syscall.h 2.9 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
/*
 * Copyright (C) 2008-2009 Red Hat, Inc.  All rights reserved.
 * Copyright 2010 Tilera Corporation. All Rights Reserved.
 *
 *   This program is free software; you can redistribute it and/or
 *   modify it under the terms of the GNU General Public License
 *   as published by the Free Software Foundation, version 2.
 *
 *   This program is distributed in the hope that it will be useful, but
 *   WITHOUT ANY WARRANTY; without even the implied warranty of
 *   MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE, GOOD TITLE or
 *   NON INFRINGEMENT.  See the GNU General Public License for
 *   more details.
 *
 * See asm-generic/syscall.h for descriptions of what we must do here.
 */

#ifndef _ASM_TILE_SYSCALL_H
#define _ASM_TILE_SYSCALL_H

#include <linux/sched.h>
#include <linux/err.h>
23 24
#include <linux/audit.h>
#include <linux/compat.h>
25 26
#include <arch/abi.h>

27 28 29 30 31 32
/* The array of function pointers for syscalls. */
extern void *sys_call_table[];
#ifdef CONFIG_COMPAT
extern void *compat_sys_call_table[];
#endif

33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65
/*
 * Only the low 32 bits of orig_r0 are meaningful, so we return int.
 * This importantly ignores the high bits on 64-bit, so comparisons
 * sign-extend the low 32 bits.
 */
static inline int syscall_get_nr(struct task_struct *t, struct pt_regs *regs)
{
	return regs->regs[TREG_SYSCALL_NR];
}

static inline void syscall_rollback(struct task_struct *task,
				    struct pt_regs *regs)
{
	regs->regs[0] = regs->orig_r0;
}

static inline long syscall_get_error(struct task_struct *task,
				     struct pt_regs *regs)
{
	unsigned long error = regs->regs[0];
	return IS_ERR_VALUE(error) ? error : 0;
}

static inline long syscall_get_return_value(struct task_struct *task,
					    struct pt_regs *regs)
{
	return regs->regs[0];
}

static inline void syscall_set_return_value(struct task_struct *task,
					    struct pt_regs *regs,
					    int error, long val)
{
66 67 68 69 70 71 72 73 74
	if (error) {
		/* R0 is the passed-in negative error, R1 is positive. */
		regs->regs[0] = error;
		regs->regs[1] = -error;
	} else {
		/* R1 set to zero to indicate no error. */
		regs->regs[0] = val;
		regs->regs[1] = 0;
	}
75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
}

static inline void syscall_get_arguments(struct task_struct *task,
					 struct pt_regs *regs,
					 unsigned int i, unsigned int n,
					 unsigned long *args)
{
	BUG_ON(i + n > 6);
	memcpy(args, &regs[i], n * sizeof(args[0]));
}

static inline void syscall_set_arguments(struct task_struct *task,
					 struct pt_regs *regs,
					 unsigned int i, unsigned int n,
					 const unsigned long *args)
{
	BUG_ON(i + n > 6);
	memcpy(&regs[i], args, n * sizeof(args[0]));
}

95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
/*
 * We don't care about endianness (__AUDIT_ARCH_LE bit) here because
 * tile has the same system calls both on little- and big- endian.
 */
static inline int syscall_get_arch(void)
{
	if (is_compat_task())
		return AUDIT_ARCH_TILEGX32;

#ifdef CONFIG_TILEGX
	return AUDIT_ARCH_TILEGX;
#else
	return AUDIT_ARCH_TILEPRO;
#endif
}

111
#endif	/* _ASM_TILE_SYSCALL_H */