| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246 | /* Make sure that all fields are freed in various scenarios. */#include <pb_decode.h>#include <pb_encode.h>#include <malloc_wrappers.h>#include <stdio.h>#include <test_helpers.h>#include "mem_release.pb.h"#define TEST(x) if (!(x)) { \    fprintf(stderr, "Test %s on line %d failed.\n", #x, __LINE__); \    return false; \    }static char *test_str_arr[] = {"1", "2", ""};static SubMessage test_msg_arr[] = {SubMessage_init_zero, SubMessage_init_zero};static pb_extension_t ext1, ext2;static void fill_TestMessage(TestMessage *msg){    msg->static_req_submsg.dynamic_str = "12345";    msg->static_req_submsg.dynamic_str_arr_count = 3;    msg->static_req_submsg.dynamic_str_arr = test_str_arr;    msg->static_req_submsg.dynamic_submsg_count = 2;    msg->static_req_submsg.dynamic_submsg = test_msg_arr;    msg->static_req_submsg.dynamic_submsg[1].dynamic_str = "abc";    msg->static_opt_submsg.dynamic_str = "abc";    msg->static_rep_submsg_count = 2;    msg->static_rep_submsg[1].dynamic_str = "abc";    msg->has_static_opt_submsg = true;    msg->dynamic_submsg = &msg->static_req_submsg;    msg->extensions = &ext1;    ext1.type = &dynamic_ext;    ext1.dest = &msg->static_req_submsg;    ext1.next = &ext2;    ext2.type = &static_ext;    ext2.dest = &msg->static_req_submsg;    ext2.next = NULL;}/* Basic fields, nested submessages, extensions */static bool test_TestMessage(){    uint8_t buffer[256];    size_t msgsize;        /* Construct a message with various fields filled in */    {        TestMessage msg = TestMessage_init_zero;        pb_ostream_t stream;        fill_TestMessage(&msg);                stream = pb_ostream_from_buffer(buffer, sizeof(buffer));        if (!pb_encode(&stream, TestMessage_fields, &msg))        {            fprintf(stderr, "Encode failed: %s\n", PB_GET_ERROR(&stream));            return false;        }        msgsize = stream.bytes_written;    }        /* Output encoded message for debug */    SET_BINARY_MODE(stdout);    fwrite(buffer, 1, msgsize, stdout);        /* Decode memory using dynamic allocation */    {        TestMessage msg = TestMessage_init_zero;        pb_istream_t stream;        SubMessage ext2_dest;        msg.extensions = &ext1;        ext1.type = &dynamic_ext;        ext1.dest = NULL;        ext1.next = &ext2;        ext2.type = &static_ext;        ext2.dest = &ext2_dest;        ext2.next = NULL;                stream = pb_istream_from_buffer(buffer, msgsize);        if (!pb_decode(&stream, TestMessage_fields, &msg))        {            fprintf(stderr, "Decode failed: %s\n", PB_GET_ERROR(&stream));            return false;        }                /* Make sure it encodes back to same data */        {            uint8_t buffer2[256];            pb_ostream_t ostream = pb_ostream_from_buffer(buffer2, sizeof(buffer2));            TEST(pb_encode(&ostream, TestMessage_fields, &msg));            TEST(ostream.bytes_written == msgsize);            TEST(memcmp(buffer, buffer2, msgsize) == 0);        }                /* Make sure that malloc counters work */        TEST(get_alloc_count() > 0);                /* Make sure that pb_release releases everything */        pb_release(TestMessage_fields, &msg);        TEST(get_alloc_count() == 0);                /* Check that double-free is a no-op */        pb_release(TestMessage_fields, &msg);        TEST(get_alloc_count() == 0);    }        return true;}/* Oneofs */static bool test_OneofMessage(){    uint8_t buffer[256];    size_t msgsize;    {        pb_ostream_t stream = pb_ostream_from_buffer(buffer, sizeof(buffer));        /* Encode first with TestMessage */        {            OneofMessage msg = OneofMessage_init_zero;            msg.which_msgs = OneofMessage_msg1_tag;            fill_TestMessage(&msg.msgs.msg1);            if (!pb_encode(&stream, OneofMessage_fields, &msg))            {                fprintf(stderr, "Encode failed: %s\n", PB_GET_ERROR(&stream));                return false;            }        }        /* Encode second with SubMessage, replacing the oneof item */        {            OneofMessage msg = OneofMessage_init_zero;            char *teststr = "1";            msg.which_msgs = OneofMessage_msg2_tag;            msg.first = 999;            msg.msgs.msg2.dynamic_str = "ABCD";            msg.msgs.msg2.dynamic_str_arr_count = 1;            msg.msgs.msg2.dynamic_str_arr = &teststr;            msg.last = 888;            if (!pb_encode(&stream, OneofMessage_fields, &msg))            {                fprintf(stderr, "Encode failed: %s\n", PB_GET_ERROR(&stream));                return false;            }        }        /* Encode second SubMessage, invoking submessage merge behavior */        {            OneofMessage msg = OneofMessage_init_zero;            char *teststr = "2";            msg.which_msgs = OneofMessage_msg2_tag;            msg.first = 99;            msg.msgs.msg2.dynamic_str = "EFGH";            msg.msgs.msg2.dynamic_str_arr_count = 1;            msg.msgs.msg2.dynamic_str_arr = &teststr;            msg.last = 88;            if (!pb_encode(&stream, OneofMessage_fields, &msg))            {                fprintf(stderr, "Encode failed: %s\n", PB_GET_ERROR(&stream));                return false;            }        }        msgsize = stream.bytes_written;    }    {        OneofMessage msg = OneofMessage_init_zero;        pb_istream_t stream = pb_istream_from_buffer(buffer, msgsize);        if (!pb_decode(&stream, OneofMessage_fields, &msg))        {            fprintf(stderr, "Decode failed: %s\n", PB_GET_ERROR(&stream));            return false;        }        TEST(msg.first == 99);        TEST(msg.which_msgs == OneofMessage_msg2_tag);        TEST(msg.msgs.msg2.dynamic_str);        TEST(strcmp(msg.msgs.msg2.dynamic_str, "EFGH") == 0);        TEST(msg.msgs.msg2.dynamic_str_arr != NULL);        TEST(msg.msgs.msg2.dynamic_str_arr_count == 2);        TEST(strcmp(msg.msgs.msg2.dynamic_str_arr[0], "1") == 0);        TEST(strcmp(msg.msgs.msg2.dynamic_str_arr[1], "2") == 0);        TEST(msg.msgs.msg2.dynamic_submsg == NULL);        TEST(msg.last == 88);        pb_release(OneofMessage_fields, &msg);        TEST(get_alloc_count() == 0);        pb_release(OneofMessage_fields, &msg);        TEST(get_alloc_count() == 0);    }    return true;}static bool dummy_decode_cb(pb_istream_t *stream, const pb_field_t *field, void **arg){    return false;}/* Garbage input */static bool test_Garbage(){    const uint8_t buffer[] = "I'm only happy when it rains";    const size_t msgsize = sizeof(buffer);    {        OneofMessage msg = OneofMessage_init_zero;        pb_istream_t stream = pb_istream_from_buffer(buffer, msgsize);        TEST(!pb_decode(&stream, OneofMessage_fields, &msg));    }    {        TestMessage msg = TestMessage_init_zero;        pb_istream_t stream = pb_istream_from_buffer(buffer, msgsize);        TEST(!pb_decode(&stream, TestMessage_fields, &msg));    }    {        RepeatedMessage msg = RepeatedMessage_init_zero;        pb_istream_t stream = pb_istream_from_buffer(buffer, msgsize);        msg.subs.arg = NULL;        msg.subs.funcs.decode = dummy_decode_cb;        TEST(!pb_decode(&stream, RepeatedMessage_fields, &msg));    }    return true;}int main(){    if (test_TestMessage() && test_OneofMessage() && test_Garbage())        return 0;    else        return 1;}
 |