generate_datatypes.py 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582
  1. #!/usr/bin/env python
  2. # This Source Code Form is subject to the terms of the Mozilla Public
  3. # License, v. 2.0. If a copy of the MPL was not distributed with this
  4. # file, You can obtain one at http://mozilla.org/MPL/2.0/.
  5. from __future__ import print_function
  6. import sys
  7. import time
  8. import platform
  9. import getpass
  10. from collections import OrderedDict
  11. import re
  12. import xml.etree.ElementTree as etree
  13. import itertools
  14. import argparse
  15. from nodeset_compiler.opaque_type_mapping import get_base_type_for_opaque
  16. types = OrderedDict() # contains types that were already parsed
  17. typedescriptions = {} # contains type nodeids
  18. excluded_types = ["NodeIdType", "InstanceNode", "TypeNode", "Node", "ObjectNode",
  19. "ObjectTypeNode", "VariableNode", "VariableTypeNode", "ReferenceTypeNode",
  20. "MethodNode", "ViewNode", "DataTypeNode",
  21. "NumericRange", "NumericRangeDimensions",
  22. "UA_ServerDiagnosticsSummaryDataType", "UA_SamplingIntervalDiagnosticsDataType",
  23. "UA_SessionSecurityDiagnosticsDataType", "UA_SubscriptionDiagnosticsDataType",
  24. "UA_SessionDiagnosticsDataType"]
  25. builtin_types = ["Boolean", "SByte", "Byte", "Int16", "UInt16", "Int32", "UInt32",
  26. "Int64", "UInt64", "Float", "Double", "String", "DateTime", "Guid",
  27. "ByteString", "XmlElement", "NodeId", "ExpandedNodeId", "StatusCode",
  28. "QualifiedName", "LocalizedText", "ExtensionObject", "DataValue",
  29. "Variant", "DiagnosticInfo"]
  30. # If the type does not contain pointers, it can be copied with memcpy
  31. # (internally, not into the protocol message). This dict contains the sizes of
  32. # pointer-free types. Parsed types are added if they apply.
  33. builtin_pointerfree = ["Boolean", "SByte", "Byte", "Int16", "UInt16", "Int32", "UInt32",
  34. "Int64", "UInt64", "Float", "Double", "DateTime", "Guid", "StatusCode"]
  35. # Some types can be memcpy'd off the binary stream. That's especially important
  36. # for arrays. But we need to check if they contain padding and whether the
  37. # endianness is correct. This dict gives the C-statement that must be true for the
  38. # type to be overlayable. Parsed types are added if they apply.
  39. builtin_overlayable = {"Boolean": "true", "SByte": "true", "Byte": "true",
  40. "Int16": "UA_BINARY_OVERLAYABLE_INTEGER", "UInt16": "UA_BINARY_OVERLAYABLE_INTEGER",
  41. "Int32": "UA_BINARY_OVERLAYABLE_INTEGER", "UInt32": "UA_BINARY_OVERLAYABLE_INTEGER",
  42. "Int64": "UA_BINARY_OVERLAYABLE_INTEGER", "UInt64": "UA_BINARY_OVERLAYABLE_INTEGER",
  43. "Float": "UA_BINARY_OVERLAYABLE_FLOAT", "Double": "UA_BINARY_OVERLAYABLE_FLOAT",
  44. "DateTime": "UA_BINARY_OVERLAYABLE_INTEGER", "StatusCode": "UA_BINARY_OVERLAYABLE_INTEGER",
  45. "Guid": "(UA_BINARY_OVERLAYABLE_INTEGER && offsetof(UA_Guid, data2) == sizeof(UA_UInt32) && " + \
  46. "offsetof(UA_Guid, data3) == (sizeof(UA_UInt16) + sizeof(UA_UInt32)) && " + \
  47. "offsetof(UA_Guid, data4) == (2*sizeof(UA_UInt32)))"}
  48. ################
  49. # Type Classes #
  50. ################
  51. class StructMember(object):
  52. def __init__(self, name, memberType, isArray):
  53. self.name = name
  54. self.memberType = memberType
  55. self.isArray = isArray
  56. class Type(object):
  57. def __init__(self, outname, xml, namespace):
  58. self.name = xml.get("Name")
  59. self.ns0 = ("true" if namespace == 0 else "false")
  60. self.typeIndex = outname.upper() + "_" + self.name.upper()
  61. self.outname = outname
  62. self.description = ""
  63. self.pointerfree = "false"
  64. self.overlayable = "false"
  65. if self.name in builtin_types:
  66. self.builtin = "true"
  67. else:
  68. self.builtin = "false"
  69. self.members = [StructMember("", self, False)] # Returns one member: itself. Overwritten by some types.
  70. for child in xml:
  71. if child.tag == "{http://opcfoundation.org/BinarySchema/}Documentation":
  72. self.description = child.text
  73. break
  74. def datatype_c(self):
  75. xmlEncodingId = "0"
  76. binaryEncodingId = "0"
  77. if self.name in typedescriptions:
  78. description = typedescriptions[self.name]
  79. typeid = "{%s, UA_NODEIDTYPE_NUMERIC, {%s}}" % (description.namespaceid, description.nodeid)
  80. xmlEncodingId = description.xmlEncodingId
  81. binaryEncodingId = description.binaryEncodingId
  82. else:
  83. typeid = "{0, UA_NODEIDTYPE_NUMERIC, {0}}"
  84. return "{\n UA_TYPENAME(\"%s\") /* .typeName */\n" % self.name + \
  85. " " + typeid + ", /* .typeId */\n" + \
  86. " sizeof(UA_" + self.name + "), /* .memSize */\n" + \
  87. " " + self.typeIndex + ", /* .typeIndex */\n" + \
  88. " " + str(len(self.members)) + ", /* .membersSize */\n" + \
  89. " " + self.builtin + ", /* .builtin */\n" + \
  90. " " + self.pointerfree + ", /* .pointerFree */\n" + \
  91. " " + self.overlayable + ", /* .overlayable */ \n" + \
  92. " " + binaryEncodingId + ", /* .binaryEncodingId */\n" + \
  93. " %s_members" % self.name + " /* .members */\n}"
  94. def members_c(self):
  95. if len(self.members)==0:
  96. return "#define %s_members NULL" % (self.name)
  97. members = "static UA_DataTypeMember %s_members[%s] = {" % (self.name, len(self.members))
  98. before = None
  99. for index, member in enumerate(self.members):
  100. m = "\n{\n UA_TYPENAME(\"%s\") /* .memberName */\n" % member.name
  101. m += " %s_%s, /* .memberTypeIndex */\n" % (member.memberType.outname.upper(), member.memberType.name.upper())
  102. m += " "
  103. if not before:
  104. m += "0,"
  105. else:
  106. if member.isArray:
  107. m += "offsetof(UA_%s, %sSize)" % (self.name, member.name)
  108. else:
  109. m += "offsetof(UA_%s, %s)" % (self.name, member.name)
  110. m += " - offsetof(UA_%s, %s)" % (self.name, before.name)
  111. if before.isArray:
  112. m += " - sizeof(void*),"
  113. else:
  114. m += " - sizeof(UA_%s)," % before.memberType.name
  115. m += " /* .padding */\n"
  116. m += " %s, /* .namespaceZero */\n" % member.memberType.ns0
  117. m += (" true" if member.isArray else " false") + " /* .isArray */\n},"
  118. members += m
  119. before = member
  120. return members + "};"
  121. def datatype_ptr(self):
  122. return "&" + self.outname.upper() + "[" + self.outname.upper() + "_" + self.name.upper() + "]"
  123. def functions_c(self):
  124. funcs = "static UA_INLINE void\nUA_%s_init(UA_%s *p) {\n memset(p, 0, sizeof(UA_%s));\n}\n\n" % (self.name, self.name, self.name)
  125. funcs += "static UA_INLINE UA_%s *\nUA_%s_new(void) {\n return (UA_%s*)UA_new(%s);\n}\n\n" % (self.name, self.name, self.name, self.datatype_ptr())
  126. if self.pointerfree == "true":
  127. funcs += "static UA_INLINE UA_StatusCode\nUA_%s_copy(const UA_%s *src, UA_%s *dst) {\n *dst = *src;\n return UA_STATUSCODE_GOOD;\n}\n\n" % (self.name, self.name, self.name)
  128. funcs += "static UA_INLINE void\nUA_%s_deleteMembers(UA_%s *p) { }\n\n" % (self.name, self.name)
  129. else:
  130. funcs += "static UA_INLINE UA_StatusCode\nUA_%s_copy(const UA_%s *src, UA_%s *dst) {\n return UA_copy(src, dst, %s);\n}\n\n" % (self.name, self.name, self.name, self.datatype_ptr())
  131. funcs += "static UA_INLINE void\nUA_%s_deleteMembers(UA_%s *p) {\n UA_deleteMembers(p, %s);\n}\n\n" % (self.name, self.name, self.datatype_ptr())
  132. funcs += "static UA_INLINE void\nUA_%s_delete(UA_%s *p) {\n UA_delete(p, %s);\n}" % (self.name, self.name, self.datatype_ptr())
  133. return funcs
  134. def encoding_h(self):
  135. enc = "static UA_INLINE UA_StatusCode\nUA_%s_encodeBinary(const UA_%s *src, UA_Byte **bufPos, const UA_Byte **bufEnd) {\n return UA_encodeBinary(src, %s, bufPos, bufEnd, NULL, NULL);\n}\n"
  136. enc += "static UA_INLINE UA_StatusCode\nUA_%s_decodeBinary(const UA_ByteString *src, size_t *offset, UA_%s *dst) {\n return UA_decodeBinary(src, offset, dst, %s, 0, NULL);\n}"
  137. return enc % tuple(list(itertools.chain(*itertools.repeat([self.name, self.name, self.datatype_ptr()], 2))))
  138. class BuiltinType(Type):
  139. def __init__(self, name):
  140. self.name = name
  141. self.ns0 = "true"
  142. self.typeIndex = "UA_TYPES_" + self.name.upper()
  143. self.outname = "ua_types"
  144. self.description = ""
  145. self.pointerfree = "false"
  146. if self.name in builtin_pointerfree:
  147. self.pointerfree = "true"
  148. self.overlayable = "false"
  149. if name in builtin_overlayable:
  150. self.overlayable = builtin_overlayable[name]
  151. self.builtin = "true"
  152. if self.name == "QualifiedName":
  153. self.members = [StructMember("namespaceIndex", types["Int16"], False), StructMember("name", types["String"], False)]
  154. elif self.name in ["String", "ByteString", "XmlElement"]:
  155. self.members = [StructMember("", types["Byte"], True)]
  156. else:
  157. self.members = [StructMember("", self, False)]
  158. class EnumerationType(Type):
  159. def __init__(self, outname, xml, namespace):
  160. Type.__init__(self, outname, xml, namespace)
  161. self.pointerfree = "true"
  162. self.overlayable = "UA_BINARY_OVERLAYABLE_INTEGER"
  163. self.members = [StructMember("", types["Int32"], False)] # encoded as uint32
  164. self.builtin = "true"
  165. self.typeIndex = "UA_TYPES_INT32"
  166. self.elements = OrderedDict()
  167. for child in xml:
  168. if child.tag == "{http://opcfoundation.org/BinarySchema/}EnumeratedValue":
  169. self.elements[child.get("Name")] = child.get("Value")
  170. def typedef_h(self):
  171. if sys.version_info[0] < 3:
  172. values = self.elements.iteritems()
  173. else:
  174. values = self.elements.items()
  175. return "typedef enum {\n " + ",\n ".join(map(lambda kv : "UA_" + self.name.upper() + "_" + kv[0].upper() + \
  176. " = " + kv[1], values)) + "\n} UA_%s;" % self.name
  177. class OpaqueType(Type):
  178. def __init__(self, outname, xml, namespace, baseType):
  179. Type.__init__(self, outname, xml, namespace)
  180. self.baseType = baseType
  181. self.members = [StructMember("", types[baseType], False)] # encoded as string
  182. def typedef_h(self):
  183. return "typedef UA_" + self.baseType + " UA_%s;" % self.name
  184. class StructType(Type):
  185. def __init__(self, outname, xml, namespace):
  186. Type.__init__(self, outname, xml, namespace)
  187. self.members = []
  188. lengthfields = [] # lengthfields of arrays are not included as members
  189. for child in xml:
  190. if child.get("LengthField"):
  191. lengthfields.append(child.get("LengthField"))
  192. for child in xml:
  193. if not child.tag == "{http://opcfoundation.org/BinarySchema/}Field":
  194. continue
  195. if child.get("Name") in lengthfields:
  196. continue
  197. memberName = child.get("Name")
  198. memberName = memberName[:1].lower() + memberName[1:]
  199. memberTypeName = child.get("TypeName")
  200. memberType = types[memberTypeName[memberTypeName.find(":")+1:]]
  201. isArray = True if child.get("LengthField") else False
  202. self.members.append(StructMember(memberName, memberType, isArray))
  203. self.pointerfree = "true"
  204. self.overlayable = "true"
  205. before = None
  206. for m in self.members:
  207. if m.isArray or m.memberType.pointerfree != "true":
  208. self.pointerfree = "false"
  209. self.overlayable = "false"
  210. else:
  211. self.overlayable += " && " + m.memberType.overlayable
  212. if before:
  213. self.overlayable += " && offsetof(UA_%s, %s) == (offsetof(UA_%s, %s) + sizeof(UA_%s))" % \
  214. (self.name, m.name, self.name, before.name, before.memberType.name)
  215. if "false" in self.overlayable:
  216. self.overlayable = "false"
  217. before = m
  218. def typedef_h(self):
  219. if len(self.members) == 0:
  220. return "typedef void * UA_%s;" % self.name
  221. returnstr = "typedef struct {\n"
  222. for member in self.members:
  223. if member.isArray:
  224. returnstr += " size_t %sSize;\n" % member.name
  225. returnstr += " UA_%s *%s;\n" % (member.memberType.name, member.name)
  226. else:
  227. returnstr += " UA_%s %s;\n" % (member.memberType.name, member.name)
  228. return returnstr + "} UA_%s;" % self.name
  229. #########################
  230. # Parse Typedefinitions #
  231. #########################
  232. def parseTypeDefinitions(outname, xmlDescription, namespace):
  233. def typeReady(element):
  234. "Are all member types defined?"
  235. for child in element:
  236. if child.tag == "{http://opcfoundation.org/BinarySchema/}Field":
  237. childname = child.get("TypeName")
  238. if childname[childname.find(":")+1:] not in types:
  239. return False
  240. return True
  241. def skipType(name):
  242. if name in excluded_types:
  243. return True
  244. if "Test" in name: # skip all test types
  245. return True
  246. if re.search("NodeId$", name) != None:
  247. return True
  248. return False
  249. snippets = {}
  250. for typeXml in etree.parse(xmlDescription).getroot():
  251. if not typeXml.get("Name"):
  252. continue
  253. name = typeXml.get("Name")
  254. snippets[name] = typeXml
  255. while(len(snippets) > 0):
  256. for name, typeXml in list(snippets.items()):
  257. if name in types or skipType(name):
  258. del snippets[name]
  259. continue
  260. if not typeReady(typeXml):
  261. continue
  262. if name in builtin_types:
  263. types[name] = BuiltinType(name)
  264. elif typeXml.tag == "{http://opcfoundation.org/BinarySchema/}EnumeratedType":
  265. types[name] = EnumerationType(outname, typeXml, namespace)
  266. elif typeXml.tag == "{http://opcfoundation.org/BinarySchema/}OpaqueType":
  267. types[name] = OpaqueType(outname, typeXml, namespace, get_base_type_for_opaque(name)['name'])
  268. elif typeXml.tag == "{http://opcfoundation.org/BinarySchema/}StructuredType":
  269. types[name] = StructType(outname, typeXml, namespace)
  270. else:
  271. raise Exception("Type not known")
  272. del snippets[name]
  273. ##########################
  274. # Parse TypeDescriptions #
  275. ##########################
  276. class TypeDescription(object):
  277. def __init__(self, name, nodeid, namespaceid):
  278. self.name = name
  279. self.nodeid = nodeid
  280. self.namespaceid = namespaceid
  281. self.xmlEncodingId = "0"
  282. self.binaryEncodingId = "0"
  283. def parseTypeDescriptions(f, namespaceid):
  284. definitions = {}
  285. input_str = f.read()
  286. input_str = input_str.replace('\r','')
  287. rows = map(lambda x:tuple(x.split(',')), input_str.split('\n'))
  288. for index, row in enumerate(rows):
  289. if len(row) < 3:
  290. continue
  291. if row[2] == "Object":
  292. # Check if node name ends with _Encoding_(DefaultXml|DefaultBinary) and store the node id in the corresponding DataType
  293. m = re.match('(.*?)_Encoding_Default(Xml|Binary)$',row[0])
  294. if (m):
  295. baseType = m.group(1)
  296. if baseType not in types:
  297. continue
  298. if m.group(2) == "Xml":
  299. definitions[baseType].xmlEncodingId = row[1]
  300. else:
  301. definitions[baseType].binaryEncodingId = row[1]
  302. continue
  303. if row[2] != "DataType":
  304. continue
  305. if row[0] == "BaseDataType":
  306. definitions["Variant"] = TypeDescription(row[0], row[1], namespaceid)
  307. elif row[0] == "Structure":
  308. definitions["ExtensionObject"] = TypeDescription(row[0], row[1], namespaceid)
  309. elif row[0] not in types:
  310. continue
  311. else:
  312. definitions[row[0]] = TypeDescription(row[0], row[1], namespaceid)
  313. return definitions
  314. def merge_dicts(*dict_args):
  315. """
  316. Given any number of dicts, shallow copy and merge into a new dict,
  317. precedence goes to key value pairs in latter dicts.
  318. """
  319. result = {}
  320. for dictionary in dict_args:
  321. result.update(dictionary)
  322. return result
  323. ###############################
  324. # Parse the Command Line Input#
  325. ###############################
  326. parser = argparse.ArgumentParser()
  327. parser.add_argument('-c', '--type-csv',
  328. metavar="<typeDescriptions>",
  329. type=argparse.FileType('r'),
  330. dest="type_csv",
  331. action='append',
  332. default=[],
  333. help='csv file with type descriptions')
  334. parser.add_argument('--namespace',
  335. type=int,
  336. dest="namespace",
  337. default=0,
  338. help='namespace id of the generated type nodeids (defaults to 0)')
  339. parser.add_argument('-s', '--selected-types',
  340. metavar="<selectedTypes>",
  341. type=argparse.FileType('r'),
  342. dest="selected_types",
  343. action='append',
  344. default=[],
  345. help='file with list of types (among those parsed) to be generated. If not given, all types are generated')
  346. parser.add_argument('--no-builtin',
  347. action='store_true',
  348. dest="no_builtin",
  349. help='Do not generate builtin types')
  350. parser.add_argument('-t', '--type-bsd',
  351. metavar="<typeBsds>",
  352. type=argparse.FileType('r'),
  353. dest="type_bsd",
  354. action='append',
  355. default=[],
  356. help='csv file with type descriptions')
  357. parser.add_argument('outfile',
  358. metavar='<outputFile>',
  359. help='output file w/o extension')
  360. args = parser.parse_args()
  361. outname = args.outfile.split("/")[-1]
  362. inname = ', '.join(list(map(lambda x:x.name.split("/")[-1], args.type_bsd)))
  363. ################
  364. # Create Types #
  365. ################
  366. for builtin in builtin_types:
  367. types[builtin] = BuiltinType(builtin)
  368. for f in args.type_bsd:
  369. parseTypeDefinitions(outname, f, args.namespace)
  370. typedescriptions = {}
  371. for f in args.type_csv:
  372. typedescriptions = merge_dicts(typedescriptions, parseTypeDescriptions(f, args.namespace))
  373. selected_types = []
  374. for f in args.selected_types:
  375. selected_types = list(filter(len, [line.strip() for line in f]))
  376. if len(selected_types) == 0:
  377. selected_types = types.keys()
  378. #############################
  379. # Write out the Definitions #
  380. #############################
  381. fh = open(args.outfile + "_generated.h",'w')
  382. ff = open(args.outfile + "_generated_handling.h",'w')
  383. fe = open(args.outfile + "_generated_encoding_binary.h",'w')
  384. fc = open(args.outfile + "_generated.c",'w')
  385. def printh(string):
  386. print(string, end='\n', file=fh)
  387. def printf(string):
  388. print(string, end='\n', file=ff)
  389. def printe(string):
  390. print(string, end='\n', file=fe)
  391. def printc(string):
  392. print(string, end='\n', file=fc)
  393. def iter_types(v):
  394. l = None
  395. if sys.version_info[0] < 3:
  396. l = list(v.itervalues())
  397. else:
  398. l = list(v.values())
  399. if len(selected_types) > 0:
  400. l = filter(lambda t: t.name in selected_types, l)
  401. if args.no_builtin:
  402. l = filter(lambda t: type(t) != BuiltinType, l)
  403. return l
  404. ################
  405. # Print Header #
  406. ################
  407. printh('''/* Generated from ''' + inname + ''' with script ''' + sys.argv[0] + '''
  408. * on host ''' + platform.uname()[1] + ''' by user ''' + getpass.getuser() + \
  409. ''' at ''' + time.strftime("%Y-%m-%d %I:%M:%S") + ''' */
  410. #ifndef ''' + outname.upper() + '''_GENERATED_H_
  411. #define ''' + outname.upper() + '''_GENERATED_H_
  412. #ifdef __cplusplus
  413. extern "C" {
  414. #endif
  415. #include "ua_types.h"
  416. ''' + ('#include "ua_types_generated.h"\n' if outname != "ua_types" else ''))
  417. filtered_types = iter_types(types)
  418. printh('''/**
  419. * Every type is assigned an index in an array containing the type descriptions.
  420. * These descriptions are used during type handling (copying, deletion,
  421. * binary encoding, ...). */''')
  422. printh("#define " + outname.upper() + "_COUNT %s" % (str(len(filtered_types))))
  423. printh("extern UA_EXPORT const UA_DataType " + outname.upper() + "[" + outname.upper() + "_COUNT];")
  424. i = 0
  425. for t in filtered_types:
  426. printh("\n/**\n * " + t.name)
  427. printh(" * " + "^" * len(t.name))
  428. if t.description == "":
  429. printh(" */")
  430. else:
  431. printh(" * " + t.description + " */")
  432. if type(t) != BuiltinType:
  433. printh(t.typedef_h() + "\n")
  434. printh("#define " + outname.upper() + "_" + t.name.upper() + " " + str(i))
  435. i += 1
  436. printh('''
  437. #ifdef __cplusplus
  438. } // extern "C"
  439. #endif\n
  440. #endif /* %s_GENERATED_H_ */''' % outname.upper())
  441. ##################
  442. # Print Handling #
  443. ##################
  444. printf('''/* Generated from ''' + inname + ''' with script ''' + sys.argv[0] + '''
  445. * on host ''' + platform.uname()[1] + ''' by user ''' + getpass.getuser() + \
  446. ''' at ''' + time.strftime("%Y-%m-%d %I:%M:%S") + ''' */
  447. #ifndef ''' + outname.upper() + '''_GENERATED_HANDLING_H_
  448. #define ''' + outname.upper() + '''_GENERATED_HANDLING_H_
  449. #ifdef __cplusplus
  450. extern "C" {
  451. #endif
  452. #include "''' + outname + '''_generated.h"
  453. #if defined(__GNUC__) && __GNUC__ >= 4 && __GNUC_MINOR__ >= 6
  454. # pragma GCC diagnostic push
  455. # pragma GCC diagnostic ignored "-Wmissing-field-initializers"
  456. # pragma GCC diagnostic ignored "-Wmissing-braces"
  457. #endif
  458. ''')
  459. for t in filtered_types:
  460. printf("\n/* " + t.name + " */")
  461. printf(t.functions_c())
  462. printf('''
  463. #if defined(__GNUC__) && __GNUC__ >= 4 && __GNUC_MINOR__ >= 6
  464. # pragma GCC diagnostic pop
  465. #endif
  466. #ifdef __cplusplus
  467. } // extern "C"
  468. #endif\n
  469. #endif /* %s_GENERATED_HANDLING_H_ */''' % outname.upper())
  470. ###########################
  471. # Print Description Array #
  472. ###########################
  473. printc('''/* Generated from ''' + inname + ''' with script ''' + sys.argv[0] + '''
  474. * on host ''' + platform.uname()[1] + ''' by user ''' + getpass.getuser() + \
  475. ''' at ''' + time.strftime("%Y-%m-%d %I:%M:%S") + ''' */
  476. #include "''' + outname + '''_generated.h"
  477. #include "ua_util.h"''')
  478. for t in filtered_types:
  479. printc("")
  480. printc("/* " + t.name + " */")
  481. printc(t.members_c())
  482. printc("const UA_DataType %s[%s_COUNT] = {" % (outname.upper(), outname.upper()))
  483. for t in filtered_types:
  484. printc("")
  485. printc("/* " + t.name + " */")
  486. printc(t.datatype_c() + ",")
  487. printc("};\n")
  488. ##################
  489. # Print Encoding #
  490. ##################
  491. printe('''/* Generated from ''' + inname + ''' with script ''' + sys.argv[0] + '''
  492. * on host ''' + platform.uname()[1] + ''' by user ''' + getpass.getuser() + \
  493. ''' at ''' + time.strftime("%Y-%m-%d %I:%M:%S") + ''' */
  494. #include "ua_types_encoding_binary.h"
  495. #include "''' + outname + '''_generated.h"''')
  496. for t in filtered_types:
  497. printe("\n/* " + t.name + " */")
  498. printe(t.encoding_h())
  499. fh.close()
  500. ff.close()
  501. fc.close()
  502. fe.close()