diff options
Diffstat (limited to 'mm/mmap.c')
-rw-r--r-- | mm/mmap.c | 43 |
1 files changed, 16 insertions, 27 deletions
@@ -1200,6 +1200,9 @@ arch_get_unmapped_area(struct file *filp, unsigned long addr, | |||
1200 | if (len > TASK_SIZE) | 1200 | if (len > TASK_SIZE) |
1201 | return -ENOMEM; | 1201 | return -ENOMEM; |
1202 | 1202 | ||
1203 | if (flags & MAP_FIXED) | ||
1204 | return addr; | ||
1205 | |||
1203 | if (addr) { | 1206 | if (addr) { |
1204 | addr = PAGE_ALIGN(addr); | 1207 | addr = PAGE_ALIGN(addr); |
1205 | vma = find_vma(mm, addr); | 1208 | vma = find_vma(mm, addr); |
@@ -1273,6 +1276,9 @@ arch_get_unmapped_area_topdown(struct file *filp, const unsigned long addr0, | |||
1273 | if (len > TASK_SIZE) | 1276 | if (len > TASK_SIZE) |
1274 | return -ENOMEM; | 1277 | return -ENOMEM; |
1275 | 1278 | ||
1279 | if (flags & MAP_FIXED) | ||
1280 | return addr; | ||
1281 | |||
1276 | /* requesting a specific address */ | 1282 | /* requesting a specific address */ |
1277 | if (addr) { | 1283 | if (addr) { |
1278 | addr = PAGE_ALIGN(addr); | 1284 | addr = PAGE_ALIGN(addr); |
@@ -1361,38 +1367,21 @@ get_unmapped_area(struct file *file, unsigned long addr, unsigned long len, | |||
1361 | unsigned long pgoff, unsigned long flags) | 1367 | unsigned long pgoff, unsigned long flags) |
1362 | { | 1368 | { |
1363 | unsigned long ret; | 1369 | unsigned long ret; |
1364 | 1370 | unsigned long (*get_area)(struct file *, unsigned long, | |
1365 | if (!(flags & MAP_FIXED)) { | 1371 | unsigned long, unsigned long, unsigned long); |
1366 | unsigned long (*get_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long); | 1372 | |
1367 | 1373 | get_area = current->mm->get_unmapped_area; | |
1368 | get_area = current->mm->get_unmapped_area; | 1374 | if (file && file->f_op && file->f_op->get_unmapped_area) |
1369 | if (file && file->f_op && file->f_op->get_unmapped_area) | 1375 | get_area = file->f_op->get_unmapped_area; |
1370 | get_area = file->f_op->get_unmapped_area; | 1376 | addr = get_area(file, addr, len, pgoff, flags); |
1371 | addr = get_area(file, addr, len, pgoff, flags); | 1377 | if (IS_ERR_VALUE(addr)) |
1372 | if (IS_ERR_VALUE(addr)) | 1378 | return addr; |
1373 | return addr; | ||
1374 | } | ||
1375 | 1379 | ||
1376 | if (addr > TASK_SIZE - len) | 1380 | if (addr > TASK_SIZE - len) |
1377 | return -ENOMEM; | 1381 | return -ENOMEM; |
1378 | if (addr & ~PAGE_MASK) | 1382 | if (addr & ~PAGE_MASK) |
1379 | return -EINVAL; | 1383 | return -EINVAL; |
1380 | if (file && is_file_hugepages(file)) { | 1384 | |
1381 | /* | ||
1382 | * Check if the given range is hugepage aligned, and | ||
1383 | * can be made suitable for hugepages. | ||
1384 | */ | ||
1385 | ret = prepare_hugepage_range(addr, len, pgoff); | ||
1386 | } else { | ||
1387 | /* | ||
1388 | * Ensure that a normal request is not falling in a | ||
1389 | * reserved hugepage range. For some archs like IA-64, | ||
1390 | * there is a separate region for hugepages. | ||
1391 | */ | ||
1392 | ret = is_hugepage_only_range(current->mm, addr, len); | ||
1393 | } | ||
1394 | if (ret) | ||
1395 | return -EINVAL; | ||
1396 | return addr; | 1385 | return addr; |
1397 | } | 1386 | } |
1398 | 1387 | ||