/* SPDX-License-Identifier: GPL-2.0-only */ #include #include #include #include #include static const uint8_t test_data_simple[] = { 0x64, 0x3b, 0x33, 0x17, 0x34, 0x74, 0x62, 0x30, 0x75, 0x73, 0xf3, 0x11, 0x30, 0x2c, 0x34, 0x35, 0x6d, 0x39, 0x69, 0x32, 0x23, 0x24, 0x76, 0x71, 0x77, 0x30, 0x39, 0x75, 0x76, 0x35, 0x71, 0x32, 0x40, 0x46, 0x34, 0x34, 0xBB, 0x03, 0x66, 0x52}; static const size_t test_data_simple_sz = ARRAY_SIZE(test_data_simple); static const uint16_t test_data_simple_checksum = 0x4267; static uint8_t test_data_zeros[1024]; static const size_t test_data_zeros_sz = ARRAY_SIZE(test_data_zeros); static const uint16_t test_data_zeros_checksum = 0xFFFF; static int setup_test_group(void **state) { memset(test_data_zeros, 0, test_data_zeros_sz); return 0; } static void test_ipchksum_zero_length(void **state) { uint16_t res = ipchksum(test_data_simple, 0); /* Expect checksum to be in initial state as there are were no data provided. */ assert_int_equal(0xFFFF, res); } static void test_ipchksum_zero_buffer(void **state) { uint16_t res = ipchksum(test_data_zeros, test_data_zeros_sz); assert_int_equal(test_data_zeros_checksum, res); } static void test_ipchksum_simple_data(void **state) { uint16_t res; uint16_t check_res; const size_t helper_buffer_size = sizeof(uint8_t) * (test_data_simple_sz + 2); char *helper_buffer = malloc(helper_buffer_size); /* Self test */ assert_non_null(helper_buffer); /* Expect function to generate the same checksum as stored in */ res = ipchksum(test_data_simple, test_data_simple_sz); assert_int_equal(test_data_simple_checksum, res); /* Copy test data and checksum to new buffer. Expect computed checksum to be zero, as it proves the data and the checksum are correct. */ memcpy(helper_buffer, test_data_simple, test_data_simple_sz); helper_buffer[helper_buffer_size - 2] = res & 0xFF; helper_buffer[helper_buffer_size - 1] = (res >> 8) & 0xFF; check_res = ipchksum(helper_buffer, helper_buffer_size); assert_int_equal(0, check_res); free(helper_buffer); } static void test_ipchksum_80kff(void **state) { /* 64K is an important boundary since naive 32-bit sum implementations that accumulate carries may run over after that point. */ size_t buffer_sz = 80 * 1024; char *buffer = malloc(buffer_sz); memset(buffer, 0xff, buffer_sz); assert_int_equal(ipchksum(buffer, buffer_sz), 0); /* Make things a bit more interesting... */ memcpy(buffer + 0x6789, test_data_simple, test_data_simple_sz); assert_int_equal(ipchksum(buffer, buffer_sz), 0x6742); free(buffer); } static void test_ipchksum_add_empty_values(void **state) { uint16_t res; res = ipchksum_add(0, 0xFFFF, 0xFFFF); assert_int_equal(0xFFFF, res); res = ipchksum_add(1, 0xFFFF, 0xFFFF); assert_int_equal(0xFFFF, res); } static void test_ipchksum_add(void **state) { uint16_t res_1 = ipchksum(test_data_simple, test_data_simple_sz / 2); uint16_t res_2 = ipchksum(test_data_simple + test_data_simple_sz / 2, test_data_simple_sz / 2); uint16_t res_sum = ipchksum_add(test_data_simple_sz / 2, res_1, res_2); assert_int_equal(0xb62e, res_1); assert_int_equal(0x8c38, res_2); assert_int_equal(test_data_simple_checksum, res_sum); /* Test some unaligned sums */ res_1 = ipchksum(test_data_simple, 17); res_2 = ipchksum(test_data_simple + 17, test_data_simple_sz - 17); res_sum = ipchksum_add(17, res_1, res_2); assert_int_equal(0x2198, res_1); assert_int_equal(0xcf20, res_2); assert_int_equal(test_data_simple_checksum, res_sum); } int main(void) { const struct CMUnitTest tests[] = { cmocka_unit_test(test_ipchksum_zero_length), cmocka_unit_test(test_ipchksum_zero_buffer), cmocka_unit_test(test_ipchksum_simple_data), cmocka_unit_test(test_ipchksum_80kff), cmocka_unit_test(test_ipchksum_add_empty_values), cmocka_unit_test(test_ipchksum_add), }; return cb_run_group_tests(tests, setup_test_group, NULL); }