compat.c 3.8 KB
Newer Older
1
/* 32-bit compatibility syscall for 64-bit systems
L
Linus Torvalds 已提交
2
 *
3
 * Copyright (C) 2004-5 Red Hat, Inc. All Rights Reserved.
L
Linus Torvalds 已提交
4 5 6 7 8 9 10 11 12 13 14
 * Written by David Howells (dhowells@redhat.com)
 *
 * 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; either version
 * 2 of the License, or (at your option) any later version.
 */

#include <linux/syscalls.h>
#include <linux/keyctl.h>
#include <linux/compat.h>
15
#include <linux/slab.h>
L
Linus Torvalds 已提交
16 17
#include "internal.h"

18 19 20 21 22 23 24 25 26
/*
 * Instantiate a key with the specified compatibility multipart payload and
 * link the key into the destination keyring if one is given.
 *
 * The caller must have the appropriate instantiation permit set for this to
 * work (see keyctl_assume_authority).  No other permissions are required.
 *
 * If successful, 0 will be returned.
 */
D
David Howells 已提交
27
static long compat_keyctl_instantiate_key_iov(
28 29 30 31 32 33
	key_serial_t id,
	const struct compat_iovec __user *_payload_iov,
	unsigned ioc,
	key_serial_t ringid)
{
	struct iovec iovstack[UIO_FASTIOV], *iov = iovstack;
34
	struct iov_iter from;
35 36
	long ret;

37 38
	if (!_payload_iov)
		ioc = 0;
39

40 41 42
	ret = compat_import_iovec(WRITE, _payload_iov, ioc,
				  ARRAY_SIZE(iovstack), &iov,
				  &from);
43
	if (ret < 0)
44
		return ret;
45

46 47
	ret = keyctl_instantiate_key_common(id, &from, ringid);
	kfree(iov);
48 49 50
	return ret;
}

L
Linus Torvalds 已提交
51
/*
52 53 54 55 56 57
 * The key control system call, 32-bit compatibility version for 64-bit archs
 *
 * This should only be called if the 64-bit arch uses weird pointers in 32-bit
 * mode or doesn't guarantee that the top 32-bits of the argument registers on
 * taking a 32-bit syscall are zero.  If you can, you should call sys_keyctl()
 * directly.
L
Linus Torvalds 已提交
58
 */
59 60
COMPAT_SYSCALL_DEFINE5(keyctl, u32, option,
		       u32, arg2, u32, arg3, u32, arg4, u32, arg5)
L
Linus Torvalds 已提交
61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106
{
	switch (option) {
	case KEYCTL_GET_KEYRING_ID:
		return keyctl_get_keyring_ID(arg2, arg3);

	case KEYCTL_JOIN_SESSION_KEYRING:
		return keyctl_join_session_keyring(compat_ptr(arg2));

	case KEYCTL_UPDATE:
		return keyctl_update_key(arg2, compat_ptr(arg3), arg4);

	case KEYCTL_REVOKE:
		return keyctl_revoke_key(arg2);

	case KEYCTL_DESCRIBE:
		return keyctl_describe_key(arg2, compat_ptr(arg3), arg4);

	case KEYCTL_CLEAR:
		return keyctl_keyring_clear(arg2);

	case KEYCTL_LINK:
		return keyctl_keyring_link(arg2, arg3);

	case KEYCTL_UNLINK:
		return keyctl_keyring_unlink(arg2, arg3);

	case KEYCTL_SEARCH:
		return keyctl_keyring_search(arg2, compat_ptr(arg3),
					     compat_ptr(arg4), arg5);

	case KEYCTL_READ:
		return keyctl_read_key(arg2, compat_ptr(arg3), arg4);

	case KEYCTL_CHOWN:
		return keyctl_chown_key(arg2, arg3, arg4);

	case KEYCTL_SETPERM:
		return keyctl_setperm_key(arg2, arg3);

	case KEYCTL_INSTANTIATE:
		return keyctl_instantiate_key(arg2, compat_ptr(arg3), arg4,
					      arg5);

	case KEYCTL_NEGATE:
		return keyctl_negate_key(arg2, arg3, arg4);

107 108 109
	case KEYCTL_SET_REQKEY_KEYRING:
		return keyctl_set_reqkey_keyring(arg2);

110 111 112
	case KEYCTL_SET_TIMEOUT:
		return keyctl_set_timeout(arg2, arg3);

113 114 115
	case KEYCTL_ASSUME_AUTHORITY:
		return keyctl_assume_authority(arg2);

116 117 118
	case KEYCTL_GET_SECURITY:
		return keyctl_get_security(arg2, compat_ptr(arg3), arg4);

119 120 121
	case KEYCTL_SESSION_TO_PARENT:
		return keyctl_session_to_parent();

122 123 124
	case KEYCTL_REJECT:
		return keyctl_reject_key(arg2, arg3, arg4, arg5);

125 126 127 128
	case KEYCTL_INSTANTIATE_IOV:
		return compat_keyctl_instantiate_key_iov(
			arg2, compat_ptr(arg3), arg4, arg5);

D
David Howells 已提交
129 130 131
	case KEYCTL_INVALIDATE:
		return keyctl_invalidate_key(arg2);

132 133 134
	case KEYCTL_GET_PERSISTENT:
		return keyctl_get_persistent(arg2, arg3);

135 136
	case KEYCTL_DH_COMPUTE:
		return keyctl_dh_compute(compat_ptr(arg2), compat_ptr(arg3),
137
					 arg4, compat_ptr(arg5));
138

139 140 141 142
	case KEYCTL_RESTRICT_KEYRING:
		return keyctl_restrict_keyring(arg2, compat_ptr(arg3),
					       compat_ptr(arg4));

L
Linus Torvalds 已提交
143 144 145
	default:
		return -EOPNOTSUPP;
	}
146
}