]> git.proxmox.com Git - mirror_qemu.git/blame - linux-user/mmap.c
Merge remote-tracking branch 'remotes/rth/tags/pull-sparc-20161031-2' into staging
[mirror_qemu.git] / linux-user / mmap.c
CommitLineData
54936004
FB
1/*
2 * mmap support for qemu
5fafdf24 3 *
54936004
FB
4 * Copyright (c) 2003 Fabrice Bellard
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
8167ee88 17 * along with this program; if not, see <http://www.gnu.org/licenses/>.
54936004 18 */
d39594e9 19#include "qemu/osdep.h"
54936004
FB
20
21#include "qemu.h"
78f5bf1e 22#include "qemu-common.h"
1652b974 23#include "translate-all.h"
54936004
FB
24
25//#define DEBUG_MMAP
26
1e6eec8b 27static pthread_mutex_t mmap_mutex = PTHREAD_MUTEX_INITIALIZER;
dfd3f85c 28static __thread int mmap_lock_count;
c8a706fe
PB
29
30void mmap_lock(void)
31{
32 if (mmap_lock_count++ == 0) {
33 pthread_mutex_lock(&mmap_mutex);
34 }
35}
36
37void mmap_unlock(void)
38{
39 if (--mmap_lock_count == 0) {
40 pthread_mutex_unlock(&mmap_mutex);
41 }
42}
d5975363 43
301e40ed
AB
44bool have_mmap_lock(void)
45{
46 return mmap_lock_count > 0 ? true : false;
47}
48
d5975363
PB
49/* Grab lock to make sure things are in a consistent state after fork(). */
50void mmap_fork_start(void)
51{
52 if (mmap_lock_count)
53 abort();
54 pthread_mutex_lock(&mmap_mutex);
55}
56
57void mmap_fork_end(int child)
58{
59 if (child)
60 pthread_mutex_init(&mmap_mutex, NULL);
61 else
62 pthread_mutex_unlock(&mmap_mutex);
63}
c8a706fe 64
53a5960a 65/* NOTE: all the constants are the HOST ones, but addresses are target. */
992f48a0 66int target_mprotect(abi_ulong start, abi_ulong len, int prot)
54936004 67{
992f48a0 68 abi_ulong end, host_start, host_end, addr;
54936004
FB
69 int prot1, ret;
70
71#ifdef DEBUG_MMAP
0bf9e31a
BS
72 printf("mprotect: start=0x" TARGET_ABI_FMT_lx
73 "len=0x" TARGET_ABI_FMT_lx " prot=%c%c%c\n", start, len,
54936004
FB
74 prot & PROT_READ ? 'r' : '-',
75 prot & PROT_WRITE ? 'w' : '-',
76 prot & PROT_EXEC ? 'x' : '-');
77#endif
78
79 if ((start & ~TARGET_PAGE_MASK) != 0)
80 return -EINVAL;
81 len = TARGET_PAGE_ALIGN(len);
82 end = start + len;
83 if (end < start)
84 return -EINVAL;
171cd1cd 85 prot &= PROT_READ | PROT_WRITE | PROT_EXEC;
54936004
FB
86 if (len == 0)
87 return 0;
3b46e624 88
c8a706fe 89 mmap_lock();
83fb7adf 90 host_start = start & qemu_host_page_mask;
54936004
FB
91 host_end = HOST_PAGE_ALIGN(end);
92 if (start > host_start) {
93 /* handle host page containing start */
94 prot1 = prot;
95 for(addr = host_start; addr < start; addr += TARGET_PAGE_SIZE) {
96 prot1 |= page_get_flags(addr);
97 }
83fb7adf 98 if (host_end == host_start + qemu_host_page_size) {
d418c81e
FB
99 for(addr = end; addr < host_end; addr += TARGET_PAGE_SIZE) {
100 prot1 |= page_get_flags(addr);
101 }
102 end = host_end;
103 }
53a5960a 104 ret = mprotect(g2h(host_start), qemu_host_page_size, prot1 & PAGE_BITS);
54936004 105 if (ret != 0)
c8a706fe 106 goto error;
83fb7adf 107 host_start += qemu_host_page_size;
54936004
FB
108 }
109 if (end < host_end) {
54936004
FB
110 prot1 = prot;
111 for(addr = end; addr < host_end; addr += TARGET_PAGE_SIZE) {
112 prot1 |= page_get_flags(addr);
113 }
5fafdf24 114 ret = mprotect(g2h(host_end - qemu_host_page_size), qemu_host_page_size,
54936004
FB
115 prot1 & PAGE_BITS);
116 if (ret != 0)
c8a706fe 117 goto error;
83fb7adf 118 host_end -= qemu_host_page_size;
54936004 119 }
3b46e624 120
54936004
FB
121 /* handle the pages in the middle */
122 if (host_start < host_end) {
53a5960a 123 ret = mprotect(g2h(host_start), host_end - host_start, prot);
54936004 124 if (ret != 0)
c8a706fe 125 goto error;
54936004 126 }
54936004 127 page_set_flags(start, start + len, prot | PAGE_VALID);
c8a706fe 128 mmap_unlock();
54936004 129 return 0;
c8a706fe
PB
130error:
131 mmap_unlock();
132 return ret;
54936004
FB
133}
134
135/* map an incomplete host page */
992f48a0
BS
136static int mmap_frag(abi_ulong real_start,
137 abi_ulong start, abi_ulong end,
138 int prot, int flags, int fd, abi_ulong offset)
54936004 139{
80210bcd 140 abi_ulong real_end, addr;
53a5960a 141 void *host_start;
54936004
FB
142 int prot1, prot_new;
143
53a5960a
PB
144 real_end = real_start + qemu_host_page_size;
145 host_start = g2h(real_start);
54936004
FB
146
147 /* get the protection of the target pages outside the mapping */
148 prot1 = 0;
53a5960a 149 for(addr = real_start; addr < real_end; addr++) {
54936004
FB
150 if (addr < start || addr >= end)
151 prot1 |= page_get_flags(addr);
152 }
3b46e624 153
54936004
FB
154 if (prot1 == 0) {
155 /* no page was there, so we allocate one */
80210bcd
TS
156 void *p = mmap(host_start, qemu_host_page_size, prot,
157 flags | MAP_ANONYMOUS, -1, 0);
158 if (p == MAP_FAILED)
159 return -1;
53a5960a 160 prot1 = prot;
54936004
FB
161 }
162 prot1 &= PAGE_BITS;
163
164 prot_new = prot | prot1;
165 if (!(flags & MAP_ANONYMOUS)) {
166 /* msync() won't work here, so we return an error if write is
167 possible while it is a shared mapping */
168 if ((flags & MAP_TYPE) == MAP_SHARED &&
169 (prot & PROT_WRITE))
ee636500 170 return -1;
54936004
FB
171
172 /* adjust protection to be able to read */
173 if (!(prot1 & PROT_WRITE))
53a5960a 174 mprotect(host_start, qemu_host_page_size, prot1 | PROT_WRITE);
3b46e624 175
54936004 176 /* read the corresponding file data */
fb7e378c
KS
177 if (pread(fd, g2h(start), end - start, offset) == -1)
178 return -1;
3b46e624 179
54936004
FB
180 /* put final protection */
181 if (prot_new != (prot1 | PROT_WRITE))
53a5960a 182 mprotect(host_start, qemu_host_page_size, prot_new);
54936004 183 } else {
54936004 184 if (prot_new != prot1) {
53a5960a 185 mprotect(host_start, qemu_host_page_size, prot_new);
54936004 186 }
e6deac9c
CG
187 if (prot_new & PROT_WRITE) {
188 memset(g2h(start), 0, end - start);
189 }
54936004
FB
190 }
191 return 0;
192}
193
14f24e14
RH
194#if HOST_LONG_BITS == 64 && TARGET_ABI_BITS == 64
195# define TASK_UNMAPPED_BASE (1ul << 38)
196#elif defined(__CYGWIN__)
a03e2d42 197/* Cygwin doesn't have a whole lot of address space. */
14f24e14 198# define TASK_UNMAPPED_BASE 0x18000000
a03e2d42 199#else
14f24e14 200# define TASK_UNMAPPED_BASE 0x40000000
a03e2d42 201#endif
59e9d91c 202abi_ulong mmap_next_start = TASK_UNMAPPED_BASE;
a03e2d42 203
0776590d
PB
204unsigned long last_brk;
205
68a1c816
PB
206/* Subroutine of mmap_find_vma, used when we have pre-allocated a chunk
207 of guest address space. */
208static abi_ulong mmap_find_vma_reserved(abi_ulong start, abi_ulong size)
209{
210 abi_ulong addr;
59e9d91c 211 abi_ulong end_addr;
68a1c816
PB
212 int prot;
213 int looped = 0;
214
b76f21a7 215 if (size > reserved_va) {
68a1c816
PB
216 return (abi_ulong)-1;
217 }
218
59e9d91c
PM
219 size = HOST_PAGE_ALIGN(size);
220 end_addr = start + size;
b76f21a7
LV
221 if (end_addr > reserved_va) {
222 end_addr = reserved_va;
59e9d91c
PM
223 }
224 addr = end_addr - qemu_host_page_size;
225
226 while (1) {
227 if (addr > end_addr) {
68a1c816
PB
228 if (looped) {
229 return (abi_ulong)-1;
230 }
b76f21a7 231 end_addr = reserved_va;
59e9d91c 232 addr = end_addr - qemu_host_page_size;
68a1c816
PB
233 looped = 1;
234 continue;
235 }
236 prot = page_get_flags(addr);
237 if (prot) {
59e9d91c
PM
238 end_addr = addr;
239 }
240 if (addr + size == end_addr) {
241 break;
68a1c816 242 }
59e9d91c 243 addr -= qemu_host_page_size;
68a1c816 244 }
59e9d91c
PM
245
246 if (start == mmap_next_start) {
247 mmap_next_start = addr;
248 }
249
250 return addr;
68a1c816
PB
251}
252
fe3b4152
KS
253/*
254 * Find and reserve a free memory area of size 'size'. The search
255 * starts at 'start'.
256 * It must be called with mmap_lock() held.
257 * Return -1 if error.
258 */
9ad197d9 259abi_ulong mmap_find_vma(abi_ulong start, abi_ulong size)
a03e2d42 260{
14f24e14 261 void *ptr, *prev;
fe3b4152 262 abi_ulong addr;
14f24e14 263 int wrapped, repeat;
fe3b4152
KS
264
265 /* If 'start' == 0, then a default start address is used. */
14f24e14 266 if (start == 0) {
fe3b4152 267 start = mmap_next_start;
14f24e14
RH
268 } else {
269 start &= qemu_host_page_mask;
270 }
271
272 size = HOST_PAGE_ALIGN(size);
fe3b4152 273
b76f21a7 274 if (reserved_va) {
68a1c816
PB
275 return mmap_find_vma_reserved(start, size);
276 }
277
a03e2d42 278 addr = start;
14f24e14
RH
279 wrapped = repeat = 0;
280 prev = 0;
fe3b4152 281
14f24e14 282 for (;; prev = ptr) {
fe3b4152
KS
283 /*
284 * Reserve needed memory area to avoid a race.
285 * It should be discarded using:
286 * - mmap() with MAP_FIXED flag
287 * - mremap() with MREMAP_FIXED flag
288 * - shmat() with SHM_REMAP flag
289 */
14f24e14 290 ptr = mmap(g2h(addr), size, PROT_NONE,
fe3b4152
KS
291 MAP_ANONYMOUS|MAP_PRIVATE|MAP_NORESERVE, -1, 0);
292
293 /* ENOMEM, if host address space has no memory */
14f24e14 294 if (ptr == MAP_FAILED) {
fe3b4152 295 return (abi_ulong)-1;
14f24e14
RH
296 }
297
298 /* Count the number of sequential returns of the same address.
299 This is used to modify the search algorithm below. */
300 repeat = (ptr == prev ? repeat + 1 : 0);
301
302 if (h2g_valid(ptr + size - 1)) {
303 addr = h2g(ptr);
fe3b4152 304
14f24e14
RH
305 if ((addr & ~TARGET_PAGE_MASK) == 0) {
306 /* Success. */
307 if (start == mmap_next_start && addr >= TASK_UNMAPPED_BASE) {
308 mmap_next_start = addr + size;
309 }
310 return addr;
311 }
fe3b4152 312
14f24e14
RH
313 /* The address is not properly aligned for the target. */
314 switch (repeat) {
315 case 0:
316 /* Assume the result that the kernel gave us is the
317 first with enough free space, so start again at the
318 next higher target page. */
319 addr = TARGET_PAGE_ALIGN(addr);
320 break;
321 case 1:
322 /* Sometimes the kernel decides to perform the allocation
323 at the top end of memory instead. */
324 addr &= TARGET_PAGE_MASK;
325 break;
326 case 2:
327 /* Start over at low memory. */
328 addr = 0;
329 break;
330 default:
331 /* Fail. This unaligned block must the last. */
332 addr = -1;
333 break;
334 }
335 } else {
336 /* Since the result the kernel gave didn't fit, start
337 again at low memory. If any repetition, fail. */
338 addr = (repeat ? -1 : 0);
339 }
340
341 /* Unmap and try again. */
fe3b4152 342 munmap(ptr, size);
fe3b4152 343
14f24e14 344 /* ENOMEM if we checked the whole of the target address space. */
d0b3e4f5 345 if (addr == (abi_ulong)-1) {
a03e2d42 346 return (abi_ulong)-1;
14f24e14
RH
347 } else if (addr == 0) {
348 if (wrapped) {
349 return (abi_ulong)-1;
350 }
351 wrapped = 1;
352 /* Don't actually use 0 when wrapping, instead indicate
8186e783 353 that we'd truly like an allocation in low memory. */
14f24e14
RH
354 addr = (mmap_min_addr > TARGET_PAGE_SIZE
355 ? TARGET_PAGE_ALIGN(mmap_min_addr)
356 : TARGET_PAGE_SIZE);
357 } else if (wrapped && addr >= start) {
358 return (abi_ulong)-1;
359 }
a03e2d42 360 }
a03e2d42
FB
361}
362
54936004 363/* NOTE: all the constants are the HOST ones */
992f48a0
BS
364abi_long target_mmap(abi_ulong start, abi_ulong len, int prot,
365 int flags, int fd, abi_ulong offset)
54936004 366{
992f48a0 367 abi_ulong ret, end, real_start, real_end, retaddr, host_offset, host_len;
54936004 368
c8a706fe 369 mmap_lock();
54936004
FB
370#ifdef DEBUG_MMAP
371 {
0bf9e31a
BS
372 printf("mmap: start=0x" TARGET_ABI_FMT_lx
373 " len=0x" TARGET_ABI_FMT_lx " prot=%c%c%c flags=",
5fafdf24 374 start, len,
54936004
FB
375 prot & PROT_READ ? 'r' : '-',
376 prot & PROT_WRITE ? 'w' : '-',
377 prot & PROT_EXEC ? 'x' : '-');
378 if (flags & MAP_FIXED)
379 printf("MAP_FIXED ");
380 if (flags & MAP_ANONYMOUS)
381 printf("MAP_ANON ");
382 switch(flags & MAP_TYPE) {
383 case MAP_PRIVATE:
384 printf("MAP_PRIVATE ");
385 break;
386 case MAP_SHARED:
387 printf("MAP_SHARED ");
388 break;
389 default:
390 printf("[MAP_TYPE=0x%x] ", flags & MAP_TYPE);
391 break;
392 }
0bf9e31a 393 printf("fd=%d offset=" TARGET_ABI_FMT_lx "\n", fd, offset);
54936004
FB
394 }
395#endif
396
e89f07d3
PB
397 if (offset & ~TARGET_PAGE_MASK) {
398 errno = EINVAL;
c8a706fe 399 goto fail;
e89f07d3 400 }
54936004
FB
401
402 len = TARGET_PAGE_ALIGN(len);
403 if (len == 0)
c8a706fe 404 goto the_end;
53a5960a 405 real_start = start & qemu_host_page_mask;
a5e7ee46
RH
406 host_offset = offset & qemu_host_page_mask;
407
408 /* If the user is asking for the kernel to find a location, do that
409 before we truncate the length for mapping files below. */
410 if (!(flags & MAP_FIXED)) {
411 host_len = len + offset - host_offset;
412 host_len = HOST_PAGE_ALIGN(host_len);
413 start = mmap_find_vma(real_start, host_len);
414 if (start == (abi_ulong)-1) {
415 errno = ENOMEM;
416 goto fail;
417 }
418 }
54936004 419
54c5a2ae
EI
420 /* When mapping files into a memory area larger than the file, accesses
421 to pages beyond the file size will cause a SIGBUS.
422
423 For example, if mmaping a file of 100 bytes on a host with 4K pages
424 emulating a target with 8K pages, the target expects to be able to
425 access the first 8K. But the host will trap us on any access beyond
426 4K.
427
428 When emulating a target with a larger page-size than the hosts, we
429 may need to truncate file maps at EOF and add extra anonymous pages
430 up to the targets page boundary. */
431
432 if ((qemu_real_host_page_size < TARGET_PAGE_SIZE)
433 && !(flags & MAP_ANONYMOUS)) {
434 struct stat sb;
435
436 if (fstat (fd, &sb) == -1)
437 goto fail;
438
439 /* Are we trying to create a map beyond EOF?. */
440 if (offset + len > sb.st_size) {
441 /* If so, truncate the file map at eof aligned with
442 the hosts real pagesize. Additional anonymous maps
443 will be created beyond EOF. */
0c2d70c4 444 len = REAL_HOST_PAGE_ALIGN(sb.st_size - offset);
54c5a2ae
EI
445 }
446 }
447
54936004 448 if (!(flags & MAP_FIXED)) {
a5e7ee46 449 unsigned long host_start;
a03e2d42 450 void *p;
a5e7ee46 451
a03e2d42
FB
452 host_len = len + offset - host_offset;
453 host_len = HOST_PAGE_ALIGN(host_len);
a5e7ee46 454
a03e2d42
FB
455 /* Note: we prefer to control the mapping address. It is
456 especially important if qemu_host_page_size >
457 qemu_real_host_page_size */
a5e7ee46
RH
458 p = mmap(g2h(start), host_len, prot,
459 flags | MAP_FIXED | MAP_ANONYMOUS, -1, 0);
a03e2d42 460 if (p == MAP_FAILED)
c8a706fe 461 goto fail;
a03e2d42
FB
462 /* update start so that it points to the file position at 'offset' */
463 host_start = (unsigned long)p;
54c5a2ae 464 if (!(flags & MAP_ANONYMOUS)) {
a5e7ee46 465 p = mmap(g2h(start), len, prot,
54c5a2ae 466 flags | MAP_FIXED, fd, host_offset);
8384274e
JB
467 if (p == MAP_FAILED) {
468 munmap(g2h(start), host_len);
469 goto fail;
470 }
a03e2d42 471 host_start += offset - host_offset;
54c5a2ae 472 }
a03e2d42
FB
473 start = h2g(host_start);
474 } else {
475 if (start & ~TARGET_PAGE_MASK) {
e89f07d3 476 errno = EINVAL;
c8a706fe 477 goto fail;
e89f07d3 478 }
a03e2d42
FB
479 end = start + len;
480 real_end = HOST_PAGE_ALIGN(end);
7ab240ad 481
45bc1f52
AJ
482 /*
483 * Test if requested memory area fits target address space
484 * It can fail only on 64-bit host with 32-bit target.
485 * On any other target/host host mmap() handles this error correctly.
486 */
487 if ((unsigned long)start + len - 1 > (abi_ulong) -1) {
488 errno = EINVAL;
489 goto fail;
490 }
491
a03e2d42
FB
492 /* worst case: we cannot map the file because the offset is not
493 aligned, so we read it */
494 if (!(flags & MAP_ANONYMOUS) &&
495 (offset & ~qemu_host_page_mask) != (start & ~qemu_host_page_mask)) {
496 /* msync() won't work here, so we return an error if write is
497 possible while it is a shared mapping */
498 if ((flags & MAP_TYPE) == MAP_SHARED &&
499 (prot & PROT_WRITE)) {
500 errno = EINVAL;
c8a706fe 501 goto fail;
a03e2d42
FB
502 }
503 retaddr = target_mmap(start, len, prot | PROT_WRITE,
504 MAP_FIXED | MAP_PRIVATE | MAP_ANONYMOUS,
505 -1, 0);
506 if (retaddr == -1)
c8a706fe 507 goto fail;
fb7e378c
KS
508 if (pread(fd, g2h(start), len, offset) == -1)
509 goto fail;
a03e2d42
FB
510 if (!(prot & PROT_WRITE)) {
511 ret = target_mprotect(start, len, prot);
86abac06 512 assert(ret == 0);
a03e2d42
FB
513 }
514 goto the_end;
54936004 515 }
a03e2d42
FB
516
517 /* handle the start of the mapping */
518 if (start > real_start) {
519 if (real_end == real_start + qemu_host_page_size) {
520 /* one single host page */
521 ret = mmap_frag(real_start, start, end,
522 prot, flags, fd, offset);
523 if (ret == -1)
c8a706fe 524 goto fail;
a03e2d42
FB
525 goto the_end1;
526 }
527 ret = mmap_frag(real_start, start, real_start + qemu_host_page_size,
54936004
FB
528 prot, flags, fd, offset);
529 if (ret == -1)
c8a706fe 530 goto fail;
a03e2d42
FB
531 real_start += qemu_host_page_size;
532 }
533 /* handle the end of the mapping */
534 if (end < real_end) {
535 ret = mmap_frag(real_end - qemu_host_page_size,
530c0032 536 real_end - qemu_host_page_size, end,
a03e2d42
FB
537 prot, flags, fd,
538 offset + real_end - qemu_host_page_size - start);
539 if (ret == -1)
c8a706fe 540 goto fail;
a03e2d42 541 real_end -= qemu_host_page_size;
54936004 542 }
3b46e624 543
a03e2d42
FB
544 /* map the middle (easier) */
545 if (real_start < real_end) {
546 void *p;
547 unsigned long offset1;
548 if (flags & MAP_ANONYMOUS)
549 offset1 = 0;
550 else
551 offset1 = offset + real_start - start;
552 p = mmap(g2h(real_start), real_end - real_start,
553 prot, flags, fd, offset1);
554 if (p == MAP_FAILED)
c8a706fe 555 goto fail;
a03e2d42 556 }
54936004
FB
557 }
558 the_end1:
559 page_set_flags(start, start + len, prot | PAGE_VALID);
560 the_end:
561#ifdef DEBUG_MMAP
0bf9e31a 562 printf("ret=0x" TARGET_ABI_FMT_lx "\n", start);
54936004
FB
563 page_dump(stdout);
564 printf("\n");
565#endif
35865339 566 tb_invalidate_phys_range(start, start + len);
c8a706fe 567 mmap_unlock();
54936004 568 return start;
c8a706fe
PB
569fail:
570 mmap_unlock();
571 return -1;
54936004
FB
572}
573
68a1c816
PB
574static void mmap_reserve(abi_ulong start, abi_ulong size)
575{
576 abi_ulong real_start;
577 abi_ulong real_end;
578 abi_ulong addr;
579 abi_ulong end;
580 int prot;
581
582 real_start = start & qemu_host_page_mask;
583 real_end = HOST_PAGE_ALIGN(start + size);
584 end = start + size;
585 if (start > real_start) {
586 /* handle host page containing start */
587 prot = 0;
588 for (addr = real_start; addr < start; addr += TARGET_PAGE_SIZE) {
589 prot |= page_get_flags(addr);
590 }
591 if (real_end == real_start + qemu_host_page_size) {
592 for (addr = end; addr < real_end; addr += TARGET_PAGE_SIZE) {
593 prot |= page_get_flags(addr);
594 }
595 end = real_end;
596 }
597 if (prot != 0)
598 real_start += qemu_host_page_size;
599 }
600 if (end < real_end) {
601 prot = 0;
602 for (addr = end; addr < real_end; addr += TARGET_PAGE_SIZE) {
603 prot |= page_get_flags(addr);
604 }
605 if (prot != 0)
606 real_end -= qemu_host_page_size;
607 }
608 if (real_start != real_end) {
609 mmap(g2h(real_start), real_end - real_start, PROT_NONE,
610 MAP_FIXED | MAP_ANONYMOUS | MAP_PRIVATE | MAP_NORESERVE,
611 -1, 0);
612 }
613}
614
992f48a0 615int target_munmap(abi_ulong start, abi_ulong len)
54936004 616{
992f48a0 617 abi_ulong end, real_start, real_end, addr;
54936004
FB
618 int prot, ret;
619
620#ifdef DEBUG_MMAP
0bf9e31a
BS
621 printf("munmap: start=0x" TARGET_ABI_FMT_lx " len=0x"
622 TARGET_ABI_FMT_lx "\n",
623 start, len);
54936004
FB
624#endif
625 if (start & ~TARGET_PAGE_MASK)
626 return -EINVAL;
627 len = TARGET_PAGE_ALIGN(len);
628 if (len == 0)
629 return -EINVAL;
c8a706fe 630 mmap_lock();
54936004 631 end = start + len;
53a5960a
PB
632 real_start = start & qemu_host_page_mask;
633 real_end = HOST_PAGE_ALIGN(end);
54936004 634
53a5960a 635 if (start > real_start) {
54936004
FB
636 /* handle host page containing start */
637 prot = 0;
53a5960a 638 for(addr = real_start; addr < start; addr += TARGET_PAGE_SIZE) {
54936004
FB
639 prot |= page_get_flags(addr);
640 }
53a5960a
PB
641 if (real_end == real_start + qemu_host_page_size) {
642 for(addr = end; addr < real_end; addr += TARGET_PAGE_SIZE) {
d418c81e
FB
643 prot |= page_get_flags(addr);
644 }
53a5960a 645 end = real_end;
d418c81e 646 }
54936004 647 if (prot != 0)
53a5960a 648 real_start += qemu_host_page_size;
54936004 649 }
53a5960a 650 if (end < real_end) {
54936004 651 prot = 0;
53a5960a 652 for(addr = end; addr < real_end; addr += TARGET_PAGE_SIZE) {
54936004
FB
653 prot |= page_get_flags(addr);
654 }
655 if (prot != 0)
53a5960a 656 real_end -= qemu_host_page_size;
54936004 657 }
3b46e624 658
c8a706fe 659 ret = 0;
54936004 660 /* unmap what we can */
53a5960a 661 if (real_start < real_end) {
b76f21a7 662 if (reserved_va) {
68a1c816
PB
663 mmap_reserve(real_start, real_end - real_start);
664 } else {
665 ret = munmap(g2h(real_start), real_end - real_start);
666 }
54936004
FB
667 }
668
77a8f1a5 669 if (ret == 0) {
c8a706fe 670 page_set_flags(start, start + len, 0);
35865339 671 tb_invalidate_phys_range(start, start + len);
77a8f1a5 672 }
c8a706fe
PB
673 mmap_unlock();
674 return ret;
54936004
FB
675}
676
992f48a0
BS
677abi_long target_mremap(abi_ulong old_addr, abi_ulong old_size,
678 abi_ulong new_size, unsigned long flags,
679 abi_ulong new_addr)
54936004
FB
680{
681 int prot;
f19412a2 682 void *host_addr;
54936004 683
c8a706fe 684 mmap_lock();
f19412a2 685
68a1c816 686 if (flags & MREMAP_FIXED) {
52956a9b
FJ
687 host_addr = mremap(g2h(old_addr), old_size, new_size,
688 flags, g2h(new_addr));
68a1c816 689
b76f21a7 690 if (reserved_va && host_addr != MAP_FAILED) {
68a1c816
PB
691 /* If new and old addresses overlap then the above mremap will
692 already have failed with EINVAL. */
693 mmap_reserve(old_addr, old_size);
694 }
695 } else if (flags & MREMAP_MAYMOVE) {
f19412a2
AJ
696 abi_ulong mmap_start;
697
698 mmap_start = mmap_find_vma(0, new_size);
699
700 if (mmap_start == -1) {
701 errno = ENOMEM;
702 host_addr = MAP_FAILED;
68a1c816 703 } else {
52956a9b
FJ
704 host_addr = mremap(g2h(old_addr), old_size, new_size,
705 flags | MREMAP_FIXED, g2h(mmap_start));
b76f21a7 706 if (reserved_va) {
c65ffe6d 707 mmap_reserve(old_addr, old_size);
708 }
68a1c816 709 }
3af72a4d 710 } else {
68a1c816 711 int prot = 0;
b76f21a7 712 if (reserved_va && old_size < new_size) {
68a1c816
PB
713 abi_ulong addr;
714 for (addr = old_addr + old_size;
715 addr < old_addr + new_size;
716 addr++) {
717 prot |= page_get_flags(addr);
718 }
719 }
720 if (prot == 0) {
721 host_addr = mremap(g2h(old_addr), old_size, new_size, flags);
b76f21a7 722 if (host_addr != MAP_FAILED && reserved_va && old_size > new_size) {
68a1c816
PB
723 mmap_reserve(old_addr + old_size, new_size - old_size);
724 }
725 } else {
726 errno = ENOMEM;
727 host_addr = MAP_FAILED;
728 }
f19412a2
AJ
729 /* Check if address fits target address space */
730 if ((unsigned long)host_addr + new_size > (abi_ulong)-1) {
731 /* Revert mremap() changes */
732 host_addr = mremap(g2h(old_addr), new_size, old_size, flags);
733 errno = ENOMEM;
734 host_addr = MAP_FAILED;
735 }
736 }
737
738 if (host_addr == MAP_FAILED) {
c8a706fe
PB
739 new_addr = -1;
740 } else {
741 new_addr = h2g(host_addr);
742 prot = page_get_flags(old_addr);
743 page_set_flags(old_addr, old_addr + old_size, 0);
744 page_set_flags(new_addr, new_addr + new_size, prot | PAGE_VALID);
745 }
35865339 746 tb_invalidate_phys_range(new_addr, new_addr + new_size);
c8a706fe 747 mmap_unlock();
54936004
FB
748 return new_addr;
749}
750
992f48a0 751int target_msync(abi_ulong start, abi_ulong len, int flags)
54936004 752{
992f48a0 753 abi_ulong end;
54936004
FB
754
755 if (start & ~TARGET_PAGE_MASK)
756 return -EINVAL;
757 len = TARGET_PAGE_ALIGN(len);
54936004 758 end = start + len;
d418c81e
FB
759 if (end < start)
760 return -EINVAL;
761 if (end == start)
762 return 0;
3b46e624 763
83fb7adf 764 start &= qemu_host_page_mask;
53a5960a 765 return msync(g2h(start), end - start, flags);
54936004 766}