Fri Sep 3 11:52:18 1993 Bob Amstadt * [windows/timer.c] Changed to use CallWindowProc() rather directly calling callback. * [windows/event.c] Implemented SetCapture() and ReleaseCapture() * [windows/keyboard.c] Created stub for GetKeyState() * [objects/linedda.c] Created stub for LineDDA() * [if1632/callback.c] Created callback handler for LineDDA callback procedure. * [if1632/callback.c] Created FreeProcInstance() Fri Sep 3 08:36:52 1993 David Metcalfe * [loader/signal.c] Patch to and code for INT 1A Thu Sep 2 00:31:54 1993 Alexandre Julliard * [objects/font.c] [objects/text.c] More text support: implemented justification and underlining. * [windows/clipping.c] [objects/clipping.c] Moved low-level clipping functions to objects/clipping.c. * [windows/clipping.c] [windows/event.c] [windows/message.c] Implemented window update regions. * [windows/dc.c] [objects/dcvalues.c] Moved some device-independent DC functions to objects/dcvalues.c. * [windows/graphics.c] Implemented InvertRect() and GetPixel(). Sat Aug 28 08:40:23 1993 Eric Youngdale * [include/neexe.h] [loader/wine.c] Added code to handle relocation type 4. * [loader/signal.h] [loader/wine.c] [loader/selector.c] Added support for dos interrupts. Thu 26 Aug 19:15:00 1993 Eric Youngdale * [loader/selector.c] Fixed bug dealing with loading DLLs. Thu Aug 26 19:22:40 1993 Alexandre Julliard * [include/gdi.h] [objects/font.c] [windows/dc.c] Beginning of real font support. * [windows/graphics.c] Implemented PatBlt(). * [memory/global.c] Corrected a bug with linked list handling in GlobalAlloc(). * [objects/bitmap.c] Corrected a bug in BITMAP_SelectObject(). Tue Aug 24 19:22:40 1993 David Metcalfe * [controls/Command*] [controls/Label*] [controls[MenuButto*] [controls/SmeMenuButt*] Change code to support & as a special character in menu item text. Tue Aug 24 19:22:40 1993 Alexandre Julliard * [include/gdi.h] [windows/dc.c] Heavily modified the DC structure for better device-independence. * [objects/bitmap.c] Implemented bitmap dimensions. * [windows/dc.c] [windows/dce.c] Implemented DC state saving and restoring. * [windows/dc.c] Implemented ROP mode. * [windows/graphics.c] Implemented FillRect(). Mon Aug 23 22:08:34 1993 Bob Amstadt (bob at pooh) * [misc/xt.c] Fixed bug in InvalidateRect(). Solitaire attempted to clear window before it was realized. * [loader/resource.c] Began rewrite of LoadBitmap(). * [loader/wine.c] Fixed code which set Argv and Argc global variables. * [loader/selector.c] Added code to set up command line arguments. * [include/neexe.h] Fixed error in PSP structure. Tue Aug 17 20:41:12 1993 Alexandre Julliard * [include/gdi.h] [windows/dc.c] Implemented device capabilities. * [objects/region.c] Implemented EqualRgn() and CombineRgn(). * [windows/clipping.c] Implemented Save/RestoreVisRgn(). * [windows/graphics.c] Implemented PaintRgn() and FillRgn(). * [windows/mapping.c] Implemented mapping modes. Tue Aug 10 14:07:38 1993 Alexandre Julliard * [if1632/user.spec] [misc/rect.c] Implemented rectangle API functions. * [if1632/gdi.spec] [include/gdi.h] [objects/region.c] Implemented regions. * [windows/class.c] Corrected a typo in UnregisterClass(). * [windows/clipping.c] [windows/dc.c] Implemented DC clipping and visible region. Tue Aug 10 20:57:56 1993 Bob Amstadt (bob at pooh) * [controls/menu.c] [windows/win.c] SetMenu(), GetMenu(), CheckMenuItem() implemented Thu Aug 5 22:33:22 1993 Bob Amstadt (bob at pooh) * [controls/menu.c] [windows/win.c] Many improvements menus. LoadMenu() should work. Wed Aug 4 14:55:36 1993 Alexandre Julliard * [objects/dib.c] Started the implementation of device-independent bitmaps. * [objects/bitmap.c] Added support for multiple bitmap depths. * [objects/brush.c] Implemented pattern brushes. * [windows/dc.c] [windows/graphics.c] Implemented some GDI graphics primitives. Tue Aug 3 21:16:47 1993 Bob Amstadt (bob at pooh) * [controls/menu.c] [windows/win.c] [include/menu.h] Code to load class menus from executable file. * [if1632/user.spec] Fixed specification of SendMessage() and PostMessage. Mon Jul 26 21:53:24 1993 Alexandre Julliard * [if1632/call.S] Corrected a bug in KERNEL_InitTask(). * [include/windows.h] Added a lot of constants. * [loader/selector.c] Corrected a bug in segment allocation in CreateSelectors(). * [objects/bitmap.c] Implemented SelectObject() for bitmaps. * [objects/brush.c] Implemented hatched brushes and SelectObject(). * [objects/gdiobj.c] Removed linked list (not needed). * [objects/palette.c] Implemented system palette creation and misc. palette API functions. * [windows/timer.c] Implemented timers. * [windows/dc.c] Implemented memory device contexts. Tue Jul 20 10:38:59 1993 Bob Amstadt (bob at pooh) * [dos.c] Split DOS3Call() out of kernel.c. Added support for get date and time functions. * [call.S] Added function ReturnFromRegisterFunc() to allow DOS calls to return values in registers. * [regfunc.h] Macros to access registers saved on stack. Tue Jul 20 10:38:59 1993 Alexandre Julliard * [win.c] Corrected allocation of the WM_CREATE data structure. * [dce.c] [dce.h] Implemented DCE handling. * [bitmap.c] [brush.c] [dc.c] [font.c] [gdi.h] [gdi.spec] [gdiobj.c] [palette.c] [pen.c] Implemented the GDI objects data structures and allocation. * [windows.h] Added several structures and constants for GDI objects. Mon Jul 19 12:51:10 1993 Bob Amstadt (bob at pooh) * [ldtlib.c] Modified system calls to match Linus' new interface for the LDT modification. * [win.c] Fixed bug with WM_CREATE message. * [heap.c] [kernel.spec] Completed local heap allocation functions. * [global.c] Created function GlobalQuickAlloc() for easy allocation from DLLs
565 lines
15 KiB
C
565 lines
15 KiB
C
static char RCSId[] = "$Id: wine.c,v 1.2 1993/07/04 04:04:21 root Exp root $";
|
||
static char Copyright[] = "Copyright Robert J. Amstadt, 1993";
|
||
|
||
#include <stdio.h>
|
||
#include <stdlib.h>
|
||
#include <sys/types.h>
|
||
#include <sys/stat.h>
|
||
#include <fcntl.h>
|
||
#include <unistd.h>
|
||
#include <linux/unistd.h>
|
||
#include <linux/head.h>
|
||
#include <linux/ldt.h>
|
||
#include <linux/segment.h>
|
||
#include <string.h>
|
||
#include <errno.h>
|
||
#include "neexe.h"
|
||
#include "segmem.h"
|
||
#include "prototypes.h"
|
||
#include "dlls.h"
|
||
#include "wine.h"
|
||
#include "windows.h"
|
||
|
||
extern int CallToInit16(unsigned long csip, unsigned long sssp,
|
||
unsigned short ds);
|
||
extern void CallTo32();
|
||
|
||
char * GetModuleName(struct w_files * wpnt, int index, char *buffer);
|
||
extern unsigned char ran_out;
|
||
unsigned short WIN_StackSize;
|
||
unsigned short WIN_HeapSize;
|
||
|
||
struct w_files * wine_files = NULL;
|
||
|
||
char **Argv;
|
||
int Argc;
|
||
struct mz_header_s *CurrentMZHeader;
|
||
struct ne_header_s *CurrentNEHeader;
|
||
int CurrentNEFile;
|
||
|
||
/**********************************************************************
|
||
* DebugPrintString
|
||
*/
|
||
int
|
||
DebugPrintString(char *str)
|
||
{
|
||
printf("%s", str);
|
||
return 0;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* myerror
|
||
*/
|
||
void
|
||
myerror(const char *s)
|
||
{
|
||
if (s == NULL)
|
||
perror("wine");
|
||
else
|
||
fprintf(stderr, "wine: %s\n", s);
|
||
|
||
exit(1);
|
||
}
|
||
|
||
|
||
/* Load one NE format executable into memory */
|
||
LoadImage(char * filename, char * modulename)
|
||
{
|
||
unsigned int read_size;
|
||
int i;
|
||
struct w_files * wpnt, *wpnt1;
|
||
unsigned int status;
|
||
|
||
/* First allocate a spot to store the info we collect, and add it to
|
||
* our linked list.
|
||
*/
|
||
|
||
wpnt = (struct w_files *) malloc(sizeof(struct w_files));
|
||
if(wine_files == NULL)
|
||
wine_files = wpnt;
|
||
else {
|
||
wpnt1 = wine_files;
|
||
while(wpnt1->next) wpnt1 = wpnt1->next;
|
||
wpnt1->next = wpnt;
|
||
};
|
||
wpnt->next = NULL;
|
||
|
||
/*
|
||
* Open file for reading.
|
||
*/
|
||
wpnt->fd = open(filename, O_RDONLY);
|
||
if (wpnt->fd < 0)
|
||
{
|
||
myerror(NULL);
|
||
}
|
||
/*
|
||
* Establish header pointers.
|
||
*/
|
||
wpnt->filename = strdup(filename);
|
||
wpnt->name = NULL;
|
||
if(modulename) wpnt->name = strdup(modulename);
|
||
|
||
wpnt->mz_header = (struct mz_header_s *) malloc(sizeof(struct mz_header_s));;
|
||
status = lseek(wpnt->fd, 0, SEEK_SET);
|
||
if (read(wpnt->fd, wpnt->mz_header, sizeof(struct mz_header_s)) !=
|
||
sizeof(struct mz_header_s))
|
||
{
|
||
myerror("Unable to read MZ header from file");
|
||
}
|
||
if (wpnt->mz_header->must_be_0x40 != 0x40)
|
||
myerror("This is not a Windows program");
|
||
|
||
wpnt->ne_header = (struct ne_header_s *) malloc(sizeof(struct ne_header_s));
|
||
status = lseek(wpnt->fd, wpnt->mz_header->ne_offset, SEEK_SET);
|
||
if (read(wpnt->fd, wpnt->ne_header, sizeof(struct ne_header_s))
|
||
!= sizeof(struct ne_header_s))
|
||
{
|
||
myerror("Unable to read NE header from file");
|
||
}
|
||
if (wpnt->ne_header->header_type[0] != 'N' ||
|
||
wpnt->ne_header->header_type[1] != 'E')
|
||
myerror("This is not a Windows program");
|
||
|
||
if(wine_files == wpnt){
|
||
CurrentMZHeader = wpnt->mz_header;
|
||
CurrentNEHeader = wpnt->ne_header;
|
||
CurrentNEFile = wpnt->fd;
|
||
|
||
WIN_StackSize = wpnt->ne_header->stack_length;
|
||
WIN_HeapSize = wpnt->ne_header->local_heap_length;
|
||
};
|
||
|
||
/*
|
||
* Create segment selectors.
|
||
*/
|
||
status = lseek(wpnt->fd, wpnt->mz_header->ne_offset +
|
||
wpnt->ne_header->segment_tab_offset,
|
||
SEEK_SET);
|
||
read_size = wpnt->ne_header->n_segment_tab *
|
||
sizeof(struct ne_segment_table_entry_s);
|
||
wpnt->seg_table = (struct ne_segment_table_entry_s *) malloc(read_size);
|
||
if (read(wpnt->fd, wpnt->seg_table, read_size) != read_size)
|
||
myerror("Unable to read segment table header from file");
|
||
wpnt->selector_table = CreateSelectors(wpnt);
|
||
|
||
/* Get the lookup table. This is used for looking up the addresses
|
||
of functions that are exported */
|
||
|
||
read_size = wpnt->ne_header->entry_tab_length;
|
||
wpnt->lookup_table = (char *) malloc(read_size);
|
||
lseek(wpnt->fd, wpnt->mz_header->ne_offset +
|
||
wpnt->ne_header->entry_tab_offset, SEEK_SET);
|
||
if (read(wpnt->fd, wpnt->lookup_table, read_size) != read_size)
|
||
myerror("Unable to read lookup table header from file");
|
||
|
||
/* Get the iname table. This is used for looking up the names
|
||
of functions that are exported */
|
||
|
||
status = lseek(wpnt->fd, wpnt->ne_header->nrname_tab_offset, SEEK_SET);
|
||
read_size = wpnt->ne_header->nrname_tab_length;
|
||
wpnt->nrname_table = (char *) malloc(read_size);
|
||
if (read(wpnt->fd, wpnt->nrname_table, read_size) != read_size)
|
||
myerror("Unable to read nrname table header from file");
|
||
|
||
status = lseek(wpnt->fd, wpnt->mz_header->ne_offset +
|
||
wpnt->ne_header->rname_tab_offset, SEEK_SET);
|
||
read_size = wpnt->ne_header->moduleref_tab_offset -
|
||
wpnt->ne_header->rname_tab_offset;
|
||
wpnt->rname_table = (char *) malloc(read_size);
|
||
if (read(wpnt->fd, wpnt->rname_table, read_size) != read_size)
|
||
myerror("Unable to read rname table header from file");
|
||
|
||
/* Now get the module name */
|
||
|
||
wpnt->name = (char*) malloc(*wpnt->rname_table + 1);
|
||
memcpy(wpnt->name, wpnt->rname_table+1, *wpnt->rname_table);
|
||
wpnt->name[*wpnt->rname_table] = 0;
|
||
|
||
/*
|
||
* Now load any DLLs that this module refers to.
|
||
*/
|
||
for(i=0; i<wpnt->ne_header->n_mod_ref_tab; i++){
|
||
char buff[14];
|
||
char buff2[14];
|
||
int fd, j;
|
||
GetModuleName(wpnt, i + 1, buff);
|
||
|
||
if(FindDLLTable(buff)) continue; /* This module already loaded */
|
||
|
||
/* The next trick is to convert the case, and add the .dll
|
||
* extension if required to find the actual library. We may want
|
||
* to use a search path at some point as well. */
|
||
|
||
/* First try the straight name */
|
||
strcpy(buff2, buff);
|
||
if(fd = open(buff2, O_RDONLY) >= 0) {
|
||
close(fd);
|
||
LoadImage(buff2, buff);
|
||
continue;
|
||
};
|
||
|
||
/* OK, that did not work, try making it lower-case, and add the .dll
|
||
extension */
|
||
|
||
for(j=0; j<strlen(buff2); j++)
|
||
if(buff2[j] >= 'A' && buff2[j] <= 'Z') buff2[j] |= 0x20;
|
||
strcat(buff2, ".dll");
|
||
|
||
if(fd = open(buff2, O_RDONLY) >= 0) {
|
||
close(fd);
|
||
LoadImage(buff2, buff);
|
||
continue;
|
||
};
|
||
|
||
fprintf(stderr,"Unable to load:%s\n", buff);
|
||
};
|
||
}
|
||
|
||
|
||
/**********************************************************************
|
||
* main
|
||
*/
|
||
_WinMain(int argc, char **argv)
|
||
{
|
||
int segment;
|
||
struct w_files * wpnt;
|
||
int cs_reg, ds_reg, ss_reg, ip_reg, sp_reg;
|
||
int i;
|
||
int rv;
|
||
|
||
Argc = argc - 1;
|
||
Argv = argv + 1;
|
||
|
||
if (argc < 2)
|
||
{
|
||
fprintf(stderr, "usage: %s FILENAME\n", argv[0]);
|
||
exit(1);
|
||
}
|
||
|
||
LoadImage(argv[1], NULL);
|
||
|
||
if(ran_out) exit(1);
|
||
#ifdef DEBUG
|
||
GetEntryDLLName("USER", "INITAPP", 0, 0);
|
||
for(i=0; i<1024; i++) {
|
||
int j;
|
||
j = GetEntryPointFromOrdinal(wine_files, i);
|
||
if(j == 0) break;
|
||
fprintf(stderr," %d %x\n", i, j);
|
||
};
|
||
#endif
|
||
/*
|
||
* Fixup references.
|
||
*/
|
||
wpnt = wine_files;
|
||
for(wpnt = wine_files; wpnt; wpnt = wpnt->next)
|
||
for (segment = 0; segment < wpnt->ne_header->n_segment_tab; segment++)
|
||
{
|
||
if (FixupSegment(wpnt, segment) < 0)
|
||
{
|
||
myerror("fixup failed.");
|
||
}
|
||
}
|
||
|
||
/*
|
||
* Fixup stack and jump to start.
|
||
*/
|
||
ds_reg = wine_files->selector_table[wine_files->ne_header->auto_data_seg-1].selector;
|
||
cs_reg = wine_files->selector_table[wine_files->ne_header->cs-1].selector;
|
||
ip_reg = wine_files->ne_header->ip;
|
||
ss_reg = wine_files->selector_table[wine_files->ne_header->ss-1].selector;
|
||
sp_reg = wine_files->ne_header->sp;
|
||
|
||
init_wine_signals();
|
||
|
||
rv = CallToInit16(cs_reg << 16 | ip_reg, ss_reg << 16 | sp_reg, ds_reg);
|
||
printf ("rv = %x\n", rv);
|
||
}
|
||
|
||
|
||
/**********************************************************************
|
||
* GetImportedName
|
||
*/
|
||
char *
|
||
GetImportedName(int fd, struct mz_header_s *mz_header,
|
||
struct ne_header_s *ne_header, int name_offset, char *buffer)
|
||
{
|
||
char *p;
|
||
int length;
|
||
int status;
|
||
int i;
|
||
|
||
status = lseek(fd, mz_header->ne_offset + ne_header->iname_tab_offset +
|
||
name_offset, SEEK_SET);
|
||
length = 0;
|
||
read(fd, &length, 1); /* Get the length byte */
|
||
read(fd, buffer, length);
|
||
buffer[length] = 0;
|
||
return buffer;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* GetModuleName
|
||
*/
|
||
char *
|
||
GetModuleName(struct w_files * wpnt, int index, char *buffer)
|
||
{
|
||
int fd = wpnt->fd;
|
||
struct mz_header_s *mz_header = wpnt->mz_header;
|
||
struct ne_header_s *ne_header = wpnt->ne_header;
|
||
char *p;
|
||
int length;
|
||
int name_offset, status;
|
||
int i;
|
||
|
||
status = lseek(fd, mz_header->ne_offset + ne_header->moduleref_tab_offset +
|
||
2*(index - 1), SEEK_SET);
|
||
name_offset = 0;
|
||
read(fd, &name_offset, 2);
|
||
status = lseek(fd, mz_header->ne_offset + ne_header->iname_tab_offset +
|
||
name_offset, SEEK_SET);
|
||
length = 0;
|
||
read(fd, &length, 1); /* Get the length byte */
|
||
read(fd, buffer, length);
|
||
buffer[length] = 0;
|
||
|
||
/* Module names are always upper case */
|
||
for(i=0; i<length; i++)
|
||
if(buffer[i] >= 'a' && buffer[i] <= 'z') buffer[i] &= ~0x20;
|
||
|
||
return buffer;
|
||
}
|
||
|
||
|
||
/**********************************************************************
|
||
* FixupSegment
|
||
*/
|
||
int
|
||
FixupSegment(struct w_files * wpnt, int segment_num)
|
||
{
|
||
int fd = wpnt->fd;
|
||
struct mz_header_s * mz_header = wpnt->mz_header;
|
||
struct ne_header_s *ne_header = wpnt->ne_header;
|
||
struct ne_segment_table_entry_s *seg_table = wpnt->seg_table;
|
||
struct segment_descriptor_s *selector_table = wpnt->selector_table;
|
||
|
||
struct relocation_entry_s *rep, *rep1;
|
||
struct ne_segment_table_entry_s *seg;
|
||
struct segment_descriptor_s *sel;
|
||
struct dll_table_entry_s *dll_table;
|
||
int status;
|
||
unsigned short *sp;
|
||
unsigned int selector, address;
|
||
unsigned int next_addr;
|
||
int ordinal;
|
||
char dll_name[257];
|
||
char func_name[257];
|
||
int i, n_entries;
|
||
|
||
seg = &seg_table[segment_num];
|
||
sel = &selector_table[segment_num];
|
||
|
||
if ((seg->seg_data_offset == 0) ||
|
||
!(seg->seg_flags & NE_SEGFLAGS_RELOC_DATA))
|
||
return 0;
|
||
|
||
/*
|
||
* Go through the relocation table on entry at a time.
|
||
*/
|
||
i = seg->seg_data_length;
|
||
if (i == 0)
|
||
i = 0x10000;
|
||
|
||
status = lseek(fd, seg->seg_data_offset *
|
||
(1 << ne_header->align_shift_count) + i, SEEK_SET);
|
||
n_entries = 0;
|
||
read(fd, &n_entries, sizeof(short int));
|
||
rep = (struct relocation_entry_s *)
|
||
malloc(n_entries * sizeof(struct relocation_entry_s));
|
||
|
||
if (read(fd,rep, n_entries * sizeof(struct relocation_entry_s)) !=
|
||
n_entries * sizeof(struct relocation_entry_s))
|
||
{
|
||
myerror("Unable to read relocation information");
|
||
}
|
||
|
||
rep1 = rep;
|
||
|
||
for (i = 0; i < n_entries; i++, rep++)
|
||
{
|
||
/*
|
||
* Get the target address corresponding to this entry.
|
||
*/
|
||
switch (rep->relocation_type)
|
||
{
|
||
case NE_RELTYPE_ORDINAL:
|
||
if (GetModuleName(wpnt, rep->target1,
|
||
dll_name) == NULL)
|
||
{
|
||
fprintf(stderr, "NE_RELTYPE_ORDINAL failed");
|
||
return -1;
|
||
}
|
||
|
||
ordinal = rep->target2;
|
||
|
||
status = GetEntryDLLOrdinal(dll_name, ordinal, &selector,
|
||
&address);
|
||
if (status)
|
||
{
|
||
char s[80];
|
||
|
||
sprintf(s, "Bad DLL name '%s.%d'", dll_name, ordinal);
|
||
myerror(s);
|
||
return -1;
|
||
}
|
||
|
||
#ifdef DEBUG_FIXUP
|
||
printf("%d: %s.%d: %04.4x:%04.4x\n", i + 1, dll_name, ordinal,
|
||
selector, address);
|
||
#endif
|
||
break;
|
||
|
||
case NE_RELTYPE_NAME:
|
||
if (GetModuleName(wpnt, rep->target1, dll_name)
|
||
== NULL)
|
||
{
|
||
fprintf(stderr,"NE_RELTYPE_NAME failed");
|
||
return -1;
|
||
}
|
||
|
||
if (GetImportedName(fd, mz_header, ne_header,
|
||
rep->target2, func_name) == NULL)
|
||
{
|
||
fprintf(stderr,"getimportedname failed");
|
||
return -1;
|
||
}
|
||
|
||
status = GetEntryDLLName(dll_name, func_name, &selector,
|
||
&address);
|
||
if (status)
|
||
{
|
||
char s[80];
|
||
|
||
sprintf(s, "Bad DLL name '%s (%s)'", dll_name,func_name);
|
||
myerror(s);
|
||
return -1;
|
||
}
|
||
|
||
#ifdef DEBUG_FIXUP
|
||
printf("%d: %s %s.%d: %04.4x:%04.4x\n", i + 1, func_name,
|
||
dll_name, ordinal, selector, address);
|
||
#endif
|
||
break;
|
||
|
||
case NE_RELTYPE_INTERNAL:
|
||
case NE_RELTYPE_INT1:
|
||
if (rep->target1 == 0x00ff)
|
||
{
|
||
address = GetEntryPointFromOrdinal(wpnt, rep->target2);
|
||
selector = (address >> 16) & 0xffff;
|
||
address &= 0xffff;
|
||
}
|
||
else
|
||
{
|
||
selector = selector_table[rep->target1-1].selector;
|
||
address = rep->target2;
|
||
}
|
||
|
||
#ifdef DEBUG_FIXUP
|
||
printf("%d: %04.4x:%04.4x\n", i + 1, selector, address);
|
||
#endif
|
||
break;
|
||
|
||
case 7:
|
||
/* 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.
|
||
*/
|
||
#ifdef DEBUG_FIXUP
|
||
printf("%d: ADDR TYPE %d, TYPE %d, OFFSET %04.4x, ",
|
||
i + 1, rep->address_type, rep->relocation_type,
|
||
rep->offset);
|
||
printf("TARGET %04.4x %04.4x\n", rep->target1, rep->target2);
|
||
#endif
|
||
continue;
|
||
|
||
default:
|
||
#ifndef DEBUG_FIXUP
|
||
printf("%d: ADDR TYPE %d, TYPE %d, OFFSET %04.4x, ",
|
||
i + 1, rep->address_type, rep->relocation_type,
|
||
rep->offset);
|
||
printf("TARGET %04.4x %04.4x\n", rep->target1, rep->target2);
|
||
#endif
|
||
free(rep1);
|
||
return -1;
|
||
}
|
||
|
||
/*
|
||
* Stuff the right size result in.
|
||
*/
|
||
sp = (unsigned short *) ((char *) sel->base_addr + rep->offset);
|
||
switch (rep->address_type)
|
||
{
|
||
case NE_RADDR_OFFSET16:
|
||
do {
|
||
next_addr = *sp;
|
||
*sp = (unsigned short) address;
|
||
sp = (unsigned short *) ((char *) sel->base_addr + next_addr);
|
||
}
|
||
while (next_addr != 0xffff);
|
||
|
||
break;
|
||
|
||
case NE_RADDR_POINTER32:
|
||
do {
|
||
next_addr = *sp;
|
||
*sp = (unsigned short) address;
|
||
*(sp+1) = (unsigned short) selector;
|
||
sp = (unsigned short *) ((char *) sel->base_addr + next_addr);
|
||
}
|
||
while (next_addr != 0xffff);
|
||
|
||
break;
|
||
|
||
case NE_RADDR_SELECTOR:
|
||
do {
|
||
next_addr = *sp;
|
||
*sp = (unsigned short) selector;
|
||
sp = (unsigned short *) ((char *) sel->base_addr + next_addr);
|
||
}
|
||
while (next_addr != 0xffff);
|
||
|
||
break;
|
||
|
||
default:
|
||
#ifdef DEBUG_FIXUP
|
||
printf("%d: ADDR TYPE %d, TYPE %d, OFFSET %04.4x, ",
|
||
i + 1, rep->address_type, rep->relocation_type,
|
||
rep->offset);
|
||
printf("TARGET %04.4x %04.4x\n", rep->target1, rep->target2);
|
||
#endif
|
||
free(rep1);
|
||
return -1;
|
||
}
|
||
}
|
||
|
||
free(rep1);
|
||
return 0;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* GetProcAddress
|
||
*/
|
||
FARPROC GetProcAddress(HINSTANCE hinstance, char *proc_name)
|
||
{
|
||
if ((int) proc_name & 0xffff0000)
|
||
printf("GetProcAddress: %#04x, '%s'\n", hinstance, proc_name);
|
||
else
|
||
printf("GetProcAddress: %#04x, %d\n", hinstance, (int) proc_name);
|
||
|
||
return NULL;
|
||
}
|