aboutsummaryrefslogtreecommitdiffstats
path: root/mm/nommu.c
diff options
context:
space:
mode:
Diffstat (limited to 'mm/nommu.c')
-rw-r--r--mm/nommu.c47
1 files changed, 47 insertions, 0 deletions
diff --git a/mm/nommu.c b/mm/nommu.c
index d99dea31e443..d08acdae0036 100644
--- a/mm/nommu.c
+++ b/mm/nommu.c
@@ -1206,3 +1206,50 @@ struct page *filemap_nopage(struct vm_area_struct *area,
1206 BUG(); 1206 BUG();
1207 return NULL; 1207 return NULL;
1208} 1208}
1209
1210/*
1211 * Access another process' address space.
1212 * - source/target buffer must be kernel space
1213 */
1214int access_process_vm(struct task_struct *tsk, unsigned long addr, void *buf, int len, int write)
1215{
1216 struct vm_list_struct *vml;
1217 struct vm_area_struct *vma;
1218 struct mm_struct *mm;
1219
1220 if (addr + len < addr)
1221 return 0;
1222
1223 mm = get_task_mm(tsk);
1224 if (!mm)
1225 return 0;
1226
1227 down_read(&mm->mmap_sem);
1228
1229 /* the access must start within one of the target process's mappings */
1230 for (vml = mm->context.vmlist; vml; vml = vml->next)
1231 if (addr >= vml->vma->vm_start && addr < vml->vma->vm_end)
1232 break;
1233
1234 if (vml) {
1235 vma = vml->vma;
1236
1237 /* don't overrun this mapping */
1238 if (addr + len >= vma->vm_end)
1239 len = vma->vm_end - addr;
1240
1241 /* only read or write mappings where it is permitted */
1242 if (write && vma->vm_flags & VM_WRITE)
1243 len -= copy_to_user((void *) addr, buf, len);
1244 else if (!write && vma->vm_flags & VM_READ)
1245 len -= copy_from_user(buf, (void *) addr, len);
1246 else
1247 len = 0;
1248 } else {
1249 len = 0;
1250 }
1251
1252 up_read(&mm->mmap_sem);
1253 mmput(mm);
1254 return len;
1255}