|  | #!/usr/bin/env python | 
|  | # | 
|  | # Copyright (c) 2005 Niels Provos <provos@citi.umich.edu> | 
|  | # All rights reserved. | 
|  | # | 
|  | # Generates marshaling code based on libevent. | 
|  |  | 
|  | import sys | 
|  | import re | 
|  |  | 
|  | # | 
|  | _NAME = "event_rpcgen.py" | 
|  | _VERSION = "0.1" | 
|  | _STRUCT_RE = '[a-z][a-z_0-9]*' | 
|  |  | 
|  | # Globals | 
|  | line_count = 0 | 
|  |  | 
|  | white = re.compile(r'^\s+') | 
|  | cppcomment = re.compile(r'\/\/.*$') | 
|  | headerdirect = [] | 
|  | cppdirect = [] | 
|  |  | 
|  | # Holds everything that makes a struct | 
|  | class Struct: | 
|  | def __init__(self, name): | 
|  | self._name = name | 
|  | self._entries = [] | 
|  | self._tags = {} | 
|  | print >>sys.stderr, '  Created struct: %s' % name | 
|  |  | 
|  | def AddEntry(self, entry): | 
|  | if self._tags.has_key(entry.Tag()): | 
|  | print >>sys.stderr, ( 'Entry "%s" duplicates tag number ' | 
|  | '%d from "%s" around line %d' ) % ( | 
|  | entry.Name(), entry.Tag(), | 
|  | self._tags[entry.Tag()], line_count) | 
|  | sys.exit(1) | 
|  | self._entries.append(entry) | 
|  | self._tags[entry.Tag()] = entry.Name() | 
|  | print >>sys.stderr, '    Added entry: %s' % entry.Name() | 
|  |  | 
|  | def Name(self): | 
|  | return self._name | 
|  |  | 
|  | def EntryTagName(self, entry): | 
|  | """Creates the name inside an enumeration for distinguishing data | 
|  | types.""" | 
|  | name = "%s_%s" % (self._name, entry.Name()) | 
|  | return name.upper() | 
|  |  | 
|  | def PrintIdented(self, file, ident, code): | 
|  | """Takes an array, add indentation to each entry and prints it.""" | 
|  | for entry in code: | 
|  | print >>file, '%s%s' % (ident, entry) | 
|  |  | 
|  | def PrintTags(self, file): | 
|  | """Prints the tag definitions for a structure.""" | 
|  | print >>file, '/* Tag definition for %s */' % self._name | 
|  | print >>file, 'enum %s_ {' % self._name.lower() | 
|  | for entry in self._entries: | 
|  | print >>file, '  %s=%d,' % (self.EntryTagName(entry), | 
|  | entry.Tag()) | 
|  | print >>file, '  %s_MAX_TAGS' % (self._name.upper()) | 
|  | print >>file, '};\n' | 
|  |  | 
|  | def PrintForwardDeclaration(self, file): | 
|  | print >>file, 'struct %s;' % self._name | 
|  |  | 
|  | def PrintDeclaration(self, file): | 
|  | print >>file, '/* Structure declaration for %s */' % self._name | 
|  | print >>file, 'struct %s_access_ {' % self._name | 
|  | for entry in self._entries: | 
|  | dcl = entry.AssignDeclaration('(*%s_assign)' % entry.Name()) | 
|  | dcl.extend( | 
|  | entry.GetDeclaration('(*%s_get)' % entry.Name())) | 
|  | if entry.Array(): | 
|  | dcl.extend( | 
|  | entry.AddDeclaration('(*%s_add)' % entry.Name())) | 
|  | self.PrintIdented(file, '  ', dcl) | 
|  | print >>file, '};\n' | 
|  |  | 
|  | print >>file, 'struct %s {' % self._name | 
|  | print >>file, '  struct %s_access_ *base;\n' % self._name | 
|  | for entry in self._entries: | 
|  | dcl = entry.Declaration() | 
|  | self.PrintIdented(file, '  ', dcl) | 
|  | print >>file, '' | 
|  | for entry in self._entries: | 
|  | print >>file, '  ev_uint8_t %s_set;' % entry.Name() | 
|  | print >>file, '};\n' | 
|  |  | 
|  | print >>file, \ | 
|  | """struct %(name)s *%(name)s_new(void); | 
|  | void %(name)s_free(struct %(name)s *); | 
|  | void %(name)s_clear(struct %(name)s *); | 
|  | void %(name)s_marshal(struct evbuffer *, const struct %(name)s *); | 
|  | int %(name)s_unmarshal(struct %(name)s *, struct evbuffer *); | 
|  | int %(name)s_complete(struct %(name)s *); | 
|  | void evtag_marshal_%(name)s(struct evbuffer *, ev_uint32_t, | 
|  | const struct %(name)s *); | 
|  | int evtag_unmarshal_%(name)s(struct evbuffer *, ev_uint32_t, | 
|  | struct %(name)s *);""" % { 'name' : self._name } | 
|  |  | 
|  |  | 
|  | # Write a setting function of every variable | 
|  | for entry in self._entries: | 
|  | self.PrintIdented(file, '', entry.AssignDeclaration( | 
|  | entry.AssignFuncName())) | 
|  | self.PrintIdented(file, '', entry.GetDeclaration( | 
|  | entry.GetFuncName())) | 
|  | if entry.Array(): | 
|  | self.PrintIdented(file, '', entry.AddDeclaration( | 
|  | entry.AddFuncName())) | 
|  |  | 
|  | print >>file, '/* --- %s done --- */\n' % self._name | 
|  |  | 
|  | def PrintCode(self, file): | 
|  | print >>file, ('/*\n' | 
|  | ' * Implementation of %s\n' | 
|  | ' */\n') % self._name | 
|  |  | 
|  | print >>file, \ | 
|  | 'static struct %(name)s_access_ __%(name)s_base = {' % \ | 
|  | { 'name' : self._name } | 
|  | for entry in self._entries: | 
|  | self.PrintIdented(file, '  ', entry.CodeBase()) | 
|  | print >>file, '};\n' | 
|  |  | 
|  | # Creation | 
|  | print >>file, ( | 
|  | 'struct %(name)s *\n' | 
|  | '%(name)s_new(void)\n' | 
|  | '{\n' | 
|  | '  struct %(name)s *tmp;\n' | 
|  | '  if ((tmp = malloc(sizeof(struct %(name)s))) == NULL) {\n' | 
|  | '    event_warn("%%s: malloc", __func__);\n' | 
|  | '    return (NULL);\n' | 
|  | '  }\n' | 
|  | '  tmp->base = &__%(name)s_base;\n') % { 'name' : self._name } | 
|  |  | 
|  | for entry in self._entries: | 
|  | self.PrintIdented(file, '  ', entry.CodeNew('tmp')) | 
|  | print >>file, '  tmp->%s_set = 0;\n' % entry.Name() | 
|  |  | 
|  | print >>file, ( | 
|  | '  return (tmp);\n' | 
|  | '}\n') | 
|  |  | 
|  | # Adding | 
|  | for entry in self._entries: | 
|  | if entry.Array(): | 
|  | self.PrintIdented(file, '', entry.CodeAdd()) | 
|  | print >>file, '' | 
|  |  | 
|  | # Assigning | 
|  | for entry in self._entries: | 
|  | self.PrintIdented(file, '', entry.CodeAssign()) | 
|  | print >>file, '' | 
|  |  | 
|  | # Getting | 
|  | for entry in self._entries: | 
|  | self.PrintIdented(file, '', entry.CodeGet()) | 
|  | print >>file, '' | 
|  |  | 
|  | # Clearing | 
|  | print >>file, ( 'void\n' | 
|  | '%(name)s_clear(struct %(name)s *tmp)\n' | 
|  | '{' | 
|  | ) % { 'name' : self._name } | 
|  | for entry in self._entries: | 
|  | self.PrintIdented(file, '  ', entry.CodeClear('tmp')) | 
|  |  | 
|  | print >>file, '}\n' | 
|  |  | 
|  | # Freeing | 
|  | print >>file, ( 'void\n' | 
|  | '%(name)s_free(struct %(name)s *tmp)\n' | 
|  | '{' | 
|  | ) % { 'name' : self._name } | 
|  |  | 
|  | for entry in self._entries: | 
|  | self.PrintIdented(file, '  ', entry.CodeFree('tmp')) | 
|  |  | 
|  | print >>file, ('  free(tmp);\n' | 
|  | '}\n') | 
|  |  | 
|  | # Marshaling | 
|  | print >>file, ('void\n' | 
|  | '%(name)s_marshal(struct evbuffer *evbuf, ' | 
|  | 'const struct %(name)s *tmp)' | 
|  | '{') % { 'name' : self._name } | 
|  | for entry in self._entries: | 
|  | indent = '  ' | 
|  | # Optional entries do not have to be set | 
|  | if entry.Optional(): | 
|  | indent += '  ' | 
|  | print >>file, '  if (tmp->%s_set) {' % entry.Name() | 
|  | self.PrintIdented( | 
|  | file, indent, | 
|  | entry.CodeMarshal('evbuf', self.EntryTagName(entry), 'tmp')) | 
|  | if entry.Optional(): | 
|  | print >>file, '  }' | 
|  |  | 
|  | print >>file, '}\n' | 
|  |  | 
|  | # Unmarshaling | 
|  | print >>file, ('int\n' | 
|  | '%(name)s_unmarshal(struct %(name)s *tmp, ' | 
|  | ' struct evbuffer *evbuf)\n' | 
|  | '{\n' | 
|  | '  ev_uint32_t tag;\n' | 
|  | '  while (EVBUFFER_LENGTH(evbuf) > 0) {\n' | 
|  | '    if (evtag_peek(evbuf, &tag) == -1)\n' | 
|  | '      return (-1);\n' | 
|  | '    switch (tag) {\n' | 
|  | ) % { 'name' : self._name } | 
|  | for entry in self._entries: | 
|  | print >>file, '      case %s:\n' % self.EntryTagName(entry) | 
|  | if not entry.Array(): | 
|  | print >>file, ( | 
|  | '        if (tmp->%s_set)\n' | 
|  | '          return (-1);' | 
|  | ) % (entry.Name()) | 
|  |  | 
|  | self.PrintIdented( | 
|  | file, '        ', | 
|  | entry.CodeUnmarshal('evbuf', | 
|  | self.EntryTagName(entry), 'tmp')) | 
|  |  | 
|  | print >>file, ( '        tmp->%s_set = 1;\n' % entry.Name() + | 
|  | '        break;\n' ) | 
|  | print >>file, ( '      default:\n' | 
|  | '        return -1;\n' | 
|  | '    }\n' | 
|  | '  }\n' ) | 
|  | # Check if it was decoded completely | 
|  | print >>file, ( '  if (%(name)s_complete(tmp) == -1)\n' | 
|  | '    return (-1);' | 
|  | ) % { 'name' : self._name } | 
|  |  | 
|  | # Successfully decoded | 
|  | print >>file, ( '  return (0);\n' | 
|  | '}\n') | 
|  |  | 
|  | # Checking if a structure has all the required data | 
|  | print >>file, ( | 
|  | 'int\n' | 
|  | '%(name)s_complete(struct %(name)s *msg)\n' | 
|  | '{' ) % { 'name' : self._name } | 
|  | for entry in self._entries: | 
|  | self.PrintIdented( | 
|  | file, '  ', | 
|  | entry.CodeComplete('msg')) | 
|  | print >>file, ( | 
|  | '  return (0);\n' | 
|  | '}\n' ) | 
|  |  | 
|  | # Complete message unmarshaling | 
|  | print >>file, ( | 
|  | 'int\n' | 
|  | 'evtag_unmarshal_%(name)s(struct evbuffer *evbuf, ' | 
|  | 'ev_uint32_t need_tag, struct %(name)s *msg)\n' | 
|  | '{\n' | 
|  | '  ev_uint32_t tag;\n' | 
|  | '  int res = -1;\n' | 
|  | '\n' | 
|  | '  struct evbuffer *tmp = evbuffer_new();\n' | 
|  | '\n' | 
|  | '  if (evtag_unmarshal(evbuf, &tag, tmp) == -1' | 
|  | ' || tag != need_tag)\n' | 
|  | '    goto error;\n' | 
|  | '\n' | 
|  | '  if (%(name)s_unmarshal(msg, tmp) == -1)\n' | 
|  | '    goto error;\n' | 
|  | '\n' | 
|  | '  res = 0;\n' | 
|  | '\n' | 
|  | ' error:\n' | 
|  | '  evbuffer_free(tmp);\n' | 
|  | '  return (res);\n' | 
|  | '}\n' ) % { 'name' : self._name } | 
|  |  | 
|  | # Complete message marshaling | 
|  | print >>file, ( | 
|  | 'void\n' | 
|  | 'evtag_marshal_%(name)s(struct evbuffer *evbuf, ev_uint32_t tag, ' | 
|  | 'const struct %(name)s *msg)\n' | 
|  | '{\n' | 
|  | '  struct evbuffer *_buf = evbuffer_new();\n' | 
|  | '  assert(_buf != NULL);\n' | 
|  | '  evbuffer_drain(_buf, -1);\n' | 
|  | '  %(name)s_marshal(_buf, msg);\n' | 
|  | '  evtag_marshal(evbuf, tag, EVBUFFER_DATA(_buf), ' | 
|  | 'EVBUFFER_LENGTH(_buf));\n' | 
|  | '  evbuffer_free(_buf);\n' | 
|  | '}\n' ) % { 'name' : self._name } | 
|  |  | 
|  | class Entry: | 
|  | def __init__(self, type, name, tag): | 
|  | self._type = type | 
|  | self._name = name | 
|  | self._tag = int(tag) | 
|  | self._ctype = type | 
|  | self._optional = 0 | 
|  | self._can_be_array = 0 | 
|  | self._array = 0 | 
|  | self._line_count = -1 | 
|  | self._struct = None | 
|  | self._refname = None | 
|  |  | 
|  | def GetTranslation(self): | 
|  | return { "parent_name" : self._struct.Name(), | 
|  | "name" : self._name, | 
|  | "ctype" : self._ctype, | 
|  | "refname" : self._refname | 
|  | } | 
|  |  | 
|  | def SetStruct(self, struct): | 
|  | self._struct = struct | 
|  |  | 
|  | def LineCount(self): | 
|  | assert self._line_count != -1 | 
|  | return self._line_count | 
|  |  | 
|  | def SetLineCount(self, number): | 
|  | self._line_count = number | 
|  |  | 
|  | def Array(self): | 
|  | return self._array | 
|  |  | 
|  | def Optional(self): | 
|  | return self._optional | 
|  |  | 
|  | def Tag(self): | 
|  | return self._tag | 
|  |  | 
|  | def Name(self): | 
|  | return self._name | 
|  |  | 
|  | def Type(self): | 
|  | return self._type | 
|  |  | 
|  | def MakeArray(self, yes=1): | 
|  | self._array = yes | 
|  |  | 
|  | def MakeOptional(self): | 
|  | self._optional = 1 | 
|  |  | 
|  | def GetFuncName(self): | 
|  | return '%s_%s_get' % (self._struct.Name(), self._name) | 
|  |  | 
|  | def GetDeclaration(self, funcname): | 
|  | code = [ 'int %s(struct %s *, %s *);' % ( | 
|  | funcname, self._struct.Name(), self._ctype ) ] | 
|  | return code | 
|  |  | 
|  | def CodeGet(self): | 
|  | code = ( | 
|  | 'int', | 
|  | '%(parent_name)s_%(name)s_get(struct %(parent_name)s *msg, ' | 
|  | '%(ctype)s *value)', | 
|  | '{', | 
|  | '  if (msg->%(name)s_set != 1)', | 
|  | '    return (-1);', | 
|  | '  *value = msg->%(name)s_data;', | 
|  | '  return (0);', | 
|  | '}' ) | 
|  | code = '\n'.join(code) | 
|  | code = code % self.GetTranslation() | 
|  | return code.split('\n') | 
|  |  | 
|  | def AssignFuncName(self): | 
|  | return '%s_%s_assign' % (self._struct.Name(), self._name) | 
|  |  | 
|  | def AddFuncName(self): | 
|  | return '%s_%s_add' % (self._struct.Name(), self._name) | 
|  |  | 
|  | def AssignDeclaration(self, funcname): | 
|  | code = [ 'int %s(struct %s *, const %s);' % ( | 
|  | funcname, self._struct.Name(), self._ctype ) ] | 
|  | return code | 
|  |  | 
|  | def CodeAssign(self): | 
|  | code = [ 'int', | 
|  | '%(parent_name)s_%(name)s_assign(struct %(parent_name)s *msg,' | 
|  | ' const %(ctype)s value)', | 
|  | '{', | 
|  | '  msg->%(name)s_set = 1;', | 
|  | '  msg->%(name)s_data = value;', | 
|  | '  return (0);', | 
|  | '}' ] | 
|  | code = '\n'.join(code) | 
|  | code = code % self.GetTranslation() | 
|  | return code.split('\n') | 
|  |  | 
|  | def CodeClear(self, structname): | 
|  | code = [ '%s->%s_set = 0;' % (structname, self.Name()) ] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def CodeComplete(self, structname): | 
|  | if self.Optional(): | 
|  | return [] | 
|  |  | 
|  | code = [ 'if (!%s->%s_set)' % (structname, self.Name()), | 
|  | '  return (-1);' ] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def CodeFree(self, name): | 
|  | return [] | 
|  |  | 
|  | def CodeBase(self): | 
|  | code = [ | 
|  | '%(parent_name)s_%(name)s_assign,', | 
|  | '%(parent_name)s_%(name)s_get,' | 
|  | ] | 
|  | if self.Array(): | 
|  | code.append('%(parent_name)s_%(name)s_add,') | 
|  |  | 
|  | code = '\n'.join(code) | 
|  | code = code % self.GetTranslation() | 
|  | return code.split('\n') | 
|  |  | 
|  | def Verify(self): | 
|  | if self.Array() and not self._can_be_array: | 
|  | print >>sys.stderr, ( | 
|  | 'Entry "%s" cannot be created as an array ' | 
|  | 'around line %d' ) % (self._name, self.LineCount()) | 
|  | sys.exit(1) | 
|  | if not self._struct: | 
|  | print >>sys.stderr, ( | 
|  | 'Entry "%s" does not know which struct it belongs to ' | 
|  | 'around line %d' ) % (self._name, self.LineCount()) | 
|  | sys.exit(1) | 
|  | if self._optional and self._array: | 
|  | print >>sys.stderr,  ( 'Entry "%s" has illegal combination of ' | 
|  | 'optional and array around line %d' ) % ( | 
|  | self._name, self.LineCount() ) | 
|  | sys.exit(1) | 
|  |  | 
|  | class EntryBytes(Entry): | 
|  | def __init__(self, type, name, tag, length): | 
|  | # Init base class | 
|  | Entry.__init__(self, type, name, tag) | 
|  |  | 
|  | self._length = length | 
|  | self._ctype = 'ev_uint8_t' | 
|  |  | 
|  | def GetDeclaration(self, funcname): | 
|  | code = [ 'int %s(struct %s *, %s **);' % ( | 
|  | funcname, self._struct.Name(), self._ctype ) ] | 
|  | return code | 
|  |  | 
|  | def AssignDeclaration(self, funcname): | 
|  | code = [ 'int %s(struct %s *, const %s *);' % ( | 
|  | funcname, self._struct.Name(), self._ctype ) ] | 
|  | return code | 
|  |  | 
|  | def Declaration(self): | 
|  | dcl  = ['ev_uint8_t %s_data[%s];' % (self._name, self._length)] | 
|  |  | 
|  | return dcl | 
|  |  | 
|  | def CodeGet(self): | 
|  | name = self._name | 
|  | code = [ 'int', | 
|  | '%s_%s_get(struct %s *msg, %s **value)' % ( | 
|  | self._struct.Name(), name, | 
|  | self._struct.Name(), self._ctype), | 
|  | '{', | 
|  | '  if (msg->%s_set != 1)' % name, | 
|  | '    return (-1);', | 
|  | '  *value = msg->%s_data;' % name, | 
|  | '  return (0);', | 
|  | '}' ] | 
|  | return code | 
|  |  | 
|  | def CodeAssign(self): | 
|  | name = self._name | 
|  | code = [ 'int', | 
|  | '%s_%s_assign(struct %s *msg, const %s *value)' % ( | 
|  | self._struct.Name(), name, | 
|  | self._struct.Name(), self._ctype), | 
|  | '{', | 
|  | '  msg->%s_set = 1;' % name, | 
|  | '  memcpy(msg->%s_data, value, %s);' % ( | 
|  | name, self._length), | 
|  | '  return (0);', | 
|  | '}' ] | 
|  | return code | 
|  |  | 
|  | def CodeUnmarshal(self, buf, tag_name, var_name): | 
|  | code = [  'if (evtag_unmarshal_fixed(%s, %s, ' % (buf, tag_name) + | 
|  | '%s->%s_data, ' % (var_name, self._name) + | 
|  | 'sizeof(%s->%s_data)) == -1) {' % ( | 
|  | var_name, self._name), | 
|  | '  event_warnx("%%s: failed to unmarshal %s", __func__);' % ( | 
|  | self._name ), | 
|  | '  return (-1);', | 
|  | '}' | 
|  | ] | 
|  | return code | 
|  |  | 
|  | def CodeMarshal(self, buf, tag_name, var_name): | 
|  | code = ['evtag_marshal(%s, %s, %s->%s_data, sizeof(%s->%s_data));' % ( | 
|  | buf, tag_name, var_name, self._name, var_name, self._name )] | 
|  | return code | 
|  |  | 
|  | def CodeClear(self, structname): | 
|  | code = [ '%s->%s_set = 0;' % (structname, self.Name()), | 
|  | 'memset(%s->%s_data, 0, sizeof(%s->%s_data));' % ( | 
|  | structname, self._name, structname, self._name)] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def CodeNew(self, name): | 
|  | code  = ['memset(%s->%s_data, 0, sizeof(%s->%s_data));' % ( | 
|  | name, self._name, name, self._name)] | 
|  | return code | 
|  |  | 
|  | def Verify(self): | 
|  | if not self._length: | 
|  | print >>sys.stderr, 'Entry "%s" needs a length around line %d' % ( | 
|  | self._name, self.LineCount() ) | 
|  | sys.exit(1) | 
|  |  | 
|  | Entry.Verify(self) | 
|  |  | 
|  | class EntryInt(Entry): | 
|  | def __init__(self, type, name, tag): | 
|  | # Init base class | 
|  | Entry.__init__(self, type, name, tag) | 
|  |  | 
|  | self._ctype = 'ev_uint32_t' | 
|  |  | 
|  | def CodeUnmarshal(self, buf, tag_name, var_name): | 
|  | code = ['if (evtag_unmarshal_int(%s, %s, &%s->%s_data) == -1) {' % ( | 
|  | buf, tag_name, var_name, self._name), | 
|  | '  event_warnx("%%s: failed to unmarshal %s", __func__);' % ( | 
|  | self._name ), | 
|  | '  return (-1);', | 
|  | '}' ] | 
|  | return code | 
|  |  | 
|  | def CodeMarshal(self, buf, tag_name, var_name): | 
|  | code = ['evtag_marshal_int(%s, %s, %s->%s_data);' % ( | 
|  | buf, tag_name, var_name, self._name)] | 
|  | return code | 
|  |  | 
|  | def Declaration(self): | 
|  | dcl  = ['ev_uint32_t %s_data;' % self._name] | 
|  |  | 
|  | return dcl | 
|  |  | 
|  | def CodeNew(self, name): | 
|  | code = ['%s->%s_data = 0;' % (name, self._name)] | 
|  | return code | 
|  |  | 
|  | class EntryString(Entry): | 
|  | def __init__(self, type, name, tag): | 
|  | # Init base class | 
|  | Entry.__init__(self, type, name, tag) | 
|  |  | 
|  | self._ctype = 'char *' | 
|  |  | 
|  | def CodeAssign(self): | 
|  | name = self._name | 
|  | code = """int | 
|  | %(parent_name)s_%(name)s_assign(struct %(parent_name)s *msg, | 
|  | const %(ctype)s value) | 
|  | { | 
|  | if (msg->%(name)s_data != NULL) | 
|  | free(msg->%(name)s_data); | 
|  | if ((msg->%(name)s_data = strdup(value)) == NULL) | 
|  | return (-1); | 
|  | msg->%(name)s_set = 1; | 
|  | return (0); | 
|  | }""" % self.GetTranslation() | 
|  |  | 
|  | return code.split('\n') | 
|  |  | 
|  | def CodeUnmarshal(self, buf, tag_name, var_name): | 
|  | code = ['if (evtag_unmarshal_string(%s, %s, &%s->%s_data) == -1) {' % ( | 
|  | buf, tag_name, var_name, self._name), | 
|  | '  event_warnx("%%s: failed to unmarshal %s", __func__);' % ( | 
|  | self._name ), | 
|  | '  return (-1);', | 
|  | '}' | 
|  | ] | 
|  | return code | 
|  |  | 
|  | def CodeMarshal(self, buf, tag_name, var_name): | 
|  | code = ['evtag_marshal_string(%s, %s, %s->%s_data);' % ( | 
|  | buf, tag_name, var_name, self._name)] | 
|  | return code | 
|  |  | 
|  | def CodeClear(self, structname): | 
|  | code = [ 'if (%s->%s_set == 1) {' % (structname, self.Name()), | 
|  | '  free (%s->%s_data);' % (structname, self.Name()), | 
|  | '  %s->%s_data = NULL;' % (structname, self.Name()), | 
|  | '  %s->%s_set = 0;' % (structname, self.Name()), | 
|  | '}' | 
|  | ] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def CodeNew(self, name): | 
|  | code  = ['%s->%s_data = NULL;' % (name, self._name)] | 
|  | return code | 
|  |  | 
|  | def CodeFree(self, name): | 
|  | code  = ['if (%s->%s_data != NULL)' % (name, self._name), | 
|  | '    free (%s->%s_data); ' % (name, self._name)] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def Declaration(self): | 
|  | dcl  = ['char *%s_data;' % self._name] | 
|  |  | 
|  | return dcl | 
|  |  | 
|  | class EntryStruct(Entry): | 
|  | def __init__(self, type, name, tag, refname): | 
|  | # Init base class | 
|  | Entry.__init__(self, type, name, tag) | 
|  |  | 
|  | self._can_be_array = 1 | 
|  | self._refname = refname | 
|  | self._ctype = 'struct %s*' % refname | 
|  |  | 
|  | def CodeGet(self): | 
|  | name = self._name | 
|  | code = [ 'int', | 
|  | '%s_%s_get(struct %s *msg, %s *value)' % ( | 
|  | self._struct.Name(), name, | 
|  | self._struct.Name(), self._ctype), | 
|  | '{', | 
|  | '  if (msg->%s_set != 1) {' % name, | 
|  | '    msg->%s_data = %s_new();' % (name, self._refname), | 
|  | '    if (msg->%s_data == NULL)' % name, | 
|  | '      return (-1);', | 
|  | '    msg->%s_set = 1;' % name, | 
|  | '  }', | 
|  | '  *value = msg->%s_data;' % name, | 
|  | '  return (0);', | 
|  | '}' ] | 
|  | return code | 
|  |  | 
|  | def CodeAssign(self): | 
|  | name = self._name | 
|  | code = """int | 
|  | %(parent_name)s_%(name)s_assign(struct %(parent_name)s *msg, | 
|  | const %(ctype)s value) | 
|  | { | 
|  | struct evbuffer *tmp = NULL; | 
|  | if (msg->%(name)s_set) { | 
|  | %(refname)s_clear(msg->%(name)s_data); | 
|  | msg->%(name)s_set = 0; | 
|  | } else { | 
|  | msg->%(name)s_data = %(refname)s_new(); | 
|  | if (msg->%(name)s_data == NULL) { | 
|  | event_warn("%%s: %(refname)s_new()", __func__); | 
|  | goto error; | 
|  | } | 
|  | } | 
|  | if ((tmp = evbuffer_new()) == NULL) { | 
|  | event_warn("%%s: evbuffer_new()", __func__); | 
|  | goto error; | 
|  | } | 
|  | %(refname)s_marshal(tmp, value); | 
|  | if (%(refname)s_unmarshal(msg->%(name)s_data, tmp) == -1) { | 
|  | event_warnx("%%s: %(refname)s_unmarshal", __func__); | 
|  | goto error; | 
|  | } | 
|  | msg->%(name)s_set = 1; | 
|  | evbuffer_free(tmp); | 
|  | return (0); | 
|  | error: | 
|  | if (tmp != NULL) | 
|  | evbuffer_free(tmp); | 
|  | if (msg->%(name)s_data != NULL) { | 
|  | %(refname)s_free(msg->%(name)s_data); | 
|  | msg->%(name)s_data = NULL; | 
|  | } | 
|  | return (-1); | 
|  | }""" % self.GetTranslation() | 
|  | return code.split('\n') | 
|  |  | 
|  | def CodeComplete(self, structname): | 
|  | if self.Optional(): | 
|  | code = [ 'if (%s->%s_set && %s_complete(%s->%s_data) == -1)' % ( | 
|  | structname, self.Name(), | 
|  | self._refname, structname, self.Name()), | 
|  | '  return (-1);' ] | 
|  | else: | 
|  | code = [ 'if (%s_complete(%s->%s_data) == -1)' % ( | 
|  | self._refname, structname, self.Name()), | 
|  | '  return (-1);' ] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def CodeUnmarshal(self, buf, tag_name, var_name): | 
|  | code = ['%s->%s_data = %s_new();' % ( | 
|  | var_name, self._name, self._refname), | 
|  | 'if (%s->%s_data == NULL)' % (var_name, self._name), | 
|  | '  return (-1);', | 
|  | 'if (evtag_unmarshal_%s(%s, %s, %s->%s_data) == -1) {' % ( | 
|  | self._refname, buf, tag_name, var_name, self._name), | 
|  | '  event_warnx("%%s: failed to unmarshal %s", __func__);' % ( | 
|  | self._name ), | 
|  | '  return (-1);', | 
|  | '}' | 
|  | ] | 
|  | return code | 
|  |  | 
|  | def CodeMarshal(self, buf, tag_name, var_name): | 
|  | code = ['evtag_marshal_%s(%s, %s, %s->%s_data);' % ( | 
|  | self._refname, buf, tag_name, var_name, self._name)] | 
|  | return code | 
|  |  | 
|  | def CodeClear(self, structname): | 
|  | code = [ 'if (%s->%s_set == 1) {' % (structname, self.Name()), | 
|  | '  %s_free(%s->%s_data);' % ( | 
|  | self._refname, structname, self.Name()), | 
|  | '  %s->%s_data = NULL;' % (structname, self.Name()), | 
|  | '  %s->%s_set = 0;' % (structname, self.Name()), | 
|  | '}' | 
|  | ] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def CodeNew(self, name): | 
|  | code  = ['%s->%s_data = NULL;' % (name, self._name)] | 
|  | return code | 
|  |  | 
|  | def CodeFree(self, name): | 
|  | code  = ['if (%s->%s_data != NULL)' % (name, self._name), | 
|  | '    %s_free(%s->%s_data); ' % ( | 
|  | self._refname, name, self._name)] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def Declaration(self): | 
|  | dcl  = ['%s %s_data;' % (self._ctype, self._name)] | 
|  |  | 
|  | return dcl | 
|  |  | 
|  | class EntryVarBytes(Entry): | 
|  | def __init__(self, type, name, tag): | 
|  | # Init base class | 
|  | Entry.__init__(self, type, name, tag) | 
|  |  | 
|  | self._ctype = 'ev_uint8_t *' | 
|  |  | 
|  | def GetDeclaration(self, funcname): | 
|  | code = [ 'int %s(struct %s *, %s *, ev_uint32_t *);' % ( | 
|  | funcname, self._struct.Name(), self._ctype ) ] | 
|  | return code | 
|  |  | 
|  | def AssignDeclaration(self, funcname): | 
|  | code = [ 'int %s(struct %s *, const %s, ev_uint32_t);' % ( | 
|  | funcname, self._struct.Name(), self._ctype ) ] | 
|  | return code | 
|  |  | 
|  | def CodeAssign(self): | 
|  | name = self._name | 
|  | code = [ 'int', | 
|  | '%s_%s_assign(struct %s *msg, ' | 
|  | 'const %s value, ev_uint32_t len)' % ( | 
|  | self._struct.Name(), name, | 
|  | self._struct.Name(), self._ctype), | 
|  | '{', | 
|  | '  if (msg->%s_data != NULL)' % name, | 
|  | '    free (msg->%s_data);' % name, | 
|  | '  msg->%s_data = malloc(len);' % name, | 
|  | '  if (msg->%s_data == NULL)' % name, | 
|  | '    return (-1);', | 
|  | '  msg->%s_set = 1;' % name, | 
|  | '  msg->%s_length = len;' % name, | 
|  | '  memcpy(msg->%s_data, value, len);' % name, | 
|  | '  return (0);', | 
|  | '}' ] | 
|  | return code | 
|  |  | 
|  | def CodeGet(self): | 
|  | name = self._name | 
|  | code = [ 'int', | 
|  | '%s_%s_get(struct %s *msg, %s *value, ev_uint32_t *plen)' % ( | 
|  | self._struct.Name(), name, | 
|  | self._struct.Name(), self._ctype), | 
|  | '{', | 
|  | '  if (msg->%s_set != 1)' % name, | 
|  | '    return (-1);', | 
|  | '  *value = msg->%s_data;' % name, | 
|  | '  *plen = msg->%s_length;' % name, | 
|  | '  return (0);', | 
|  | '}' ] | 
|  | return code | 
|  |  | 
|  | def CodeUnmarshal(self, buf, tag_name, var_name): | 
|  | code = ['if (evtag_payload_length(%s, &%s->%s_length) == -1)' % ( | 
|  | buf, var_name, self._name), | 
|  | '  return (-1);', | 
|  | # We do not want DoS opportunities | 
|  | 'if (%s->%s_length > EVBUFFER_LENGTH(%s))' % ( | 
|  | var_name, self._name, buf), | 
|  | '  return (-1);', | 
|  | 'if ((%s->%s_data = malloc(%s->%s_length)) == NULL)' % ( | 
|  | var_name, self._name, var_name, self._name), | 
|  | '  return (-1);', | 
|  | 'if (evtag_unmarshal_fixed(%s, %s, %s->%s_data, ' | 
|  | '%s->%s_length) == -1) {' % ( | 
|  | buf, tag_name, var_name, self._name, var_name, self._name), | 
|  | '  event_warnx("%%s: failed to unmarshal %s", __func__);' % ( | 
|  | self._name ), | 
|  | '  return (-1);', | 
|  | '}' | 
|  | ] | 
|  | return code | 
|  |  | 
|  | def CodeMarshal(self, buf, tag_name, var_name): | 
|  | code = ['evtag_marshal(%s, %s, %s->%s_data, %s->%s_length);' % ( | 
|  | buf, tag_name, var_name, self._name, var_name, self._name)] | 
|  | return code | 
|  |  | 
|  | def CodeClear(self, structname): | 
|  | code = [ 'if (%s->%s_set == 1) {' % (structname, self.Name()), | 
|  | '  free (%s->%s_data);' % (structname, self.Name()), | 
|  | '  %s->%s_data = NULL;' % (structname, self.Name()), | 
|  | '  %s->%s_length = 0;' % (structname, self.Name()), | 
|  | '  %s->%s_set = 0;' % (structname, self.Name()), | 
|  | '}' | 
|  | ] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def CodeNew(self, name): | 
|  | code  = ['%s->%s_data = NULL;' % (name, self._name), | 
|  | '%s->%s_length = 0;' % (name, self._name) ] | 
|  | return code | 
|  |  | 
|  | def CodeFree(self, name): | 
|  | code  = ['if (%s->%s_data != NULL)' % (name, self._name), | 
|  | '    free (%s->%s_data); ' % (name, self._name)] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def Declaration(self): | 
|  | dcl  = ['ev_uint8_t *%s_data;' % self._name, | 
|  | 'ev_uint32_t %s_length;' % self._name] | 
|  |  | 
|  | return dcl | 
|  |  | 
|  | class EntryArray(Entry): | 
|  | def __init__(self, entry): | 
|  | # Init base class | 
|  | Entry.__init__(self, entry._type, entry._name, entry._tag) | 
|  |  | 
|  | self._entry = entry | 
|  | self._refname = entry._refname | 
|  | self._ctype = 'struct %s *' % self._refname | 
|  |  | 
|  | def GetDeclaration(self, funcname): | 
|  | """Allows direct access to elements of the array.""" | 
|  | translate = self.GetTranslation() | 
|  | translate["funcname"] = funcname | 
|  | code = [ | 
|  | 'int %(funcname)s(struct %(parent_name)s *, int, %(ctype)s *);' % | 
|  | translate ] | 
|  | return code | 
|  |  | 
|  | def AssignDeclaration(self, funcname): | 
|  | code = [ 'int %s(struct %s *, int, const %s);' % ( | 
|  | funcname, self._struct.Name(), self._ctype ) ] | 
|  | return code | 
|  |  | 
|  | def AddDeclaration(self, funcname): | 
|  | code = [ '%s %s(struct %s *);' % ( | 
|  | self._ctype, funcname, self._struct.Name() ) ] | 
|  | return code | 
|  |  | 
|  | def CodeGet(self): | 
|  | code = """int | 
|  | %(parent_name)s_%(name)s_get(struct %(parent_name)s *msg, int offset, | 
|  | %(ctype)s *value) | 
|  | { | 
|  | if (!msg->%(name)s_set || offset < 0 || offset >= msg->%(name)s_length) | 
|  | return (-1); | 
|  | *value = msg->%(name)s_data[offset]; | 
|  | return (0); | 
|  | }""" % self.GetTranslation() | 
|  |  | 
|  | return code.split('\n') | 
|  |  | 
|  | def CodeAssign(self): | 
|  | code = """int | 
|  | %(parent_name)s_%(name)s_assign(struct %(parent_name)s *msg, int off, | 
|  | const %(ctype)s value) | 
|  | { | 
|  | struct evbuffer *tmp = NULL; | 
|  | if (!msg->%(name)s_set || off < 0 || off >= msg->%(name)s_length) | 
|  | return (-1); | 
|  | %(refname)s_clear(msg->%(name)s_data[off]); | 
|  | if ((tmp = evbuffer_new()) == NULL) { | 
|  | event_warn("%%s: evbuffer_new()", __func__); | 
|  | goto error; | 
|  | } | 
|  | %(refname)s_marshal(tmp, value); | 
|  | if (%(refname)s_unmarshal(msg->%(name)s_data[off], tmp) == -1) { | 
|  | event_warnx("%%s: %(refname)s_unmarshal", __func__); | 
|  | goto error; | 
|  | } | 
|  | evbuffer_free(tmp); | 
|  | return (0); | 
|  | error: | 
|  | if (tmp != NULL) | 
|  | evbuffer_free(tmp); | 
|  | %(refname)s_clear(msg->%(name)s_data[off]); | 
|  | return (-1); | 
|  | }""" % self.GetTranslation() | 
|  |  | 
|  | return code.split('\n') | 
|  |  | 
|  | def CodeAdd(self): | 
|  | code = \ | 
|  | """%(ctype)s | 
|  | %(parent_name)s_%(name)s_add(struct %(parent_name)s *msg) | 
|  | { | 
|  | if (++msg->%(name)s_length >= msg->%(name)s_num_allocated) { | 
|  | int tobe_allocated = msg->%(name)s_num_allocated; | 
|  | %(ctype)s* new_data = NULL; | 
|  | tobe_allocated = !tobe_allocated ? 1 : tobe_allocated << 1; | 
|  | new_data = (%(ctype)s*) realloc(msg->%(name)s_data, | 
|  | tobe_allocated * sizeof(%(ctype)s)); | 
|  | if (new_data == NULL) | 
|  | goto error; | 
|  | msg->%(name)s_data = new_data; | 
|  | msg->%(name)s_num_allocated = tobe_allocated; | 
|  | } | 
|  | msg->%(name)s_data[msg->%(name)s_length - 1] = %(refname)s_new(); | 
|  | if (msg->%(name)s_data[msg->%(name)s_length - 1] == NULL) | 
|  | goto error; | 
|  | msg->%(name)s_set = 1; | 
|  | return (msg->%(name)s_data[msg->%(name)s_length - 1]); | 
|  | error: | 
|  | --msg->%(name)s_length; | 
|  | return (NULL); | 
|  | } | 
|  | """ % self.GetTranslation() | 
|  |  | 
|  | return code.split('\n') | 
|  |  | 
|  | def CodeComplete(self, structname): | 
|  | code = [] | 
|  | translate = self.GetTranslation() | 
|  |  | 
|  | if self.Optional(): | 
|  | code.append( 'if (%(structname)s->%(name)s_set)'  % translate) | 
|  |  | 
|  | translate["structname"] = structname | 
|  | tmp = """{ | 
|  | int i; | 
|  | for (i = 0; i < %(structname)s->%(name)s_length; ++i) { | 
|  | if (%(refname)s_complete(%(structname)s->%(name)s_data[i]) == -1) | 
|  | return (-1); | 
|  | } | 
|  | }""" % translate | 
|  | code.extend(tmp.split('\n')) | 
|  |  | 
|  | return code | 
|  |  | 
|  | def CodeUnmarshal(self, buf, tag_name, var_name): | 
|  | translate = self.GetTranslation() | 
|  | translate["var_name"] = var_name | 
|  | translate["buf"] = buf | 
|  | translate["tag_name"] = tag_name | 
|  | code = """if (%(parent_name)s_%(name)s_add(%(var_name)s) == NULL) | 
|  | return (-1); | 
|  | if (evtag_unmarshal_%(refname)s(%(buf)s, %(tag_name)s, | 
|  | %(var_name)s->%(name)s_data[%(var_name)s->%(name)s_length - 1]) == -1) { | 
|  | --%(var_name)s->%(name)s_length; | 
|  | event_warnx("%%s: failed to unmarshal %(name)s", __func__); | 
|  | return (-1); | 
|  | }""" % translate | 
|  |  | 
|  | return code.split('\n') | 
|  |  | 
|  | def CodeMarshal(self, buf, tag_name, var_name): | 
|  | code = ['{', | 
|  | '  int i;', | 
|  | '  for (i = 0; i < %s->%s_length; ++i) {' % ( | 
|  | var_name, self._name), | 
|  | '    evtag_marshal_%s(%s, %s, %s->%s_data[i]);' % ( | 
|  | self._refname, buf, tag_name, var_name, self._name), | 
|  | '  }', | 
|  | '}' | 
|  | ] | 
|  | return code | 
|  |  | 
|  | def CodeClear(self, structname): | 
|  | code = [ 'if (%s->%s_set == 1) {' % (structname, self.Name()), | 
|  | '  int i;', | 
|  | '  for (i = 0; i < %s->%s_length; ++i) {' % ( | 
|  | structname, self.Name()), | 
|  | '    %s_free(%s->%s_data[i]);' % ( | 
|  | self._refname, structname, self.Name()), | 
|  | '  }', | 
|  | '  free(%s->%s_data);' % (structname, self.Name()), | 
|  | '  %s->%s_data = NULL;' % (structname, self.Name()), | 
|  | '  %s->%s_set = 0;' % (structname, self.Name()), | 
|  | '  %s->%s_length = 0;' % (structname, self.Name()), | 
|  | '  %s->%s_num_allocated = 0;' % (structname, self.Name()), | 
|  | '}' | 
|  | ] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def CodeNew(self, name): | 
|  | code  = ['%s->%s_data = NULL;' % (name, self._name), | 
|  | '%s->%s_length = 0;' % (name, self._name), | 
|  | '%s->%s_num_allocated = 0;' % (name, self._name)] | 
|  | return code | 
|  |  | 
|  | def CodeFree(self, name): | 
|  | code  = ['if (%s->%s_data != NULL) {' % (name, self._name), | 
|  | '  int i;', | 
|  | '  for (i = 0; i < %s->%s_length; ++i) {' % ( | 
|  | name, self._name), | 
|  | '    %s_free(%s->%s_data[i]); ' % ( | 
|  | self._refname, name, self._name), | 
|  | '    %s->%s_data[i] = NULL;' % (name, self._name), | 
|  | '  }', | 
|  | '  free(%s->%s_data);' % (name, self._name), | 
|  | '  %s->%s_data = NULL;' % (name, self._name), | 
|  | '  %s->%s_length = 0;' % (name, self._name), | 
|  | '  %s->%s_num_allocated = 0;' % (name, self._name), | 
|  | '}' | 
|  | ] | 
|  |  | 
|  | return code | 
|  |  | 
|  | def Declaration(self): | 
|  | dcl  = ['struct %s **%s_data;' % (self._refname, self._name), | 
|  | 'int %s_length;' % self._name, | 
|  | 'int %s_num_allocated;' % self._name ] | 
|  |  | 
|  | return dcl | 
|  |  | 
|  | def NormalizeLine(line): | 
|  | global white | 
|  | global cppcomment | 
|  |  | 
|  | line = cppcomment.sub('', line) | 
|  | line = line.strip() | 
|  | line = white.sub(' ', line) | 
|  |  | 
|  | return line | 
|  |  | 
|  | def ProcessOneEntry(newstruct, entry): | 
|  | optional = 0 | 
|  | array = 0 | 
|  | entry_type = '' | 
|  | name = '' | 
|  | tag = '' | 
|  | tag_set = None | 
|  | separator = '' | 
|  | fixed_length = '' | 
|  |  | 
|  | tokens = entry.split(' ') | 
|  | while tokens: | 
|  | token = tokens[0] | 
|  | tokens = tokens[1:] | 
|  |  | 
|  | if not entry_type: | 
|  | if not optional and token == 'optional': | 
|  | optional = 1 | 
|  | continue | 
|  |  | 
|  | if not array and token == 'array': | 
|  | array = 1 | 
|  | continue | 
|  |  | 
|  | if not entry_type: | 
|  | entry_type = token | 
|  | continue | 
|  |  | 
|  | if not name: | 
|  | res = re.match(r'^([^\[\]]+)(\[.*\])?$', token) | 
|  | if not res: | 
|  | print >>sys.stderr, 'Cannot parse name: \"%s\" around %d' % ( | 
|  | entry, line_count) | 
|  | sys.exit(1) | 
|  | name = res.group(1) | 
|  | fixed_length = res.group(2) | 
|  | if fixed_length: | 
|  | fixed_length = fixed_length[1:-1] | 
|  | continue | 
|  |  | 
|  | if not separator: | 
|  | separator = token | 
|  | if separator != '=': | 
|  | print >>sys.stderr, 'Expected "=" after name \"%s\" got %s' % ( | 
|  | name, token) | 
|  | sys.exit(1) | 
|  | continue | 
|  |  | 
|  | if not tag_set: | 
|  | tag_set = 1 | 
|  | if not re.match(r'^(0x)?[0-9]+$', token): | 
|  | print >>sys.stderr, 'Expected tag number: \"%s\"' % entry | 
|  | sys.exit(1) | 
|  | tag = int(token, 0) | 
|  | continue | 
|  |  | 
|  | print >>sys.stderr, 'Cannot parse \"%s\"' % entry | 
|  | sys.exit(1) | 
|  |  | 
|  | if not tag_set: | 
|  | print >>sys.stderr, 'Need tag number: \"%s\"' % entry | 
|  | sys.exit(1) | 
|  |  | 
|  | # Create the right entry | 
|  | if entry_type == 'bytes': | 
|  | if fixed_length: | 
|  | newentry = EntryBytes(entry_type, name, tag, fixed_length) | 
|  | else: | 
|  | newentry = EntryVarBytes(entry_type, name, tag) | 
|  | elif entry_type == 'int' and not fixed_length: | 
|  | newentry = EntryInt(entry_type, name, tag) | 
|  | elif entry_type == 'string' and not fixed_length: | 
|  | newentry = EntryString(entry_type, name, tag) | 
|  | else: | 
|  | res = re.match(r'^struct\[(%s)\]$' % _STRUCT_RE, | 
|  | entry_type, re.IGNORECASE) | 
|  | if res: | 
|  | # References another struct defined in our file | 
|  | newentry = EntryStruct(entry_type, name, tag, res.group(1)) | 
|  | else: | 
|  | print >>sys.stderr, 'Bad type: "%s" in "%s"' % (entry_type, entry) | 
|  | sys.exit(1) | 
|  |  | 
|  | structs = [] | 
|  |  | 
|  | if optional: | 
|  | newentry.MakeOptional() | 
|  | if array: | 
|  | newentry.MakeArray() | 
|  |  | 
|  | newentry.SetStruct(newstruct) | 
|  | newentry.SetLineCount(line_count) | 
|  | newentry.Verify() | 
|  |  | 
|  | if array: | 
|  | # We need to encapsulate this entry into a struct | 
|  | newname = newentry.Name()+ '_array' | 
|  |  | 
|  | # Now borgify the new entry. | 
|  | newentry = EntryArray(newentry) | 
|  | newentry.SetStruct(newstruct) | 
|  | newentry.SetLineCount(line_count) | 
|  | newentry.MakeArray() | 
|  |  | 
|  | newstruct.AddEntry(newentry) | 
|  |  | 
|  | return structs | 
|  |  | 
|  | def ProcessStruct(data): | 
|  | tokens = data.split(' ') | 
|  |  | 
|  | # First three tokens are: 'struct' 'name' '{' | 
|  | newstruct = Struct(tokens[1]) | 
|  |  | 
|  | inside = ' '.join(tokens[3:-1]) | 
|  |  | 
|  | tokens = inside.split(';') | 
|  |  | 
|  | structs = [] | 
|  |  | 
|  | for entry in tokens: | 
|  | entry = NormalizeLine(entry) | 
|  | if not entry: | 
|  | continue | 
|  |  | 
|  | # It's possible that new structs get defined in here | 
|  | structs.extend(ProcessOneEntry(newstruct, entry)) | 
|  |  | 
|  | structs.append(newstruct) | 
|  | return structs | 
|  |  | 
|  | def GetNextStruct(file): | 
|  | global line_count | 
|  | global cppdirect | 
|  |  | 
|  | got_struct = 0 | 
|  |  | 
|  | processed_lines = [] | 
|  |  | 
|  | have_c_comment = 0 | 
|  | data = '' | 
|  | while 1: | 
|  | line = file.readline() | 
|  | if not line: | 
|  | break | 
|  |  | 
|  | line_count += 1 | 
|  | line = line[:-1] | 
|  |  | 
|  | if not have_c_comment and re.search(r'/\*', line): | 
|  | if re.search(r'/\*.*\*/', line): | 
|  | line = re.sub(r'/\*.*\*/', '', line) | 
|  | else: | 
|  | line = re.sub(r'/\*.*$', '', line) | 
|  | have_c_comment = 1 | 
|  |  | 
|  | if have_c_comment: | 
|  | if not re.search(r'\*/', line): | 
|  | continue | 
|  | have_c_comment = 0 | 
|  | line = re.sub(r'^.*\*/', '', line) | 
|  |  | 
|  | line = NormalizeLine(line) | 
|  |  | 
|  | if not line: | 
|  | continue | 
|  |  | 
|  | if not got_struct: | 
|  | if re.match(r'#include ["<].*[>"]', line): | 
|  | cppdirect.append(line) | 
|  | continue | 
|  |  | 
|  | if re.match(r'^#(if( |def)|endif)', line): | 
|  | cppdirect.append(line) | 
|  | continue | 
|  |  | 
|  | if re.match(r'^#define', line): | 
|  | headerdirect.append(line) | 
|  | continue | 
|  |  | 
|  | if not re.match(r'^struct %s {$' % _STRUCT_RE, | 
|  | line, re.IGNORECASE): | 
|  | print >>sys.stderr, 'Missing struct on line %d: %s' % ( | 
|  | line_count, line) | 
|  | sys.exit(1) | 
|  | else: | 
|  | got_struct = 1 | 
|  | data += line | 
|  | continue | 
|  |  | 
|  | # We are inside the struct | 
|  | tokens = line.split('}') | 
|  | if len(tokens) == 1: | 
|  | data += ' ' + line | 
|  | continue | 
|  |  | 
|  | if len(tokens[1]): | 
|  | print >>sys.stderr, 'Trailing garbage after struct on line %d' % ( | 
|  | line_count ) | 
|  | sys.exit(1) | 
|  |  | 
|  | # We found the end of the struct | 
|  | data += ' %s}' % tokens[0] | 
|  | break | 
|  |  | 
|  | # Remove any comments, that might be in there | 
|  | data = re.sub(r'/\*.*\*/', '', data) | 
|  |  | 
|  | return data | 
|  |  | 
|  |  | 
|  | def Parse(file): | 
|  | """ | 
|  | Parses the input file and returns C code and corresponding header file. | 
|  | """ | 
|  |  | 
|  | entities = [] | 
|  |  | 
|  | while 1: | 
|  | # Just gets the whole struct nicely formatted | 
|  | data = GetNextStruct(file) | 
|  |  | 
|  | if not data: | 
|  | break | 
|  |  | 
|  | entities.extend(ProcessStruct(data)) | 
|  |  | 
|  | return entities | 
|  |  | 
|  | def GuardName(name): | 
|  | name = '_'.join(name.split('.')) | 
|  | name = '_'.join(name.split('/')) | 
|  | guard = '_'+name.upper()+'_' | 
|  |  | 
|  | return guard | 
|  |  | 
|  | def HeaderPreamble(name): | 
|  | guard = GuardName(name) | 
|  | pre = ( | 
|  | '/*\n' | 
|  | ' * Automatically generated from %s\n' | 
|  | ' */\n\n' | 
|  | '#ifndef %s\n' | 
|  | '#define %s\n\n' ) % ( | 
|  | name, guard, guard) | 
|  |  | 
|  | # insert stdint.h - let's hope everyone has it | 
|  | pre += ( | 
|  | '#include <event-config.h>\n' | 
|  | '#ifdef _EVENT_HAVE_STDINT_H\n' | 
|  | '#include <stdint.h>\n' | 
|  | '#endif\n' ) | 
|  |  | 
|  | for statement in headerdirect: | 
|  | pre += '%s\n' % statement | 
|  | if headerdirect: | 
|  | pre += '\n' | 
|  |  | 
|  | pre += ( | 
|  | '#define EVTAG_HAS(msg, member) ((msg)->member##_set == 1)\n' | 
|  | '#ifdef __GNUC__\n' | 
|  | '#define EVTAG_ASSIGN(msg, member, args...) ' | 
|  | '(*(msg)->base->member##_assign)(msg, ## args)\n' | 
|  | '#define EVTAG_GET(msg, member, args...) ' | 
|  | '(*(msg)->base->member##_get)(msg, ## args)\n' | 
|  | '#else\n' | 
|  | '#define EVTAG_ASSIGN(msg, member, ...) ' | 
|  | '(*(msg)->base->member##_assign)(msg, ## __VA_ARGS__)\n' | 
|  | '#define EVTAG_GET(msg, member, ...) ' | 
|  | '(*(msg)->base->member##_get)(msg, ## __VA_ARGS__)\n' | 
|  | '#endif\n' | 
|  | '#define EVTAG_ADD(msg, member) (*(msg)->base->member##_add)(msg)\n' | 
|  | '#define EVTAG_LEN(msg, member) ((msg)->member##_length)\n' | 
|  | ) | 
|  |  | 
|  | return pre | 
|  |  | 
|  |  | 
|  | def HeaderPostamble(name): | 
|  | guard = GuardName(name) | 
|  | return '#endif  /* %s */' % guard | 
|  |  | 
|  | def BodyPreamble(name): | 
|  | global _NAME | 
|  | global _VERSION | 
|  |  | 
|  | header_file = '.'.join(name.split('.')[:-1]) + '.gen.h' | 
|  |  | 
|  | pre = ( '/*\n' | 
|  | ' * Automatically generated from %s\n' | 
|  | ' * by %s/%s.  DO NOT EDIT THIS FILE.\n' | 
|  | ' */\n\n' ) % (name, _NAME, _VERSION) | 
|  | pre += ( '#include <sys/types.h>\n' | 
|  | '#ifdef _EVENT_HAVE_SYS_TIME_H\n' | 
|  | '#include <sys/time.h>\n' | 
|  | '#endif\n' | 
|  | '#include <stdlib.h>\n' | 
|  | '#include <string.h>\n' | 
|  | '#include <assert.h>\n' | 
|  | '#define EVENT_NO_STRUCT\n' | 
|  | '#include <event.h>\n\n' | 
|  | '#ifdef _EVENT___func__\n' | 
|  | '#define __func__ _EVENT___func__\n' | 
|  | '#endif\n' ) | 
|  |  | 
|  | for statement in cppdirect: | 
|  | pre += '%s\n' % statement | 
|  |  | 
|  | pre += '\n#include "%s"\n\n' % header_file | 
|  |  | 
|  | pre += 'void event_err(int eval, const char *fmt, ...);\n' | 
|  | pre += 'void event_warn(const char *fmt, ...);\n' | 
|  | pre += 'void event_errx(int eval, const char *fmt, ...);\n' | 
|  | pre += 'void event_warnx(const char *fmt, ...);\n\n' | 
|  |  | 
|  | return pre | 
|  |  | 
|  | def main(argv): | 
|  | if len(argv) < 2 or not argv[1]: | 
|  | print >>sys.stderr, 'Need RPC description file as first argument.' | 
|  | sys.exit(1) | 
|  |  | 
|  | filename = argv[1] | 
|  |  | 
|  | ext = filename.split('.')[-1] | 
|  | if ext != 'rpc': | 
|  | print >>sys.stderr, 'Unrecognized file extension: %s' % ext | 
|  | sys.exit(1) | 
|  |  | 
|  | print >>sys.stderr, 'Reading \"%s\"' % filename | 
|  |  | 
|  | fp = open(filename, 'r') | 
|  | entities = Parse(fp) | 
|  | fp.close() | 
|  |  | 
|  | header_file = '.'.join(filename.split('.')[:-1]) + '.gen.h' | 
|  | impl_file = '.'.join(filename.split('.')[:-1]) + '.gen.c' | 
|  |  | 
|  | print >>sys.stderr, '... creating "%s"' % header_file | 
|  | header_fp = open(header_file, 'w') | 
|  | print >>header_fp, HeaderPreamble(filename) | 
|  |  | 
|  | # Create forward declarations: allows other structs to reference | 
|  | # each other | 
|  | for entry in entities: | 
|  | entry.PrintForwardDeclaration(header_fp) | 
|  | print >>header_fp, '' | 
|  |  | 
|  | for entry in entities: | 
|  | entry.PrintTags(header_fp) | 
|  | entry.PrintDeclaration(header_fp) | 
|  | print >>header_fp, HeaderPostamble(filename) | 
|  | header_fp.close() | 
|  |  | 
|  | print >>sys.stderr, '... creating "%s"' % impl_file | 
|  | impl_fp = open(impl_file, 'w') | 
|  | print >>impl_fp, BodyPreamble(filename) | 
|  | for entry in entities: | 
|  | entry.PrintCode(impl_fp) | 
|  | impl_fp.close() | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | main(sys.argv) |