mirror of
https://github.com/torvalds/linux.git
synced 2026-04-19 07:13:56 -04:00
Add NUMA helpers to probe for support/availability and to check if the test is running on a multi-node system. The APIs will be used to verify guest_memfd NUMA support. Signed-off-by: Shivank Garg <shivankg@amd.com> [sean: land helpers in numaif.h, add comments, tweak names] Link: https://lore.kernel.org/r/20251016172853.52451-11-seanjc@google.com Signed-off-by: Sean Christopherson <seanjc@google.com>
84 lines
2.1 KiB
C
84 lines
2.1 KiB
C
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
/* Copyright (C) 2020, Google LLC. */
|
|
|
|
#ifndef SELFTEST_KVM_NUMAIF_H
|
|
#define SELFTEST_KVM_NUMAIF_H
|
|
|
|
#include <dirent.h>
|
|
|
|
#include <linux/mempolicy.h>
|
|
|
|
#include "kvm_syscalls.h"
|
|
|
|
KVM_SYSCALL_DEFINE(get_mempolicy, 5, int *, policy, const unsigned long *, nmask,
|
|
unsigned long, maxnode, void *, addr, int, flags);
|
|
|
|
KVM_SYSCALL_DEFINE(set_mempolicy, 3, int, mode, const unsigned long *, nmask,
|
|
unsigned long, maxnode);
|
|
|
|
KVM_SYSCALL_DEFINE(set_mempolicy_home_node, 4, unsigned long, start,
|
|
unsigned long, len, unsigned long, home_node,
|
|
unsigned long, flags);
|
|
|
|
KVM_SYSCALL_DEFINE(migrate_pages, 4, int, pid, unsigned long, maxnode,
|
|
const unsigned long *, frommask, const unsigned long *, tomask);
|
|
|
|
KVM_SYSCALL_DEFINE(move_pages, 6, int, pid, unsigned long, count, void *, pages,
|
|
const int *, nodes, int *, status, int, flags);
|
|
|
|
KVM_SYSCALL_DEFINE(mbind, 6, void *, addr, unsigned long, size, int, mode,
|
|
const unsigned long *, nodemask, unsigned long, maxnode,
|
|
unsigned int, flags);
|
|
|
|
static inline int get_max_numa_node(void)
|
|
{
|
|
struct dirent *de;
|
|
int max_node = 0;
|
|
DIR *d;
|
|
|
|
/*
|
|
* Assume there's a single node if the kernel doesn't support NUMA,
|
|
* or if no nodes are found.
|
|
*/
|
|
d = opendir("/sys/devices/system/node");
|
|
if (!d)
|
|
return 0;
|
|
|
|
while ((de = readdir(d)) != NULL) {
|
|
int node_id;
|
|
char *endptr;
|
|
|
|
if (strncmp(de->d_name, "node", 4) != 0)
|
|
continue;
|
|
|
|
node_id = strtol(de->d_name + 4, &endptr, 10);
|
|
if (*endptr != '\0')
|
|
continue;
|
|
|
|
if (node_id > max_node)
|
|
max_node = node_id;
|
|
}
|
|
closedir(d);
|
|
|
|
return max_node;
|
|
}
|
|
|
|
static bool is_numa_available(void)
|
|
{
|
|
/*
|
|
* Probe for NUMA by doing a dummy get_mempolicy(). If the syscall
|
|
* fails with ENOSYS, then the kernel was built without NUMA support.
|
|
* if the syscall fails with EPERM, then the process/user lacks the
|
|
* necessary capabilities (CAP_SYS_NICE).
|
|
*/
|
|
return !get_mempolicy(NULL, NULL, 0, NULL, 0) ||
|
|
(errno != ENOSYS && errno != EPERM);
|
|
}
|
|
|
|
static inline bool is_multi_numa_node_system(void)
|
|
{
|
|
return is_numa_available() && get_max_numa_node() >= 1;
|
|
}
|
|
|
|
#endif /* SELFTEST_KVM_NUMAIF_H */
|