// SPDX-License-Identifier: BSD-2-Clause-Patent /* * test-argv.c - test our loader_opts parsing */ #pragma GCC diagnostic ignored "-Wunused-variable" #pragma GCC diagnostic error "-Wnonnull" #pragma GCC diagnostic error "-Wunused-function" #pragma GCC diagnostic warning "-Wcpp" #ifndef SHIM_UNIT_TEST #define SHIM_UNIT_TEST #endif #include "shim.h" #include #include #include EFI_DEVICE_PATH * make_file_dp(char *filename) { void *filedp = NULL; size_t filedpsz = 0, filedpneeded = 0; filedpneeded = efidp_make_file(filedp, filedpneeded, filename); assert_positive_return(filedpneeded, NULL, "\n"); filedp = calloc(1, filedpneeded + 4); assert_nonzero_return(filedp, NULL, "\n"); filedpsz = efidp_make_file(filedp, filedpneeded, filename); assert_equal_goto(filedpsz, filedpneeded, err, "got %zu expected %zu\n"); efidp_make_end_entire((uint8_t *)filedp + filedpneeded, 4); return filedp; err: free(filedp); return NULL; } int test_parse_load_options(char *load_option_data, size_t load_option_data_size, char *file_path, CHAR16 *target_second_stage, char *target_remaining, size_t target_remaining_size) { EFI_STATUS status = EFI_SUCCESS; EFI_LOADED_IMAGE li = { .LoadOptions = load_option_data, .LoadOptionsSize = load_option_data_size, .FilePath = make_file_dp(file_path), }; CHAR16 *dummy_second_stage = calloc(1, 8); int rc = -1; assert_nonzero_goto(li.FilePath, err, "\n"); assert_nonzero_goto(dummy_second_stage, err, "\n"); StrCat(dummy_second_stage, L"foo"); second_stage = dummy_second_stage; status = parse_load_options(&li); assert_false_goto(EFI_ERROR(status), err, "\n"); assert_nonzero_goto(second_stage, err, "\n"); if (target_second_stage) { assert_not_equal_goto(second_stage, dummy_second_stage, err, "%p == %p\n"); assert_zero_goto(StrnCmp(second_stage, target_second_stage, 90), err_print_second_stage, "%d != 0\n"); } else assert_equal_goto(second_stage, dummy_second_stage, err, "%p != %p\n"); assert_equal_goto(load_options_size, target_remaining_size, err_remaining, "%zu != %zu\n"); assert_equal_goto(load_options, target_remaining, err_remaining, "%p != %p\n"); assert_zero_goto(memcmp(load_options, target_remaining, load_options_size), err_remaining, "\n"); rc = 0; err_remaining: if (rc != 0) { printf("expected remaining:%p\n", target_remaining); for (size_t i = 0; i < target_remaining_size; i++) printf("0x%02hhx ", target_remaining[i]); printf("\n"); printf("actual remaining:%p\n", load_options); for (size_t i = 0; i < load_options_size; i++) printf("0x%02hhx ", ((char *)load_options)[i]); printf("\n"); } err_print_second_stage: if (rc != 0) { printf("second stage:\""); for(int i = 0; second_stage[i] != 0; i++) printf("%c", second_stage[i]); printf("\"\nexpected:\""); for(int i = 0; target_second_stage[i] != 0; i++) printf("%c", target_second_stage[i]); printf("\"\n"); } err: if (rc != 0) { print_traceback(0); } if (li.FilePath) { free(li.FilePath); } if (dummy_second_stage && dummy_second_stage != second_stage) { free(dummy_second_stage); } second_stage = NULL; return rc; } int __attribute__((__flatten__)) test_efi_shell_0(void) { /* 00000000 5c 00 45 00 46 00 49 00 5c 00 66 00 65 00 64 00 |\.E.F.I.\.f.e.d.| 00000010 6f 00 72 00 61 00 5c 00 73 00 68 00 69 00 6d 00 |o.r.a.\.s.h.i.m.| 00000020 78 00 36 00 34 00 2e 00 65 00 66 00 69 00 20 00 |x.6.4...e.f.i. .| 00000030 5c 00 45 00 46 00 49 00 5c 00 66 00 65 00 64 00 |\.E.F.I.\.f.e.d.| 00000040 6f 00 72 00 61 00 5c 00 66 00 77 00 75 00 70 00 |o.r.a.\.f.w.u.p.| 00000050 64 00 61 00 74 00 65 00 2e 00 65 00 66 00 69 00 |d.a.t.e...e.f.i.| 00000060 20 00 00 00 66 00 73 00 30 00 3a 00 5c 00 00 00 | ...f.s.0.:.\...| */ char load_option_data[] = { 0x5c, 0x00, 0x45, 0x00, 0x46, 0x00, 0x49, 0x00, 0x5c, 0x00, 0x66, 0x00, 0x65, 0x00, 0x64, 0x00, 0x6f, 0x00, 0x72, 0x00, 0x61, 0x00, 0x5c, 0x00, 0x73, 0x00, 0x68, 0x00, 0x69, 0x00, 0x6d, 0x00, 0x78, 0x00, 0x36, 0x00, 0x34, 0x00, 0x2e, 0x00, 0x65, 0x00, 0x66, 0x00, 0x69, 0x00, 0x20, 0x00, 0x5c, 0x00, 0x45, 0x00, 0x46, 0x00, 0x49, 0x00, 0x5c, 0x00, 0x66, 0x00, 0x65, 0x00, 0x64, 0x00, 0x6f, 0x00, 0x72, 0x00, 0x61, 0x00, 0x5c, 0x00, 0x66, 0x00, 0x77, 0x00, 0x75, 0x00, 0x70, 0x00, 0x64, 0x00, 0x61, 0x00, 0x74, 0x00, 0x65, 0x00, 0x2e, 0x00, 0x65, 0x00, 0x66, 0x00, 0x69, 0x00, 0x20, 0x00, 0x00, 0x00, 0x66, 0x00, 0x73, 0x00, 0x30, 0x00, 0x3a, 0x00, 0x5c, 0x00, 0x00, 0x00, }; size_t load_option_data_size = sizeof(load_option_data); char *remaining = &load_option_data[sizeof(load_option_data)-14]; size_t remaining_size = 14; return test_parse_load_options(load_option_data, load_option_data_size, "\\EFI\\fedora\\shimx64.efi", L"\\EFI\\fedora\\fwupdate.efi", remaining, remaining_size); } int __attribute__((__flatten__)) test_bds_0(void) { /* 00000000 01 00 00 00 62 00 4c 00 69 00 6e 00 75 00 78 00 |....b.L.i.n.u.x.| 00000010 20 00 46 00 69 00 72 00 6d 00 77 00 61 00 72 00 | .F.i.r.m.w.a.r.| 00000020 65 00 20 00 55 00 70 00 64 00 61 00 74 00 65 00 |e. .U.p.d.a.t.e.| 00000030 72 00 00 00 40 01 2a 00 01 00 00 00 00 08 00 00 |r.....*.........| 00000040 00 00 00 00 00 40 06 00 00 00 00 00 1a 9e 55 bf |.....@........U.| 00000050 04 57 f2 4f b4 4a ed 26 4a 40 6a 94 02 02 04 04 |.W.O.:.&J@j.....| 00000060 34 00 5c 00 45 00 46 00 49 00 5c 00 66 00 65 00 |4.\.E.F.I.\.f.e.| 00000070 64 00 6f 00 72 00 61 00 5c 00 73 00 68 00 69 00 |d.o.r.a.\.s.h.i.| 00000080 6d 00 78 00 36 00 34 00 2e 00 65 00 66 00 69 00 |m.x.6.4...e.f.i.| 00000090 00 00 7f ff 04 00 20 00 5c 00 66 00 77 00 75 00 |...... .\.f.w.u.| 000000a0 70 00 78 00 36 00 34 00 2e 00 65 00 66 00 69 00 |p.x.6.4...e.f.i.| 000000b0 00 00 |..| */ char load_option_data [] = { 0x01, 0x00, 0x00, 0x00, 0x62, 0x00, 0x4c, 0x00, 0x69, 0x00, 0x6e, 0x00, 0x75, 0x00, 0x78, 0x00, 0x20, 0x00, 0x46, 0x00, 0x69, 0x00, 0x72, 0x00, 0x6d, 0x00, 0x77, 0x00, 0x61, 0x00, 0x72, 0x00, 0x65, 0x00, 0x20, 0x00, 0x55, 0x00, 0x70, 0x00, 0x64, 0x00, 0x61, 0x00, 0x74, 0x00, 0x65, 0x00, 0x72, 0x00, 0x00, 0x00, 0x40, 0x01, 0x2a, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1a, 0x9e, 0x55, 0xbf, 0x04, 0x57, 0xf2, 0x4f, 0xb4, 0x4a, 0xed, 0x26, 0x4a, 0x40, 0x6a, 0x94, 0x02, 0x02, 0x04, 0x04, 0x34, 0x00, 0x5c, 0x00, 0x45, 0x00, 0x46, 0x00, 0x49, 0x00, 0x5c, 0x00, 0x66, 0x00, 0x65, 0x00, 0x64, 0x00, 0x6f, 0x00, 0x72, 0x00, 0x61, 0x00, 0x5c, 0x00, 0x73, 0x00, 0x68, 0x00, 0x69, 0x00, 0x6d, 0x00, 0x78, 0x00, 0x36, 0x00, 0x34, 0x00, 0x2e, 0x00, 0x65, 0x00, 0x66, 0x00, 0x69, 0x00, 0x00, 0x00, 0x7f, 0xff, 0x04, 0x00, 0x20, 0x00, 0x5c, 0x00, 0x66, 0x00, 0x77, 0x00, 0x75, 0x00, 0x70, 0x00, 0x78, 0x00, 0x36, 0x00, 0x34, 0x00, 0x2e, 0x00, 0x65, 0x00, 0x66, 0x00, 0x69, 0x00, 0x00, 0x00 }; size_t load_option_data_size = sizeof(load_option_data); return test_parse_load_options(load_option_data, load_option_data_size, "\\EFI\\fedora\\shimx64.efi", L"\\fwupx64.efi", NULL, 0); } int __attribute__((__flatten__)) test_bds_1(void) { /* 00000000 5c 00 66 00 77 00 75 00 70 00 78 00 36 00 34 00 |\.f.w.u.p.x.6.4.| 00000010 2e 00 65 00 66 00 69 00 00 00 |..e.f.i...| 0000001a */ char load_option_data [] = { 0x5c, 0x00, 0x66, 0x00, 0x77, 0x00, 0x75, 0x00, 0x70, 0x00, 0x78, 0x00, 0x36, 0x00, 0x34, 0x00, 0x2e, 0x00, 0x65, 0x00, 0x66, 0x00, 0x69, 0x00, 0x00, 0x00 }; size_t load_option_data_size = sizeof(load_option_data); return test_parse_load_options(load_option_data, load_option_data_size, "\\EFI\\fedora\\shimx64.efi", L"\\fwupx64.efi", NULL, 0); } int test_bds_2(void) { /* 00000000 74 00 65 00 73 00 74 00 2E 00 65 00 66 00 69 00 |t.e.s.t...e.f.i.| 00000010 20 00 6F 00 6E 00 65 00 20 00 74 00 77 00 6F 00 |..o.n.e...t.w.o.| 00000020 20 00 74 00 68 00 72 00 65 00 65 00 00 00 |..t.h.r.e.e...| */ char load_option_data [] = { 0x74, 0x00, 0x65, 0x00, 0x73, 0x00, 0x74, 0x00, 0x2E, 0x00, 0x65, 0x00, 0x66, 0x00, 0x69, 0x00, 0x20, 0x00, 0x6F, 0x00, 0x6E, 0x00, 0x65, 0x00, 0x20, 0x00, 0x74, 0x00, 0x77, 0x00, 0x6F, 0x00, 0x20, 0x00, 0x74, 0x00, 0x68, 0x00, 0x72, 0x00, 0x65, 0x00, 0x65, 0x00, 0x00, 0x00 }; size_t load_option_data_size = sizeof(load_option_data); char *target_remaining = &load_option_data[26]; size_t target_remaining_size = 20; return test_parse_load_options(load_option_data, load_option_data_size, "test.efi", L"one", target_remaining, target_remaining_size); } int test_multi_end_dp(void) { /* 00000000 01 00 00 00 d0 00 4f 00 6e 00 62 00 6f 00 61 00 |......O.n.b.o.a.| 00000010 72 00 64 00 20 00 4e 00 49 00 43 00 28 00 49 00 |r.d. .N.I.C.(.I.| 00000020 50 00 56 00 34 00 29 00 00 00 02 01 0c 00 d0 41 |P.V.4.)........A| 00000030 03 0a 00 00 00 00 01 01 06 00 06 1f 03 0b 25 00 |..............%.| 00000040 2c ea 7f 0a 9f 69 00 00 00 00 00 00 00 00 00 00 |,....i..........| 00000050 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 |................| 00000060 00 03 0c 1b 00 00 00 00 00 00 00 00 00 00 00 00 |................| 00000070 00 00 00 00 00 00 00 00 00 00 00 00 7f ff 04 00 |................| 00000080 01 04 76 00 ef 47 64 2d c9 3b a0 41 ac 19 4d 51 |..v..Gd-.;.A..MQ| 00000090 d0 1b 4c e6 50 00 58 00 45 00 20 00 49 00 50 00 |..L.P.X.E. .I.P.| 000000a0 34 00 20 00 49 00 6e 00 74 00 65 00 6c 00 28 00 |4. .I.n.t.e.l.(.| 000000b0 52 00 29 00 20 00 45 00 74 00 68 00 65 00 72 00 |R.). .E.t.h.e.r.| 000000c0 6e 00 65 00 74 00 20 00 43 00 6f 00 6e 00 6e 00 |n.e.t. .C.o.n.n.| 000000d0 65 00 63 00 74 00 69 00 6f 00 6e 00 20 00 28 00 |e.c.t.i.o.n. .(.| 000000e0 36 00 29 00 20 00 49 00 32 00 31 00 39 00 2d 00 |6.). .I.2.1.9.-.| 000000f0 4c 00 4d 00 00 00 7f ff 04 00 00 00 42 4f |L.M.........BO| */ char load_option_data [] = { 0x01, 0x00, 0x00, 0x00, 0xd0, 0x00, 0x4f, 0x00, 0x6e, 0x00, 0x62, 0x00, 0x6f, 0x00, 0x61, 0x00, 0x72, 0x00, 0x64, 0x00, 0x20, 0x00, 0x4e, 0x00, 0x49, 0x00, 0x43, 0x00, 0x28, 0x00, 0x49, 0x00, 0x50, 0x00, 0x56, 0x00, 0x34, 0x00, 0x29, 0x00, 0x00, 0x00, 0x02, 0x01, 0x0c, 0x00, 0xd0, 0x41, 0x03, 0x0a, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x06, 0x00, 0x06, 0x1f, 0x03, 0x0b, 0x25, 0x00, 0x2c, 0xea, 0x7f, 0x0a, 0x9f, 0x69, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0x0c, 0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7f, 0xff, 0x04, 0x00, 0x01, 0x04, 0x76, 0x00, 0xef, 0x47, 0x64, 0x2d, 0xc9, 0x3b, 0xa0, 0x41, 0xac, 0x19, 0x4d, 0x51, 0xd0, 0x1b, 0x4c, 0xe6, 0x50, 0x00, 0x58, 0x00, 0x45, 0x00, 0x20, 0x00, 0x49, 0x00, 0x50, 0x00, 0x34, 0x00, 0x20, 0x00, 0x49, 0x00, 0x6e, 0x00, 0x74, 0x00, 0x65, 0x00, 0x6c, 0x00, 0x28, 0x00, 0x52, 0x00, 0x29, 0x00, 0x20, 0x00, 0x45, 0x00, 0x74, 0x00, 0x68, 0x00, 0x65, 0x00, 0x72, 0x00, 0x6e, 0x00, 0x65, 0x00, 0x74, 0x00, 0x20, 0x00, 0x43, 0x00, 0x6f, 0x00, 0x6e, 0x00, 0x6e, 0x00, 0x65, 0x00, 0x63, 0x00, 0x74, 0x00, 0x69, 0x00, 0x6f, 0x00, 0x6e, 0x00, 0x20, 0x00, 0x28, 0x00, 0x36, 0x00, 0x29, 0x00, 0x20, 0x00, 0x49, 0x00, 0x32, 0x00, 0x31, 0x00, 0x39, 0x00, 0x2d, 0x00, 0x4c, 0x00, 0x4d, 0x00, 0x00, 0x00, 0x7f, 0xff, 0x04, 0x00, 0x00, 0x00, 0x42, 0x4f }; size_t load_option_data_size = sizeof(load_option_data); char *target_remaining = &load_option_data[load_option_data_size - 2]; size_t target_remaining_size = 2; return test_parse_load_options(load_option_data, load_option_data_size, "test.efi", NULL, target_remaining, target_remaining_size); } int main(void) { int status = 0; test(test_efi_shell_0); test(test_bds_0); test(test_bds_1); test(test_bds_2); test(test_multi_end_dp); return status; } // vim:fenc=utf-8:tw=75:noet