Compare commits
13 Commits
nanopb-0.2
...
nanopb-0.2
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7f397b067f | ||
|
|
586777b52f | ||
|
|
542463dbaa | ||
|
|
fe0bf121eb | ||
|
|
bd22cf2777 | ||
|
|
057165966c | ||
|
|
235219a295 | ||
|
|
d3ed0744d0 | ||
|
|
879860be19 | ||
|
|
a2f8112166 | ||
|
|
5efeb392e0 | ||
|
|
a46ed9f475 | ||
|
|
843fc4b11a |
@@ -1,3 +1,11 @@
|
||||
nanopb-0.2.6 (2014-02-15)
|
||||
Fix generator error with bytes callback fields (issue 99)
|
||||
Fix warnings about large integer constants (issue 102)
|
||||
Add comments to where STATIC_ASSERT is used (issue 96)
|
||||
Add warning about unknown field names on .options (issue 105)
|
||||
Move descriptor.proto to google/protobuf subdirectory (issue 104)
|
||||
Improved tests
|
||||
|
||||
nanopb-0.2.5 (2014-01-01)
|
||||
Fix a bug with encoding negative values in int32 fields (issue 97)
|
||||
Create binary packages of the generator + dependencies (issue 47)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
#!/usr/bin/python
|
||||
|
||||
'''Generate header file for nanopb from a ProtoBuf FileDescriptorSet.'''
|
||||
nanopb_version = "nanopb-0.2.5"
|
||||
nanopb_version = "nanopb-0.2.6"
|
||||
|
||||
import sys
|
||||
|
||||
@@ -15,6 +15,7 @@ except:
|
||||
|
||||
try:
|
||||
import google.protobuf.text_format as text_format
|
||||
import google.protobuf.descriptor_pb2 as descriptor
|
||||
except:
|
||||
sys.stderr.write('''
|
||||
*************************************************************
|
||||
@@ -26,7 +27,7 @@ except:
|
||||
|
||||
try:
|
||||
import proto.nanopb_pb2 as nanopb_pb2
|
||||
import proto.descriptor_pb2 as descriptor
|
||||
import proto.plugin_pb2 as plugin_pb2
|
||||
except:
|
||||
sys.stderr.write('''
|
||||
********************************************************************
|
||||
@@ -36,7 +37,6 @@ except:
|
||||
''' + '\n')
|
||||
raise
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Generation of single fields
|
||||
# ---------------------------------------------------------------------------
|
||||
@@ -169,6 +169,7 @@ class Field:
|
||||
self.max_count = None
|
||||
self.array_decl = ""
|
||||
self.enc_size = None
|
||||
self.ctype = None
|
||||
|
||||
# Parse field options
|
||||
if field_options.HasField("max_size"):
|
||||
@@ -314,6 +315,12 @@ class Field:
|
||||
data = self.default.decode('string_escape')
|
||||
data = ['0x%02x' % ord(c) for c in data]
|
||||
default = '{%d, {%s}}' % (len(data), ','.join(data))
|
||||
elif self.pbtype in ['FIXED32', 'UINT32']:
|
||||
default += 'u'
|
||||
elif self.pbtype in ['FIXED64', 'UINT64']:
|
||||
default += 'ull'
|
||||
elif self.pbtype in ['SFIXED64', 'INT64']:
|
||||
default += 'll'
|
||||
|
||||
if declaration_only:
|
||||
return 'extern const %s %s_default%s;' % (ctype, self.struct_name + self.name, array_decl)
|
||||
@@ -815,16 +822,6 @@ def generate_source(headername, enums, messages, extensions, options):
|
||||
if worst > 255 or checks:
|
||||
yield '\n/* Check that field information fits in pb_field_t */\n'
|
||||
|
||||
if worst < 65536:
|
||||
yield '#if !defined(PB_FIELD_16BIT) && !defined(PB_FIELD_32BIT)\n'
|
||||
if worst > 255:
|
||||
yield '#error Field descriptor for %s is too large. Define PB_FIELD_16BIT to fix this.\n' % worst_field
|
||||
else:
|
||||
assertion = ' && '.join(str(c) + ' < 256' for c in checks)
|
||||
msgs = '_'.join(str(n) for n in checks_msgnames)
|
||||
yield 'STATIC_ASSERT((%s), YOU_MUST_DEFINE_PB_FIELD_16BIT_FOR_MESSAGES_%s)\n'%(assertion,msgs)
|
||||
yield '#endif\n\n'
|
||||
|
||||
if worst > 65535 or checks:
|
||||
yield '#if !defined(PB_FIELD_32BIT)\n'
|
||||
if worst > 65535:
|
||||
@@ -832,8 +829,32 @@ def generate_source(headername, enums, messages, extensions, options):
|
||||
else:
|
||||
assertion = ' && '.join(str(c) + ' < 65536' for c in checks)
|
||||
msgs = '_'.join(str(n) for n in checks_msgnames)
|
||||
yield '/* If you get an error here, it means that you need to define PB_FIELD_32BIT\n'
|
||||
yield ' * compile-time option. You can do that in pb.h or on compiler command line.\n'
|
||||
yield ' * \n'
|
||||
yield ' * The reason you need to do this is that some of your messages contain tag\n'
|
||||
yield ' * numbers or field sizes that are larger than what can fit in 8 or 16 bit\n'
|
||||
yield ' * field descriptors.\n'
|
||||
yield ' */\n'
|
||||
yield 'STATIC_ASSERT((%s), YOU_MUST_DEFINE_PB_FIELD_32BIT_FOR_MESSAGES_%s)\n'%(assertion,msgs)
|
||||
yield '#endif\n'
|
||||
yield '#endif\n\n'
|
||||
|
||||
if worst < 65536:
|
||||
yield '#if !defined(PB_FIELD_16BIT) && !defined(PB_FIELD_32BIT)\n'
|
||||
if worst > 255:
|
||||
yield '#error Field descriptor for %s is too large. Define PB_FIELD_16BIT to fix this.\n' % worst_field
|
||||
else:
|
||||
assertion = ' && '.join(str(c) + ' < 256' for c in checks)
|
||||
msgs = '_'.join(str(n) for n in checks_msgnames)
|
||||
yield '/* If you get an error here, it means that you need to define PB_FIELD_16BIT\n'
|
||||
yield ' * compile-time option. You can do that in pb.h or on compiler command line.\n'
|
||||
yield ' * \n'
|
||||
yield ' * The reason you need to do this is that some of your messages contain tag\n'
|
||||
yield ' * numbers or field sizes that are larger than what can fit in the default\n'
|
||||
yield ' * 8 bit descriptors.\n'
|
||||
yield ' */\n'
|
||||
yield 'STATIC_ASSERT((%s), YOU_MUST_DEFINE_PB_FIELD_16BIT_FOR_MESSAGES_%s)\n'%(assertion,msgs)
|
||||
yield '#endif\n\n'
|
||||
|
||||
# Add check for sizeof(double)
|
||||
has_double = False
|
||||
@@ -879,6 +900,7 @@ class Globals:
|
||||
'''Ugly global variables, should find a good way to pass these.'''
|
||||
verbose_options = False
|
||||
separate_options = []
|
||||
matched_namemasks = set()
|
||||
|
||||
def get_nanopb_suboptions(subdesc, options, name):
|
||||
'''Get copy of options, and merge information from subdesc.'''
|
||||
@@ -889,6 +911,7 @@ def get_nanopb_suboptions(subdesc, options, name):
|
||||
dotname = '.'.join(name.parts)
|
||||
for namemask, options in Globals.separate_options:
|
||||
if fnmatch(dotname, namemask):
|
||||
Globals.matched_namemasks.add(namemask)
|
||||
new_options.MergeFrom(options)
|
||||
|
||||
# Handle options defined in .proto
|
||||
@@ -908,8 +931,8 @@ def get_nanopb_suboptions(subdesc, options, name):
|
||||
new_options.MergeFrom(ext)
|
||||
|
||||
if Globals.verbose_options:
|
||||
print "Options for " + dotname + ":"
|
||||
print text_format.MessageToString(new_options)
|
||||
sys.stderr.write("Options for " + dotname + ": ")
|
||||
sys.stderr.write(text_format.MessageToString(new_options) + "\n")
|
||||
|
||||
return new_options
|
||||
|
||||
@@ -973,13 +996,14 @@ def process_file(filename, fdesc, options):
|
||||
# No %s specified, use the filename as-is
|
||||
optfilename = options.options_file
|
||||
|
||||
if options.verbose:
|
||||
print 'Reading options from ' + optfilename
|
||||
|
||||
if os.path.isfile(optfilename):
|
||||
if options.verbose:
|
||||
sys.stderr.write('Reading options from ' + optfilename + '\n')
|
||||
|
||||
Globals.separate_options = read_options_file(open(optfilename, "rU"))
|
||||
else:
|
||||
Globals.separate_options = []
|
||||
Globals.matched_namemasks = set()
|
||||
|
||||
# Parse the file
|
||||
file_options = get_nanopb_suboptions(fdesc, toplevel_options, Names([filename]))
|
||||
@@ -1002,6 +1026,14 @@ def process_file(filename, fdesc, options):
|
||||
sourcedata = ''.join(generate_source(headerbasename, enums,
|
||||
messages, extensions, options))
|
||||
|
||||
# Check if there were any lines in .options that did not match a member
|
||||
unmatched = [n for n,o in Globals.separate_options if n not in Globals.matched_namemasks]
|
||||
if unmatched and not options.quiet:
|
||||
sys.stderr.write("Following patterns in " + optfilename + " did not match any fields: "
|
||||
+ ', '.join(unmatched) + "\n")
|
||||
if not Globals.verbose_options:
|
||||
sys.stderr.write("Use protoc --nanopb-out=-v:. to see a list of the field names.\n")
|
||||
|
||||
return {'headername': headername, 'headerdata': headerdata,
|
||||
'sourcename': sourcename, 'sourcedata': sourcedata}
|
||||
|
||||
@@ -1023,7 +1055,8 @@ def main_cli():
|
||||
results = process_file(filename, None, options)
|
||||
|
||||
if not options.quiet:
|
||||
print "Writing to " + results['headername'] + " and " + results['sourcename']
|
||||
sys.stderr.write("Writing to " + results['headername'] + " and "
|
||||
+ results['sourcename'] + "\n")
|
||||
|
||||
open(results['headername'], 'w').write(results['headerdata'])
|
||||
open(results['sourcename'], 'w').write(results['sourcedata'])
|
||||
@@ -1038,7 +1071,6 @@ def main_plugin():
|
||||
msvcrt.setmode(sys.stdin.fileno(), os.O_BINARY)
|
||||
msvcrt.setmode(sys.stdout.fileno(), os.O_BINARY)
|
||||
|
||||
import proto.plugin_pb2 as plugin_pb2
|
||||
data = sys.stdin.read()
|
||||
request = plugin_pb2.CodeGeneratorRequest.FromString(data)
|
||||
|
||||
@@ -1046,10 +1078,7 @@ def main_plugin():
|
||||
args = shlex.split(request.parameter)
|
||||
options, dummy = optparser.parse_args(args)
|
||||
|
||||
# We can't go printing stuff to stdout
|
||||
Globals.verbose_options = False
|
||||
options.verbose = False
|
||||
options.quiet = True
|
||||
Globals.verbose_options = options.verbose
|
||||
|
||||
response = plugin_pb2.CodeGeneratorResponse()
|
||||
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
all: nanopb_pb2.py plugin_pb2.py descriptor_pb2.py
|
||||
all: nanopb_pb2.py plugin_pb2.py
|
||||
|
||||
%_pb2.py: %.proto
|
||||
protoc --python_out=. $<
|
||||
|
||||
@@ -5,7 +5,7 @@
|
||||
// These are used by nanopb to generate statically allocable structures
|
||||
// for memory-limited environments.
|
||||
|
||||
import "descriptor.proto";
|
||||
import "google/protobuf/descriptor.proto";
|
||||
|
||||
option java_package = "fi.kapsi.koti.jpa.nanopb";
|
||||
|
||||
|
||||
@@ -46,7 +46,7 @@
|
||||
|
||||
package google.protobuf.compiler;
|
||||
|
||||
import "descriptor.proto";
|
||||
import "google/protobuf/descriptor.proto";
|
||||
|
||||
// An encoded CodeGeneratorRequest is written to the plugin's stdin.
|
||||
message CodeGeneratorRequest {
|
||||
|
||||
18
pb.h
18
pb.h
@@ -43,7 +43,7 @@
|
||||
|
||||
/* Version of the nanopb library. Just in case you want to check it in
|
||||
* your own program. */
|
||||
#define NANOPB_VERSION nanopb-0.2.5
|
||||
#define NANOPB_VERSION nanopb-0.2.6
|
||||
|
||||
/* Include all the system headers needed by nanopb. You will need the
|
||||
* definitions of the following:
|
||||
@@ -96,8 +96,14 @@
|
||||
#endif
|
||||
|
||||
/* Compile-time assertion, used for checking compatible compilation options.
|
||||
* If this fails on your compiler for some reason, use #define STATIC_ASSERT
|
||||
* to disable it. */
|
||||
* If this does not work properly on your compiler, use #define STATIC_ASSERT
|
||||
* to disable it.
|
||||
*
|
||||
* But before doing that, check carefully the error message / place where it
|
||||
* comes from to see if the error has a real cause. Unfortunately the error
|
||||
* message is not always very clear to read, but you can see the reason better
|
||||
* in the place where the STATIC_ASSERT macro was called.
|
||||
*/
|
||||
#ifndef STATIC_ASSERT
|
||||
#define STATIC_ASSERT(COND,MSG) typedef char STATIC_ASSERT_MSG(MSG, __LINE__, __COUNTER__)[(COND)?1:-1];
|
||||
#define STATIC_ASSERT_MSG(MSG, LINE, COUNTER) STATIC_ASSERT_MSG_(MSG, LINE, COUNTER)
|
||||
@@ -210,7 +216,11 @@ struct _pb_field_t {
|
||||
PB_PACKED_STRUCT_END
|
||||
|
||||
/* Make sure that the standard integer types are of the expected sizes.
|
||||
* All kinds of things may break otherwise.. atleast all fixed* types. */
|
||||
* All kinds of things may break otherwise.. atleast all fixed* types.
|
||||
*
|
||||
* If you get errors here, it probably means that your stdint.h is not
|
||||
* correct for your platform.
|
||||
*/
|
||||
STATIC_ASSERT(sizeof(int8_t) == 1, INT8_T_WRONG_SIZE)
|
||||
STATIC_ASSERT(sizeof(uint8_t) == 1, UINT8_T_WRONG_SIZE)
|
||||
STATIC_ASSERT(sizeof(int16_t) == 2, INT16_T_WRONG_SIZE)
|
||||
|
||||
@@ -74,6 +74,7 @@ if not env.GetOption('clean'):
|
||||
|
||||
# Check if we can use extra strict warning flags (only with GCC)
|
||||
extra = '-Wcast-qual -Wlogical-op -Wconversion'
|
||||
extra += ' -fstrict-aliasing -Wstrict-aliasing=1'
|
||||
extra += ' -Wmissing-prototypes -Wmissing-declarations -Wredundant-decls'
|
||||
extra += ' -Wstack-protector '
|
||||
if 'gcc' in env['CC']:
|
||||
|
||||
@@ -14,16 +14,16 @@ enum HugeEnum {
|
||||
}
|
||||
|
||||
message Limits {
|
||||
required int32 int32_min = 1;
|
||||
required int32 int32_max = 2;
|
||||
required uint32 uint32_min = 3;
|
||||
required uint32 uint32_max = 4;
|
||||
required int64 int64_min = 5;
|
||||
required int64 int64_max = 6;
|
||||
required uint64 uint64_min = 7;
|
||||
required uint64 uint64_max = 8;
|
||||
required HugeEnum enum_min = 9;
|
||||
required HugeEnum enum_max = 10;
|
||||
required int32 int32_min = 1 [default = 2147483647];
|
||||
required int32 int32_max = 2 [default = -2147483647];
|
||||
required uint32 uint32_min = 3 [default = 4294967295];
|
||||
required uint32 uint32_max = 4 [default = 0];
|
||||
required int64 int64_min = 5 [default = 9223372036854775807];
|
||||
required int64 int64_max = 6 [default = -9223372036854775807];
|
||||
required uint64 uint64_min = 7 [default = 18446744073709551615];
|
||||
required uint64 uint64_max = 8 [default = 0];
|
||||
required HugeEnum enum_min = 9 [default = Positive];
|
||||
required HugeEnum enum_max = 10 [default = Negative];
|
||||
}
|
||||
|
||||
enum MyEnum {
|
||||
|
||||
23
tests/alltypes_callback/SConscript
Normal file
23
tests/alltypes_callback/SConscript
Normal file
@@ -0,0 +1,23 @@
|
||||
# Test the AllTypes encoding & decoding using callbacks for all fields.
|
||||
|
||||
Import("env")
|
||||
|
||||
c = Copy("$TARGET", "$SOURCE")
|
||||
env.Command("alltypes.proto", "#alltypes/alltypes.proto", c)
|
||||
|
||||
env.NanopbProto(["alltypes", "alltypes.options"])
|
||||
enc = env.Program(["encode_alltypes_callback.c", "alltypes.pb.c", "$COMMON/pb_encode.o"])
|
||||
dec = env.Program(["decode_alltypes_callback.c", "alltypes.pb.c", "$COMMON/pb_decode.o"])
|
||||
|
||||
refdec = "$BUILD/alltypes/decode_alltypes$PROGSUFFIX"
|
||||
|
||||
# Encode and compare results
|
||||
env.RunTest(enc)
|
||||
env.RunTest("decode_alltypes.output", [refdec, "encode_alltypes_callback.output"])
|
||||
env.RunTest("decode_alltypes_callback.output", [dec, "encode_alltypes_callback.output"])
|
||||
|
||||
# Do the same thing with the optional fields present
|
||||
env.RunTest("optionals.output", enc, ARGS = ['1'])
|
||||
env.RunTest("optionals.refdecout", [refdec, "optionals.output"], ARGS = ['1'])
|
||||
env.RunTest("optionals.decout", [dec, "optionals.output"], ARGS = ['1'])
|
||||
|
||||
3
tests/alltypes_callback/alltypes.options
Normal file
3
tests/alltypes_callback/alltypes.options
Normal file
@@ -0,0 +1,3 @@
|
||||
# Generate all fields as callbacks.
|
||||
AllTypes.* type:FT_CALLBACK
|
||||
SubMessage.substuff1 max_size:16
|
||||
423
tests/alltypes_callback/decode_alltypes_callback.c
Normal file
423
tests/alltypes_callback/decode_alltypes_callback.c
Normal file
@@ -0,0 +1,423 @@
|
||||
/* Attempts to test all the datatypes supported by ProtoBuf when used as callback fields.
|
||||
* Note that normally there would be no reason to use callback fields for this,
|
||||
* because each encoder defined here only gives a single field.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
#include <pb_decode.h>
|
||||
#include "alltypes.pb.h"
|
||||
#include "test_helpers.h"
|
||||
|
||||
#define TEST(x) if (!(x)) { \
|
||||
printf("Test " #x " failed (in field %d).\n", field->tag); \
|
||||
return false; \
|
||||
}
|
||||
|
||||
static bool read_varint(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
uint64_t value;
|
||||
if (!pb_decode_varint(stream, &value))
|
||||
return false;
|
||||
|
||||
TEST((int64_t)value == (long)*arg);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_svarint(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
int64_t value;
|
||||
if (!pb_decode_svarint(stream, &value))
|
||||
return false;
|
||||
|
||||
TEST(value == (long)*arg);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_fixed32(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
uint32_t value;
|
||||
if (!pb_decode_fixed32(stream, &value))
|
||||
return false;
|
||||
|
||||
TEST(value == *(uint32_t*)*arg);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_fixed64(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
uint64_t value;
|
||||
if (!pb_decode_fixed64(stream, &value))
|
||||
return false;
|
||||
|
||||
TEST(value == *(uint64_t*)*arg);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_string(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
uint8_t buf[16] = {0};
|
||||
size_t len = stream->bytes_left;
|
||||
|
||||
if (len > sizeof(buf) - 1 || !pb_read(stream, buf, len))
|
||||
return false;
|
||||
|
||||
TEST(strcmp((char*)buf, *arg) == 0);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_submsg(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
SubMessage submsg = {""};
|
||||
|
||||
if (!pb_decode(stream, SubMessage_fields, &submsg))
|
||||
return false;
|
||||
|
||||
TEST(memcmp(&submsg, *arg, sizeof(submsg)));
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_emptymsg(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
EmptyMessage emptymsg = {0};
|
||||
return pb_decode(stream, EmptyMessage_fields, &emptymsg);
|
||||
}
|
||||
|
||||
static bool read_repeated_varint(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
int32_t** expected = (int32_t**)arg;
|
||||
uint64_t value;
|
||||
if (!pb_decode_varint(stream, &value))
|
||||
return false;
|
||||
|
||||
TEST(*(*expected)++ == value);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_repeated_svarint(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
int32_t** expected = (int32_t**)arg;
|
||||
int64_t value;
|
||||
if (!pb_decode_svarint(stream, &value))
|
||||
return false;
|
||||
|
||||
TEST(*(*expected)++ == value);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_repeated_fixed32(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
uint32_t** expected = (uint32_t**)arg;
|
||||
uint32_t value;
|
||||
if (!pb_decode_fixed32(stream, &value))
|
||||
return false;
|
||||
|
||||
TEST(*(*expected)++ == value);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_repeated_fixed64(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
uint64_t** expected = (uint64_t**)arg;
|
||||
uint64_t value;
|
||||
if (!pb_decode_fixed64(stream, &value))
|
||||
return false;
|
||||
|
||||
TEST(*(*expected)++ == value);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_repeated_string(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
uint8_t*** expected = (uint8_t***)arg;
|
||||
uint8_t buf[16] = {0};
|
||||
size_t len = stream->bytes_left;
|
||||
|
||||
if (len > sizeof(buf) - 1 || !pb_read(stream, buf, len))
|
||||
return false;
|
||||
|
||||
TEST(strcmp((char*)*(*expected)++, (char*)buf) == 0);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_repeated_submsg(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
SubMessage** expected = (SubMessage**)arg;
|
||||
SubMessage decoded = {""};
|
||||
if (!pb_decode(stream, SubMessage_fields, &decoded))
|
||||
return false;
|
||||
|
||||
TEST(memcmp((*expected)++, &decoded, sizeof(decoded)) == 0);
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool read_limits(pb_istream_t *stream, const pb_field_t *field, void **arg)
|
||||
{
|
||||
Limits decoded = {0};
|
||||
if (!pb_decode(stream, Limits_fields, &decoded))
|
||||
return false;
|
||||
|
||||
TEST(decoded.int32_min == INT32_MIN);
|
||||
TEST(decoded.int32_max == INT32_MAX);
|
||||
TEST(decoded.uint32_min == 0);
|
||||
TEST(decoded.uint32_max == UINT32_MAX);
|
||||
TEST(decoded.int64_min == INT64_MIN);
|
||||
TEST(decoded.int64_max == INT64_MAX);
|
||||
TEST(decoded.uint64_min == 0);
|
||||
TEST(decoded.uint64_max == UINT64_MAX);
|
||||
TEST(decoded.enum_min == HugeEnum_Negative);
|
||||
TEST(decoded.enum_max == HugeEnum_Positive);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/* This function is called once from main(), it handles
|
||||
the decoding and checks the fields. */
|
||||
bool check_alltypes(pb_istream_t *stream, int mode)
|
||||
{
|
||||
/* Values for use from callbacks through pointers. */
|
||||
uint32_t req_fixed32 = 1008;
|
||||
int32_t req_sfixed32 = -1009;
|
||||
float req_float = 1010.0f;
|
||||
uint64_t req_fixed64 = 1011;
|
||||
int64_t req_sfixed64 = -1012;
|
||||
double req_double = 1013.0;
|
||||
SubMessage req_submsg = {"1016", 1016};
|
||||
|
||||
int32_t rep_int32[5] = {0, 0, 0, 0, -2001};
|
||||
int32_t rep_int64[5] = {0, 0, 0, 0, -2002};
|
||||
int32_t rep_uint32[5] = {0, 0, 0, 0, 2003};
|
||||
int32_t rep_uint64[5] = {0, 0, 0, 0, 2004};
|
||||
int32_t rep_sint32[5] = {0, 0, 0, 0, -2005};
|
||||
int32_t rep_sint64[5] = {0, 0, 0, 0, -2006};
|
||||
int32_t rep_bool[5] = {false, false, false, false, true};
|
||||
uint32_t rep_fixed32[5] = {0, 0, 0, 0, 2008};
|
||||
int32_t rep_sfixed32[5] = {0, 0, 0, 0, -2009};
|
||||
float rep_float[5] = {0, 0, 0, 0, 2010.0f};
|
||||
uint64_t rep_fixed64[5] = {0, 0, 0, 0, 2011};
|
||||
int64_t rep_sfixed64[5] = {0, 0, 0, 0, -2012};
|
||||
double rep_double[5] = {0, 0, 0, 0, 2013.0};
|
||||
char* rep_string[5] = {"", "", "", "", "2014"};
|
||||
char* rep_bytes[5] = {"", "", "", "", "2015"};
|
||||
SubMessage rep_submsg[5] = {{"", 0, 0, 3},
|
||||
{"", 0, 0, 3},
|
||||
{"", 0, 0, 3},
|
||||
{"", 0, 0, 3},
|
||||
{"2016", 2016, true, 2016}};
|
||||
int32_t rep_enum[5] = {0, 0, 0, 0, MyEnum_Truth};
|
||||
|
||||
uint32_t opt_fixed32 = 3048;
|
||||
int32_t opt_sfixed32 = 3049;
|
||||
float opt_float = 3050.0f;
|
||||
uint64_t opt_fixed64 = 3051;
|
||||
int64_t opt_sfixed64 = 3052;
|
||||
double opt_double = 3053.0f;
|
||||
SubMessage opt_submsg = {"3056", 3056};
|
||||
|
||||
/* Bind callbacks for required fields */
|
||||
AllTypes alltypes;
|
||||
|
||||
/* Fill with garbage to better detect initialization errors */
|
||||
memset(&alltypes, 0xAA, sizeof(alltypes));
|
||||
|
||||
alltypes.req_int32.funcs.decode = &read_varint;
|
||||
alltypes.req_int32.arg = (void*)-1001;
|
||||
|
||||
alltypes.req_int64.funcs.decode = &read_varint;
|
||||
alltypes.req_int64.arg = (void*)-1002;
|
||||
|
||||
alltypes.req_uint32.funcs.decode = &read_varint;
|
||||
alltypes.req_uint32.arg = (void*)1003;
|
||||
|
||||
alltypes.req_uint32.funcs.decode = &read_varint;
|
||||
alltypes.req_uint32.arg = (void*)1003;
|
||||
|
||||
alltypes.req_uint64.funcs.decode = &read_varint;
|
||||
alltypes.req_uint64.arg = (void*)1004;
|
||||
|
||||
alltypes.req_sint32.funcs.decode = &read_svarint;
|
||||
alltypes.req_sint32.arg = (void*)-1005;
|
||||
|
||||
alltypes.req_sint64.funcs.decode = &read_svarint;
|
||||
alltypes.req_sint64.arg = (void*)-1006;
|
||||
|
||||
alltypes.req_bool.funcs.decode = &read_varint;
|
||||
alltypes.req_bool.arg = (void*)true;
|
||||
|
||||
alltypes.req_fixed32.funcs.decode = &read_fixed32;
|
||||
alltypes.req_fixed32.arg = &req_fixed32;
|
||||
|
||||
alltypes.req_sfixed32.funcs.decode = &read_fixed32;
|
||||
alltypes.req_sfixed32.arg = &req_sfixed32;
|
||||
|
||||
alltypes.req_float.funcs.decode = &read_fixed32;
|
||||
alltypes.req_float.arg = &req_float;
|
||||
|
||||
alltypes.req_fixed64.funcs.decode = &read_fixed64;
|
||||
alltypes.req_fixed64.arg = &req_fixed64;
|
||||
|
||||
alltypes.req_sfixed64.funcs.decode = &read_fixed64;
|
||||
alltypes.req_sfixed64.arg = &req_sfixed64;
|
||||
|
||||
alltypes.req_double.funcs.decode = &read_fixed64;
|
||||
alltypes.req_double.arg = &req_double;
|
||||
|
||||
alltypes.req_string.funcs.decode = &read_string;
|
||||
alltypes.req_string.arg = "1014";
|
||||
|
||||
alltypes.req_bytes.funcs.decode = &read_string;
|
||||
alltypes.req_bytes.arg = "1015";
|
||||
|
||||
alltypes.req_submsg.funcs.decode = &read_submsg;
|
||||
alltypes.req_submsg.arg = &req_submsg;
|
||||
|
||||
alltypes.req_enum.funcs.decode = &read_varint;
|
||||
alltypes.req_enum.arg = (void*)MyEnum_Truth;
|
||||
|
||||
alltypes.req_emptymsg.funcs.decode = &read_emptymsg;
|
||||
|
||||
/* Bind callbacks for repeated fields */
|
||||
alltypes.rep_int32.funcs.decode = &read_repeated_varint;
|
||||
alltypes.rep_int32.arg = rep_int32;
|
||||
|
||||
alltypes.rep_int64.funcs.decode = &read_repeated_varint;
|
||||
alltypes.rep_int64.arg = rep_int64;
|
||||
|
||||
alltypes.rep_uint32.funcs.decode = &read_repeated_varint;
|
||||
alltypes.rep_uint32.arg = rep_uint32;
|
||||
|
||||
alltypes.rep_uint64.funcs.decode = &read_repeated_varint;
|
||||
alltypes.rep_uint64.arg = rep_uint64;
|
||||
|
||||
alltypes.rep_sint32.funcs.decode = &read_repeated_svarint;
|
||||
alltypes.rep_sint32.arg = rep_sint32;
|
||||
|
||||
alltypes.rep_sint64.funcs.decode = &read_repeated_svarint;
|
||||
alltypes.rep_sint64.arg = rep_sint64;
|
||||
|
||||
alltypes.rep_bool.funcs.decode = &read_repeated_varint;
|
||||
alltypes.rep_bool.arg = rep_bool;
|
||||
|
||||
alltypes.rep_fixed32.funcs.decode = &read_repeated_fixed32;
|
||||
alltypes.rep_fixed32.arg = rep_fixed32;
|
||||
|
||||
alltypes.rep_sfixed32.funcs.decode = &read_repeated_fixed32;
|
||||
alltypes.rep_sfixed32.arg = rep_sfixed32;
|
||||
|
||||
alltypes.rep_float.funcs.decode = &read_repeated_fixed32;
|
||||
alltypes.rep_float.arg = rep_float;
|
||||
|
||||
alltypes.rep_fixed64.funcs.decode = &read_repeated_fixed64;
|
||||
alltypes.rep_fixed64.arg = rep_fixed64;
|
||||
|
||||
alltypes.rep_sfixed64.funcs.decode = &read_repeated_fixed64;
|
||||
alltypes.rep_sfixed64.arg = rep_sfixed64;
|
||||
|
||||
alltypes.rep_double.funcs.decode = &read_repeated_fixed64;
|
||||
alltypes.rep_double.arg = rep_double;
|
||||
|
||||
alltypes.rep_string.funcs.decode = &read_repeated_string;
|
||||
alltypes.rep_string.arg = rep_string;
|
||||
|
||||
alltypes.rep_bytes.funcs.decode = &read_repeated_string;
|
||||
alltypes.rep_bytes.arg = rep_bytes;
|
||||
|
||||
alltypes.rep_submsg.funcs.decode = &read_repeated_submsg;
|
||||
alltypes.rep_submsg.arg = rep_submsg;
|
||||
|
||||
alltypes.rep_enum.funcs.decode = &read_repeated_varint;
|
||||
alltypes.rep_enum.arg = rep_enum;
|
||||
|
||||
alltypes.rep_emptymsg.funcs.decode = &read_emptymsg;
|
||||
|
||||
alltypes.req_limits.funcs.decode = &read_limits;
|
||||
|
||||
alltypes.end.funcs.decode = &read_varint;
|
||||
alltypes.end.arg = (void*)1099;
|
||||
|
||||
/* Bind callbacks for optional fields */
|
||||
if (mode == 1)
|
||||
{
|
||||
alltypes.opt_int32.funcs.decode = &read_varint;
|
||||
alltypes.opt_int32.arg = (void*)3041;
|
||||
|
||||
alltypes.opt_int64.funcs.decode = &read_varint;
|
||||
alltypes.opt_int64.arg = (void*)3042;
|
||||
|
||||
alltypes.opt_uint32.funcs.decode = &read_varint;
|
||||
alltypes.opt_uint32.arg = (void*)3043;
|
||||
|
||||
alltypes.opt_uint64.funcs.decode = &read_varint;
|
||||
alltypes.opt_uint64.arg = (void*)3044;
|
||||
|
||||
alltypes.opt_sint32.funcs.decode = &read_svarint;
|
||||
alltypes.opt_sint32.arg = (void*)3045;
|
||||
|
||||
alltypes.opt_sint64.funcs.decode = &read_svarint;
|
||||
alltypes.opt_sint64.arg = (void*)3046;
|
||||
|
||||
alltypes.opt_bool.funcs.decode = &read_varint;
|
||||
alltypes.opt_bool.arg = (void*)true;
|
||||
|
||||
alltypes.opt_fixed32.funcs.decode = &read_fixed32;
|
||||
alltypes.opt_fixed32.arg = &opt_fixed32;
|
||||
|
||||
alltypes.opt_sfixed32.funcs.decode = &read_fixed32;
|
||||
alltypes.opt_sfixed32.arg = &opt_sfixed32;
|
||||
|
||||
alltypes.opt_float.funcs.decode = &read_fixed32;
|
||||
alltypes.opt_float.arg = &opt_float;
|
||||
|
||||
alltypes.opt_fixed64.funcs.decode = &read_fixed64;
|
||||
alltypes.opt_fixed64.arg = &opt_fixed64;
|
||||
|
||||
alltypes.opt_sfixed64.funcs.decode = &read_fixed64;
|
||||
alltypes.opt_sfixed64.arg = &opt_sfixed64;
|
||||
|
||||
alltypes.opt_double.funcs.decode = &read_fixed64;
|
||||
alltypes.opt_double.arg = &opt_double;
|
||||
|
||||
alltypes.opt_string.funcs.decode = &read_string;
|
||||
alltypes.opt_string.arg = "3054";
|
||||
|
||||
alltypes.opt_bytes.funcs.decode = &read_string;
|
||||
alltypes.opt_bytes.arg = "3055";
|
||||
|
||||
alltypes.opt_submsg.funcs.decode = &read_submsg;
|
||||
alltypes.opt_submsg.arg = &opt_submsg;
|
||||
|
||||
alltypes.opt_enum.funcs.decode = &read_varint;
|
||||
alltypes.opt_enum.arg = (void*)MyEnum_Truth;
|
||||
|
||||
alltypes.opt_emptymsg.funcs.decode = &read_emptymsg;
|
||||
}
|
||||
|
||||
return pb_decode(stream, AllTypes_fields, &alltypes);
|
||||
}
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
uint8_t buffer[1024];
|
||||
size_t count;
|
||||
pb_istream_t stream;
|
||||
|
||||
/* Whether to expect the optional values or the default values. */
|
||||
int mode = (argc > 1) ? atoi(argv[1]) : 0;
|
||||
|
||||
/* Read the data into buffer */
|
||||
SET_BINARY_MODE(stdin);
|
||||
count = fread(buffer, 1, sizeof(buffer), stdin);
|
||||
|
||||
/* Construct a pb_istream_t for reading from the buffer */
|
||||
stream = pb_istream_from_buffer(buffer, count);
|
||||
|
||||
/* Decode and print out the stuff */
|
||||
if (!check_alltypes(&stream, mode))
|
||||
{
|
||||
printf("Parsing failed: %s\n", PB_GET_ERROR(&stream));
|
||||
return 1;
|
||||
} else {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
397
tests/alltypes_callback/encode_alltypes_callback.c
Normal file
397
tests/alltypes_callback/encode_alltypes_callback.c
Normal file
@@ -0,0 +1,397 @@
|
||||
/* Attempts to test all the datatypes supported by ProtoBuf when used as callback fields.
|
||||
* Note that normally there would be no reason to use callback fields for this,
|
||||
* because each encoder defined here only gives a single field.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <pb_encode.h>
|
||||
#include "alltypes.pb.h"
|
||||
#include "test_helpers.h"
|
||||
|
||||
static bool write_varint(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_varint(stream, (long)*arg);
|
||||
}
|
||||
|
||||
static bool write_svarint(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_svarint(stream, (long)*arg);
|
||||
}
|
||||
|
||||
static bool write_fixed32(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_fixed32(stream, *arg);
|
||||
}
|
||||
|
||||
static bool write_fixed64(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_fixed64(stream, *arg);
|
||||
}
|
||||
|
||||
static bool write_string(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_string(stream, *arg, strlen(*arg));
|
||||
}
|
||||
|
||||
static bool write_submsg(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, SubMessage_fields, *arg);
|
||||
}
|
||||
|
||||
static bool write_emptymsg(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
EmptyMessage emptymsg = {0};
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, EmptyMessage_fields, &emptymsg);
|
||||
}
|
||||
|
||||
static bool write_repeated_varint(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_varint(stream, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_varint(stream, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_varint(stream, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_varint(stream, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_varint(stream, (long)*arg);
|
||||
}
|
||||
|
||||
static bool write_repeated_svarint(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_svarint(stream, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_svarint(stream, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_svarint(stream, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_svarint(stream, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_svarint(stream, (long)*arg);
|
||||
}
|
||||
|
||||
static bool write_repeated_fixed32(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
uint32_t dummy = 0;
|
||||
|
||||
/* Make it a packed field */
|
||||
return pb_encode_tag(stream, PB_WT_STRING, field->tag) &&
|
||||
pb_encode_varint(stream, 5 * 4) && /* Number of bytes */
|
||||
pb_encode_fixed32(stream, &dummy) &&
|
||||
pb_encode_fixed32(stream, &dummy) &&
|
||||
pb_encode_fixed32(stream, &dummy) &&
|
||||
pb_encode_fixed32(stream, &dummy) &&
|
||||
pb_encode_fixed32(stream, *arg);
|
||||
}
|
||||
|
||||
static bool write_repeated_fixed64(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
uint64_t dummy = 0;
|
||||
|
||||
/* Make it a packed field */
|
||||
return pb_encode_tag(stream, PB_WT_STRING, field->tag) &&
|
||||
pb_encode_varint(stream, 5 * 8) && /* Number of bytes */
|
||||
pb_encode_fixed64(stream, &dummy) &&
|
||||
pb_encode_fixed64(stream, &dummy) &&
|
||||
pb_encode_fixed64(stream, &dummy) &&
|
||||
pb_encode_fixed64(stream, &dummy) &&
|
||||
pb_encode_fixed64(stream, *arg);
|
||||
}
|
||||
|
||||
static bool write_repeated_string(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_string(stream, 0, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_string(stream, 0, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_string(stream, 0, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_string(stream, 0, 0) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_string(stream, *arg, strlen(*arg));
|
||||
}
|
||||
|
||||
static bool write_repeated_submsg(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
SubMessage dummy = {""};
|
||||
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, SubMessage_fields, &dummy) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, SubMessage_fields, &dummy) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, SubMessage_fields, &dummy) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, SubMessage_fields, &dummy) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, SubMessage_fields, *arg);
|
||||
}
|
||||
|
||||
static bool write_limits(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
Limits limits = {0};
|
||||
limits.int32_min = INT32_MIN;
|
||||
limits.int32_max = INT32_MAX;
|
||||
limits.uint32_min = 0;
|
||||
limits.uint32_max = UINT32_MAX;
|
||||
limits.int64_min = INT64_MIN;
|
||||
limits.int64_max = INT64_MAX;
|
||||
limits.uint64_min = 0;
|
||||
limits.uint64_max = UINT64_MAX;
|
||||
limits.enum_min = HugeEnum_Negative;
|
||||
limits.enum_max = HugeEnum_Positive;
|
||||
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, Limits_fields, &limits);
|
||||
}
|
||||
|
||||
static bool write_repeated_emptymsg(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
|
||||
{
|
||||
EmptyMessage emptymsg = {0};
|
||||
return pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, EmptyMessage_fields, &emptymsg) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, EmptyMessage_fields, &emptymsg) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, EmptyMessage_fields, &emptymsg) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, EmptyMessage_fields, &emptymsg) &&
|
||||
pb_encode_tag_for_field(stream, field) &&
|
||||
pb_encode_submessage(stream, EmptyMessage_fields, &emptymsg);
|
||||
}
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
int mode = (argc > 1) ? atoi(argv[1]) : 0;
|
||||
|
||||
/* Values for use from callbacks through pointers. */
|
||||
uint32_t req_fixed32 = 1008;
|
||||
int32_t req_sfixed32 = -1009;
|
||||
float req_float = 1010.0f;
|
||||
uint64_t req_fixed64 = 1011;
|
||||
int64_t req_sfixed64 = -1012;
|
||||
double req_double = 1013.0;
|
||||
SubMessage req_submsg = {"1016", 1016};
|
||||
|
||||
uint32_t rep_fixed32 = 2008;
|
||||
int32_t rep_sfixed32 = -2009;
|
||||
float rep_float = 2010.0f;
|
||||
uint64_t rep_fixed64 = 2011;
|
||||
int64_t rep_sfixed64 = -2012;
|
||||
double rep_double = 2013.0;
|
||||
SubMessage rep_submsg = {"2016", 2016, true, 2016};
|
||||
|
||||
uint32_t opt_fixed32 = 3048;
|
||||
int32_t opt_sfixed32 = 3049;
|
||||
float opt_float = 3050.0f;
|
||||
uint64_t opt_fixed64 = 3051;
|
||||
int64_t opt_sfixed64 = 3052;
|
||||
double opt_double = 3053.0f;
|
||||
SubMessage opt_submsg = {"3056", 3056};
|
||||
|
||||
/* Bind callbacks for required fields */
|
||||
AllTypes alltypes = {{{0}}};
|
||||
|
||||
alltypes.req_int32.funcs.encode = &write_varint;
|
||||
alltypes.req_int32.arg = (void*)-1001;
|
||||
|
||||
alltypes.req_int64.funcs.encode = &write_varint;
|
||||
alltypes.req_int64.arg = (void*)-1002;
|
||||
|
||||
alltypes.req_uint32.funcs.encode = &write_varint;
|
||||
alltypes.req_uint32.arg = (void*)1003;
|
||||
|
||||
alltypes.req_uint32.funcs.encode = &write_varint;
|
||||
alltypes.req_uint32.arg = (void*)1003;
|
||||
|
||||
alltypes.req_uint64.funcs.encode = &write_varint;
|
||||
alltypes.req_uint64.arg = (void*)1004;
|
||||
|
||||
alltypes.req_sint32.funcs.encode = &write_svarint;
|
||||
alltypes.req_sint32.arg = (void*)-1005;
|
||||
|
||||
alltypes.req_sint64.funcs.encode = &write_svarint;
|
||||
alltypes.req_sint64.arg = (void*)-1006;
|
||||
|
||||
alltypes.req_bool.funcs.encode = &write_varint;
|
||||
alltypes.req_bool.arg = (void*)true;
|
||||
|
||||
alltypes.req_fixed32.funcs.encode = &write_fixed32;
|
||||
alltypes.req_fixed32.arg = &req_fixed32;
|
||||
|
||||
alltypes.req_sfixed32.funcs.encode = &write_fixed32;
|
||||
alltypes.req_sfixed32.arg = &req_sfixed32;
|
||||
|
||||
alltypes.req_float.funcs.encode = &write_fixed32;
|
||||
alltypes.req_float.arg = &req_float;
|
||||
|
||||
alltypes.req_fixed64.funcs.encode = &write_fixed64;
|
||||
alltypes.req_fixed64.arg = &req_fixed64;
|
||||
|
||||
alltypes.req_sfixed64.funcs.encode = &write_fixed64;
|
||||
alltypes.req_sfixed64.arg = &req_sfixed64;
|
||||
|
||||
alltypes.req_double.funcs.encode = &write_fixed64;
|
||||
alltypes.req_double.arg = &req_double;
|
||||
|
||||
alltypes.req_string.funcs.encode = &write_string;
|
||||
alltypes.req_string.arg = "1014";
|
||||
|
||||
alltypes.req_bytes.funcs.encode = &write_string;
|
||||
alltypes.req_bytes.arg = "1015";
|
||||
|
||||
alltypes.req_submsg.funcs.encode = &write_submsg;
|
||||
alltypes.req_submsg.arg = &req_submsg;
|
||||
|
||||
alltypes.req_enum.funcs.encode = &write_varint;
|
||||
alltypes.req_enum.arg = (void*)MyEnum_Truth;
|
||||
|
||||
alltypes.req_emptymsg.funcs.encode = &write_emptymsg;
|
||||
|
||||
/* Bind callbacks for repeated fields */
|
||||
alltypes.rep_int32.funcs.encode = &write_repeated_varint;
|
||||
alltypes.rep_int32.arg = (void*)-2001;
|
||||
|
||||
alltypes.rep_int64.funcs.encode = &write_repeated_varint;
|
||||
alltypes.rep_int64.arg = (void*)-2002;
|
||||
|
||||
alltypes.rep_uint32.funcs.encode = &write_repeated_varint;
|
||||
alltypes.rep_uint32.arg = (void*)2003;
|
||||
|
||||
alltypes.rep_uint64.funcs.encode = &write_repeated_varint;
|
||||
alltypes.rep_uint64.arg = (void*)2004;
|
||||
|
||||
alltypes.rep_sint32.funcs.encode = &write_repeated_svarint;
|
||||
alltypes.rep_sint32.arg = (void*)-2005;
|
||||
|
||||
alltypes.rep_sint64.funcs.encode = &write_repeated_svarint;
|
||||
alltypes.rep_sint64.arg = (void*)-2006;
|
||||
|
||||
alltypes.rep_bool.funcs.encode = &write_repeated_varint;
|
||||
alltypes.rep_bool.arg = (void*)true;
|
||||
|
||||
alltypes.rep_fixed32.funcs.encode = &write_repeated_fixed32;
|
||||
alltypes.rep_fixed32.arg = &rep_fixed32;
|
||||
|
||||
alltypes.rep_sfixed32.funcs.encode = &write_repeated_fixed32;
|
||||
alltypes.rep_sfixed32.arg = &rep_sfixed32;
|
||||
|
||||
alltypes.rep_float.funcs.encode = &write_repeated_fixed32;
|
||||
alltypes.rep_float.arg = &rep_float;
|
||||
|
||||
alltypes.rep_fixed64.funcs.encode = &write_repeated_fixed64;
|
||||
alltypes.rep_fixed64.arg = &rep_fixed64;
|
||||
|
||||
alltypes.rep_sfixed64.funcs.encode = &write_repeated_fixed64;
|
||||
alltypes.rep_sfixed64.arg = &rep_sfixed64;
|
||||
|
||||
alltypes.rep_double.funcs.encode = &write_repeated_fixed64;
|
||||
alltypes.rep_double.arg = &rep_double;
|
||||
|
||||
alltypes.rep_string.funcs.encode = &write_repeated_string;
|
||||
alltypes.rep_string.arg = "2014";
|
||||
|
||||
alltypes.rep_bytes.funcs.encode = &write_repeated_string;
|
||||
alltypes.rep_bytes.arg = "2015";
|
||||
|
||||
alltypes.rep_submsg.funcs.encode = &write_repeated_submsg;
|
||||
alltypes.rep_submsg.arg = &rep_submsg;
|
||||
|
||||
alltypes.rep_enum.funcs.encode = &write_repeated_varint;
|
||||
alltypes.rep_enum.arg = (void*)MyEnum_Truth;
|
||||
|
||||
alltypes.rep_emptymsg.funcs.encode = &write_repeated_emptymsg;
|
||||
|
||||
alltypes.req_limits.funcs.encode = &write_limits;
|
||||
|
||||
/* Bind callbacks for optional fields */
|
||||
if (mode != 0)
|
||||
{
|
||||
alltypes.opt_int32.funcs.encode = &write_varint;
|
||||
alltypes.opt_int32.arg = (void*)3041;
|
||||
|
||||
alltypes.opt_int64.funcs.encode = &write_varint;
|
||||
alltypes.opt_int64.arg = (void*)3042;
|
||||
|
||||
alltypes.opt_uint32.funcs.encode = &write_varint;
|
||||
alltypes.opt_uint32.arg = (void*)3043;
|
||||
|
||||
alltypes.opt_uint64.funcs.encode = &write_varint;
|
||||
alltypes.opt_uint64.arg = (void*)3044;
|
||||
|
||||
alltypes.opt_sint32.funcs.encode = &write_svarint;
|
||||
alltypes.opt_sint32.arg = (void*)3045;
|
||||
|
||||
alltypes.opt_sint64.funcs.encode = &write_svarint;
|
||||
alltypes.opt_sint64.arg = (void*)3046;
|
||||
|
||||
alltypes.opt_bool.funcs.encode = &write_varint;
|
||||
alltypes.opt_bool.arg = (void*)true;
|
||||
|
||||
alltypes.opt_fixed32.funcs.encode = &write_fixed32;
|
||||
alltypes.opt_fixed32.arg = &opt_fixed32;
|
||||
|
||||
alltypes.opt_sfixed32.funcs.encode = &write_fixed32;
|
||||
alltypes.opt_sfixed32.arg = &opt_sfixed32;
|
||||
|
||||
alltypes.opt_float.funcs.encode = &write_fixed32;
|
||||
alltypes.opt_float.arg = &opt_float;
|
||||
|
||||
alltypes.opt_fixed64.funcs.encode = &write_fixed64;
|
||||
alltypes.opt_fixed64.arg = &opt_fixed64;
|
||||
|
||||
alltypes.opt_sfixed64.funcs.encode = &write_fixed64;
|
||||
alltypes.opt_sfixed64.arg = &opt_sfixed64;
|
||||
|
||||
alltypes.opt_double.funcs.encode = &write_fixed64;
|
||||
alltypes.opt_double.arg = &opt_double;
|
||||
|
||||
alltypes.opt_string.funcs.encode = &write_string;
|
||||
alltypes.opt_string.arg = "3054";
|
||||
|
||||
alltypes.opt_bytes.funcs.encode = &write_string;
|
||||
alltypes.opt_bytes.arg = "3055";
|
||||
|
||||
alltypes.opt_submsg.funcs.encode = &write_submsg;
|
||||
alltypes.opt_submsg.arg = &opt_submsg;
|
||||
|
||||
alltypes.opt_enum.funcs.encode = &write_varint;
|
||||
alltypes.opt_enum.arg = (void*)MyEnum_Truth;
|
||||
|
||||
alltypes.opt_emptymsg.funcs.encode = &write_emptymsg;
|
||||
}
|
||||
|
||||
alltypes.end.funcs.encode = &write_varint;
|
||||
alltypes.end.arg = (void*)1099;
|
||||
|
||||
{
|
||||
uint8_t buffer[2048];
|
||||
pb_ostream_t stream = pb_ostream_from_buffer(buffer, sizeof(buffer));
|
||||
|
||||
/* Now encode it and check if we succeeded. */
|
||||
if (pb_encode(&stream, AllTypes_fields, &alltypes))
|
||||
{
|
||||
SET_BINARY_MODE(stdout);
|
||||
fwrite(buffer, 1, stream.bytes_written, stdout);
|
||||
return 0; /* Success */
|
||||
}
|
||||
else
|
||||
{
|
||||
fprintf(stderr, "Encoding failed: %s\n", PB_GET_ERROR(&stream));
|
||||
return 1; /* Failure */
|
||||
}
|
||||
}
|
||||
}
|
||||
36
tests/package_name/SConscript
Normal file
36
tests/package_name/SConscript
Normal file
@@ -0,0 +1,36 @@
|
||||
# Check that alltypes test case works also when the .proto file defines
|
||||
# a package name.
|
||||
|
||||
Import("env")
|
||||
|
||||
# Build a modified alltypes.proto
|
||||
def modify_proto(target, source, env):
|
||||
'''Add a "package test.package;" directive to the beginning of the .proto file.'''
|
||||
data = open(str(source[0]), 'r').read()
|
||||
open(str(target[0]), 'w').write("package test.package;\n\n" + data)
|
||||
return 0
|
||||
|
||||
env.Command("alltypes.proto", "#alltypes/alltypes.proto", modify_proto)
|
||||
env.Command("alltypes.options", "#alltypes/alltypes.options", Copy("$TARGET", "$SOURCE"))
|
||||
env.NanopbProto(["alltypes", "alltypes.options"])
|
||||
|
||||
# Build a modified encode_alltypes.c
|
||||
def modify_c(target, source, env):
|
||||
'''Add package name to type names in .c file.'''
|
||||
|
||||
data = open(str(source[0]), 'r').read()
|
||||
|
||||
type_names = ['AllTypes', 'MyEnum', 'HugeEnum']
|
||||
for name in type_names:
|
||||
data = data.replace(name, 'test_package_' + name)
|
||||
|
||||
open(str(target[0]), 'w').write(data)
|
||||
return 0
|
||||
env.Command("encode_alltypes.c", "#alltypes/encode_alltypes.c", modify_c)
|
||||
|
||||
# Encode and compare results to original alltypes testcase
|
||||
enc = env.Program(["encode_alltypes.c", "alltypes.pb.c", "$COMMON/pb_encode.o"])
|
||||
refdec = "$BUILD/alltypes/decode_alltypes$PROGSUFFIX"
|
||||
env.RunTest(enc)
|
||||
env.Compare(["encode_alltypes.output", "$BUILD/alltypes/encode_alltypes.output"])
|
||||
|
||||
Reference in New Issue
Block a user