* heap.cc (sbrk): Add a FIXME comment to VirtualFree call. Fix memory
reservation and commit strategy when more memory is requested than available on the heap. Release newly reserved memory if commiting it fails. Add more comments to explain what we do.
This commit is contained in:
parent
a723366660
commit
b03bd1f41c
@ -245,7 +245,7 @@ extern "C" void *
|
|||||||
sbrk (ptrdiff_t n)
|
sbrk (ptrdiff_t n)
|
||||||
{
|
{
|
||||||
char *newtop, *newbrk;
|
char *newtop, *newbrk;
|
||||||
SIZE_T commitbytes, newbrksize;
|
SIZE_T commitbytes, newbrksize, reservebytes;
|
||||||
|
|
||||||
if (n == 0)
|
if (n == 0)
|
||||||
return cygheap->user_heap.ptr; /* Just wanted to find current cygheap->user_heap.ptr address */
|
return cygheap->user_heap.ptr; /* Just wanted to find current cygheap->user_heap.ptr address */
|
||||||
@ -261,6 +261,9 @@ sbrk (ptrdiff_t n)
|
|||||||
{ /* Freeing memory */
|
{ /* Freeing memory */
|
||||||
assert (newtop < cygheap->user_heap.top);
|
assert (newtop < cygheap->user_heap.top);
|
||||||
n = (char *) cygheap->user_heap.top - newtop;
|
n = (char *) cygheap->user_heap.top - newtop;
|
||||||
|
/* FIXME: This doesn't work if we cross a virtual memory reservation
|
||||||
|
border. If that happens, we have to free the space in multiple
|
||||||
|
VirtualFree calls, aligned to the former reservation borders. */
|
||||||
if (VirtualFree (newtop, n, MEM_DECOMMIT)) /* Give it back to OS */
|
if (VirtualFree (newtop, n, MEM_DECOMMIT)) /* Give it back to OS */
|
||||||
goto good;
|
goto good;
|
||||||
goto err; /* Didn't take */
|
goto err; /* Didn't take */
|
||||||
@ -277,27 +280,49 @@ sbrk (ptrdiff_t n)
|
|||||||
reserved. */
|
reserved. */
|
||||||
if (newtop <= cygheap->user_heap.max)
|
if (newtop <= cygheap->user_heap.max)
|
||||||
{
|
{
|
||||||
if (VirtualAlloc (cygheap->user_heap.top, commitbytes, MEM_COMMIT, PAGE_READWRITE) != NULL)
|
if (VirtualAlloc (cygheap->user_heap.top, commitbytes,
|
||||||
|
MEM_COMMIT, PAGE_READWRITE) != NULL)
|
||||||
goto good;
|
goto good;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Couldn't allocate memory. Maybe we can reserve some more.
|
/* The remainder of the existing heap is too small to fulfil the memory
|
||||||
Reserve either the maximum of the standard cygwin_shared->heap_chunk_size ()
|
request. We have to extend the heap, so we reserve some more memory
|
||||||
or the requested amount. Then attempt to actually allocate it. */
|
and then commit the remainder of the old heap, if any, and the rest of
|
||||||
if ((newbrksize = cygheap->user_heap.chunk) < commitbytes)
|
the required space from the extended heap. */
|
||||||
newbrksize = commitbytes;
|
|
||||||
|
|
||||||
if ((VirtualAlloc (cygheap->user_heap.top, newbrksize,
|
/* Reserve either the maximum of the standard heap chunk size
|
||||||
MEM_RESERVE, PAGE_NOACCESS)
|
or the requested amount. Then attempt to actually allocate it. */
|
||||||
|| VirtualAlloc (cygheap->user_heap.top, newbrksize = commitbytes,
|
reservebytes = commitbytes - ((ptrdiff_t) cygheap->user_heap.max
|
||||||
MEM_RESERVE, PAGE_NOACCESS))
|
- (ptrdiff_t) cygheap->user_heap.top);
|
||||||
&& VirtualAlloc (cygheap->user_heap.top, commitbytes,
|
commitbytes -= reservebytes;
|
||||||
MEM_COMMIT, PAGE_READWRITE) != NULL)
|
if ((newbrksize = cygheap->user_heap.chunk) < reservebytes)
|
||||||
{
|
newbrksize = reservebytes;
|
||||||
cygheap->user_heap.max = (char *) cygheap->user_heap.max
|
|
||||||
+ pround (newbrksize);
|
if (VirtualAlloc (cygheap->user_heap.max, newbrksize,
|
||||||
goto good;
|
MEM_RESERVE, PAGE_NOACCESS)
|
||||||
}
|
|| VirtualAlloc (cygheap->user_heap.max, newbrksize = reservebytes,
|
||||||
|
MEM_RESERVE, PAGE_NOACCESS))
|
||||||
|
{
|
||||||
|
/* Now commit the requested memory. Windows keeps all virtual
|
||||||
|
reservations separate, so we can't commit the two regions in a single,
|
||||||
|
combined call or we suffer an ERROR_INVALID_ADDRESS. The same error
|
||||||
|
is returned when trying to VirtualAlloc 0 bytes, which would occur if
|
||||||
|
the existing heap was already full. */
|
||||||
|
if ((!commitbytes || VirtualAlloc (cygheap->user_heap.top, commitbytes,
|
||||||
|
MEM_COMMIT, PAGE_READWRITE))
|
||||||
|
&& VirtualAlloc (cygheap->user_heap.max, reservebytes,
|
||||||
|
MEM_COMMIT, PAGE_READWRITE))
|
||||||
|
{
|
||||||
|
cygheap->user_heap.max = (char *) cygheap->user_heap.max
|
||||||
|
+ pround (newbrksize);
|
||||||
|
goto good;
|
||||||
|
}
|
||||||
|
/* If committing the memory failed, we must free the extendend reserved
|
||||||
|
region, otherwise any other try to fetch memory (for instance by using
|
||||||
|
mmap) may fail just because we still reserve memory we don't even know
|
||||||
|
about. */
|
||||||
|
VirtualFree (cygheap->user_heap.max, newbrksize, MEM_RELEASE);
|
||||||
|
}
|
||||||
|
|
||||||
err:
|
err:
|
||||||
set_errno (ENOMEM);
|
set_errno (ENOMEM);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user