blob: 51f56ad087c489cdbaf382161d8ade0cced0e0ec [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")
Courtney Goeltzenleuchterf83cd4a2015-06-26 15:05:29 -0600122 stmts.append("memset(table, 0, sizeof(*table));")
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600123 stmts.append("table->GetDeviceProcAddr =(PFN_vkGetDeviceProcAddr) gpa(device,\"vkGetDeviceProcAddr\");")
Jon Ashburnd9564002015-05-07 10:27:37 -0600124 for proto in self.protos:
Tony Barbour426b9052015-06-24 16:06:58 -0600125 if proto.name == "CreateInstance" or proto.name == "GetGlobalExtensionProperties" or proto.name == "GetGlobalExtensionCount" or proto.params[0].ty == "VkInstance" or proto.params[0].ty == "VkPhysicalDevice":
Jon Ashburn2666e2f2015-05-15 15:09:35 -0600126 continue
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600127 if proto.name != "GetDeviceProcAddr":
128 stmts.append("table->%s = (PFN_vk%s) gpa(baseDevice, \"vk%s\");" %
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800129 (proto.name, proto.name, proto.name))
Jon Ashburnd9564002015-05-07 10:27:37 -0600130 func.append("static inline void %s_initialize_dispatch_table(VkLayerDispatchTable *table,"
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800131 % self.prefix)
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600132 func.append("%s const VkBaseLayerObject *devw)"
Jon Ashburnd9564002015-05-07 10:27:37 -0600133 % (" " * len(self.prefix)))
134 else:
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600135 # GPA has to be first one and uses wrapped object
136 stmts.append("VkInstance instance = (VkInstance) instw->nextObject;")
137 stmts.append("PFN_vkGetInstanceProcAddr gpa = (PFN_vkGetInstanceProcAddr) instw->pGPA;")
138 stmts.append("VkInstance baseInstance = (VkInstance) instw->baseObject;")
139 stmts.append("// GPA has to be first entry inited and uses wrapped object since it triggers init")
140 stmts.append("table->GetInstanceProcAddr =(PFN_vkGetInstanceProcAddr) gpa(instance,\"vkGetInstanceProcAddr\");")
Jon Ashburnd9564002015-05-07 10:27:37 -0600141 for proto in self.protos:
Jon Ashburnba4a1952015-06-16 12:44:51 -0600142 if proto.name != "CreateInstance" and proto.params[0].ty != "VkInstance" and proto.params[0].ty != "VkPhysicalDevice":
Jon Ashburnd9564002015-05-07 10:27:37 -0600143 continue
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600144 if proto.name != "GetInstanceProcAddr":
145 stmts.append("table->%s = (PFN_vk%s) gpa(baseInstance, \"vk%s\");" %
Jon Ashburnd9564002015-05-07 10:27:37 -0600146 (proto.name, proto.name, proto.name))
147 func.append("static inline void %s_init_instance_dispatch_table(VkLayerInstanceDispatchTable *table,"
148 % self.prefix)
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600149 func.append("%s const VkBaseLayerObject *instw)"
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800150 % (" " * len(self.prefix)))
151 func.append("{")
152 func.append(" %s" % "\n ".join(stmts))
153 func.append("}")
154
155 return "\n".join(func)
156
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800157 def generate_body(self):
Jon Ashburn4f2575f2015-05-28 16:25:02 -0600158 body = [self._generate_init_dispatch("device"),
159 self._generate_init_dispatch("instance")]
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800160
161 return "\n\n".join(body)
162
Chia-I Wu731517d2015-01-03 23:48:15 +0800163class IcdDummyEntrypointsSubcommand(Subcommand):
Chia-I Wu30c78292014-08-04 10:08:08 +0800164 def run(self):
Chia-I Wu731517d2015-01-03 23:48:15 +0800165 if len(self.argv) == 1:
166 self.prefix = self.argv[0]
167 self.qual = "static"
168 else:
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600169 self.prefix = "vk"
Chia-I Wu731517d2015-01-03 23:48:15 +0800170 self.qual = "ICD_EXPORT"
Chia-I Wu30c78292014-08-04 10:08:08 +0800171
172 super().run()
173
174 def generate_header(self):
175 return "#include \"icd.h\""
176
177 def _generate_stub_decl(self, proto):
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600178 return proto.c_pretty_decl(self.prefix + proto.name, attr="VKAPI")
Chia-I Wu30c78292014-08-04 10:08:08 +0800179
180 def _generate_stubs(self):
181 stubs = []
182 for proto in self.protos:
Chia-I Wu30c78292014-08-04 10:08:08 +0800183 decl = self._generate_stub_decl(proto)
Mark Lobodzinskie2d07a52015-01-29 08:55:56 -0600184 if proto.ret != "void":
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600185 stmt = " return VK_ERROR_UNKNOWN;\n"
Chia-I Wu30c78292014-08-04 10:08:08 +0800186 else:
187 stmt = ""
188
Chia-I Wu731517d2015-01-03 23:48:15 +0800189 stubs.append("%s %s\n{\n%s}" % (self.qual, decl, stmt))
Chia-I Wu30c78292014-08-04 10:08:08 +0800190
191 return "\n\n".join(stubs)
192
Chia-I Wu30c78292014-08-04 10:08:08 +0800193 def generate_body(self):
Chia-I Wu731517d2015-01-03 23:48:15 +0800194 return self._generate_stubs()
Chia-I Wu30c78292014-08-04 10:08:08 +0800195
Chia-I Wu58f20852015-01-04 00:11:17 +0800196class IcdGetProcAddrSubcommand(IcdDummyEntrypointsSubcommand):
197 def generate_header(self):
198 return "\n".join(["#include <string.h>", "#include \"icd.h\""])
199
200 def generate_body(self):
201 for proto in self.protos:
Jon Ashburn1245cec2015-05-18 13:20:15 -0600202 if proto.name == "GetDeviceProcAddr":
Chia-I Wu58f20852015-01-04 00:11:17 +0800203 gpa_proto = proto
Jon Ashburn53c16772015-05-06 10:15:07 -0600204 if proto.name == "GetInstanceProcAddr":
205 gpa_instance_proto = proto
Chia-I Wu58f20852015-01-04 00:11:17 +0800206
Jon Ashburn53c16772015-05-06 10:15:07 -0600207 gpa_instance_decl = self._generate_stub_decl(gpa_instance_proto)
Chia-I Wu58f20852015-01-04 00:11:17 +0800208 gpa_decl = self._generate_stub_decl(gpa_proto)
209 gpa_pname = gpa_proto.params[-1].name
210
211 lookups = []
212 for proto in self.protos:
213 lookups.append("if (!strcmp(%s, \"%s\"))" %
214 (gpa_pname, proto.name))
215 lookups.append(" return (%s) %s%s;" %
216 (gpa_proto.ret, self.prefix, proto.name))
217
218 body = []
Jon Ashburn53c16772015-05-06 10:15:07 -0600219 body.append("%s %s" % (self.qual, gpa_instance_decl))
220 body.append("{")
221 body.append(" return NULL;")
222 body.append("}")
223 body.append("")
224
Chia-I Wu58f20852015-01-04 00:11:17 +0800225 body.append("%s %s" % (self.qual, gpa_decl))
226 body.append("{")
Chia-I Wuf7d6d3c2015-01-05 12:55:13 +0800227 body.append(generate_get_proc_addr_check(gpa_pname))
Chia-I Wu58f20852015-01-04 00:11:17 +0800228 body.append("")
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600229 body.append(" %s += 2;" % gpa_pname)
Chia-I Wu58f20852015-01-04 00:11:17 +0800230 body.append(" %s" % "\n ".join(lookups))
231 body.append("")
232 body.append(" return NULL;")
233 body.append("}")
234
235 return "\n".join(body)
236
Chia-I Wud98a23c2015-04-11 10:56:50 +0800237class WinDefFileSubcommand(Subcommand):
238 def run(self):
239 library_exports = {
240 "all": [],
241 "icd": [
Tony Barbour8205d902015-04-16 15:59:00 -0600242 "EnumeratePhysicalDevices",
Chia-I Wud98a23c2015-04-11 10:56:50 +0800243 "CreateInstance",
244 "DestroyInstance",
Jon Ashburn1245cec2015-05-18 13:20:15 -0600245 "GetDeviceProcAddr",
246 "GetInstanceProcAddr",
Chia-I Wud98a23c2015-04-11 10:56:50 +0800247 ],
248 "layer": [
Jon Ashburn3feb7312015-05-06 14:49:55 -0600249 "GetInstanceProcAddr",
Jon Ashburn1245cec2015-05-18 13:20:15 -0600250 "GetDeviceProcAddr",
Chia-I Wud98a23c2015-04-11 10:56:50 +0800251 "EnumerateLayers",
Tony Barbour426b9052015-06-24 16:06:58 -0600252 "GetGlobalExtensionCount",
253 "GetGlobalExtensionProperties"
Chia-I Wud98a23c2015-04-11 10:56:50 +0800254 ],
255 }
256
257 if len(self.argv) != 2 or self.argv[1] not in library_exports:
258 print("WinDefFileSubcommand: <library-name> {%s}" %
259 "|".join(library_exports.keys()))
260 return
261
262 self.library = self.argv[0]
263 self.exports = library_exports[self.argv[1]]
264
265 super().run()
266
267 def generate_copyright(self):
268 return """; THIS FILE IS GENERATED. DO NOT EDIT.
269
270;;;; Begin Copyright Notice ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600271; Vulkan
Chia-I Wud98a23c2015-04-11 10:56:50 +0800272;
273; Copyright (C) 2015 LunarG, Inc.
274;
275; Permission is hereby granted, free of charge, to any person obtaining a
276; copy of this software and associated documentation files (the "Software"),
277; to deal in the Software without restriction, including without limitation
278; the rights to use, copy, modify, merge, publish, distribute, sublicense,
279; and/or sell copies of the Software, and to permit persons to whom the
280; Software is furnished to do so, subject to the following conditions:
281;
282; The above copyright notice and this permission notice shall be included
283; in all copies or substantial portions of the Software.
284;
285; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
286; IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
287; FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
288; THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
289; LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
290; FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
291; DEALINGS IN THE SOFTWARE.
292;;;; End Copyright Notice ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;"""
293
294 def generate_header(self):
295 return "; The following is required on Windows, for exporting symbols from the DLL"
296
297 def generate_body(self):
298 body = []
299
300 body.append("LIBRARY " + self.library)
301 body.append("EXPORTS")
302
303 for proto in self.protos:
304 if self.exports and proto.name not in self.exports:
305 continue
Courtney Goeltzenleuchter9cc421e2015-04-08 15:36:08 -0600306 body.append(" vk" + proto.name)
Chia-I Wud98a23c2015-04-11 10:56:50 +0800307
308 return "\n".join(body)
309
Chia-I Wufb2559d2014-08-01 11:19:52 +0800310def main():
311 subcommands = {
Chia-I Wu28b8d8c2015-01-04 10:19:50 +0800312 "dispatch-table-ops": DispatchTableOpsSubcommand,
Chia-I Wu731517d2015-01-03 23:48:15 +0800313 "icd-dummy-entrypoints": IcdDummyEntrypointsSubcommand,
Chia-I Wu58f20852015-01-04 00:11:17 +0800314 "icd-get-proc-addr": IcdGetProcAddrSubcommand,
Chia-I Wud98a23c2015-04-11 10:56:50 +0800315 "win-def-file": WinDefFileSubcommand,
Chia-I Wufb2559d2014-08-01 11:19:52 +0800316 }
317
318 if len(sys.argv) < 2 or sys.argv[1] not in subcommands:
319 print("Usage: %s <subcommand> [options]" % sys.argv[0])
320 print
321 print("Available sucommands are: %s" % " ".join(subcommands))
322 exit(1)
323
324 subcmd = subcommands[sys.argv[1]](sys.argv[2:])
325 subcmd.run()
326
327if __name__ == "__main__":
328 main()