3 # Copyright © 2019-2024 Google, Inc.
5 # SPDX-License-Identifier: MIT
7 import xml.parsers.expat
15 class Error(Exception):
16 def __init__(self, message):
17 self.message = message
20 def __init__(self, name):
24 def has_name(self, name):
25 for (n, value) in self.values:
31 return [n for (n, value) in self.values]
35 for (name, value) in self.values:
39 print("enum %s {" % self.name)
40 for (name, value) in self.values:
42 print("\t%s = 0x%08x," % (name, value))
44 print("\t%s = %d," % (name, value))
47 def dump_pack_struct(self):
51 def __init__(self, name, low, high, shr, type, parser):
58 builtin_types = [ None, "a3xx_regid", "boolean", "uint", "hex", "int", "fixed", "ufixed", "float", "address", "waddress" ]
60 maxpos = parser.current_bitsize - 1
62 if low < 0 or low > maxpos:
63 raise parser.error("low attribute out of range: %d" % low)
64 if high < 0 or high > maxpos:
65 raise parser.error("high attribute out of range: %d" % high)
67 raise parser.error("low is greater than high: low=%d, high=%d" % (low, high))
68 if self.type == "boolean" and not low == high:
69 raise parser.error("booleans should be 1 bit fields")
70 elif self.type == "float" and not (high - low == 31 or high - low == 15):
71 raise parser.error("floats should be 16 or 32 bit fields")
72 elif not self.type in builtin_types and not self.type in parser.enums:
73 raise parser.error("unknown type '%s'" % self.type)
75 def ctype(self, var_name):
79 elif self.type == "boolean":
82 elif self.type == "uint" or self.type == "hex" or self.type == "a3xx_regid":
85 elif self.type == "int":
88 elif self.type == "fixed":
90 val = "((int32_t)(%s * %d.0))" % (var_name, 1 << self.radix)
91 elif self.type == "ufixed":
93 val = "((uint32_t)(%s * %d.0))" % (var_name, 1 << self.radix)
94 elif self.type == "float" and self.high - self.low == 31:
96 val = "fui(%s)" % var_name
97 elif self.type == "float" and self.high - self.low == 15:
99 val = "_mesa_float_to_half(%s)" % var_name
100 elif self.type in [ "address", "waddress" ]:
104 type = "enum %s" % self.type
108 val = "(%s >> %d)" % (val, self.shr)
112 def tab_to(name, value):
113 tab_count = (68 - (len(name) & ~7)) // 8
116 print(name + ('\t' * tab_count) + value)
119 return ((0xffffffffffffffff >> (64 - (high + 1 - low))) << low)
121 def field_name(reg, f):
123 name = f.name.lower()
125 # We hit this path when a reg is defined with no bitset fields, ie.
126 # <reg32 offset="0x88db" name="RB_BLIT_DST_ARRAY_PITCH" low="0" high="28" shr="6" type="uint"/>
127 name = reg.name.lower()
129 if (name in [ "double", "float", "int" ]) or not (name[0].isalpha()):
134 # indices - array of (ctype, stride, __offsets_NAME)
135 def indices_varlist(indices):
136 return ", ".join(["i%d" % i for i in range(len(indices))])
138 def indices_prototype(indices):
139 return ", ".join(["%s i%d" % (ctype, idx)
140 for (idx, (ctype, stride, offset)) in enumerate(indices)])
142 def indices_strides(indices):
143 return " + ".join(["0x%x*i%d" % (stride, idx)
145 "%s(i%d)" % (offset, idx)
146 for (idx, (ctype, stride, offset)) in enumerate(indices)])
148 class Bitset(object):
149 def __init__(self, name, template):
153 self.fields = template.fields[:]
157 # Get address field if there is one in the bitset, else return None:
158 def get_address_field(self):
159 for f in self.fields:
160 if f.type in [ "address", "waddress" ]:
164 def dump_regpair_builder(self, reg):
165 print("#ifndef NDEBUG")
167 for f in self.fields:
168 known_mask |= mask(f.low, f.high)
169 if f.type in [ "boolean", "address", "waddress" ]:
171 type, val = f.ctype("fields.%s" % field_name(reg, f))
172 print(" assert((%-40s & 0x%08x) == 0);" % (val, 0xffffffff ^ mask(0 , f.high - f.low)))
173 print(" assert((%-40s & 0x%08x) == 0);" % ("fields.unknown", known_mask))
176 print(" return (struct fd_reg_pair) {")
178 print(" .reg = REG_%s(__i)," % reg.full_name)
180 print(" .reg = REG_%s," % reg.full_name)
183 for f in self.fields:
184 if f.type in [ "address", "waddress" ]:
187 type, val = f.ctype("fields.%s" % field_name(reg, f))
188 print(" (%-40s << %2d) |" % (val, f.low))
190 if reg.bit_size == 64:
192 print(" fields.unknown | fields.%s," % (value_name,))
194 address = self.get_address_field()
196 print(" .bo = fields.bo,")
197 print(" .is_address = true,")
198 if f.type == "waddress":
199 print(" .bo_write = true,")
200 print(" .bo_offset = fields.bo_offset,")
201 print(" .bo_shift = %d," % address.shr)
202 print(" .bo_low = %d," % address.low)
206 def dump_pack_struct(self, reg=None):
210 prefix = reg.full_name
212 print("struct %s {" % prefix)
213 for f in self.fields:
214 if f.type in [ "address", "waddress" ]:
215 tab_to(" __bo_type", "bo;")
216 tab_to(" uint32_t", "bo_offset;")
218 name = field_name(reg, f)
220 type, val = f.ctype("var")
222 tab_to(" %s" % type, "%s;" % name)
223 if reg.bit_size == 64:
224 tab_to(" uint64_t", "unknown;")
225 tab_to(" uint64_t", "qword;")
227 tab_to(" uint32_t", "unknown;")
228 tab_to(" uint32_t", "dword;")
232 print("static inline struct fd_reg_pair\npack_%s(uint32_t __i, struct %s fields)\n{" %
235 print("static inline struct fd_reg_pair\npack_%s(struct %s fields)\n{" %
238 self.dump_regpair_builder(reg)
242 if self.get_address_field():
243 skip = ", { .reg = 0 }"
248 print("#define %s(__i, ...) pack_%s(__i, __struct_cast(%s) { __VA_ARGS__ })%s\n" %
249 (prefix, prefix, prefix, skip))
251 print("#define %s(...) pack_%s(__struct_cast(%s) { __VA_ARGS__ })%s\n" %
252 (prefix, prefix, prefix, skip))
255 def dump(self, prefix=None):
258 for f in self.fields:
260 name = prefix + "_" + f.name
264 if not f.name and f.low == 0 and f.shr == 0 and not f.type in ["float", "fixed", "ufixed"]:
266 elif f.type == "boolean" or (f.type == None and f.low == f.high):
267 tab_to("#define %s" % name, "0x%08x" % (1 << f.low))
269 tab_to("#define %s__MASK" % name, "0x%08x" % mask(f.low, f.high))
270 tab_to("#define %s__SHIFT" % name, "%d" % f.low)
271 type, val = f.ctype("val")
273 print("static inline uint32_t %s(%s val)\n{" % (name, type))
275 print("\tassert(!(val & 0x%x));" % mask(0, f.shr - 1))
276 print("\treturn ((%s) << %s__SHIFT) & %s__MASK;\n}" % (val, name, name))
280 def __init__(self, attrs, domain, variant, parent, index_type):
282 self.local_name = attrs["name"]
286 self.variant = variant
289 self.name = self.parent.name + "_" + self.local_name
291 self.name = self.local_name
292 if "offsets" in attrs:
293 self.offsets = map(lambda i: "0x%08x" % int(i, 0), attrs["offsets"].split(","))
294 self.fixed_offsets = True
295 elif "doffsets" in attrs:
296 self.offsets = map(lambda s: "(%s)" % s , attrs["doffsets"].split(","))
297 self.fixed_offsets = True
299 self.offset = int(attrs["offset"], 0)
300 self.stride = int(attrs["stride"], 0)
301 self.fixed_offsets = False
303 self.index_type = index_type
305 self.index_type = None
306 self.length = int(attrs["length"], 0)
308 self.usages = attrs["usage"].split(',')
312 def index_ctype(self):
313 if not self.index_type:
316 return "enum %s" % self.index_type.name
318 # Generate array of (ctype, stride, __offsets_NAME)
321 indices = self.parent.indices()
325 if self.fixed_offsets:
326 indices.append((self.index_ctype(), None, "__offset_%s" % self.local_name))
328 indices.append((self.index_ctype(), self.stride, None))
331 def total_offset(self):
333 if not self.fixed_offsets:
334 offset += self.offset
336 offset += self.parent.total_offset()
340 proto = indices_varlist(self.indices())
341 strides = indices_strides(self.indices())
342 array_offset = self.total_offset()
343 if self.fixed_offsets:
344 print("static inline uint32_t __offset_%s(%s idx)" % (self.local_name, self.index_ctype()))
345 print("{\n\tswitch (idx) {")
347 for val, offset in zip(self.index_type.names(), self.offsets):
348 print("\t\tcase %s: return %s;" % (val, offset))
350 for idx, offset in enumerate(self.offsets):
351 print("\t\tcase %d: return %s;" % (idx, offset))
352 print("\t\tdefault: return INVALID_IDX(idx);")
355 tab_to("#define REG_%s_%s" % (self.domain, self.name), "0x%08x\n" % array_offset)
357 tab_to("#define REG_%s_%s(%s)" % (self.domain, self.name, proto), "(0x%08x + %s )\n" % (array_offset, strides))
359 def dump_pack_struct(self):
362 def dump_regpair_builder(self):
366 def __init__(self, attrs, domain, array, bit_size):
367 self.name = attrs["name"]
370 self.offset = int(attrs["offset"], 0)
372 self.bit_size = bit_size
374 self.name = array.name + "_" + self.name
375 self.full_name = self.domain + "_" + self.name
376 if "stride" in attrs:
377 self.stride = int(attrs["stride"], 0)
378 self.length = int(attrs["length"], 0)
383 # Generate array of (ctype, stride, __offsets_NAME)
386 indices = self.array.indices()
390 indices.append(("uint32_t", self.stride, None))
393 def total_offset(self):
395 return self.array.total_offset() + self.offset
400 proto = indices_prototype(self.indices())
401 strides = indices_strides(self.indices())
402 offset = self.total_offset()
404 tab_to("#define REG_%s" % self.full_name, "0x%08x" % offset)
406 print("static inline uint32_t REG_%s(%s) { return 0x%08x + %s; }" % (self.full_name, proto, offset, strides))
408 if self.bitset.inline:
409 self.bitset.dump(self.full_name)
411 def dump_pack_struct(self):
412 if self.bitset.inline:
413 self.bitset.dump_pack_struct(self)
415 def dump_regpair_builder(self):
416 if self.bitset.inline:
417 self.bitset.dump_regpair_builder(self)
420 print("\tREG_%s = 0x%08x" % (self.full_name, self.offset))
423 class Parser(object):
425 self.current_array = None
426 self.current_domain = None
427 self.current_prefix = None
428 self.current_prefix_type = None
429 self.current_stripe = None
430 self.current_bitset = None
431 self.current_bitsize = 32
432 # The varset attribute on the domain specifies the enum which
433 # specifies all possible hw variants:
434 self.current_varset = None
435 # Regs that have multiple variants.. we only generated the C++
436 # template based struct-packers for these
437 self.variant_regs = {}
438 # Information in which contexts regs are used, to be used in
440 self.usage_regs = collections.defaultdict(list)
443 self.variants = set()
446 self.copyright_year = None
450 def error(self, message):
451 parser, filename = self.stack[-1]
452 return Error("%s:%d:%d: %s" % (filename, parser.CurrentLineNumber, parser.CurrentColumnNumber, message))
454 def prefix(self, variant=None):
455 if self.current_prefix_type == "variant" and variant:
457 elif self.current_stripe:
458 return self.current_stripe + "_" + self.current_domain
459 elif self.current_prefix:
460 return self.current_prefix + "_" + self.current_domain
462 return self.current_domain
464 def parse_field(self, name, attrs):
467 high = low = int(attrs["pos"], 0)
468 elif "high" in attrs and "low" in attrs:
469 high = int(attrs["high"], 0)
470 low = int(attrs["low"], 0)
473 high = self.current_bitsize - 1
481 shr = int(attrs["shr"], 0)
485 b = Field(name, low, high, shr, type, self)
487 if type == "fixed" or type == "ufixed":
488 b.radix = int(attrs["radix"], 0)
490 self.current_bitset.fields.append(b)
491 except ValueError as e:
494 def parse_varset(self, attrs):
495 # Inherit the varset from the enclosing domain if not overriden:
496 varset = self.current_varset
497 if "varset" in attrs:
498 varset = self.enums[attrs["varset"]]
501 def parse_variants(self, attrs):
502 if not "variants" in attrs:
504 variant = attrs["variants"].split(",")[0]
506 variant = variant[:variant.index("-")]
508 varset = self.parse_varset(attrs)
510 assert varset.has_name(variant)
514 def add_all_variants(self, reg, attrs, parent_variant):
515 # TODO this should really handle *all* variants, including dealing
516 # with open ended ranges (ie. "A2XX,A4XX-") (we have the varset
517 # enum now to make that possible)
518 variant = self.parse_variants(attrs)
520 variant = parent_variant
522 if reg.name not in self.variant_regs:
523 self.variant_regs[reg.name] = {}
525 # All variants must be same size:
526 v = next(iter(self.variant_regs[reg.name]))
527 assert self.variant_regs[reg.name][v].bit_size == reg.bit_size
529 self.variant_regs[reg.name][variant] = reg
531 def add_all_usages(self, reg, usages):
536 self.usage_regs[usage].append(reg)
538 self.variants.add(reg.domain)
540 def do_validate(self, schemafile):
541 if not self.validate:
545 from lxml import etree
547 parser, filename = self.stack[-1]
548 dirname = os.path.dirname(filename)
550 # we expect this to look like <namespace url> schema.xsd.. I think
551 # technically it is supposed to be just a URL, but that doesn't
552 # quite match up to what we do.. Just skip over everything up to
553 # and including the first whitespace character:
554 schemafile = schemafile[schemafile.rindex(" ")+1:]
556 # this is a bit cheezy, but the xml file to validate could be
557 # in a child director, ie. we don't really know where the schema
558 # file is, the way the rnn C code does. So if it doesn't exist
559 # just look one level up
560 if not os.path.exists(dirname + "/" + schemafile):
561 schemafile = "../" + schemafile
563 if not os.path.exists(dirname + "/" + schemafile):
564 raise self.error("Cannot find schema for: " + filename)
566 xmlschema_doc = etree.parse(dirname + "/" + schemafile)
567 xmlschema = etree.XMLSchema(xmlschema_doc)
569 xml_doc = etree.parse(filename)
570 if not xmlschema.validate(xml_doc):
571 error_str = str(xmlschema.error_log.filter_from_errors()[0])
572 raise self.error("Schema validation failed for: " + filename + "\n" + error_str)
573 except ImportError as e:
577 print("lxml not found, skipping validation", file=sys.stderr)
579 def do_parse(self, filename):
580 filepath = os.path.abspath(filename)
581 if filepath in self.xml_files:
583 self.xml_files.append(filepath)
584 file = open(filename, "rb")
585 parser = xml.parsers.expat.ParserCreate()
586 self.stack.append((parser, filename))
587 parser.StartElementHandler = self.start_element
588 parser.EndElementHandler = self.end_element
589 parser.CharacterDataHandler = self.character_data
590 parser.buffer_text = True
591 parser.ParseFile(file)
595 def parse(self, rnn_path, filename, validate):
598 self.validate = validate
599 self.do_parse(filename)
601 def parse_reg(self, attrs, bit_size):
602 self.current_bitsize = bit_size
603 if "type" in attrs and attrs["type"] in self.bitsets:
604 bitset = self.bitsets[attrs["type"]]
606 self.current_bitset = Bitset(attrs["name"], bitset)
607 self.current_bitset.inline = True
609 self.current_bitset = bitset
611 self.current_bitset = Bitset(attrs["name"], None)
612 self.current_bitset.inline = True
614 self.parse_field(None, attrs)
616 variant = self.parse_variants(attrs)
617 if not variant and self.current_array:
618 variant = self.current_array.variant
620 self.current_reg = Reg(attrs, self.prefix(variant), self.current_array, bit_size)
621 self.current_reg.bitset = self.current_bitset
623 if len(self.stack) == 1:
624 self.file.append(self.current_reg)
626 if variant is not None:
627 self.add_all_variants(self.current_reg, attrs, variant)
631 usages = attrs["usage"].split(',')
632 elif self.current_array:
633 usages = self.current_array.usages
635 self.add_all_usages(self.current_reg, usages)
637 def start_element(self, name, attrs):
640 filename = attrs["file"]
641 self.do_parse(os.path.join(self.path, filename))
642 elif name == "domain":
643 self.current_domain = attrs["name"]
644 if "prefix" in attrs:
645 self.current_prefix = self.parse_variants(attrs)
646 self.current_prefix_type = attrs["prefix"]
648 self.current_prefix = None
649 self.current_prefix_type = None
650 if "varset" in attrs:
651 self.current_varset = self.enums[attrs["varset"]]
652 elif name == "stripe":
653 self.current_stripe = self.parse_variants(attrs)
655 self.current_enum_value = 0
656 self.current_enum = Enum(attrs["name"])
657 self.enums[attrs["name"]] = self.current_enum
658 if len(self.stack) == 1:
659 self.file.append(self.current_enum)
660 elif name == "value":
662 value = int(attrs["value"], 0)
664 value = self.current_enum_value
665 self.current_enum.values.append((attrs["name"], value))
666 elif name == "reg32":
667 self.parse_reg(attrs, 32)
668 elif name == "reg64":
669 self.parse_reg(attrs, 64)
670 elif name == "array":
671 self.current_bitsize = 32
672 variant = self.parse_variants(attrs)
673 index_type = self.enums[attrs["index"]] if "index" in attrs else None
674 self.current_array = Array(attrs, self.prefix(variant), variant, self.current_array, index_type)
675 if len(self.stack) == 1:
676 self.file.append(self.current_array)
677 elif name == "bitset":
678 self.current_bitset = Bitset(attrs["name"], None)
679 if "inline" in attrs and attrs["inline"] == "yes":
680 self.current_bitset.inline = True
681 self.bitsets[self.current_bitset.name] = self.current_bitset
682 if len(self.stack) == 1 and not self.current_bitset.inline:
683 self.file.append(self.current_bitset)
684 elif name == "bitfield" and self.current_bitset:
685 self.parse_field(attrs["name"], attrs)
686 elif name == "database":
687 self.do_validate(attrs["xsi:schemaLocation"])
688 elif name == "copyright":
689 self.copyright_year = attrs["year"]
690 elif name == "author":
691 self.authors.append(attrs["name"] + " <" + attrs["email"] + "> " + attrs["name"])
693 def end_element(self, name):
695 self.current_domain = None
696 self.current_prefix = None
697 self.current_prefix_type = None
698 elif name == "stripe":
699 self.current_stripe = None
700 elif name == "bitset":
701 self.current_bitset = None
702 elif name == "reg32":
703 self.current_reg = None
704 elif name == "array":
705 self.current_array = self.current_array.parent
707 self.current_enum = None
708 elif name == "license":
709 self.license = self.cdata
711 def character_data(self, data):
714 def dump_reg_usages(self):
715 d = collections.defaultdict(list)
716 for usage, regs in self.usage_regs.items():
718 variants = self.variant_regs.get(reg.name)
720 for variant, vreg in variants.items():
722 d[(usage, variant)].append(reg)
724 for variant in self.variants:
725 d[(usage, variant)].append(reg)
727 print("#ifdef __cplusplus")
729 for usage, regs in self.usage_regs.items():
730 print("template<chip CHIP> constexpr inline uint16_t %s_REGS[] = {};" % (usage.upper()))
732 for (usage, variant), regs in d.items():
737 for i in range(reg.array.length):
738 offsets.append(reg.array.offset + reg.offset + i * reg.array.stride)
739 if reg.bit_size == 64:
740 offsets.append(offsets[-1] + 1)
742 offsets.append(reg.offset)
743 if reg.bit_size == 64:
744 offsets.append(offsets[-1] + 1)
748 print("template<> constexpr inline uint16_t %s_REGS<%s>[] = {" % (usage.upper(), variant))
749 for offset in offsets:
750 print("\t%s," % hex(offset))
760 if isinstance(e, Enum):
762 elif isinstance(e, Bitset):
767 for e in enums + bitsets + regs:
770 self.dump_reg_usages()
773 def dump_regs_py(self):
776 if isinstance(e, Reg):
783 def dump_reg_variants(self, regname, variants):
784 # Don't bother for things that only have a single variant:
785 if len(variants) == 1:
787 print("#ifdef __cplusplus")
788 print("struct __%s {" % regname)
789 # TODO be more clever.. we should probably figure out which
790 # fields have the same type in all variants (in which they
791 # appear) and stuff everything else in a variant specific
797 for variant in variants.keys():
798 print(" /* %s fields: */" % variant)
799 reg = variants[variant]
800 bit_size = reg.bit_size
802 for f in reg.bitset.fields:
803 fld_name = field_name(reg, f)
804 if fld_name in seen_fields:
806 seen_fields.append(fld_name)
807 name = fld_name.lower()
808 if f.type in [ "address", "waddress" ]:
812 tab_to(" __bo_type", "bo;")
813 tab_to(" uint32_t", "bo_offset;")
815 type, val = f.ctype("var")
816 tab_to(" %s" %type, "%s;" %name)
817 print(" /* fallback fields: */")
819 tab_to(" uint64_t", "unknown;")
820 tab_to(" uint64_t", "qword;")
822 tab_to(" uint32_t", "unknown;")
823 tab_to(" uint32_t", "dword;")
825 # TODO don't hardcode the varset enum name
827 print("template <%s %s>" % (varenum, varenum.upper()))
828 print("static inline struct fd_reg_pair")
834 print("__%s(%sstruct __%s fields) {" % (regname, xtra, regname))
835 for variant in variants.keys():
836 print(" if (%s == %s) {" % (varenum.upper(), variant))
837 reg = variants[variant]
838 reg.dump_regpair_builder()
840 print(" assert(!\"invalid variant\");")
844 skip = ", { .reg = 0 }"
848 print("#define %s(VARIANT, %s...) __%s<VARIANT>(%s{__VA_ARGS__})%s" % (regname, xtravar, regname, xtravar, skip))
849 print("#endif /* __cplusplus */")
851 def dump_structs(self):
855 for regname in self.variant_regs:
856 self.dump_reg_variants(regname, self.variant_regs[regname])
859 def dump_c(args, guard, func):
863 p.parse(args.rnn, args.xml, args.validate)
865 print(e, file=sys.stderr)
868 print("#ifndef %s\n#define %s\n" % (guard, guard))
870 print("""/* Autogenerated file, DO NOT EDIT manually!
872 This file was generated by the rules-ng-ng gen_header.py tool in this git repository:
873 http://gitlab.freedesktop.org/mesa/mesa/
874 git clone https://gitlab.freedesktop.org/mesa/mesa.git
876 The rules-ng-ng source files this header was generated from are:
879 for filepath in p.xml_files:
880 maxlen = max(maxlen, len(filepath))
881 for filepath in p.xml_files:
882 pad = " " * (maxlen - len(filepath))
883 filesize = str(os.path.getsize(filepath))
884 filesize = " " * (7 - len(filesize)) + filesize
885 filetime = time.ctime(os.path.getmtime(filepath))
886 print("- " + filepath + pad + " (" + filesize + " bytes, from " + filetime + ")")
888 current_year = str(datetime.date.today().year)
890 print("Copyright (C) %s-%s by the following authors:" % (p.copyright_year, current_year))
891 for author in p.authors:
898 print("#ifdef __KERNEL__")
899 print("#include <linux/bug.h>")
900 print("#define assert(x) BUG_ON(!(x))")
902 print("#include <assert.h>")
906 print("#ifdef __cplusplus")
907 print("#define __struct_cast(X)")
909 print("#define __struct_cast(X) (struct X)")
915 print("\n#endif /* %s */" % guard)
918 def dump_c_defines(args):
919 guard = str.replace(os.path.basename(args.xml), '.', '_').upper()
920 dump_c(args, guard, lambda p: p.dump())
923 def dump_c_pack_structs(args):
924 guard = str.replace(os.path.basename(args.xml), '.', '_').upper() + '_STRUCTS'
925 dump_c(args, guard, lambda p: p.dump_structs())
928 def dump_py_defines(args):
932 p.parse(args.rnn, args.xml)
934 print(e, file=sys.stderr)
937 file_name = os.path.splitext(os.path.basename(args.xml))[0]
939 print("from enum import IntEnum")
940 print("class %sRegs(IntEnum):" % file_name.upper())
942 os.path.basename(args.xml)
948 parser = argparse.ArgumentParser()
949 parser.add_argument('--rnn', type=str, required=True)
950 parser.add_argument('--xml', type=str, required=True)
951 parser.add_argument('--validate', default=False, action='store_true')
952 parser.add_argument('--no-validate', dest='validate', action='store_false')
954 subparsers = parser.add_subparsers()
955 subparsers.required = True
957 parser_c_defines = subparsers.add_parser('c-defines')
958 parser_c_defines.set_defaults(func=dump_c_defines)
960 parser_c_pack_structs = subparsers.add_parser('c-pack-structs')
961 parser_c_pack_structs.set_defaults(func=dump_c_pack_structs)
963 parser_py_defines = subparsers.add_parser('py-defines')
964 parser_py_defines.set_defaults(func=dump_py_defines)
966 args = parser.parse_args()
970 if __name__ == '__main__':