1570af302Sopenharmony_ci/*
2570af302Sopenharmony_ci * Copyright (c) 2024 Huawei Device Co., Ltd.
3570af302Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4570af302Sopenharmony_ci * you may not use this file except in compliance with the License.
5570af302Sopenharmony_ci * You may obtain a copy of the License at
6570af302Sopenharmony_ci *
7570af302Sopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
8570af302Sopenharmony_ci *
9570af302Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10570af302Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11570af302Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12570af302Sopenharmony_ci * See the License for the specific language governing permissions and
13570af302Sopenharmony_ci * limitations under the License.
14570af302Sopenharmony_ci */
15570af302Sopenharmony_ci
16570af302Sopenharmony_ci#include <stdio.h>
17570af302Sopenharmony_ci#include <syscall.h>
18570af302Sopenharmony_ci#include <assert.h>
19570af302Sopenharmony_ci#include <syscall_hooks.h>
20570af302Sopenharmony_ci
21570af302Sopenharmony_ci#include "libc.h"
22570af302Sopenharmony_ci#include "pthread_impl.h"
23570af302Sopenharmony_ci
24570af302Sopenharmony_ci#define HIJACK_SYSCALL_MAX  1024
25570af302Sopenharmony_ci
26570af302Sopenharmony_civolatile const char *g_syscall_hooks_table __attribute__((aligned(8))) = NULL;
27570af302Sopenharmony_civolatile void *g_syscall_hooks_entry __attribute__((aligned(8))) = NULL;
28570af302Sopenharmony_ci
29570af302Sopenharmony_ci/*
30570af302Sopenharmony_ci * @brief
31570af302Sopenharmony_ci *     reset == 0: g_syscall_hooks_table and g_syscall_hooks_entry will be the function
32570af302Sopenharmony_ci *                 input parameters hooks_table and hooks_entry.
33570af302Sopenharmony_ci *     reset != 0: g_syscall_hooks_table and g_syscall_hooks_entry will be default values
34570af302Sopenharmony_ci *
35570af302Sopenharmony_ci * @param
36570af302Sopenharmony_ci *     hooks_table: pointer to a syscall table
37570af302Sopenharmony_ci *     hooks_entry: pointer to svc0_entry
38570af302Sopenharmony_ci *
39570af302Sopenharmony_ci * @return
40570af302Sopenharmony_ci *     -EINVAL: input parameters are invalid
41570af302Sopenharmony_ci *     -EPERM: operation not allowed
42570af302Sopenharmony_ci *     0: set_syscall_hooks success
43570af302Sopenharmony_ci*/
44570af302Sopenharmony_ci
45570af302Sopenharmony_ciint set_syscall_hooks(const char *hooks_table, int table_len, void *hooks_entry, int reset, int **tid_addr)
46570af302Sopenharmony_ci{
47570af302Sopenharmony_ci	int ret = 0;
48570af302Sopenharmony_ci
49570af302Sopenharmony_ci	if (reset == 0) {
50570af302Sopenharmony_ci		if (hooks_table == NULL || hooks_entry == NULL || tid_addr == NULL ||
51570af302Sopenharmony_ci			table_len < HIJACK_SYSCALL_MAX) {
52570af302Sopenharmony_ci				ret = -EINVAL;
53570af302Sopenharmony_ci			}
54570af302Sopenharmony_ci	}
55570af302Sopenharmony_ci
56570af302Sopenharmony_ci	if (ret == 0) {
57570af302Sopenharmony_ci		__tl_lock();
58570af302Sopenharmony_ci		if (!libc.threads_minus_1) {
59570af302Sopenharmony_ci			if (reset) {
60570af302Sopenharmony_ci				g_syscall_hooks_entry = NULL;
61570af302Sopenharmony_ci				g_syscall_hooks_table = NULL;
62570af302Sopenharmony_ci			} else {
63570af302Sopenharmony_ci				g_syscall_hooks_entry = hooks_entry;
64570af302Sopenharmony_ci				g_syscall_hooks_table = hooks_table;
65570af302Sopenharmony_ci			}
66570af302Sopenharmony_ci			*tid_addr = &__thread_list_lock;
67570af302Sopenharmony_ci		} else {
68570af302Sopenharmony_ci			ret = -EPERM;
69570af302Sopenharmony_ci		}
70570af302Sopenharmony_ci		__tl_unlock();
71570af302Sopenharmony_ci	}
72570af302Sopenharmony_ci
73570af302Sopenharmony_ci	return ret;
74570af302Sopenharmony_ci}