blob: 5bc2073435a886d8646be46a3bebe74cb33f928a [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 */
142static void check_tlv_match(uint8_t **data, size_t *data_len,
143 uint8_t tag, size_t exp_len, const uint8_t *exp_val)
144{
145 uint8_t *value;
146 size_t value_len;
147 int rc;
148
149 rc = gprs_match_tlv(data, data_len, tag ^ 1, NULL, NULL);
150 OSMO_ASSERT(rc == 0);
151
152 rc = gprs_match_tlv(data, data_len, tag, &value, &value_len);
153 OSMO_ASSERT(rc == (int)value_len + 2);
154 OSMO_ASSERT(value_len == exp_len);
155 OSMO_ASSERT(memcmp(value, exp_val, exp_len) == 0);
156}
157
158static void check_tv_fixed_match(uint8_t **data, size_t *data_len,
159 uint8_t tag, size_t len, const uint8_t *exp_val)
160{
161 uint8_t *value;
162 int rc;
163
164 rc = gprs_match_tv_fixed(data, data_len, tag ^ 1, len, NULL);
165 OSMO_ASSERT(rc == 0);
166
167 rc = gprs_match_tv_fixed(data, data_len, tag, len, &value);
168 OSMO_ASSERT(rc == (int)len + 1);
169 OSMO_ASSERT(memcmp(value, exp_val, len) == 0);
170}
171
172static void check_v_fixed_shift(uint8_t **data, size_t *data_len,
173 size_t len, const uint8_t *exp_val)
174{
175 uint8_t *value;
176 int rc;
177
178 rc = gprs_shift_v_fixed(data, data_len, len, &value);
179 OSMO_ASSERT(rc == (int)len);
180 OSMO_ASSERT(memcmp(value, exp_val, len) == 0);
181}
182
183static void check_lv_shift(uint8_t **data, size_t *data_len,
184 size_t exp_len, const uint8_t *exp_val)
185{
186 uint8_t *value;
187 size_t value_len;
188 int rc;
189
190 rc = gprs_shift_lv(data, data_len, &value, &value_len);
191 OSMO_ASSERT(rc == (int)value_len + 1);
192 OSMO_ASSERT(value_len == exp_len);
193 OSMO_ASSERT(memcmp(value, exp_val, exp_len) == 0);
194}
195
196static void check_tlv_match_data_len(size_t data_len, uint8_t tag, size_t len,
197 const uint8_t *test_data)
198{
199 uint8_t buf[300] = {0};
200
201 uint8_t *unchanged_ptr = buf - 1;
202 size_t unchanged_len = 0xdead;
203 size_t tmp_data_len = data_len;
204 uint8_t *value = unchanged_ptr;
205 size_t value_len = unchanged_len;
206 uint8_t *data = buf;
207
208 OSMO_ASSERT(data_len <= sizeof(buf));
209
210 tlv_put(data, tag, len, test_data);
211 if (data_len < len + 2) {
212 OSMO_ASSERT(-1 == gprs_match_tlv(&data, &tmp_data_len,
213 tag, &value, &value_len));
214 OSMO_ASSERT(tmp_data_len == 0);
215 OSMO_ASSERT(data == buf + data_len);
216 OSMO_ASSERT(value == unchanged_ptr);
217 OSMO_ASSERT(value_len == unchanged_len);
218 } else {
219 OSMO_ASSERT(0 <= gprs_match_tlv(&data, &tmp_data_len,
220 tag, &value, &value_len));
221 OSMO_ASSERT(value != unchanged_ptr);
222 OSMO_ASSERT(value_len != unchanged_len);
223 }
224}
225
226static void check_tv_fixed_match_data_len(size_t data_len,
227 uint8_t tag, size_t len,
228 const uint8_t *test_data)
229{
230 uint8_t buf[300] = {0};
231
232 uint8_t *unchanged_ptr = buf - 1;
233 size_t tmp_data_len = data_len;
234 uint8_t *value = unchanged_ptr;
235 uint8_t *data = buf;
236
237 OSMO_ASSERT(data_len <= sizeof(buf));
238
239 tv_fixed_put(data, tag, len, test_data);
240
241 if (data_len < len + 1) {
242 OSMO_ASSERT(-1 == gprs_match_tv_fixed(&data, &tmp_data_len,
243 tag, len, &value));
244 OSMO_ASSERT(tmp_data_len == 0);
245 OSMO_ASSERT(data == buf + data_len);
246 OSMO_ASSERT(value == unchanged_ptr);
247 } else {
248 OSMO_ASSERT(0 <= gprs_match_tv_fixed(&data, &tmp_data_len,
249 tag, len, &value));
250 OSMO_ASSERT(value != unchanged_ptr);
251 }
252}
253
254static void check_v_fixed_shift_data_len(size_t data_len,
255 size_t len, const uint8_t *test_data)
256{
257 uint8_t buf[300] = {0};
258
259 uint8_t *unchanged_ptr = buf - 1;
260 size_t tmp_data_len = data_len;
261 uint8_t *value = unchanged_ptr;
262 uint8_t *data = buf;
263
264 OSMO_ASSERT(data_len <= sizeof(buf));
265
266 memcpy(data, test_data, len);
267
268 if (data_len < len) {
269 OSMO_ASSERT(-1 == gprs_shift_v_fixed(&data, &tmp_data_len,
270 len, &value));
271 OSMO_ASSERT(tmp_data_len == 0);
272 OSMO_ASSERT(data == buf + data_len);
273 OSMO_ASSERT(value == unchanged_ptr);
274 } else {
275 OSMO_ASSERT(0 <= gprs_shift_v_fixed(&data, &tmp_data_len,
276 len, &value));
277 OSMO_ASSERT(value != unchanged_ptr);
278 }
279}
280
281static void check_lv_shift_data_len(size_t data_len,
282 size_t len, const uint8_t *test_data)
283{
284 uint8_t buf[300] = {0};
285
286 uint8_t *unchanged_ptr = buf - 1;
287 size_t unchanged_len = 0xdead;
288 size_t tmp_data_len = data_len;
289 uint8_t *value = unchanged_ptr;
290 size_t value_len = unchanged_len;
291 uint8_t *data = buf;
292
293 lv_put(data, len, test_data);
294 if (data_len < len + 1) {
295 OSMO_ASSERT(-1 == gprs_shift_lv(&data, &tmp_data_len,
296 &value, &value_len));
297 OSMO_ASSERT(tmp_data_len == 0);
298 OSMO_ASSERT(data == buf + data_len);
299 OSMO_ASSERT(value == unchanged_ptr);
300 OSMO_ASSERT(value_len == unchanged_len);
301 } else {
302 OSMO_ASSERT(0 <= gprs_shift_lv(&data, &tmp_data_len,
303 &value, &value_len));
304 OSMO_ASSERT(value != unchanged_ptr);
305 OSMO_ASSERT(value_len != unchanged_len);
306 }
307}
308
309static void test_tlv_shift_functions()
310{
311 uint8_t test_data[1024];
312 uint8_t buf[1024];
313 uint8_t *data_end;
314 unsigned i, len;
315 uint8_t *data;
316 size_t data_len;
317 const uint8_t tag = 0x1a;
318
319 printf("Test shift functions\n");
320
321 for (i = 0; i < ARRAY_SIZE(test_data); i++)
322 test_data[i] = (uint8_t)i;
323
324 for (len = 0; len < 256; len++) {
325 const unsigned iterations = sizeof(buf) / (len + 2) / 4;
326
327 memset(buf, 0xee, sizeof(buf));
328 data_end = data = buf;
329
330 for (i = 0; i < iterations; i++) {
331 data_end = tlv_put(data_end, tag, len, test_data);
332 data_end = tv_fixed_put(data_end, tag, len, test_data);
333 /* v_fixed_put */
334 memcpy(data_end, test_data, len);
335 data_end += len;
336 data_end = lv_put(data_end, len, test_data);
337 }
338
339 data_len = data_end - data;
340 OSMO_ASSERT(data_len <= sizeof(buf));
341
342 for (i = 0; i < iterations; i++) {
343 check_tlv_match(&data, &data_len, tag, len, test_data);
344 check_tv_fixed_match(&data, &data_len, tag, len, test_data);
345 check_v_fixed_shift(&data, &data_len, len, test_data);
346 check_lv_shift(&data, &data_len, len, test_data);
347 }
348
349 OSMO_ASSERT(data == data_end);
350
351 /* Test at end of data */
352
353 OSMO_ASSERT(-1 == gprs_match_tlv(&data, &data_len, tag, NULL, NULL));
354 OSMO_ASSERT(-1 == gprs_match_tv_fixed(&data, &data_len, tag, len, NULL));
355 OSMO_ASSERT((len ? -1 : 0) == gprs_shift_v_fixed(&data, &data_len, len, NULL));
356 OSMO_ASSERT(-1 == gprs_shift_lv(&data, &data_len, NULL, NULL));
357
358 /* Test invalid data_len */
359 for (data_len = 0; data_len <= len + 2 + 1; data_len += 1) {
360 check_tlv_match_data_len(data_len, tag, len, test_data);
361 check_tv_fixed_match_data_len(data_len, tag, len, test_data);
362 check_v_fixed_shift_data_len(data_len, len, test_data);
363 check_lv_shift_data_len(data_len, len, test_data);
364 }
365 }
366}
367
Holger Hans Peter Freytherfaf1f642011-06-23 17:53:27 -0400368int main(int argc, char **argv)
369{
370 test_8_4_2();
Holger Hans Peter Freytherce1b22e2014-08-04 14:22:13 +0200371 test_gsm_03_03_apn();
Jacob Erlbeckdcfd4562014-12-11 11:01:46 +0100372 test_tlv_shift_functions();
Holger Hans Peter Freytherfaf1f642011-06-23 17:53:27 -0400373
374 printf("Done.\n");
375 return EXIT_SUCCESS;
376}