| COPYRIGHT = """\ |
| /* |
| * Copyright (C) 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 (including the next |
| * paragraph) 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. |
| */ |
| """ |
| |
| import argparse |
| import json |
| from sys import stdout |
| from mako.template import Template |
| |
| def collect_data(spirv, kind): |
| for x in spirv["operand_kinds"]: |
| if x["kind"] == kind: |
| operands = x |
| break |
| |
| values = {} |
| for x in operands["enumerants"]: |
| val = x["value"] |
| assert(val not in values) |
| names = [x["enumerant"]] |
| if "aliases" in x: |
| names.extend(x["aliases"]) |
| values[val] = names |
| |
| return (kind, list(values.values()), operands["category"]) |
| |
| def collect_opcodes(spirv): |
| seen = set() |
| values = [] |
| for x in spirv["instructions"]: |
| opcode = x["opcode"] |
| assert(opcode not in seen) |
| seen.add(opcode) |
| |
| name = x["opname"] |
| assert name.startswith("Op") |
| values.append([name[2:]]) |
| |
| return ("Op", values, None) |
| |
| def parse_args(): |
| p = argparse.ArgumentParser() |
| p.add_argument('--out-c', required=True, help='Output C file.') |
| p.add_argument('--out-h', required=True, help='Output H file.') |
| p.add_argument('--json', required=True, help='SPIR-V JSON file.') |
| return p.parse_args() |
| |
| TEMPLATE_H = Template("""\ |
| /* DO NOT EDIT - This file is generated automatically by spirv_info_c.py script */ |
| |
| """ + COPYRIGHT + """\ |
| |
| #ifndef _SPIRV_INFO_H_ |
| #define _SPIRV_INFO_H_ |
| |
| #include <stdbool.h> |
| |
| #include "compiler/spirv/spirv.h" |
| |
| % for kind,values,category in info: |
| % if kind == "Capability": |
| struct spirv_capabilities { |
| % for names in values: |
| % if len(names) == 1: |
| bool ${names[0]}; |
| % else: |
| union { |
| % for name in names: |
| bool ${name}; |
| % endfor |
| }; |
| % endif |
| % endfor |
| }; |
| % endif |
| % endfor |
| |
| bool spirv_capabilities_get(const struct spirv_capabilities *caps, |
| SpvCapability cap); |
| void spirv_capabilities_set(struct spirv_capabilities *caps, |
| SpvCapability cap, bool enabled); |
| |
| % for kind,values,category in info: |
| % if category == "BitEnum": |
| const char *spirv_${kind.lower()}_to_string(Spv${kind}Mask v); |
| % else: |
| const char *spirv_${kind.lower()}_to_string(Spv${kind} v); |
| % endif |
| % endfor |
| |
| #endif /* SPIRV_INFO_H */ |
| """) |
| |
| TEMPLATE_C = Template("""\ |
| /* DO NOT EDIT - This file is generated automatically by spirv_info_c.py script */ |
| |
| """ + COPYRIGHT + """\ |
| #include "spirv_info.h" |
| |
| #include "util/macros.h" |
| |
| % for kind,values,category in info: |
| % if kind == "Capability": |
| bool |
| spirv_capabilities_get(const struct spirv_capabilities *caps, |
| SpvCapability cap) |
| { |
| switch (cap) { |
| % for names in values: |
| case SpvCapability${names[0]}: return caps->${names[0]}; |
| % endfor |
| default: |
| return false; |
| } |
| } |
| |
| void |
| spirv_capabilities_set(struct spirv_capabilities *caps, |
| SpvCapability cap, bool enabled) |
| { |
| switch (cap) { |
| % for names in values: |
| case SpvCapability${names[0]}: caps->${names[0]} = enabled; break; |
| % endfor |
| default: |
| unreachable("Unknown capability"); |
| } |
| } |
| % endif |
| % endfor |
| |
| % for kind,values,category in info: |
| |
| % if category == "BitEnum": |
| const char * |
| spirv_${kind.lower()}_to_string(Spv${kind}Mask v) |
| { |
| switch (v) { |
| % for names in values: |
| %if names[0] != "None": |
| case Spv${kind}${names[0]}Mask: return "Spv${kind}${names[0]}"; |
| % else: |
| case Spv${kind}MaskNone: return "Spv${kind}${names[0]}"; |
| % endif |
| % endfor |
| } |
| |
| return "unknown"; |
| } |
| % else: |
| const char * |
| spirv_${kind.lower()}_to_string(Spv${kind} v) |
| { |
| switch (v) { |
| % for names in values: |
| case Spv${kind}${names[0]}: return "Spv${kind}${names[0]}"; |
| % endfor |
| case Spv${kind}Max: break; /* silence warnings about unhandled enums. */ |
| } |
| |
| return "unknown"; |
| } |
| % endif |
| % endfor |
| """) |
| |
| if __name__ == "__main__": |
| pargs = parse_args() |
| |
| spirv_info = json.JSONDecoder().decode(open(pargs.json, "r").read()) |
| |
| info = [ |
| collect_data(spirv_info, "AddressingModel"), |
| collect_data(spirv_info, "BuiltIn"), |
| collect_data(spirv_info, "Capability"), |
| collect_data(spirv_info, "Decoration"), |
| collect_data(spirv_info, "Dim"), |
| collect_data(spirv_info, "ExecutionMode"), |
| collect_data(spirv_info, "ExecutionModel"), |
| collect_data(spirv_info, "FPRoundingMode"), |
| collect_data(spirv_info, "FunctionParameterAttribute"), |
| collect_data(spirv_info, "ImageFormat"), |
| collect_data(spirv_info, "ImageOperands"), |
| collect_data(spirv_info, "MemoryModel"), |
| collect_data(spirv_info, "StorageClass"), |
| collect_opcodes(spirv_info), |
| ] |
| |
| with open(pargs.out_h, 'w', encoding='utf-8') as f: |
| f.write(TEMPLATE_H.render(info=info)) |
| with open(pargs.out_c, 'w', encoding='utf-8') as f: |
| f.write(TEMPLATE_C.render(info=info)) |