/* SPDX-License-Identifier: GPL-2.0 */ /* * Copyright (c) 2024 Meta Platforms, Inc. and affiliates. * Copyright (c) 2024 David Vernet */ #include #include #include #include #include #include #include "hotplug_test.h" #include "hotplug.bpf.skel.h" #include "scx_test.h" #include "util.h" const char *online_path = "/sys/devices/system/cpu/cpu1/online"; static bool is_cpu_online(void) { return file_read_long(online_path) > 0; } static void toggle_online_status(bool online) { long val = online ? 1 : 0; int ret; ret = file_write_long(online_path, val); if (ret != 0) fprintf(stderr, "Failed to bring CPU %s (%s)", online ? "online" : "offline", strerror(errno)); } static enum scx_test_status setup(void **ctx) { if (!is_cpu_online()) return SCX_TEST_SKIP; return SCX_TEST_PASS; } static enum scx_test_status test_hotplug(bool onlining, bool cbs_defined) { struct hotplug *skel; struct bpf_link *link; long kind, code; SCX_ASSERT(is_cpu_online()); skel = hotplug__open_and_load(); SCX_ASSERT(skel); /* Testing the offline -> online path, so go offline before starting */ if (onlining) toggle_online_status(0); if (cbs_defined) { kind = SCX_KIND_VAL(SCX_EXIT_UNREG_BPF); code = SCX_ECODE_VAL(SCX_ECODE_ACT_RESTART) | HOTPLUG_EXIT_RSN; if (onlining) code |= HOTPLUG_ONLINING; } else { kind = SCX_KIND_VAL(SCX_EXIT_UNREG_KERN); code = SCX_ECODE_VAL(SCX_ECODE_ACT_RESTART) | SCX_ECODE_VAL(SCX_ECODE_RSN_HOTPLUG); } if (cbs_defined) link = bpf_map__attach_struct_ops(skel->maps.hotplug_cb_ops); else link = bpf_map__attach_struct_ops(skel->maps.hotplug_nocb_ops); if (!link) { SCX_ERR("Failed to attach scheduler"); hotplug__destroy(skel); return SCX_TEST_FAIL; } toggle_online_status(onlining ? 1 : 0); while (!UEI_EXITED(skel, uei)) sched_yield(); SCX_EQ(skel->data->uei.kind, kind); SCX_EQ(UEI_REPORT(skel, uei), code); if (!onlining) toggle_online_status(1); bpf_link__destroy(link); hotplug__destroy(skel); return SCX_TEST_PASS; } static enum scx_test_status test_hotplug_attach(void) { struct hotplug *skel; struct bpf_link *link; enum scx_test_status status = SCX_TEST_PASS; long kind, code; SCX_ASSERT(is_cpu_online()); SCX_ASSERT(scx_hotplug_seq() > 0); skel = SCX_OPS_OPEN(hotplug_nocb_ops, hotplug); SCX_ASSERT(skel); SCX_OPS_LOAD(skel, hotplug_nocb_ops, hotplug, uei); /* * Take the CPU offline to increment the global hotplug seq, which * should cause attach to fail due to us setting the hotplug seq above */ toggle_online_status(0); link = bpf_map__attach_struct_ops(skel->maps.hotplug_nocb_ops); toggle_online_status(1); SCX_ASSERT(link); while (!UEI_EXITED(skel, uei)) sched_yield(); kind = SCX_KIND_VAL(SCX_EXIT_UNREG_KERN); code = SCX_ECODE_VAL(SCX_ECODE_ACT_RESTART) | SCX_ECODE_VAL(SCX_ECODE_RSN_HOTPLUG); SCX_EQ(skel->data->uei.kind, kind); SCX_EQ(UEI_REPORT(skel, uei), code); bpf_link__destroy(link); hotplug__destroy(skel); return status; } static enum scx_test_status run(void *ctx) { #define HP_TEST(__onlining, __cbs_defined) ({ \ if (test_hotplug(__onlining, __cbs_defined) != SCX_TEST_PASS) \ return SCX_TEST_FAIL; \ }) HP_TEST(true, true); HP_TEST(false, true); HP_TEST(true, false); HP_TEST(false, false); #undef HP_TEST return test_hotplug_attach(); } static void cleanup(void *ctx) { toggle_online_status(1); } struct scx_test hotplug_test = { .name = "hotplug", .description = "Verify hotplug behavior", .setup = setup, .run = run, .cleanup = cleanup, }; REGISTER_SCX_TEST(&hotplug_test)