Commit 00451d5e authored by Rémi Bernon's avatar Rémi Bernon Committed by Alexandre Julliard

ntdll: Clarify NtAllocateVirtualMemory zero_bits parameter semantics.

This parameter was misinterpreted as an alignment parameter for the lower bits of the allocated memory region, although it is a constraint on the higher bits. Signed-off-by: 's avatarRémi Bernon <rbernon@codeweavers.com> Signed-off-by: 's avatarHuw Davies <huw@codeweavers.com> Signed-off-by: 's avatarAlexandre Julliard <julliard@winehq.org>
parent 2481b53d
...@@ -1603,14 +1603,14 @@ static KERNEL_DIRENT *start_vfat_ioctl( int fd ) ...@@ -1603,14 +1603,14 @@ static KERNEL_DIRENT *start_vfat_ioctl( int fd )
SIZE_T size = 2 * sizeof(*de) + page_size; SIZE_T size = 2 * sizeof(*de) + page_size;
void *addr = NULL; void *addr = NULL;
if (NtAllocateVirtualMemory( GetCurrentProcess(), &addr, 1, &size, MEM_RESERVE, PAGE_READWRITE )) if (virtual_alloc_aligned( &addr, 0, &size, MEM_RESERVE, PAGE_READWRITE, 1 ))
return NULL; return NULL;
/* commit only the size needed for the dir entries */ /* commit only the size needed for the dir entries */
/* this leaves an extra unaccessible page, which should make the kernel */ /* this leaves an extra unaccessible page, which should make the kernel */
/* fail with -EFAULT before it stomps all over our memory */ /* fail with -EFAULT before it stomps all over our memory */
de = addr; de = addr;
size = 2 * sizeof(*de); size = 2 * sizeof(*de);
NtAllocateVirtualMemory( GetCurrentProcess(), &addr, 1, &size, MEM_COMMIT, PAGE_READWRITE ); virtual_alloc_aligned( &addr, 0, &size, MEM_COMMIT, PAGE_READWRITE, 1 );
} }
/* set d_reclen to 65535 to work around an AFS kernel bug */ /* set d_reclen to 65535 to work around an AFS kernel bug */
......
...@@ -726,8 +726,7 @@ static void *allocate_large_block( HEAP *heap, DWORD flags, SIZE_T size ) ...@@ -726,8 +726,7 @@ static void *allocate_large_block( HEAP *heap, DWORD flags, SIZE_T size )
LPVOID address = NULL; LPVOID address = NULL;
if (block_size < size) return NULL; /* overflow */ if (block_size < size) return NULL; /* overflow */
if (NtAllocateVirtualMemory( NtCurrentProcess(), &address, 5, if (virtual_alloc_aligned( &address, 0, &block_size, MEM_COMMIT, get_protection_type( flags ), 5 ))
&block_size, MEM_COMMIT, get_protection_type( flags ) ))
{ {
WARN("Could not allocate block for %08lx bytes\n", size ); WARN("Could not allocate block for %08lx bytes\n", size );
return NULL; return NULL;
...@@ -1521,7 +1520,7 @@ void heap_set_debug_flags( HANDLE handle ) ...@@ -1521,7 +1520,7 @@ void heap_set_debug_flags( HANDLE handle )
void *ptr = NULL; void *ptr = NULL;
SIZE_T size = MAX_FREE_PENDING * sizeof(*heap->pending_free); SIZE_T size = MAX_FREE_PENDING * sizeof(*heap->pending_free);
if (!NtAllocateVirtualMemory( NtCurrentProcess(), &ptr, 4, &size, MEM_COMMIT, PAGE_READWRITE )) if (!virtual_alloc_aligned( &ptr, 0, &size, MEM_COMMIT, PAGE_READWRITE, 4 ))
{ {
heap->pending_free = ptr; heap->pending_free = ptr;
heap->pending_pos = 0; heap->pending_pos = 0;
......
...@@ -168,6 +168,8 @@ extern NTSTATUS nt_to_unix_file_name_attr( const OBJECT_ATTRIBUTES *attr, ANSI_S ...@@ -168,6 +168,8 @@ extern NTSTATUS nt_to_unix_file_name_attr( const OBJECT_ATTRIBUTES *attr, ANSI_S
UINT disposition ) DECLSPEC_HIDDEN; UINT disposition ) DECLSPEC_HIDDEN;
/* virtual memory */ /* virtual memory */
extern NTSTATUS virtual_alloc_aligned( PVOID *ret, ULONG zero_bits, SIZE_T *size_ptr,
ULONG type, ULONG protect, ULONG alignment ) DECLSPEC_HIDDEN;
extern NTSTATUS virtual_map_section( HANDLE handle, PVOID *addr_ptr, ULONG zero_bits, SIZE_T commit_size, extern NTSTATUS virtual_map_section( HANDLE handle, PVOID *addr_ptr, ULONG zero_bits, SIZE_T commit_size,
const LARGE_INTEGER *offset_ptr, SIZE_T *size_ptr, ULONG protect, const LARGE_INTEGER *offset_ptr, SIZE_T *size_ptr, ULONG protect,
pe_image_info_t *image_info ) DECLSPEC_HIDDEN; pe_image_info_t *image_info ) DECLSPEC_HIDDEN;
......
...@@ -967,22 +967,22 @@ int CDECL __wine_set_signal_handler(unsigned int sig, wine_signal_handler wsh) ...@@ -967,22 +967,22 @@ int CDECL __wine_set_signal_handler(unsigned int sig, wine_signal_handler wsh)
*/ */
NTSTATUS signal_alloc_thread( TEB **teb ) NTSTATUS signal_alloc_thread( TEB **teb )
{ {
static size_t sigstack_zero_bits; static size_t sigstack_alignment;
SIZE_T size; SIZE_T size;
NTSTATUS status; NTSTATUS status;
if (!sigstack_zero_bits) if (!sigstack_alignment)
{ {
size_t min_size = page_size; size_t min_size = page_size;
/* find the first power of two not smaller than min_size */ /* find the first power of two not smaller than min_size */
while ((1u << sigstack_zero_bits) < min_size) sigstack_zero_bits++; while ((1u << sigstack_alignment) < min_size) sigstack_alignment++;
assert( sizeof(TEB) <= min_size ); assert( sizeof(TEB) <= min_size );
} }
size = 1 << sigstack_zero_bits; size = 1 << sigstack_alignment;
*teb = NULL; *teb = NULL;
if (!(status = NtAllocateVirtualMemory( NtCurrentProcess(), (void **)teb, sigstack_zero_bits, if (!(status = virtual_alloc_aligned( (void **)teb, 0, &size, MEM_COMMIT | MEM_TOP_DOWN,
&size, MEM_COMMIT | MEM_TOP_DOWN, PAGE_READWRITE ))) PAGE_READWRITE, sigstack_alignment )))
{ {
(*teb)->Tib.Self = &(*teb)->Tib; (*teb)->Tib.Self = &(*teb)->Tib;
(*teb)->Tib.ExceptionList = (void *)~0UL; (*teb)->Tib.ExceptionList = (void *)~0UL;
......
...@@ -871,24 +871,24 @@ int CDECL __wine_set_signal_handler(unsigned int sig, wine_signal_handler wsh) ...@@ -871,24 +871,24 @@ int CDECL __wine_set_signal_handler(unsigned int sig, wine_signal_handler wsh)
*/ */
NTSTATUS signal_alloc_thread( TEB **teb ) NTSTATUS signal_alloc_thread( TEB **teb )
{ {
static size_t sigstack_zero_bits; static size_t sigstack_alignment;
SIZE_T size; SIZE_T size;
NTSTATUS status; NTSTATUS status;
if (!sigstack_zero_bits) if (!sigstack_alignment)
{ {
size_t min_size = teb_size + max( MINSIGSTKSZ, 8192 ); size_t min_size = teb_size + max( MINSIGSTKSZ, 8192 );
/* find the first power of two not smaller than min_size */ /* find the first power of two not smaller than min_size */
sigstack_zero_bits = 12; sigstack_alignment = 12;
while ((1u << sigstack_zero_bits) < min_size) sigstack_zero_bits++; while ((1u << sigstack_alignment) < min_size) sigstack_alignment++;
signal_stack_size = (1 << sigstack_zero_bits) - teb_size; signal_stack_size = (1 << sigstack_alignment) - teb_size;
assert( sizeof(TEB) <= teb_size ); assert( sizeof(TEB) <= teb_size );
} }
size = 1 << sigstack_zero_bits; size = 1 << sigstack_alignment;
*teb = NULL; *teb = NULL;
if (!(status = NtAllocateVirtualMemory( NtCurrentProcess(), (void **)teb, sigstack_zero_bits, if (!(status = virtual_alloc_aligned( (void **)teb, 0, &size, MEM_COMMIT | MEM_TOP_DOWN,
&size, MEM_COMMIT | MEM_TOP_DOWN, PAGE_READWRITE ))) PAGE_READWRITE, sigstack_alignment )))
{ {
(*teb)->Tib.Self = &(*teb)->Tib; (*teb)->Tib.Self = &(*teb)->Tib;
(*teb)->Tib.ExceptionList = (void *)~0UL; (*teb)->Tib.ExceptionList = (void *)~0UL;
......
...@@ -2312,25 +2312,25 @@ static void ldt_unlock(void) ...@@ -2312,25 +2312,25 @@ static void ldt_unlock(void)
*/ */
NTSTATUS signal_alloc_thread( TEB **teb ) NTSTATUS signal_alloc_thread( TEB **teb )
{ {
static size_t sigstack_zero_bits; static size_t sigstack_alignment;
struct x86_thread_data *thread_data; struct x86_thread_data *thread_data;
SIZE_T size; SIZE_T size;
void *addr = NULL; void *addr = NULL;
NTSTATUS status; NTSTATUS status;
if (!sigstack_zero_bits) if (!sigstack_alignment)
{ {
size_t min_size = teb_size + max( MINSIGSTKSZ, 8192 ); size_t min_size = teb_size + max( MINSIGSTKSZ, 8192 );
/* find the first power of two not smaller than min_size */ /* find the first power of two not smaller than min_size */
sigstack_zero_bits = 12; sigstack_alignment = 12;
while ((1u << sigstack_zero_bits) < min_size) sigstack_zero_bits++; while ((1u << sigstack_alignment) < min_size) sigstack_alignment++;
signal_stack_mask = (1 << sigstack_zero_bits) - 1; signal_stack_mask = (1 << sigstack_alignment) - 1;
signal_stack_size = (1 << sigstack_zero_bits) - teb_size; signal_stack_size = (1 << sigstack_alignment) - teb_size;
} }
size = signal_stack_mask + 1; size = signal_stack_mask + 1;
if (!(status = NtAllocateVirtualMemory( NtCurrentProcess(), &addr, sigstack_zero_bits, if (!(status = virtual_alloc_aligned( &addr, 0, &size, MEM_COMMIT | MEM_TOP_DOWN,
&size, MEM_COMMIT | MEM_TOP_DOWN, PAGE_READWRITE ))) PAGE_READWRITE, sigstack_alignment )))
{ {
*teb = addr; *teb = addr;
(*teb)->Tib.Self = &(*teb)->Tib; (*teb)->Tib.Self = &(*teb)->Tib;
......
...@@ -1018,22 +1018,22 @@ int CDECL __wine_set_signal_handler(unsigned int sig, wine_signal_handler wsh) ...@@ -1018,22 +1018,22 @@ int CDECL __wine_set_signal_handler(unsigned int sig, wine_signal_handler wsh)
*/ */
NTSTATUS signal_alloc_thread( TEB **teb ) NTSTATUS signal_alloc_thread( TEB **teb )
{ {
static size_t sigstack_zero_bits; static size_t sigstack_alignment;
SIZE_T size; SIZE_T size;
NTSTATUS status; NTSTATUS status;
if (!sigstack_zero_bits) if (!sigstack_alignment)
{ {
size_t min_size = page_size; /* this is just for the TEB, we don't use a signal stack yet */ size_t min_size = page_size; /* this is just for the TEB, we don't use a signal stack yet */
/* find the first power of two not smaller than min_size */ /* find the first power of two not smaller than min_size */
while ((1u << sigstack_zero_bits) < min_size) sigstack_zero_bits++; while ((1u << sigstack_alignment) < min_size) sigstack_alignment++;
assert( sizeof(TEB) <= min_size ); assert( sizeof(TEB) <= min_size );
} }
size = 1 << sigstack_zero_bits; size = 1 << sigstack_alignment;
*teb = NULL; *teb = NULL;
if (!(status = NtAllocateVirtualMemory( NtCurrentProcess(), (void **)teb, sigstack_zero_bits, if (!(status = virtual_alloc_aligned( (void **)teb, 0, &size, MEM_COMMIT | MEM_TOP_DOWN,
&size, MEM_COMMIT | MEM_TOP_DOWN, PAGE_READWRITE ))) PAGE_READWRITE, sigstack_alignment )))
{ {
(*teb)->Tib.Self = &(*teb)->Tib; (*teb)->Tib.Self = &(*teb)->Tib;
(*teb)->Tib.ExceptionList = (void *)~0UL; (*teb)->Tib.ExceptionList = (void *)~0UL;
......
...@@ -3263,24 +3263,24 @@ int CDECL __wine_set_signal_handler(unsigned int sig, wine_signal_handler wsh) ...@@ -3263,24 +3263,24 @@ int CDECL __wine_set_signal_handler(unsigned int sig, wine_signal_handler wsh)
*/ */
NTSTATUS signal_alloc_thread( TEB **teb ) NTSTATUS signal_alloc_thread( TEB **teb )
{ {
static size_t sigstack_zero_bits; static size_t sigstack_alignment;
SIZE_T size; SIZE_T size;
NTSTATUS status; NTSTATUS status;
if (!sigstack_zero_bits) if (!sigstack_alignment)
{ {
size_t min_size = teb_size + max( MINSIGSTKSZ, 8192 ); size_t min_size = teb_size + max( MINSIGSTKSZ, 8192 );
/* find the first power of two not smaller than min_size */ /* find the first power of two not smaller than min_size */
sigstack_zero_bits = 12; sigstack_alignment = 12;
while ((1u << sigstack_zero_bits) < min_size) sigstack_zero_bits++; while ((1u << sigstack_alignment) < min_size) sigstack_alignment++;
signal_stack_size = (1 << sigstack_zero_bits) - teb_size; signal_stack_size = (1 << sigstack_alignment) - teb_size;
assert( sizeof(TEB) <= teb_size ); assert( sizeof(TEB) <= teb_size );
} }
size = 1 << sigstack_zero_bits; size = 1 << sigstack_alignment;
*teb = NULL; *teb = NULL;
if (!(status = NtAllocateVirtualMemory( NtCurrentProcess(), (void **)teb, sigstack_zero_bits, if (!(status = virtual_alloc_aligned( (void **)teb, 0, &size, MEM_COMMIT | MEM_TOP_DOWN,
&size, MEM_COMMIT | MEM_TOP_DOWN, PAGE_READWRITE ))) PAGE_READWRITE, sigstack_alignment )))
{ {
(*teb)->Tib.Self = &(*teb)->Tib; (*teb)->Tib.Self = &(*teb)->Tib;
(*teb)->Tib.ExceptionList = (void *)~0UL; (*teb)->Tib.ExceptionList = (void *)~0UL;
......
...@@ -55,7 +55,6 @@ static void test_AllocateVirtualMemory(void) ...@@ -55,7 +55,6 @@ static void test_AllocateVirtualMemory(void)
addr2 = (char *)addr1 + 0x1000; addr2 = (char *)addr1 + 0x1000;
status = NtAllocateVirtualMemory(NtCurrentProcess(), &addr2, 12, &size, status = NtAllocateVirtualMemory(NtCurrentProcess(), &addr2, 12, &size,
MEM_RESERVE | MEM_COMMIT, PAGE_EXECUTE_READWRITE); MEM_RESERVE | MEM_COMMIT, PAGE_EXECUTE_READWRITE);
todo_wine
ok(status == STATUS_CONFLICTING_ADDRESSES, "NtAllocateVirtualMemory returned %08x\n", status); ok(status == STATUS_CONFLICTING_ADDRESSES, "NtAllocateVirtualMemory returned %08x\n", status);
if (status == STATUS_SUCCESS) if (status == STATUS_SUCCESS)
{ {
...@@ -141,12 +140,11 @@ static void test_AllocateVirtualMemory(void) ...@@ -141,12 +140,11 @@ static void test_AllocateVirtualMemory(void)
} }
else else
{ {
todo_wine
ok(status == STATUS_SUCCESS || status == STATUS_NO_MEMORY, ok(status == STATUS_SUCCESS || status == STATUS_NO_MEMORY,
"NtAllocateVirtualMemory returned %08x\n", status); "NtAllocateVirtualMemory returned %08x\n", status);
if (status == STATUS_SUCCESS) if (status == STATUS_SUCCESS)
{ {
todo_wine todo_wine_if((UINT_PTR)addr2 & ~zero_bits)
ok(((UINT_PTR)addr2 & ~zero_bits) == 0, ok(((UINT_PTR)addr2 & ~zero_bits) == 0,
"NtAllocateVirtualMemory returned address %p\n", addr2); "NtAllocateVirtualMemory returned address %p\n", addr2);
......
...@@ -184,8 +184,7 @@ void thread_init(void) ...@@ -184,8 +184,7 @@ void thread_init(void)
addr = NULL; addr = NULL;
size = sizeof(*peb); size = sizeof(*peb);
NtAllocateVirtualMemory( NtCurrentProcess(), &addr, 1, &size, virtual_alloc_aligned( &addr, 0, &size, MEM_COMMIT | MEM_TOP_DOWN, PAGE_READWRITE, 1 );
MEM_COMMIT | MEM_TOP_DOWN, PAGE_READWRITE );
peb = addr; peb = addr;
peb->FastPebLock = &peb_lock; peb->FastPebLock = &peb_lock;
......
...@@ -1083,7 +1083,7 @@ static NTSTATUS map_fixed_area( void *base, size_t size, unsigned int vprot ) ...@@ -1083,7 +1083,7 @@ static NTSTATUS map_fixed_area( void *base, size_t size, unsigned int vprot )
* The csVirtual section must be held by caller. * The csVirtual section must be held by caller.
*/ */
static NTSTATUS map_view( struct file_view **view_ret, void *base, size_t size, size_t mask, static NTSTATUS map_view( struct file_view **view_ret, void *base, size_t size, size_t mask,
int top_down, unsigned int vprot ) int top_down, unsigned int vprot, size_t zero_bits )
{ {
void *ptr; void *ptr;
NTSTATUS status; NTSTATUS status;
...@@ -1101,6 +1101,9 @@ static NTSTATUS map_view( struct file_view **view_ret, void *base, size_t size, ...@@ -1101,6 +1101,9 @@ static NTSTATUS map_view( struct file_view **view_ret, void *base, size_t size,
size_t view_size = size + mask + 1; size_t view_size = size + mask + 1;
struct alloc_area alloc; struct alloc_area alloc;
if (zero_bits)
FIXME("Unimplemented zero_bits parameter value\n");
alloc.size = size; alloc.size = size;
alloc.mask = mask; alloc.mask = mask;
alloc.top_down = top_down; alloc.top_down = top_down;
...@@ -1284,7 +1287,7 @@ static NTSTATUS allocate_dos_memory( struct file_view **view, unsigned int vprot ...@@ -1284,7 +1287,7 @@ static NTSTATUS allocate_dos_memory( struct file_view **view, unsigned int vprot
if (addr != low_64k) if (addr != low_64k)
{ {
if (addr != (void *)-1) munmap( addr, dosmem_size - 0x10000 ); if (addr != (void *)-1) munmap( addr, dosmem_size - 0x10000 );
return map_view( view, NULL, dosmem_size, 0xffff, 0, vprot ); return map_view( view, NULL, dosmem_size, 0xffff, 0, vprot, 0 );
} }
} }
...@@ -1388,11 +1391,11 @@ static NTSTATUS map_image( HANDLE hmapping, ACCESS_MASK access, int fd, SIZE_T m ...@@ -1388,11 +1391,11 @@ static NTSTATUS map_image( HANDLE hmapping, ACCESS_MASK access, int fd, SIZE_T m
if (base >= (char *)address_space_start) /* make sure the DOS area remains free */ if (base >= (char *)address_space_start) /* make sure the DOS area remains free */
status = map_view( &view, base, total_size, mask, FALSE, SEC_IMAGE | SEC_FILE | status = map_view( &view, base, total_size, mask, FALSE, SEC_IMAGE | SEC_FILE |
VPROT_COMMITTED | VPROT_READ | VPROT_EXEC | VPROT_WRITECOPY ); VPROT_COMMITTED | VPROT_READ | VPROT_EXEC | VPROT_WRITECOPY, 0 );
if (status != STATUS_SUCCESS) if (status != STATUS_SUCCESS)
status = map_view( &view, NULL, total_size, mask, FALSE, SEC_IMAGE | SEC_FILE | status = map_view( &view, NULL, total_size, mask, FALSE, SEC_IMAGE | SEC_FILE |
VPROT_COMMITTED | VPROT_READ | VPROT_EXEC | VPROT_WRITECOPY ); VPROT_COMMITTED | VPROT_READ | VPROT_EXEC | VPROT_WRITECOPY, 0 );
if (status != STATUS_SUCCESS) goto error; if (status != STATUS_SUCCESS) goto error;
...@@ -1713,7 +1716,7 @@ NTSTATUS virtual_map_section( HANDLE handle, PVOID *addr_ptr, ULONG zero_bits, S ...@@ -1713,7 +1716,7 @@ NTSTATUS virtual_map_section( HANDLE handle, PVOID *addr_ptr, ULONG zero_bits, S
get_vprot_flags( protect, &vprot, sec_flags & SEC_IMAGE ); get_vprot_flags( protect, &vprot, sec_flags & SEC_IMAGE );
vprot |= sec_flags; vprot |= sec_flags;
if (!(sec_flags & SEC_RESERVE)) vprot |= VPROT_COMMITTED; if (!(sec_flags & SEC_RESERVE)) vprot |= VPROT_COMMITTED;
res = map_view( &view, *addr_ptr, size, mask, FALSE, vprot ); res = map_view( &view, *addr_ptr, size, mask, FALSE, vprot, 0 );
if (res) if (res)
{ {
server_leave_uninterrupted_section( &csVirtual, &sigset ); server_leave_uninterrupted_section( &csVirtual, &sigset );
...@@ -1946,7 +1949,7 @@ NTSTATUS virtual_alloc_thread_stack( TEB *teb, SIZE_T reserve_size, SIZE_T commi ...@@ -1946,7 +1949,7 @@ NTSTATUS virtual_alloc_thread_stack( TEB *teb, SIZE_T reserve_size, SIZE_T commi
server_enter_uninterrupted_section( &csVirtual, &sigset ); server_enter_uninterrupted_section( &csVirtual, &sigset );
if ((status = map_view( &view, NULL, size + extra_size, 0xffff, 0, if ((status = map_view( &view, NULL, size + extra_size, 0xffff, 0,
VPROT_READ | VPROT_WRITE | VPROT_COMMITTED )) != STATUS_SUCCESS) VPROT_READ | VPROT_WRITE | VPROT_COMMITTED, 0 )) != STATUS_SUCCESS)
goto done; goto done;
#ifdef VALGRIND_STACK_REGISTER #ifdef VALGRIND_STACK_REGISTER
...@@ -2461,19 +2464,14 @@ void virtual_set_large_address_space(void) ...@@ -2461,19 +2464,14 @@ void virtual_set_large_address_space(void)
NTSTATUS WINAPI NtAllocateVirtualMemory( HANDLE process, PVOID *ret, ULONG zero_bits, NTSTATUS WINAPI NtAllocateVirtualMemory( HANDLE process, PVOID *ret, ULONG zero_bits,
SIZE_T *size_ptr, ULONG type, ULONG protect ) SIZE_T *size_ptr, ULONG type, ULONG protect )
{ {
void *base;
unsigned int vprot;
SIZE_T size = *size_ptr; SIZE_T size = *size_ptr;
SIZE_T mask = get_mask( zero_bits );
NTSTATUS status = STATUS_SUCCESS; NTSTATUS status = STATUS_SUCCESS;
BOOL is_dos_memory = FALSE;
struct file_view *view;
sigset_t sigset;
TRACE("%p %p %08lx %x %08x\n", process, *ret, size, type, protect ); TRACE("%p %p %08lx %x %08x\n", process, *ret, size, type, protect );
if (!size) return STATUS_INVALID_PARAMETER; if (!size) return STATUS_INVALID_PARAMETER;
if (!mask) return STATUS_INVALID_PARAMETER_3; if (zero_bits > 21 && zero_bits < 32) return STATUS_INVALID_PARAMETER_3;
if (!is_win64 && !is_wow64 && zero_bits >= 32) return STATUS_INVALID_PARAMETER_3;
if (process != NtCurrentProcess()) if (process != NtCurrentProcess())
{ {
...@@ -2499,6 +2497,27 @@ NTSTATUS WINAPI NtAllocateVirtualMemory( HANDLE process, PVOID *ret, ULONG zero_ ...@@ -2499,6 +2497,27 @@ NTSTATUS WINAPI NtAllocateVirtualMemory( HANDLE process, PVOID *ret, ULONG zero_
return result.virtual_alloc.status; return result.virtual_alloc.status;
} }
return virtual_alloc_aligned( ret, zero_bits, size_ptr, type, protect, 0 );
}
/***********************************************************************
* virtual_alloc_aligned (NTDLL.@)
*
* Same as NtAllocateVirtualMemory but with an alignment parameter
*/
NTSTATUS virtual_alloc_aligned( PVOID *ret, ULONG zero_bits, SIZE_T *size_ptr,
ULONG type, ULONG protect, ULONG alignment )
{
void *base;
unsigned int vprot;
SIZE_T size = *size_ptr;
SIZE_T mask = get_mask( alignment );
NTSTATUS status = STATUS_SUCCESS;
BOOL is_dos_memory = FALSE;
struct file_view *view;
sigset_t sigset;
/* Round parameters to a page boundary */ /* Round parameters to a page boundary */
if (is_beyond_limit( 0, size, working_set_limit )) return STATUS_WORKING_SET_LIMIT_RANGE; if (is_beyond_limit( 0, size, working_set_limit )) return STATUS_WORKING_SET_LIMIT_RANGE;
...@@ -2550,7 +2569,7 @@ NTSTATUS WINAPI NtAllocateVirtualMemory( HANDLE process, PVOID *ret, ULONG zero_ ...@@ -2550,7 +2569,7 @@ NTSTATUS WINAPI NtAllocateVirtualMemory( HANDLE process, PVOID *ret, ULONG zero_
if (vprot & VPROT_WRITECOPY) status = STATUS_INVALID_PAGE_PROTECTION; if (vprot & VPROT_WRITECOPY) status = STATUS_INVALID_PAGE_PROTECTION;
else if (is_dos_memory) status = allocate_dos_memory( &view, vprot ); else if (is_dos_memory) status = allocate_dos_memory( &view, vprot );
else status = map_view( &view, base, size, mask, type & MEM_TOP_DOWN, vprot ); else status = map_view( &view, base, size, mask, type & MEM_TOP_DOWN, vprot, zero_bits );
if (status == STATUS_SUCCESS) base = view->base; if (status == STATUS_SUCCESS) base = view->base;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment