protobuf-c/t/generated-code2/test-generated-code2.c

1716 lines
56 KiB
C
Raw Normal View History

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include "t/test-full.pb-c.h"
#include "t/generated-code2/test-full-cxx-output.inc"
#define TEST_ENUM_SMALL_TYPE_NAME Foo__TestEnumSmall
#define TEST_ENUM_SMALL(shortname) FOO__TEST_ENUM_SMALL__##shortname
#define TEST_ENUM_TYPE_NAME Foo__TestEnum
#include "common-test-arrays.h"
#define N_ELEMENTS(arr) (sizeof(arr)/sizeof((arr)[0]))
/* ==== helper functions ==== */
static protobuf_c_boolean
are_bytes_equal (unsigned actual_len,
const uint8_t *actual_data,
unsigned expected_len,
const uint8_t *expected_data)
{
if (actual_len != expected_len)
return 0;
return memcmp (actual_data, expected_data, actual_len) == 0;
}
static void
dump_bytes_stderr (size_t len, const uint8_t *data)
{
size_t i;
for (i = 0; i < len; i++)
fprintf(stderr," %02x", data[i]);
fprintf(stderr,"\n");
}
static void
test_versus_static_array(unsigned actual_len,
const uint8_t *actual_data,
unsigned expected_len,
const uint8_t *expected_data,
const char *static_buf_name,
const char *filename,
unsigned lineno)
{
if (are_bytes_equal (actual_len, actual_data, expected_len, expected_data))
return;
fprintf (stderr, "test_versus_static_array failed:\n"
"actual: [length=%u]\n", actual_len);
dump_bytes_stderr (actual_len, actual_data);
fprintf (stderr, "expected: [length=%u] [buffer %s]\n",
expected_len, static_buf_name);
dump_bytes_stderr (expected_len, expected_data);
fprintf (stderr, "at line %u of %s.\n", lineno, filename);
abort();
}
#define TEST_VERSUS_STATIC_ARRAY(actual_len, actual_data, buf) \
test_versus_static_array(actual_len,actual_data, \
sizeof(buf), buf, #buf, __FILE__, __LINE__)
/* rv is unpacked message */
static void *
test_compare_pack_methods (ProtobufCMessage *message,
size_t *packed_len_out,
uint8_t **packed_out)
{
unsigned char scratch[16];
ProtobufCBufferSimple bs = PROTOBUF_C_BUFFER_SIMPLE_INIT (scratch);
size_t siz1 = protobuf_c_message_get_packed_size (message);
size_t siz2;
size_t siz3 = protobuf_c_message_pack_to_buffer (message, &bs.base);
void *packed1 = malloc (siz1);
void *rv;
assert (packed1 != NULL);
assert (siz1 == siz3);
siz2 = protobuf_c_message_pack (message, packed1);
assert (siz1 == siz2);
assert (bs.len == siz1);
assert (memcmp (bs.data, packed1, siz1) == 0);
rv = protobuf_c_message_unpack (message->descriptor, NULL, siz1, packed1);
assert (rv != NULL);
PROTOBUF_C_BUFFER_SIMPLE_CLEAR (&bs);
*packed_len_out = siz1;
*packed_out = packed1;
return rv;
}
#define NUMERIC_EQUALS(a,b) ((a) == (b))
#define STRING_EQUALS(a,b) (strcmp((a),(b))==0)
static protobuf_c_boolean
binary_data_equals (ProtobufCBinaryData a, ProtobufCBinaryData b)
{
if (a.len != b.len)
return 0;
return memcmp (a.data, b.data, a.len) == 0;
}
static protobuf_c_boolean
submesses_equals (Foo__SubMess *a, Foo__SubMess *b)
{
assert(a->base.descriptor == &foo__sub_mess__descriptor);
assert(b->base.descriptor == &foo__sub_mess__descriptor);
return a->test == b->test;
}
/* === the actual tests === */
/* === misc fencepost tests === */
static void test_enum_small (void)
{
#define DO_TEST(UC_VALUE) \
do{ \
Foo__TestMessRequiredEnumSmall small = FOO__TEST_MESS_REQUIRED_ENUM_SMALL__INIT; \
size_t len; \
uint8_t *data; \
Foo__TestMessRequiredEnumSmall *unpacked; \
small.test = FOO__TEST_ENUM_SMALL__##UC_VALUE; \
unpacked = test_compare_pack_methods ((ProtobufCMessage*)&small, &len, &data); \
assert (unpacked->test == FOO__TEST_ENUM_SMALL__##UC_VALUE); \
foo__test_mess_required_enum_small__free_unpacked (unpacked, NULL); \
TEST_VERSUS_STATIC_ARRAY (len, data, test_enum_small_##UC_VALUE); \
free (data); \
}while(0)
assert (sizeof (Foo__TestEnumSmall) == 4);
DO_TEST(VALUE);
DO_TEST(OTHER_VALUE);
#undef DO_TEST
}
static void test_enum_big (void)
{
Foo__TestMessRequiredEnum big = FOO__TEST_MESS_REQUIRED_ENUM__INIT;
size_t len;
uint8_t *data;
Foo__TestMessRequiredEnum *unpacked;
assert (sizeof (Foo__TestEnum) == 4);
#define DO_ONE_TEST(shortname, numeric_value, encoded_len) \
do{ \
big.test = FOO__TEST_ENUM__##shortname; \
unpacked = test_compare_pack_methods ((ProtobufCMessage*)&big, &len, &data); \
assert (unpacked->test == FOO__TEST_ENUM__##shortname); \
foo__test_mess_required_enum__free_unpacked (unpacked, NULL); \
TEST_VERSUS_STATIC_ARRAY (len, data, test_enum_big_##shortname); \
assert (encoded_len + 1 == len); \
assert (big.test == numeric_value); \
free (data); \
}while(0)
DO_ONE_TEST(VALUE0, 0, 1);
DO_ONE_TEST(VALUE127, 127, 1);
DO_ONE_TEST(VALUE128, 128, 2);
DO_ONE_TEST(VALUE16383, 16383, 2);
DO_ONE_TEST(VALUE16384, 16384, 3);
DO_ONE_TEST(VALUE2097151, 2097151, 3);
DO_ONE_TEST(VALUE2097152, 2097152, 4);
DO_ONE_TEST(VALUE268435455, 268435455, 4);
DO_ONE_TEST(VALUE268435456, 268435456, 5);
#undef DO_ONE_TEST
}
static void test_field_numbers (void)
{
size_t len;
uint8_t *data;
#define DO_ONE_TEST(num, exp_len) \
{ \
Foo__TestFieldNo##num t = FOO__TEST_FIELD_NO##num##__INIT; \
Foo__TestFieldNo##num *t2; \
t.test = "tst"; \
t2 = test_compare_pack_methods ((ProtobufCMessage*)(&t), &len, &data); \
assert (strcmp (t2->test, "tst") == 0); \
TEST_VERSUS_STATIC_ARRAY (len, data, test_field_number_##num); \
assert (len == exp_len); \
free (data); \
foo__test_field_no##num##__free_unpacked (t2, NULL); \
}
DO_ONE_TEST (15, 1 + 1 + 3);
DO_ONE_TEST (16, 2 + 1 + 3);
DO_ONE_TEST (2047, 2 + 1 + 3);
DO_ONE_TEST (2048, 3 + 1 + 3);
DO_ONE_TEST (262143, 3 + 1 + 3);
DO_ONE_TEST (262144, 4 + 1 + 3);
DO_ONE_TEST (33554431, 4 + 1 + 3);
DO_ONE_TEST (33554432, 5 + 1 + 3);
#undef DO_ONE_TEST
}
/* === Required type fields === */
#define DO_TEST_REQUIRED(Type, TYPE, type, value, example_packed_data, equal_func) \
do{ \
Foo__TestMessRequired##Type opt = FOO__TEST_MESS_REQUIRED_##TYPE##__INIT; \
Foo__TestMessRequired##Type *mess; \
size_t len; uint8_t *data; \
opt.test = value; \
mess = test_compare_pack_methods (&opt.base, &len, &data); \
TEST_VERSUS_STATIC_ARRAY (len, data, example_packed_data); \
assert (equal_func (mess->test, value)); \
foo__test_mess_required_##type##__free_unpacked (mess, NULL); \
free (data); \
}while(0)
static void test_required_int32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(Int32, INT32, int32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(INT32_MIN, test_required_int32_min);
DO_TEST(-1000, test_required_int32_m1000);
DO_TEST(0, test_required_int32_0);
DO_TEST(INT32_MAX, test_required_int32_max);
#undef DO_TEST
}
static void test_required_sint32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(SInt32, SINT32, sint32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(INT32_MIN, test_required_sint32_min);
DO_TEST(-1000, test_required_sint32_m1000);
DO_TEST(0, test_required_sint32_0);
DO_TEST(INT32_MAX, test_required_sint32_max);
#undef DO_TEST
}
static void test_required_sfixed32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(SFixed32, SFIXED32, sfixed32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(INT32_MIN, test_required_sfixed32_min);
DO_TEST(-1000, test_required_sfixed32_m1000);
DO_TEST(0, test_required_sfixed32_0);
DO_TEST(INT32_MAX, test_required_sfixed32_max);
#undef DO_TEST
}
static void test_required_uint32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(UInt32, UINT32, uint32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(0, test_required_uint32_0);
DO_TEST(MILLION, test_required_uint32_million);
DO_TEST(UINT32_MAX, test_required_uint32_max);
#undef DO_TEST
}
static void test_required_fixed32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(Fixed32, FIXED32, fixed32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(0, test_required_fixed32_0);
DO_TEST(MILLION, test_required_fixed32_million);
DO_TEST(UINT32_MAX, test_required_fixed32_max);
#undef DO_TEST
}
static void test_required_int64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(Int64, INT64, int64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(INT64_MIN, test_required_int64_min);
DO_TEST(-TRILLION, test_required_int64_mtril);
DO_TEST(0, test_required_int64_0);
DO_TEST(QUADRILLION, test_required_int64_quad);
DO_TEST(INT64_MAX, test_required_int64_max);
#undef DO_TEST
}
static void test_required_sint64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(SInt64, SINT64, sint64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(INT64_MIN, test_required_sint64_min);
DO_TEST(-TRILLION, test_required_sint64_mtril);
DO_TEST(0, test_required_sint64_0);
DO_TEST(QUADRILLION, test_required_sint64_quad);
DO_TEST(INT64_MAX, test_required_sint64_max);
#undef DO_TEST
}
static void test_required_sfixed64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(SFixed64, SFIXED64, sfixed64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(INT64_MIN, test_required_sfixed64_min);
DO_TEST(-TRILLION, test_required_sfixed64_mtril);
DO_TEST(0, test_required_sfixed64_0);
DO_TEST(QUADRILLION, test_required_sfixed64_quad);
DO_TEST(INT64_MAX, test_required_sfixed64_max);
#undef DO_TEST
}
static void test_required_uint64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(UInt64, UINT64, uint64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(0, test_required_uint64_0);
DO_TEST(THOUSAND, test_required_uint64_thou);
DO_TEST(MILLION, test_required_uint64_mill);
DO_TEST(BILLION, test_required_uint64_bill);
DO_TEST(TRILLION, test_required_uint64_tril);
DO_TEST(QUADRILLION, test_required_uint64_quad);
DO_TEST(QUINTILLION, test_required_uint64_quint);
DO_TEST(UINT64_MAX, test_required_uint64_max);
#undef DO_TEST
}
static void test_required_fixed64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(Fixed64, FIXED64, fixed64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(0, test_required_fixed64_0);
DO_TEST(THOUSAND, test_required_fixed64_thou);
DO_TEST(MILLION, test_required_fixed64_mill);
DO_TEST(BILLION, test_required_fixed64_bill);
DO_TEST(TRILLION, test_required_fixed64_tril);
DO_TEST(QUADRILLION, test_required_fixed64_quad);
DO_TEST(QUINTILLION, test_required_fixed64_quint);
DO_TEST(UINT64_MAX, test_required_fixed64_max);
#undef DO_TEST
}
static void test_required_float (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(Float, FLOAT, float, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(-THOUSAND, test_required_float_mthou);
DO_TEST(0, test_required_float_0);
DO_TEST(420, test_required_float_420);
#undef DO_TEST
}
static void test_required_double (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(Double, DOUBLE, double, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(-THOUSAND, test_required_double_mthou);
DO_TEST(0, test_required_double_0);
DO_TEST(420, test_required_double_420);
#undef DO_TEST
}
static void test_required_bool (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(Bool, BOOL, bool, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(0, test_required_bool_0);
DO_TEST(1, test_required_bool_1);
#undef DO_TEST
}
static void test_required_TestEnumSmall (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(EnumSmall, ENUM_SMALL, enum_small, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST(FOO__TEST_ENUM_SMALL__VALUE, test_required_enum_small_VALUE);
DO_TEST(FOO__TEST_ENUM_SMALL__OTHER_VALUE, test_required_enum_small_OTHER_VALUE);
#undef DO_TEST
}
static void test_required_TestEnum (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(Enum, ENUM, enum, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (FOO__TEST_ENUM__VALUE0, test_required_enum_0);
DO_TEST (FOO__TEST_ENUM__VALUE1, test_required_enum_1);
DO_TEST (FOO__TEST_ENUM__VALUE127, test_required_enum_127);
DO_TEST (FOO__TEST_ENUM__VALUE128, test_required_enum_128);
DO_TEST (FOO__TEST_ENUM__VALUE16383, test_required_enum_16383);
DO_TEST (FOO__TEST_ENUM__VALUE16384, test_required_enum_16384);
DO_TEST (FOO__TEST_ENUM__VALUE2097151, test_required_enum_2097151);
DO_TEST (FOO__TEST_ENUM__VALUE2097152, test_required_enum_2097152);
DO_TEST (FOO__TEST_ENUM__VALUE268435455, test_required_enum_268435455);
DO_TEST (FOO__TEST_ENUM__VALUE268435456, test_required_enum_268435456);
#undef DO_TEST
}
static void test_required_string (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED(String, STRING, string, value, example_packed_data, STRING_EQUALS)
DO_TEST("", test_required_string_empty);
DO_TEST("hello", test_required_string_hello);
DO_TEST("two hundred xs follow: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx", test_required_string_long);
#undef DO_TEST
}
static void test_required_bytes (void)
{
static ProtobufCBinaryData bd_empty = { 0, (uint8_t*)"" };
static ProtobufCBinaryData bd_hello = { 5, (uint8_t*)"hello" };
static ProtobufCBinaryData bd_random = { 5, (uint8_t*)"\1\0\375\2\4" };
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED (Bytes, BYTES, bytes, value, example_packed_data, binary_data_equals)
DO_TEST (bd_empty, test_required_bytes_empty);
DO_TEST (bd_hello, test_required_bytes_hello);
DO_TEST (bd_random, test_required_bytes_random);
#undef DO_TEST
}
static void test_required_SubMess (void)
{
Foo__SubMess submess = FOO__SUB_MESS__INIT;
#define DO_TEST(value, example_packed_data) \
DO_TEST_REQUIRED (Message, MESSAGE, message, value, example_packed_data, submesses_equals)
submess.test = 0;
DO_TEST (&submess, test_required_submess_0);
submess.test = 42;
DO_TEST (&submess, test_required_submess_42);
#undef DO_TEST
}
/* === Optional type fields === */
static void test_empty_optional (void)
{
Foo__TestMessOptional mess = FOO__TEST_MESS_OPTIONAL__INIT;
size_t len;
uint8_t *data;
Foo__TestMessOptional *mess2 = test_compare_pack_methods (&mess.base, &len, &data);
assert (len == 0);
free (data);
foo__test_mess_optional__free_unpacked (mess2, NULL);
}
#define DO_TEST_OPTIONAL(base_member, value, example_packed_data, equal_func) \
do{ \
Foo__TestMessOptional opt = FOO__TEST_MESS_OPTIONAL__INIT; \
Foo__TestMessOptional *mess; \
size_t len; uint8_t *data; \
opt.has_##base_member = 1; \
opt.base_member = value; \
mess = test_compare_pack_methods (&opt.base, &len, &data); \
TEST_VERSUS_STATIC_ARRAY (len, data, example_packed_data); \
assert (mess->has_##base_member); \
assert (equal_func (mess->base_member, value)); \
foo__test_mess_optional__free_unpacked (mess, NULL); \
free (data); \
}while(0)
static void test_optional_int32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_int32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (INT32_MIN, test_optional_int32_min);
DO_TEST (-1, test_optional_int32_m1);
DO_TEST (0, test_optional_int32_0);
DO_TEST (666, test_optional_int32_666);
DO_TEST (INT32_MAX, test_optional_int32_max);
#undef DO_TEST
}
static void test_optional_sint32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_sint32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (INT32_MIN, test_optional_sint32_min);
DO_TEST (-1, test_optional_sint32_m1);
DO_TEST (0, test_optional_sint32_0);
DO_TEST (666, test_optional_sint32_666);
DO_TEST (INT32_MAX, test_optional_sint32_max);
#undef DO_TEST
}
static void test_optional_sfixed32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_sfixed32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (INT32_MIN, test_optional_sfixed32_min);
DO_TEST (-1, test_optional_sfixed32_m1);
DO_TEST (0, test_optional_sfixed32_0);
DO_TEST (666, test_optional_sfixed32_666);
DO_TEST (INT32_MAX, test_optional_sfixed32_max);
#undef DO_TEST
}
static void test_optional_int64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_int64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (INT64_MIN, test_optional_int64_min);
DO_TEST (-1111111111LL, test_optional_int64_m1111111111LL);
DO_TEST (0, test_optional_int64_0);
DO_TEST (QUINTILLION, test_optional_int64_quintillion);
DO_TEST (INT64_MAX, test_optional_int64_max);
#undef DO_TEST
}
static void test_optional_sint64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_sint64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (INT64_MIN, test_optional_sint64_min);
DO_TEST (-1111111111LL, test_optional_sint64_m1111111111LL);
DO_TEST (0, test_optional_sint64_0);
DO_TEST (QUINTILLION, test_optional_sint64_quintillion);
DO_TEST (INT64_MAX, test_optional_sint64_max);
#undef DO_TEST
}
static void test_optional_sfixed64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_sfixed64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (INT64_MIN, test_optional_sfixed64_min);
DO_TEST (-1111111111LL, test_optional_sfixed64_m1111111111LL);
DO_TEST (0, test_optional_sfixed64_0);
DO_TEST (QUINTILLION, test_optional_sfixed64_quintillion);
DO_TEST (INT64_MAX, test_optional_sfixed64_max);
#undef DO_TEST
}
static void test_optional_uint32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_uint32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (0, test_optional_uint32_0);
DO_TEST (669, test_optional_uint32_669);
DO_TEST (UINT32_MAX, test_optional_uint32_max);
#undef DO_TEST
}
static void test_optional_fixed32 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_fixed32, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (0, test_optional_fixed32_0);
DO_TEST (669, test_optional_fixed32_669);
DO_TEST (UINT32_MAX, test_optional_fixed32_max);
#undef DO_TEST
}
static void test_optional_uint64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_uint64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (0, test_optional_uint64_0);
DO_TEST (669669669669669ULL, test_optional_uint64_669669669669669);
DO_TEST (UINT64_MAX, test_optional_uint64_max);
#undef DO_TEST
}
static void test_optional_fixed64 (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_fixed64, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (0, test_optional_fixed64_0);
DO_TEST (669669669669669ULL, test_optional_fixed64_669669669669669);
DO_TEST (UINT64_MAX, test_optional_fixed64_max);
#undef DO_TEST
}
static void test_optional_float (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_float, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (-100, test_optional_float_m100);
DO_TEST (0, test_optional_float_0);
DO_TEST (141243, test_optional_float_141243);
#undef DO_TEST
}
static void test_optional_double (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_double, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (-100, test_optional_double_m100);
DO_TEST (0, test_optional_double_0);
DO_TEST (141243, test_optional_double_141243);
#undef DO_TEST
}
static void test_optional_bool (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_boolean, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (0, test_optional_bool_0);
DO_TEST (1, test_optional_bool_1);
#undef DO_TEST
}
static void test_optional_TestEnumSmall (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_enum_small, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (0, test_optional_enum_small_0);
DO_TEST (1, test_optional_enum_small_1);
#undef DO_TEST
}
static void test_optional_TestEnum (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL(test_enum, value, example_packed_data, NUMERIC_EQUALS)
DO_TEST (FOO__TEST_ENUM__VALUE0, test_optional_enum_0);
DO_TEST (FOO__TEST_ENUM__VALUE1, test_optional_enum_1);
DO_TEST (FOO__TEST_ENUM__VALUE127, test_optional_enum_127);
DO_TEST (FOO__TEST_ENUM__VALUE128, test_optional_enum_128);
DO_TEST (FOO__TEST_ENUM__VALUE16383, test_optional_enum_16383);
DO_TEST (FOO__TEST_ENUM__VALUE16384, test_optional_enum_16384);
DO_TEST (FOO__TEST_ENUM__VALUE2097151, test_optional_enum_2097151);
DO_TEST (FOO__TEST_ENUM__VALUE2097152, test_optional_enum_2097152);
DO_TEST (FOO__TEST_ENUM__VALUE268435455, test_optional_enum_268435455);
DO_TEST (FOO__TEST_ENUM__VALUE268435456, test_optional_enum_268435456);
#undef DO_TEST
}
#define DO_TEST_OPTIONAL__NO_HAS(base_member, value, example_packed_data, equal_func) \
do{ \
Foo__TestMessOptional opt = FOO__TEST_MESS_OPTIONAL__INIT; \
Foo__TestMessOptional *mess; \
size_t len; uint8_t *data; \
opt.base_member = value; \
mess = test_compare_pack_methods (&opt.base, &len, &data); \
TEST_VERSUS_STATIC_ARRAY (len, data, example_packed_data); \
assert (mess->base_member != NULL); \
assert (equal_func (mess->base_member, value)); \
foo__test_mess_optional__free_unpacked (mess, NULL); \
free (data); \
}while(0)
static void test_optional_string (void)
{
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL__NO_HAS (test_string, value, example_packed_data, STRING_EQUALS)
DO_TEST ("", test_optional_string_empty);
DO_TEST ("hello", test_optional_string_hello);
#undef DO_TEST
}
static void test_optional_bytes (void)
{
static ProtobufCBinaryData bd_empty = { 0, (uint8_t*)"" };
static ProtobufCBinaryData bd_hello = { 5, (uint8_t*)"hello" };
static ProtobufCBinaryData bd_random = { 5, (uint8_t*)"\1\0\375\2\4" };
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL (test_bytes, value, example_packed_data, binary_data_equals)
DO_TEST (bd_empty, test_optional_bytes_empty);
DO_TEST (bd_hello, test_optional_bytes_hello);
DO_TEST (bd_random, test_optional_bytes_random);
#undef DO_TEST
}
static void test_optional_SubMess (void)
{
Foo__SubMess submess = FOO__SUB_MESS__INIT;
#define DO_TEST(value, example_packed_data) \
DO_TEST_OPTIONAL__NO_HAS (test_message, value, example_packed_data, submesses_equals)
submess.test = 0;
DO_TEST (&submess, test_optional_submess_0);
submess.test = 42;
DO_TEST (&submess, test_optional_submess_42);
#undef DO_TEST
}
/* === repeated type fields === */
#define DO_TEST_REPEATED(lc_member_name, cast, \
static_array, example_packed_data, \
equals_macro) \
do{ \
Foo__TestMess mess = FOO__TEST_MESS__INIT; \
Foo__TestMess *mess2; \
size_t len; \
uint8_t *data; \
unsigned i; \
mess.n_##lc_member_name = N_ELEMENTS (static_array); \
mess.lc_member_name = cast static_array; \
mess2 = test_compare_pack_methods ((ProtobufCMessage*)(&mess), &len, &data); \
assert(mess2->n_##lc_member_name == N_ELEMENTS (static_array)); \
for (i = 0; i < N_ELEMENTS (static_array); i++) \
assert(equals_macro(mess2->lc_member_name[i], static_array[i])); \
TEST_VERSUS_STATIC_ARRAY (len, data, example_packed_data); \
free (data); \
foo__test_mess__free_unpacked (mess2, NULL); \
}while(0)
static void test_empty_repeated (void)
{
Foo__TestMess mess = FOO__TEST_MESS__INIT;
size_t len;
uint8_t *data;
Foo__TestMess *mess2 = test_compare_pack_methods (&mess.base, &len, &data);
assert (len == 0);
free (data);
foo__test_mess__free_unpacked (mess2, NULL);
}
static void test_repeated_int32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_int32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int32_arr0, test_repeated_int32_arr0);
DO_TEST (int32_arr1, test_repeated_int32_arr1);
DO_TEST (int32_arr_min_max, test_repeated_int32_arr_min_max);
#undef DO_TEST
}
static void test_repeated_sint32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_sint32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int32_arr0, test_repeated_sint32_arr0);
DO_TEST (int32_arr1, test_repeated_sint32_arr1);
DO_TEST (int32_arr_min_max, test_repeated_sint32_arr_min_max);
#undef DO_TEST
}
static void test_repeated_sfixed32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_sfixed32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int32_arr0, test_repeated_sfixed32_arr0);
DO_TEST (int32_arr1, test_repeated_sfixed32_arr1);
DO_TEST (int32_arr_min_max, test_repeated_sfixed32_arr_min_max);
#undef DO_TEST
}
static void test_repeated_uint32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_uint32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (uint32_roundnumbers, test_repeated_uint32_roundnumbers);
DO_TEST (uint32_0_max, test_repeated_uint32_0_max);
#undef DO_TEST
}
static void test_repeated_fixed32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_fixed32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (uint32_roundnumbers, test_repeated_fixed32_roundnumbers);
DO_TEST (uint32_0_max, test_repeated_fixed32_0_max);
#undef DO_TEST
}
static void test_repeated_int64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_int64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int64_roundnumbers, test_repeated_int64_roundnumbers);
DO_TEST (int64_min_max, test_repeated_int64_min_max);
#undef DO_TEST
}
static void test_repeated_sint64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_sint64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int64_roundnumbers, test_repeated_sint64_roundnumbers);
DO_TEST (int64_min_max, test_repeated_sint64_min_max);
#undef DO_TEST
}
static void test_repeated_sfixed64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_sfixed64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int64_roundnumbers, test_repeated_sfixed64_roundnumbers);
DO_TEST (int64_min_max, test_repeated_sfixed64_min_max);
#undef DO_TEST
}
static void test_repeated_uint64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_uint64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(uint64_roundnumbers, test_repeated_uint64_roundnumbers);
DO_TEST(uint64_0_1_max, test_repeated_uint64_0_1_max);
DO_TEST(uint64_random, test_repeated_uint64_random);
#undef DO_TEST
}
static void test_repeated_fixed64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_fixed64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(uint64_roundnumbers, test_repeated_fixed64_roundnumbers);
DO_TEST(uint64_0_1_max, test_repeated_fixed64_0_1_max);
DO_TEST(uint64_random, test_repeated_fixed64_random);
#undef DO_TEST
}
static void test_repeated_float (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_float, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(float_random, test_repeated_float_random);
#undef DO_TEST
}
static void test_repeated_double (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_double, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(double_random, test_repeated_double_random);
#undef DO_TEST
}
static void test_repeated_boolean (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_boolean, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(boolean_0, test_repeated_boolean_0);
DO_TEST(boolean_1, test_repeated_boolean_1);
DO_TEST(boolean_random, test_repeated_boolean_random);
#undef DO_TEST
}
static void test_repeated_TestEnumSmall (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_enum_small, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(enum_small_0, test_repeated_enum_small_0);
DO_TEST(enum_small_1, test_repeated_enum_small_1);
DO_TEST(enum_small_random, test_repeated_enum_small_random);
#undef DO_TEST
}
static void test_repeated_TestEnum (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_enum, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(enum_0, test_repeated_enum_0);
DO_TEST(enum_1, test_repeated_enum_1);
DO_TEST(enum_random, test_repeated_enum_random);
#undef DO_TEST
}
static void test_repeated_string (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_string, (char **), \
static_array, example_packed_data, \
STRING_EQUALS)
DO_TEST(repeated_strings_0, test_repeated_strings_0);
DO_TEST(repeated_strings_1, test_repeated_strings_1);
DO_TEST(repeated_strings_2, test_repeated_strings_2);
DO_TEST(repeated_strings_3, test_repeated_strings_3);
#undef DO_TEST
}
static void test_repeated_bytes (void)
{
static ProtobufCBinaryData test_binary_data_0[] = {
{ 4, (uint8_t *) "text" },
{ 9, (uint8_t *) "str\1\2\3\4\5\0" },
{ 10, (uint8_t *) "gobble\0foo" }
};
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_bytes, , \
static_array, example_packed_data, \
binary_data_equals)
DO_TEST (test_binary_data_0, test_repeated_bytes_0);
#undef DO_TEST
}
static void test_repeated_SubMess (void)
{
static Foo__SubMess submess0 = FOO__SUB_MESS__INIT;
static Foo__SubMess submess1 = FOO__SUB_MESS__INIT;
static Foo__SubMess submess2 = FOO__SUB_MESS__INIT;
static Foo__SubMess *submesses[3] = { &submess0, &submess1, &submess2 };
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_REPEATED(test_message, , \
static_array, example_packed_data, \
submesses_equals)
DO_TEST (submesses, test_repeated_submess_0);
submess0.test = 42;
submess1.test = -10000;
submess2.test = 667;
DO_TEST (submesses, test_repeated_submess_1);
#undef DO_TEST
}
#define DO_TEST_PACKED_REPEATED(lc_member_name, cast, \
static_array, example_packed_data, \
equals_macro) \
do{ \
Foo__TestMessPacked mess = FOO__TEST_MESS_PACKED__INIT; \
Foo__TestMessPacked *mess2; \
size_t len; \
uint8_t *data; \
unsigned i; \
mess.n_##lc_member_name = N_ELEMENTS (static_array); \
mess.lc_member_name = cast static_array; \
mess2 = test_compare_pack_methods ((ProtobufCMessage*)(&mess), &len, &data); \
TEST_VERSUS_STATIC_ARRAY (len, data, example_packed_data); \
assert(mess2->n_##lc_member_name == N_ELEMENTS (static_array)); \
for (i = 0; i < N_ELEMENTS (static_array); i++) \
assert(equals_macro(mess2->lc_member_name[i], static_array[i])); \
free (data); \
foo__test_mess_packed__free_unpacked (mess2, NULL); \
}while(0)
static void test_packed_repeated_int32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_int32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int32_arr0, test_packed_repeated_int32_arr0);
DO_TEST (int32_arr1, test_packed_repeated_int32_arr1);
DO_TEST (int32_arr_min_max, test_packed_repeated_int32_arr_min_max);
#undef DO_TEST
}
static void test_packed_repeated_sint32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_sint32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int32_arr0, test_packed_repeated_sint32_arr0);
DO_TEST (int32_arr1, test_packed_repeated_sint32_arr1);
DO_TEST (int32_arr_min_max, test_packed_repeated_sint32_arr_min_max);
#undef DO_TEST
}
static void test_packed_repeated_sfixed32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_sfixed32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int32_arr0, test_packed_repeated_sfixed32_arr0);
DO_TEST (int32_arr1, test_packed_repeated_sfixed32_arr1);
DO_TEST (int32_arr_min_max, test_packed_repeated_sfixed32_arr_min_max);
#undef DO_TEST
}
static void test_packed_repeated_uint32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_uint32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (uint32_roundnumbers, test_packed_repeated_uint32_roundnumbers);
DO_TEST (uint32_0_max, test_packed_repeated_uint32_0_max);
#undef DO_TEST
}
static void test_packed_repeated_fixed32 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_fixed32, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (uint32_roundnumbers, test_packed_repeated_fixed32_roundnumbers);
DO_TEST (uint32_0_max, test_packed_repeated_fixed32_0_max);
#undef DO_TEST
}
static void test_packed_repeated_int64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_int64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int64_roundnumbers, test_packed_repeated_int64_roundnumbers);
DO_TEST (int64_min_max, test_packed_repeated_int64_min_max);
#undef DO_TEST
}
static void test_packed_repeated_sint64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_sint64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int64_roundnumbers, test_packed_repeated_sint64_roundnumbers);
DO_TEST (int64_min_max, test_packed_repeated_sint64_min_max);
#undef DO_TEST
}
static void test_packed_repeated_sfixed64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_sfixed64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST (int64_roundnumbers, test_packed_repeated_sfixed64_roundnumbers);
DO_TEST (int64_min_max, test_packed_repeated_sfixed64_min_max);
#undef DO_TEST
}
static void test_packed_repeated_uint64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_uint64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(uint64_roundnumbers, test_packed_repeated_uint64_roundnumbers);
DO_TEST(uint64_0_1_max, test_packed_repeated_uint64_0_1_max);
DO_TEST(uint64_random, test_packed_repeated_uint64_random);
#undef DO_TEST
}
static void test_packed_repeated_fixed64 (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_fixed64, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(uint64_roundnumbers, test_packed_repeated_fixed64_roundnumbers);
DO_TEST(uint64_0_1_max, test_packed_repeated_fixed64_0_1_max);
DO_TEST(uint64_random, test_packed_repeated_fixed64_random);
#undef DO_TEST
}
static void test_packed_repeated_float (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_float, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(float_random, test_packed_repeated_float_random);
#undef DO_TEST
}
static void test_packed_repeated_double (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_double, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(double_random, test_packed_repeated_double_random);
#undef DO_TEST
}
static void test_packed_repeated_boolean (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_boolean, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(boolean_0, test_packed_repeated_boolean_0);
DO_TEST(boolean_1, test_packed_repeated_boolean_1);
DO_TEST(boolean_random, test_packed_repeated_boolean_random);
#undef DO_TEST
}
static void test_packed_repeated_TestEnumSmall (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_enum_small, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(enum_small_0, test_packed_repeated_enum_small_0);
DO_TEST(enum_small_1, test_packed_repeated_enum_small_1);
DO_TEST(enum_small_random, test_packed_repeated_enum_small_random);
#undef DO_TEST
}
static void test_packed_repeated_TestEnum (void)
{
#define DO_TEST(static_array, example_packed_data) \
DO_TEST_PACKED_REPEATED(test_enum, , \
static_array, example_packed_data, \
NUMERIC_EQUALS)
DO_TEST(enum_0, test_packed_repeated_enum_0);
DO_TEST(enum_1, test_packed_repeated_enum_1);
DO_TEST(enum_random, test_packed_repeated_enum_random);
#undef DO_TEST
}
static void test_unknown_fields (void)
{
static Foo__EmptyMess mess = FOO__EMPTY_MESS__INIT;
static Foo__EmptyMess *mess2;
ProtobufCMessageUnknownField fields[2];
size_t len; uint8_t *data;
mess.base.n_unknown_fields = 2;
mess.base.unknown_fields = fields;
fields[0].tag = 5454;
fields[0].wire_type = PROTOBUF_C_WIRE_TYPE_VARINT;
fields[0].len = 2;
fields[0].data = (uint8_t*)"\377\1";
fields[1].tag = 5555;
fields[1].wire_type = PROTOBUF_C_WIRE_TYPE_32BIT;
fields[1].len = 4;
fields[1].data = (uint8_t*)"\4\1\0\0";
mess2 = test_compare_pack_methods (&mess.base, &len, &data);
assert (mess2->base.n_unknown_fields == 2);
assert (mess2->base.unknown_fields[0].tag == 5454);
assert (mess2->base.unknown_fields[0].wire_type == PROTOBUF_C_WIRE_TYPE_VARINT);
assert (mess2->base.unknown_fields[0].len == 2);
assert (memcmp (mess2->base.unknown_fields[0].data, fields[0].data, 2) == 0);
assert (mess2->base.unknown_fields[1].tag == 5555);
assert (mess2->base.unknown_fields[1].wire_type == PROTOBUF_C_WIRE_TYPE_32BIT);
assert (mess2->base.unknown_fields[1].len == 4);
assert (memcmp (mess2->base.unknown_fields[1].data, fields[1].data, 4) == 0);
TEST_VERSUS_STATIC_ARRAY (len, data, test_unknown_fields_0);
free (data);
foo__empty_mess__free_unpacked (mess2, NULL);
fields[0].tag = 6666;
fields[0].wire_type = PROTOBUF_C_WIRE_TYPE_LENGTH_PREFIXED;
fields[0].len = 9;
fields[0].data = (uint8_t*)"\10xxxxxxxx";
fields[1].tag = 7777;
fields[1].wire_type = PROTOBUF_C_WIRE_TYPE_64BIT;
fields[1].len = 8;
fields[1].data = (uint8_t*)"\1\1\1\0\0\0\0\0";
mess2 = test_compare_pack_methods (&mess.base, &len, &data);
assert (mess2->base.n_unknown_fields == 2);
assert (mess2->base.unknown_fields[0].tag == 6666);
assert (mess2->base.unknown_fields[0].wire_type == PROTOBUF_C_WIRE_TYPE_LENGTH_PREFIXED);
assert (mess2->base.unknown_fields[0].len == 9);
assert (memcmp (mess2->base.unknown_fields[0].data, fields[0].data, 9) == 0);
assert (mess2->base.unknown_fields[1].tag == 7777);
assert (mess2->base.unknown_fields[1].wire_type == PROTOBUF_C_WIRE_TYPE_64BIT);
assert (mess2->base.unknown_fields[1].len == 8);
assert (memcmp (mess2->base.unknown_fields[1].data, fields[1].data, 8) == 0);
TEST_VERSUS_STATIC_ARRAY (len, data, test_unknown_fields_1);
free (data);
foo__empty_mess__free_unpacked (mess2, NULL);
}
static void
test_enum_descriptor (const ProtobufCEnumDescriptor *desc)
{
unsigned i;
for (i = 0; i < desc->n_values; i++)
{
const ProtobufCEnumValue *sv = desc->values + i;
const ProtobufCEnumValue *vv;
const ProtobufCEnumValue *vn;
vv = protobuf_c_enum_descriptor_get_value (desc, sv->value);
vn = protobuf_c_enum_descriptor_get_value_by_name (desc, sv->name);
assert (sv == vv);
assert (sv == vn);
}
for (i = 0; i < desc->n_value_names; i++)
{
const char *name = desc->values_by_name[i].name;
const ProtobufCEnumValue *v;
v = protobuf_c_enum_descriptor_get_value_by_name (desc, name);
assert (v != NULL);
}
}
static void
test_enum_by_name (const ProtobufCEnumDescriptor *desc,
const char *name,
int expected_value)
{
const ProtobufCEnumValue *v;
v = protobuf_c_enum_descriptor_get_value_by_name (desc, name);
assert (v != NULL);
assert (v->value == expected_value);
}
static void
test_enum_lookups (void)
{
test_enum_descriptor (&foo__test_enum__descriptor);
test_enum_descriptor (&foo__test_enum_small__descriptor);
test_enum_descriptor (&foo__test_enum_dup_values__descriptor);
#define TEST_ENUM_DUP_VALUES(str, shortname) \
test_enum_by_name (&foo__test_enum_dup_values__descriptor, \
str, FOO__TEST_ENUM_DUP_VALUES__##shortname)
TEST_ENUM_DUP_VALUES ("VALUE_A", VALUE_A);
TEST_ENUM_DUP_VALUES ("VALUE_B", VALUE_B);
TEST_ENUM_DUP_VALUES ("VALUE_C", VALUE_C);
TEST_ENUM_DUP_VALUES ("VALUE_D", VALUE_D);
TEST_ENUM_DUP_VALUES ("VALUE_E", VALUE_E);
TEST_ENUM_DUP_VALUES ("VALUE_F", VALUE_F);
TEST_ENUM_DUP_VALUES ("VALUE_AA", VALUE_AA);
TEST_ENUM_DUP_VALUES ("VALUE_BB", VALUE_BB);
#undef TEST_ENUM_DUP_VALUES
}
static void
test_message_descriptor (const ProtobufCMessageDescriptor *desc)
{
unsigned i;
for (i = 0; i < desc->n_fields; i++)
{
const ProtobufCFieldDescriptor *f = desc->fields + i;
const ProtobufCFieldDescriptor *fv;
const ProtobufCFieldDescriptor *fn;
fv = protobuf_c_message_descriptor_get_field (desc, f->id);
fn = protobuf_c_message_descriptor_get_field_by_name (desc, f->name);
assert (f == fv);
assert (f == fn);
}
}
static void
test_message_lookups (void)
{
test_message_descriptor (&foo__test_mess__descriptor);
test_message_descriptor (&foo__test_mess_optional__descriptor);
test_message_descriptor (&foo__test_mess_required_enum__descriptor);
}
static void
assert_required_default_values_are_default (Foo__DefaultRequiredValues *mess)
{
assert (mess->v_int32 == -42);
assert (mess->v_uint32 == 666);
assert (mess->v_int64 == 100000);
assert (mess->v_uint64 == 100001);
assert (mess->v_float == 2.5);
assert (mess->v_double == 4.5);
assert (strcmp (mess->v_string, "hi mom\n") == 0);
assert (mess->v_bytes.len = /* a */ 1
+ /* space */ 1
+ /* NUL */ 1
+ /* space */ 1
+ /* "character" */ 9);
assert (memcmp (mess->v_bytes.data, "a \0 character", 13) == 0);
}
static void
test_required_default_values (void)
{
Foo__DefaultRequiredValues mess = FOO__DEFAULT_REQUIRED_VALUES__INIT;
Foo__DefaultRequiredValues *mess2;
size_t len; uint8_t *data;
assert_required_default_values_are_default (&mess);
mess2 = test_compare_pack_methods (&mess.base, &len, &data);
free (data);
assert_required_default_values_are_default (mess2);
foo__default_required_values__free_unpacked (mess2, NULL);
}
static void
assert_optional_default_values_are_default (Foo__DefaultOptionalValues *mess)
{
assert (!mess->has_v_int32);
assert (mess->v_int32 == -42);
assert (!mess->has_v_uint32);
assert (mess->v_uint32 == 666);
assert (!mess->has_v_int64);
assert (mess->v_int64 == 100000);
assert (!mess->has_v_uint64);
assert (mess->v_uint64 == 100001);
assert (!mess->has_v_float);
assert (mess->v_float == 2.5);
assert (!mess->has_v_double);
assert (mess->v_double == 4.5);
assert (strcmp (mess->v_string, "hi mom\n") == 0);
assert (!mess->has_v_bytes);
assert (mess->v_bytes.len = /* a */ 1
+ /* space */ 1
+ /* NUL */ 1
+ /* space */ 1
+ /* "character" */ 9);
assert (memcmp (mess->v_bytes.data, "a \0 character", 13) == 0);
}
static void
test_optional_default_values (void)
{
Foo__DefaultOptionalValues mess = FOO__DEFAULT_OPTIONAL_VALUES__INIT;
Foo__DefaultOptionalValues *mess2;
size_t len; uint8_t *data;
assert_optional_default_values_are_default (&mess);
mess2 = test_compare_pack_methods (&mess.base, &len, &data);
assert (len == 0); /* no non-default values */
free (data);
assert_optional_default_values_are_default (mess2);
foo__default_optional_values__free_unpacked (mess2, NULL);
}
static void
assert_optional_lowercase_enum_default_value_is_default (Foo__LowerCase *mess)
{
assert (!mess->has_value);
assert (mess->value == 2);
}
static void
test_optional_lowercase_enum_default_value (void)
{
Foo__LowerCase mess = FOO__LOWER_CASE__INIT;
Foo__LowerCase *mess2;
size_t len; uint8_t *data;
assert_optional_lowercase_enum_default_value_is_default (&mess);
mess2 = test_compare_pack_methods (&mess.base, &len, &data);
assert (len == 0); /* no non-default values */
free (data);
assert_optional_lowercase_enum_default_value_is_default (mess2);
foo__lower_case__free_unpacked (mess2, NULL);
}
static void
test_field_merge (void)
{
Foo__TestMessOptional msg1 = FOO__TEST_MESS_OPTIONAL__INIT;
Foo__SubMess sub1 = FOO__SUB_MESS__INIT;
Foo__SubMess__SubSubMess subsub1 = FOO__SUB_MESS__SUB_SUB_MESS__INIT;
msg1.has_test_int32 = 1;
msg1.test_int32 = 12345;
msg1.has_test_sint32 = 1;
msg1.test_sint32 = -12345;
msg1.has_test_int64 = 1;
msg1.test_int64 = 232;
msg1.test_string = "123";
msg1.test_message = &sub1;
sub1.has_val1 = 1;
sub1.val1 = 4;
sub1.has_val2 = 1;
sub1.val2 = 5;
int32_t arr1[] = {0, 1};
sub1.n_rep = 2;
sub1.rep = arr1;
sub1.sub1 = &subsub1;
sub1.sub2 = &subsub1;
subsub1.n_rep = 2;
subsub1.rep = arr1;
subsub1.str1 = "test";
size_t msg_size = foo__test_mess_optional__get_packed_size (&msg1);
Foo__TestMessOptional msg2 = FOO__TEST_MESS_OPTIONAL__INIT;
Foo__SubMess sub2 = FOO__SUB_MESS__INIT;
Foo__SubMess__SubSubMess subsub2 = FOO__SUB_MESS__SUB_SUB_MESS__INIT;
msg2.has_test_int64 = 1;
msg2.test_int64 = 2;
msg2.has_test_enum = 1;
msg2.test_enum = FOO__TEST_ENUM__VALUE128;
msg2.test_string = "456";
msg2.test_message = &sub2;
sub2.has_val2 = 1;
sub2.val2 = 666;
int32_t arr2[] = {2, 3};
sub2.n_rep = 2;
sub2.rep = arr2;
sub2.sub1 = &subsub2;
subsub2.has_val1 = 1;
subsub2.val1 = 1;
msg_size += foo__test_mess_optional__get_packed_size (&msg2);
uint8_t *packed_buffer = (uint8_t *)malloc (msg_size);
size_t packed_size = foo__test_mess_optional__pack (&msg1, packed_buffer);
packed_size += foo__test_mess_optional__pack (&msg2, packed_buffer + packed_size);
assert (packed_size == msg_size);
Foo__TestMessOptional *merged = foo__test_mess_optional__unpack
(NULL, msg_size, packed_buffer);
assert (merged->has_test_int32 && merged->test_int32 == msg1.test_int32);
assert (merged->has_test_sint32 && merged->test_sint32 == msg1.test_sint32);
assert (merged->has_test_int64 && merged->test_int64 == msg2.test_int64);
assert (merged->has_test_enum && merged->test_enum == msg2.test_enum);
assert (strcmp (merged->test_string, msg2.test_string) == 0);
assert (merged->test_message->has_val1 && merged->test_message->val1 == sub1.val1);
assert (merged->test_message->has_val2 && merged->test_message->val2 == sub2.val2);
/* Repeated fields should get concatenated */
int32_t merged_arr[] = {2, 3, 0, 1};
assert (merged->test_message->n_rep == 4 &&
memcmp(merged->test_message->rep, merged_arr, sizeof(merged_arr)) == 0);
assert (merged->test_message->sub1->val1 == subsub2.val1);
assert (memcmp(merged->test_message->sub1->bytes1.data,
subsub2.bytes1.data, subsub2.bytes1.len) == 0);
assert (strcmp(merged->test_message->sub1->str1, subsub1.str1) == 0);
assert (merged->test_message->sub1->n_rep == subsub1.n_rep &&
memcmp(merged->test_message->sub1->rep, arr1, sizeof(arr1)) == 0);
assert (!merged->test_message->sub2->has_val1
&& merged->test_message->sub2->val1 == subsub1.val1);
free (packed_buffer);
foo__test_mess_optional__free_unpacked (merged, NULL);
}
static struct alloc_data {
uint32_t alloc_count;
int32_t allocs_left;
} test_allocator_data;
static void *test_alloc(void *allocator_data, size_t size)
{
struct alloc_data *ad = allocator_data;
void *rv = NULL;
if (ad->allocs_left-- > 0)
rv = malloc (size);
/* fprintf (stderr, "alloc %d = %p\n", size, rv); */
if (rv)
ad->alloc_count++;
return rv;
}
static void test_free (void *allocator_data, void *data)
{
struct alloc_data *ad = allocator_data;
/* fprintf (stderr, "free %p\n", data); */
free (data);
if (data)
ad->alloc_count--;
}
static ProtobufCAllocator test_allocator = {
.alloc = test_alloc,
.free = test_free,
.allocator_data = &test_allocator_data,
};
#define SETUP_TEST_ALLOC_BUFFER(pbuf, len) \
uint8_t bytes[] = "some bytes", *pbuf; \
size_t len, _len2; \
Foo__DefaultRequiredValues _req = FOO__DEFAULT_REQUIRED_VALUES__INIT; \
Foo__AllocValues _mess = FOO__ALLOC_VALUES__INIT; \
_mess.a_string = "some string"; \
_mess.r_string = repeated_strings_2; \
_mess.n_r_string = sizeof(repeated_strings_2) / sizeof(*repeated_strings_2); \
_mess.a_bytes.len = sizeof(bytes); \
_mess.a_bytes.data = bytes; \
_mess.a_mess = &_req; \
len = foo__alloc_values__get_packed_size (&_mess); \
pbuf = malloc (len); \
assert (pbuf); \
_len2 = foo__alloc_values__pack (&_mess, pbuf); \
assert (len == _len2);
static void
test_alloc_graceful_cleanup (uint8_t *packed, size_t len, int good_allocs)
{
Foo__AllocValues *mess;
test_allocator_data.alloc_count = 0;
test_allocator_data.allocs_left = good_allocs;
mess = foo__alloc_values__unpack (&test_allocator, len, packed);
assert (test_allocator_data.allocs_left < 0 ? !mess : !!mess);
if (mess)
foo__alloc_values__free_unpacked (mess, &test_allocator);
assert (0 == test_allocator_data.alloc_count);
}
static void
test_alloc_free_all (void)
{
SETUP_TEST_ALLOC_BUFFER (packed, len);
test_alloc_graceful_cleanup (packed, len, INT32_MAX);
free (packed);
}
/* TODO: test alloc failure for slab, unknown fields */
static void
test_alloc_fail (void)
{
int i = 0;
SETUP_TEST_ALLOC_BUFFER (packed, len);
do test_alloc_graceful_cleanup (packed, len, i++);
while (test_allocator_data.allocs_left < 0);
free (packed);
}
/* === simple testing framework === */
typedef void (*TestFunc) (void);
typedef struct {
const char *name;
TestFunc func;
} Test;
static Test tests[] =
{
{ "small enums", test_enum_small },
{ "big enums", test_enum_big },
{ "test field numbers", test_field_numbers },
{ "test required int32", test_required_int32 },
{ "test required sint32", test_required_sint32 },
{ "test required sfixed32", test_required_sfixed32 },
{ "test required int64", test_required_int64 },
{ "test required sint64", test_required_sint64 },
{ "test required sfixed64", test_required_sfixed64 },
{ "test required uint32", test_required_uint32 },
{ "test required fixed32", test_required_fixed32 },
{ "test required uint64", test_required_uint64 },
{ "test required fixed64", test_required_fixed64 },
{ "test required float", test_required_float },
{ "test required double", test_required_double },
{ "test required bool", test_required_bool },
{ "test required TestEnumSmall", test_required_TestEnumSmall },
{ "test required TestEnum", test_required_TestEnum },
{ "test required string", test_required_string },
{ "test required bytes", test_required_bytes },
{ "test required SubMess", test_required_SubMess },
{ "test empty optional" ,test_empty_optional },
{ "test optional int32", test_optional_int32 },
{ "test optional sint32", test_optional_sint32 },
{ "test optional sfixed32", test_optional_sfixed32 },
{ "test optional int64", test_optional_int64 },
{ "test optional sint64", test_optional_sint64 },
{ "test optional sfixed64", test_optional_sfixed64 },
{ "test optional uint32", test_optional_uint32 },
{ "test optional fixed32", test_optional_fixed32 },
{ "test optional uint64", test_optional_uint64 },
{ "test optional fixed64", test_optional_fixed64 },
{ "test optional float", test_optional_float },
{ "test optional double", test_optional_double },
{ "test optional bool", test_optional_bool },
{ "test optional TestEnumSmall", test_optional_TestEnumSmall },
{ "test optional TestEnum", test_optional_TestEnum },
{ "test optional string", test_optional_string },
{ "test optional bytes", test_optional_bytes },
{ "test optional SubMess", test_optional_SubMess },
{ "test empty repeated" ,test_empty_repeated },
{ "test repeated int32" ,test_repeated_int32 },
{ "test repeated sint32" ,test_repeated_sint32 },
{ "test repeated sfixed32" ,test_repeated_sfixed32 },
{ "test repeated uint32", test_repeated_uint32 },
{ "test repeated int64", test_repeated_int64 },
{ "test repeated sint64", test_repeated_sint64 },
{ "test repeated sfixed64", test_repeated_sfixed64 },
{ "test repeated fixed32", test_repeated_fixed32 },
{ "test repeated uint64", test_repeated_uint64 },
{ "test repeated fixed64", test_repeated_fixed64 },
{ "test repeated float", test_repeated_float },
{ "test repeated double", test_repeated_double },
{ "test repeated boolean", test_repeated_boolean },
{ "test repeated TestEnumSmall", test_repeated_TestEnumSmall },
{ "test repeated TestEnum", test_repeated_TestEnum },
{ "test repeated string", test_repeated_string },
{ "test repeated bytes", test_repeated_bytes },
{ "test repeated SubMess", test_repeated_SubMess },
{ "test packed repeated int32", test_packed_repeated_int32 },
{ "test packed repeated sint32", test_packed_repeated_sint32 },
{ "test packed repeated sfixed32" ,test_packed_repeated_sfixed32 },
{ "test packed repeated uint32", test_packed_repeated_uint32 },
{ "test packed repeated int64", test_packed_repeated_int64 },
{ "test packed repeated sint64", test_packed_repeated_sint64 },
{ "test packed repeated sfixed64", test_packed_repeated_sfixed64 },
{ "test packed repeated fixed32", test_packed_repeated_fixed32 },
{ "test packed repeated uint64", test_packed_repeated_uint64 },
{ "test packed repeated fixed64", test_packed_repeated_fixed64 },
{ "test packed repeated float", test_packed_repeated_float },
{ "test packed repeated double", test_packed_repeated_double },
{ "test packed repeated boolean", test_packed_repeated_boolean },
{ "test packed repeated TestEnumSmall", test_packed_repeated_TestEnumSmall },
{ "test packed repeated TestEnum", test_packed_repeated_TestEnum },
{ "test unknown fields", test_unknown_fields },
{ "test enum lookups", test_enum_lookups },
{ "test message lookups", test_message_lookups },
{ "test required default values", test_required_default_values },
{ "test optional default values", test_optional_default_values },
{ "test optional lowercase enum default value", test_optional_lowercase_enum_default_value },
{ "test field merge", test_field_merge },
{ "test free unpacked", test_alloc_free_all },
{ "test alloc failure", test_alloc_fail },
};
#define n_tests (sizeof(tests)/sizeof(Test))
int main ()
{
unsigned i;
for (i = 0; i < n_tests; i++)
{
fprintf (stderr, "Test: %s... ", tests[i].name);
tests[i].func ();
fprintf (stderr, " done.\n");
}
return 0;
}