Blame view

security/keys/compat.c 3.62 KB
973c9f4f4   David Howells   KEYS: Fix up comm...
1
  /* 32-bit compatibility syscall for 64-bit systems
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2
   *
3e30148c3   David Howells   [PATCH] Keys: Mak...
3
   * Copyright (C) 2004-5 Red Hat, Inc. All Rights Reserved.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
4
5
6
7
8
9
10
   * 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.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
11
12
13
  #include <linux/syscalls.h>
  #include <linux/keyctl.h>
  #include <linux/compat.h>
ee009e4a0   David Howells   KEYS: Add an iove...
14
  #include <linux/slab.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
15
  #include "internal.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
16
  /*
ee009e4a0   David Howells   KEYS: Add an iove...
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
   * 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.
   */
  long compat_keyctl_instantiate_key_iov(
  	key_serial_t id,
  	const struct compat_iovec __user *_payload_iov,
  	unsigned ioc,
  	key_serial_t ringid)
  {
  	struct iovec iovstack[UIO_FASTIOV], *iov = iovstack;
  	long ret;
  
  	if (_payload_iov == 0 || ioc == 0)
  		goto no_payload;
  
  	ret = compat_rw_copy_check_uvector(WRITE, _payload_iov, ioc,
  					   ARRAY_SIZE(iovstack),
fcf634098   Christopher Yeoh   Cross Memory Attach
39
  					   iovstack, &iov, 1);
ee009e4a0   David Howells   KEYS: Add an iove...
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
  	if (ret < 0)
  		return ret;
  	if (ret == 0)
  		goto no_payload_free;
  
  	ret = keyctl_instantiate_key_common(id, iov, ioc, ret, ringid);
  
  	if (iov != iovstack)
  		kfree(iov);
  	return ret;
  
  no_payload_free:
  	if (iov != iovstack)
  		kfree(iov);
  no_payload:
  	return keyctl_instantiate_key_common(id, NULL, 0, 0, ringid);
  }
  
  /*
973c9f4f4   David Howells   KEYS: Fix up comm...
59
60
61
62
63
64
   * 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.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
65
66
   */
  asmlinkage long compat_sys_keyctl(u32 option,
3e30148c3   David Howells   [PATCH] Keys: Mak...
67
  				  u32 arg2, u32 arg3, u32 arg4, u32 arg5)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
107
108
109
110
111
112
  {
  	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);
3e30148c3   David Howells   [PATCH] Keys: Mak...
113
114
  	case KEYCTL_SET_REQKEY_KEYRING:
  		return keyctl_set_reqkey_keyring(arg2);
017679c4d   David Howells   [PATCH] keys: Per...
115
116
  	case KEYCTL_SET_TIMEOUT:
  		return keyctl_set_timeout(arg2, arg3);
b5f545c88   David Howells   [PATCH] keys: Per...
117
118
  	case KEYCTL_ASSUME_AUTHORITY:
  		return keyctl_assume_authority(arg2);
70a5bb72b   David Howells   keys: add keyctl ...
119
120
  	case KEYCTL_GET_SECURITY:
  		return keyctl_get_security(arg2, compat_ptr(arg3), arg4);
ee18d64c1   David Howells   KEYS: Add a keyct...
121
122
  	case KEYCTL_SESSION_TO_PARENT:
  		return keyctl_session_to_parent();
fdd1b9458   David Howells   KEYS: Add a new k...
123
124
  	case KEYCTL_REJECT:
  		return keyctl_reject_key(arg2, arg3, arg4, arg5);
ee009e4a0   David Howells   KEYS: Add an iove...
125
126
127
  	case KEYCTL_INSTANTIATE_IOV:
  		return compat_keyctl_instantiate_key_iov(
  			arg2, compat_ptr(arg3), arg4, arg5);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
128
129
130
  	default:
  		return -EOPNOTSUPP;
  	}
a8b17ed01   David Howells   KEYS: Do some sty...
131
  }