Path: blob/master/tools/testing/selftests/futex/functional/futex_numa_mpol.c
29271 views
// SPDX-License-Identifier: GPL-2.0-or-later1/*2* Copyright (C) 2025 Sebastian Andrzej Siewior <[email protected]>3*/45#define _GNU_SOURCE67#include <errno.h>8#include <pthread.h>9#include <stdio.h>10#include <stdlib.h>11#include <unistd.h>12#include <numa.h>13#include <numaif.h>1415#include <linux/futex.h>16#include <sys/mman.h>1718#include "futextest.h"19#include "futex2test.h"20#include "../../kselftest_harness.h"2122#define MAX_THREADS 642324static pthread_barrier_t barrier_main;25static pthread_t threads[MAX_THREADS];2627struct thread_args {28void *futex_ptr;29unsigned int flags;30int result;31};3233static struct thread_args thread_args[MAX_THREADS];3435#ifndef FUTEX_NO_NODE36#define FUTEX_NO_NODE (-1)37#endif3839#ifndef FUTEX2_MPOL40#define FUTEX2_MPOL 0x0841#endif4243static void *thread_lock_fn(void *arg)44{45struct thread_args *args = arg;46int ret;4748pthread_barrier_wait(&barrier_main);49ret = futex2_wait(args->futex_ptr, 0, args->flags, NULL, 0);50args->result = ret;51return NULL;52}5354static void create_max_threads(void *futex_ptr)55{56int i, ret;5758for (i = 0; i < MAX_THREADS; i++) {59thread_args[i].futex_ptr = futex_ptr;60thread_args[i].flags = FUTEX2_SIZE_U32 | FUTEX_PRIVATE_FLAG | FUTEX2_NUMA;61thread_args[i].result = 0;62ret = pthread_create(&threads[i], NULL, thread_lock_fn, &thread_args[i]);63if (ret)64ksft_exit_fail_msg("pthread_create failed\n");65}66}6768static void join_max_threads(void)69{70int i, ret;7172for (i = 0; i < MAX_THREADS; i++) {73ret = pthread_join(threads[i], NULL);74if (ret)75ksft_exit_fail_msg("pthread_join failed for thread %d\n", i);76}77}7879static void __test_futex(void *futex_ptr, int err_value, unsigned int futex_flags)80{81int to_wake, ret, i, need_exit = 0;8283pthread_barrier_init(&barrier_main, NULL, MAX_THREADS + 1);84create_max_threads(futex_ptr);85pthread_barrier_wait(&barrier_main);86to_wake = MAX_THREADS;8788do {89ret = futex2_wake(futex_ptr, to_wake, futex_flags);9091if (err_value) {92if (ret >= 0)93ksft_exit_fail_msg("futex2_wake(%d, 0x%x) should fail, but didn't\n",94to_wake, futex_flags);9596if (errno != err_value)97ksft_exit_fail_msg("futex2_wake(%d, 0x%x) expected error was %d, but returned %d (%s)\n",98to_wake, futex_flags, err_value, errno, strerror(errno));99100break;101}102if (ret < 0) {103ksft_exit_fail_msg("Failed futex2_wake(%d, 0x%x): %m\n",104to_wake, futex_flags);105}106if (!ret)107usleep(50);108to_wake -= ret;109110} while (to_wake);111join_max_threads();112113for (i = 0; i < MAX_THREADS; i++) {114if (err_value && thread_args[i].result != -1) {115ksft_print_msg("Thread %d should fail but succeeded (%d)\n",116i, thread_args[i].result);117need_exit = 1;118}119if (!err_value && thread_args[i].result != 0) {120ksft_print_msg("Thread %d failed (%d)\n", i, thread_args[i].result);121need_exit = 1;122}123}124if (need_exit)125ksft_exit_fail_msg("Aborting due to earlier errors.\n");126}127128static void test_futex(void *futex_ptr, int err_value)129{130__test_futex(futex_ptr, err_value, FUTEX2_SIZE_U32 | FUTEX_PRIVATE_FLAG | FUTEX2_NUMA);131}132133static void test_futex_mpol(void *futex_ptr, int err_value)134{135__test_futex(futex_ptr, err_value, FUTEX2_SIZE_U32 | FUTEX_PRIVATE_FLAG | FUTEX2_NUMA | FUTEX2_MPOL);136}137138TEST(futex_numa_mpol)139{140struct futex32_numa *futex_numa;141void *futex_ptr;142int mem_size;143144mem_size = sysconf(_SC_PAGE_SIZE);145futex_ptr = mmap(NULL, mem_size * 2, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, 0, 0);146if (futex_ptr == MAP_FAILED)147ksft_exit_fail_msg("mmap() for %d bytes failed\n", mem_size);148149/* Create an invalid memory region for the "Memory out of range" test */150mprotect(futex_ptr + mem_size, mem_size, PROT_NONE);151152futex_numa = futex_ptr;153154ksft_print_msg("Regular test\n");155futex_numa->futex = 0;156futex_numa->numa = FUTEX_NO_NODE;157test_futex(futex_ptr, 0);158159if (futex_numa->numa == FUTEX_NO_NODE)160ksft_exit_fail_msg("NUMA node is left uninitialized\n");161162/* FUTEX2_NUMA futex must be 8-byte aligned */163ksft_print_msg("Mis-aligned futex\n");164test_futex(futex_ptr + mem_size - 4, EINVAL);165166ksft_print_msg("Memory out of range\n");167test_futex(futex_ptr + mem_size, EFAULT);168169futex_numa->numa = FUTEX_NO_NODE;170mprotect(futex_ptr, mem_size, PROT_READ);171ksft_print_msg("Memory, RO\n");172test_futex(futex_ptr, EFAULT);173174mprotect(futex_ptr, mem_size, PROT_NONE);175ksft_print_msg("Memory, no access\n");176test_futex(futex_ptr, EFAULT);177178mprotect(futex_ptr, mem_size, PROT_READ | PROT_WRITE);179ksft_print_msg("Memory back to RW\n");180test_futex(futex_ptr, 0);181182ksft_test_result_pass("futex2 memory boundary tests passed\n");183184/* MPOL test. Does not work as expected */185#ifdef LIBNUMA_VER_SUFFICIENT186for (int i = 0; i < 4; i++) {187unsigned long nodemask;188int ret;189190nodemask = 1 << i;191ret = mbind(futex_ptr, mem_size, MPOL_BIND, &nodemask,192sizeof(nodemask) * 8, 0);193if (ret == 0) {194ret = numa_set_mempolicy_home_node(futex_ptr, mem_size, i, 0);195if (ret != 0)196ksft_exit_fail_msg("Failed to set home node: %m, %d\n", errno);197198ksft_print_msg("Node %d test\n", i);199futex_numa->futex = 0;200futex_numa->numa = FUTEX_NO_NODE;201202ret = futex2_wake(futex_ptr, 0, FUTEX2_SIZE_U32 | FUTEX_PRIVATE_FLAG | FUTEX2_NUMA | FUTEX2_MPOL);203if (ret < 0)204ksft_test_result_fail("Failed to wake 0 with MPOL: %m\n");205if (futex_numa->numa != i) {206ksft_exit_fail_msg("Returned NUMA node is %d expected %d\n",207futex_numa->numa, i);208}209}210}211ksft_test_result_pass("futex2 MPOL hints test passed\n");212#else213ksft_test_result_skip("futex2 MPOL hints test requires libnuma 2.0.16+\n");214#endif215munmap(futex_ptr, mem_size * 2);216}217218TEST_HARNESS_MAIN219220221