/* SPDX-License-Identifier: LGPL-2.1+ */ #include "alloc-util.h" //#include "env-util.h" #include "log.h" #include "macro.h" #include "proc-cmdline.h" //#include "special.h" #include "string-util.h" #include "util.h" static int obj; static int parse_item(const char *key, const char *value, void *data) { assert_se(key); assert_se(data == &obj); log_info("kernel cmdline option <%s> = <%s>", key, strna(value)); return 0; } static void test_proc_cmdline_parse(void) { log_info("/* %s */", __func__); assert_se(proc_cmdline_parse(parse_item, &obj, PROC_CMDLINE_STRIP_RD_PREFIX) >= 0); } #if 0 /// UNNEEDED by elogind static void test_proc_cmdline_override(void) { log_info("/* %s */", __func__); assert_se(putenv((char*) "SYSTEMD_PROC_CMDLINE=foo_bar=quux wuff-piep=tuet zumm") == 0); /* Test if the override works */ _cleanup_free_ char *line = NULL, *value = NULL; assert_se(proc_cmdline(&line) >= 0); /* Test if parsing makes uses of the override */ assert_se(streq(line, "foo_bar=quux wuff-piep=tuet zumm")); assert_se(proc_cmdline_get_key("foo_bar", 0, &value) > 0 && streq_ptr(value, "quux")); } static int parse_item_given(const char *key, const char *value, void *data) { assert_se(key); assert_se(data); bool *strip = data; log_info("%s: option <%s> = <%s>", __func__, key, strna(value)); if (streq(key, "foo_bar")) assert_se(streq(value, "quux")); else if (streq(key, "wuff-piep")) assert_se(streq(value, "tuet ")); else if (in_initrd() && *strip && streq(key, "zumm")) assert_se(!value); else if (in_initrd() && !*strip && streq(key, "rd.zumm")) assert_se(!value); else assert_not_reached("Bad key!"); return 0; } static void test_proc_cmdline_given(bool flip_initrd) { log_info("/* %s (flip: %s) */", __func__, yes_no(flip_initrd)); if (flip_initrd) in_initrd_force(!in_initrd()); bool t = true, f = false; assert_se(proc_cmdline_parse_given("foo_bar=quux wuff-piep=\"tuet \" rd.zumm", parse_item_given, &t, PROC_CMDLINE_STRIP_RD_PREFIX) >= 0); assert_se(proc_cmdline_parse_given("foo_bar=quux wuff-piep=\"tuet \" rd.zumm", parse_item_given, &f, 0) >= 0); if (flip_initrd) in_initrd_force(!in_initrd()); } static void test_proc_cmdline_get_key(void) { _cleanup_free_ char *value = NULL; log_info("/* %s */", __func__); putenv((char*) "SYSTEMD_PROC_CMDLINE=foo_bar=quux wuff-piep=tuet zumm"); assert_se(proc_cmdline_get_key("", 0, &value) == -EINVAL); assert_se(proc_cmdline_get_key("abc", 0, NULL) == 0); assert_se(proc_cmdline_get_key("abc", 0, &value) == 0 && value == NULL); assert_se(proc_cmdline_get_key("abc", PROC_CMDLINE_VALUE_OPTIONAL, &value) == 0 && value == NULL); assert_se(proc_cmdline_get_key("foo_bar", 0, &value) > 0 && streq_ptr(value, "quux")); value = mfree(value); assert_se(proc_cmdline_get_key("foo_bar", PROC_CMDLINE_VALUE_OPTIONAL, &value) > 0 && streq_ptr(value, "quux")); value = mfree(value); assert_se(proc_cmdline_get_key("foo-bar", 0, &value) > 0 && streq_ptr(value, "quux")); value = mfree(value); assert_se(proc_cmdline_get_key("foo-bar", PROC_CMDLINE_VALUE_OPTIONAL, &value) > 0 && streq_ptr(value, "quux")); value = mfree(value); assert_se(proc_cmdline_get_key("foo-bar", 0, NULL) == 0); assert_se(proc_cmdline_get_key("foo-bar", PROC_CMDLINE_VALUE_OPTIONAL, NULL) == -EINVAL); assert_se(proc_cmdline_get_key("wuff-piep", 0, &value) > 0 && streq_ptr(value, "tuet")); value = mfree(value); assert_se(proc_cmdline_get_key("wuff-piep", PROC_CMDLINE_VALUE_OPTIONAL, &value) > 0 && streq_ptr(value, "tuet")); value = mfree(value); assert_se(proc_cmdline_get_key("wuff_piep", 0, &value) > 0 && streq_ptr(value, "tuet")); value = mfree(value); assert_se(proc_cmdline_get_key("wuff_piep", PROC_CMDLINE_VALUE_OPTIONAL, &value) > 0 && streq_ptr(value, "tuet")); value = mfree(value); assert_se(proc_cmdline_get_key("wuff_piep", 0, NULL) == 0); assert_se(proc_cmdline_get_key("wuff_piep", PROC_CMDLINE_VALUE_OPTIONAL, NULL) == -EINVAL); assert_se(proc_cmdline_get_key("zumm", 0, &value) == 0 && value == NULL); assert_se(proc_cmdline_get_key("zumm", PROC_CMDLINE_VALUE_OPTIONAL, &value) > 0 && value == NULL); assert_se(proc_cmdline_get_key("zumm", 0, NULL) > 0); } static void test_proc_cmdline_get_bool(void) { bool value = false; log_info("/* %s */", __func__); putenv((char*) "SYSTEMD_PROC_CMDLINE=foo_bar bar-waldo=1 x_y-z=0 quux=miep"); assert_se(proc_cmdline_get_bool("", &value) == -EINVAL); assert_se(proc_cmdline_get_bool("abc", &value) == 0 && value == false); assert_se(proc_cmdline_get_bool("foo_bar", &value) > 0 && value == true); assert_se(proc_cmdline_get_bool("foo-bar", &value) > 0 && value == true); assert_se(proc_cmdline_get_bool("bar-waldo", &value) > 0 && value == true); assert_se(proc_cmdline_get_bool("bar_waldo", &value) > 0 && value == true); assert_se(proc_cmdline_get_bool("x_y-z", &value) > 0 && value == false); assert_se(proc_cmdline_get_bool("x-y-z", &value) > 0 && value == false); assert_se(proc_cmdline_get_bool("x-y_z", &value) > 0 && value == false); assert_se(proc_cmdline_get_bool("x_y_z", &value) > 0 && value == false); assert_se(proc_cmdline_get_bool("quux", &value) == -EINVAL && value == false); } #endif // 0 static void test_proc_cmdline_key_streq(void) { log_info("/* %s */", __func__); assert_se(proc_cmdline_key_streq("", "")); assert_se(proc_cmdline_key_streq("a", "a")); assert_se(!proc_cmdline_key_streq("", "a")); assert_se(!proc_cmdline_key_streq("a", "")); assert_se(proc_cmdline_key_streq("a", "a")); assert_se(!proc_cmdline_key_streq("a", "b")); assert_se(proc_cmdline_key_streq("x-y-z", "x-y-z")); assert_se(proc_cmdline_key_streq("x-y-z", "x_y_z")); assert_se(proc_cmdline_key_streq("x-y-z", "x-y_z")); assert_se(proc_cmdline_key_streq("x-y-z", "x_y-z")); assert_se(proc_cmdline_key_streq("x_y-z", "x-y_z")); assert_se(!proc_cmdline_key_streq("x_y-z", "x-z_z")); } #if 0 /// UNNEEDED by elogind static void test_proc_cmdline_key_startswith(void) { log_info("/* %s */", __func__); assert_se(proc_cmdline_key_startswith("", "")); assert_se(proc_cmdline_key_startswith("x", "")); assert_se(!proc_cmdline_key_startswith("", "x")); assert_se(proc_cmdline_key_startswith("x", "x")); assert_se(!proc_cmdline_key_startswith("x", "y")); assert_se(!proc_cmdline_key_startswith("foo-bar", "quux")); assert_se(proc_cmdline_key_startswith("foo-bar", "foo")); assert_se(proc_cmdline_key_startswith("foo-bar", "foo-bar")); assert_se(proc_cmdline_key_startswith("foo-bar", "foo_bar")); assert_se(proc_cmdline_key_startswith("foo-bar", "foo_")); assert_se(!proc_cmdline_key_startswith("foo-bar", "foo_xx")); } static void test_runlevel_to_target(void) { log_info("/* %s */", __func__); in_initrd_force(false); assert_se(streq_ptr(runlevel_to_target(NULL), NULL)); assert_se(streq_ptr(runlevel_to_target("unknown-runlevel"), NULL)); assert_se(streq_ptr(runlevel_to_target("rd.unknown-runlevel"), NULL)); assert_se(streq_ptr(runlevel_to_target("3"), SPECIAL_MULTI_USER_TARGET)); assert_se(streq_ptr(runlevel_to_target("rd.rescue"), NULL)); in_initrd_force(true); assert_se(streq_ptr(runlevel_to_target(NULL), NULL)); assert_se(streq_ptr(runlevel_to_target("unknown-runlevel"), NULL)); assert_se(streq_ptr(runlevel_to_target("rd.unknown-runlevel"), NULL)); assert_se(streq_ptr(runlevel_to_target("3"), NULL)); assert_se(streq_ptr(runlevel_to_target("rd.rescue"), SPECIAL_RESCUE_TARGET)); } #endif // 0 int main(void) { log_parse_environment(); log_open(); test_proc_cmdline_parse(); #if 0 /// UNNEEDED by elogind test_proc_cmdline_override(); test_proc_cmdline_given(false); /* Repeat the same thing, but now flip our ininitrdness */ test_proc_cmdline_given(true); #endif // 0 test_proc_cmdline_key_streq(); #if 0 /// UNNEEDED by elogind test_proc_cmdline_key_startswith(); test_proc_cmdline_get_key(); test_proc_cmdline_get_bool(); test_runlevel_to_target(); #endif // 0 return 0; }