Reformat generated .pb.c files using macros.
This has the following advantages: 1) Easier to modify pb_field_t encoding 2) Simpler generator logic 3) Tidier looking, easier to read .pb.c files Update issue 58 Status: FixedInGit
This commit is contained in:
@@ -35,22 +35,22 @@ except:
|
|||||||
import time
|
import time
|
||||||
import os.path
|
import os.path
|
||||||
|
|
||||||
# Values are tuple (c type, pb ltype)
|
# Values are tuple (c type, pb type)
|
||||||
FieldD = descriptor.FieldDescriptorProto
|
FieldD = descriptor.FieldDescriptorProto
|
||||||
datatypes = {
|
datatypes = {
|
||||||
FieldD.TYPE_BOOL: ('bool', 'PB_LTYPE_VARINT'),
|
FieldD.TYPE_BOOL: ('bool', 'BOOL'),
|
||||||
FieldD.TYPE_DOUBLE: ('double', 'PB_LTYPE_FIXED64'),
|
FieldD.TYPE_DOUBLE: ('double', 'DOUBLE'),
|
||||||
FieldD.TYPE_FIXED32: ('uint32_t', 'PB_LTYPE_FIXED32'),
|
FieldD.TYPE_FIXED32: ('uint32_t', 'FIXED32'),
|
||||||
FieldD.TYPE_FIXED64: ('uint64_t', 'PB_LTYPE_FIXED64'),
|
FieldD.TYPE_FIXED64: ('uint64_t', 'FIXED64'),
|
||||||
FieldD.TYPE_FLOAT: ('float', 'PB_LTYPE_FIXED32'),
|
FieldD.TYPE_FLOAT: ('float', 'FLOAT'),
|
||||||
FieldD.TYPE_INT32: ('int32_t', 'PB_LTYPE_VARINT'),
|
FieldD.TYPE_INT32: ('int32_t', 'INT32'),
|
||||||
FieldD.TYPE_INT64: ('int64_t', 'PB_LTYPE_VARINT'),
|
FieldD.TYPE_INT64: ('int64_t', 'INT64'),
|
||||||
FieldD.TYPE_SFIXED32: ('int32_t', 'PB_LTYPE_FIXED32'),
|
FieldD.TYPE_SFIXED32: ('int32_t', 'SFIXED32'),
|
||||||
FieldD.TYPE_SFIXED64: ('int64_t', 'PB_LTYPE_FIXED64'),
|
FieldD.TYPE_SFIXED64: ('int64_t', 'SFIXED64'),
|
||||||
FieldD.TYPE_SINT32: ('int32_t', 'PB_LTYPE_SVARINT'),
|
FieldD.TYPE_SINT32: ('int32_t', 'SINT32'),
|
||||||
FieldD.TYPE_SINT64: ('int64_t', 'PB_LTYPE_SVARINT'),
|
FieldD.TYPE_SINT64: ('int64_t', 'SINT64'),
|
||||||
FieldD.TYPE_UINT32: ('uint32_t', 'PB_LTYPE_VARINT'),
|
FieldD.TYPE_UINT32: ('uint32_t', 'UINT32'),
|
||||||
FieldD.TYPE_UINT64: ('uint64_t', 'PB_LTYPE_VARINT')
|
FieldD.TYPE_UINT64: ('uint64_t', 'UINT64')
|
||||||
}
|
}
|
||||||
|
|
||||||
class Names:
|
class Names:
|
||||||
@@ -123,49 +123,44 @@ class Field:
|
|||||||
if desc.HasField('default_value'):
|
if desc.HasField('default_value'):
|
||||||
self.default = desc.default_value
|
self.default = desc.default_value
|
||||||
|
|
||||||
# Decide HTYPE
|
# Check field rules, i.e. required/optional/repeated.
|
||||||
# HTYPE is the high-order nibble of nanopb field description,
|
|
||||||
# defining whether value is required/optional/repeated.
|
|
||||||
can_be_static = True
|
can_be_static = True
|
||||||
if desc.label == FieldD.LABEL_REQUIRED:
|
if desc.label == FieldD.LABEL_REQUIRED:
|
||||||
self.htype = 'PB_HTYPE_REQUIRED'
|
self.rules = 'REQUIRED'
|
||||||
elif desc.label == FieldD.LABEL_OPTIONAL:
|
elif desc.label == FieldD.LABEL_OPTIONAL:
|
||||||
self.htype = 'PB_HTYPE_OPTIONAL'
|
self.rules = 'OPTIONAL'
|
||||||
elif desc.label == FieldD.LABEL_REPEATED:
|
elif desc.label == FieldD.LABEL_REPEATED:
|
||||||
|
self.rules = 'REPEATED'
|
||||||
if self.max_count is None:
|
if self.max_count is None:
|
||||||
can_be_static = False
|
can_be_static = False
|
||||||
else:
|
else:
|
||||||
self.htype = 'PB_HTYPE_ARRAY'
|
|
||||||
self.array_decl = '[%d]' % self.max_count
|
self.array_decl = '[%d]' % self.max_count
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(desc.label)
|
raise NotImplementedError(desc.label)
|
||||||
|
|
||||||
# Decide LTYPE and CTYPE
|
# Decide the C data type to use in the struct.
|
||||||
# LTYPE is the low-order nibble of nanopb field description,
|
|
||||||
# defining how to decode an individual value.
|
|
||||||
# CTYPE is the name of the c type to use in the struct.
|
|
||||||
if datatypes.has_key(desc.type):
|
if datatypes.has_key(desc.type):
|
||||||
self.ctype, self.ltype = datatypes[desc.type]
|
self.ctype, self.pbtype = datatypes[desc.type]
|
||||||
elif desc.type == FieldD.TYPE_ENUM:
|
elif desc.type == FieldD.TYPE_ENUM:
|
||||||
self.ltype = 'PB_LTYPE_VARINT'
|
self.pbtype = 'ENUM'
|
||||||
self.ctype = names_from_type_name(desc.type_name)
|
self.ctype = names_from_type_name(desc.type_name)
|
||||||
if self.default is not None:
|
if self.default is not None:
|
||||||
self.default = self.ctype + self.default
|
self.default = self.ctype + self.default
|
||||||
elif desc.type == FieldD.TYPE_STRING:
|
elif desc.type == FieldD.TYPE_STRING:
|
||||||
self.ltype = 'PB_LTYPE_STRING'
|
self.pbtype = 'STRING'
|
||||||
if self.max_size is None:
|
if self.max_size is None:
|
||||||
can_be_static = False
|
can_be_static = False
|
||||||
else:
|
else:
|
||||||
self.ctype = 'char'
|
self.ctype = 'char'
|
||||||
self.array_decl += '[%d]' % self.max_size
|
self.array_decl += '[%d]' % self.max_size
|
||||||
elif desc.type == FieldD.TYPE_BYTES:
|
elif desc.type == FieldD.TYPE_BYTES:
|
||||||
self.ltype = 'PB_LTYPE_BYTES'
|
self.pbtype = 'BYTES'
|
||||||
if self.max_size is None:
|
if self.max_size is None:
|
||||||
can_be_static = False
|
can_be_static = False
|
||||||
else:
|
else:
|
||||||
self.ctype = self.struct_name + self.name + 't'
|
self.ctype = self.struct_name + self.name + 't'
|
||||||
elif desc.type == FieldD.TYPE_MESSAGE:
|
elif desc.type == FieldD.TYPE_MESSAGE:
|
||||||
self.ltype = 'PB_LTYPE_SUBMESSAGE'
|
self.pbtype = 'MESSAGE'
|
||||||
self.ctype = self.submsgname = names_from_type_name(desc.type_name)
|
self.ctype = self.submsgname = names_from_type_name(desc.type_name)
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(desc.type)
|
raise NotImplementedError(desc.type)
|
||||||
@@ -179,18 +174,22 @@ class Field:
|
|||||||
if field_options.type == nanopb_pb2.FT_STATIC and not can_be_static:
|
if field_options.type == nanopb_pb2.FT_STATIC and not can_be_static:
|
||||||
raise Exception("Field %s is defined as static, but max_size or max_count is not given." % self.name)
|
raise Exception("Field %s is defined as static, but max_size or max_count is not given." % self.name)
|
||||||
|
|
||||||
if field_options.type == nanopb_pb2.FT_CALLBACK:
|
if field_options.type == nanopb_pb2.FT_STATIC:
|
||||||
self.htype = 'PB_HTYPE_CALLBACK'
|
self.allocation = 'STATIC'
|
||||||
|
elif field_options.type == nanopb_pb2.FT_CALLBACK:
|
||||||
|
self.allocation = 'CALLBACK'
|
||||||
self.ctype = 'pb_callback_t'
|
self.ctype = 'pb_callback_t'
|
||||||
self.array_decl = ''
|
self.array_decl = ''
|
||||||
|
else:
|
||||||
|
raise NotImplementedError(field_options.type)
|
||||||
|
|
||||||
def __cmp__(self, other):
|
def __cmp__(self, other):
|
||||||
return cmp(self.tag, other.tag)
|
return cmp(self.tag, other.tag)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
if self.htype == 'PB_HTYPE_OPTIONAL':
|
if self.rules == 'OPTIONAL':
|
||||||
result = ' bool has_' + self.name + ';\n'
|
result = ' bool has_' + self.name + ';\n'
|
||||||
elif self.htype == 'PB_HTYPE_ARRAY':
|
elif self.rules == 'REPEATED' and self.allocation == 'STATIC':
|
||||||
result = ' size_t ' + self.name + '_count;\n'
|
result = ' size_t ' + self.name + '_count;\n'
|
||||||
else:
|
else:
|
||||||
result = ''
|
result = ''
|
||||||
@@ -199,7 +198,7 @@ class Field:
|
|||||||
|
|
||||||
def types(self):
|
def types(self):
|
||||||
'''Return definitions for any special types this field might need.'''
|
'''Return definitions for any special types this field might need.'''
|
||||||
if self.ltype == 'PB_LTYPE_BYTES' and self.max_size is not None:
|
if self.pbtype == 'BYTES' and self.allocation == 'STATIC':
|
||||||
result = 'typedef struct {\n'
|
result = 'typedef struct {\n'
|
||||||
result += ' size_t size;\n'
|
result += ' size_t size;\n'
|
||||||
result += ' uint8_t bytes[%d];\n' % self.max_size
|
result += ' uint8_t bytes[%d];\n' % self.max_size
|
||||||
@@ -212,30 +211,25 @@ class Field:
|
|||||||
'''Return definition for this field's default value.'''
|
'''Return definition for this field's default value.'''
|
||||||
if self.default is None:
|
if self.default is None:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
ctype, default = self.ctype, self.default
|
||||||
|
array_decl = ''
|
||||||
|
|
||||||
if self.ltype == 'PB_LTYPE_STRING':
|
if self.pbtype == 'STRING':
|
||||||
ctype = 'char'
|
if self.allocation != 'STATIC':
|
||||||
if self.max_size is None:
|
|
||||||
return None # Not implemented
|
return None # Not implemented
|
||||||
else:
|
|
||||||
array_decl = '[%d]' % (self.max_size + 1)
|
array_decl = '[%d]' % self.max_size
|
||||||
default = str(self.default).encode('string_escape')
|
default = str(self.default).encode('string_escape')
|
||||||
default = default.replace('"', '\\"')
|
default = default.replace('"', '\\"')
|
||||||
default = '"' + default + '"'
|
default = '"' + default + '"'
|
||||||
elif self.ltype == 'PB_LTYPE_BYTES':
|
elif self.pbtype == 'BYTES':
|
||||||
|
if self.allocation != 'STATIC':
|
||||||
|
return None # Not implemented
|
||||||
|
|
||||||
data = self.default.decode('string_escape')
|
data = self.default.decode('string_escape')
|
||||||
data = ['0x%02x' % ord(c) for c in data]
|
data = ['0x%02x' % ord(c) for c in data]
|
||||||
|
|
||||||
if self.max_size is None:
|
|
||||||
return None # Not implemented
|
|
||||||
else:
|
|
||||||
ctype = self.ctype
|
|
||||||
|
|
||||||
default = '{%d, {%s}}' % (len(data), ','.join(data))
|
default = '{%d, {%s}}' % (len(data), ','.join(data))
|
||||||
array_decl = ''
|
|
||||||
else:
|
|
||||||
ctype, default = self.ctype, self.default
|
|
||||||
array_decl = ''
|
|
||||||
|
|
||||||
if declaration_only:
|
if declaration_only:
|
||||||
return 'extern const %s %s_default%s;' % (ctype, self.struct_name + self.name, array_decl)
|
return 'extern const %s %s_default%s;' % (ctype, self.struct_name + self.name, array_decl)
|
||||||
@@ -246,47 +240,30 @@ class Field:
|
|||||||
'''Return the pb_field_t initializer to use in the constant array.
|
'''Return the pb_field_t initializer to use in the constant array.
|
||||||
prev_field_name is the name of the previous field or None.
|
prev_field_name is the name of the previous field or None.
|
||||||
'''
|
'''
|
||||||
result = ' {%d, ' % self.tag
|
result = ' PB_FIELD(%3d, ' % self.tag
|
||||||
result += '(pb_type_t) ((int) ' + self.htype
|
result += '%-8s, ' % self.pbtype
|
||||||
if self.ltype is not None:
|
result += '%s, ' % self.rules
|
||||||
result += ' | (int) ' + self.ltype
|
result += '%s, ' % self.allocation
|
||||||
result += '),\n'
|
result += '%s, ' % self.struct_name
|
||||||
|
result += '%s, ' % self.name
|
||||||
|
result += '%s, ' % (prev_field_name or self.name)
|
||||||
|
|
||||||
if prev_field_name is None:
|
if self.pbtype == 'MESSAGE':
|
||||||
result += ' offsetof(%s, %s),' % (self.struct_name, self.name)
|
result += '&%s_fields)' % self.submsgname
|
||||||
|
elif self.default is None:
|
||||||
|
result += '0)'
|
||||||
|
elif self.pbtype in ['BYTES', 'STRING'] and self.allocation != 'STATIC':
|
||||||
|
result += '0)' # Arbitrary size default values not implemented
|
||||||
else:
|
else:
|
||||||
result += ' pb_delta_end(%s, %s, %s),' % (self.struct_name, self.name, prev_field_name)
|
result += '&%s_default)' % (self.struct_name + self.name)
|
||||||
|
|
||||||
if self.htype == 'PB_HTYPE_OPTIONAL':
|
|
||||||
result += '\n pb_delta(%s, has_%s, %s),' % (self.struct_name, self.name, self.name)
|
|
||||||
elif self.htype == 'PB_HTYPE_ARRAY':
|
|
||||||
result += '\n pb_delta(%s, %s_count, %s),' % (self.struct_name, self.name, self.name)
|
|
||||||
else:
|
|
||||||
result += ' 0,'
|
|
||||||
|
|
||||||
|
|
||||||
if self.htype == 'PB_HTYPE_ARRAY':
|
|
||||||
result += '\n pb_membersize(%s, %s[0]),' % (self.struct_name, self.name)
|
|
||||||
result += ('\n pb_membersize(%s, %s) / pb_membersize(%s, %s[0]),'
|
|
||||||
% (self.struct_name, self.name, self.struct_name, self.name))
|
|
||||||
else:
|
|
||||||
result += '\n pb_membersize(%s, %s),' % (self.struct_name, self.name)
|
|
||||||
result += ' 0,'
|
|
||||||
|
|
||||||
if self.ltype == 'PB_LTYPE_SUBMESSAGE':
|
|
||||||
result += '\n &%s_fields}' % self.submsgname
|
|
||||||
elif self.default is None or self.htype == 'PB_HTYPE_CALLBACK':
|
|
||||||
result += ' 0}'
|
|
||||||
else:
|
|
||||||
result += '\n &%s_default}' % (self.struct_name + self.name)
|
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
def largest_field_value(self):
|
def largest_field_value(self):
|
||||||
'''Determine if this field needs 16bit or 32bit pb_field_t structure to compile properly.
|
'''Determine if this field needs 16bit or 32bit pb_field_t structure to compile properly.
|
||||||
Returns numeric value or a C-expression for assert.'''
|
Returns numeric value or a C-expression for assert.'''
|
||||||
if self.ltype == 'PB_LTYPE_SUBMESSAGE':
|
if self.pbtype == 'MESSAGE':
|
||||||
if self.htype == 'PB_HTYPE_ARRAY':
|
if self.rules == 'REPEATED' and self.allocation == 'STATIC':
|
||||||
return 'pb_membersize(%s, %s[0])' % (self.struct_name, self.name)
|
return 'pb_membersize(%s, %s[0])' % (self.struct_name, self.name)
|
||||||
else:
|
else:
|
||||||
return 'pb_membersize(%s, %s)' % (self.struct_name, self.name)
|
return 'pb_membersize(%s, %s)' % (self.struct_name, self.name)
|
||||||
@@ -358,7 +335,7 @@ class Message:
|
|||||||
prev = None
|
prev = None
|
||||||
for field in self.ordered_fields:
|
for field in self.ordered_fields:
|
||||||
result += field.pb_field_t(prev)
|
result += field.pb_field_t(prev)
|
||||||
result += ',\n\n'
|
result += ',\n'
|
||||||
prev = field.name
|
prev = field.name
|
||||||
|
|
||||||
result += ' PB_LAST_FIELD\n};'
|
result += ' PB_LAST_FIELD\n};'
|
||||||
@@ -502,7 +479,7 @@ def generate_header(dependencies, headername, enums, messages):
|
|||||||
yield msg.fields_declaration() + '\n'
|
yield msg.fields_declaration() + '\n'
|
||||||
|
|
||||||
if messages:
|
if messages:
|
||||||
count_required_fields = lambda m: len([f for f in msg.fields if f.htype == 'PB_HTYPE_REQUIRED'])
|
count_required_fields = lambda m: len([f for f in msg.fields if f.rules == 'REQUIRED'])
|
||||||
largest_msg = max(messages, key = count_required_fields)
|
largest_msg = max(messages, key = count_required_fields)
|
||||||
largest_count = count_required_fields(largest_msg)
|
largest_count = count_required_fields(largest_msg)
|
||||||
if largest_count > 64:
|
if largest_count > 64:
|
||||||
|
|||||||
77
pb.h
77
pb.h
@@ -204,9 +204,84 @@ typedef enum {
|
|||||||
#define pb_membersize(st, m) (sizeof ((st*)0)->m)
|
#define pb_membersize(st, m) (sizeof ((st*)0)->m)
|
||||||
#define pb_arraysize(st, m) (pb_membersize(st, m) / pb_membersize(st, m[0]))
|
#define pb_arraysize(st, m) (pb_membersize(st, m) / pb_membersize(st, m[0]))
|
||||||
#define pb_delta(st, m1, m2) ((int)offsetof(st, m1) - (int)offsetof(st, m2))
|
#define pb_delta(st, m1, m2) ((int)offsetof(st, m1) - (int)offsetof(st, m2))
|
||||||
#define pb_delta_end(st, m1, m2) (offsetof(st, m1) - offsetof(st, m2) - pb_membersize(st, m2))
|
#define pb_delta_end(st, m1, m2) (offsetof(st, m1) == offsetof(st, m2) \
|
||||||
|
? offsetof(st, m1) \
|
||||||
|
: offsetof(st, m1) - offsetof(st, m2) - pb_membersize(st, m2))
|
||||||
#define PB_LAST_FIELD {0,(pb_type_t) 0,0,0,0,0,0}
|
#define PB_LAST_FIELD {0,(pb_type_t) 0,0,0,0,0,0}
|
||||||
|
|
||||||
|
/* Required fields are the simplest. They just have delta (padding) from
|
||||||
|
* previous field end, and the size of the field. Pointer is used for
|
||||||
|
* submessages and default values.
|
||||||
|
*/
|
||||||
|
#define PB_REQUIRED_STATIC(tag, st, m, pm, ltype, ptr) \
|
||||||
|
{tag, PB_HTYPE_REQUIRED | ltype, \
|
||||||
|
pb_delta_end(st, m, pm), 0, pb_membersize(st, m), 0, ptr}
|
||||||
|
|
||||||
|
/* Optional fields add the delta to the has_ variable. */
|
||||||
|
#define PB_OPTIONAL_STATIC(tag, st, m, pm, ltype, ptr) \
|
||||||
|
{tag, PB_HTYPE_OPTIONAL | ltype, \
|
||||||
|
pb_delta_end(st, m, pm), \
|
||||||
|
pb_delta(st, has_ ## m, m), \
|
||||||
|
pb_membersize(st, m), 0, ptr}
|
||||||
|
|
||||||
|
/* Repeated fields have a _count field and also the maximum number of entries. */
|
||||||
|
#define PB_REPEATED_STATIC(tag, st, m, pm, ltype, ptr) \
|
||||||
|
{tag, PB_HTYPE_ARRAY | ltype, \
|
||||||
|
pb_delta_end(st, m, pm), \
|
||||||
|
pb_delta(st, m ## _count, m), \
|
||||||
|
pb_membersize(st, m[0]), \
|
||||||
|
pb_arraysize(st, m), ptr}
|
||||||
|
|
||||||
|
/* Callbacks are much like required fields except with special datatype. */
|
||||||
|
#define PB_REQUIRED_CALLBACK(tag, st, m, pm, ltype, ptr) \
|
||||||
|
{tag, PB_HTYPE_CALLBACK | ltype, \
|
||||||
|
pb_delta_end(st, m, pm), 0, pb_membersize(st, m), 0, ptr}
|
||||||
|
|
||||||
|
#define PB_OPTIONAL_CALLBACK(tag, st, m, pm, ltype, ptr) \
|
||||||
|
{tag, PB_HTYPE_CALLBACK | ltype, \
|
||||||
|
pb_delta_end(st, m, pm), 0, pb_membersize(st, m), 0, ptr}
|
||||||
|
|
||||||
|
#define PB_REPEATED_CALLBACK(tag, st, m, pm, ltype, ptr) \
|
||||||
|
{tag, PB_HTYPE_CALLBACK | ltype, \
|
||||||
|
pb_delta_end(st, m, pm), 0, pb_membersize(st, m), 0, ptr}
|
||||||
|
|
||||||
|
/* The mapping from protobuf types to LTYPEs is done using these macros. */
|
||||||
|
#define PB_LTYPE_MAP_BOOL PB_LTYPE_VARINT
|
||||||
|
#define PB_LTYPE_MAP_BYTES PB_LTYPE_BYTES
|
||||||
|
#define PB_LTYPE_MAP_DOUBLE PB_LTYPE_FIXED64
|
||||||
|
#define PB_LTYPE_MAP_ENUM PB_LTYPE_VARINT
|
||||||
|
#define PB_LTYPE_MAP_FIXED32 PB_LTYPE_FIXED32
|
||||||
|
#define PB_LTYPE_MAP_FIXED64 PB_LTYPE_FIXED64
|
||||||
|
#define PB_LTYPE_MAP_FLOAT PB_LTYPE_FIXED32
|
||||||
|
#define PB_LTYPE_MAP_INT32 PB_LTYPE_VARINT
|
||||||
|
#define PB_LTYPE_MAP_INT64 PB_LTYPE_VARINT
|
||||||
|
#define PB_LTYPE_MAP_MESSAGE PB_LTYPE_SUBMESSAGE
|
||||||
|
#define PB_LTYPE_MAP_SFIXED32 PB_LTYPE_FIXED32
|
||||||
|
#define PB_LTYPE_MAP_SFIXED64 PB_LTYPE_FIXED64
|
||||||
|
#define PB_LTYPE_MAP_SINT32 PB_LTYPE_SVARINT
|
||||||
|
#define PB_LTYPE_MAP_SINT64 PB_LTYPE_SVARINT
|
||||||
|
#define PB_LTYPE_MAP_STRING PB_LTYPE_STRING
|
||||||
|
#define PB_LTYPE_MAP_UINT32 PB_LTYPE_VARINT
|
||||||
|
#define PB_LTYPE_MAP_UINT64 PB_LTYPE_VARINT
|
||||||
|
|
||||||
|
/* This is the actual macro used in field descriptions.
|
||||||
|
* It takes these arguments:
|
||||||
|
* - Field tag number
|
||||||
|
* - Field type: BOOL, BYTES, DOUBLE, ENUM, FIXED32, FIXED64,
|
||||||
|
* FLOAT, INT32, INT64, MESSAGE, SFIXED32, SFIXED64
|
||||||
|
* SINT32, SINT64, STRING, UINT32 or UINT64
|
||||||
|
* - Field rules: REQUIRED, OPTIONAL or REPEATED
|
||||||
|
* - Allocation: STATIC or CALLBACK
|
||||||
|
* - Message name
|
||||||
|
* - Field name
|
||||||
|
* - Previous field name (or field name again for first field)
|
||||||
|
* - Pointer to default value or submsg fields.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define PB_FIELD(tag, type, rules, allocation, message, field, prevfield, ptr) \
|
||||||
|
PB_ ## rules ## _ ## allocation(tag, message, field, prevfield, \
|
||||||
|
PB_LTYPE_MAP_ ## type, ptr)
|
||||||
|
|
||||||
/* These macros are used for giving out error messages.
|
/* These macros are used for giving out error messages.
|
||||||
* They are mostly a debugging aid; the main error information
|
* They are mostly a debugging aid; the main error information
|
||||||
* is the true/false return value from functions.
|
* is the true/false return value from functions.
|
||||||
|
|||||||
Reference in New Issue
Block a user