aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/sysctl_binary.c
diff options
context:
space:
mode:
authorEric W. Biederman <ebiederm@xmission.com>2009-04-03 03:36:27 -0400
committerEric W. Biederman <ebiederm@xmission.com>2009-11-06 06:52:55 -0500
commitda3f6f9b3e0d1e73975ca81ae124406bf1587d40 (patch)
treeab2e5bb6366b0bdc00ec1bc5c02e610c1c77195d /kernel/sysctl_binary.c
parent2830b68361a9f58354ad043c6d85043ea917f907 (diff)
sysctl: Introduce a generic compat sysctl sysctl
This uses compat_alloc_userspace to remove the various hacks to allow do_sysctl to write to throuh oldlenp. The rest of our mature compat syscall helper facitilies are used as well to ensure we have a nice clean maintainable compat syscall that can be used on all architectures. The motiviation for a generic compat sysctl (besides the obvious hack removal) is to reduce the number of compat sysctl defintions out there so I can refactor the binary sysctl implementation. ppc already used the name compat_sys_sysctl so I remove the ppcs version here. Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org> Cc: Paul Mackerras <paulus@samba.org> Acked-by: Arnd Bergmann <arnd@arndb.de> Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
Diffstat (limited to 'kernel/sysctl_binary.c')
-rw-r--r--kernel/sysctl_binary.c50
1 files changed, 50 insertions, 0 deletions
diff --git a/kernel/sysctl_binary.c b/kernel/sysctl_binary.c
index 930a31cd708b..775cc49da622 100644
--- a/kernel/sysctl_binary.c
+++ b/kernel/sysctl_binary.c
@@ -176,3 +176,53 @@ SYSCALL_DEFINE1(sysctl, struct __sysctl_args __user *, args)
176 176
177 return error; 177 return error;
178} 178}
179
180#ifdef CONFIG_COMPAT
181#include <asm/compat.h>
182
183struct compat_sysctl_args {
184 compat_uptr_t name;
185 int nlen;
186 compat_uptr_t oldval;
187 compat_uptr_t oldlenp;
188 compat_uptr_t newval;
189 compat_size_t newlen;
190 compat_ulong_t __unused[4];
191};
192
193asmlinkage long compat_sys_sysctl(struct compat_sysctl_args __user *args)
194{
195 struct compat_sysctl_args tmp;
196 compat_size_t __user *compat_oldlenp;
197 size_t __user *oldlenp = NULL;
198 size_t oldlen = 0;
199 ssize_t result;
200
201 if (copy_from_user(&tmp, args, sizeof(tmp)))
202 return -EFAULT;
203
204 compat_oldlenp = compat_ptr(tmp.oldlenp);
205 if (compat_oldlenp) {
206 oldlenp = compat_alloc_user_space(sizeof(*compat_oldlenp));
207
208 if (get_user(oldlen, compat_oldlenp) ||
209 put_user(oldlen, oldlenp))
210 return -EFAULT;
211 }
212
213 lock_kernel();
214 result = do_sysctl(compat_ptr(tmp.name), tmp.nlen,
215 compat_ptr(tmp.oldval), oldlenp,
216 compat_ptr(tmp.newval), tmp.newlen);
217 unlock_kernel();
218
219 if (oldlenp && !result) {
220 if (get_user(oldlen, oldlenp) ||
221 put_user(oldlen, compat_oldlenp))
222 return -EFAULT;
223 }
224
225 return result;
226}
227
228#endif /* CONFIG_COMPAT */