blob: 790a543f614d8990a4ce0a005784b1ce144621a2 [file] [log] [blame]
Chia-I Wufb2559d2014-08-01 11:19:52 +08001#!/usr/bin/env python3
Chia-I Wu44e42362014-09-02 08:32:09 +08002#
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -06003# VK
Chia-I Wu44e42362014-09-02 08:32:09 +08004#
5# Copyright (C) 2014 LunarG, Inc.
6#
7# Permission is hereby granted, free of charge, to any person obtaining a
8# copy of this software and associated documentation files (the "Software"),
9# to deal in the Software without restriction, including without limitation
10# the rights to use, copy, modify, merge, publish, distribute, sublicense,
11# and/or sell copies of the Software, and to permit persons to whom the
12# Software is furnished to do so, subject to the following conditions:
13#
14# The above copyright notice and this permission notice shall be included
15# in all copies or substantial portions of the Software.
16#
17# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
20# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
22# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
23# DEALINGS IN THE SOFTWARE.
24#
25# Authors:
26# Chia-I Wu <olv@lunarg.com>
Chia-I Wufb2559d2014-08-01 11:19:52 +080027
28import sys
29
Courtney Goeltzenleuchtera8c06282015-04-14 14:55:44 -060030import vulkan
Chia-I Wufb2559d2014-08-01 11:19:52 +080031
Chia-I Wuf7d6d3c2015-01-05 12:55:13 +080032def generate_get_proc_addr_check(name):
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -060033 return " if (!%s || %s[0] != 'v' || %s[1] != 'k')\n" \
34 " return NULL;" % ((name,) * 3)
Chia-I Wuf7d6d3c2015-01-05 12:55:13 +080035
Chia-I Wufb2559d2014-08-01 11:19:52 +080036class Subcommand(object):
37 def __init__(self, argv):
38 self.argv = argv
Courtney Goeltzenleuchtera8c06282015-04-14 14:55:44 -060039 self.headers = vulkan.headers
40 self.protos = vulkan.protos
Chia-I Wufb2559d2014-08-01 11:19:52 +080041
42 def run(self):
Chia-I Wufb2559d2014-08-01 11:19:52 +080043 print(self.generate())
44
45 def generate(self):
46 copyright = self.generate_copyright()
47 header = self.generate_header()
48 body = self.generate_body()
49 footer = self.generate_footer()
50
51 contents = []
52 if copyright:
53 contents.append(copyright)
54 if header:
55 contents.append(header)
56 if body:
57 contents.append(body)
58 if footer:
59 contents.append(footer)
60
61 return "\n\n".join(contents)
62
63 def generate_copyright(self):
64 return """/* THIS FILE IS GENERATED. DO NOT EDIT. */
65
66/*
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -060067 * Vulkan
Chia-I Wufb2559d2014-08-01 11:19:52 +080068 *
69 * Copyright (C) 2014 LunarG, Inc.
70 *
71 * Permission is hereby granted, free of charge, to any person obtaining a
72 * copy of this software and associated documentation files (the "Software"),
73 * to deal in the Software without restriction, including without limitation
74 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
75 * and/or sell copies of the Software, and to permit persons to whom the
76 * Software is furnished to do so, subject to the following conditions:
77 *
78 * The above copyright notice and this permission notice shall be included
79 * in all copies or substantial portions of the Software.
80 *
81 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
82 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
83 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
84 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
85 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
86 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
87 * DEALINGS IN THE SOFTWARE.
88 */"""
89
90 def generate_header(self):
Chia-I Wu6ae460f2014-09-13 13:36:06 +080091 return "\n".join(["#include <" + h + ">" for h in self.headers])
Chia-I Wufb2559d2014-08-01 11:19:52 +080092
93 def generate_body(self):
94 pass
95
96 def generate_footer(self):
97 pass
98
Chia-I Wu28b8d8c2015-01-04 10:19:50 +080099class DispatchTableOpsSubcommand(Subcommand):
100 def run(self):
101 if len(self.argv) != 1:
102 print("DispatchTableOpsSubcommand: <prefix> unspecified")
103 return
104
105 self.prefix = self.argv[0]
106 super().run()
107
108 def generate_header(self):
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600109 return "\n".join(["#include <vulkan.h>",
110 "#include <vkLayer.h>",
Jon Ashburnaef65882015-05-04 09:16:41 -0600111 "#include <string.h>"])
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800112
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600113 def _generate_init_dispatch(self, type):
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800114 stmts = []
Jon Ashburnd9564002015-05-07 10:27:37 -0600115 func = []
116 if type == "device":
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600117 # GPA has to be first one and uses wrapped object
118 stmts.append("VkDevice device = (VkDevice) devw->nextObject;")
119 stmts.append("PFN_vkGetDeviceProcAddr gpa = (PFN_vkGetDeviceProcAddr) devw->pGPA;")
120 stmts.append("VkDevice baseDevice = (VkDevice) devw->baseObject;")
121 stmts.append("// GPA has to be first entry inited and uses wrapped object since it triggers init")
122 stmts.append("table->GetDeviceProcAddr =(PFN_vkGetDeviceProcAddr) gpa(device,\"vkGetDeviceProcAddr\");")
Jon Ashburnd9564002015-05-07 10:27:37 -0600123 for proto in self.protos:
Jon Ashburn2666e2f2015-05-15 15:09:35 -0600124 if proto.name == "CreateInstance" or proto.name == "GetGlobalExtensionInfo" or proto.name == "GetDisplayInfoWSI" or proto.params[0].ty == "VkInstance" or proto.params[0].ty == "VkPhysicalDevice":
125 continue
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600126 if proto.name != "GetDeviceProcAddr":
127 stmts.append("table->%s = (PFN_vk%s) gpa(baseDevice, \"vk%s\");" %
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800128 (proto.name, proto.name, proto.name))
Jon Ashburnd9564002015-05-07 10:27:37 -0600129 func.append("static inline void %s_initialize_dispatch_table(VkLayerDispatchTable *table,"
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800130 % self.prefix)
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600131 func.append("%s const VkBaseLayerObject *devw)"
Jon Ashburnd9564002015-05-07 10:27:37 -0600132 % (" " * len(self.prefix)))
133 else:
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600134 # GPA has to be first one and uses wrapped object
135 stmts.append("VkInstance instance = (VkInstance) instw->nextObject;")
136 stmts.append("PFN_vkGetInstanceProcAddr gpa = (PFN_vkGetInstanceProcAddr) instw->pGPA;")
137 stmts.append("VkInstance baseInstance = (VkInstance) instw->baseObject;")
138 stmts.append("// GPA has to be first entry inited and uses wrapped object since it triggers init")
139 stmts.append("table->GetInstanceProcAddr =(PFN_vkGetInstanceProcAddr) gpa(instance,\"vkGetInstanceProcAddr\");")
Jon Ashburnd9564002015-05-07 10:27:37 -0600140 for proto in self.protos:
Jon Ashburn2666e2f2015-05-15 15:09:35 -0600141 if proto.name != "CreateInstance" and proto.name != "GetGlobalExtensionInfo" and proto.name != "GetDisplayInfoWSI" and proto.params[0].ty != "VkInstance" and proto.params[0].ty != "VkPhysicalDevice":
Jon Ashburnd9564002015-05-07 10:27:37 -0600142 continue
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600143 if proto.name != "GetInstanceProcAddr":
144 stmts.append("table->%s = (PFN_vk%s) gpa(baseInstance, \"vk%s\");" %
Jon Ashburnd9564002015-05-07 10:27:37 -0600145 (proto.name, proto.name, proto.name))
146 func.append("static inline void %s_init_instance_dispatch_table(VkLayerInstanceDispatchTable *table,"
147 % self.prefix)
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600148 func.append("%s const VkBaseLayerObject *instw)"
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800149 % (" " * len(self.prefix)))
150 func.append("{")
151 func.append(" %s" % "\n ".join(stmts))
152 func.append("}")
153
154 return "\n".join(func)
155
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800156 def generate_body(self):
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600157 body = [self._generate_init_dispatch("device"),
158 self._generate_init_dispatch("instance")]
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800159
160 return "\n\n".join(body)
161
Chia-I Wu731517d2015-01-03 23:48:15 +0800162class IcdDummyEntrypointsSubcommand(Subcommand):
Chia-I Wu30c78292014-08-04 10:08:08 +0800163 def run(self):
Chia-I Wu731517d2015-01-03 23:48:15 +0800164 if len(self.argv) == 1:
165 self.prefix = self.argv[0]
166 self.qual = "static"
167 else:
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600168 self.prefix = "vk"
Chia-I Wu731517d2015-01-03 23:48:15 +0800169 self.qual = "ICD_EXPORT"
Chia-I Wu30c78292014-08-04 10:08:08 +0800170
171 super().run()
172
173 def generate_header(self):
174 return "#include \"icd.h\""
175
176 def _generate_stub_decl(self, proto):
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600177 return proto.c_pretty_decl(self.prefix + proto.name, attr="VKAPI")
Chia-I Wu30c78292014-08-04 10:08:08 +0800178
179 def _generate_stubs(self):
180 stubs = []
181 for proto in self.protos:
Chia-I Wu30c78292014-08-04 10:08:08 +0800182 decl = self._generate_stub_decl(proto)
Mark Lobodzinskie2d07a52015-01-29 08:55:56 -0600183 if proto.ret != "void":
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600184 stmt = " return VK_ERROR_UNKNOWN;\n"
Chia-I Wu30c78292014-08-04 10:08:08 +0800185 else:
186 stmt = ""
187
Chia-I Wu731517d2015-01-03 23:48:15 +0800188 stubs.append("%s %s\n{\n%s}" % (self.qual, decl, stmt))
Chia-I Wu30c78292014-08-04 10:08:08 +0800189
190 return "\n\n".join(stubs)
191
Chia-I Wu30c78292014-08-04 10:08:08 +0800192 def generate_body(self):
Chia-I Wu731517d2015-01-03 23:48:15 +0800193 return self._generate_stubs()
Chia-I Wu30c78292014-08-04 10:08:08 +0800194
Chia-I Wu58f20852015-01-04 00:11:17 +0800195class IcdGetProcAddrSubcommand(IcdDummyEntrypointsSubcommand):
196 def generate_header(self):
197 return "\n".join(["#include <string.h>", "#include \"icd.h\""])
198
199 def generate_body(self):
200 for proto in self.protos:
Jon Ashburn1245cec2015-05-18 13:20:15 -0600201 if proto.name == "GetDeviceProcAddr":
Chia-I Wu58f20852015-01-04 00:11:17 +0800202 gpa_proto = proto
Jon Ashburn53c16772015-05-06 10:15:07 -0600203 if proto.name == "GetInstanceProcAddr":
204 gpa_instance_proto = proto
Chia-I Wu58f20852015-01-04 00:11:17 +0800205
Jon Ashburn53c16772015-05-06 10:15:07 -0600206 gpa_instance_decl = self._generate_stub_decl(gpa_instance_proto)
Chia-I Wu58f20852015-01-04 00:11:17 +0800207 gpa_decl = self._generate_stub_decl(gpa_proto)
208 gpa_pname = gpa_proto.params[-1].name
209
210 lookups = []
211 for proto in self.protos:
212 lookups.append("if (!strcmp(%s, \"%s\"))" %
213 (gpa_pname, proto.name))
214 lookups.append(" return (%s) %s%s;" %
215 (gpa_proto.ret, self.prefix, proto.name))
216
217 body = []
Jon Ashburn53c16772015-05-06 10:15:07 -0600218 body.append("%s %s" % (self.qual, gpa_instance_decl))
219 body.append("{")
220 body.append(" return NULL;")
221 body.append("}")
222 body.append("")
223
Chia-I Wu58f20852015-01-04 00:11:17 +0800224 body.append("%s %s" % (self.qual, gpa_decl))
225 body.append("{")
Chia-I Wuf7d6d3c2015-01-05 12:55:13 +0800226 body.append(generate_get_proc_addr_check(gpa_pname))
Chia-I Wu58f20852015-01-04 00:11:17 +0800227 body.append("")
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600228 body.append(" %s += 2;" % gpa_pname)
Chia-I Wu58f20852015-01-04 00:11:17 +0800229 body.append(" %s" % "\n ".join(lookups))
230 body.append("")
231 body.append(" return NULL;")
232 body.append("}")
233
234 return "\n".join(body)
235
Chia-I Wud98a23c2015-04-11 10:56:50 +0800236class WinDefFileSubcommand(Subcommand):
237 def run(self):
238 library_exports = {
239 "all": [],
240 "icd": [
Tony Barbour8205d902015-04-16 15:59:00 -0600241 "EnumeratePhysicalDevices",
Chia-I Wud98a23c2015-04-11 10:56:50 +0800242 "CreateInstance",
243 "DestroyInstance",
Jon Ashburn1245cec2015-05-18 13:20:15 -0600244 "GetDeviceProcAddr",
245 "GetInstanceProcAddr",
Chia-I Wud98a23c2015-04-11 10:56:50 +0800246 ],
247 "layer": [
Jon Ashburn3feb7312015-05-06 14:49:55 -0600248 "GetInstanceProcAddr",
Jon Ashburn1245cec2015-05-18 13:20:15 -0600249 "GetDeviceProcAddr",
Chia-I Wud98a23c2015-04-11 10:56:50 +0800250 "EnumerateLayers",
Jon Ashburnb67859d2015-04-24 14:10:50 -0700251 "GetGlobalExtensionInfo",
Chia-I Wud98a23c2015-04-11 10:56:50 +0800252 ],
253 }
254
255 if len(self.argv) != 2 or self.argv[1] not in library_exports:
256 print("WinDefFileSubcommand: <library-name> {%s}" %
257 "|".join(library_exports.keys()))
258 return
259
260 self.library = self.argv[0]
261 self.exports = library_exports[self.argv[1]]
262
263 super().run()
264
265 def generate_copyright(self):
266 return """; THIS FILE IS GENERATED. DO NOT EDIT.
267
268;;;; Begin Copyright Notice ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600269; Vulkan
Chia-I Wud98a23c2015-04-11 10:56:50 +0800270;
271; Copyright (C) 2015 LunarG, Inc.
272;
273; Permission is hereby granted, free of charge, to any person obtaining a
274; copy of this software and associated documentation files (the "Software"),
275; to deal in the Software without restriction, including without limitation
276; the rights to use, copy, modify, merge, publish, distribute, sublicense,
277; and/or sell copies of the Software, and to permit persons to whom the
278; Software is furnished to do so, subject to the following conditions:
279;
280; The above copyright notice and this permission notice shall be included
281; in all copies or substantial portions of the Software.
282;
283; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
284; IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
285; FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
286; THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
287; LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
288; FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
289; DEALINGS IN THE SOFTWARE.
290;;;; End Copyright Notice ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;"""
291
292 def generate_header(self):
293 return "; The following is required on Windows, for exporting symbols from the DLL"
294
295 def generate_body(self):
296 body = []
297
298 body.append("LIBRARY " + self.library)
299 body.append("EXPORTS")
300
301 for proto in self.protos:
302 if self.exports and proto.name not in self.exports:
303 continue
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600304 body.append(" vk" + proto.name)
Chia-I Wud98a23c2015-04-11 10:56:50 +0800305
306 return "\n".join(body)
307
Chia-I Wufb2559d2014-08-01 11:19:52 +0800308def main():
309 subcommands = {
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800310 "dispatch-table-ops": DispatchTableOpsSubcommand,
Chia-I Wu731517d2015-01-03 23:48:15 +0800311 "icd-dummy-entrypoints": IcdDummyEntrypointsSubcommand,
Chia-I Wu58f20852015-01-04 00:11:17 +0800312 "icd-get-proc-addr": IcdGetProcAddrSubcommand,
Chia-I Wud98a23c2015-04-11 10:56:50 +0800313 "win-def-file": WinDefFileSubcommand,
Chia-I Wufb2559d2014-08-01 11:19:52 +0800314 }
315
316 if len(sys.argv) < 2 or sys.argv[1] not in subcommands:
317 print("Usage: %s <subcommand> [options]" % sys.argv[0])
318 print
319 print("Available sucommands are: %s" % " ".join(subcommands))
320 exit(1)
321
322 subcmd = subcommands[sys.argv[1]](sys.argv[2:])
323 subcmd.run()
324
325if __name__ == "__main__":
326 main()