/* * Copyright 2002 Dmitry Timoshkov for Codeweavers * * 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 "winbase.h" #include "winnt.h" #include "winternl.h" #include "module.h" #include "wine/exception.h" #include "excpt.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(ntdll); /* filter for page-fault exceptions */ static WINE_EXCEPTION_FILTER(page_fault) { if (GetExceptionCode() == EXCEPTION_ACCESS_VIOLATION) return EXCEPTION_EXECUTE_HANDLER; return EXCEPTION_CONTINUE_SEARCH; } /****************************************************************** * LdrDisableThreadCalloutsForDll (NTDLL.@) * */ NTSTATUS WINAPI LdrDisableThreadCalloutsForDll(HMODULE hModule) { WINE_MODREF *wm; NTSTATUS ret = STATUS_SUCCESS; RtlEnterCriticalSection( &loader_section ); wm = MODULE32_LookupHMODULE( hModule ); if ( !wm ) ret = STATUS_DLL_NOT_FOUND; else wm->flags |= WINE_MODREF_NO_DLL_CALLS; RtlLeaveCriticalSection( &loader_section ); return ret; } /* FIXME : MODULE_FindModule should depend on LdrGetDllHandle, not vice-versa */ NTSTATUS WINAPI LdrGetDllHandle(ULONG x, ULONG y, PUNICODE_STRING name, HMODULE *base) { STRING str; WINE_MODREF *wm; FIXME("%08lx %08lx %s %p : partial stub\n",x,y,debugstr_wn(name->Buffer,name->Length),base); *base = 0; RtlUnicodeStringToAnsiString(&str, name, TRUE); wm = MODULE_FindModule(str.Buffer); if(!wm) return STATUS_DLL_NOT_FOUND; *base = (PVOID) wm->module; return STATUS_SUCCESS; } /* FIXME : MODULE_GetProcAddress should depend on LdrGetProcedureAddress, not vice-versa */ NTSTATUS WINAPI LdrGetProcedureAddress(HMODULE base, PANSI_STRING name, ULONG ord, PVOID *address) { WARN("%p %s %ld %p\n", base, debugstr_an(name->Buffer,name->Length), ord, address); if(name) *address = MODULE_GetProcAddress( base, name->Buffer, -1, FALSE); else *address = MODULE_GetProcAddress( base, (LPSTR) ord, -1, FALSE); return (*address) ? STATUS_SUCCESS : STATUS_DLL_NOT_FOUND; } /****************************************************************** * LdrShutdownProcess (NTDLL.@) * */ NTSTATUS WINAPI LdrShutdownProcess(void) { TRACE("()\n"); MODULE_DllProcessDetach( TRUE, (LPVOID)1 ); return STATUS_SUCCESS; /* FIXME */ } /****************************************************************** * LdrShutdownThread (NTDLL.@) * */ NTSTATUS WINAPI LdrShutdownThread(void) { WINE_MODREF *wm; TRACE("()\n"); /* don't do any detach calls if process is exiting */ if (process_detaching) return STATUS_SUCCESS; /* FIXME: there is still a race here */ RtlEnterCriticalSection( &loader_section ); for ( wm = MODULE_modref_list; wm; wm = wm->next ) { if ( !(wm->flags & WINE_MODREF_PROCESS_ATTACHED) ) continue; if ( wm->flags & WINE_MODREF_NO_DLL_CALLS ) continue; MODULE_InitDLL( wm, DLL_THREAD_DETACH, NULL ); } RtlLeaveCriticalSection( &loader_section ); return STATUS_SUCCESS; /* FIXME */ } /*********************************************************************** * RtlImageNtHeader (NTDLL.@) */ PIMAGE_NT_HEADERS WINAPI RtlImageNtHeader(HMODULE hModule) { IMAGE_NT_HEADERS *ret; __TRY { IMAGE_DOS_HEADER *dos = (IMAGE_DOS_HEADER *)hModule; ret = NULL; if (dos->e_magic == IMAGE_DOS_SIGNATURE) { ret = (IMAGE_NT_HEADERS *)((char *)dos + dos->e_lfanew); if (ret->Signature != IMAGE_NT_SIGNATURE) ret = NULL; } } __EXCEPT(page_fault) { return NULL; } __ENDTRY return ret; } /*********************************************************************** * RtlImageDirectoryEntryToData (NTDLL.@) */ PVOID WINAPI RtlImageDirectoryEntryToData( HMODULE module, BOOL image, WORD dir, ULONG *size ) { const IMAGE_NT_HEADERS *nt; DWORD addr; if ((ULONG_PTR)module & 1) /* mapped as data file */ { module = (HMODULE)((ULONG_PTR)module & ~1); image = FALSE; } if (!(nt = RtlImageNtHeader( module ))) return NULL; if (dir >= nt->OptionalHeader.NumberOfRvaAndSizes) return NULL; if (!(addr = nt->OptionalHeader.DataDirectory[dir].VirtualAddress)) return NULL; *size = nt->OptionalHeader.DataDirectory[dir].Size; if (image || addr < nt->OptionalHeader.SizeOfHeaders) return (char *)module + addr; /* not mapped as image, need to find the section containing the virtual address */ return RtlImageRvaToVa( nt, module, addr, NULL ); } /*********************************************************************** * RtlImageRvaToSection (NTDLL.@) */ PIMAGE_SECTION_HEADER WINAPI RtlImageRvaToSection( const IMAGE_NT_HEADERS *nt, HMODULE module, DWORD rva ) { int i; IMAGE_SECTION_HEADER *sec = (IMAGE_SECTION_HEADER*)((char*)&nt->OptionalHeader + nt->FileHeader.SizeOfOptionalHeader); for (i = 0; i < nt->FileHeader.NumberOfSections; i++, sec++) { if ((sec->VirtualAddress <= rva) && (sec->VirtualAddress + sec->SizeOfRawData > rva)) return sec; } return NULL; } /*********************************************************************** * RtlImageRvaToVa (NTDLL.@) */ PVOID WINAPI RtlImageRvaToVa( const IMAGE_NT_HEADERS *nt, HMODULE module, DWORD rva, IMAGE_SECTION_HEADER **section ) { IMAGE_SECTION_HEADER *sec; if (section && *section) /* try this section first */ { sec = *section; if ((sec->VirtualAddress <= rva) && (sec->VirtualAddress + sec->SizeOfRawData > rva)) goto found; } if (!(sec = RtlImageRvaToSection( nt, module, rva ))) return NULL; found: if (section) *section = sec; return (char *)module + sec->PointerToRawData + (rva - sec->VirtualAddress); }