Sun Nov 16 07:42:44 1997 Alex Korobka <alex@trantor.pharm.sunysb.edu> * [windows/dce.c] [windows/clipboard.c] [windows/nonclient.c] Bug fixes. * [misc/shell.c] [resources/*] New "About" dialog. Sat Nov 15 17:30:18 1997 Alexandre Julliard <julliard@lrc.epfl.ch> * [configure.in] [Makefile.in] Replaced --with-library option by --disable-emulator. The default is now to build both the library and the emulator. Renamed --with options to --enable to follow autoconf guidelines. * [loader/main.c] [miscemu/main.c] (New file) Split initialization in WinelibInit/EmulatorInit. * [loader/*.c] Removed all remaining #ifdef's WINELIB. * [controls/widgets.c] [windows/mdi.c] Converted MDIClientWndProc to 32-bit. * [debugger/break.c] [if1632/signal.c] [include/selectors.h] [scheduler/thread.c] Code and data selector values are now computed at run-time. * [library/libres.c] Moved to loader/ directory. * [misc/main.c] [misc/version.c] (New file) Moved all version stuff to version.c. Cleaned up a bit. * [msdos/dpmi.c] Update the REALMODECALL structure on return from real-mode interrupt. * [windows/event.c] [windows/keyboard.c] Changed the way event coordinates are determined. Don't rely on the ConfigureNotify event values. This should fix all problems with cursor position in -desktop and -managed modes. Sat Nov 15 16:09:36 1997 Slaven Rezic <eserte@cs.tu-berlin.de> * [controls/button.c] (BUTTON_CheckAutoRadioButton): Prevent possible endless loop. Wed Nov 12 03:42:45 1997 Chris Faherty <chrisf@america.com> * [misc/ver.c] Changed VerInstall32A to assume srcdir as destination if destdir is blank. This was causing alot of DLL installation into SYSTEM directory to fail. * [loader/ne_image.c] NE_LoadSegment buffer[100] was too small and getting overruns. Changed it to buffer[200]. Sat Nov 8 06:09:57 1997 Len White <phreak@cgocable.net> * [misc/ddeml.c] [include/ddeml.h] [if1632/ddeml.spec] Added stub functions DdeConnectList(), DdeQueryNextServer(), DdeDisconnectList(), DdeSetUserHandle(), DdeAbandonTransaction(), DdePostAdvise(), DdeCreateDataHandle(), DdeAddData(), DdeGetData(), DdeAccessData(), DdeUnaccessData(), DdeEnableCallback(), DdeCmpStringHandles(). Fri Nov 7 19:44:26 1997 Olaf Flebbe <o.flebbe@science-computing.de> * [files/directory.c] Fix typo in directory.c [broke loading of cdplayer on nt40] * [misc/main.c] Implemented -winver nt40. * [loader/resource.c] [user32.spec] Stubs for CopyAcceleratorTable, Destroy AcceleratorTable. Thu Nov 6 22:37:04 1997 Morten Welinder <welinder@rentec.com> * [files/drive.c] (GetDiskFreeSpace32A): Cap at 2GB. * [include/windows.h] Prototype DrawIconEx and CreateDIBSection32. Define OBM_RADIOCHECK. Add DI_* macros. * [objects/dib.c] [if1632/gdi.spec] CreateDIBSection is a WINAPI. Renamed to CreateDIBSection32. Implement CreateDIBSection16. * [if1632/user.spec] [if1632/user32.spec] Add DrawIconEx. * [objects/cursoricon.c] (CopyIcon32): Fix bogus implementation. * [objects/bitmap.c] (CopyBitmap32): New function. (CopyImage32): Do bitmaps. * [graphics/x11drv/text.c] (X11DRV_ExtTextOut): Change ascent and descent default to avoid zero-thinkness overstrike line. * [include/debugstr.h] [misc/debugstr.c] New files. * [msdos/dpmi.c] Don't prototype do_mscdex. In INT_Int31Handler, handle real-mode int 0x21, ah=0x52. * [msdos/int2f.c] Add dummys for 0x1681 and 0x1682. * [misc/registry.c] Fix memory leaks in RegDeleteKey32W. * [objects/text.c] In TEXT_NextLine, fix another off-by-one bug. * [include/bitmaps/obm_radiocheck] New file. (It a small circle used to radio-button menu items when selected.) * [objects/oembitmap.c] Add obm_radiocheck. * [include/windows.h] [if1632/user32.spec] [controls/menu.c] [if1632/user.spec] Define CheckMenuRadioItem{16,32}. Define GetMenuItemRect{16,32}. Wed Nov 5 11:30:14 1997 Marcus Meissner <msmeissn@cip.informatik.uni-erlangen.de> * [misc/main.c] Auto adjust versions depending on binary. Tue Nov 4 15:21:00 1997 Kristian Nielsen <kristian.nielsen@risoe.dk> * [controls/listbox.c] Paint full background in listbox items with tab stops enabled. * [if1632/thunk.c] Copy some more message parameter structures (DRAWITEMSTRUCT16, COMPAREITEMSTRUCT16) to the stack segment to fix broken programs that need this. * [windows/dce.c] Only clip sibling windows when the parent has the WS_CLIPSIBLINGS style set. * [windows/focus.c] Make order of events in FOCUS_SwitchFocus() reflect API docs. * [windows/defdlg.c] Fix problem with loss of focus in some dialogs. * [win32/code_page.c] Fix return value for MultiByteToWideChar(). * [BUGS] BCW now works.
677 lines
23 KiB
C
677 lines
23 KiB
C
/*
|
|
* NE modules
|
|
*
|
|
* Copyright 1993 Robert J. Amstadt
|
|
* Copyright 1995 Alexandre Julliard
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <ctype.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include "neexe.h"
|
|
#include "windows.h"
|
|
#include "global.h"
|
|
#include "task.h"
|
|
#include "selectors.h"
|
|
#include "callback.h"
|
|
#include "file.h"
|
|
#include "module.h"
|
|
#include "stackframe.h"
|
|
#include "stddebug.h"
|
|
#include "debug.h"
|
|
#include "xmalloc.h"
|
|
|
|
|
|
/***********************************************************************
|
|
* NE_LoadSegment
|
|
*/
|
|
BOOL32 NE_LoadSegment( NE_MODULE *pModule, WORD segnum )
|
|
{
|
|
SEGTABLEENTRY *pSegTable, *pSeg;
|
|
WORD *pModuleTable;
|
|
WORD count, i, offset;
|
|
HMODULE16 module;
|
|
FARPROC16 address;
|
|
int fd;
|
|
struct relocation_entry_s *rep, *reloc_entries;
|
|
BYTE *func_name;
|
|
int size;
|
|
char* mem;
|
|
|
|
char buffer[256];
|
|
int ordinal, additive;
|
|
unsigned short *sp;
|
|
|
|
pSegTable = NE_SEG_TABLE( pModule );
|
|
pSeg = pSegTable + segnum - 1;
|
|
pModuleTable = NE_MODULE_TABLE( pModule );
|
|
|
|
if (!pSeg->filepos) return TRUE; /* No file image, just return */
|
|
|
|
fd = MODULE_OpenFile( pModule->self );
|
|
dprintf_module( stddeb, "Loading segment %d, selector=%04x\n",
|
|
segnum, pSeg->selector );
|
|
lseek( fd, pSeg->filepos << pModule->alignment, SEEK_SET );
|
|
if (pSeg->size) size = pSeg->size;
|
|
else if (pSeg->minsize) size = pSeg->minsize;
|
|
else size = 0x10000;
|
|
mem = GlobalLock16(pSeg->selector);
|
|
if (pModule->flags & NE_FFLAGS_SELFLOAD && segnum > 1)
|
|
{
|
|
/* Implement self loading segments */
|
|
SELFLOADHEADER *selfloadheader;
|
|
STACK16FRAME *stack16Top;
|
|
DWORD oldstack;
|
|
WORD oldselector, newselector;
|
|
HFILE32 hf = FILE_DupUnixHandle( fd );
|
|
|
|
selfloadheader = (SELFLOADHEADER *)
|
|
PTR_SEG_OFF_TO_LIN(pSegTable->selector,0);
|
|
oldstack = IF1632_Saved16_ss_sp;
|
|
oldselector = pSeg->selector;
|
|
IF1632_Saved16_ss_sp = PTR_SEG_OFF_TO_SEGPTR(pModule->self_loading_sel,
|
|
0xff00 - sizeof(*stack16Top));
|
|
stack16Top = CURRENT_STACK16;
|
|
stack16Top->saved_ss_sp = 0;
|
|
stack16Top->ds = stack16Top->es = pModule->self_loading_sel;
|
|
stack16Top->entry_point = 0;
|
|
stack16Top->entry_ip = 0;
|
|
stack16Top->entry_cs = 0;
|
|
stack16Top->bp = 0;
|
|
stack16Top->ip = 0;
|
|
stack16Top->cs = 0;
|
|
newselector = Callbacks->CallLoadAppSegProc(selfloadheader->LoadAppSeg,
|
|
pModule->self, hf, segnum );
|
|
_lclose32( hf );
|
|
if (newselector != oldselector) {
|
|
/* Self loaders like creating their own selectors;
|
|
* they love asking for trouble to Wine developers
|
|
*/
|
|
if (segnum == pModule->dgroup) {
|
|
memcpy(PTR_SEG_OFF_TO_LIN(oldselector,0),
|
|
PTR_SEG_OFF_TO_LIN(newselector,0),
|
|
pSeg->minsize ? pSeg->minsize : 0x10000);
|
|
FreeSelector(newselector);
|
|
pSeg->selector = oldselector;
|
|
fprintf(stderr, "A new selector was allocated for the dgroup segment\n"
|
|
"Old selector is %d, new one is %d", oldselector, newselector);
|
|
} else {
|
|
FreeSelector(pSeg->selector);
|
|
pSeg->selector = newselector;
|
|
}
|
|
}
|
|
|
|
IF1632_Saved16_ss_sp = oldstack;
|
|
}
|
|
else if (!(pSeg->flags & NE_SEGFLAGS_ITERATED))
|
|
read(fd, mem, size);
|
|
else {
|
|
/*
|
|
The following bit of code for "iterated segments" was written without
|
|
any documentation on the format of these segments. It seems to work,
|
|
but may be missing something. If you have any doco please either send
|
|
it to me or fix the code yourself. gfm@werple.mira.net.au
|
|
*/
|
|
char* buff = xmalloc(size);
|
|
char* curr = buff;
|
|
read(fd, buff, size);
|
|
while(curr < buff + size) {
|
|
unsigned int rept = *((short*) curr)++;
|
|
unsigned int len = *((short*) curr)++;
|
|
for(; rept > 0; rept--) {
|
|
char* bytes = curr;
|
|
unsigned int byte;
|
|
for(byte = 0; byte < len; byte++)
|
|
*mem++ = *bytes++;
|
|
}
|
|
curr += len;
|
|
}
|
|
free(buff);
|
|
}
|
|
|
|
pSeg->flags |= NE_SEGFLAGS_LOADED;
|
|
if (!(pSeg->flags & NE_SEGFLAGS_RELOC_DATA))
|
|
return TRUE; /* No relocation data, we are done */
|
|
|
|
read( fd, &count, sizeof(count) );
|
|
if (!count) return TRUE;
|
|
|
|
dprintf_fixup( stddeb, "Fixups for %*.*s, segment %d, selector %04x\n",
|
|
*((BYTE *)pModule + pModule->name_table),
|
|
*((BYTE *)pModule + pModule->name_table),
|
|
(char *)pModule + pModule->name_table + 1,
|
|
segnum, pSeg->selector );
|
|
|
|
reloc_entries = (struct relocation_entry_s *)xmalloc(count * sizeof(struct relocation_entry_s));
|
|
if (read( fd, reloc_entries, count * sizeof(struct relocation_entry_s)) !=
|
|
count * sizeof(struct relocation_entry_s))
|
|
{
|
|
dprintf_fixup( stddeb, "Unable to read relocation information\n" );
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
* Go through the relocation table on entry at a time.
|
|
*/
|
|
rep = reloc_entries;
|
|
for (i = 0; i < count; i++, rep++)
|
|
{
|
|
/*
|
|
* Get the target address corresponding to this entry.
|
|
*/
|
|
|
|
/* If additive, there is no target chain list. Instead, add source
|
|
and target */
|
|
additive = rep->relocation_type & NE_RELFLAG_ADDITIVE;
|
|
rep->relocation_type &= 0x3;
|
|
|
|
switch (rep->relocation_type)
|
|
{
|
|
case NE_RELTYPE_ORDINAL:
|
|
module = pModuleTable[rep->target1-1];
|
|
ordinal = rep->target2;
|
|
address = MODULE_GetEntryPoint( module, ordinal );
|
|
if (!address)
|
|
{
|
|
NE_MODULE *pTarget = MODULE_GetPtr( module );
|
|
if (!pTarget)
|
|
fprintf( stderr, "Module not found: %04x, reference %d of module %*.*s\n",
|
|
module, rep->target1,
|
|
*((BYTE *)pModule + pModule->name_table),
|
|
*((BYTE *)pModule + pModule->name_table),
|
|
(char *)pModule + pModule->name_table + 1 );
|
|
else
|
|
fprintf( stderr, "Warning: no handler for %*.*s.%d, setting to 0:0\n",
|
|
*((BYTE *)pTarget + pTarget->name_table),
|
|
*((BYTE *)pTarget + pTarget->name_table),
|
|
(char *)pTarget + pTarget->name_table + 1,
|
|
ordinal );
|
|
}
|
|
if (debugging_fixup)
|
|
{
|
|
NE_MODULE *pTarget = MODULE_GetPtr( module );
|
|
fprintf( stddeb,"%d: %*.*s.%d=%04x:%04x\n", i + 1,
|
|
*((BYTE *)pTarget + pTarget->name_table),
|
|
*((BYTE *)pTarget + pTarget->name_table),
|
|
(char *)pTarget + pTarget->name_table + 1,
|
|
ordinal, HIWORD(address), LOWORD(address) );
|
|
}
|
|
break;
|
|
|
|
case NE_RELTYPE_NAME:
|
|
module = pModuleTable[rep->target1-1];
|
|
func_name = (char *)pModule + pModule->import_table + rep->target2;
|
|
memcpy( buffer, func_name+1, *func_name );
|
|
buffer[*func_name] = '\0';
|
|
func_name = buffer;
|
|
ordinal = MODULE_GetOrdinal( module, func_name );
|
|
|
|
address = MODULE_GetEntryPoint( module, ordinal );
|
|
|
|
if (!address)
|
|
{
|
|
NE_MODULE *pTarget = MODULE_GetPtr( module );
|
|
fprintf( stderr, "Warning: no handler for %.*s.%s, setting to 0:0\n",
|
|
*((BYTE *)pTarget + pTarget->name_table),
|
|
(char *)pTarget + pTarget->name_table + 1, func_name );
|
|
}
|
|
if (debugging_fixup)
|
|
{
|
|
NE_MODULE *pTarget = MODULE_GetPtr( module );
|
|
fprintf( stddeb,"%d: %.*s.%s=%04x:%04x\n", i + 1,
|
|
*((BYTE *)pTarget + pTarget->name_table),
|
|
(char *)pTarget + pTarget->name_table + 1,
|
|
func_name, HIWORD(address), LOWORD(address) );
|
|
}
|
|
break;
|
|
|
|
case NE_RELTYPE_INTERNAL:
|
|
if ((rep->target1 & 0xff) == 0xff)
|
|
{
|
|
address = MODULE_GetEntryPoint( pModule->self, rep->target2 );
|
|
}
|
|
else
|
|
{
|
|
address = (FARPROC16)PTR_SEG_OFF_TO_SEGPTR( pSegTable[rep->target1-1].selector, rep->target2 );
|
|
}
|
|
|
|
dprintf_fixup(stddeb,"%d: %04x:%04x\n",
|
|
i + 1, HIWORD(address), LOWORD(address) );
|
|
break;
|
|
|
|
case NE_RELTYPE_OSFIXUP:
|
|
/* Relocation type 7:
|
|
*
|
|
* These appear to be used as fixups for the Windows
|
|
* floating point emulator. Let's just ignore them and
|
|
* try to use the hardware floating point. Linux should
|
|
* successfully emulate the coprocessor if it doesn't
|
|
* exist.
|
|
*/
|
|
dprintf_fixup(stddeb,
|
|
"%d: ADDR TYPE %d, TYPE %d, OFFSET %04x, ",
|
|
i + 1, rep->address_type, rep->relocation_type,
|
|
rep->offset);
|
|
dprintf_fixup(stddeb,"TARGET %04x %04x\n",
|
|
rep->target1, rep->target2);
|
|
continue;
|
|
|
|
default:
|
|
dprintf_fixup(stddeb,
|
|
"WARNING: %d: ADDR TYPE %d, unknown TYPE %d, OFFSET %04x, ",
|
|
i + 1, rep->address_type, rep->relocation_type,
|
|
rep->offset);
|
|
dprintf_fixup(stddeb,"TARGET %04x %04x\n",
|
|
rep->target1, rep->target2);
|
|
free(reloc_entries);
|
|
return FALSE;
|
|
}
|
|
|
|
offset = rep->offset;
|
|
|
|
/* Apparently, high bit of address_type is sometimes set; */
|
|
/* we ignore it for now */
|
|
if (rep->address_type > NE_RADDR_OFFSET32)
|
|
fprintf( stderr, "WARNING: module %s: unknown reloc addr type = 0x%02x. Please report.\n",
|
|
MODULE_GetModuleName(pModule->self), rep->address_type );
|
|
|
|
switch (rep->address_type & 0x7f)
|
|
{
|
|
case NE_RADDR_LOWBYTE:
|
|
do {
|
|
sp = PTR_SEG_OFF_TO_LIN( pSeg->selector, offset );
|
|
dprintf_fixup(stddeb," %04x:%04x:%04x BYTE%s\n",
|
|
pSeg->selector, offset, *sp, additive ? " additive":"");
|
|
offset = *sp;
|
|
if(additive)
|
|
*(unsigned char*)sp = (unsigned char)(((int)address+offset) & 0xFF);
|
|
else
|
|
*(unsigned char*)sp = (unsigned char)((int)address & 0xFF);
|
|
}
|
|
while (offset != 0xffff && !additive);
|
|
break;
|
|
|
|
case NE_RADDR_OFFSET16:
|
|
do {
|
|
sp = PTR_SEG_OFF_TO_LIN( pSeg->selector, offset );
|
|
dprintf_fixup(stddeb," %04x:%04x:%04x OFFSET16%s\n",
|
|
pSeg->selector, offset, *sp, additive ? " additive" : "" );
|
|
offset = *sp;
|
|
*sp = LOWORD(address);
|
|
if (additive) *sp += offset;
|
|
}
|
|
while (offset != 0xffff && !additive);
|
|
break;
|
|
|
|
case NE_RADDR_POINTER32:
|
|
do {
|
|
sp = PTR_SEG_OFF_TO_LIN( pSeg->selector, offset );
|
|
dprintf_fixup(stddeb," %04x:%04x:%04x POINTER32%s\n",
|
|
pSeg->selector, offset, *sp, additive ? " additive" : "" );
|
|
offset = *sp;
|
|
*sp = LOWORD(address);
|
|
if (additive) *sp += offset;
|
|
*(sp+1) = HIWORD(address);
|
|
}
|
|
while (offset != 0xffff && !additive);
|
|
break;
|
|
|
|
case NE_RADDR_SELECTOR:
|
|
do {
|
|
sp = PTR_SEG_OFF_TO_LIN( pSeg->selector, offset );
|
|
dprintf_fixup(stddeb," %04x:%04x:%04x SELECTOR%s\n",
|
|
pSeg->selector, offset, *sp, additive ? " additive" : "" );
|
|
offset = *sp;
|
|
*sp = HIWORD(address);
|
|
/* Borland creates additive records with offset zero. Strange, but OK */
|
|
if(additive && offset)
|
|
fprintf(stderr,"Additive selector to %4.4x.Please report\n",offset);
|
|
}
|
|
/* FIXME: Quicken 5 has a zero offset fixup. This seems to work */
|
|
while (offset && offset != 0xffff && !additive);
|
|
break;
|
|
|
|
default:
|
|
dprintf_fixup(stddeb,
|
|
"WARNING: %d: unknown ADDR TYPE %d, TYPE %d, OFFSET %04x, ",
|
|
i + 1, rep->address_type, rep->relocation_type,
|
|
rep->offset);
|
|
dprintf_fixup(stddeb,
|
|
"TARGET %04x %04x\n", rep->target1, rep->target2);
|
|
free(reloc_entries);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
free(reloc_entries);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* NE_LoadAllSegments
|
|
*/
|
|
BOOL32 NE_LoadAllSegments( NE_MODULE *pModule )
|
|
{
|
|
int i;
|
|
|
|
if (pModule->flags & NE_FFLAGS_SELFLOAD)
|
|
{
|
|
HFILE32 hf;
|
|
/* Handle self loading modules */
|
|
SEGTABLEENTRY * pSegTable = (SEGTABLEENTRY *) NE_SEG_TABLE(pModule);
|
|
SELFLOADHEADER *selfloadheader;
|
|
STACK16FRAME *stack16Top;
|
|
HMODULE16 hselfload = GetModuleHandle16("WPROCS");
|
|
DWORD oldstack;
|
|
WORD saved_dgroup = pSegTable[pModule->dgroup - 1].selector;
|
|
|
|
dprintf_module(stddeb, "MODULE_Load: %.*s is a self-loading module!\n",
|
|
*((BYTE*)pModule + pModule->name_table),
|
|
(char *)pModule + pModule->name_table + 1);
|
|
if (!NE_LoadSegment( pModule, 1 )) return FALSE;
|
|
selfloadheader = (SELFLOADHEADER *)
|
|
PTR_SEG_OFF_TO_LIN(pSegTable->selector, 0);
|
|
selfloadheader->EntryAddrProc = MODULE_GetEntryPoint(hselfload,27);
|
|
selfloadheader->MyAlloc = MODULE_GetEntryPoint(hselfload,28);
|
|
selfloadheader->SetOwner = MODULE_GetEntryPoint(GetModuleHandle16("KERNEL"),403);
|
|
pModule->self_loading_sel = GlobalHandleToSel(GLOBAL_Alloc(GMEM_ZEROINIT, 0xFF00, pModule->self, FALSE, FALSE, FALSE));
|
|
oldstack = IF1632_Saved16_ss_sp;
|
|
IF1632_Saved16_ss_sp = PTR_SEG_OFF_TO_SEGPTR(pModule->self_loading_sel,
|
|
0xff00 - sizeof(*stack16Top) );
|
|
stack16Top = CURRENT_STACK16;
|
|
stack16Top->saved_ss_sp = 0;
|
|
stack16Top->ebp = 0;
|
|
stack16Top->ds = stack16Top->es = pModule->self_loading_sel;
|
|
stack16Top->entry_point = 0;
|
|
stack16Top->entry_ip = 0;
|
|
stack16Top->entry_cs = 0;
|
|
stack16Top->bp = 0;
|
|
stack16Top->ip = 0;
|
|
stack16Top->cs = 0;
|
|
|
|
hf = FILE_DupUnixHandle( MODULE_OpenFile( pModule->self ) );
|
|
Callbacks->CallBootAppProc(selfloadheader->BootApp, pModule->self, hf);
|
|
_lclose32(hf);
|
|
/* some BootApp procs overwrite the selector of dgroup */
|
|
pSegTable[pModule->dgroup - 1].selector = saved_dgroup;
|
|
IF1632_Saved16_ss_sp = oldstack;
|
|
for (i = 2; i <= pModule->seg_count; i++)
|
|
if (!NE_LoadSegment( pModule, i )) return FALSE;
|
|
}
|
|
else
|
|
{
|
|
for (i = 1; i <= pModule->seg_count; i++)
|
|
if (!NE_LoadSegment( pModule, i )) return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* NE_LoadDLLs
|
|
*/
|
|
BOOL32 NE_LoadDLLs( NE_MODULE *pModule )
|
|
{
|
|
int i;
|
|
WORD *pModRef = (WORD *)((char *)pModule + pModule->modref_table);
|
|
WORD *pDLLs = (WORD *)GlobalLock16( pModule->dlls_to_init );
|
|
|
|
for (i = 0; i < pModule->modref_count; i++, pModRef++)
|
|
{
|
|
char buffer[256];
|
|
BYTE *pstr = (BYTE *)pModule + pModule->import_table + *pModRef;
|
|
memcpy( buffer, pstr + 1, *pstr );
|
|
strcpy( buffer + *pstr, ".dll" );
|
|
dprintf_module( stddeb, "Loading '%s'\n", buffer );
|
|
if (!(*pModRef = MODULE_FindModule( buffer )))
|
|
{
|
|
/* If the DLL is not loaded yet, load it and store */
|
|
/* its handle in the list of DLLs to initialize. */
|
|
HMODULE16 hDLL;
|
|
|
|
if ((hDLL = MODULE_Load( buffer, (LPVOID)-1, NE_FFLAGS_IMPLICIT )) == 2)
|
|
{
|
|
/* file not found */
|
|
char *p;
|
|
|
|
/* Try with prepending the path of the current module */
|
|
GetModuleFileName16( pModule->self, buffer, sizeof(buffer) );
|
|
if (!(p = strrchr( buffer, '\\' ))) p = buffer;
|
|
memcpy( p + 1, pstr + 1, *pstr );
|
|
strcpy( p + 1 + *pstr, ".dll" );
|
|
hDLL = MODULE_Load( buffer, (LPVOID)-1, NE_FFLAGS_IMPLICIT );
|
|
}
|
|
if (hDLL < 32)
|
|
{
|
|
/* FIXME: cleanup what was done */
|
|
|
|
fprintf( stderr, "Could not load '%s' required by '%.*s', error = %d\n",
|
|
buffer, *((BYTE*)pModule + pModule->name_table),
|
|
(char *)pModule + pModule->name_table + 1, hDLL );
|
|
return FALSE;
|
|
}
|
|
*pModRef = MODULE_HANDLEtoHMODULE16( hDLL );
|
|
*pDLLs++ = *pModRef;
|
|
}
|
|
else /* Increment the reference count of the DLL */
|
|
{
|
|
NE_MODULE *pOldDLL = MODULE_GetPtr( *pModRef );
|
|
if (pOldDLL) pOldDLL->count++;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* NE_FixupPrologs
|
|
*
|
|
* Fixup the exported functions prologs.
|
|
*/
|
|
void NE_FixupPrologs( NE_MODULE *pModule )
|
|
{
|
|
SEGTABLEENTRY *pSegTable;
|
|
WORD dgroup = 0;
|
|
WORD sel;
|
|
BYTE *p, *fixup_ptr, count;
|
|
|
|
pSegTable = NE_SEG_TABLE(pModule);
|
|
if (pModule->flags & NE_FFLAGS_SINGLEDATA)
|
|
dgroup = pSegTable[pModule->dgroup-1].selector;
|
|
|
|
dprintf_module( stddeb, "MODULE_FixupPrologs(%04x)\n", pModule->self );
|
|
p = (BYTE *)pModule + pModule->entry_table;
|
|
while (*p)
|
|
{
|
|
if (p[1] == 0) /* Unused entry */
|
|
{
|
|
p += 2; /* Skip it */
|
|
continue;
|
|
}
|
|
if (p[1] == 0xfe) /* Constant entry */
|
|
{
|
|
p += 2 + *p * 3; /* Skip it */
|
|
continue;
|
|
}
|
|
|
|
/* Now fixup the entries of this bundle */
|
|
count = *p;
|
|
sel = p[1];
|
|
p += 2;
|
|
while (count-- > 0)
|
|
{
|
|
dprintf_module( stddeb,"Flags: %04x, sel %02x ", *p, sel);
|
|
/* According to the output generated by TDUMP, the flags mean:
|
|
* 0x0001 function is exported
|
|
* 0x0002 Single data (seems to occur only in DLLs)
|
|
*/
|
|
if (sel == 0xff) { /* moveable */
|
|
dprintf_module( stddeb, "(%02x) o %04x ", p[3], *(WORD *)(p+4) );
|
|
fixup_ptr = (char *)GET_SEL_BASE(pSegTable[p[3]-1].selector) + *(WORD *)(p + 4);
|
|
} else { /* fixed */
|
|
dprintf_module( stddeb, "offset %04x ", *(WORD *)(p+1) );
|
|
fixup_ptr = (char *)GET_SEL_BASE(pSegTable[sel-1].selector) + *(WORD *)(p + 1);
|
|
}
|
|
dprintf_module( stddeb, "Signature: %02x %02x %02x,ff %x\n",
|
|
fixup_ptr[0], fixup_ptr[1], fixup_ptr[2],
|
|
pModule->flags );
|
|
if (*p & 0x0001)
|
|
{
|
|
/* Verify the signature */
|
|
if (((fixup_ptr[0] == 0x1e && fixup_ptr[1] == 0x58)
|
|
|| (fixup_ptr[0] == 0x8c && fixup_ptr[1] == 0xd8))
|
|
&& fixup_ptr[2] == 0x90)
|
|
{
|
|
if (*p & 0x0002)
|
|
{
|
|
if (pModule->flags & NE_FFLAGS_MULTIPLEDATA)
|
|
{
|
|
/* can this happen? */
|
|
fprintf( stderr, "FixupPrologs got confused\n" );
|
|
}
|
|
else if (pModule->flags & NE_FFLAGS_SINGLEDATA)
|
|
{
|
|
*fixup_ptr = 0xb8; /* MOV AX, */
|
|
*(WORD *)(fixup_ptr+1) = dgroup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (pModule->flags & NE_FFLAGS_MULTIPLEDATA) {
|
|
fixup_ptr[0] = 0x90; /* non-library: NOPs */
|
|
fixup_ptr[1] = 0x90;
|
|
fixup_ptr[2] = 0x90;
|
|
}
|
|
}
|
|
} else {
|
|
dprintf_fixup( stddeb, "Unknown signature\n" );
|
|
}
|
|
}
|
|
else
|
|
dprintf_module( stddeb,"\n");
|
|
p += (sel == 0xff) ? 6 : 3;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* NE_InitDLL
|
|
*
|
|
* Call the DLL initialization code
|
|
*/
|
|
static BOOL32 NE_InitDLL( TDB* pTask, HMODULE16 hModule )
|
|
{
|
|
NE_MODULE *pModule;
|
|
SEGTABLEENTRY *pSegTable;
|
|
CONTEXT context;
|
|
|
|
/* Registers at initialization must be:
|
|
* cx heap size
|
|
* di library instance
|
|
* ds data segment if any
|
|
* es:si command line (always 0)
|
|
*/
|
|
|
|
if (!(pModule = MODULE_GetPtr( hModule ))) return FALSE;
|
|
pSegTable = NE_SEG_TABLE( pModule );
|
|
|
|
if (!(pModule->flags & NE_FFLAGS_LIBMODULE) ||
|
|
(pModule->flags & NE_FFLAGS_WIN32)) return TRUE; /*not a library*/
|
|
|
|
/* Call USER signal handler. This is necessary to install a
|
|
* proper loader for HICON and HCURSOR resources that this DLL
|
|
* may contain. InitApp() does this for task modules. */
|
|
|
|
if (pTask && pTask->userhandler)
|
|
{
|
|
pTask->userhandler( hModule, USIG_DLL_LOAD, 0, pTask->hInstance,
|
|
pTask->hQueue );
|
|
}
|
|
|
|
if (!pModule->cs) return TRUE; /* no initialization code */
|
|
|
|
memset( &context, 0, sizeof(context) );
|
|
|
|
if (!(pModule->flags & NE_FFLAGS_SINGLEDATA))
|
|
{
|
|
if (pModule->flags & NE_FFLAGS_MULTIPLEDATA || pModule->dgroup)
|
|
{
|
|
/* Not SINGLEDATA */
|
|
fprintf(stderr, "Library is not marked SINGLEDATA\n");
|
|
exit(1);
|
|
}
|
|
else /* DATA NONE DLL */
|
|
{
|
|
DS_reg(&context) = 0;
|
|
ECX_reg(&context) = 0;
|
|
}
|
|
}
|
|
else /* DATA SINGLE DLL */
|
|
{
|
|
DS_reg(&context) = pSegTable[pModule->dgroup-1].selector;
|
|
ECX_reg(&context) = pModule->heap_size;
|
|
}
|
|
|
|
CS_reg(&context) = pSegTable[pModule->cs-1].selector;
|
|
EIP_reg(&context) = pModule->ip;
|
|
EBP_reg(&context) = OFFSETOF(IF1632_Saved16_ss_sp)
|
|
+ (WORD)&((STACK16FRAME*)0)->bp;
|
|
EDI_reg(&context) = DS_reg(&context) ? DS_reg(&context) : hModule;
|
|
|
|
|
|
pModule->cs = 0; /* Don't initialize it twice */
|
|
dprintf_dll( stddeb, "Calling LibMain, cs:ip=%04lx:%04x ds=%04lx di=%04x cx=%04x\n",
|
|
CS_reg(&context), IP_reg(&context), DS_reg(&context),
|
|
DI_reg(&context), CX_reg(&context) );
|
|
Callbacks->CallRegisterProc( &context, 0 );
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* NE_InitializeDLLs
|
|
*
|
|
* Recursively initialize all DLLs (according to the order in which
|
|
* they where loaded).
|
|
*/
|
|
void NE_InitializeDLLs( HMODULE16 hModule )
|
|
{
|
|
TDB* pTask = (TDB*)GlobalLock16(GetCurrentTask());
|
|
NE_MODULE *pModule;
|
|
HMODULE16 *pDLL;
|
|
|
|
if (!(pModule = MODULE_GetPtr( hModule ))) return;
|
|
if (pModule->flags & NE_FFLAGS_WIN32) return;
|
|
|
|
if (pModule->dlls_to_init)
|
|
{
|
|
HGLOBAL16 to_init = pModule->dlls_to_init;
|
|
pModule->dlls_to_init = 0;
|
|
for (pDLL = (HMODULE16 *)GlobalLock16( to_init ); *pDLL; pDLL++)
|
|
{
|
|
NE_InitializeDLLs( *pDLL );
|
|
}
|
|
GlobalFree16( to_init );
|
|
}
|
|
NE_InitDLL( pTask, hModule );
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* PatchCodeHandle
|
|
*
|
|
* Needed for self-loading modules.
|
|
*/
|
|
|
|
/* It does nothing */
|
|
void WINAPI PatchCodeHandle(HANDLE16 hSel)
|
|
{
|
|
}
|