xref: /openbmc/linux/tools/perf/util/syscalltbl.c (revision 1ac731c529cd4d6adbce134754b51ff7d822b145)
12025cf9eSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2fd0db102SArnaldo Carvalho de Melo /*
3fd0db102SArnaldo Carvalho de Melo  * System call table mapper
4fd0db102SArnaldo Carvalho de Melo  *
5fd0db102SArnaldo Carvalho de Melo  * (C) 2016 Arnaldo Carvalho de Melo <acme@redhat.com>
6fd0db102SArnaldo Carvalho de Melo  */
7fd0db102SArnaldo Carvalho de Melo 
8fd0db102SArnaldo Carvalho de Melo #include "syscalltbl.h"
95af56fabSArnaldo Carvalho de Melo #include <stdlib.h>
10090657c9SAkemi Yagi #include <linux/compiler.h>
11a9e8c1f8SArnaldo Carvalho de Melo #include <linux/zalloc.h>
12fd0db102SArnaldo Carvalho de Melo 
1322e9af4eSJin Yao #ifdef HAVE_SYSCALL_TABLE_SUPPORT
145af56fabSArnaldo Carvalho de Melo #include <string.h>
1589be3f8aSArnaldo Carvalho de Melo #include "string2.h"
165af56fabSArnaldo Carvalho de Melo 
171b700c99SArnaldo Carvalho de Melo #if defined(__x86_64__)
181b700c99SArnaldo Carvalho de Melo #include <asm/syscalls_64.c>
191b700c99SArnaldo Carvalho de Melo const int syscalltbl_native_max_id = SYSCALLTBL_x86_64_MAX_ID;
201b700c99SArnaldo Carvalho de Melo static const char *const *syscalltbl_native = syscalltbl_x86_64;
21901bb028SHendrik Brueckner #elif defined(__s390x__)
22901bb028SHendrik Brueckner #include <asm/syscalls_64.c>
23901bb028SHendrik Brueckner const int syscalltbl_native_max_id = SYSCALLTBL_S390_64_MAX_ID;
24901bb028SHendrik Brueckner static const char *const *syscalltbl_native = syscalltbl_s390_64;
254281da23SRavi Bangoria #elif defined(__powerpc64__)
264281da23SRavi Bangoria #include <asm/syscalls_64.c>
274281da23SRavi Bangoria const int syscalltbl_native_max_id = SYSCALLTBL_POWERPC_64_MAX_ID;
284281da23SRavi Bangoria static const char *const *syscalltbl_native = syscalltbl_powerpc_64;
294281da23SRavi Bangoria #elif defined(__powerpc__)
304281da23SRavi Bangoria #include <asm/syscalls_32.c>
314281da23SRavi Bangoria const int syscalltbl_native_max_id = SYSCALLTBL_POWERPC_32_MAX_ID;
324281da23SRavi Bangoria static const char *const *syscalltbl_native = syscalltbl_powerpc_32;
33a7f660d6SKim Phillips #elif defined(__aarch64__)
34a7f660d6SKim Phillips #include <asm/syscalls.c>
35a7f660d6SKim Phillips const int syscalltbl_native_max_id = SYSCALLTBL_ARM64_MAX_ID;
36a7f660d6SKim Phillips static const char *const *syscalltbl_native = syscalltbl_arm64;
37d9fd5a71STiezhu Yang #elif defined(__mips__)
38d9fd5a71STiezhu Yang #include <asm/syscalls_n64.c>
39d9fd5a71STiezhu Yang const int syscalltbl_native_max_id = SYSCALLTBL_MIPS_N64_MAX_ID;
40d9fd5a71STiezhu Yang static const char *const *syscalltbl_native = syscalltbl_mips_n64;
41*2fa5ebe3SHuacai Chen #elif defined(__loongarch__)
42*2fa5ebe3SHuacai Chen #include <asm/syscalls.c>
43*2fa5ebe3SHuacai Chen const int syscalltbl_native_max_id = SYSCALLTBL_LOONGARCH_MAX_ID;
44*2fa5ebe3SHuacai Chen static const char *const *syscalltbl_native = syscalltbl_loongarch;
451b700c99SArnaldo Carvalho de Melo #endif
461b700c99SArnaldo Carvalho de Melo 
475af56fabSArnaldo Carvalho de Melo struct syscall {
485af56fabSArnaldo Carvalho de Melo 	int id;
495af56fabSArnaldo Carvalho de Melo 	const char *name;
505af56fabSArnaldo Carvalho de Melo };
515af56fabSArnaldo Carvalho de Melo 
syscallcmpname(const void * vkey,const void * ventry)525af56fabSArnaldo Carvalho de Melo static int syscallcmpname(const void *vkey, const void *ventry)
535af56fabSArnaldo Carvalho de Melo {
545af56fabSArnaldo Carvalho de Melo 	const char *key = vkey;
555af56fabSArnaldo Carvalho de Melo 	const struct syscall *entry = ventry;
565af56fabSArnaldo Carvalho de Melo 
575af56fabSArnaldo Carvalho de Melo 	return strcmp(key, entry->name);
585af56fabSArnaldo Carvalho de Melo }
595af56fabSArnaldo Carvalho de Melo 
syscallcmp(const void * va,const void * vb)605af56fabSArnaldo Carvalho de Melo static int syscallcmp(const void *va, const void *vb)
615af56fabSArnaldo Carvalho de Melo {
625af56fabSArnaldo Carvalho de Melo 	const struct syscall *a = va, *b = vb;
635af56fabSArnaldo Carvalho de Melo 
645af56fabSArnaldo Carvalho de Melo 	return strcmp(a->name, b->name);
655af56fabSArnaldo Carvalho de Melo }
665af56fabSArnaldo Carvalho de Melo 
syscalltbl__init_native(struct syscalltbl * tbl)675af56fabSArnaldo Carvalho de Melo static int syscalltbl__init_native(struct syscalltbl *tbl)
685af56fabSArnaldo Carvalho de Melo {
695af56fabSArnaldo Carvalho de Melo 	int nr_entries = 0, i, j;
705af56fabSArnaldo Carvalho de Melo 	struct syscall *entries;
715af56fabSArnaldo Carvalho de Melo 
725af56fabSArnaldo Carvalho de Melo 	for (i = 0; i <= syscalltbl_native_max_id; ++i)
735af56fabSArnaldo Carvalho de Melo 		if (syscalltbl_native[i])
745af56fabSArnaldo Carvalho de Melo 			++nr_entries;
755af56fabSArnaldo Carvalho de Melo 
765af56fabSArnaldo Carvalho de Melo 	entries = tbl->syscalls.entries = malloc(sizeof(struct syscall) * nr_entries);
775af56fabSArnaldo Carvalho de Melo 	if (tbl->syscalls.entries == NULL)
785af56fabSArnaldo Carvalho de Melo 		return -1;
795af56fabSArnaldo Carvalho de Melo 
805af56fabSArnaldo Carvalho de Melo 	for (i = 0, j = 0; i <= syscalltbl_native_max_id; ++i) {
815af56fabSArnaldo Carvalho de Melo 		if (syscalltbl_native[i]) {
825af56fabSArnaldo Carvalho de Melo 			entries[j].name = syscalltbl_native[i];
835af56fabSArnaldo Carvalho de Melo 			entries[j].id = i;
845af56fabSArnaldo Carvalho de Melo 			++j;
855af56fabSArnaldo Carvalho de Melo 		}
865af56fabSArnaldo Carvalho de Melo 	}
875af56fabSArnaldo Carvalho de Melo 
885af56fabSArnaldo Carvalho de Melo 	qsort(tbl->syscalls.entries, nr_entries, sizeof(struct syscall), syscallcmp);
895af56fabSArnaldo Carvalho de Melo 	tbl->syscalls.nr_entries = nr_entries;
9030a910d7SArnaldo Carvalho de Melo 	tbl->syscalls.max_id	 = syscalltbl_native_max_id;
915af56fabSArnaldo Carvalho de Melo 	return 0;
925af56fabSArnaldo Carvalho de Melo }
93fd0db102SArnaldo Carvalho de Melo 
syscalltbl__new(void)94fd0db102SArnaldo Carvalho de Melo struct syscalltbl *syscalltbl__new(void)
95fd0db102SArnaldo Carvalho de Melo {
96fd0db102SArnaldo Carvalho de Melo 	struct syscalltbl *tbl = malloc(sizeof(*tbl));
97fd0db102SArnaldo Carvalho de Melo 	if (tbl) {
985af56fabSArnaldo Carvalho de Melo 		if (syscalltbl__init_native(tbl)) {
995af56fabSArnaldo Carvalho de Melo 			free(tbl);
1005af56fabSArnaldo Carvalho de Melo 			return NULL;
101fd0db102SArnaldo Carvalho de Melo 		}
1025af56fabSArnaldo Carvalho de Melo 	}
1035af56fabSArnaldo Carvalho de Melo 	return tbl;
1045af56fabSArnaldo Carvalho de Melo }
1055af56fabSArnaldo Carvalho de Melo 
syscalltbl__delete(struct syscalltbl * tbl)1065af56fabSArnaldo Carvalho de Melo void syscalltbl__delete(struct syscalltbl *tbl)
1075af56fabSArnaldo Carvalho de Melo {
1085af56fabSArnaldo Carvalho de Melo 	zfree(&tbl->syscalls.entries);
1095af56fabSArnaldo Carvalho de Melo 	free(tbl);
1105af56fabSArnaldo Carvalho de Melo }
1115af56fabSArnaldo Carvalho de Melo 
syscalltbl__name(const struct syscalltbl * tbl __maybe_unused,int id)1125af56fabSArnaldo Carvalho de Melo const char *syscalltbl__name(const struct syscalltbl *tbl __maybe_unused, int id)
1135af56fabSArnaldo Carvalho de Melo {
1145af56fabSArnaldo Carvalho de Melo 	return id <= syscalltbl_native_max_id ? syscalltbl_native[id]: NULL;
1155af56fabSArnaldo Carvalho de Melo }
1165af56fabSArnaldo Carvalho de Melo 
syscalltbl__id(struct syscalltbl * tbl,const char * name)1175af56fabSArnaldo Carvalho de Melo int syscalltbl__id(struct syscalltbl *tbl, const char *name)
1185af56fabSArnaldo Carvalho de Melo {
1195af56fabSArnaldo Carvalho de Melo 	struct syscall *sc = bsearch(name, tbl->syscalls.entries,
1205af56fabSArnaldo Carvalho de Melo 				     tbl->syscalls.nr_entries, sizeof(*sc),
1215af56fabSArnaldo Carvalho de Melo 				     syscallcmpname);
1225af56fabSArnaldo Carvalho de Melo 
1235af56fabSArnaldo Carvalho de Melo 	return sc ? sc->id : -1;
1245af56fabSArnaldo Carvalho de Melo }
1255af56fabSArnaldo Carvalho de Melo 
syscalltbl__strglobmatch_next(struct syscalltbl * tbl,const char * syscall_glob,int * idx)12689be3f8aSArnaldo Carvalho de Melo int syscalltbl__strglobmatch_next(struct syscalltbl *tbl, const char *syscall_glob, int *idx)
12789be3f8aSArnaldo Carvalho de Melo {
12889be3f8aSArnaldo Carvalho de Melo 	int i;
12989be3f8aSArnaldo Carvalho de Melo 	struct syscall *syscalls = tbl->syscalls.entries;
13089be3f8aSArnaldo Carvalho de Melo 
13189be3f8aSArnaldo Carvalho de Melo 	for (i = *idx + 1; i < tbl->syscalls.nr_entries; ++i) {
13289be3f8aSArnaldo Carvalho de Melo 		if (strglobmatch(syscalls[i].name, syscall_glob)) {
13389be3f8aSArnaldo Carvalho de Melo 			*idx = i;
13489be3f8aSArnaldo Carvalho de Melo 			return syscalls[i].id;
13589be3f8aSArnaldo Carvalho de Melo 		}
13689be3f8aSArnaldo Carvalho de Melo 	}
13789be3f8aSArnaldo Carvalho de Melo 
13889be3f8aSArnaldo Carvalho de Melo 	return -1;
13989be3f8aSArnaldo Carvalho de Melo }
14089be3f8aSArnaldo Carvalho de Melo 
syscalltbl__strglobmatch_first(struct syscalltbl * tbl,const char * syscall_glob,int * idx)14189be3f8aSArnaldo Carvalho de Melo int syscalltbl__strglobmatch_first(struct syscalltbl *tbl, const char *syscall_glob, int *idx)
14289be3f8aSArnaldo Carvalho de Melo {
14389be3f8aSArnaldo Carvalho de Melo 	*idx = -1;
14489be3f8aSArnaldo Carvalho de Melo 	return syscalltbl__strglobmatch_next(tbl, syscall_glob, idx);
14589be3f8aSArnaldo Carvalho de Melo }
14689be3f8aSArnaldo Carvalho de Melo 
14722e9af4eSJin Yao #else /* HAVE_SYSCALL_TABLE_SUPPORT */
1485af56fabSArnaldo Carvalho de Melo 
1495af56fabSArnaldo Carvalho de Melo #include <libaudit.h>
1505af56fabSArnaldo Carvalho de Melo 
syscalltbl__new(void)1515af56fabSArnaldo Carvalho de Melo struct syscalltbl *syscalltbl__new(void)
1525af56fabSArnaldo Carvalho de Melo {
153a9e8c1f8SArnaldo Carvalho de Melo 	struct syscalltbl *tbl = zalloc(sizeof(*tbl));
1545af56fabSArnaldo Carvalho de Melo 	if (tbl)
1555af56fabSArnaldo Carvalho de Melo 		tbl->audit_machine = audit_detect_machine();
156fd0db102SArnaldo Carvalho de Melo 	return tbl;
157fd0db102SArnaldo Carvalho de Melo }
158fd0db102SArnaldo Carvalho de Melo 
syscalltbl__delete(struct syscalltbl * tbl)159fd0db102SArnaldo Carvalho de Melo void syscalltbl__delete(struct syscalltbl *tbl)
160fd0db102SArnaldo Carvalho de Melo {
161fd0db102SArnaldo Carvalho de Melo 	free(tbl);
162fd0db102SArnaldo Carvalho de Melo }
163fd0db102SArnaldo Carvalho de Melo 
syscalltbl__name(const struct syscalltbl * tbl,int id)164fd0db102SArnaldo Carvalho de Melo const char *syscalltbl__name(const struct syscalltbl *tbl, int id)
165fd0db102SArnaldo Carvalho de Melo {
166fd0db102SArnaldo Carvalho de Melo 	return audit_syscall_to_name(id, tbl->audit_machine);
167fd0db102SArnaldo Carvalho de Melo }
168fd0db102SArnaldo Carvalho de Melo 
syscalltbl__id(struct syscalltbl * tbl,const char * name)169fd0db102SArnaldo Carvalho de Melo int syscalltbl__id(struct syscalltbl *tbl, const char *name)
170fd0db102SArnaldo Carvalho de Melo {
171fd0db102SArnaldo Carvalho de Melo 	return audit_name_to_syscall(name, tbl->audit_machine);
172fd0db102SArnaldo Carvalho de Melo }
17389be3f8aSArnaldo Carvalho de Melo 
syscalltbl__strglobmatch_next(struct syscalltbl * tbl __maybe_unused,const char * syscall_glob __maybe_unused,int * idx __maybe_unused)17489be3f8aSArnaldo Carvalho de Melo int syscalltbl__strglobmatch_next(struct syscalltbl *tbl __maybe_unused,
17589be3f8aSArnaldo Carvalho de Melo 				  const char *syscall_glob __maybe_unused, int *idx __maybe_unused)
17689be3f8aSArnaldo Carvalho de Melo {
17789be3f8aSArnaldo Carvalho de Melo 	return -1;
17889be3f8aSArnaldo Carvalho de Melo }
17989be3f8aSArnaldo Carvalho de Melo 
syscalltbl__strglobmatch_first(struct syscalltbl * tbl,const char * syscall_glob,int * idx)18089be3f8aSArnaldo Carvalho de Melo int syscalltbl__strglobmatch_first(struct syscalltbl *tbl, const char *syscall_glob, int *idx)
18189be3f8aSArnaldo Carvalho de Melo {
18289be3f8aSArnaldo Carvalho de Melo 	return syscalltbl__strglobmatch_next(tbl, syscall_glob, idx);
18389be3f8aSArnaldo Carvalho de Melo }
18422e9af4eSJin Yao #endif /* HAVE_SYSCALL_TABLE_SUPPORT */
185