2020-01-28 04:11:05 +00:00
|
|
|
# encoding=utf-8
|
|
|
|
# Copyright © 2017 Intel Corporation
|
|
|
|
|
|
|
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
# of this software and associated documentation files (the "Software"), to deal
|
|
|
|
# in the Software without restriction, including without limitation the rights
|
|
|
|
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
# copies of the Software, and to permit persons to whom the Software is
|
|
|
|
# furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
# The above copyright notice and this permission notice shall be included in
|
|
|
|
# all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
# SOFTWARE.
|
|
|
|
|
|
|
|
"""Create enum to string functions for vulkan using vk.xml."""
|
|
|
|
|
|
|
|
from __future__ import print_function
|
|
|
|
import argparse
|
|
|
|
import os
|
|
|
|
import textwrap
|
2020-06-13 14:44:20 +00:00
|
|
|
import xml.etree.ElementTree as et
|
2020-01-28 04:11:05 +00:00
|
|
|
|
|
|
|
from mako.template import Template
|
|
|
|
|
|
|
|
COPYRIGHT = textwrap.dedent(u"""\
|
|
|
|
* Copyright © 2017 Intel Corporation
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.""")
|
|
|
|
|
|
|
|
C_TEMPLATE = Template(textwrap.dedent(u"""\
|
|
|
|
/* Autogenerated file -- do not edit
|
|
|
|
* generated by ${file}
|
|
|
|
*
|
|
|
|
${copyright}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <vulkan/vulkan.h>
|
|
|
|
#include "../src/mesa/util/macros.h"
|
|
|
|
#include "vk_enum_to_str.h"
|
|
|
|
|
|
|
|
% for enum in enums:
|
|
|
|
|
|
|
|
% if enum.guard:
|
|
|
|
#ifdef ${enum.guard}
|
|
|
|
% endif
|
|
|
|
const char *
|
|
|
|
vk_${enum.name[2:]}_to_str(${enum.name} input)
|
|
|
|
{
|
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wswitch"
|
|
|
|
switch(input) {
|
|
|
|
% for v in sorted(enum.values.keys()):
|
|
|
|
case ${v}:
|
|
|
|
return "${enum.values[v]}";
|
|
|
|
% endfor
|
|
|
|
}
|
|
|
|
#pragma GCC diagnostic pop
|
|
|
|
unreachable("Undefined enum value.");
|
|
|
|
}
|
|
|
|
|
|
|
|
% if enum.guard:
|
|
|
|
#endif
|
|
|
|
% endif
|
|
|
|
%endfor
|
|
|
|
|
|
|
|
size_t vk_structure_type_size(const struct VkBaseInStructure *item)
|
|
|
|
{
|
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wswitch"
|
|
|
|
switch(item->sType) {
|
|
|
|
% for struct in structs:
|
|
|
|
% if struct.extension is not None and struct.extension.define is not None:
|
|
|
|
#ifdef ${struct.extension.define}
|
|
|
|
case ${struct.stype}: return sizeof(${struct.name});
|
|
|
|
#endif
|
|
|
|
% else:
|
|
|
|
case ${struct.stype}: return sizeof(${struct.name});
|
|
|
|
% endif
|
|
|
|
%endfor
|
|
|
|
}
|
|
|
|
#pragma GCC diagnostic pop
|
|
|
|
unreachable("Undefined struct type.");
|
|
|
|
}
|
|
|
|
|
|
|
|
void vk_load_instance_commands(VkInstance instance,
|
|
|
|
PFN_vkGetInstanceProcAddr gpa,
|
|
|
|
struct vk_instance_dispatch_table *table)
|
|
|
|
{
|
|
|
|
memset(table, 0, sizeof(*table));
|
|
|
|
table->GetInstanceProcAddr = gpa;
|
|
|
|
% for cmd in commands:
|
|
|
|
% if not cmd.device_entrypoint and cmd.name != 'vkGetInstanceProcAddr':
|
|
|
|
% if cmd.extension is not None and cmd.extension.define is not None:
|
|
|
|
#ifdef ${cmd.extension.define}
|
|
|
|
table->${cmd.name[2:]} = (PFN_${cmd.name}) gpa(instance, "${cmd.name}");
|
|
|
|
#endif
|
|
|
|
% else:
|
|
|
|
table->${cmd.name[2:]} = (PFN_${cmd.name}) gpa(instance, "${cmd.name}");
|
|
|
|
% endif
|
|
|
|
% endif
|
|
|
|
%endfor
|
|
|
|
}
|
|
|
|
|
|
|
|
void vk_load_device_commands(VkDevice device,
|
|
|
|
PFN_vkGetDeviceProcAddr gpa,
|
|
|
|
struct vk_device_dispatch_table *table)
|
|
|
|
{
|
|
|
|
memset(table, 0, sizeof(*table));
|
|
|
|
table->GetDeviceProcAddr = gpa;
|
|
|
|
% for cmd in commands:
|
|
|
|
% if cmd.device_entrypoint and cmd.name != 'vkGetDeviceProcAddr':
|
|
|
|
% if cmd.extension is not None and cmd.extension.define is not None:
|
|
|
|
#ifdef ${cmd.extension.define}
|
|
|
|
table->${cmd.name[2:]} = (PFN_${cmd.name}) gpa(device, "${cmd.name}");
|
|
|
|
#endif
|
|
|
|
% else:
|
|
|
|
table->${cmd.name[2:]} = (PFN_${cmd.name}) gpa(device, "${cmd.name}");
|
|
|
|
% endif
|
|
|
|
% endif
|
|
|
|
%endfor
|
|
|
|
}
|
|
|
|
"""),
|
|
|
|
output_encoding='utf-8')
|
|
|
|
|
|
|
|
H_TEMPLATE = Template(textwrap.dedent(u"""\
|
|
|
|
/* Autogenerated file -- do not edit
|
|
|
|
* generated by ${file}
|
|
|
|
*
|
|
|
|
${copyright}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef MESA_VK_ENUM_TO_STR_H
|
|
|
|
#define MESA_VK_ENUM_TO_STR_H
|
|
|
|
|
|
|
|
#include <vulkan/vulkan.h>
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
% for ext in extensions:
|
|
|
|
#define _${ext.name}_number (${ext.number})
|
|
|
|
% endfor
|
|
|
|
|
|
|
|
% for enum in enums:
|
|
|
|
% if enum.guard:
|
|
|
|
#ifdef ${enum.guard}
|
|
|
|
% endif
|
|
|
|
const char * vk_${enum.name[2:]}_to_str(${enum.name} input);
|
|
|
|
% if enum.guard:
|
|
|
|
#endif
|
|
|
|
% endif
|
|
|
|
% endfor
|
|
|
|
|
|
|
|
size_t vk_structure_type_size(const struct VkBaseInStructure *item);
|
|
|
|
|
|
|
|
struct vk_instance_dispatch_table {
|
|
|
|
PFN_vkGetInstanceProcAddr GetInstanceProcAddr;
|
|
|
|
% for cmd in commands:
|
|
|
|
% if not cmd.device_entrypoint and cmd.name != 'vkGetInstanceProcAddr':
|
|
|
|
% if cmd.extension is not None and cmd.extension.define is not None:
|
|
|
|
#ifdef ${cmd.extension.define}
|
|
|
|
PFN_${cmd.name} ${cmd.name[2:]};
|
|
|
|
#endif
|
|
|
|
% else:
|
|
|
|
PFN_${cmd.name} ${cmd.name[2:]};
|
|
|
|
% endif
|
|
|
|
% endif
|
|
|
|
%endfor
|
|
|
|
};
|
|
|
|
|
|
|
|
struct vk_device_dispatch_table {
|
|
|
|
PFN_vkGetDeviceProcAddr GetDeviceProcAddr;
|
|
|
|
% for cmd in commands:
|
|
|
|
% if cmd.device_entrypoint and cmd.name != 'vkGetDeviceProcAddr':
|
|
|
|
% if cmd.extension is not None and cmd.extension.define is not None:
|
|
|
|
#ifdef ${cmd.extension.define}
|
|
|
|
PFN_${cmd.name} ${cmd.name[2:]};
|
|
|
|
#endif
|
|
|
|
% else:
|
|
|
|
PFN_${cmd.name} ${cmd.name[2:]};
|
|
|
|
% endif
|
|
|
|
% endif
|
|
|
|
%endfor
|
|
|
|
};
|
|
|
|
|
|
|
|
void vk_load_instance_commands(VkInstance instance, PFN_vkGetInstanceProcAddr gpa, struct vk_instance_dispatch_table *table);
|
|
|
|
void vk_load_device_commands(VkDevice device, PFN_vkGetDeviceProcAddr gpa, struct vk_device_dispatch_table *table);
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
} /* extern "C" */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif"""),
|
|
|
|
output_encoding='utf-8')
|
|
|
|
|
|
|
|
|
|
|
|
class NamedFactory(object):
|
|
|
|
"""Factory for creating enums."""
|
|
|
|
|
|
|
|
def __init__(self, type_):
|
|
|
|
self.registry = {}
|
|
|
|
self.type = type_
|
|
|
|
|
|
|
|
def __call__(self, name, **kwargs):
|
|
|
|
try:
|
|
|
|
return self.registry[name]
|
|
|
|
except KeyError:
|
|
|
|
n = self.registry[name] = self.type(name, **kwargs)
|
|
|
|
return n
|
|
|
|
|
|
|
|
def get(self, name):
|
|
|
|
return self.registry.get(name)
|
|
|
|
|
|
|
|
|
|
|
|
class VkExtension(object):
|
|
|
|
"""Simple struct-like class representing extensions"""
|
|
|
|
|
|
|
|
def __init__(self, name, number=None, define=None):
|
|
|
|
self.name = name
|
|
|
|
self.number = number
|
|
|
|
self.define = define
|
|
|
|
|
|
|
|
|
|
|
|
class VkEnum(object):
|
|
|
|
"""Simple struct-like class representing a single Vulkan Enum."""
|
|
|
|
|
|
|
|
def __init__(self, name, values=None):
|
|
|
|
self.name = name
|
|
|
|
self.extension = None
|
|
|
|
# Maps numbers to names
|
|
|
|
self.values = values or dict()
|
|
|
|
self.name_to_value = dict()
|
|
|
|
self.guard = None
|
|
|
|
self.name_to_alias_list = {}
|
|
|
|
|
|
|
|
def add_value(self, name, value=None,
|
|
|
|
extnum=None, offset=None, alias=None,
|
|
|
|
error=False):
|
|
|
|
if alias is not None:
|
|
|
|
assert value is None and offset is None
|
|
|
|
if alias not in self.name_to_value:
|
|
|
|
# We don't have this alias yet. Just record the alias and
|
|
|
|
# we'll deal with it later.
|
|
|
|
alias_list = self.name_to_alias_list.get(alias, [])
|
|
|
|
alias_list.append(name);
|
|
|
|
return
|
|
|
|
|
|
|
|
# Use the value from the alias
|
|
|
|
value = self.name_to_value[alias]
|
|
|
|
|
|
|
|
assert value is not None or extnum is not None
|
|
|
|
if value is None:
|
|
|
|
value = 1000000000 + (extnum - 1) * 1000 + offset
|
|
|
|
if error:
|
|
|
|
value = -value
|
|
|
|
|
|
|
|
self.name_to_value[name] = value
|
|
|
|
if value not in self.values:
|
|
|
|
self.values[value] = name
|
|
|
|
elif len(self.values[value]) > len(name):
|
|
|
|
self.values[value] = name
|
|
|
|
|
|
|
|
# Now that the value has been fully added, resolve aliases, if any.
|
|
|
|
if name in self.name_to_alias_list:
|
|
|
|
for alias in self.name_to_alias_list[name]:
|
|
|
|
add_value(alias, value)
|
|
|
|
del self.name_to_alias_list[name]
|
|
|
|
|
|
|
|
def add_value_from_xml(self, elem, extension=None):
|
|
|
|
self.extension = extension
|
|
|
|
if 'value' in elem.attrib:
|
|
|
|
self.add_value(elem.attrib['name'],
|
|
|
|
value=int(elem.attrib['value'], base=0))
|
|
|
|
elif 'alias' in elem.attrib:
|
|
|
|
self.add_value(elem.attrib['name'], alias=elem.attrib['alias'])
|
|
|
|
else:
|
|
|
|
error = 'dir' in elem.attrib and elem.attrib['dir'] == '-'
|
|
|
|
if 'extnumber' in elem.attrib:
|
|
|
|
extnum = int(elem.attrib['extnumber'])
|
|
|
|
else:
|
|
|
|
extnum = extension.number
|
|
|
|
self.add_value(elem.attrib['name'],
|
|
|
|
extnum=extnum,
|
|
|
|
offset=int(elem.attrib['offset']),
|
|
|
|
error=error)
|
|
|
|
|
|
|
|
def set_guard(self, g):
|
|
|
|
self.guard = g
|
|
|
|
|
|
|
|
|
|
|
|
class VkCommand(object):
|
|
|
|
"""Simple struct-like class representing a single Vulkan command"""
|
|
|
|
|
|
|
|
def __init__(self, name, device_entrypoint=False):
|
|
|
|
self.name = name
|
|
|
|
self.device_entrypoint = device_entrypoint
|
|
|
|
self.extension = None
|
|
|
|
|
|
|
|
|
|
|
|
class VkChainStruct(object):
|
|
|
|
"""Simple struct-like class representing a single Vulkan struct identified with a VkStructureType"""
|
|
|
|
def __init__(self, name, stype):
|
|
|
|
self.name = name
|
|
|
|
self.stype = stype
|
|
|
|
self.extension = None
|
|
|
|
|
|
|
|
|
|
|
|
def struct_get_stype(xml_node):
|
|
|
|
for member in xml_node.findall('./member'):
|
|
|
|
name = member.findall('./name')
|
|
|
|
if len(name) > 0 and name[0].text == "sType":
|
|
|
|
return member.get('values')
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
def parse_xml(cmd_factory, enum_factory, ext_factory, struct_factory, filename):
|
|
|
|
"""Parse the XML file. Accumulate results into the factories.
|
|
|
|
|
|
|
|
This parser is a memory efficient iterative XML parser that returns a list
|
|
|
|
of VkEnum objects.
|
|
|
|
"""
|
|
|
|
|
|
|
|
xml = et.parse(filename)
|
|
|
|
|
|
|
|
for enum_type in xml.findall('./enums[@type="enum"]'):
|
|
|
|
enum = enum_factory(enum_type.attrib['name'])
|
|
|
|
for value in enum_type.findall('./enum'):
|
|
|
|
enum.add_value_from_xml(value)
|
|
|
|
|
|
|
|
for value in xml.findall('./feature/require/enum[@extends]'):
|
|
|
|
enum = enum_factory.get(value.attrib['extends'])
|
|
|
|
if enum is not None:
|
|
|
|
enum.add_value_from_xml(value)
|
|
|
|
|
|
|
|
for command in xml.findall('./commands/command'):
|
|
|
|
name = command.find('./proto/name')
|
2020-02-13 18:09:47 +00:00
|
|
|
if name is not None and "ANDROID" in name.text:
|
|
|
|
continue
|
2020-01-28 04:11:05 +00:00
|
|
|
first_arg = command.find('./param/type')
|
|
|
|
# Some commands are alias KHR -> nonKHR, ignore those
|
|
|
|
if name is not None:
|
|
|
|
cmd_factory(name.text,
|
|
|
|
device_entrypoint=(first_arg.text in ('VkDevice', 'VkCommandBuffer', 'VkQueue')))
|
|
|
|
|
|
|
|
for struct_type in xml.findall('./types/type[@category="struct"]'):
|
|
|
|
name = struct_type.attrib['name']
|
2020-02-13 18:09:47 +00:00
|
|
|
if name is not None and "ANDROID" in name:
|
|
|
|
continue
|
2020-01-28 04:11:05 +00:00
|
|
|
stype = struct_get_stype(struct_type)
|
|
|
|
if stype is not None:
|
|
|
|
struct_factory(name, stype=stype)
|
|
|
|
|
|
|
|
platform_define = {}
|
|
|
|
for platform in xml.findall('./platforms/platform'):
|
|
|
|
name = platform.attrib['name']
|
|
|
|
define = platform.attrib['protect']
|
|
|
|
platform_define[name] = define
|
|
|
|
|
|
|
|
for ext_elem in xml.findall('./extensions/extension[@supported="vulkan"]'):
|
|
|
|
define = None
|
|
|
|
if "platform" in ext_elem.attrib:
|
|
|
|
define = platform_define[ext_elem.attrib['platform']]
|
|
|
|
extension = ext_factory(ext_elem.attrib['name'],
|
|
|
|
number=int(ext_elem.attrib['number']),
|
|
|
|
define=define)
|
|
|
|
|
|
|
|
for value in ext_elem.findall('./require/enum[@extends]'):
|
|
|
|
enum = enum_factory.get(value.attrib['extends'])
|
|
|
|
if enum is not None:
|
|
|
|
enum.add_value_from_xml(value, extension)
|
|
|
|
for t in ext_elem.findall('./require/type'):
|
|
|
|
struct = struct_factory.get(t.attrib['name'])
|
|
|
|
if struct is not None:
|
|
|
|
struct.extension = extension
|
|
|
|
|
|
|
|
if define:
|
|
|
|
for value in ext_elem.findall('./require/type[@name]'):
|
|
|
|
enum = enum_factory.get(value.attrib['name'])
|
|
|
|
if enum is not None:
|
|
|
|
enum.set_guard(define)
|
|
|
|
|
|
|
|
for t in ext_elem.findall('./require/command'):
|
|
|
|
command = cmd_factory.get(t.attrib['name'])
|
|
|
|
if command is not None:
|
|
|
|
command.extension = extension
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument('--xml', required=True,
|
|
|
|
help='Vulkan API XML files',
|
|
|
|
action='append',
|
|
|
|
dest='xml_files')
|
|
|
|
parser.add_argument('--outdir',
|
|
|
|
help='Directory to put the generated files in',
|
|
|
|
required=True)
|
|
|
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
command_factory = NamedFactory(VkCommand)
|
|
|
|
enum_factory = NamedFactory(VkEnum)
|
|
|
|
ext_factory = NamedFactory(VkExtension)
|
|
|
|
struct_factory = NamedFactory(VkChainStruct)
|
|
|
|
for filename in args.xml_files:
|
|
|
|
parse_xml(command_factory, enum_factory, ext_factory, struct_factory, filename)
|
|
|
|
commands = sorted(command_factory.registry.values(), key=lambda e: e.name)
|
|
|
|
enums = sorted(enum_factory.registry.values(), key=lambda e: e.name)
|
|
|
|
extensions = sorted(ext_factory.registry.values(), key=lambda e: e.name)
|
|
|
|
structs = sorted(struct_factory.registry.values(), key=lambda e: e.name)
|
|
|
|
|
|
|
|
for template, file_ in [(C_TEMPLATE, os.path.join(args.outdir, 'vk_enum_to_str.c')),
|
|
|
|
(H_TEMPLATE, os.path.join(args.outdir, 'vk_enum_to_str.h'))]:
|
|
|
|
with open(file_, 'wb') as f:
|
|
|
|
f.write(template.render(
|
|
|
|
file=os.path.basename(__file__),
|
|
|
|
commands=commands,
|
|
|
|
enums=enums,
|
|
|
|
extensions=extensions,
|
|
|
|
structs=structs,
|
|
|
|
copyright=COPYRIGHT))
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|