mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-05 16:54:27 +00:00

Arnd Bergmann described that MIPS system calls don't necessarily start from 0 as an ABI prefix is applied: https://lore.kernel.org/lkml/8ed7dfb2-1e4d-4aa4-a04b-0397a89365d1@app.fastmail.com/ When decoding the "id" (aka system call number) for MIPS ignore values greater-than 1000. Signed-off-by: Ian Rogers <irogers@google.com> Reviewed-by: Namhyung Kim <namhyung@kernel.org> Acked-by: Arnaldo Carvalho de Melo <acme@kernel.org> Link: https://lore.kernel.org/r/20250319050741.269828-12-irogers@google.com Signed-off-by: Namhyung Kim <namhyung@kernel.org>
133 lines
3 KiB
C
133 lines
3 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* System call table mapper
|
|
*
|
|
* (C) 2016 Arnaldo Carvalho de Melo <acme@redhat.com>
|
|
*/
|
|
|
|
#include "syscalltbl.h"
|
|
#include <stdlib.h>
|
|
#include <asm/bitsperlong.h>
|
|
#include <linux/compiler.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/zalloc.h>
|
|
|
|
#include <string.h>
|
|
#include "string2.h"
|
|
|
|
#include "trace/beauty/generated/syscalltbl.c"
|
|
|
|
static const struct syscalltbl *find_table(int e_machine)
|
|
{
|
|
static const struct syscalltbl *last_table;
|
|
static int last_table_machine = EM_NONE;
|
|
|
|
/* Tables only exist for EM_SPARC. */
|
|
if (e_machine == EM_SPARCV9)
|
|
e_machine = EM_SPARC;
|
|
|
|
if (last_table_machine == e_machine && last_table != NULL)
|
|
return last_table;
|
|
|
|
for (size_t i = 0; i < ARRAY_SIZE(syscalltbls); i++) {
|
|
const struct syscalltbl *entry = &syscalltbls[i];
|
|
|
|
if (entry->e_machine != e_machine && entry->e_machine != EM_NONE)
|
|
continue;
|
|
|
|
last_table = entry;
|
|
last_table_machine = e_machine;
|
|
return entry;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
const char *syscalltbl__name(int e_machine, int id)
|
|
{
|
|
const struct syscalltbl *table = find_table(e_machine);
|
|
|
|
if (e_machine == EM_MIPS && id > 1000) {
|
|
/*
|
|
* MIPS may encode the N32/64/O32 type in the high part of
|
|
* syscall number. Mask this off if present. See the values of
|
|
* __NR_N32_Linux, __NR_64_Linux, __NR_O32_Linux and __NR_Linux.
|
|
*/
|
|
id = id % 1000;
|
|
}
|
|
if (table && id >= 0 && id < table->num_to_name_len)
|
|
return table->num_to_name[id];
|
|
return NULL;
|
|
}
|
|
|
|
struct syscall_cmp_key {
|
|
const char *name;
|
|
const char *const *tbl;
|
|
};
|
|
|
|
static int syscallcmpname(const void *vkey, const void *ventry)
|
|
{
|
|
const struct syscall_cmp_key *key = vkey;
|
|
const uint16_t *entry = ventry;
|
|
|
|
return strcmp(key->name, key->tbl[*entry]);
|
|
}
|
|
|
|
int syscalltbl__id(int e_machine, const char *name)
|
|
{
|
|
const struct syscalltbl *table = find_table(e_machine);
|
|
struct syscall_cmp_key key;
|
|
const uint16_t *id;
|
|
|
|
if (!table)
|
|
return -1;
|
|
|
|
key.name = name;
|
|
key.tbl = table->num_to_name;
|
|
id = bsearch(&key, table->sorted_names, table->sorted_names_len,
|
|
sizeof(table->sorted_names[0]), syscallcmpname);
|
|
|
|
return id ? *id : -1;
|
|
}
|
|
|
|
int syscalltbl__num_idx(int e_machine)
|
|
{
|
|
const struct syscalltbl *table = find_table(e_machine);
|
|
|
|
if (!table)
|
|
return 0;
|
|
|
|
return table->sorted_names_len;
|
|
}
|
|
|
|
int syscalltbl__id_at_idx(int e_machine, int idx)
|
|
{
|
|
const struct syscalltbl *table = find_table(e_machine);
|
|
|
|
if (!table)
|
|
return -1;
|
|
|
|
assert(idx >= 0 && idx < table->sorted_names_len);
|
|
return table->sorted_names[idx];
|
|
}
|
|
|
|
int syscalltbl__strglobmatch_next(int e_machine, const char *syscall_glob, int *idx)
|
|
{
|
|
const struct syscalltbl *table = find_table(e_machine);
|
|
|
|
for (int i = *idx + 1; table && i < table->sorted_names_len; ++i) {
|
|
const char *name = table->num_to_name[table->sorted_names[i]];
|
|
|
|
if (strglobmatch(name, syscall_glob)) {
|
|
*idx = i;
|
|
return table->sorted_names[i];
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
int syscalltbl__strglobmatch_first(int e_machine, const char *syscall_glob, int *idx)
|
|
{
|
|
*idx = -1;
|
|
return syscalltbl__strglobmatch_next(e_machine, syscall_glob, idx);
|
|
}
|