blob: 70ec3156f45a7077ef3e2c51bfe9a23a2647cc87 [file] [log] [blame]
Christopher Yeohfcf63402011-10-31 17:06:39 -07001/*
2 * linux/mm/process_vm_access.c
3 *
4 * Copyright (C) 2010-2011 Christopher Yeoh <cyeoh@au1.ibm.com>, IBM Corp.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version
9 * 2 of the License, or (at your option) any later version.
10 */
11
12#include <linux/mm.h>
13#include <linux/uio.h>
14#include <linux/sched.h>
15#include <linux/highmem.h>
16#include <linux/ptrace.h>
17#include <linux/slab.h>
18#include <linux/syscalls.h>
19
20#ifdef CONFIG_COMPAT
21#include <linux/compat.h>
22#endif
23
24/**
25 * process_vm_rw_pages - read/write pages from task specified
26 * @task: task to read/write from
27 * @mm: mm for task
28 * @process_pages: struct pages area that can store at least
29 * nr_pages_to_copy struct page pointers
30 * @pa: address of page in task to start copying from/to
31 * @start_offset: offset in page to start copying from/to
32 * @len: number of bytes to copy
33 * @lvec: iovec array specifying where to copy to/from
34 * @lvec_cnt: number of elements in iovec array
35 * @lvec_current: index in iovec array we are up to
36 * @lvec_offset: offset in bytes from current iovec iov_base we are up to
37 * @vm_write: 0 means copy from, 1 means copy to
38 * @nr_pages_to_copy: number of pages to copy
39 * @bytes_copied: returns number of bytes successfully copied
40 * Returns 0 on success, error code otherwise
41 */
42static int process_vm_rw_pages(struct task_struct *task,
43 struct mm_struct *mm,
44 struct page **process_pages,
45 unsigned long pa,
46 unsigned long start_offset,
47 unsigned long len,
Al Viro9f78bdf2014-02-05 11:51:53 -050048 struct iov_iter *iter,
Christopher Yeohfcf63402011-10-31 17:06:39 -070049 int vm_write,
50 unsigned int nr_pages_to_copy,
51 ssize_t *bytes_copied)
52{
53 int pages_pinned;
Christopher Yeohfcf63402011-10-31 17:06:39 -070054 int pgs_copied = 0;
55 int j;
56 int ret;
57 ssize_t bytes_to_copy;
58 ssize_t rc = 0;
59
60 *bytes_copied = 0;
61
62 /* Get the pages we're interested in */
63 down_read(&mm->mmap_sem);
64 pages_pinned = get_user_pages(task, mm, pa,
65 nr_pages_to_copy,
66 vm_write, 0, process_pages, NULL);
67 up_read(&mm->mmap_sem);
68
69 if (pages_pinned != nr_pages_to_copy) {
70 rc = -EFAULT;
71 goto end;
72 }
73
74 /* Do the copy for each page */
75 for (pgs_copied = 0;
Al Viro240f3902014-02-05 12:14:11 -050076 (pgs_copied < nr_pages_to_copy) && iov_iter_count(iter);
Christopher Yeohfcf63402011-10-31 17:06:39 -070077 pgs_copied++) {
Al Viro240f3902014-02-05 12:14:11 -050078 struct page *page = process_pages[pgs_copied];
79 bytes_to_copy = min_t(ssize_t, PAGE_SIZE - start_offset, len);
Christopher Yeohfcf63402011-10-31 17:06:39 -070080
Al Viro240f3902014-02-05 12:14:11 -050081 if (vm_write) {
82 if (bytes_to_copy > iov_iter_count(iter))
83 bytes_to_copy = iov_iter_count(iter);
84 ret = iov_iter_copy_from_user(page,
85 iter, start_offset, bytes_to_copy);
86 iov_iter_advance(iter, ret);
87 set_page_dirty_lock(page);
Christopher Yeohfcf63402011-10-31 17:06:39 -070088 } else {
Al Viro240f3902014-02-05 12:14:11 -050089 ret = copy_page_to_iter(page, start_offset,
90 bytes_to_copy, iter);
Christopher Yeohfcf63402011-10-31 17:06:39 -070091 }
Al Viro240f3902014-02-05 12:14:11 -050092 *bytes_copied += ret;
93 len -= ret;
94 if (ret < bytes_to_copy && iov_iter_count(iter)) {
95 rc = -EFAULT;
96 break;
97 }
98 start_offset = 0;
Christopher Yeohfcf63402011-10-31 17:06:39 -070099 }
100
101end:
Al Viro240f3902014-02-05 12:14:11 -0500102 for (j = 0; j < pages_pinned; j++)
103 put_page(process_pages[j]);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700104 return rc;
105}
106
107/* Maximum number of pages kmalloc'd to hold struct page's during copy */
108#define PVM_MAX_KMALLOC_PAGES (PAGE_SIZE * 2)
109
110/**
111 * process_vm_rw_single_vec - read/write pages from task specified
112 * @addr: start memory address of target process
113 * @len: size of area to copy to/from
114 * @lvec: iovec array specifying where to copy to/from locally
115 * @lvec_cnt: number of elements in iovec array
116 * @lvec_current: index in iovec array we are up to
117 * @lvec_offset: offset in bytes from current iovec iov_base we are up to
118 * @process_pages: struct pages area that can store at least
119 * nr_pages_to_copy struct page pointers
120 * @mm: mm for task
121 * @task: task to read/write from
122 * @vm_write: 0 means copy from, 1 means copy to
123 * @bytes_copied: returns number of bytes successfully copied
124 * Returns 0 on success or on failure error code
125 */
126static int process_vm_rw_single_vec(unsigned long addr,
127 unsigned long len,
Al Viro9f78bdf2014-02-05 11:51:53 -0500128 struct iov_iter *iter,
Christopher Yeohfcf63402011-10-31 17:06:39 -0700129 struct page **process_pages,
130 struct mm_struct *mm,
131 struct task_struct *task,
132 int vm_write,
133 ssize_t *bytes_copied)
134{
135 unsigned long pa = addr & PAGE_MASK;
136 unsigned long start_offset = addr - pa;
137 unsigned long nr_pages;
138 ssize_t bytes_copied_loop;
139 ssize_t rc = 0;
140 unsigned long nr_pages_copied = 0;
141 unsigned long nr_pages_to_copy;
142 unsigned long max_pages_per_loop = PVM_MAX_KMALLOC_PAGES
143 / sizeof(struct pages *);
144
145 *bytes_copied = 0;
146
147 /* Work out address and page range required */
148 if (len == 0)
149 return 0;
150 nr_pages = (addr + len - 1) / PAGE_SIZE - addr / PAGE_SIZE + 1;
151
Al Viro240f3902014-02-05 12:14:11 -0500152 while ((nr_pages_copied < nr_pages) && iov_iter_count(iter)) {
Christopher Yeohfcf63402011-10-31 17:06:39 -0700153 nr_pages_to_copy = min(nr_pages - nr_pages_copied,
154 max_pages_per_loop);
155
156 rc = process_vm_rw_pages(task, mm, process_pages, pa,
Al Viro9f78bdf2014-02-05 11:51:53 -0500157 start_offset, len, iter,
Christopher Yeohfcf63402011-10-31 17:06:39 -0700158 vm_write, nr_pages_to_copy,
159 &bytes_copied_loop);
160 start_offset = 0;
161 *bytes_copied += bytes_copied_loop;
162
163 if (rc < 0) {
164 return rc;
165 } else {
166 len -= bytes_copied_loop;
167 nr_pages_copied += nr_pages_to_copy;
168 pa += nr_pages_to_copy * PAGE_SIZE;
169 }
170 }
171
172 return rc;
173}
174
175/* Maximum number of entries for process pages array
176 which lives on stack */
177#define PVM_MAX_PP_ARRAY_COUNT 16
178
179/**
180 * process_vm_rw_core - core of reading/writing pages from task specified
181 * @pid: PID of process to read/write from/to
182 * @lvec: iovec array specifying where to copy to/from locally
183 * @liovcnt: size of lvec array
184 * @rvec: iovec array specifying where to copy to/from in the other process
185 * @riovcnt: size of rvec array
186 * @flags: currently unused
187 * @vm_write: 0 if reading from other process, 1 if writing to other process
188 * Returns the number of bytes read/written or error code. May
189 * return less bytes than expected if an error occurs during the copying
190 * process.
191 */
Al Viro9f78bdf2014-02-05 11:51:53 -0500192static ssize_t process_vm_rw_core(pid_t pid, struct iov_iter *iter,
Christopher Yeohfcf63402011-10-31 17:06:39 -0700193 const struct iovec *rvec,
194 unsigned long riovcnt,
195 unsigned long flags, int vm_write)
196{
197 struct task_struct *task;
198 struct page *pp_stack[PVM_MAX_PP_ARRAY_COUNT];
199 struct page **process_pages = pp_stack;
200 struct mm_struct *mm;
201 unsigned long i;
202 ssize_t rc = 0;
203 ssize_t bytes_copied_loop;
204 ssize_t bytes_copied = 0;
205 unsigned long nr_pages = 0;
206 unsigned long nr_pages_iov;
Christopher Yeohfcf63402011-10-31 17:06:39 -0700207 ssize_t iov_len;
208
209 /*
210 * Work out how many pages of struct pages we're going to need
211 * when eventually calling get_user_pages
212 */
213 for (i = 0; i < riovcnt; i++) {
214 iov_len = rvec[i].iov_len;
215 if (iov_len > 0) {
216 nr_pages_iov = ((unsigned long)rvec[i].iov_base
217 + iov_len)
218 / PAGE_SIZE - (unsigned long)rvec[i].iov_base
219 / PAGE_SIZE + 1;
220 nr_pages = max(nr_pages, nr_pages_iov);
221 }
222 }
223
224 if (nr_pages == 0)
225 return 0;
226
227 if (nr_pages > PVM_MAX_PP_ARRAY_COUNT) {
228 /* For reliability don't try to kmalloc more than
229 2 pages worth */
230 process_pages = kmalloc(min_t(size_t, PVM_MAX_KMALLOC_PAGES,
231 sizeof(struct pages *)*nr_pages),
232 GFP_KERNEL);
233
234 if (!process_pages)
235 return -ENOMEM;
236 }
237
238 /* Get process information */
239 rcu_read_lock();
240 task = find_task_by_vpid(pid);
241 if (task)
242 get_task_struct(task);
243 rcu_read_unlock();
244 if (!task) {
245 rc = -ESRCH;
246 goto free_proc_pages;
247 }
248
Christopher Yeoh8cdb8782012-02-02 11:34:09 +1030249 mm = mm_access(task, PTRACE_MODE_ATTACH);
250 if (!mm || IS_ERR(mm)) {
251 rc = IS_ERR(mm) ? PTR_ERR(mm) : -ESRCH;
252 /*
253 * Explicitly map EACCES to EPERM as EPERM is a more a
254 * appropriate error code for process_vw_readv/writev
255 */
256 if (rc == -EACCES)
257 rc = -EPERM;
Christopher Yeohfcf63402011-10-31 17:06:39 -0700258 goto put_task_struct;
259 }
Christopher Yeohfcf63402011-10-31 17:06:39 -0700260
Al Viro240f3902014-02-05 12:14:11 -0500261 for (i = 0; i < riovcnt && iov_iter_count(iter); i++) {
Christopher Yeohfcf63402011-10-31 17:06:39 -0700262 rc = process_vm_rw_single_vec(
263 (unsigned long)rvec[i].iov_base, rvec[i].iov_len,
Al Viro9f78bdf2014-02-05 11:51:53 -0500264 iter, process_pages, mm, task, vm_write,
265 &bytes_copied_loop);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700266 bytes_copied += bytes_copied_loop;
267 if (rc != 0) {
268 /* If we have managed to copy any data at all then
269 we return the number of bytes copied. Otherwise
270 we return the error code */
271 if (bytes_copied)
272 rc = bytes_copied;
273 goto put_mm;
274 }
275 }
276
277 rc = bytes_copied;
278put_mm:
279 mmput(mm);
280
281put_task_struct:
282 put_task_struct(task);
283
284free_proc_pages:
285 if (process_pages != pp_stack)
286 kfree(process_pages);
287 return rc;
288}
289
290/**
291 * process_vm_rw - check iovecs before calling core routine
292 * @pid: PID of process to read/write from/to
293 * @lvec: iovec array specifying where to copy to/from locally
294 * @liovcnt: size of lvec array
295 * @rvec: iovec array specifying where to copy to/from in the other process
296 * @riovcnt: size of rvec array
297 * @flags: currently unused
298 * @vm_write: 0 if reading from other process, 1 if writing to other process
299 * Returns the number of bytes read/written or error code. May
300 * return less bytes than expected if an error occurs during the copying
301 * process.
302 */
303static ssize_t process_vm_rw(pid_t pid,
304 const struct iovec __user *lvec,
305 unsigned long liovcnt,
306 const struct iovec __user *rvec,
307 unsigned long riovcnt,
308 unsigned long flags, int vm_write)
309{
310 struct iovec iovstack_l[UIO_FASTIOV];
311 struct iovec iovstack_r[UIO_FASTIOV];
312 struct iovec *iov_l = iovstack_l;
313 struct iovec *iov_r = iovstack_r;
Al Viro9f78bdf2014-02-05 11:51:53 -0500314 struct iov_iter iter;
Christopher Yeohfcf63402011-10-31 17:06:39 -0700315 ssize_t rc;
316
317 if (flags != 0)
318 return -EINVAL;
319
320 /* Check iovecs */
321 if (vm_write)
322 rc = rw_copy_check_uvector(WRITE, lvec, liovcnt, UIO_FASTIOV,
Christopher Yeohac34ebb2012-05-31 16:26:42 -0700323 iovstack_l, &iov_l);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700324 else
325 rc = rw_copy_check_uvector(READ, lvec, liovcnt, UIO_FASTIOV,
Christopher Yeohac34ebb2012-05-31 16:26:42 -0700326 iovstack_l, &iov_l);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700327 if (rc <= 0)
328 goto free_iovecs;
329
Al Viro9f78bdf2014-02-05 11:51:53 -0500330 iov_iter_init(&iter, iov_l, liovcnt, rc, 0);
331
Christopher Yeohac34ebb2012-05-31 16:26:42 -0700332 rc = rw_copy_check_uvector(CHECK_IOVEC_ONLY, rvec, riovcnt, UIO_FASTIOV,
333 iovstack_r, &iov_r);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700334 if (rc <= 0)
335 goto free_iovecs;
336
Al Viro9f78bdf2014-02-05 11:51:53 -0500337 rc = process_vm_rw_core(pid, &iter, iov_r, riovcnt, flags, vm_write);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700338
339free_iovecs:
340 if (iov_r != iovstack_r)
341 kfree(iov_r);
342 if (iov_l != iovstack_l)
343 kfree(iov_l);
344
345 return rc;
346}
347
348SYSCALL_DEFINE6(process_vm_readv, pid_t, pid, const struct iovec __user *, lvec,
349 unsigned long, liovcnt, const struct iovec __user *, rvec,
350 unsigned long, riovcnt, unsigned long, flags)
351{
352 return process_vm_rw(pid, lvec, liovcnt, rvec, riovcnt, flags, 0);
353}
354
355SYSCALL_DEFINE6(process_vm_writev, pid_t, pid,
356 const struct iovec __user *, lvec,
357 unsigned long, liovcnt, const struct iovec __user *, rvec,
358 unsigned long, riovcnt, unsigned long, flags)
359{
360 return process_vm_rw(pid, lvec, liovcnt, rvec, riovcnt, flags, 1);
361}
362
363#ifdef CONFIG_COMPAT
364
365asmlinkage ssize_t
366compat_process_vm_rw(compat_pid_t pid,
367 const struct compat_iovec __user *lvec,
368 unsigned long liovcnt,
369 const struct compat_iovec __user *rvec,
370 unsigned long riovcnt,
371 unsigned long flags, int vm_write)
372{
373 struct iovec iovstack_l[UIO_FASTIOV];
374 struct iovec iovstack_r[UIO_FASTIOV];
375 struct iovec *iov_l = iovstack_l;
376 struct iovec *iov_r = iovstack_r;
Al Viro9f78bdf2014-02-05 11:51:53 -0500377 struct iov_iter iter;
Christopher Yeohfcf63402011-10-31 17:06:39 -0700378 ssize_t rc = -EFAULT;
379
380 if (flags != 0)
381 return -EINVAL;
382
Christopher Yeohfcf63402011-10-31 17:06:39 -0700383 if (vm_write)
384 rc = compat_rw_copy_check_uvector(WRITE, lvec, liovcnt,
385 UIO_FASTIOV, iovstack_l,
Christopher Yeohac34ebb2012-05-31 16:26:42 -0700386 &iov_l);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700387 else
388 rc = compat_rw_copy_check_uvector(READ, lvec, liovcnt,
389 UIO_FASTIOV, iovstack_l,
Christopher Yeohac34ebb2012-05-31 16:26:42 -0700390 &iov_l);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700391 if (rc <= 0)
392 goto free_iovecs;
Al Viro9f78bdf2014-02-05 11:51:53 -0500393 iov_iter_init(&iter, iov_l, liovcnt, rc, 0);
Christopher Yeohac34ebb2012-05-31 16:26:42 -0700394 rc = compat_rw_copy_check_uvector(CHECK_IOVEC_ONLY, rvec, riovcnt,
Christopher Yeohfcf63402011-10-31 17:06:39 -0700395 UIO_FASTIOV, iovstack_r,
Christopher Yeohac34ebb2012-05-31 16:26:42 -0700396 &iov_r);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700397 if (rc <= 0)
398 goto free_iovecs;
399
Al Viro9f78bdf2014-02-05 11:51:53 -0500400 rc = process_vm_rw_core(pid, &iter, iov_r, riovcnt, flags, vm_write);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700401
402free_iovecs:
403 if (iov_r != iovstack_r)
404 kfree(iov_r);
405 if (iov_l != iovstack_l)
406 kfree(iov_l);
Christopher Yeohfcf63402011-10-31 17:06:39 -0700407 return rc;
408}
409
410asmlinkage ssize_t
411compat_sys_process_vm_readv(compat_pid_t pid,
412 const struct compat_iovec __user *lvec,
413 unsigned long liovcnt,
414 const struct compat_iovec __user *rvec,
415 unsigned long riovcnt,
416 unsigned long flags)
417{
418 return compat_process_vm_rw(pid, lvec, liovcnt, rvec,
419 riovcnt, flags, 0);
420}
421
422asmlinkage ssize_t
423compat_sys_process_vm_writev(compat_pid_t pid,
424 const struct compat_iovec __user *lvec,
425 unsigned long liovcnt,
426 const struct compat_iovec __user *rvec,
427 unsigned long riovcnt,
428 unsigned long flags)
429{
430 return compat_process_vm_rw(pid, lvec, liovcnt, rvec,
431 riovcnt, flags, 1);
432}
433
434#endif