forked from Minki/linux
0d55303c51
All the current architecture specific defines for these are the same. Refactor these common defines to a common header file. The new common linux/compat_time.h is also useful as it will eventually be used to hold all the defines that are needed for compat time types that support non y2038 safe types. New architectures need not have to define these new types as they will only use new y2038 safe syscalls. This file can be deleted after y2038 when we stop supporting non y2038 safe syscalls. The patch also requires an operation similar to: git grep "asm/compat\.h" | cut -d ":" -f 1 | xargs -n 1 sed -i -e "s%asm/compat.h%linux/compat.h%g" Cc: acme@kernel.org Cc: benh@kernel.crashing.org Cc: borntraeger@de.ibm.com Cc: catalin.marinas@arm.com Cc: cmetcalf@mellanox.com Cc: cohuck@redhat.com Cc: davem@davemloft.net Cc: deller@gmx.de Cc: devel@driverdev.osuosl.org Cc: gerald.schaefer@de.ibm.com Cc: gregkh@linuxfoundation.org Cc: heiko.carstens@de.ibm.com Cc: hoeppner@linux.vnet.ibm.com Cc: hpa@zytor.com Cc: jejb@parisc-linux.org Cc: jwi@linux.vnet.ibm.com Cc: linux-kernel@vger.kernel.org Cc: linux-mips@linux-mips.org Cc: linux-parisc@vger.kernel.org Cc: linuxppc-dev@lists.ozlabs.org Cc: linux-s390@vger.kernel.org Cc: mark.rutland@arm.com Cc: mingo@redhat.com Cc: mpe@ellerman.id.au Cc: oberpar@linux.vnet.ibm.com Cc: oprofile-list@lists.sf.net Cc: paulus@samba.org Cc: peterz@infradead.org Cc: ralf@linux-mips.org Cc: rostedt@goodmis.org Cc: rric@kernel.org Cc: schwidefsky@de.ibm.com Cc: sebott@linux.vnet.ibm.com Cc: sparclinux@vger.kernel.org Cc: sth@linux.vnet.ibm.com Cc: ubraun@linux.vnet.ibm.com Cc: will.deacon@arm.com Cc: x86@kernel.org Signed-off-by: Arnd Bergmann <arnd@arndb.de> Signed-off-by: Deepa Dinamani <deepa.kernel@gmail.com> Acked-by: Steven Rostedt (VMware) <rostedt@goodmis.org> Acked-by: Catalin Marinas <catalin.marinas@arm.com> Acked-by: James Hogan <jhogan@kernel.org> Acked-by: Helge Deller <deller@gmx.de> Signed-off-by: Arnd Bergmann <arnd@arndb.de>
149 lines
3.3 KiB
C
149 lines
3.3 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Hypervisor filesystem for Linux on s390.
|
|
* Set Partition-Resource Parameter interface.
|
|
*
|
|
* Copyright IBM Corp. 2013
|
|
* Author(s): Martin Schwidefsky <schwidefsky@de.ibm.com>
|
|
*/
|
|
|
|
#include <linux/compat.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/gfp.h>
|
|
#include <linux/string.h>
|
|
#include <linux/types.h>
|
|
#include <linux/uaccess.h>
|
|
#include <asm/diag.h>
|
|
#include <asm/sclp.h>
|
|
#include "hypfs.h"
|
|
|
|
#define DIAG304_SET_WEIGHTS 0
|
|
#define DIAG304_QUERY_PRP 1
|
|
#define DIAG304_SET_CAPPING 2
|
|
|
|
#define DIAG304_CMD_MAX 2
|
|
|
|
static inline unsigned long __hypfs_sprp_diag304(void *data, unsigned long cmd)
|
|
{
|
|
register unsigned long _data asm("2") = (unsigned long) data;
|
|
register unsigned long _rc asm("3");
|
|
register unsigned long _cmd asm("4") = cmd;
|
|
|
|
asm volatile("diag %1,%2,0x304\n"
|
|
: "=d" (_rc) : "d" (_data), "d" (_cmd) : "memory");
|
|
|
|
return _rc;
|
|
}
|
|
|
|
static unsigned long hypfs_sprp_diag304(void *data, unsigned long cmd)
|
|
{
|
|
diag_stat_inc(DIAG_STAT_X304);
|
|
return __hypfs_sprp_diag304(data, cmd);
|
|
}
|
|
|
|
static void hypfs_sprp_free(const void *data)
|
|
{
|
|
free_page((unsigned long) data);
|
|
}
|
|
|
|
static int hypfs_sprp_create(void **data_ptr, void **free_ptr, size_t *size)
|
|
{
|
|
unsigned long rc;
|
|
void *data;
|
|
|
|
data = (void *) get_zeroed_page(GFP_KERNEL);
|
|
if (!data)
|
|
return -ENOMEM;
|
|
rc = hypfs_sprp_diag304(data, DIAG304_QUERY_PRP);
|
|
if (rc != 1) {
|
|
*data_ptr = *free_ptr = NULL;
|
|
*size = 0;
|
|
free_page((unsigned long) data);
|
|
return -EIO;
|
|
}
|
|
*data_ptr = *free_ptr = data;
|
|
*size = PAGE_SIZE;
|
|
return 0;
|
|
}
|
|
|
|
static int __hypfs_sprp_ioctl(void __user *user_area)
|
|
{
|
|
struct hypfs_diag304 diag304;
|
|
unsigned long cmd;
|
|
void __user *udata;
|
|
void *data;
|
|
int rc;
|
|
|
|
if (copy_from_user(&diag304, user_area, sizeof(diag304)))
|
|
return -EFAULT;
|
|
if ((diag304.args[0] >> 8) != 0 || diag304.args[1] > DIAG304_CMD_MAX)
|
|
return -EINVAL;
|
|
|
|
data = (void *) get_zeroed_page(GFP_KERNEL | GFP_DMA);
|
|
if (!data)
|
|
return -ENOMEM;
|
|
|
|
udata = (void __user *)(unsigned long) diag304.data;
|
|
if (diag304.args[1] == DIAG304_SET_WEIGHTS ||
|
|
diag304.args[1] == DIAG304_SET_CAPPING)
|
|
if (copy_from_user(data, udata, PAGE_SIZE)) {
|
|
rc = -EFAULT;
|
|
goto out;
|
|
}
|
|
|
|
cmd = *(unsigned long *) &diag304.args[0];
|
|
diag304.rc = hypfs_sprp_diag304(data, cmd);
|
|
|
|
if (diag304.args[1] == DIAG304_QUERY_PRP)
|
|
if (copy_to_user(udata, data, PAGE_SIZE)) {
|
|
rc = -EFAULT;
|
|
goto out;
|
|
}
|
|
|
|
rc = copy_to_user(user_area, &diag304, sizeof(diag304)) ? -EFAULT : 0;
|
|
out:
|
|
free_page((unsigned long) data);
|
|
return rc;
|
|
}
|
|
|
|
static long hypfs_sprp_ioctl(struct file *file, unsigned int cmd,
|
|
unsigned long arg)
|
|
{
|
|
void __user *argp;
|
|
|
|
if (!capable(CAP_SYS_ADMIN))
|
|
return -EACCES;
|
|
if (is_compat_task())
|
|
argp = compat_ptr(arg);
|
|
else
|
|
argp = (void __user *) arg;
|
|
switch (cmd) {
|
|
case HYPFS_DIAG304:
|
|
return __hypfs_sprp_ioctl(argp);
|
|
default: /* unknown ioctl number */
|
|
return -ENOTTY;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static struct hypfs_dbfs_file hypfs_sprp_file = {
|
|
.name = "diag_304",
|
|
.data_create = hypfs_sprp_create,
|
|
.data_free = hypfs_sprp_free,
|
|
.unlocked_ioctl = hypfs_sprp_ioctl,
|
|
};
|
|
|
|
int hypfs_sprp_init(void)
|
|
{
|
|
if (!sclp.has_sprp)
|
|
return 0;
|
|
return hypfs_dbfs_create_file(&hypfs_sprp_file);
|
|
}
|
|
|
|
void hypfs_sprp_exit(void)
|
|
{
|
|
if (!sclp.has_sprp)
|
|
return;
|
|
hypfs_dbfs_remove_file(&hypfs_sprp_file);
|
|
}
|