From ec3982175062ff19eaaf2f2a12dcfa73fb286f87 Mon Sep 17 00:00:00 2001 From: Alexandre Julliard Date: Thu, 18 Sep 2003 20:58:41 +0000 Subject: [PATCH] Moved 16-bit global heap, DOS memory and selector functions to dlls/kernel. --- dlls/kernel/Makefile.in | 3 + {msdos => dlls/kernel}/dosmem.c | 0 dlls/kernel/global16.c | 1046 ++++++++++++++++++++++++++ dlls/kernel/heap.c | 185 +++++ {memory => dlls/kernel}/selector.c | 0 dlls/ntdll/Makefile.in | 2 - memory/global.c | 1119 +--------------------------- 7 files changed, 1236 insertions(+), 1119 deletions(-) rename {msdos => dlls/kernel}/dosmem.c (100%) create mode 100644 dlls/kernel/global16.c rename {memory => dlls/kernel}/selector.c (100%) diff --git a/dlls/kernel/Makefile.in b/dlls/kernel/Makefile.in index 776152e70b3..3ad890951ae 100644 --- a/dlls/kernel/Makefile.in +++ b/dlls/kernel/Makefile.in @@ -27,12 +27,14 @@ C_SRCS = \ computername.c \ console.c \ debugger.c \ + dosmem.c \ editline.c \ except.c \ fiber.c \ file.c \ file16.c \ format_msg.c \ + global16.c \ heap.c \ instr.c \ kernel_main.c \ @@ -48,6 +50,7 @@ C_SRCS = \ relay16.c \ resource.c \ resource16.c \ + selector.c \ snoop16.c \ stress.c \ string.c \ diff --git a/msdos/dosmem.c b/dlls/kernel/dosmem.c similarity index 100% rename from msdos/dosmem.c rename to dlls/kernel/dosmem.c diff --git a/dlls/kernel/global16.c b/dlls/kernel/global16.c new file mode 100644 index 00000000000..4c106f797ae --- /dev/null +++ b/dlls/kernel/global16.c @@ -0,0 +1,1046 @@ +/* + * Global heap functions + * + * Copyright 1995 Alexandre Julliard + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2.1 of the License, or (at your option) any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +/* 0xffff sometimes seems to mean: CURRENT_DS */ + +#include "config.h" +#include "wine/port.h" + +#include +#include +#include +#include +#ifdef HAVE_UNISTD_H +# include +#endif +#include +#ifdef HAVE_SYS_PARAM_H +#include +#endif +#ifdef HAVE_SYS_SYSCTL_H +#include +#endif + +#include "wine/winbase16.h" +#include "ntstatus.h" +#include "global.h" +#include "toolhelp.h" +#include "selectors.h" +#include "miscemu.h" +#include "stackframe.h" +#include "wine/debug.h" +#include "winerror.h" + +WINE_DEFAULT_DEBUG_CHANNEL(global); + + /* Global arena block */ +typedef struct +{ + DWORD base; /* Base address (0 if discarded) */ + DWORD size; /* Size in bytes (0 indicates a free block) */ + HGLOBAL16 handle; /* Handle for this block */ + HGLOBAL16 hOwner; /* Owner of this block */ + BYTE lockCount; /* Count of GlobalFix() calls */ + BYTE pageLockCount; /* Count of GlobalPageLock() calls */ + BYTE flags; /* Allocation flags */ + BYTE selCount; /* Number of selectors allocated for this block */ +} GLOBALARENA; + + /* Flags definitions */ +#define GA_MOVEABLE 0x02 /* same as GMEM_MOVEABLE */ +#define GA_DGROUP 0x04 +#define GA_DISCARDABLE 0x08 +#define GA_IPCSHARE 0x10 /* same as GMEM_DDESHARE */ +#define GA_DOSMEM 0x20 + +/* Arena array (FIXME) */ +extern GLOBALARENA *pGlobalArena; +extern int globalArenaSize; + +#define GLOBAL_MAX_ALLOC_SIZE 0x00ff0000 /* Largest allocation is 16M - 64K */ + +#define VALID_HANDLE(handle) (((handle)>>__AHSHIFT)> __AHSHIFT)) + + +/*********************************************************************** + * GLOBAL_GetArena + * + * Return the arena for a given selector, growing the arena array if needed. + */ +static GLOBALARENA *GLOBAL_GetArena( WORD sel, WORD selcount ) +{ + if (((sel >> __AHSHIFT) + selcount) > globalArenaSize) + { + int newsize = ((sel >> __AHSHIFT) + selcount + 0xff) & ~0xff; + GLOBALARENA *pNewArena = realloc( pGlobalArena, + newsize * sizeof(GLOBALARENA) ); + if (!pNewArena) return 0; + pGlobalArena = pNewArena; + memset( pGlobalArena + globalArenaSize, 0, + (newsize - globalArenaSize) * sizeof(GLOBALARENA) ); + globalArenaSize = newsize; + } + return pGlobalArena + (sel >> __AHSHIFT); +} + +void debug_handles(void) +{ + int printed=0; + int i; + for (i = globalArenaSize-1 ; i>=0 ; i--) { + if (pGlobalArena[i].size!=0 && (pGlobalArena[i].handle & 0x8000)){ + printed=1; + DPRINTF("0x%08x, ",pGlobalArena[i].handle); + } + } + if (printed) + DPRINTF("\n"); +} + + +/*********************************************************************** + * GLOBAL_CreateBlock + * + * Create a global heap block for a fixed range of linear memory. + */ +HGLOBAL16 GLOBAL_CreateBlock( WORD flags, const void *ptr, DWORD size, + HGLOBAL16 hOwner, unsigned char selflags ) +{ + WORD sel, selcount; + GLOBALARENA *pArena; + + /* Allocate the selector(s) */ + + sel = SELECTOR_AllocBlock( ptr, size, selflags ); + if (!sel) return 0; + selcount = (size + 0xffff) / 0x10000; + + if (!(pArena = GLOBAL_GetArena( sel, selcount ))) + { + SELECTOR_FreeBlock( sel ); + return 0; + } + + /* Fill the arena block */ + + pArena->base = (DWORD)ptr; + pArena->size = GetSelectorLimit16(sel) + 1; + pArena->handle = (flags & GMEM_MOVEABLE) ? sel - 1 : sel; + pArena->hOwner = hOwner; + pArena->lockCount = 0; + pArena->pageLockCount = 0; + pArena->flags = flags & GA_MOVEABLE; + if (flags & GMEM_DISCARDABLE) pArena->flags |= GA_DISCARDABLE; + if (flags & GMEM_DDESHARE) pArena->flags |= GA_IPCSHARE; + if (!(selflags & (WINE_LDT_FLAGS_CODE^WINE_LDT_FLAGS_DATA))) pArena->flags |= GA_DGROUP; + pArena->selCount = selcount; + if (selcount > 1) /* clear the next arena blocks */ + memset( pArena + 1, 0, (selcount - 1) * sizeof(GLOBALARENA) ); + + return pArena->handle; +} + + +/*********************************************************************** + * GLOBAL_FreeBlock + * + * Free a block allocated by GLOBAL_CreateBlock, without touching + * the associated linear memory range. + */ +BOOL16 GLOBAL_FreeBlock( HGLOBAL16 handle ) +{ + WORD sel; + GLOBALARENA *pArena; + + if (!handle) return TRUE; + sel = GlobalHandleToSel16( handle ); + if (!VALID_HANDLE(sel)) return FALSE; + pArena = GET_ARENA_PTR(sel); + SELECTOR_FreeBlock( sel ); + memset( pArena, 0, sizeof(GLOBALARENA) ); + return TRUE; +} + +/*********************************************************************** + * GLOBAL_MoveBlock + */ +BOOL16 GLOBAL_MoveBlock( HGLOBAL16 handle, const void *ptr, DWORD size ) +{ + WORD sel; + GLOBALARENA *pArena; + + if (!handle) return TRUE; + sel = GlobalHandleToSel16( handle ); + if (!VALID_HANDLE(sel)) return FALSE; + pArena = GET_ARENA_PTR(sel); + if (pArena->selCount != 1) + return FALSE; + + pArena->base = (DWORD)ptr; + pArena->size = size; + SELECTOR_ReallocBlock( sel, ptr, size ); + return TRUE; +} + +/*********************************************************************** + * GLOBAL_Alloc + * + * Implementation of GlobalAlloc16() + */ +HGLOBAL16 GLOBAL_Alloc( UINT16 flags, DWORD size, HGLOBAL16 hOwner, unsigned char selflags ) +{ + void *ptr; + HGLOBAL16 handle; + + TRACE("%ld flags=%04x\n", size, flags ); + + /* If size is 0, create a discarded block */ + + if (size == 0) return GLOBAL_CreateBlock( flags, NULL, 1, hOwner, selflags ); + + /* Fixup the size */ + + if (size >= GLOBAL_MAX_ALLOC_SIZE - 0x1f) return 0; + size = (size + 0x1f) & ~0x1f; + + /* Allocate the linear memory */ + ptr = HeapAlloc( GetProcessHeap(), 0, size ); + /* FIXME: free discardable blocks and try again? */ + if (!ptr) return 0; + + /* Allocate the selector(s) */ + + handle = GLOBAL_CreateBlock( flags, ptr, size, hOwner, selflags ); + if (!handle) + { + HeapFree( GetProcessHeap(), 0, ptr ); + return 0; + } + + if (flags & GMEM_ZEROINIT) memset( ptr, 0, size ); + return handle; +} + +/*********************************************************************** + * GlobalAlloc (KERNEL.15) + * GlobalAlloc16 (KERNEL32.24) + * RETURNS + * Handle: Success + * NULL: Failure + */ +HGLOBAL16 WINAPI GlobalAlloc16( + UINT16 flags, /* [in] Object allocation attributes */ + DWORD size /* [in] Number of bytes to allocate */ +) { + HANDLE16 owner = GetCurrentPDB16(); + + if (flags & GMEM_DDESHARE) + owner = GetExePtr(owner); /* Make it a module handle */ + return GLOBAL_Alloc( flags, size, owner, WINE_LDT_FLAGS_DATA ); +} + + +/*********************************************************************** + * GlobalReAlloc (KERNEL.16) + * RETURNS + * Handle: Success + * NULL: Failure + */ +HGLOBAL16 WINAPI GlobalReAlloc16( + HGLOBAL16 handle, /* [in] Handle of global memory object */ + DWORD size, /* [in] New size of block */ + UINT16 flags /* [in] How to reallocate object */ +) { + WORD selcount; + DWORD oldsize; + void *ptr, *newptr; + GLOBALARENA *pArena, *pNewArena; + WORD sel = GlobalHandleToSel16( handle ); + + TRACE("%04x %ld flags=%04x\n", + handle, size, flags ); + if (!handle) return 0; + + if (!VALID_HANDLE(handle)) + { + WARN("Invalid handle 0x%04x!\n", handle); + return 0; + } + pArena = GET_ARENA_PTR( handle ); + + /* Discard the block if requested */ + + if ((size == 0) && (flags & GMEM_MOVEABLE) && !(flags & GMEM_MODIFY)) + { + if (!(pArena->flags & GA_MOVEABLE) || + !(pArena->flags & GA_DISCARDABLE) || + (pArena->lockCount > 0) || (pArena->pageLockCount > 0)) return 0; + if (pArena->flags & GA_DOSMEM) + DOSMEM_FreeBlock( (void *)pArena->base ); + else + HeapFree( GetProcessHeap(), 0, (void *)pArena->base ); + pArena->base = 0; + + /* Note: we rely on the fact that SELECTOR_ReallocBlock won't + * change the selector if we are shrinking the block. + * FIXME: shouldn't we keep selectors until the block is deleted? + */ + SELECTOR_ReallocBlock( sel, 0, 1 ); + return handle; + } + + /* Fixup the size */ + + if (size > GLOBAL_MAX_ALLOC_SIZE - 0x20) return 0; + if (size == 0) size = 0x20; + else size = (size + 0x1f) & ~0x1f; + + /* Change the flags */ + + if (flags & GMEM_MODIFY) + { + /* Change the flags, leaving GA_DGROUP alone */ + pArena->flags = (pArena->flags & GA_DGROUP) | (flags & GA_MOVEABLE); + if (flags & GMEM_DISCARDABLE) pArena->flags |= GA_DISCARDABLE; + return handle; + } + + /* Reallocate the linear memory */ + + ptr = (void *)pArena->base; + oldsize = pArena->size; + TRACE("oldbase %p oldsize %08lx newsize %08lx\n", ptr,oldsize,size); + if (ptr && (size == oldsize)) return handle; /* Nothing to do */ + + if (pArena->flags & GA_DOSMEM) + { + if (DOSMEM_ResizeBlock(ptr, size, TRUE) == size) + newptr = ptr; + else if(pArena->pageLockCount > 0) + newptr = 0; + else + { + newptr = DOSMEM_GetBlock( size, 0 ); + if (newptr) + { + memcpy( newptr, ptr, oldsize ); + DOSMEM_FreeBlock( ptr ); + } + } + } + else + { + /* + * if more than one reader (e.g. some pointer has been + * given out by GetVDMPointer32W16), + * only try to realloc in place + */ + newptr = HeapReAlloc( GetProcessHeap(), + (pArena->pageLockCount > 0) ? + HEAP_REALLOC_IN_PLACE_ONLY : 0, + ptr, size ); + } + + if (!newptr) + { + FIXME("Realloc failed lock %d\n",pArena->pageLockCount); + if (pArena->pageLockCount <1) + { + if (pArena->flags & GA_DOSMEM) + DOSMEM_FreeBlock( (void *)pArena->base ); + else + HeapFree( GetProcessHeap(), 0, ptr ); + SELECTOR_FreeBlock( sel ); + memset( pArena, 0, sizeof(GLOBALARENA) ); + } + return 0; + } + ptr = newptr; + + /* Reallocate the selector(s) */ + + sel = SELECTOR_ReallocBlock( sel, ptr, size ); + if (!sel) + { + if (pArena->flags & GA_DOSMEM) + DOSMEM_FreeBlock( (void *)pArena->base ); + else + HeapFree( GetProcessHeap(), 0, ptr ); + memset( pArena, 0, sizeof(GLOBALARENA) ); + return 0; + } + selcount = (size + 0xffff) / 0x10000; + + if (!(pNewArena = GLOBAL_GetArena( sel, selcount ))) + { + if (pArena->flags & GA_DOSMEM) + DOSMEM_FreeBlock( (void *)pArena->base ); + else + HeapFree( GetProcessHeap(), 0, ptr ); + SELECTOR_FreeBlock( sel ); + return 0; + } + + /* Fill the new arena block + As we may have used HEAP_REALLOC_IN_PLACE_ONLY, areas may overlap*/ + + if (pNewArena != pArena) memmove( pNewArena, pArena, sizeof(GLOBALARENA) ); + pNewArena->base = (DWORD)ptr; + pNewArena->size = GetSelectorLimit16(sel) + 1; + pNewArena->selCount = selcount; + pNewArena->handle = (pNewArena->flags & GA_MOVEABLE) ? sel - 1 : sel; + + if (selcount > 1) /* clear the next arena blocks */ + memset( pNewArena + 1, 0, (selcount - 1) * sizeof(GLOBALARENA) ); + + if ((oldsize < size) && (flags & GMEM_ZEROINIT)) + memset( (char *)ptr + oldsize, 0, size - oldsize ); + return pNewArena->handle; +} + + +/*********************************************************************** + * GlobalFree (KERNEL.17) + * GlobalFree16 (KERNEL32.31) + * RETURNS + * NULL: Success + * Handle: Failure + */ +HGLOBAL16 WINAPI GlobalFree16( + HGLOBAL16 handle /* [in] Handle of global memory object */ +) { + void *ptr; + + if (!VALID_HANDLE(handle)) + { + WARN("Invalid handle 0x%04x passed to GlobalFree16!\n",handle); + return 0; + } + ptr = (void *)GET_ARENA_PTR(handle)->base; + + TRACE("%04x\n", handle ); + if (!GLOBAL_FreeBlock( handle )) return handle; /* failed */ + if (ptr) HeapFree( GetProcessHeap(), 0, ptr ); + return 0; +} + + +/********************************************************************** + * K32WOWGlobalLock16 (KERNEL32.60) + */ +SEGPTR WINAPI K32WOWGlobalLock16( HGLOBAL16 handle ) +{ + WORD sel = GlobalHandleToSel16( handle ); + TRACE("(%04x) -> %08lx\n", handle, MAKELONG( 0, sel ) ); + + if (handle) + { + if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; + + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to WIN16_GlobalLock16!\n",handle); + sel = 0; + } + else if (!GET_ARENA_PTR(handle)->base) + sel = 0; + else + GET_ARENA_PTR(handle)->lockCount++; + } + + return MAKESEGPTR( sel, 0 ); + +} + + +/*********************************************************************** + * GlobalLock (KERNEL.18) + * + * This is the GlobalLock16() function used by 16-bit code. + */ +SEGPTR WINAPI WIN16_GlobalLock16( HGLOBAL16 handle ) +{ + SEGPTR ret = K32WOWGlobalLock16( handle ); + CURRENT_STACK16->ecx = SELECTOROF(ret); /* selector must be returned in CX as well */ + return ret; +} + + +/*********************************************************************** + * GlobalLock16 (KERNEL32.25) + * + * This is the GlobalLock16() function used by 32-bit code. + * + * RETURNS + * Pointer to first byte of memory block + * NULL: Failure + */ +LPVOID WINAPI GlobalLock16( + HGLOBAL16 handle /* [in] Handle of global memory object */ +) { + if (!handle) return 0; + if (!VALID_HANDLE(handle)) + return 0; + GET_ARENA_PTR(handle)->lockCount++; + return (LPVOID)GET_ARENA_PTR(handle)->base; +} + + +/*********************************************************************** + * GlobalUnlock (KERNEL.19) + * GlobalUnlock16 (KERNEL32.26) + * NOTES + * Should the return values be cast to booleans? + * + * RETURNS + * TRUE: Object is still locked + * FALSE: Object is unlocked + */ +BOOL16 WINAPI GlobalUnlock16( + HGLOBAL16 handle /* [in] Handle of global memory object */ +) { + GLOBALARENA *pArena = GET_ARENA_PTR(handle); + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to GlobalUnlock16!\n",handle); + return 0; + } + TRACE("%04x\n", handle ); + if (pArena->lockCount) pArena->lockCount--; + return pArena->lockCount; +} + +/*********************************************************************** + * GlobalChangeLockCount (KERNEL.365) + * + * This is declared as a register function as it has to preserve + * *all* registers, even AX/DX ! + * + */ +void WINAPI GlobalChangeLockCount16( HGLOBAL16 handle, INT16 delta, + CONTEXT86 *context ) +{ + if ( delta == 1 ) + GlobalLock16( handle ); + else if ( delta == -1 ) + GlobalUnlock16( handle ); + else + ERR("(%04X, %d): strange delta value\n", handle, delta ); +} + +/*********************************************************************** + * GlobalSize (KERNEL.20) + * GlobalSize16 (KERNEL32.32) + * RETURNS + * Size in bytes of object + * 0: Failure + */ +DWORD WINAPI GlobalSize16( + HGLOBAL16 handle /* [in] Handle of global memory object */ +) { + TRACE("%04x\n", handle ); + if (!handle) return 0; + if (!VALID_HANDLE(handle)) + return 0; + return GET_ARENA_PTR(handle)->size; +} + + +/*********************************************************************** + * GlobalHandle (KERNEL.21) + * NOTES + * Why is GlobalHandleToSel used here with the sel as input? + * + * RETURNS + * Handle: Success + * NULL: Failure + */ +DWORD WINAPI GlobalHandle16( + WORD sel /* [in] Address of global memory block */ +) { + TRACE("%04x\n", sel ); + if (!VALID_HANDLE(sel)) { + WARN("Invalid handle 0x%04x passed to GlobalHandle16!\n",sel); + return 0; + } + return MAKELONG( GET_ARENA_PTR(sel)->handle, GlobalHandleToSel16(sel) ); +} + +/*********************************************************************** + * GlobalHandleNoRIP (KERNEL.159) + */ +DWORD WINAPI GlobalHandleNoRIP16( WORD sel ) +{ + int i; + for (i = globalArenaSize-1 ; i>=0 ; i--) { + if (pGlobalArena[i].size!=0 && pGlobalArena[i].handle == sel) + return MAKELONG( GET_ARENA_PTR(sel)->handle, GlobalHandleToSel16(sel) ); + } + return 0; +} + + +/*********************************************************************** + * GlobalFlags (KERNEL.22) + * + * NOTES + * Should this return GMEM_INVALID_HANDLE instead of 0 on invalid + * handle? + * + * RETURNS + * Value specifying flags and lock count + * GMEM_INVALID_HANDLE: Invalid handle + */ +UINT16 WINAPI GlobalFlags16( + HGLOBAL16 handle /* [in] Handle of global memory object */ +) { + GLOBALARENA *pArena; + + TRACE("%04x\n", handle ); + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to GlobalFlags16!\n",handle); + return 0; + } + pArena = GET_ARENA_PTR(handle); + return pArena->lockCount | + ((pArena->flags & GA_DISCARDABLE) ? GMEM_DISCARDABLE : 0) | + ((pArena->base == 0) ? GMEM_DISCARDED : 0); +} + + +/*********************************************************************** + * LockSegment (KERNEL.23) + */ +HGLOBAL16 WINAPI LockSegment16( HGLOBAL16 handle ) +{ + TRACE("%04x\n", handle ); + if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to LockSegment16!\n",handle); + return 0; + } + GET_ARENA_PTR(handle)->lockCount++; + return handle; +} + + +/*********************************************************************** + * UnlockSegment (KERNEL.24) + */ +void WINAPI UnlockSegment16( HGLOBAL16 handle ) +{ + TRACE("%04x\n", handle ); + if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to UnlockSegment16!\n",handle); + return; + } + GET_ARENA_PTR(handle)->lockCount--; + /* FIXME: this ought to return the lock count in CX (go figure...) */ +} + + +/*********************************************************************** + * GlobalCompact (KERNEL.25) + */ +DWORD WINAPI GlobalCompact16( DWORD desired ) +{ + return GLOBAL_MAX_ALLOC_SIZE; +} + + +/*********************************************************************** + * GlobalFreeAll (KERNEL.26) + */ +void WINAPI GlobalFreeAll16( HGLOBAL16 owner ) +{ + DWORD i; + GLOBALARENA *pArena; + + pArena = pGlobalArena; + for (i = 0; i < globalArenaSize; i++, pArena++) + { + if ((pArena->size != 0) && (pArena->hOwner == owner)) + GlobalFree16( pArena->handle ); + } +} + + +/*********************************************************************** + * GlobalWire (KERNEL.111) + * GlobalWire16 (KERNEL32.29) + */ +SEGPTR WINAPI GlobalWire16( HGLOBAL16 handle ) +{ + return WIN16_GlobalLock16( handle ); +} + + +/*********************************************************************** + * GlobalUnWire (KERNEL.112) + * GlobalUnWire16 (KERNEL32.30) + */ +BOOL16 WINAPI GlobalUnWire16( HGLOBAL16 handle ) +{ + return !GlobalUnlock16( handle ); +} + + +/*********************************************************************** + * SetSwapAreaSize (KERNEL.106) + */ +LONG WINAPI SetSwapAreaSize16( WORD size ) +{ + FIXME("(%d) - stub!\n", size ); + return MAKELONG( size, 0xffff ); +} + + +/*********************************************************************** + * GlobalLRUOldest (KERNEL.163) + */ +HGLOBAL16 WINAPI GlobalLRUOldest16( HGLOBAL16 handle ) +{ + TRACE("%04x\n", handle ); + if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; + return handle; +} + + +/*********************************************************************** + * GlobalLRUNewest (KERNEL.164) + */ +HGLOBAL16 WINAPI GlobalLRUNewest16( HGLOBAL16 handle ) +{ + TRACE("%04x\n", handle ); + if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; + return handle; +} + + +/*********************************************************************** + * GetFreeSpace (KERNEL.169) + */ +DWORD WINAPI GetFreeSpace16( UINT16 wFlags ) +{ + MEMORYSTATUS ms; + GlobalMemoryStatus( &ms ); + return ms.dwAvailVirtual; +} + +/*********************************************************************** + * GlobalDOSAlloc (KERNEL.184) + * RETURNS + * Address (HW=Paragraph segment; LW=Selector) + */ +DWORD WINAPI GlobalDOSAlloc16( + DWORD size /* [in] Number of bytes to be allocated */ +) { + UINT16 uParagraph; + LPVOID lpBlock = DOSMEM_GetBlock( size, &uParagraph ); + + if( lpBlock ) + { + HMODULE16 hModule = GetModuleHandle16("KERNEL"); + WORD wSelector; + GLOBALARENA *pArena; + + wSelector = GLOBAL_CreateBlock(GMEM_FIXED, lpBlock, size, hModule, WINE_LDT_FLAGS_DATA ); + pArena = GET_ARENA_PTR(wSelector); + pArena->flags |= GA_DOSMEM; + return MAKELONG(wSelector,uParagraph); + } + return 0; +} + + +/*********************************************************************** + * GlobalDOSFree (KERNEL.185) + * RETURNS + * NULL: Success + * sel: Failure + */ +WORD WINAPI GlobalDOSFree16( + WORD sel /* [in] Selector */ +) { + DWORD block = GetSelectorBase(sel); + + if( block && block < 0x100000 ) + { + LPVOID lpBlock = DOSMEM_MapDosToLinear( block ); + if( DOSMEM_FreeBlock( lpBlock ) ) + GLOBAL_FreeBlock( sel ); + sel = 0; + } + return sel; +} + + +/*********************************************************************** + * GlobalPageLock (KERNEL.191) + * GlobalSmartPageLock(KERNEL.230) + */ +WORD WINAPI GlobalPageLock16( HGLOBAL16 handle ) +{ + TRACE("%04x\n", handle ); + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to GlobalPageLock!\n",handle); + return 0; + } + return ++(GET_ARENA_PTR(handle)->pageLockCount); +} + + +/*********************************************************************** + * GlobalPageUnlock (KERNEL.192) + * GlobalSmartPageUnlock(KERNEL.231) + */ +WORD WINAPI GlobalPageUnlock16( HGLOBAL16 handle ) +{ + TRACE("%04x\n", handle ); + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to GlobalPageUnlock!\n",handle); + return 0; + } + return --(GET_ARENA_PTR(handle)->pageLockCount); +} + + +/*********************************************************************** + * GlobalFix (KERNEL.197) + * GlobalFix16 (KERNEL32.27) + */ +WORD WINAPI GlobalFix16( HGLOBAL16 handle ) +{ + TRACE("%04x\n", handle ); + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to GlobalFix16!\n",handle); + return 0; + } + GET_ARENA_PTR(handle)->lockCount++; + + return GlobalHandleToSel16(handle); +} + + +/*********************************************************************** + * GlobalUnfix (KERNEL.198) + * GlobalUnfix16 (KERNEL32.28) + */ +void WINAPI GlobalUnfix16( HGLOBAL16 handle ) +{ + TRACE("%04x\n", handle ); + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to GlobalUnfix16!\n",handle); + return; + } + GET_ARENA_PTR(handle)->lockCount--; +} + + +/*********************************************************************** + * FarSetOwner (KERNEL.403) + */ +void WINAPI FarSetOwner16( HGLOBAL16 handle, HANDLE16 hOwner ) +{ + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to FarSetOwner!\n",handle); + return; + } + GET_ARENA_PTR(handle)->hOwner = hOwner; +} + + +/*********************************************************************** + * FarGetOwner (KERNEL.404) + */ +HANDLE16 WINAPI FarGetOwner16( HGLOBAL16 handle ) +{ + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to FarGetOwner!\n",handle); + return 0; + } + return GET_ARENA_PTR(handle)->hOwner; +} + + +/*********************************************************************** + * GlobalHandleToSel (TOOLHELP.50) + */ +WORD WINAPI GlobalHandleToSel16( HGLOBAL16 handle ) +{ + if (!handle) return 0; + if (!VALID_HANDLE(handle)) { + WARN("Invalid handle 0x%04x passed to GlobalHandleToSel!\n",handle); + return 0; + } + if (!(handle & 7)) + { + WARN("Program attempted invalid selector conversion\n" ); + return handle - 1; + } + return handle | 7; +} + + +/*********************************************************************** + * GlobalFirst (TOOLHELP.51) + */ +BOOL16 WINAPI GlobalFirst16( GLOBALENTRY *pGlobal, WORD wFlags ) +{ + if (wFlags == GLOBAL_LRU) return FALSE; + pGlobal->dwNext = 0; + return GlobalNext16( pGlobal, wFlags ); +} + + +/*********************************************************************** + * GlobalNext (TOOLHELP.52) + */ +BOOL16 WINAPI GlobalNext16( GLOBALENTRY *pGlobal, WORD wFlags) +{ + GLOBALARENA *pArena; + + if (pGlobal->dwNext >= globalArenaSize) return FALSE; + pArena = pGlobalArena + pGlobal->dwNext; + if (wFlags == GLOBAL_FREE) /* only free blocks */ + { + int i; + for (i = pGlobal->dwNext; i < globalArenaSize; i++, pArena++) + if (pArena->size == 0) break; /* block is free */ + if (i >= globalArenaSize) return FALSE; + pGlobal->dwNext = i; + } + + pGlobal->dwAddress = pArena->base; + pGlobal->dwBlockSize = pArena->size; + pGlobal->hBlock = pArena->handle; + pGlobal->wcLock = pArena->lockCount; + pGlobal->wcPageLock = pArena->pageLockCount; + pGlobal->wFlags = (GetCurrentPDB16() == pArena->hOwner); + pGlobal->wHeapPresent = FALSE; + pGlobal->hOwner = pArena->hOwner; + pGlobal->wType = GT_UNKNOWN; + pGlobal->wData = 0; + pGlobal->dwNext++; + return TRUE; +} + + +/*********************************************************************** + * GlobalInfo (TOOLHELP.53) + */ +BOOL16 WINAPI GlobalInfo16( GLOBALINFO *pInfo ) +{ + int i; + GLOBALARENA *pArena; + + pInfo->wcItems = globalArenaSize; + pInfo->wcItemsFree = 0; + pInfo->wcItemsLRU = 0; + for (i = 0, pArena = pGlobalArena; i < globalArenaSize; i++, pArena++) + if (pArena->size == 0) pInfo->wcItemsFree++; + return TRUE; +} + + +/*********************************************************************** + * GlobalEntryHandle (TOOLHELP.54) + */ +BOOL16 WINAPI GlobalEntryHandle16( GLOBALENTRY *pGlobal, HGLOBAL16 hItem ) +{ + GLOBALARENA *pArena = GET_ARENA_PTR(hItem); + + pGlobal->dwAddress = pArena->base; + pGlobal->dwBlockSize = pArena->size; + pGlobal->hBlock = pArena->handle; + pGlobal->wcLock = pArena->lockCount; + pGlobal->wcPageLock = pArena->pageLockCount; + pGlobal->wFlags = (GetCurrentPDB16() == pArena->hOwner); + pGlobal->wHeapPresent = FALSE; + pGlobal->hOwner = pArena->hOwner; + pGlobal->wType = GT_UNKNOWN; + pGlobal->wData = 0; + pGlobal->dwNext++; + return TRUE; +} + + +/*********************************************************************** + * GlobalEntryModule (TOOLHELP.55) + */ +BOOL16 WINAPI GlobalEntryModule16( GLOBALENTRY *pGlobal, HMODULE16 hModule, + WORD wSeg ) +{ + FIXME("(%p, 0x%04x, 0x%04x), stub.\n", pGlobal, hModule, wSeg); + return FALSE; +} + + +/*********************************************************************** + * MemManInfo (TOOLHELP.72) + */ +BOOL16 WINAPI MemManInfo16( MEMMANINFO *info ) +{ + MEMORYSTATUS status; + + /* + * Not unsurprisingly although the documention says you + * _must_ provide the size in the dwSize field, this function + * (under Windows) always fills the structure and returns true. + */ + GlobalMemoryStatus( &status ); + info->wPageSize = getpagesize(); + info->dwLargestFreeBlock = status.dwAvailVirtual; + info->dwMaxPagesAvailable = info->dwLargestFreeBlock / info->wPageSize; + info->dwMaxPagesLockable = info->dwMaxPagesAvailable; + info->dwTotalLinearSpace = status.dwTotalVirtual / info->wPageSize; + info->dwTotalUnlockedPages = info->dwTotalLinearSpace; + info->dwFreePages = info->dwMaxPagesAvailable; + info->dwTotalPages = info->dwTotalLinearSpace; + info->dwFreeLinearSpace = info->dwMaxPagesAvailable; + info->dwSwapFilePages = status.dwTotalPageFile / info->wPageSize; + return TRUE; +} + +/*********************************************************************** + * GetFreeMemInfo (KERNEL.316) + */ +DWORD WINAPI GetFreeMemInfo16(void) +{ + MEMMANINFO info; + MemManInfo16( &info ); + return MAKELONG( info.dwTotalLinearSpace, info.dwMaxPagesAvailable ); +} + +/*********************************************************************** + * A20Proc (KERNEL.165) + * A20_Proc (SYSTEM.20) + */ +void WINAPI A20Proc16( WORD unused ) +{ + /* this is also a NOP in Windows */ +} + +/*********************************************************************** + * LimitEMSPages (KERNEL.156) + */ +DWORD WINAPI LimitEMSPages16( DWORD unused ) +{ + return 0; +} diff --git a/dlls/kernel/heap.c b/dlls/kernel/heap.c index b6d2e8d9e43..9675e77c6b8 100644 --- a/dlls/kernel/heap.c +++ b/dlls/kernel/heap.c @@ -27,6 +27,17 @@ #include #include #include +#include +#include +#ifdef HAVE_SYS_PARAM_H +#include +#endif +#ifdef HAVE_SYS_SYSCTL_H +#include +#endif +#ifdef HAVE_UNISTD_H +# include +#endif #include "windef.h" #include "winbase.h" @@ -957,3 +968,177 @@ void WINAPI FreeMappedBuffer( CONTEXT86 *context ) GlobalFree((HGLOBAL)buffer[0]); } } + +/*********************************************************************** + * GlobalMemoryStatus (KERNEL32.@) + * Provides information about the status of the memory, so apps can tell + * roughly how much they are able to allocate + * + * RETURNS + * None + */ +VOID WINAPI GlobalMemoryStatus( + LPMEMORYSTATUS lpmem +) { + static MEMORYSTATUS cached_memstatus; + static int cache_lastchecked = 0; + SYSTEM_INFO si; +#ifdef linux + FILE *f; +#endif +#if defined(__FreeBSD__) || defined(__NetBSD__) + int *tmp; + int size_sys; + int mib[2] = { CTL_HW }; +#endif + if (time(NULL)==cache_lastchecked) { + memcpy(lpmem,&cached_memstatus,sizeof(MEMORYSTATUS)); + return; + } + cache_lastchecked = time(NULL); + + lpmem->dwMemoryLoad = 0; + lpmem->dwTotalPhys = 16*1024*1024; + lpmem->dwAvailPhys = 16*1024*1024; + lpmem->dwTotalPageFile = 16*1024*1024; + lpmem->dwAvailPageFile = 16*1024*1024; + +#ifdef linux + f = fopen( "/proc/meminfo", "r" ); + if (f) + { + char buffer[256]; + int total, used, free, shared, buffers, cached; + + lpmem->dwLength = sizeof(MEMORYSTATUS); + lpmem->dwTotalPhys = lpmem->dwAvailPhys = 0; + lpmem->dwTotalPageFile = lpmem->dwAvailPageFile = 0; + while (fgets( buffer, sizeof(buffer), f )) + { + /* old style /proc/meminfo ... */ + if (sscanf( buffer, "Mem: %d %d %d %d %d %d", &total, &used, &free, &shared, &buffers, &cached )) + { + lpmem->dwTotalPhys += total; + lpmem->dwAvailPhys += free + buffers + cached; + } + if (sscanf( buffer, "Swap: %d %d %d", &total, &used, &free )) + { + lpmem->dwTotalPageFile += total; + lpmem->dwAvailPageFile += free; + } + + /* new style /proc/meminfo ... */ + if (sscanf(buffer, "MemTotal: %d", &total)) + lpmem->dwTotalPhys = total*1024; + if (sscanf(buffer, "MemFree: %d", &free)) + lpmem->dwAvailPhys = free*1024; + if (sscanf(buffer, "SwapTotal: %d", &total)) + lpmem->dwTotalPageFile = total*1024; + if (sscanf(buffer, "SwapFree: %d", &free)) + lpmem->dwAvailPageFile = free*1024; + if (sscanf(buffer, "Buffers: %d", &buffers)) + lpmem->dwAvailPhys += buffers*1024; + if (sscanf(buffer, "Cached: %d", &cached)) + lpmem->dwAvailPhys += cached*1024; + } + fclose( f ); + + if (lpmem->dwTotalPhys) + { + DWORD TotalPhysical = lpmem->dwTotalPhys+lpmem->dwTotalPageFile; + DWORD AvailPhysical = lpmem->dwAvailPhys+lpmem->dwAvailPageFile; + lpmem->dwMemoryLoad = (TotalPhysical-AvailPhysical) + / (TotalPhysical / 100); + } + } +#elif defined(__FreeBSD__) || defined(__NetBSD__) + mib[1] = HW_PHYSMEM; + sysctl(mib, 2, NULL, &size_sys, NULL, 0); + tmp = malloc(size_sys * sizeof(int)); + sysctl(mib, 2, tmp, &size_sys, NULL, 0); + if (tmp && *tmp) + { + lpmem->dwTotalPhys = *tmp; + free(tmp); + mib[1] = HW_USERMEM; + sysctl(mib, 2, NULL, &size_sys, NULL, 0); + tmp = malloc(size_sys * sizeof(int)); + sysctl(mib, 2, tmp, &size_sys, NULL, 0); + if (tmp && *tmp) + { + lpmem->dwAvailPhys = *tmp; + lpmem->dwTotalPageFile = *tmp; + lpmem->dwAvailPageFile = *tmp; + lpmem->dwMemoryLoad = lpmem->dwTotalPhys - lpmem->dwAvailPhys; + } else + { + lpmem->dwAvailPhys = lpmem->dwTotalPhys; + lpmem->dwTotalPageFile = lpmem->dwTotalPhys; + lpmem->dwAvailPageFile = lpmem->dwTotalPhys; + lpmem->dwMemoryLoad = 0; + } + free(tmp); + + } +#endif + /* Some applications (e.g. QuickTime 6) crash if we tell them there + * is more than 2GB of physical memory. + */ + if (lpmem->dwTotalPhys>2U*1024*1024*1024) + { + lpmem->dwTotalPhys=2U*1024*1024*1024; + lpmem->dwAvailPhys=2U*1024*1024*1024; + } + + /* FIXME: should do something for other systems */ + GetSystemInfo(&si); + lpmem->dwTotalVirtual = (char*)si.lpMaximumApplicationAddress-(char*)si.lpMinimumApplicationAddress; + /* FIXME: we should track down all the already allocated VM pages and substract them, for now arbitrarily remove 64KB so that it matches NT */ + lpmem->dwAvailVirtual = lpmem->dwTotalVirtual-64*1024; + memcpy(&cached_memstatus,lpmem,sizeof(MEMORYSTATUS)); + + /* it appears some memory display programs want to divide by these values */ + if(lpmem->dwTotalPageFile==0) + lpmem->dwTotalPageFile++; + + if(lpmem->dwAvailPageFile==0) + lpmem->dwAvailPageFile++; + + TRACE("<-- LPMEMORYSTATUS: dwLength %ld, dwMemoryLoad %ld, dwTotalPhys %ld, dwAvailPhys %ld," + " dwTotalPageFile %ld, dwAvailPageFile %ld, dwTotalVirtual %ld, dwAvailVirtual %ld\n", + lpmem->dwLength, lpmem->dwMemoryLoad, lpmem->dwTotalPhys, lpmem->dwAvailPhys, + lpmem->dwTotalPageFile, lpmem->dwAvailPageFile, lpmem->dwTotalVirtual, + lpmem->dwAvailVirtual); +} + +/*********************************************************************** + * GlobalMemoryStatusEx (KERNEL32.@) + * A version of GlobalMemoryStatus that can deal with memory over 4GB + * + * RETURNS + * None + */ +BOOL WINAPI GlobalMemoryStatusEx( LPMEMORYSTATUSEX lpBuffer ) +{ + MEMORYSTATUS memstatus; + + /* Because GlobalMemoryStatusEx is identical to GlobalMemoryStatus save + for one extra field in the struct, and the lack of a bug, we simply + call GlobalMemoryStatus and copy the values across. */ + FIXME("we should emulate the 4GB bug here, as per MSDN\n"); + GlobalMemoryStatus(&memstatus); + lpBuffer->dwMemoryLoad = memstatus.dwMemoryLoad; + lpBuffer->ullTotalPhys = memstatus.dwTotalPhys; + lpBuffer->ullAvailPhys = memstatus.dwAvailPhys; + lpBuffer->ullTotalPageFile = memstatus.dwTotalPageFile; + lpBuffer->ullAvailPageFile = memstatus.dwAvailPageFile; + lpBuffer->ullTotalVirtual = memstatus.dwTotalVirtual; + lpBuffer->ullAvailVirtual = memstatus.dwAvailVirtual; + /* MSDN says about AvailExtendedVirtual: Size of unreserved and uncommitted + memory in the extended portion of the virtual address space of the calling + process, in bytes. + However, I don't know what this means, so set it to zero :( + */ + lpBuffer->ullAvailExtendedVirtual = 0; + return 1; +} diff --git a/memory/selector.c b/dlls/kernel/selector.c similarity index 100% rename from memory/selector.c rename to dlls/kernel/selector.c diff --git a/dlls/ntdll/Makefile.in b/dlls/ntdll/Makefile.in index 6b5ada3563e..f114a19200b 100644 --- a/dlls/ntdll/Makefile.in +++ b/dlls/ntdll/Makefile.in @@ -21,14 +21,12 @@ C_SRCS = \ $(TOPOBJDIR)/memory/environ.c \ $(TOPOBJDIR)/memory/global.c \ $(TOPOBJDIR)/memory/heap.c \ - $(TOPOBJDIR)/memory/selector.c \ $(TOPOBJDIR)/memory/string.c \ $(TOPOBJDIR)/memory/virtual.c \ $(TOPOBJDIR)/misc/cpu.c \ $(TOPOBJDIR)/misc/options.c \ $(TOPOBJDIR)/misc/registry.c \ $(TOPOBJDIR)/misc/version.c \ - $(TOPOBJDIR)/msdos/dosmem.c \ $(TOPOBJDIR)/msdos/dpmi.c \ $(TOPOBJDIR)/relay32/builtin32.c \ $(TOPOBJDIR)/relay32/relay386.c \ diff --git a/memory/global.c b/memory/global.c index 79452901267..7d2a95e15ea 100644 --- a/memory/global.c +++ b/memory/global.c @@ -62,424 +62,14 @@ typedef struct BYTE selCount; /* Number of selectors allocated for this block */ } GLOBALARENA; - /* Flags definitions */ -#define GA_MOVEABLE 0x02 /* same as GMEM_MOVEABLE */ -#define GA_DGROUP 0x04 -#define GA_DISCARDABLE 0x08 -#define GA_IPCSHARE 0x10 /* same as GMEM_DDESHARE */ -#define GA_DOSMEM 0x20 - /* Arena array */ -static GLOBALARENA *pGlobalArena = NULL; -static int globalArenaSize = 0; - -#define GLOBAL_MAX_ALLOC_SIZE 0x00ff0000 /* Largest allocation is 16M - 64K */ +/*static*/ GLOBALARENA *pGlobalArena = NULL; +/*static*/ int globalArenaSize = 0; #define VALID_HANDLE(handle) (((handle)>>__AHSHIFT)> __AHSHIFT)) -/*********************************************************************** - * GLOBAL_GetArena - * - * Return the arena for a given selector, growing the arena array if needed. - */ -static GLOBALARENA *GLOBAL_GetArena( WORD sel, WORD selcount ) -{ - if (((sel >> __AHSHIFT) + selcount) > globalArenaSize) - { - int newsize = ((sel >> __AHSHIFT) + selcount + 0xff) & ~0xff; - GLOBALARENA *pNewArena = realloc( pGlobalArena, - newsize * sizeof(GLOBALARENA) ); - if (!pNewArena) return 0; - pGlobalArena = pNewArena; - memset( pGlobalArena + globalArenaSize, 0, - (newsize - globalArenaSize) * sizeof(GLOBALARENA) ); - globalArenaSize = newsize; - } - return pGlobalArena + (sel >> __AHSHIFT); -} - -void debug_handles(void) -{ - int printed=0; - int i; - for (i = globalArenaSize-1 ; i>=0 ; i--) { - if (pGlobalArena[i].size!=0 && (pGlobalArena[i].handle & 0x8000)){ - printed=1; - DPRINTF("0x%08x, ",pGlobalArena[i].handle); - } - } - if (printed) - DPRINTF("\n"); -} - - -/*********************************************************************** - * GLOBAL_CreateBlock - * - * Create a global heap block for a fixed range of linear memory. - */ -HGLOBAL16 GLOBAL_CreateBlock( WORD flags, const void *ptr, DWORD size, - HGLOBAL16 hOwner, unsigned char selflags ) -{ - WORD sel, selcount; - GLOBALARENA *pArena; - - /* Allocate the selector(s) */ - - sel = SELECTOR_AllocBlock( ptr, size, selflags ); - if (!sel) return 0; - selcount = (size + 0xffff) / 0x10000; - - if (!(pArena = GLOBAL_GetArena( sel, selcount ))) - { - SELECTOR_FreeBlock( sel ); - return 0; - } - - /* Fill the arena block */ - - pArena->base = (DWORD)ptr; - pArena->size = GetSelectorLimit16(sel) + 1; - pArena->handle = (flags & GMEM_MOVEABLE) ? sel - 1 : sel; - pArena->hOwner = hOwner; - pArena->lockCount = 0; - pArena->pageLockCount = 0; - pArena->flags = flags & GA_MOVEABLE; - if (flags & GMEM_DISCARDABLE) pArena->flags |= GA_DISCARDABLE; - if (flags & GMEM_DDESHARE) pArena->flags |= GA_IPCSHARE; - if (!(selflags & (WINE_LDT_FLAGS_CODE^WINE_LDT_FLAGS_DATA))) pArena->flags |= GA_DGROUP; - pArena->selCount = selcount; - if (selcount > 1) /* clear the next arena blocks */ - memset( pArena + 1, 0, (selcount - 1) * sizeof(GLOBALARENA) ); - - return pArena->handle; -} - - -/*********************************************************************** - * GLOBAL_FreeBlock - * - * Free a block allocated by GLOBAL_CreateBlock, without touching - * the associated linear memory range. - */ -BOOL16 GLOBAL_FreeBlock( HGLOBAL16 handle ) -{ - WORD sel; - GLOBALARENA *pArena; - - if (!handle) return TRUE; - sel = GlobalHandleToSel16( handle ); - if (!VALID_HANDLE(sel)) - return FALSE; - pArena = GET_ARENA_PTR(sel); - SELECTOR_FreeBlock( sel ); - memset( pArena, 0, sizeof(GLOBALARENA) ); - return TRUE; -} - -/*********************************************************************** - * GLOBAL_MoveBlock - */ -BOOL16 GLOBAL_MoveBlock( HGLOBAL16 handle, const void *ptr, DWORD size ) -{ - WORD sel; - GLOBALARENA *pArena; - - if (!handle) return TRUE; - sel = GlobalHandleToSel16( handle ); - if (!VALID_HANDLE(sel)) - return FALSE; - pArena = GET_ARENA_PTR(sel); - if (pArena->selCount != 1) - return FALSE; - - pArena->base = (DWORD)ptr; - pArena->size = size; - SELECTOR_ReallocBlock( sel, ptr, size ); - return TRUE; -} - -/*********************************************************************** - * GLOBAL_Alloc - * - * Implementation of GlobalAlloc16() - */ -HGLOBAL16 GLOBAL_Alloc( UINT16 flags, DWORD size, HGLOBAL16 hOwner, unsigned char selflags ) -{ - void *ptr; - HGLOBAL16 handle; - - TRACE("%ld flags=%04x\n", size, flags ); - - /* If size is 0, create a discarded block */ - - if (size == 0) return GLOBAL_CreateBlock( flags, NULL, 1, hOwner, selflags ); - - /* Fixup the size */ - - if (size >= GLOBAL_MAX_ALLOC_SIZE - 0x1f) return 0; - size = (size + 0x1f) & ~0x1f; - - /* Allocate the linear memory */ - ptr = HeapAlloc( GetProcessHeap(), 0, size ); - /* FIXME: free discardable blocks and try again? */ - if (!ptr) return 0; - - /* Allocate the selector(s) */ - - handle = GLOBAL_CreateBlock( flags, ptr, size, hOwner, selflags ); - if (!handle) - { - HeapFree( GetProcessHeap(), 0, ptr ); - return 0; - } - - if (flags & GMEM_ZEROINIT) memset( ptr, 0, size ); - return handle; -} - -/*********************************************************************** - * GlobalAlloc (KERNEL.15) - * GlobalAlloc16 (KERNEL32.24) - * RETURNS - * Handle: Success - * NULL: Failure - */ -HGLOBAL16 WINAPI GlobalAlloc16( - UINT16 flags, /* [in] Object allocation attributes */ - DWORD size /* [in] Number of bytes to allocate */ -) { - HANDLE16 owner = GetCurrentPDB16(); - - if (flags & GMEM_DDESHARE) - owner = GetExePtr(owner); /* Make it a module handle */ - return GLOBAL_Alloc( flags, size, owner, WINE_LDT_FLAGS_DATA ); -} - - -/*********************************************************************** - * GlobalReAlloc (KERNEL.16) - * RETURNS - * Handle: Success - * NULL: Failure - */ -HGLOBAL16 WINAPI GlobalReAlloc16( - HGLOBAL16 handle, /* [in] Handle of global memory object */ - DWORD size, /* [in] New size of block */ - UINT16 flags /* [in] How to reallocate object */ -) { - WORD selcount; - DWORD oldsize; - void *ptr, *newptr; - GLOBALARENA *pArena, *pNewArena; - WORD sel = GlobalHandleToSel16( handle ); - - TRACE("%04x %ld flags=%04x\n", - handle, size, flags ); - if (!handle) return 0; - - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x!\n", handle); - return 0; - } - pArena = GET_ARENA_PTR( handle ); - - /* Discard the block if requested */ - - if ((size == 0) && (flags & GMEM_MOVEABLE) && !(flags & GMEM_MODIFY)) - { - if (!(pArena->flags & GA_MOVEABLE) || - !(pArena->flags & GA_DISCARDABLE) || - (pArena->lockCount > 0) || (pArena->pageLockCount > 0)) return 0; - if (pArena->flags & GA_DOSMEM) - DOSMEM_FreeBlock( (void *)pArena->base ); - else - HeapFree( GetProcessHeap(), 0, (void *)pArena->base ); - pArena->base = 0; - - /* Note: we rely on the fact that SELECTOR_ReallocBlock won't - * change the selector if we are shrinking the block. - * FIXME: shouldn't we keep selectors until the block is deleted? - */ - SELECTOR_ReallocBlock( sel, 0, 1 ); - return handle; - } - - /* Fixup the size */ - - if (size > GLOBAL_MAX_ALLOC_SIZE - 0x20) return 0; - if (size == 0) size = 0x20; - else size = (size + 0x1f) & ~0x1f; - - /* Change the flags */ - - if (flags & GMEM_MODIFY) - { - /* Change the flags, leaving GA_DGROUP alone */ - pArena->flags = (pArena->flags & GA_DGROUP) | (flags & GA_MOVEABLE); - if (flags & GMEM_DISCARDABLE) pArena->flags |= GA_DISCARDABLE; - return handle; - } - - /* Reallocate the linear memory */ - - ptr = (void *)pArena->base; - oldsize = pArena->size; - TRACE("oldbase %p oldsize %08lx newsize %08lx\n", ptr,oldsize,size); - if (ptr && (size == oldsize)) return handle; /* Nothing to do */ - - if (pArena->flags & GA_DOSMEM) - { - if (DOSMEM_ResizeBlock(ptr, size, TRUE) == size) - newptr = ptr; - else if(pArena->pageLockCount > 0) - newptr = 0; - else - { - newptr = DOSMEM_GetBlock( size, 0 ); - if (newptr) - { - memcpy( newptr, ptr, oldsize ); - DOSMEM_FreeBlock( ptr ); - } - } - } - else - { - /* - * if more than one reader (e.g. some pointer has been - * given out by GetVDMPointer32W16), - * only try to realloc in place - */ - newptr = HeapReAlloc( GetProcessHeap(), - (pArena->pageLockCount > 0) ? - HEAP_REALLOC_IN_PLACE_ONLY : 0, - ptr, size ); - } - - if (!newptr) - { - FIXME("Realloc failed lock %d\n",pArena->pageLockCount); - if (pArena->pageLockCount <1) - { - if (pArena->flags & GA_DOSMEM) - DOSMEM_FreeBlock( (void *)pArena->base ); - else - HeapFree( GetProcessHeap(), 0, ptr ); - SELECTOR_FreeBlock( sel ); - memset( pArena, 0, sizeof(GLOBALARENA) ); - } - return 0; - } - ptr = newptr; - - /* Reallocate the selector(s) */ - - sel = SELECTOR_ReallocBlock( sel, ptr, size ); - if (!sel) - { - if (pArena->flags & GA_DOSMEM) - DOSMEM_FreeBlock( (void *)pArena->base ); - else - HeapFree( GetProcessHeap(), 0, ptr ); - memset( pArena, 0, sizeof(GLOBALARENA) ); - return 0; - } - selcount = (size + 0xffff) / 0x10000; - - if (!(pNewArena = GLOBAL_GetArena( sel, selcount ))) - { - if (pArena->flags & GA_DOSMEM) - DOSMEM_FreeBlock( (void *)pArena->base ); - else - HeapFree( GetProcessHeap(), 0, ptr ); - SELECTOR_FreeBlock( sel ); - return 0; - } - - /* Fill the new arena block - As we may have used HEAP_REALLOC_IN_PLACE_ONLY, areas may overlap*/ - - if (pNewArena != pArena) memmove( pNewArena, pArena, sizeof(GLOBALARENA) ); - pNewArena->base = (DWORD)ptr; - pNewArena->size = GetSelectorLimit16(sel) + 1; - pNewArena->selCount = selcount; - pNewArena->handle = (pNewArena->flags & GA_MOVEABLE) ? sel - 1 : sel; - - if (selcount > 1) /* clear the next arena blocks */ - memset( pNewArena + 1, 0, (selcount - 1) * sizeof(GLOBALARENA) ); - - if ((oldsize < size) && (flags & GMEM_ZEROINIT)) - memset( (char *)ptr + oldsize, 0, size - oldsize ); - return pNewArena->handle; -} - - -/*********************************************************************** - * GlobalFree (KERNEL.17) - * GlobalFree16 (KERNEL32.31) - * RETURNS - * NULL: Success - * Handle: Failure - */ -HGLOBAL16 WINAPI GlobalFree16( - HGLOBAL16 handle /* [in] Handle of global memory object */ -) { - void *ptr; - - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to GlobalFree16!\n",handle); - return 0; - } - ptr = (void *)GET_ARENA_PTR(handle)->base; - - TRACE("%04x\n", handle ); - if (!GLOBAL_FreeBlock( handle )) return handle; /* failed */ - if (ptr) HeapFree( GetProcessHeap(), 0, ptr ); - return 0; -} - - -/*********************************************************************** - * GlobalLock (KERNEL.18) - * - * This is the GlobalLock16() function used by 16-bit code. - */ -SEGPTR WINAPI WIN16_GlobalLock16( HGLOBAL16 handle ) -{ - WORD sel = GlobalHandleToSel16( handle ); - TRACE("(%04x) -> %08lx\n", handle, MAKELONG( 0, sel ) ); - - if (handle) - { - if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; - - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to WIN16_GlobalLock16!\n",handle); - sel = 0; - } - else if (!GET_ARENA_PTR(handle)->base) - sel = 0; - else - GET_ARENA_PTR(handle)->lockCount++; - } - - CURRENT_STACK16->ecx = sel; /* selector must be returned in CX as well */ - return MAKESEGPTR( sel, 0 ); -} - - -/********************************************************************** - * K32WOWGlobalLock16 (KERNEL32.60) - */ -SEGPTR WINAPI K32WOWGlobalLock16( HGLOBAL16 hMem ) -{ - return WIN16_GlobalLock16( hMem ); -} - - /*********************************************************************** * GlobalLock16 (KERNEL32.25) * @@ -500,370 +90,6 @@ LPVOID WINAPI GlobalLock16( } -/*********************************************************************** - * GlobalUnlock (KERNEL.19) - * GlobalUnlock16 (KERNEL32.26) - * NOTES - * Should the return values be cast to booleans? - * - * RETURNS - * TRUE: Object is still locked - * FALSE: Object is unlocked - */ -BOOL16 WINAPI GlobalUnlock16( - HGLOBAL16 handle /* [in] Handle of global memory object */ -) { - GLOBALARENA *pArena = GET_ARENA_PTR(handle); - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to GlobalUnlock16!\n",handle); - return 0; - } - TRACE("%04x\n", handle ); - if (pArena->lockCount) pArena->lockCount--; - return pArena->lockCount; -} - -/*********************************************************************** - * GlobalChangeLockCount (KERNEL.365) - * - * This is declared as a register function as it has to preserve - * *all* registers, even AX/DX ! - * - */ -void WINAPI GlobalChangeLockCount16( HGLOBAL16 handle, INT16 delta, - CONTEXT86 *context ) -{ - if ( delta == 1 ) - GlobalLock16( handle ); - else if ( delta == -1 ) - GlobalUnlock16( handle ); - else - ERR("(%04X, %d): strange delta value\n", handle, delta ); -} - -/*********************************************************************** - * GlobalSize (KERNEL.20) - * GlobalSize16 (KERNEL32.32) - * RETURNS - * Size in bytes of object - * 0: Failure - */ -DWORD WINAPI GlobalSize16( - HGLOBAL16 handle /* [in] Handle of global memory object */ -) { - TRACE("%04x\n", handle ); - if (!handle) return 0; - if (!VALID_HANDLE(handle)) - return 0; - return GET_ARENA_PTR(handle)->size; -} - - -/*********************************************************************** - * GlobalHandle (KERNEL.21) - * NOTES - * Why is GlobalHandleToSel used here with the sel as input? - * - * RETURNS - * Handle: Success - * NULL: Failure - */ -DWORD WINAPI GlobalHandle16( - WORD sel /* [in] Address of global memory block */ -) { - TRACE("%04x\n", sel ); - if (!VALID_HANDLE(sel)) { - WARN("Invalid handle 0x%04x passed to GlobalHandle16!\n",sel); - return 0; - } - return MAKELONG( GET_ARENA_PTR(sel)->handle, GlobalHandleToSel16(sel) ); -} - -/*********************************************************************** - * GlobalHandleNoRIP (KERNEL.159) - */ -DWORD WINAPI GlobalHandleNoRIP16( WORD sel ) -{ - int i; - for (i = globalArenaSize-1 ; i>=0 ; i--) { - if (pGlobalArena[i].size!=0 && pGlobalArena[i].handle == sel) - return MAKELONG( GET_ARENA_PTR(sel)->handle, GlobalHandleToSel16(sel) ); - } - return 0; -} - - -/*********************************************************************** - * GlobalFlags (KERNEL.22) - * - * NOTES - * Should this return GMEM_INVALID_HANDLE instead of 0 on invalid - * handle? - * - * RETURNS - * Value specifying flags and lock count - * GMEM_INVALID_HANDLE: Invalid handle - */ -UINT16 WINAPI GlobalFlags16( - HGLOBAL16 handle /* [in] Handle of global memory object */ -) { - GLOBALARENA *pArena; - - TRACE("%04x\n", handle ); - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to GlobalFlags16!\n",handle); - return 0; - } - pArena = GET_ARENA_PTR(handle); - return pArena->lockCount | - ((pArena->flags & GA_DISCARDABLE) ? GMEM_DISCARDABLE : 0) | - ((pArena->base == 0) ? GMEM_DISCARDED : 0); -} - - -/*********************************************************************** - * LockSegment (KERNEL.23) - */ -HGLOBAL16 WINAPI LockSegment16( HGLOBAL16 handle ) -{ - TRACE("%04x\n", handle ); - if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to LockSegment16!\n",handle); - return 0; - } - GET_ARENA_PTR(handle)->lockCount++; - return handle; -} - - -/*********************************************************************** - * UnlockSegment (KERNEL.24) - */ -void WINAPI UnlockSegment16( HGLOBAL16 handle ) -{ - TRACE("%04x\n", handle ); - if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to UnlockSegment16!\n",handle); - return; - } - GET_ARENA_PTR(handle)->lockCount--; - /* FIXME: this ought to return the lock count in CX (go figure...) */ -} - - -/*********************************************************************** - * GlobalCompact (KERNEL.25) - */ -DWORD WINAPI GlobalCompact16( DWORD desired ) -{ - return GLOBAL_MAX_ALLOC_SIZE; -} - - -/*********************************************************************** - * GlobalFreeAll (KERNEL.26) - */ -void WINAPI GlobalFreeAll16( HGLOBAL16 owner ) -{ - DWORD i; - GLOBALARENA *pArena; - - pArena = pGlobalArena; - for (i = 0; i < globalArenaSize; i++, pArena++) - { - if ((pArena->size != 0) && (pArena->hOwner == owner)) - GlobalFree16( pArena->handle ); - } -} - - -/*********************************************************************** - * GlobalWire (KERNEL.111) - * GlobalWire16 (KERNEL32.29) - */ -SEGPTR WINAPI GlobalWire16( HGLOBAL16 handle ) -{ - return WIN16_GlobalLock16( handle ); -} - - -/*********************************************************************** - * GlobalUnWire (KERNEL.112) - * GlobalUnWire16 (KERNEL32.30) - */ -BOOL16 WINAPI GlobalUnWire16( HGLOBAL16 handle ) -{ - return !GlobalUnlock16( handle ); -} - - -/*********************************************************************** - * SetSwapAreaSize (KERNEL.106) - */ -LONG WINAPI SetSwapAreaSize16( WORD size ) -{ - FIXME("(%d) - stub!\n", size ); - return MAKELONG( size, 0xffff ); -} - - -/*********************************************************************** - * GlobalLRUOldest (KERNEL.163) - */ -HGLOBAL16 WINAPI GlobalLRUOldest16( HGLOBAL16 handle ) -{ - TRACE("%04x\n", handle ); - if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; - return handle; -} - - -/*********************************************************************** - * GlobalLRUNewest (KERNEL.164) - */ -HGLOBAL16 WINAPI GlobalLRUNewest16( HGLOBAL16 handle ) -{ - TRACE("%04x\n", handle ); - if (handle == (HGLOBAL16)-1) handle = CURRENT_DS; - return handle; -} - - -/*********************************************************************** - * GetFreeSpace (KERNEL.169) - */ -DWORD WINAPI GetFreeSpace16( UINT16 wFlags ) -{ - MEMORYSTATUS ms; - GlobalMemoryStatus( &ms ); - return ms.dwAvailVirtual; -} - -/*********************************************************************** - * GlobalDOSAlloc (KERNEL.184) - * RETURNS - * Address (HW=Paragraph segment; LW=Selector) - */ -DWORD WINAPI GlobalDOSAlloc16( - DWORD size /* [in] Number of bytes to be allocated */ -) { - UINT16 uParagraph; - LPVOID lpBlock = DOSMEM_GetBlock( size, &uParagraph ); - - if( lpBlock ) - { - HMODULE16 hModule = GetModuleHandle16("KERNEL"); - WORD wSelector; - GLOBALARENA *pArena; - - wSelector = GLOBAL_CreateBlock(GMEM_FIXED, lpBlock, size, hModule, WINE_LDT_FLAGS_DATA ); - pArena = GET_ARENA_PTR(wSelector); - pArena->flags |= GA_DOSMEM; - return MAKELONG(wSelector,uParagraph); - } - return 0; -} - - -/*********************************************************************** - * GlobalDOSFree (KERNEL.185) - * RETURNS - * NULL: Success - * sel: Failure - */ -WORD WINAPI GlobalDOSFree16( - WORD sel /* [in] Selector */ -) { - DWORD block = GetSelectorBase(sel); - - if( block && block < 0x100000 ) - { - LPVOID lpBlock = DOSMEM_MapDosToLinear( block ); - if( DOSMEM_FreeBlock( lpBlock ) ) - GLOBAL_FreeBlock( sel ); - sel = 0; - } - return sel; -} - - -/*********************************************************************** - * GlobalPageLock (KERNEL.191) - * GlobalSmartPageLock(KERNEL.230) - */ -WORD WINAPI GlobalPageLock16( HGLOBAL16 handle ) -{ - TRACE("%04x\n", handle ); - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to GlobalPageLock!\n",handle); - return 0; - } - return ++(GET_ARENA_PTR(handle)->pageLockCount); -} - - -/*********************************************************************** - * GlobalPageUnlock (KERNEL.192) - * GlobalSmartPageUnlock(KERNEL.231) - */ -WORD WINAPI GlobalPageUnlock16( HGLOBAL16 handle ) -{ - TRACE("%04x\n", handle ); - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to GlobalPageUnlock!\n",handle); - return 0; - } - return --(GET_ARENA_PTR(handle)->pageLockCount); -} - - -/*********************************************************************** - * GlobalFix (KERNEL.197) - * GlobalFix16 (KERNEL32.27) - */ -WORD WINAPI GlobalFix16( HGLOBAL16 handle ) -{ - TRACE("%04x\n", handle ); - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to GlobalFix16!\n",handle); - return 0; - } - GET_ARENA_PTR(handle)->lockCount++; - - return GlobalHandleToSel16(handle); -} - - -/*********************************************************************** - * GlobalUnfix (KERNEL.198) - * GlobalUnfix16 (KERNEL32.28) - */ -void WINAPI GlobalUnfix16( HGLOBAL16 handle ) -{ - TRACE("%04x\n", handle ); - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to GlobalUnfix16!\n",handle); - return; - } - GET_ARENA_PTR(handle)->lockCount--; -} - - -/*********************************************************************** - * FarSetOwner (KERNEL.403) - */ -void WINAPI FarSetOwner16( HGLOBAL16 handle, HANDLE16 hOwner ) -{ - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to FarSetOwner!\n",handle); - return; - } - GET_ARENA_PTR(handle)->hOwner = hOwner; -} - - /*********************************************************************** * FarGetOwner (KERNEL.404) */ @@ -875,344 +101,3 @@ HANDLE16 WINAPI FarGetOwner16( HGLOBAL16 handle ) } return GET_ARENA_PTR(handle)->hOwner; } - - -/*********************************************************************** - * GlobalHandleToSel (TOOLHELP.50) - */ -WORD WINAPI GlobalHandleToSel16( HGLOBAL16 handle ) -{ - if (!handle) return 0; - if (!VALID_HANDLE(handle)) { - WARN("Invalid handle 0x%04x passed to GlobalHandleToSel!\n",handle); - return 0; - } - if (!(handle & 7)) - { - WARN("Program attempted invalid selector conversion\n" ); - return handle - 1; - } - return handle | 7; -} - - -/*********************************************************************** - * GlobalFirst (TOOLHELP.51) - */ -BOOL16 WINAPI GlobalFirst16( GLOBALENTRY *pGlobal, WORD wFlags ) -{ - if (wFlags == GLOBAL_LRU) return FALSE; - pGlobal->dwNext = 0; - return GlobalNext16( pGlobal, wFlags ); -} - - -/*********************************************************************** - * GlobalNext (TOOLHELP.52) - */ -BOOL16 WINAPI GlobalNext16( GLOBALENTRY *pGlobal, WORD wFlags) -{ - GLOBALARENA *pArena; - - if (pGlobal->dwNext >= globalArenaSize) return FALSE; - pArena = pGlobalArena + pGlobal->dwNext; - if (wFlags == GLOBAL_FREE) /* only free blocks */ - { - int i; - for (i = pGlobal->dwNext; i < globalArenaSize; i++, pArena++) - if (pArena->size == 0) break; /* block is free */ - if (i >= globalArenaSize) return FALSE; - pGlobal->dwNext = i; - } - - pGlobal->dwAddress = pArena->base; - pGlobal->dwBlockSize = pArena->size; - pGlobal->hBlock = pArena->handle; - pGlobal->wcLock = pArena->lockCount; - pGlobal->wcPageLock = pArena->pageLockCount; - pGlobal->wFlags = (GetCurrentPDB16() == pArena->hOwner); - pGlobal->wHeapPresent = FALSE; - pGlobal->hOwner = pArena->hOwner; - pGlobal->wType = GT_UNKNOWN; - pGlobal->wData = 0; - pGlobal->dwNext++; - return TRUE; -} - - -/*********************************************************************** - * GlobalInfo (TOOLHELP.53) - */ -BOOL16 WINAPI GlobalInfo16( GLOBALINFO *pInfo ) -{ - int i; - GLOBALARENA *pArena; - - pInfo->wcItems = globalArenaSize; - pInfo->wcItemsFree = 0; - pInfo->wcItemsLRU = 0; - for (i = 0, pArena = pGlobalArena; i < globalArenaSize; i++, pArena++) - if (pArena->size == 0) pInfo->wcItemsFree++; - return TRUE; -} - - -/*********************************************************************** - * GlobalEntryHandle (TOOLHELP.54) - */ -BOOL16 WINAPI GlobalEntryHandle16( GLOBALENTRY *pGlobal, HGLOBAL16 hItem ) -{ - GLOBALARENA *pArena = GET_ARENA_PTR(hItem); - - pGlobal->dwAddress = pArena->base; - pGlobal->dwBlockSize = pArena->size; - pGlobal->hBlock = pArena->handle; - pGlobal->wcLock = pArena->lockCount; - pGlobal->wcPageLock = pArena->pageLockCount; - pGlobal->wFlags = (GetCurrentPDB16() == pArena->hOwner); - pGlobal->wHeapPresent = FALSE; - pGlobal->hOwner = pArena->hOwner; - pGlobal->wType = GT_UNKNOWN; - pGlobal->wData = 0; - pGlobal->dwNext++; - return TRUE; -} - - -/*********************************************************************** - * GlobalEntryModule (TOOLHELP.55) - */ -BOOL16 WINAPI GlobalEntryModule16( GLOBALENTRY *pGlobal, HMODULE16 hModule, - WORD wSeg ) -{ - FIXME("(%p, 0x%04x, 0x%04x), stub.\n", pGlobal, hModule, wSeg); - return FALSE; -} - - -/*********************************************************************** - * MemManInfo (TOOLHELP.72) - */ -BOOL16 WINAPI MemManInfo16( MEMMANINFO *info ) -{ - MEMORYSTATUS status; - - /* - * Not unsurprisingly although the documention says you - * _must_ provide the size in the dwSize field, this function - * (under Windows) always fills the structure and returns true. - */ - GlobalMemoryStatus( &status ); - info->wPageSize = getpagesize(); - info->dwLargestFreeBlock = status.dwAvailVirtual; - info->dwMaxPagesAvailable = info->dwLargestFreeBlock / info->wPageSize; - info->dwMaxPagesLockable = info->dwMaxPagesAvailable; - info->dwTotalLinearSpace = status.dwTotalVirtual / info->wPageSize; - info->dwTotalUnlockedPages = info->dwTotalLinearSpace; - info->dwFreePages = info->dwMaxPagesAvailable; - info->dwTotalPages = info->dwTotalLinearSpace; - info->dwFreeLinearSpace = info->dwMaxPagesAvailable; - info->dwSwapFilePages = status.dwTotalPageFile / info->wPageSize; - return TRUE; -} - -/*********************************************************************** - * GetFreeMemInfo (KERNEL.316) - */ -DWORD WINAPI GetFreeMemInfo16(void) -{ - MEMMANINFO info; - MemManInfo16( &info ); - return MAKELONG( info.dwTotalLinearSpace, info.dwMaxPagesAvailable ); -} - -/*********************************************************************** - * GlobalMemoryStatus (KERNEL32.@) - * Provides information about the status of the memory, so apps can tell - * roughly how much they are able to allocate - * - * RETURNS - * None - */ -VOID WINAPI GlobalMemoryStatus( - LPMEMORYSTATUS lpmem -) { - static MEMORYSTATUS cached_memstatus; - static int cache_lastchecked = 0; - SYSTEM_INFO si; -#ifdef linux - FILE *f; -#endif -#if defined(__FreeBSD__) || defined(__NetBSD__) - int *tmp; - int size_sys; - int mib[2] = { CTL_HW }; -#endif - if (time(NULL)==cache_lastchecked) { - memcpy(lpmem,&cached_memstatus,sizeof(MEMORYSTATUS)); - return; - } - cache_lastchecked = time(NULL); - - lpmem->dwMemoryLoad = 0; - lpmem->dwTotalPhys = 16*1024*1024; - lpmem->dwAvailPhys = 16*1024*1024; - lpmem->dwTotalPageFile = 16*1024*1024; - lpmem->dwAvailPageFile = 16*1024*1024; - -#ifdef linux - f = fopen( "/proc/meminfo", "r" ); - if (f) - { - char buffer[256]; - int total, used, free, shared, buffers, cached; - - lpmem->dwLength = sizeof(MEMORYSTATUS); - lpmem->dwTotalPhys = lpmem->dwAvailPhys = 0; - lpmem->dwTotalPageFile = lpmem->dwAvailPageFile = 0; - while (fgets( buffer, sizeof(buffer), f )) - { - /* old style /proc/meminfo ... */ - if (sscanf( buffer, "Mem: %d %d %d %d %d %d", &total, &used, &free, &shared, &buffers, &cached )) - { - lpmem->dwTotalPhys += total; - lpmem->dwAvailPhys += free + buffers + cached; - } - if (sscanf( buffer, "Swap: %d %d %d", &total, &used, &free )) - { - lpmem->dwTotalPageFile += total; - lpmem->dwAvailPageFile += free; - } - - /* new style /proc/meminfo ... */ - if (sscanf(buffer, "MemTotal: %d", &total)) - lpmem->dwTotalPhys = total*1024; - if (sscanf(buffer, "MemFree: %d", &free)) - lpmem->dwAvailPhys = free*1024; - if (sscanf(buffer, "SwapTotal: %d", &total)) - lpmem->dwTotalPageFile = total*1024; - if (sscanf(buffer, "SwapFree: %d", &free)) - lpmem->dwAvailPageFile = free*1024; - if (sscanf(buffer, "Buffers: %d", &buffers)) - lpmem->dwAvailPhys += buffers*1024; - if (sscanf(buffer, "Cached: %d", &cached)) - lpmem->dwAvailPhys += cached*1024; - } - fclose( f ); - - if (lpmem->dwTotalPhys) - { - DWORD TotalPhysical = lpmem->dwTotalPhys+lpmem->dwTotalPageFile; - DWORD AvailPhysical = lpmem->dwAvailPhys+lpmem->dwAvailPageFile; - lpmem->dwMemoryLoad = (TotalPhysical-AvailPhysical) - / (TotalPhysical / 100); - } - } -#elif defined(__FreeBSD__) || defined(__NetBSD__) - mib[1] = HW_PHYSMEM; - sysctl(mib, 2, NULL, &size_sys, NULL, 0); - tmp = malloc(size_sys * sizeof(int)); - sysctl(mib, 2, tmp, &size_sys, NULL, 0); - if (tmp && *tmp) - { - lpmem->dwTotalPhys = *tmp; - free(tmp); - mib[1] = HW_USERMEM; - sysctl(mib, 2, NULL, &size_sys, NULL, 0); - tmp = malloc(size_sys * sizeof(int)); - sysctl(mib, 2, tmp, &size_sys, NULL, 0); - if (tmp && *tmp) - { - lpmem->dwAvailPhys = *tmp; - lpmem->dwTotalPageFile = *tmp; - lpmem->dwAvailPageFile = *tmp; - lpmem->dwMemoryLoad = lpmem->dwTotalPhys - lpmem->dwAvailPhys; - } else - { - lpmem->dwAvailPhys = lpmem->dwTotalPhys; - lpmem->dwTotalPageFile = lpmem->dwTotalPhys; - lpmem->dwAvailPageFile = lpmem->dwTotalPhys; - lpmem->dwMemoryLoad = 0; - } - free(tmp); - - } -#endif - /* Some applications (e.g. QuickTime 6) crash if we tell them there - * is more than 2GB of physical memory. - */ - if (lpmem->dwTotalPhys>2U*1024*1024*1024) - { - lpmem->dwTotalPhys=2U*1024*1024*1024; - lpmem->dwAvailPhys=2U*1024*1024*1024; - } - - /* FIXME: should do something for other systems */ - GetSystemInfo(&si); - lpmem->dwTotalVirtual = (char*)si.lpMaximumApplicationAddress-(char*)si.lpMinimumApplicationAddress; - /* FIXME: we should track down all the already allocated VM pages and substract them, for now arbitrarily remove 64KB so that it matches NT */ - lpmem->dwAvailVirtual = lpmem->dwTotalVirtual-64*1024; - memcpy(&cached_memstatus,lpmem,sizeof(MEMORYSTATUS)); - - /* it appears some memory display programs want to divide by these values */ - if(lpmem->dwTotalPageFile==0) - lpmem->dwTotalPageFile++; - - if(lpmem->dwAvailPageFile==0) - lpmem->dwAvailPageFile++; - - TRACE("<-- LPMEMORYSTATUS: dwLength %ld, dwMemoryLoad %ld, dwTotalPhys %ld, dwAvailPhys %ld," - " dwTotalPageFile %ld, dwAvailPageFile %ld, dwTotalVirtual %ld, dwAvailVirtual %ld\n", - lpmem->dwLength, lpmem->dwMemoryLoad, lpmem->dwTotalPhys, lpmem->dwAvailPhys, - lpmem->dwTotalPageFile, lpmem->dwAvailPageFile, lpmem->dwTotalVirtual, - lpmem->dwAvailVirtual); -} - -/*********************************************************************** - * GlobalMemoryStatusEx (KERNEL32.@) - * A version of GlobalMemoryStatus that can deal with memory over 4GB - * - * RETURNS - * None - */ -BOOL WINAPI GlobalMemoryStatusEx( LPMEMORYSTATUSEX lpBuffer ) { - MEMORYSTATUS memstatus; - - /* Because GlobalMemoryStatusEx is identical to GlobalMemoryStatus save - for one extra field in the struct, and the lack of a bug, we simply - call GlobalMemoryStatus and copy the values across. */ - FIXME("we should emulate the 4GB bug here, as per MSDN\n"); - GlobalMemoryStatus(&memstatus); - lpBuffer->dwMemoryLoad = memstatus.dwMemoryLoad; - lpBuffer->ullTotalPhys = memstatus.dwTotalPhys; - lpBuffer->ullAvailPhys = memstatus.dwAvailPhys; - lpBuffer->ullTotalPageFile = memstatus.dwTotalPageFile; - lpBuffer->ullAvailPageFile = memstatus.dwAvailPageFile; - lpBuffer->ullTotalVirtual = memstatus.dwTotalVirtual; - lpBuffer->ullAvailVirtual = memstatus.dwAvailVirtual; - /* MSDN says about AvailExtendedVirtual: Size of unreserved and uncommitted - memory in the extended portion of the virtual address space of the calling - process, in bytes. - - However, I don't know what this means, so set it to zero :( - */ - lpBuffer->ullAvailExtendedVirtual = 0; - return 1; -} - -/*********************************************************************** - * A20Proc (KERNEL.165) - * A20_Proc (SYSTEM.20) - */ -void WINAPI A20Proc16( WORD unused ) -{ - /* this is also a NOP in Windows */ -} - -/*********************************************************************** - * LimitEMSPages (KERNEL.156) - */ -DWORD WINAPI LimitEMSPages16( DWORD unused ) -{ - return 0; -}