blob: 1b20db147216cda90c2028b1d57a275868dd5865 [file] [log] [blame]
Holger Hans Peter Freytherfaf1f642011-06-23 17:53:27 -04001#include <stdio.h>
2#include <stdlib.h>
3#include <inttypes.h>
4
5#include <openbsc/gprs_llc.h>
Holger Hans Peter Freytherce1b22e2014-08-04 14:22:13 +02006#include <openbsc/gprs_utils.h>
Holger Hans Peter Freytherfaf1f642011-06-23 17:53:27 -04007
8#define ASSERT_FALSE(x) if (x) { printf("Should have returned false.\n"); abort(); }
9#define ASSERT_TRUE(x) if (!x) { printf("Should have returned true.\n"); abort(); }
10
11/**
12 * GSM 04.64 8.4.2 Receipt of unacknowledged information
13 */
14static int nu_is_retransmission(uint16_t nu, uint16_t vur)
15{
16 int ret = gprs_llc_is_retransmit(nu, vur);
17 printf("N(U) = %d, V(UR) = %d => %s\n", nu, vur,
18 ret == 1 ? "retransmit" : "new");
19 return ret;
20}
21
22static void test_8_4_2()
23{
24 printf("Testing gprs_llc_is_retransmit.\n");
25
26 ASSERT_FALSE(nu_is_retransmission(0, 0));
27 ASSERT_TRUE (nu_is_retransmission(0, 1));
28
29 /* expect 1... check for retransmissions */
30 ASSERT_TRUE (nu_is_retransmission(0, 1));
31 ASSERT_TRUE (nu_is_retransmission(511, 1));
32 ASSERT_TRUE (nu_is_retransmission(483, 1));
33 ASSERT_TRUE (nu_is_retransmission(482, 1));
34 ASSERT_FALSE(nu_is_retransmission(481, 1));
35
36 /* expect 511... check for retransmissions */
37 ASSERT_FALSE(nu_is_retransmission(0, 240)); // ahead
38 ASSERT_FALSE(nu_is_retransmission(0, 511)); // ahead
39 ASSERT_FALSE(nu_is_retransmission(1, 511)); // ahead
40 ASSERT_FALSE(nu_is_retransmission(511, 511)); // same
41 ASSERT_TRUE (nu_is_retransmission(510, 511)); // behind
42 ASSERT_TRUE (nu_is_retransmission(481, 511)); // behind
43 ASSERT_FALSE(nu_is_retransmission(479, 511)); // wrapped
44}
45
Holger Hans Peter Freytherce1b22e2014-08-04 14:22:13 +020046static void apn_round_trip(const uint8_t *input, size_t len, const char *wanted_output)
47{
48 char output[len ? len : 1];
49 uint8_t encoded[len + 50];
50 char *out_str;
51 int enc_len;
52
53 /* decode and verify we have what we want */
54 out_str = gprs_apn_to_str(output, input, len);
55 OSMO_ASSERT(out_str);
56 OSMO_ASSERT(out_str == &output[0]);
57 OSMO_ASSERT(strlen(out_str) == strlen(wanted_output));
58 OSMO_ASSERT(strcmp(out_str, wanted_output) == 0);
59
60 /* encode and verify it */
61 if (len != 0) {
62 enc_len = gprs_str_to_apn(encoded, ARRAY_SIZE(encoded), wanted_output);
63 OSMO_ASSERT(enc_len == len);
64 OSMO_ASSERT(memcmp(encoded, input, enc_len) == 0);
65 } else {
66 enc_len = gprs_str_to_apn(encoded, 0, wanted_output);
67 OSMO_ASSERT(enc_len == -1);
68 }
69}
70
71static void test_gsm_03_03_apn(void)
72{
73
74 {
75 /* test invalid writes */
76 const uint8_t ref[10] = { 0xAB, 0xAC, 0xAD, 0xAE, 0xAF, 0xAB, 0xAC, 0xAD, 0xAE, 0xAF };
77 uint8_t output[10];
78 int enc_len;
79
80 memcpy(output, ref, ARRAY_SIZE(output));
81 enc_len = gprs_str_to_apn(output, 0, "");
82 OSMO_ASSERT(enc_len == -1);
83 OSMO_ASSERT(memcmp(ref, output, ARRAY_SIZE(ref)) == 0);
84
85 memcpy(output, ref, ARRAY_SIZE(output));
86 enc_len = gprs_str_to_apn(output, 0, "foo");
87 OSMO_ASSERT(enc_len == -1);
88 OSMO_ASSERT(memcmp(ref, output, ARRAY_SIZE(ref)) == 0);
89
90 memcpy(output, ref, ARRAY_SIZE(output));
91 enc_len = gprs_str_to_apn(output, 1, "foo");
92 OSMO_ASSERT(enc_len == -1);
93 OSMO_ASSERT(memcmp(ref + 1, output + 1, ARRAY_SIZE(ref) - 1) == 0);
94
95 memcpy(output, ref, ARRAY_SIZE(output));
96 enc_len = gprs_str_to_apn(output, 2, "foo");
97 OSMO_ASSERT(enc_len == -1);
98 OSMO_ASSERT(memcmp(ref + 2, output + 2, ARRAY_SIZE(ref) - 2) == 0);
99
100 memcpy(output, ref, ARRAY_SIZE(output));
101 enc_len = gprs_str_to_apn(output, 3, "foo");
102 OSMO_ASSERT(enc_len == -1);
103 OSMO_ASSERT(memcmp(ref + 3, output + 3, ARRAY_SIZE(ref) - 3) == 0);
104 }
105
106 {
107 /* single empty label */
108 uint8_t input[] = { 0x0 };
109 const char *output = "";
110 apn_round_trip(input, ARRAY_SIZE(input), output);
111 }
112
113 {
114 /* no label */
115 uint8_t input[] = { };
116 const char *output = "";
117 apn_round_trip(input, ARRAY_SIZE(input), output);
118 }
119
120 {
121 /* single label with A */
122 uint8_t input[] = { 0x1, 65 };
123 const char *output = "A";
124 apn_round_trip(input, ARRAY_SIZE(input), output);
125 OSMO_ASSERT(gprs_apn_to_str(NULL, input, ARRAY_SIZE(input) - 1) == NULL);
126 }
127
128 {
129 uint8_t input[] = { 0x3, 65, 66, 67, 0x2, 90, 122 };
130 const char *output = "ABC.Zz";
131 char tmp[strlen(output) + 1];
132 apn_round_trip(input, ARRAY_SIZE(input), output);
133 OSMO_ASSERT(gprs_apn_to_str(tmp, input, ARRAY_SIZE(input) - 1) == NULL);
134 OSMO_ASSERT(gprs_apn_to_str(tmp, input, ARRAY_SIZE(input) - 2) == NULL);
135 OSMO_ASSERT(gprs_apn_to_str(tmp, input, ARRAY_SIZE(input) - 4) == NULL);
136 OSMO_ASSERT(gprs_apn_to_str(tmp, input, ARRAY_SIZE(input) - 5) == NULL);
137 OSMO_ASSERT(gprs_apn_to_str(tmp, input, ARRAY_SIZE(input) - 6) == NULL);
138 }
139}
140
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100141/* TODO: Move tlv testing to libosmocore */
Jacob Erlbeck697a5342014-12-11 12:05:29 +0100142static void check_tlv_parse(uint8_t **data, size_t *data_len,
143 uint8_t exp_tag, size_t exp_len, const uint8_t *exp_val)
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100144{
145 uint8_t *value;
146 size_t value_len;
Jacob Erlbeck697a5342014-12-11 12:05:29 +0100147 uint8_t tag;
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100148 int rc;
Jacob Erlbeck697a5342014-12-11 12:05:29 +0100149 uint8_t *saved_data = *data;
150 size_t saved_data_len = *data_len;
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100151
Jacob Erlbeck697a5342014-12-11 12:05:29 +0100152 rc = gprs_match_tlv(data, data_len, exp_tag ^ 1, NULL, NULL);
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100153 OSMO_ASSERT(rc == 0);
154
Jacob Erlbeck697a5342014-12-11 12:05:29 +0100155 rc = gprs_match_tlv(data, data_len, exp_tag, &value, &value_len);
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100156 OSMO_ASSERT(rc == (int)value_len + 2);
157 OSMO_ASSERT(value_len == exp_len);
158 OSMO_ASSERT(memcmp(value, exp_val, exp_len) == 0);
Jacob Erlbeck697a5342014-12-11 12:05:29 +0100159
160 /* restore data/data_len */
161 *data = saved_data;
162 *data_len = saved_data_len;
163
164 rc = gprs_shift_tlv(data, data_len, &tag, &value, &value_len);
165 OSMO_ASSERT(rc == (int)value_len + 2);
166 OSMO_ASSERT(tag == exp_tag);
167 OSMO_ASSERT(value_len == exp_len);
168 OSMO_ASSERT(memcmp(value, exp_val, exp_len) == 0);
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100169}
170
171static void check_tv_fixed_match(uint8_t **data, size_t *data_len,
172 uint8_t tag, size_t len, const uint8_t *exp_val)
173{
174 uint8_t *value;
175 int rc;
176
177 rc = gprs_match_tv_fixed(data, data_len, tag ^ 1, len, NULL);
178 OSMO_ASSERT(rc == 0);
179
180 rc = gprs_match_tv_fixed(data, data_len, tag, len, &value);
181 OSMO_ASSERT(rc == (int)len + 1);
182 OSMO_ASSERT(memcmp(value, exp_val, len) == 0);
183}
184
185static void check_v_fixed_shift(uint8_t **data, size_t *data_len,
186 size_t len, const uint8_t *exp_val)
187{
188 uint8_t *value;
189 int rc;
190
191 rc = gprs_shift_v_fixed(data, data_len, len, &value);
192 OSMO_ASSERT(rc == (int)len);
193 OSMO_ASSERT(memcmp(value, exp_val, len) == 0);
194}
195
196static void check_lv_shift(uint8_t **data, size_t *data_len,
197 size_t exp_len, const uint8_t *exp_val)
198{
199 uint8_t *value;
200 size_t value_len;
201 int rc;
202
203 rc = gprs_shift_lv(data, data_len, &value, &value_len);
204 OSMO_ASSERT(rc == (int)value_len + 1);
205 OSMO_ASSERT(value_len == exp_len);
206 OSMO_ASSERT(memcmp(value, exp_val, exp_len) == 0);
207}
208
209static void check_tlv_match_data_len(size_t data_len, uint8_t tag, size_t len,
210 const uint8_t *test_data)
211{
212 uint8_t buf[300] = {0};
213
214 uint8_t *unchanged_ptr = buf - 1;
215 size_t unchanged_len = 0xdead;
216 size_t tmp_data_len = data_len;
217 uint8_t *value = unchanged_ptr;
218 size_t value_len = unchanged_len;
219 uint8_t *data = buf;
220
221 OSMO_ASSERT(data_len <= sizeof(buf));
222
223 tlv_put(data, tag, len, test_data);
224 if (data_len < len + 2) {
225 OSMO_ASSERT(-1 == gprs_match_tlv(&data, &tmp_data_len,
226 tag, &value, &value_len));
227 OSMO_ASSERT(tmp_data_len == 0);
228 OSMO_ASSERT(data == buf + data_len);
229 OSMO_ASSERT(value == unchanged_ptr);
230 OSMO_ASSERT(value_len == unchanged_len);
231 } else {
232 OSMO_ASSERT(0 <= gprs_match_tlv(&data, &tmp_data_len,
233 tag, &value, &value_len));
234 OSMO_ASSERT(value != unchanged_ptr);
235 OSMO_ASSERT(value_len != unchanged_len);
236 }
237}
238
239static void check_tv_fixed_match_data_len(size_t data_len,
240 uint8_t tag, size_t len,
241 const uint8_t *test_data)
242{
243 uint8_t buf[300] = {0};
244
245 uint8_t *unchanged_ptr = buf - 1;
246 size_t tmp_data_len = data_len;
247 uint8_t *value = unchanged_ptr;
248 uint8_t *data = buf;
249
250 OSMO_ASSERT(data_len <= sizeof(buf));
251
252 tv_fixed_put(data, tag, len, test_data);
253
254 if (data_len < len + 1) {
255 OSMO_ASSERT(-1 == gprs_match_tv_fixed(&data, &tmp_data_len,
256 tag, len, &value));
257 OSMO_ASSERT(tmp_data_len == 0);
258 OSMO_ASSERT(data == buf + data_len);
259 OSMO_ASSERT(value == unchanged_ptr);
260 } else {
261 OSMO_ASSERT(0 <= gprs_match_tv_fixed(&data, &tmp_data_len,
262 tag, len, &value));
263 OSMO_ASSERT(value != unchanged_ptr);
264 }
265}
266
267static void check_v_fixed_shift_data_len(size_t data_len,
268 size_t len, const uint8_t *test_data)
269{
270 uint8_t buf[300] = {0};
271
272 uint8_t *unchanged_ptr = buf - 1;
273 size_t tmp_data_len = data_len;
274 uint8_t *value = unchanged_ptr;
275 uint8_t *data = buf;
276
277 OSMO_ASSERT(data_len <= sizeof(buf));
278
279 memcpy(data, test_data, len);
280
281 if (data_len < len) {
282 OSMO_ASSERT(-1 == gprs_shift_v_fixed(&data, &tmp_data_len,
283 len, &value));
284 OSMO_ASSERT(tmp_data_len == 0);
285 OSMO_ASSERT(data == buf + data_len);
286 OSMO_ASSERT(value == unchanged_ptr);
287 } else {
288 OSMO_ASSERT(0 <= gprs_shift_v_fixed(&data, &tmp_data_len,
289 len, &value));
290 OSMO_ASSERT(value != unchanged_ptr);
291 }
292}
293
294static void check_lv_shift_data_len(size_t data_len,
295 size_t len, const uint8_t *test_data)
296{
297 uint8_t buf[300] = {0};
298
299 uint8_t *unchanged_ptr = buf - 1;
300 size_t unchanged_len = 0xdead;
301 size_t tmp_data_len = data_len;
302 uint8_t *value = unchanged_ptr;
303 size_t value_len = unchanged_len;
304 uint8_t *data = buf;
305
306 lv_put(data, len, test_data);
307 if (data_len < len + 1) {
308 OSMO_ASSERT(-1 == gprs_shift_lv(&data, &tmp_data_len,
309 &value, &value_len));
310 OSMO_ASSERT(tmp_data_len == 0);
311 OSMO_ASSERT(data == buf + data_len);
312 OSMO_ASSERT(value == unchanged_ptr);
313 OSMO_ASSERT(value_len == unchanged_len);
314 } else {
315 OSMO_ASSERT(0 <= gprs_shift_lv(&data, &tmp_data_len,
316 &value, &value_len));
317 OSMO_ASSERT(value != unchanged_ptr);
318 OSMO_ASSERT(value_len != unchanged_len);
319 }
320}
321
322static void test_tlv_shift_functions()
323{
324 uint8_t test_data[1024];
325 uint8_t buf[1024];
326 uint8_t *data_end;
327 unsigned i, len;
328 uint8_t *data;
329 size_t data_len;
330 const uint8_t tag = 0x1a;
331
332 printf("Test shift functions\n");
333
334 for (i = 0; i < ARRAY_SIZE(test_data); i++)
335 test_data[i] = (uint8_t)i;
336
337 for (len = 0; len < 256; len++) {
338 const unsigned iterations = sizeof(buf) / (len + 2) / 4;
339
340 memset(buf, 0xee, sizeof(buf));
341 data_end = data = buf;
342
343 for (i = 0; i < iterations; i++) {
344 data_end = tlv_put(data_end, tag, len, test_data);
345 data_end = tv_fixed_put(data_end, tag, len, test_data);
346 /* v_fixed_put */
347 memcpy(data_end, test_data, len);
348 data_end += len;
349 data_end = lv_put(data_end, len, test_data);
350 }
351
352 data_len = data_end - data;
353 OSMO_ASSERT(data_len <= sizeof(buf));
354
355 for (i = 0; i < iterations; i++) {
Jacob Erlbeck697a5342014-12-11 12:05:29 +0100356 check_tlv_parse(&data, &data_len, tag, len, test_data);
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100357 check_tv_fixed_match(&data, &data_len, tag, len, test_data);
358 check_v_fixed_shift(&data, &data_len, len, test_data);
359 check_lv_shift(&data, &data_len, len, test_data);
360 }
361
362 OSMO_ASSERT(data == data_end);
363
364 /* Test at end of data */
365
366 OSMO_ASSERT(-1 == gprs_match_tlv(&data, &data_len, tag, NULL, NULL));
367 OSMO_ASSERT(-1 == gprs_match_tv_fixed(&data, &data_len, tag, len, NULL));
368 OSMO_ASSERT((len ? -1 : 0) == gprs_shift_v_fixed(&data, &data_len, len, NULL));
369 OSMO_ASSERT(-1 == gprs_shift_lv(&data, &data_len, NULL, NULL));
370
371 /* Test invalid data_len */
372 for (data_len = 0; data_len <= len + 2 + 1; data_len += 1) {
373 check_tlv_match_data_len(data_len, tag, len, test_data);
374 check_tv_fixed_match_data_len(data_len, tag, len, test_data);
375 check_v_fixed_shift_data_len(data_len, len, test_data);
376 check_lv_shift_data_len(data_len, len, test_data);
377 }
378 }
379}
380
Holger Hans Peter Freytherfaf1f642011-06-23 17:53:27 -0400381int main(int argc, char **argv)
382{
383 test_8_4_2();
Holger Hans Peter Freytherce1b22e2014-08-04 14:22:13 +0200384 test_gsm_03_03_apn();
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100385 test_tlv_shift_functions();
Holger Hans Peter Freytherfaf1f642011-06-23 17:53:27 -0400386
387 printf("Done.\n");
388 return EXIT_SUCCESS;
389}