2002-07-09 16:24:59 +04:00
|
|
|
/*
|
2008-03-15 00:58:17 +03:00
|
|
|
* Copyright 2008, Ingo Weinhold, ingo_weinhold@gmx.de.
|
2006-06-29 13:37:06 +04:00
|
|
|
* Copyright 2004-2006, Haiku Inc. All rights reserved.
|
2004-12-01 07:26:10 +03:00
|
|
|
* Distributed under the terms of the MIT License.
|
|
|
|
*/
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-12-01 07:26:10 +03:00
|
|
|
/* Big case statement for dispatching syscalls */
|
2003-10-28 16:29:29 +03:00
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <kernel.h>
|
|
|
|
#include <ksyscalls.h>
|
2002-10-23 21:31:10 +04:00
|
|
|
#include <syscalls.h>
|
2004-12-14 18:49:20 +03:00
|
|
|
#include <generic_syscall.h>
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <debug.h>
|
2008-01-18 05:13:27 +03:00
|
|
|
#include <int.h>
|
|
|
|
#include <elf.h>
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <vfs.h>
|
2004-12-14 02:02:18 +03:00
|
|
|
#include <vm.h>
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <thread.h>
|
|
|
|
#include <sem.h>
|
|
|
|
#include <port.h>
|
|
|
|
#include <cpu.h>
|
2004-09-10 19:20:37 +04:00
|
|
|
#include <arch_config.h>
|
2004-10-29 05:37:37 +04:00
|
|
|
#include <disk_device_manager/ddm_userland_interface.h>
|
2003-09-21 00:47:27 +04:00
|
|
|
#include <sys/resource.h>
|
2005-03-19 04:58:05 +03:00
|
|
|
#include <fs/fd.h>
|
2003-01-18 17:18:04 +03:00
|
|
|
#include <fs/node_monitor.h>
|
2003-01-12 19:27:03 +03:00
|
|
|
#include <kimage.h>
|
2003-01-27 06:07:30 +03:00
|
|
|
#include <ksignal.h>
|
2003-10-28 16:29:29 +03:00
|
|
|
#include <real_time_clock.h>
|
2008-01-18 05:13:27 +03:00
|
|
|
#include <safemode.h>
|
2004-04-22 02:57:39 +04:00
|
|
|
#include <system_info.h>
|
2008-01-18 05:13:27 +03:00
|
|
|
#include <tracing.h>
|
2003-08-22 03:02:00 +04:00
|
|
|
#include <user_atomic.h>
|
2004-12-01 07:26:10 +03:00
|
|
|
#include <arch/system_info.h>
|
2005-01-24 04:23:20 +03:00
|
|
|
#include <messaging.h>
|
2005-05-30 00:23:00 +04:00
|
|
|
#include <frame_buffer_console.h>
|
2008-03-11 20:12:02 +03:00
|
|
|
#include <usergroup.h>
|
2007-10-02 23:47:31 +04:00
|
|
|
#include <wait_for_objects.h>
|
2004-12-14 20:09:06 +03:00
|
|
|
|
2004-12-14 18:49:20 +03:00
|
|
|
#include <malloc.h>
|
2004-12-14 20:09:06 +03:00
|
|
|
#include <string.h>
|
2003-08-20 06:34:42 +04:00
|
|
|
|
2008-02-01 15:15:00 +03:00
|
|
|
#include "syscall_numbers.h"
|
|
|
|
|
2003-10-28 16:29:29 +03:00
|
|
|
|
2004-12-14 18:49:20 +03:00
|
|
|
typedef struct generic_syscall generic_syscall;
|
|
|
|
|
|
|
|
struct generic_syscall {
|
|
|
|
list_link link;
|
2004-12-14 20:09:06 +03:00
|
|
|
char subsystem[B_FILE_NAME_LENGTH];
|
2004-12-14 18:49:20 +03:00
|
|
|
syscall_hook hook;
|
|
|
|
uint32 version;
|
|
|
|
uint32 flags;
|
|
|
|
generic_syscall *previous;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct mutex sGenericSyscallLock;
|
|
|
|
static struct list sGenericSyscalls;
|
|
|
|
|
|
|
|
|
|
|
|
static generic_syscall *
|
2004-12-14 20:09:06 +03:00
|
|
|
find_generic_syscall(const char *subsystem)
|
2004-12-14 18:49:20 +03:00
|
|
|
{
|
|
|
|
generic_syscall *syscall = NULL;
|
|
|
|
|
|
|
|
ASSERT_LOCKED_MUTEX(&sGenericSyscallLock);
|
|
|
|
|
2008-01-18 05:13:27 +03:00
|
|
|
while ((syscall = (generic_syscall*)list_get_next_item(&sGenericSyscalls,
|
|
|
|
syscall)) != NULL) {
|
2004-12-14 20:09:06 +03:00
|
|
|
if (!strcmp(syscall->subsystem, subsystem))
|
2004-12-14 18:49:20 +03:00
|
|
|
return syscall;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** Calls the generic syscall subsystem if any.
|
|
|
|
* Also handles the special generic syscall function \c B_SYSCALL_INFO.
|
|
|
|
* Returns \c B_NAME_NOT_FOUND if either the subsystem was not found, or
|
|
|
|
* the subsystem does not support the requested function.
|
|
|
|
* All other return codes are depending on the generic syscall implementation.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static inline status_t
|
2004-12-14 20:09:06 +03:00
|
|
|
_user_generic_syscall(const char *userSubsystem, uint32 function,
|
|
|
|
void *buffer, size_t bufferSize)
|
2004-12-14 18:49:20 +03:00
|
|
|
{
|
2004-12-14 20:09:06 +03:00
|
|
|
char subsystem[B_FILE_NAME_LENGTH];
|
2004-12-14 18:49:20 +03:00
|
|
|
generic_syscall *syscall;
|
2005-01-20 20:58:58 +03:00
|
|
|
status_t status = B_NAME_NOT_FOUND;
|
2004-12-14 18:49:20 +03:00
|
|
|
|
2004-12-14 20:09:06 +03:00
|
|
|
if (!IS_USER_ADDRESS(userSubsystem)
|
|
|
|
|| user_strlcpy(subsystem, userSubsystem, sizeof(subsystem)) < B_OK)
|
|
|
|
return B_BAD_ADDRESS;
|
|
|
|
|
|
|
|
//dprintf("generic_syscall(subsystem = \"%s\", function = %lu)\n", subsystem, function);
|
2004-12-14 18:49:20 +03:00
|
|
|
|
|
|
|
mutex_lock(&sGenericSyscallLock);
|
|
|
|
|
|
|
|
syscall = find_generic_syscall(subsystem);
|
2005-01-20 20:58:58 +03:00
|
|
|
if (syscall == NULL)
|
2004-12-14 18:49:20 +03:00
|
|
|
goto out;
|
|
|
|
|
2004-12-14 20:09:06 +03:00
|
|
|
if (function >= B_RESERVED_SYSCALL_BASE) {
|
|
|
|
if (function != B_SYSCALL_INFO) {
|
|
|
|
// this is all we know
|
|
|
|
status = B_NAME_NOT_FOUND;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2004-12-14 18:49:20 +03:00
|
|
|
// special info syscall
|
|
|
|
if (bufferSize != sizeof(uint32))
|
|
|
|
status = B_BAD_VALUE;
|
|
|
|
else {
|
|
|
|
uint32 requestedVersion;
|
|
|
|
|
|
|
|
// retrieve old version
|
|
|
|
status = user_memcpy(&requestedVersion, buffer, sizeof(uint32));
|
|
|
|
if (status == B_OK && requestedVersion != 0 && requestedVersion < syscall->version)
|
|
|
|
status = B_BAD_TYPE;
|
|
|
|
|
|
|
|
// return current version
|
|
|
|
if (status == B_OK)
|
|
|
|
status = user_memcpy(buffer, &syscall->version, sizeof(uint32));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
while (syscall != NULL) {
|
|
|
|
generic_syscall *next;
|
|
|
|
|
|
|
|
mutex_unlock(&sGenericSyscallLock);
|
|
|
|
|
|
|
|
status = syscall->hook(subsystem, function, buffer, bufferSize);
|
|
|
|
|
|
|
|
mutex_lock(&sGenericSyscallLock);
|
|
|
|
if (status != B_BAD_HANDLER)
|
|
|
|
break;
|
|
|
|
|
|
|
|
// the syscall may have been removed in the mean time
|
|
|
|
next = find_generic_syscall(subsystem);
|
|
|
|
if (next == syscall)
|
|
|
|
syscall = syscall->previous;
|
|
|
|
else
|
|
|
|
syscall = next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (syscall == NULL)
|
|
|
|
status = B_NAME_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&sGenericSyscallLock);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static inline int
|
2007-02-05 00:08:35 +03:00
|
|
|
_user_is_computer_on(void)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2007-02-05 00:08:35 +03:00
|
|
|
return 1;
|
2004-08-29 00:34:43 +04:00
|
|
|
}
|
2003-08-20 06:34:42 +04:00
|
|
|
|
2004-10-01 03:25:55 +04:00
|
|
|
// map to the arch specific call
|
2004-12-14 18:49:20 +03:00
|
|
|
|
|
|
|
static inline int64
|
2004-08-29 00:34:43 +04:00
|
|
|
_user_restore_signal_frame()
|
|
|
|
{
|
2008-01-11 03:36:44 +03:00
|
|
|
syscall_64_bit_return_value();
|
|
|
|
|
2004-08-29 00:34:43 +04:00
|
|
|
return arch_restore_signal_frame();
|
|
|
|
}
|
2003-01-18 17:18:04 +03:00
|
|
|
|
2003-10-28 16:29:29 +03:00
|
|
|
|
2004-12-14 18:49:20 +03:00
|
|
|
// #pragma mark -
|
|
|
|
|
|
|
|
|
|
|
|
int32
|
|
|
|
syscall_dispatcher(uint32 call_num, void *args, uint64 *call_ret)
|
2004-08-29 00:34:43 +04:00
|
|
|
{
|
2005-02-10 06:06:05 +03:00
|
|
|
bigtime_t startTime;
|
|
|
|
|
2004-08-29 00:34:43 +04:00
|
|
|
// dprintf("syscall_dispatcher: thread 0x%x call 0x%x, arg0 0x%x, arg1 0x%x arg2 0x%x arg3 0x%x arg4 0x%x\n",
|
|
|
|
// thread_get_current_thread_id(), call_num, arg0, arg1, arg2, arg3, arg4);
|
2003-08-22 03:02:00 +04:00
|
|
|
|
2005-02-10 06:06:05 +03:00
|
|
|
user_debug_pre_syscall(call_num, args);
|
|
|
|
|
|
|
|
startTime = system_time();
|
|
|
|
|
2004-08-29 00:34:43 +04:00
|
|
|
switch (call_num) {
|
|
|
|
// the cases are auto-generated
|
|
|
|
#include "syscall_dispatcher.h"
|
2003-08-22 03:02:00 +04:00
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
default:
|
2008-01-18 05:13:27 +03:00
|
|
|
*call_ret = (uint64)B_BAD_VALUE;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2005-02-10 06:06:05 +03:00
|
|
|
user_debug_post_syscall(call_num, args, *call_ret, startTime);
|
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
// dprintf("syscall_dispatcher: done with syscall 0x%x\n", call_num);
|
|
|
|
|
2007-02-18 07:53:15 +03:00
|
|
|
return B_HANDLED_INTERRUPT;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
2004-12-14 18:49:20 +03:00
|
|
|
|
|
|
|
|
|
|
|
status_t
|
|
|
|
generic_syscall_init(void)
|
|
|
|
{
|
|
|
|
list_init(&sGenericSyscalls);
|
|
|
|
return mutex_init(&sGenericSyscallLock, "generic syscall");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
// public API
|
|
|
|
|
|
|
|
|
|
|
|
status_t
|
2004-12-14 20:09:06 +03:00
|
|
|
register_generic_syscall(const char *subsystem, syscall_hook hook,
|
2004-12-14 18:49:20 +03:00
|
|
|
uint32 version, uint32 flags)
|
|
|
|
{
|
|
|
|
struct generic_syscall *previous, *syscall;
|
|
|
|
status_t status;
|
|
|
|
|
|
|
|
if (hook == NULL)
|
|
|
|
return B_BAD_VALUE;
|
|
|
|
|
|
|
|
mutex_lock(&sGenericSyscallLock);
|
|
|
|
|
|
|
|
previous = find_generic_syscall(subsystem);
|
|
|
|
if (previous != NULL) {
|
|
|
|
if ((flags & B_DO_NOT_REPLACE_SYSCALL) != 0
|
|
|
|
|| version < previous->version) {
|
|
|
|
status = B_NAME_IN_USE;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (previous->flags & B_SYSCALL_NOT_REPLACEABLE) {
|
|
|
|
status = B_NOT_ALLOWED;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
syscall = (generic_syscall *)malloc(sizeof(struct generic_syscall));
|
|
|
|
if (syscall == NULL) {
|
|
|
|
status = B_NO_MEMORY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2004-12-14 20:09:06 +03:00
|
|
|
strlcpy(syscall->subsystem, subsystem, sizeof(syscall->subsystem));
|
2004-12-14 18:49:20 +03:00
|
|
|
syscall->hook = hook;
|
|
|
|
syscall->version = version;
|
|
|
|
syscall->flags = flags;
|
|
|
|
syscall->previous = previous;
|
|
|
|
list_add_item(&sGenericSyscalls, syscall);
|
|
|
|
|
|
|
|
if (previous != NULL)
|
|
|
|
list_remove_link(&previous->link);
|
|
|
|
|
|
|
|
status = B_OK;
|
|
|
|
|
|
|
|
out:
|
|
|
|
mutex_unlock(&sGenericSyscallLock);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
status_t
|
2004-12-14 20:09:06 +03:00
|
|
|
unregister_generic_syscall(const char *subsystem, uint32 version)
|
2004-12-14 18:49:20 +03:00
|
|
|
{
|
|
|
|
// ToDo: we should only remove the syscall with the matching version
|
|
|
|
generic_syscall *syscall;
|
|
|
|
status_t status;
|
|
|
|
|
|
|
|
mutex_lock(&sGenericSyscallLock);
|
|
|
|
|
|
|
|
syscall = find_generic_syscall(subsystem);
|
|
|
|
if (syscall != NULL) {
|
|
|
|
if (syscall->previous != NULL) {
|
|
|
|
// reestablish the old syscall
|
|
|
|
list_add_item(&sGenericSyscalls, syscall->previous);
|
|
|
|
}
|
|
|
|
list_remove_link(&syscall->link);
|
|
|
|
free(syscall);
|
|
|
|
status = B_OK;
|
|
|
|
} else
|
|
|
|
status = B_NAME_NOT_FOUND;
|
|
|
|
|
|
|
|
mutex_unlock(&sGenericSyscallLock);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2008-01-18 05:13:27 +03:00
|
|
|
|
|
|
|
// #pragma mark - syscall tracing
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SYSCALL_TRACING
|
|
|
|
|
|
|
|
namespace SyscallTracing {
|
|
|
|
|
|
|
|
|
|
|
|
static const char*
|
|
|
|
get_syscall_name(uint32 syscall)
|
|
|
|
{
|
|
|
|
if (syscall >= (uint32)kSyscallCount)
|
|
|
|
return "<invalid syscall number>";
|
|
|
|
|
2008-01-21 18:29:00 +03:00
|
|
|
return kExtendedSyscallInfos[syscall].name;
|
2008-01-18 05:13:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class PreSyscall : public AbstractTraceEntry {
|
|
|
|
public:
|
|
|
|
PreSyscall(uint32 syscall, const void* parameters)
|
|
|
|
:
|
|
|
|
fSyscall(syscall),
|
|
|
|
fParameters(NULL)
|
|
|
|
{
|
|
|
|
if (syscall < (uint32)kSyscallCount) {
|
|
|
|
fParameters = alloc_tracing_buffer_memcpy(parameters,
|
|
|
|
kSyscallInfos[syscall].parameter_size, false);
|
2008-01-18 22:33:09 +03:00
|
|
|
|
|
|
|
// copy string parameters, if any
|
2008-02-01 15:15:00 +03:00
|
|
|
if (fParameters != NULL && syscall != SYSCALL_KTRACE_OUTPUT) {
|
2008-01-18 22:33:09 +03:00
|
|
|
int32 stringIndex = 0;
|
2008-01-21 18:29:00 +03:00
|
|
|
const extended_syscall_info& syscallInfo
|
|
|
|
= kExtendedSyscallInfos[fSyscall];
|
|
|
|
for (int i = 0; i < syscallInfo.parameter_count; i++) {
|
2008-01-18 22:33:09 +03:00
|
|
|
const syscall_parameter_info& paramInfo
|
2008-01-21 18:29:00 +03:00
|
|
|
= syscallInfo.parameters[i];
|
2008-01-18 22:33:09 +03:00
|
|
|
if (paramInfo.type != B_STRING_TYPE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const uint8* data
|
|
|
|
= (uint8*)fParameters + paramInfo.offset;
|
|
|
|
if (stringIndex < MAX_PARAM_STRINGS) {
|
|
|
|
fParameterStrings[stringIndex++]
|
|
|
|
= alloc_tracing_buffer_strcpy(
|
|
|
|
*(const char**)data, 64, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-01-18 05:13:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Initialized();
|
|
|
|
}
|
|
|
|
|
2008-01-19 01:17:15 +03:00
|
|
|
virtual void AddDump(TraceOutput& out)
|
2008-01-18 05:13:27 +03:00
|
|
|
{
|
2008-01-19 01:17:15 +03:00
|
|
|
out.Print("syscall pre: %s(", get_syscall_name(fSyscall));
|
2008-01-18 05:13:27 +03:00
|
|
|
|
|
|
|
if (fParameters != NULL) {
|
2008-01-18 22:33:09 +03:00
|
|
|
int32 stringIndex = 0;
|
2008-01-21 18:29:00 +03:00
|
|
|
const extended_syscall_info& syscallInfo
|
|
|
|
= kExtendedSyscallInfos[fSyscall];
|
|
|
|
for (int i = 0; i < syscallInfo.parameter_count; i++) {
|
2008-01-18 22:33:09 +03:00
|
|
|
const syscall_parameter_info& paramInfo
|
2008-01-21 18:29:00 +03:00
|
|
|
= syscallInfo.parameters[i];
|
2008-01-18 22:33:09 +03:00
|
|
|
const uint8* data = (uint8*)fParameters + paramInfo.offset;
|
|
|
|
uint64 value = 0;
|
|
|
|
bool printValue = true;
|
|
|
|
switch (paramInfo.type) {
|
|
|
|
case B_INT8_TYPE:
|
|
|
|
value = *(uint8*)data;
|
|
|
|
break;
|
|
|
|
case B_INT16_TYPE:
|
|
|
|
value = *(uint16*)data;
|
|
|
|
break;
|
|
|
|
case B_INT32_TYPE:
|
|
|
|
value = *(uint32*)data;
|
|
|
|
break;
|
|
|
|
case B_INT64_TYPE:
|
|
|
|
value = *(uint64*)data;
|
|
|
|
break;
|
|
|
|
case B_POINTER_TYPE:
|
|
|
|
value = (uint64)*(void**)data;
|
|
|
|
break;
|
|
|
|
case B_STRING_TYPE:
|
2008-02-01 15:15:00 +03:00
|
|
|
if (stringIndex < MAX_PARAM_STRINGS
|
|
|
|
&& fSyscall != SYSCALL_KTRACE_OUTPUT) {
|
2008-01-19 01:17:15 +03:00
|
|
|
out.Print("%s\"%s\"",
|
2008-01-18 22:33:09 +03:00
|
|
|
(i == 0 ? "" : ", "),
|
|
|
|
fParameterStrings[stringIndex++]);
|
|
|
|
printValue = false;
|
|
|
|
} else
|
|
|
|
value = (uint64)*(void**)data;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-01-19 01:17:15 +03:00
|
|
|
if (printValue)
|
|
|
|
out.Print("%s0x%llx", (i == 0 ? "" : ", "), value);
|
2008-01-18 05:13:27 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-01-19 01:17:15 +03:00
|
|
|
out.Print(")");
|
2008-01-18 05:13:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2008-01-18 22:33:09 +03:00
|
|
|
enum { MAX_PARAM_STRINGS = 3 };
|
|
|
|
|
|
|
|
uint32 fSyscall;
|
|
|
|
void* fParameters;
|
|
|
|
const char* fParameterStrings[MAX_PARAM_STRINGS];
|
2008-01-18 05:13:27 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class PostSyscall : public AbstractTraceEntry {
|
|
|
|
public:
|
|
|
|
PostSyscall(uint32 syscall, uint64 returnValue)
|
|
|
|
:
|
|
|
|
fSyscall(syscall),
|
|
|
|
fReturnValue(returnValue)
|
|
|
|
{
|
|
|
|
Initialized();
|
2008-02-07 19:04:24 +03:00
|
|
|
#if 0
|
2008-02-07 22:07:26 +03:00
|
|
|
if (syscall < (uint32)kSyscallCount
|
|
|
|
&& returnValue != (returnValue & 0xffffffff)
|
2008-02-07 19:04:24 +03:00
|
|
|
&& kExtendedSyscallInfos[syscall].return_type.size <= 4) {
|
|
|
|
panic("syscall return value 64 bit although it should be 32 "
|
|
|
|
"bit");
|
|
|
|
}
|
|
|
|
#endif
|
2008-01-18 05:13:27 +03:00
|
|
|
}
|
|
|
|
|
2008-01-19 01:17:15 +03:00
|
|
|
virtual void AddDump(TraceOutput& out)
|
2008-01-18 05:13:27 +03:00
|
|
|
{
|
2008-01-19 01:17:15 +03:00
|
|
|
out.Print("syscall post: %s() -> 0x%llx",
|
2008-01-18 05:13:27 +03:00
|
|
|
get_syscall_name(fSyscall), fReturnValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
uint32 fSyscall;
|
|
|
|
uint64 fReturnValue;
|
|
|
|
};
|
|
|
|
|
2008-01-19 01:17:15 +03:00
|
|
|
} // namespace SyscallTracing
|
2008-01-18 05:13:27 +03:00
|
|
|
|
|
|
|
|
|
|
|
extern "C" void trace_pre_syscall(uint32 syscallNumber, const void* parameters);
|
|
|
|
|
|
|
|
void
|
|
|
|
trace_pre_syscall(uint32 syscallNumber, const void* parameters)
|
|
|
|
{
|
2008-03-15 00:58:17 +03:00
|
|
|
#ifdef SYSCALL_TRACING_IGNORE_KTRACE_OUTPUT
|
|
|
|
if (syscallNumber != SYSCALL_KTRACE_OUTPUT)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
new(std::nothrow) SyscallTracing::PreSyscall(syscallNumber, parameters);
|
|
|
|
}
|
2008-01-18 05:13:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
extern "C" void trace_post_syscall(int syscallNumber, uint64 returnValue);
|
|
|
|
|
|
|
|
void
|
|
|
|
trace_post_syscall(int syscallNumber, uint64 returnValue)
|
|
|
|
{
|
2008-03-15 00:58:17 +03:00
|
|
|
#ifdef SYSCALL_TRACING_IGNORE_KTRACE_OUTPUT
|
|
|
|
if (syscallNumber != SYSCALL_KTRACE_OUTPUT)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
new(std::nothrow) SyscallTracing::PostSyscall(syscallNumber,
|
|
|
|
returnValue);
|
|
|
|
}
|
2008-01-18 05:13:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#endif // SYSCALL_TRACING
|
|
|
|
|
|
|
|
|
2007-02-18 07:53:15 +03:00
|
|
|
/*
|
|
|
|
* kSyscallCount and kSyscallInfos here
|
|
|
|
*/
|
|
|
|
// generated by gensyscalls
|
2005-02-11 07:56:11 +03:00
|
|
|
#include "syscall_table.h"
|