/* * Resources * * Copyright 1993 Robert J. Amstadt * 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 */ #include "config.h" #include #include #include #include #include #include #ifdef HAVE_UNISTD_H # include #endif #include "windef.h" #include "winbase.h" #include "wine/winbase16.h" #include "wine/exception.h" #include "heap.h" #include "cursoricon.h" #include "module.h" #include "file.h" #include "wine/debug.h" #include "winerror.h" #include "winnls.h" #include "msvcrt/excpt.h" WINE_DEFAULT_DEBUG_CHANNEL(resource); #define HRSRC_MAP_BLOCKSIZE 16 typedef struct _HRSRC_ELEM { HANDLE hRsrc; WORD type; } HRSRC_ELEM; typedef struct _HRSRC_MAP { int nAlloc; int nUsed; HRSRC_ELEM *elem; } HRSRC_MAP; /********************************************************************** * MapHRsrc32To16 */ static HRSRC MapHRsrc32To16( NE_MODULE *pModule, HANDLE hRsrc32, WORD type ) { HRSRC_MAP *map = (HRSRC_MAP *)pModule->hRsrcMap; HRSRC_ELEM *newElem; int i; /* On first call, initialize HRSRC map */ if ( !map ) { if ( !(map = (HRSRC_MAP *)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(HRSRC_MAP) ) ) ) { ERR("Cannot allocate HRSRC map\n" ); return 0; } pModule->hRsrcMap = (LPVOID)map; } /* Check whether HRSRC32 already in map */ for ( i = 0; i < map->nUsed; i++ ) if ( map->elem[i].hRsrc == hRsrc32 ) return (HRSRC)(i + 1); /* If no space left, grow table */ if ( map->nUsed == map->nAlloc ) { if ( !(newElem = (HRSRC_ELEM *)HeapReAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, map->elem, (map->nAlloc + HRSRC_MAP_BLOCKSIZE) * sizeof(HRSRC_ELEM) ) )) { ERR("Cannot grow HRSRC map\n" ); return 0; } map->elem = newElem; map->nAlloc += HRSRC_MAP_BLOCKSIZE; } /* Add HRSRC32 to table */ map->elem[map->nUsed].hRsrc = hRsrc32; map->elem[map->nUsed].type = type; map->nUsed++; return (HRSRC)map->nUsed; } /********************************************************************** * MapHRsrc16To32 */ static HRSRC MapHRsrc16To32( NE_MODULE *pModule, HRSRC hRsrc16 ) { HRSRC_MAP *map = (HRSRC_MAP *)pModule->hRsrcMap; if ( !map || !hRsrc16 || (int)hRsrc16 > map->nUsed ) return 0; return map->elem[(int)hRsrc16-1].hRsrc; } /********************************************************************** * MapHRsrc16ToType */ static WORD MapHRsrc16ToType( NE_MODULE *pModule, HRSRC hRsrc16 ) { HRSRC_MAP *map = (HRSRC_MAP *)pModule->hRsrcMap; if ( !map || !hRsrc16 || (int)hRsrc16 > map->nUsed ) return 0; return map->elem[(int)hRsrc16-1].type; } /* filter for page-fault exceptions */ static WINE_EXCEPTION_FILTER(page_fault) { if (GetExceptionCode() == EXCEPTION_ACCESS_VIOLATION) return EXCEPTION_EXECUTE_HANDLER; return EXCEPTION_CONTINUE_SEARCH; } static HRSRC RES_FindResource2( HMODULE hModule, LPCSTR type, LPCSTR name, WORD lang, BOOL bUnicode, BOOL bRet16 ) { HRSRC hRsrc = 0; TRACE("(%08x, %08x%s, %08x%s, %04x, %s, %s)\n", hModule, (UINT)type, HIWORD(type)? (bUnicode? debugstr_w((LPWSTR)type) : debugstr_a(type)) : "", (UINT)name, HIWORD(name)? (bUnicode? debugstr_w((LPWSTR)name) : debugstr_a(name)) : "", lang, bUnicode? "W" : "A", bRet16? "NE" : "PE" ); if (!HIWORD(hModule)) { HMODULE16 hMod16 = MapHModuleLS( hModule ); NE_MODULE *pModule = NE_GetPtr( hMod16 ); if (!pModule) return 0; if (!pModule->module32) { /* 16-bit NE module */ LPSTR typeStr, nameStr; if ( HIWORD( type ) && bUnicode ) typeStr = HEAP_strdupWtoA( GetProcessHeap(), 0, (LPCWSTR)type ); else typeStr = (LPSTR)type; if ( HIWORD( name ) && bUnicode ) nameStr = HEAP_strdupWtoA( GetProcessHeap(), 0, (LPCWSTR)name ); else nameStr = (LPSTR)name; hRsrc = NE_FindResource( pModule, nameStr, typeStr ); if ( HIWORD( type ) && bUnicode ) HeapFree( GetProcessHeap(), 0, typeStr ); if ( HIWORD( name ) && bUnicode ) HeapFree( GetProcessHeap(), 0, nameStr ); /* If we need to return 32-bit HRSRC, no conversion is necessary, we simply use the 16-bit HRSRC as 32-bit HRSRC */ } else { /* 32-bit PE module */ hRsrc = RES_FindResource2( pModule->module32, type, name, lang, bUnicode, FALSE ); /* If we need to return 16-bit HRSRC, perform conversion */ if ( bRet16 ) hRsrc = MapHRsrc32To16( pModule, hRsrc, HIWORD( type )? 0 : LOWORD( type ) ); } } else { /* 32-bit PE module */ LPWSTR typeStr, nameStr; if ( HIWORD( type ) && !bUnicode ) typeStr = HEAP_strdupAtoW( GetProcessHeap(), 0, type ); else typeStr = (LPWSTR)type; if ( HIWORD( name ) && !bUnicode ) nameStr = HEAP_strdupAtoW( GetProcessHeap(), 0, name ); else nameStr = (LPWSTR)name; /* Here is the real difference between FindResouce and FindResourceEx */ if(lang == MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL) || lang == MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT) || lang == MAKELANGID(LANG_NEUTRAL, SUBLANG_SYS_DEFAULT) || lang == MAKELANGID(LANG_NEUTRAL, 3)) /* FIXME: real name? */ hRsrc = PE_FindResourceW( hModule, nameStr, typeStr ); else hRsrc = PE_FindResourceExW( hModule, nameStr, typeStr, lang ); if ( HIWORD( type ) && !bUnicode ) HeapFree( GetProcessHeap(), 0, typeStr ); if ( HIWORD( name ) && !bUnicode ) HeapFree( GetProcessHeap(), 0, nameStr ); } return hRsrc; } /********************************************************************** * RES_FindResource */ static HRSRC RES_FindResource( HMODULE hModule, LPCSTR type, LPCSTR name, WORD lang, BOOL bUnicode, BOOL bRet16 ) { HRSRC hRsrc; __TRY { hRsrc = RES_FindResource2(hModule, type, name, lang, bUnicode, bRet16); } __EXCEPT(page_fault) { WARN("page fault\n"); SetLastError(ERROR_INVALID_PARAMETER); return 0; } __ENDTRY return hRsrc; } /********************************************************************** * RES_SizeofResource */ static DWORD RES_SizeofResource( HMODULE hModule, HRSRC hRsrc, BOOL bRet16 ) { if (!hRsrc) return 0; TRACE("(%08x, %08x, %s)\n", hModule, hRsrc, bRet16? "NE" : "PE" ); if (!HIWORD(hModule)) { HMODULE16 hMod16 = MapHModuleLS( hModule ); NE_MODULE *pModule = NE_GetPtr( hMod16 ); if (!pModule) return 0; if (!pModule->module32) /* 16-bit NE module */ { /* If we got a 32-bit hRsrc, we don't need to convert it */ return NE_SizeofResource( pModule, hRsrc ); } /* If we got a 16-bit hRsrc, convert it */ if (!HIWORD(hRsrc)) hRsrc = MapHRsrc16To32( pModule, hRsrc ); } /* 32-bit PE module */ return PE_SizeofResource( hRsrc ); } /********************************************************************** * RES_LoadResource */ static HGLOBAL RES_LoadResource( HMODULE hModule, HRSRC hRsrc, BOOL bRet16 ) { HGLOBAL hMem = 0; TRACE("(%08x, %08x, %s)\n", hModule, hRsrc, bRet16? "NE" : "PE" ); if (!hRsrc) return 0; if (!HIWORD(hModule)) { HMODULE16 hMod16 = MapHModuleLS( hModule ); NE_MODULE *pModule = NE_GetPtr( hMod16 ); if (!pModule) return 0; if (!pModule->module32) { /* 16-bit NE module */ /* If we got a 32-bit hRsrc, we don't need to convert it */ hMem = NE_LoadResource( pModule, LOWORD(hRsrc) ); /* If we are to return a 32-bit resource, we should probably convert it but we don't for now. FIXME !!! */ return hMem; } else { /* If we got a 16-bit hRsrc, convert it */ HRSRC hRsrc32 = HIWORD(hRsrc)? hRsrc : MapHRsrc16To32( pModule, hRsrc ); hMem = PE_LoadResource( pModule->module32, hRsrc32 ); /* If we need to return a 16-bit resource, convert it */ if ( bRet16 ) { WORD type = MapHRsrc16ToType( pModule, hRsrc ); DWORD size = SizeofResource( hModule, hRsrc ); LPVOID bits = LockResource( hMem ); hMem = NE_LoadPEResource( pModule, type, bits, size ); } } } else { /* 32-bit PE module */ hMem = PE_LoadResource( hModule, hRsrc ); } return hMem; } /********************************************************************** * FindResource (KERNEL.60) * FindResource16 (KERNEL32.@) */ HRSRC16 WINAPI FindResource16( HMODULE16 hModule, LPCSTR name, LPCSTR type ) { return LOWORD( RES_FindResource( hModule, type, name, MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL), FALSE, TRUE ) ); } /********************************************************************** * FindResourceA (KERNEL32.@) */ HRSRC WINAPI FindResourceA( HMODULE hModule, LPCSTR name, LPCSTR type ) { return RES_FindResource( hModule, type, name, MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL), FALSE, FALSE ); } /********************************************************************** * FindResourceExA (KERNEL32.@) */ HRSRC WINAPI FindResourceExA( HMODULE hModule, LPCSTR type, LPCSTR name, WORD lang ) { return RES_FindResource( hModule, type, name, lang, FALSE, FALSE ); } /********************************************************************** * FindResourceExW (KERNEL32.@) */ HRSRC WINAPI FindResourceExW( HMODULE hModule, LPCWSTR type, LPCWSTR name, WORD lang ) { return RES_FindResource( hModule, (LPCSTR)type, (LPCSTR)name, lang, TRUE, FALSE ); } /********************************************************************** * FindResourceW (KERNEL32.@) */ HRSRC WINAPI FindResourceW(HINSTANCE hModule, LPCWSTR name, LPCWSTR type) { return RES_FindResource( hModule, (LPCSTR)type, (LPCSTR)name, MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL), TRUE, FALSE ); } /********************************************************************** * LoadResource (KERNEL.61) * LoadResource16 (KERNEL32.@) */ HGLOBAL16 WINAPI LoadResource16( HMODULE16 hModule, HRSRC16 hRsrc ) { return RES_LoadResource( hModule, hRsrc, TRUE ); } /********************************************************************** * LoadResource (KERNEL32.@) */ HGLOBAL WINAPI LoadResource( HINSTANCE hModule, HRSRC hRsrc ) { return RES_LoadResource( hModule, hRsrc, FALSE ); } /********************************************************************** * LockResource (KERNEL.62) */ SEGPTR WINAPI WIN16_LockResource16( HGLOBAL16 handle ) { TRACE("(%04x)\n", handle ); /* May need to reload the resource if discarded */ return K32WOWGlobalLock16( handle ); } /********************************************************************** * LockResource16 (KERNEL32.@) */ LPVOID WINAPI LockResource16( HGLOBAL16 handle ) { return MapSL( WIN16_LockResource16(handle) ); } /********************************************************************** * LockResource (KERNEL32.@) */ LPVOID WINAPI LockResource( HGLOBAL handle ) { TRACE("(%08x)\n", handle ); if (HIWORD( handle )) /* 32-bit memory handle */ return (LPVOID)handle; /* 16-bit memory handle */ return LockResource16( LOWORD(handle) ); } typedef WORD (WINAPI *pDestroyIcon32Proc)( HGLOBAL16 handle, UINT16 flags ); /********************************************************************** * FreeResource (KERNEL.63) * FreeResource16 (KERNEL32.@) */ BOOL16 WINAPI FreeResource16( HGLOBAL16 handle ) { HGLOBAL16 retv = handle; NE_MODULE *pModule = NE_GetPtr( FarGetOwner16( handle ) ); TRACE("(%04x)\n", handle ); /* Try NE resource first */ retv = NE_FreeResource( pModule, handle ); /* If this failed, call USER.DestroyIcon32; this will check whether it is a shared cursor/icon; if not it will call GlobalFree16() */ if ( retv ) { pDestroyIcon32Proc proc; HMODULE user = GetModuleHandleA( "user32.dll" ); if (user && (proc = (pDestroyIcon32Proc)GetProcAddress( user, "DestroyIcon32" ))) retv = proc( handle, CID_RESOURCE ); else retv = GlobalFree16( handle ); } return (BOOL)retv; } /********************************************************************** * FreeResource (KERNEL32.@) */ BOOL WINAPI FreeResource( HGLOBAL handle ) { if (HIWORD(handle)) return 0; /* 32-bit memory handle: nothing to do */ return FreeResource16( LOWORD(handle) ); } /********************************************************************** * SizeofResource (KERNEL.65) * SizeofResource16 (KERNEL32.@) */ DWORD WINAPI SizeofResource16( HMODULE16 hModule, HRSRC16 hRsrc ) { return RES_SizeofResource( hModule, hRsrc, TRUE ); } /********************************************************************** * SizeofResource (KERNEL32.@) */ DWORD WINAPI SizeofResource( HINSTANCE hModule, HRSRC hRsrc ) { return RES_SizeofResource( hModule, hRsrc, FALSE ); }