1*c8d645caSAndroid Build Coastguard Worker /* Make sure that all fields are freed in various scenarios. */
2*c8d645caSAndroid Build Coastguard Worker
3*c8d645caSAndroid Build Coastguard Worker #include <pb_decode.h>
4*c8d645caSAndroid Build Coastguard Worker #include <pb_encode.h>
5*c8d645caSAndroid Build Coastguard Worker #include <malloc_wrappers.h>
6*c8d645caSAndroid Build Coastguard Worker #include <stdio.h>
7*c8d645caSAndroid Build Coastguard Worker #include <test_helpers.h>
8*c8d645caSAndroid Build Coastguard Worker #include "mem_release.pb.h"
9*c8d645caSAndroid Build Coastguard Worker
10*c8d645caSAndroid Build Coastguard Worker #define TEST(x) if (!(x)) { \
11*c8d645caSAndroid Build Coastguard Worker fprintf(stderr, "Test " #x " on line %d failed.\n", __LINE__); \
12*c8d645caSAndroid Build Coastguard Worker return false; \
13*c8d645caSAndroid Build Coastguard Worker }
14*c8d645caSAndroid Build Coastguard Worker
15*c8d645caSAndroid Build Coastguard Worker static char *test_str_arr[] = {"1", "2", ""};
16*c8d645caSAndroid Build Coastguard Worker static SubMessage test_msg_arr[] = {SubMessage_init_zero, SubMessage_init_zero};
17*c8d645caSAndroid Build Coastguard Worker static pb_extension_t ext1, ext2;
18*c8d645caSAndroid Build Coastguard Worker
fill_TestMessage(TestMessage * msg)19*c8d645caSAndroid Build Coastguard Worker static void fill_TestMessage(TestMessage *msg)
20*c8d645caSAndroid Build Coastguard Worker {
21*c8d645caSAndroid Build Coastguard Worker msg->static_req_submsg.dynamic_str = "12345";
22*c8d645caSAndroid Build Coastguard Worker msg->static_req_submsg.dynamic_str_arr_count = 3;
23*c8d645caSAndroid Build Coastguard Worker msg->static_req_submsg.dynamic_str_arr = test_str_arr;
24*c8d645caSAndroid Build Coastguard Worker msg->static_req_submsg.dynamic_submsg_count = 2;
25*c8d645caSAndroid Build Coastguard Worker msg->static_req_submsg.dynamic_submsg = test_msg_arr;
26*c8d645caSAndroid Build Coastguard Worker msg->static_req_submsg.dynamic_submsg[1].dynamic_str = "abc";
27*c8d645caSAndroid Build Coastguard Worker msg->static_opt_submsg.dynamic_str = "abc";
28*c8d645caSAndroid Build Coastguard Worker msg->static_rep_submsg_count = 2;
29*c8d645caSAndroid Build Coastguard Worker msg->static_rep_submsg[1].dynamic_str = "abc";
30*c8d645caSAndroid Build Coastguard Worker msg->has_static_opt_submsg = true;
31*c8d645caSAndroid Build Coastguard Worker msg->dynamic_submsg = &msg->static_req_submsg;
32*c8d645caSAndroid Build Coastguard Worker
33*c8d645caSAndroid Build Coastguard Worker msg->extensions = &ext1;
34*c8d645caSAndroid Build Coastguard Worker ext1.type = &dynamic_ext;
35*c8d645caSAndroid Build Coastguard Worker ext1.dest = &msg->static_req_submsg;
36*c8d645caSAndroid Build Coastguard Worker ext1.next = &ext2;
37*c8d645caSAndroid Build Coastguard Worker ext2.type = &static_ext;
38*c8d645caSAndroid Build Coastguard Worker ext2.dest = &msg->static_req_submsg;
39*c8d645caSAndroid Build Coastguard Worker ext2.next = NULL;
40*c8d645caSAndroid Build Coastguard Worker }
41*c8d645caSAndroid Build Coastguard Worker
42*c8d645caSAndroid Build Coastguard Worker /* Basic fields, nested submessages, extensions */
test_TestMessage()43*c8d645caSAndroid Build Coastguard Worker static bool test_TestMessage()
44*c8d645caSAndroid Build Coastguard Worker {
45*c8d645caSAndroid Build Coastguard Worker uint8_t buffer[256];
46*c8d645caSAndroid Build Coastguard Worker size_t msgsize;
47*c8d645caSAndroid Build Coastguard Worker
48*c8d645caSAndroid Build Coastguard Worker /* Construct a message with various fields filled in */
49*c8d645caSAndroid Build Coastguard Worker {
50*c8d645caSAndroid Build Coastguard Worker TestMessage msg = TestMessage_init_zero;
51*c8d645caSAndroid Build Coastguard Worker pb_ostream_t stream;
52*c8d645caSAndroid Build Coastguard Worker
53*c8d645caSAndroid Build Coastguard Worker fill_TestMessage(&msg);
54*c8d645caSAndroid Build Coastguard Worker
55*c8d645caSAndroid Build Coastguard Worker stream = pb_ostream_from_buffer(buffer, sizeof(buffer));
56*c8d645caSAndroid Build Coastguard Worker if (!pb_encode(&stream, TestMessage_fields, &msg))
57*c8d645caSAndroid Build Coastguard Worker {
58*c8d645caSAndroid Build Coastguard Worker fprintf(stderr, "Encode failed: %s\n", PB_GET_ERROR(&stream));
59*c8d645caSAndroid Build Coastguard Worker return false;
60*c8d645caSAndroid Build Coastguard Worker }
61*c8d645caSAndroid Build Coastguard Worker msgsize = stream.bytes_written;
62*c8d645caSAndroid Build Coastguard Worker }
63*c8d645caSAndroid Build Coastguard Worker
64*c8d645caSAndroid Build Coastguard Worker /* Output encoded message for debug */
65*c8d645caSAndroid Build Coastguard Worker SET_BINARY_MODE(stdout);
66*c8d645caSAndroid Build Coastguard Worker fwrite(buffer, 1, msgsize, stdout);
67*c8d645caSAndroid Build Coastguard Worker
68*c8d645caSAndroid Build Coastguard Worker /* Decode memory using dynamic allocation */
69*c8d645caSAndroid Build Coastguard Worker {
70*c8d645caSAndroid Build Coastguard Worker TestMessage msg = TestMessage_init_zero;
71*c8d645caSAndroid Build Coastguard Worker pb_istream_t stream;
72*c8d645caSAndroid Build Coastguard Worker SubMessage ext2_dest;
73*c8d645caSAndroid Build Coastguard Worker
74*c8d645caSAndroid Build Coastguard Worker msg.extensions = &ext1;
75*c8d645caSAndroid Build Coastguard Worker ext1.type = &dynamic_ext;
76*c8d645caSAndroid Build Coastguard Worker ext1.dest = NULL;
77*c8d645caSAndroid Build Coastguard Worker ext1.next = &ext2;
78*c8d645caSAndroid Build Coastguard Worker ext2.type = &static_ext;
79*c8d645caSAndroid Build Coastguard Worker ext2.dest = &ext2_dest;
80*c8d645caSAndroid Build Coastguard Worker ext2.next = NULL;
81*c8d645caSAndroid Build Coastguard Worker
82*c8d645caSAndroid Build Coastguard Worker stream = pb_istream_from_buffer(buffer, msgsize);
83*c8d645caSAndroid Build Coastguard Worker if (!pb_decode(&stream, TestMessage_fields, &msg))
84*c8d645caSAndroid Build Coastguard Worker {
85*c8d645caSAndroid Build Coastguard Worker fprintf(stderr, "Decode failed: %s\n", PB_GET_ERROR(&stream));
86*c8d645caSAndroid Build Coastguard Worker return false;
87*c8d645caSAndroid Build Coastguard Worker }
88*c8d645caSAndroid Build Coastguard Worker
89*c8d645caSAndroid Build Coastguard Worker /* Make sure it encodes back to same data */
90*c8d645caSAndroid Build Coastguard Worker {
91*c8d645caSAndroid Build Coastguard Worker uint8_t buffer2[256];
92*c8d645caSAndroid Build Coastguard Worker pb_ostream_t ostream = pb_ostream_from_buffer(buffer2, sizeof(buffer2));
93*c8d645caSAndroid Build Coastguard Worker TEST(pb_encode(&ostream, TestMessage_fields, &msg));
94*c8d645caSAndroid Build Coastguard Worker TEST(ostream.bytes_written == msgsize);
95*c8d645caSAndroid Build Coastguard Worker TEST(memcmp(buffer, buffer2, msgsize) == 0);
96*c8d645caSAndroid Build Coastguard Worker }
97*c8d645caSAndroid Build Coastguard Worker
98*c8d645caSAndroid Build Coastguard Worker /* Make sure that malloc counters work */
99*c8d645caSAndroid Build Coastguard Worker TEST(get_alloc_count() > 0);
100*c8d645caSAndroid Build Coastguard Worker
101*c8d645caSAndroid Build Coastguard Worker /* Make sure that pb_release releases everything */
102*c8d645caSAndroid Build Coastguard Worker pb_release(TestMessage_fields, &msg);
103*c8d645caSAndroid Build Coastguard Worker TEST(get_alloc_count() == 0);
104*c8d645caSAndroid Build Coastguard Worker
105*c8d645caSAndroid Build Coastguard Worker /* Check that double-free is a no-op */
106*c8d645caSAndroid Build Coastguard Worker pb_release(TestMessage_fields, &msg);
107*c8d645caSAndroid Build Coastguard Worker TEST(get_alloc_count() == 0);
108*c8d645caSAndroid Build Coastguard Worker }
109*c8d645caSAndroid Build Coastguard Worker
110*c8d645caSAndroid Build Coastguard Worker return true;
111*c8d645caSAndroid Build Coastguard Worker }
112*c8d645caSAndroid Build Coastguard Worker
113*c8d645caSAndroid Build Coastguard Worker /* Oneofs */
test_OneofMessage()114*c8d645caSAndroid Build Coastguard Worker static bool test_OneofMessage()
115*c8d645caSAndroid Build Coastguard Worker {
116*c8d645caSAndroid Build Coastguard Worker uint8_t buffer[256];
117*c8d645caSAndroid Build Coastguard Worker size_t msgsize;
118*c8d645caSAndroid Build Coastguard Worker
119*c8d645caSAndroid Build Coastguard Worker {
120*c8d645caSAndroid Build Coastguard Worker pb_ostream_t stream = pb_ostream_from_buffer(buffer, sizeof(buffer));
121*c8d645caSAndroid Build Coastguard Worker
122*c8d645caSAndroid Build Coastguard Worker /* Encode first with TestMessage */
123*c8d645caSAndroid Build Coastguard Worker {
124*c8d645caSAndroid Build Coastguard Worker OneofMessage msg = OneofMessage_init_zero;
125*c8d645caSAndroid Build Coastguard Worker msg.which_msgs = OneofMessage_msg1_tag;
126*c8d645caSAndroid Build Coastguard Worker
127*c8d645caSAndroid Build Coastguard Worker fill_TestMessage(&msg.msgs.msg1);
128*c8d645caSAndroid Build Coastguard Worker
129*c8d645caSAndroid Build Coastguard Worker if (!pb_encode(&stream, OneofMessage_fields, &msg))
130*c8d645caSAndroid Build Coastguard Worker {
131*c8d645caSAndroid Build Coastguard Worker fprintf(stderr, "Encode failed: %s\n", PB_GET_ERROR(&stream));
132*c8d645caSAndroid Build Coastguard Worker return false;
133*c8d645caSAndroid Build Coastguard Worker }
134*c8d645caSAndroid Build Coastguard Worker }
135*c8d645caSAndroid Build Coastguard Worker
136*c8d645caSAndroid Build Coastguard Worker /* Encode second with SubMessage, replacing the oneof item */
137*c8d645caSAndroid Build Coastguard Worker {
138*c8d645caSAndroid Build Coastguard Worker OneofMessage msg = OneofMessage_init_zero;
139*c8d645caSAndroid Build Coastguard Worker char *teststr = "1";
140*c8d645caSAndroid Build Coastguard Worker msg.which_msgs = OneofMessage_msg2_tag;
141*c8d645caSAndroid Build Coastguard Worker
142*c8d645caSAndroid Build Coastguard Worker msg.first = 999;
143*c8d645caSAndroid Build Coastguard Worker msg.msgs.msg2.dynamic_str = "ABCD";
144*c8d645caSAndroid Build Coastguard Worker msg.msgs.msg2.dynamic_str_arr_count = 1;
145*c8d645caSAndroid Build Coastguard Worker msg.msgs.msg2.dynamic_str_arr = &teststr;
146*c8d645caSAndroid Build Coastguard Worker msg.last = 888;
147*c8d645caSAndroid Build Coastguard Worker
148*c8d645caSAndroid Build Coastguard Worker if (!pb_encode(&stream, OneofMessage_fields, &msg))
149*c8d645caSAndroid Build Coastguard Worker {
150*c8d645caSAndroid Build Coastguard Worker fprintf(stderr, "Encode failed: %s\n", PB_GET_ERROR(&stream));
151*c8d645caSAndroid Build Coastguard Worker return false;
152*c8d645caSAndroid Build Coastguard Worker }
153*c8d645caSAndroid Build Coastguard Worker }
154*c8d645caSAndroid Build Coastguard Worker
155*c8d645caSAndroid Build Coastguard Worker /* Encode second SubMessage, invoking submessage merge behavior */
156*c8d645caSAndroid Build Coastguard Worker {
157*c8d645caSAndroid Build Coastguard Worker OneofMessage msg = OneofMessage_init_zero;
158*c8d645caSAndroid Build Coastguard Worker char *teststr = "2";
159*c8d645caSAndroid Build Coastguard Worker msg.which_msgs = OneofMessage_msg2_tag;
160*c8d645caSAndroid Build Coastguard Worker
161*c8d645caSAndroid Build Coastguard Worker msg.first = 99;
162*c8d645caSAndroid Build Coastguard Worker msg.msgs.msg2.dynamic_str = "EFGH";
163*c8d645caSAndroid Build Coastguard Worker msg.msgs.msg2.dynamic_str_arr_count = 1;
164*c8d645caSAndroid Build Coastguard Worker msg.msgs.msg2.dynamic_str_arr = &teststr;
165*c8d645caSAndroid Build Coastguard Worker msg.last = 88;
166*c8d645caSAndroid Build Coastguard Worker
167*c8d645caSAndroid Build Coastguard Worker if (!pb_encode(&stream, OneofMessage_fields, &msg))
168*c8d645caSAndroid Build Coastguard Worker {
169*c8d645caSAndroid Build Coastguard Worker fprintf(stderr, "Encode failed: %s\n", PB_GET_ERROR(&stream));
170*c8d645caSAndroid Build Coastguard Worker return false;
171*c8d645caSAndroid Build Coastguard Worker }
172*c8d645caSAndroid Build Coastguard Worker }
173*c8d645caSAndroid Build Coastguard Worker msgsize = stream.bytes_written;
174*c8d645caSAndroid Build Coastguard Worker }
175*c8d645caSAndroid Build Coastguard Worker
176*c8d645caSAndroid Build Coastguard Worker {
177*c8d645caSAndroid Build Coastguard Worker OneofMessage msg = OneofMessage_init_zero;
178*c8d645caSAndroid Build Coastguard Worker pb_istream_t stream = pb_istream_from_buffer(buffer, msgsize);
179*c8d645caSAndroid Build Coastguard Worker if (!pb_decode(&stream, OneofMessage_fields, &msg))
180*c8d645caSAndroid Build Coastguard Worker {
181*c8d645caSAndroid Build Coastguard Worker fprintf(stderr, "Decode failed: %s\n", PB_GET_ERROR(&stream));
182*c8d645caSAndroid Build Coastguard Worker return false;
183*c8d645caSAndroid Build Coastguard Worker }
184*c8d645caSAndroid Build Coastguard Worker
185*c8d645caSAndroid Build Coastguard Worker TEST(msg.first == 99);
186*c8d645caSAndroid Build Coastguard Worker TEST(msg.which_msgs == OneofMessage_msg2_tag);
187*c8d645caSAndroid Build Coastguard Worker TEST(msg.msgs.msg2.dynamic_str);
188*c8d645caSAndroid Build Coastguard Worker TEST(strcmp(msg.msgs.msg2.dynamic_str, "EFGH") == 0);
189*c8d645caSAndroid Build Coastguard Worker TEST(msg.msgs.msg2.dynamic_str_arr != NULL);
190*c8d645caSAndroid Build Coastguard Worker TEST(msg.msgs.msg2.dynamic_str_arr_count == 2);
191*c8d645caSAndroid Build Coastguard Worker TEST(strcmp(msg.msgs.msg2.dynamic_str_arr[0], "1") == 0);
192*c8d645caSAndroid Build Coastguard Worker TEST(strcmp(msg.msgs.msg2.dynamic_str_arr[1], "2") == 0);
193*c8d645caSAndroid Build Coastguard Worker TEST(msg.msgs.msg2.dynamic_submsg == NULL);
194*c8d645caSAndroid Build Coastguard Worker TEST(msg.last == 88);
195*c8d645caSAndroid Build Coastguard Worker
196*c8d645caSAndroid Build Coastguard Worker pb_release(OneofMessage_fields, &msg);
197*c8d645caSAndroid Build Coastguard Worker TEST(get_alloc_count() == 0);
198*c8d645caSAndroid Build Coastguard Worker pb_release(OneofMessage_fields, &msg);
199*c8d645caSAndroid Build Coastguard Worker TEST(get_alloc_count() == 0);
200*c8d645caSAndroid Build Coastguard Worker }
201*c8d645caSAndroid Build Coastguard Worker
202*c8d645caSAndroid Build Coastguard Worker return true;
203*c8d645caSAndroid Build Coastguard Worker }
204*c8d645caSAndroid Build Coastguard Worker
dummy_decode_cb(pb_istream_t * stream,const pb_field_t * field,void ** arg)205*c8d645caSAndroid Build Coastguard Worker static bool dummy_decode_cb(pb_istream_t *stream, const pb_field_t *field, void **arg)
206*c8d645caSAndroid Build Coastguard Worker {
207*c8d645caSAndroid Build Coastguard Worker return false;
208*c8d645caSAndroid Build Coastguard Worker }
209*c8d645caSAndroid Build Coastguard Worker
210*c8d645caSAndroid Build Coastguard Worker /* Garbage input */
test_Garbage()211*c8d645caSAndroid Build Coastguard Worker static bool test_Garbage()
212*c8d645caSAndroid Build Coastguard Worker {
213*c8d645caSAndroid Build Coastguard Worker const uint8_t buffer[] = "I'm only happy when it rains";
214*c8d645caSAndroid Build Coastguard Worker const size_t msgsize = sizeof(buffer);
215*c8d645caSAndroid Build Coastguard Worker
216*c8d645caSAndroid Build Coastguard Worker {
217*c8d645caSAndroid Build Coastguard Worker OneofMessage msg = OneofMessage_init_zero;
218*c8d645caSAndroid Build Coastguard Worker pb_istream_t stream = pb_istream_from_buffer(buffer, msgsize);
219*c8d645caSAndroid Build Coastguard Worker TEST(!pb_decode(&stream, OneofMessage_fields, &msg));
220*c8d645caSAndroid Build Coastguard Worker }
221*c8d645caSAndroid Build Coastguard Worker
222*c8d645caSAndroid Build Coastguard Worker {
223*c8d645caSAndroid Build Coastguard Worker TestMessage msg = TestMessage_init_zero;
224*c8d645caSAndroid Build Coastguard Worker pb_istream_t stream = pb_istream_from_buffer(buffer, msgsize);
225*c8d645caSAndroid Build Coastguard Worker TEST(!pb_decode(&stream, TestMessage_fields, &msg));
226*c8d645caSAndroid Build Coastguard Worker }
227*c8d645caSAndroid Build Coastguard Worker
228*c8d645caSAndroid Build Coastguard Worker {
229*c8d645caSAndroid Build Coastguard Worker RepeatedMessage msg = RepeatedMessage_init_zero;
230*c8d645caSAndroid Build Coastguard Worker pb_istream_t stream = pb_istream_from_buffer(buffer, msgsize);
231*c8d645caSAndroid Build Coastguard Worker msg.subs.arg = NULL;
232*c8d645caSAndroid Build Coastguard Worker msg.subs.funcs.decode = dummy_decode_cb;
233*c8d645caSAndroid Build Coastguard Worker TEST(!pb_decode(&stream, RepeatedMessage_fields, &msg));
234*c8d645caSAndroid Build Coastguard Worker }
235*c8d645caSAndroid Build Coastguard Worker
236*c8d645caSAndroid Build Coastguard Worker return true;
237*c8d645caSAndroid Build Coastguard Worker }
238*c8d645caSAndroid Build Coastguard Worker
main()239*c8d645caSAndroid Build Coastguard Worker int main()
240*c8d645caSAndroid Build Coastguard Worker {
241*c8d645caSAndroid Build Coastguard Worker if (test_TestMessage() && test_OneofMessage() && test_Garbage())
242*c8d645caSAndroid Build Coastguard Worker return 0;
243*c8d645caSAndroid Build Coastguard Worker else
244*c8d645caSAndroid Build Coastguard Worker return 1;
245*c8d645caSAndroid Build Coastguard Worker }
246*c8d645caSAndroid Build Coastguard Worker
247