mirror of
git://source.winehq.org/git/wine.git
synced 2024-11-02 13:27:35 +00:00
234bc24db1
Wed Dec 7 14:52:25 1994 Alexandre Julliard (julliard@lamisun.epfl.ch) * [controls/listbox.c] Fixed problems due to new scroll-bar code. * [loader/signal.c] [miscemu/ioports.c] Handle I/O opcodes that use an absolute address. * [objects/text.c] Implemented TabbedTextOut(). Sat Dec 3 18:53:08 1994 Kenneth MacDonald <K.MacDonald@ed.ac.uk> * [objects/metafile.c] Implemented GetMetafile(). Fixed bug in PlayMetaFile() when reading disc based metafile records. Added META_POLYPOLYGON, META_DELETEOBJECT and META_EOF to PlayMetaFileRecord(). Wed Nov 30 06:32:25 1994 Martin von Loewis (martin@cs.csufresno.edu) * [Imakefile] wine.sym: Remove gcc2_compiled and friends * [controls/listbox.c][if1632/relay.c][if1632/relay.c] [loader/resource.c][memory/heap.c][objects/dib.c][windows/dialog.c] Replace #ifdef DEBUG_XXX with if(debugging_xxx){ * [if1632/call.S] CallToLibMain: New function * [if1632/relay.c][include/options.h][misc/main.c] [miscemu/int1a.c][miscemu/int21.c][miscemu/kernel.c] removed Options.relay_debug * [include/heap.h] HEAP_OWNER: Use ds instead of cs:ip * [loader/ne_image.c] LoadNEImage: Remember current exe, handle nodata dlls InitNEDLL: handle nodata dlls, call CallToLibMain * [loader/selector.c] CreateSelectors: Initialize auto_data_sel with 0 * [memory/heap.c] HEAP_CheckHeap: Check prev HEAP_CheckLocalHeaps: new function * [misc/profile] Remember and dump only changed profiles * [tools/makedebug] Introduce debugging_xxx flags Sun Nov 27 23:13:22 MET 1994 <erik@xs4all.nl> * [clipboard.h color.h dc.h dos_fs.h event.h font.h graphics.h if1632.h kernel.h library.h miscemu.h ne_image.h nonclient.h pe_image.h selectors.h wintypes.h] Added. * [*/*] - Commented all 'static char copyright statements', see misc/main.c - moved prototypes to headers files, fixed wrong prototypes. - *please* add a header file for each .c if you need to export things. * [misc/main.c] Added one static string which list the names of the contributors. Fri Nov 25 16:24:27 MET 1994 Dag Asheim (dash@ifi.uio.no) * [Configure] Made the support for multiple languages more automatic. Added a [fonts] section to the wine.conf file. Made the defaults better. Generally cleaned it up. * [rc/sysres_No.rc] [rc/sysres_De.rc] [rc/sysres.c] Norwegian resources and small fixes to the german resources. Wed Nov 23 20:28:59 1994 Martin von Loewis (martin@cs.csufresno.edu) * [debugger/break.c] bark(), toggle_next(), should_continue(): New functions insert_break(): Fixed, adds write access to page before writing wine_bp.next_addr: new structure field * [debugger/dbg.y] Changed symbol's value to be it's value instead of the value pointed to by the symbol. Changed SIGTRAP handling to allow continuation after break point * [misc/shell.c] ShellAbout(): Load resource from memory
479 lines
14 KiB
C
479 lines
14 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 <ctype.h>
|
||
#include <string.h>
|
||
#include <errno.h>
|
||
#include "neexe.h"
|
||
#include "segmem.h"
|
||
#include "dlls.h"
|
||
#include "windows.h"
|
||
#include "arch.h"
|
||
#include "library.h"
|
||
#include "if1632.h"
|
||
#include "selectors.h"
|
||
#include "ne_image.h"
|
||
#include "prototypes.h"
|
||
#include "stddebug.h"
|
||
#include "debug.h"
|
||
|
||
extern unsigned short WIN_StackSize;
|
||
extern unsigned short WIN_HeapSize;
|
||
|
||
void FixupFunctionPrologs(struct w_files *);
|
||
|
||
/**********************************************************************
|
||
* GetImportedName
|
||
*/
|
||
static
|
||
char *NE_GetImportedName(struct w_files *wpnt, int name_offset, char *buffer)
|
||
{
|
||
BYTE length;
|
||
|
||
lseek(wpnt->fd, wpnt->mz_header->ne_offset +
|
||
wpnt->ne->ne_header->iname_tab_offset + name_offset, SEEK_SET);
|
||
read(wpnt->fd, &length, 1); /* Get the length byte */
|
||
read(wpnt->fd, buffer, length);
|
||
buffer[length] = 0;
|
||
|
||
return buffer;
|
||
}
|
||
|
||
struct w_files *current_exe;
|
||
WORD current_nodata=0xfd00;
|
||
/**********************************************************************
|
||
* GetModuleName
|
||
*/
|
||
static char *NE_GetModuleName(struct w_files *wpnt, int index, char *buffer)
|
||
{
|
||
BYTE length;
|
||
WORD name_offset;
|
||
int i;
|
||
|
||
lseek(wpnt->fd, wpnt->mz_header->ne_offset +
|
||
wpnt->ne->ne_header->moduleref_tab_offset + 2 * (index - 1), SEEK_SET);
|
||
read(wpnt->fd, &name_offset, 2);
|
||
name_offset = CONV_SHORT (name_offset);
|
||
|
||
lseek(wpnt->fd, wpnt->mz_header->ne_offset +
|
||
wpnt->ne->ne_header->iname_tab_offset + name_offset, SEEK_SET);
|
||
read(wpnt->fd, &length, 1); /* Get the length byte */
|
||
read(wpnt->fd, buffer, length);
|
||
buffer[length] = 0;
|
||
|
||
/* Module names are always upper case */
|
||
for(i=0; i<length; i++)
|
||
if (islower(buffer[i]))
|
||
buffer[i] = toupper(buffer[i]);
|
||
|
||
return buffer;
|
||
}
|
||
|
||
#ifndef WINELIB
|
||
/**********************************************************************
|
||
* NE_FixupSegment
|
||
*/
|
||
int NE_FixupSegment(struct w_files *wpnt, int segment_num)
|
||
{
|
||
struct segment_descriptor_s *selector_table = wpnt->ne->selector_table;
|
||
struct relocation_entry_s *rep, *rep1;
|
||
struct ne_segment_table_entry_s *seg;
|
||
struct segment_descriptor_s *sel;
|
||
int status, ordinal, i, n_entries, additive;
|
||
unsigned short *sp;
|
||
unsigned int selector, address, next_addr;
|
||
unsigned char dll_name[257], func_name[257];
|
||
|
||
seg = &wpnt->ne->seg_table[segment_num];
|
||
sel = &selector_table[segment_num];
|
||
|
||
dprintf_fixup(stddeb, "Segment fixups for %s, segment %d, selector %x\n",
|
||
wpnt->name, segment_num, (int) sel->base_addr >> 16);
|
||
|
||
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(wpnt->fd, seg->seg_data_offset *
|
||
(1 << wpnt->ne->ne_header->align_shift_count) + i, SEEK_SET);
|
||
n_entries = 0;
|
||
read(wpnt->fd, &n_entries, sizeof(short int));
|
||
rep = (struct relocation_entry_s *)
|
||
malloc(n_entries * sizeof(struct relocation_entry_s));
|
||
|
||
if (read(wpnt->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.
|
||
*/
|
||
additive = 0;
|
||
|
||
switch (rep->relocation_type)
|
||
{
|
||
case NE_RELTYPE_ORDINALADD:
|
||
additive = 1;
|
||
|
||
case NE_RELTYPE_ORDINAL:
|
||
if (NE_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;
|
||
}
|
||
|
||
dprintf_fixup(stddeb,"%d: %s.%d: %04x:%04x\n", i + 1,
|
||
dll_name, ordinal, selector, address);
|
||
break;
|
||
|
||
case NE_RELTYPE_NAMEADD:
|
||
additive = 1;
|
||
|
||
case NE_RELTYPE_NAME:
|
||
if (NE_GetModuleName(wpnt, rep->target1, dll_name) == NULL) {
|
||
fprintf(stderr,"NE_RELTYPE_NAME failed");
|
||
return -1;
|
||
}
|
||
|
||
if (NE_GetImportedName(wpnt, rep->target2, func_name) == NULL) {
|
||
fprintf(stderr,"NE_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;
|
||
}
|
||
/* dprintf_fixup(stddeb,"%d: %s %s.%d: %04x:%04x\n", i + 1,
|
||
func_name, dll_name, ordinal, selector, address);*/
|
||
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;
|
||
}
|
||
|
||
dprintf_fixup(stddeb,"%d: %04x:%04x\n",
|
||
i + 1, selector, address);
|
||
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.
|
||
*/
|
||
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,
|
||
"%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);
|
||
free(rep1);
|
||
return -1;
|
||
}
|
||
|
||
/*
|
||
* Stuff the right size result in.
|
||
*/
|
||
sp = (unsigned short *) ((char *) sel->base_addr + rep->offset);
|
||
if (additive)
|
||
{
|
||
if (FindDLLTable(dll_name) == NULL)
|
||
additive = 2;
|
||
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);
|
||
dprintf_fixup(stddeb, " Additive = %d\n", additive);
|
||
}
|
||
|
||
switch (rep->address_type)
|
||
{
|
||
case NE_RADDR_OFFSET16:
|
||
do {
|
||
dprintf_fixup(stddeb," %04x:%04x:%04x OFFSET16\n",
|
||
(unsigned int) sp >> 16, (int) sp & 0xFFFF, *sp);
|
||
next_addr = *sp;
|
||
*sp = (unsigned short) address;
|
||
if (additive == 2)
|
||
*sp += next_addr;
|
||
sp = (unsigned short *) ((char *) sel->base_addr + next_addr);
|
||
}
|
||
while (next_addr != 0xffff && !additive);
|
||
|
||
break;
|
||
|
||
case NE_RADDR_POINTER32:
|
||
do {
|
||
dprintf_fixup(stddeb," %04x:%04x:%04x POINTER32\n",
|
||
(unsigned int) sp >> 16, (int) sp & 0xFFFF, *sp);
|
||
next_addr = *sp;
|
||
*sp = (unsigned short) address;
|
||
if (additive == 2)
|
||
*sp += next_addr;
|
||
*(sp+1) = (unsigned short) selector;
|
||
sp = (unsigned short *) ((char *) sel->base_addr + next_addr);
|
||
}
|
||
while (next_addr != 0xffff && !additive);
|
||
|
||
break;
|
||
|
||
case NE_RADDR_SELECTOR:
|
||
do {
|
||
dprintf_fixup(stddeb," %04x:%04x:%04x SELECTOR\n",
|
||
(unsigned int) sp >> 16, (int) sp & 0xFFFF, *sp);
|
||
next_addr = *sp;
|
||
*sp = (unsigned short) selector;
|
||
sp = (unsigned short *) ((char *) sel->base_addr + next_addr);
|
||
if (rep->relocation_type == NE_RELTYPE_INT1)
|
||
break;
|
||
}
|
||
while (next_addr != 0xffff && !additive);
|
||
|
||
break;
|
||
|
||
default:
|
||
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);
|
||
free(rep1);
|
||
return -1;
|
||
}
|
||
}
|
||
|
||
free(rep1);
|
||
return 0;
|
||
}
|
||
|
||
int NE_unloadImage(struct w_files *wpnt)
|
||
{
|
||
dprintf_fixup(stdnimp, "NEunloadImage() called!\n");
|
||
/* free resources, image */
|
||
return 1;
|
||
}
|
||
|
||
int NE_StartProgram(struct w_files *wpnt)
|
||
{
|
||
int cs_reg, ds_reg, ss_reg, ip_reg, sp_reg;
|
||
/*
|
||
* Fixup stack and jump to start.
|
||
*/
|
||
WIN_StackSize = wpnt->ne->ne_header->stack_length;
|
||
WIN_HeapSize = wpnt->ne->ne_header->local_heap_length;
|
||
|
||
ds_reg = wpnt->ne->selector_table[wpnt->ne->ne_header->auto_data_seg-1].selector;
|
||
cs_reg = wpnt->ne->selector_table[wpnt->ne->ne_header->cs-1].selector;
|
||
ip_reg = wpnt->ne->ne_header->ip;
|
||
ss_reg = wpnt->ne->selector_table[wpnt->ne->ne_header->ss-1].selector;
|
||
sp_reg = wpnt->ne->ne_header->sp;
|
||
|
||
return CallToInit16(cs_reg << 16 | ip_reg, ss_reg << 16 | sp_reg, ds_reg);
|
||
}
|
||
|
||
void NE_InitDLL(struct w_files *wpnt)
|
||
{
|
||
int cs_reg, ds_reg, ip_reg, cx_reg, di_reg, rv;
|
||
extern struct w_files *current_exe;
|
||
/*
|
||
* Is this a library?
|
||
*/
|
||
if (wpnt->ne->ne_header->format_flags & 0x8000)
|
||
{
|
||
if (!(wpnt->ne->ne_header->format_flags & 0x0001))
|
||
if(wpnt->ne->ne_header->format_flags & NE_FFLAGS_MULTIPLEDATA
|
||
|| wpnt->ne->ne_header->auto_data_seg)
|
||
{
|
||
/* Not SINGLEDATA */
|
||
fprintf(stderr, "Library is not marked SINGLEDATA\n");
|
||
exit(1);
|
||
} else { /* DATA NONE DLL */
|
||
ds_reg = current_exe->ne->selector_table[
|
||
current_exe->ne->ne_header->auto_data_seg-1].selector;
|
||
cx_reg = 0;
|
||
} else { /* DATA SINGLE DLL */
|
||
ds_reg = wpnt->ne->selector_table[wpnt->ne->
|
||
ne_header->auto_data_seg-1].selector;
|
||
cx_reg = wpnt->ne->ne_header->local_heap_length;
|
||
}
|
||
|
||
cs_reg = wpnt->ne->selector_table[wpnt->ne->ne_header->cs-1].selector;
|
||
ip_reg = wpnt->ne->ne_header->ip;
|
||
|
||
di_reg = wpnt->hinstance;
|
||
|
||
if (cs_reg) {
|
||
dprintf_dll(stddeb,"Initializing %s, cs:ip %04x:%04x, ds %04x, cx %04x\n",
|
||
wpnt->name, cs_reg, ip_reg, ds_reg, cx_reg);
|
||
|
||
rv = CallTo16cx(cs_reg << 16 | ip_reg, ds_reg | (cx_reg<<16));
|
||
dprintf_exec(stddeb,"rv = %x\n", rv);
|
||
} else
|
||
dprintf_exec(stddeb,"%s skipped\n", wpnt->name);
|
||
}
|
||
}
|
||
|
||
/**********************************************************************
|
||
* NE_LoadImage
|
||
* Load one NE format executable into memory
|
||
*/
|
||
HINSTANCE NE_LoadImage(struct w_files *wpnt)
|
||
{
|
||
unsigned int read_size, status, segment;
|
||
int i;
|
||
|
||
wpnt->ne = malloc(sizeof(struct ne_data));
|
||
wpnt->ne->resnamtab = NULL;
|
||
wpnt->ne->ne_header = malloc(sizeof(struct ne_header_s));
|
||
|
||
lseek(wpnt->fd, wpnt->mz_header->ne_offset, SEEK_SET);
|
||
if (read(wpnt->fd, wpnt->ne->ne_header, sizeof(struct ne_header_s))
|
||
!= sizeof(struct ne_header_s))
|
||
myerror("Unable to read NE header from file");
|
||
if(!(wpnt->ne->ne_header->format_flags & NE_FFLAGS_LIBMODULE)){
|
||
if(current_exe)printf("Warning: more than one EXE\n");
|
||
current_exe=wpnt;
|
||
}
|
||
|
||
#ifndef WINELIB
|
||
/*
|
||
* Create segment selectors.
|
||
*/
|
||
status = lseek(wpnt->fd, wpnt->mz_header->ne_offset +
|
||
wpnt->ne->ne_header->segment_tab_offset,
|
||
SEEK_SET);
|
||
read_size = wpnt->ne->ne_header->n_segment_tab *
|
||
sizeof(struct ne_segment_table_entry_s);
|
||
wpnt->ne->seg_table = (struct ne_segment_table_entry_s *) malloc(read_size);
|
||
if (read(wpnt->fd, wpnt->ne->seg_table, read_size) != read_size)
|
||
myerror("Unable to read segment table header from file");
|
||
wpnt->ne->selector_table = CreateSelectors(wpnt);
|
||
if(wpnt->ne->ne_header->auto_data_seg==0)
|
||
{
|
||
printf("DATA NONE DLL %s\n",wpnt->name);
|
||
wpnt->hinstance=current_nodata++;
|
||
} else
|
||
wpnt->hinstance = (wpnt->ne->
|
||
selector_table[wpnt->ne->ne_header->auto_data_seg-1].
|
||
selector);
|
||
if (wpnt->hinstance == 0)
|
||
wpnt->hinstance = 0xf000;
|
||
#endif
|
||
/* Get the lookup table. This is used for looking up the addresses
|
||
of functions that are exported */
|
||
|
||
read_size = wpnt->ne->ne_header->entry_tab_length;
|
||
wpnt->ne->lookup_table = (char *) malloc(read_size);
|
||
lseek(wpnt->fd, wpnt->mz_header->ne_offset +
|
||
wpnt->ne->ne_header->entry_tab_offset, SEEK_SET);
|
||
if (read(wpnt->fd, wpnt->ne->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->ne_header->nrname_tab_offset, SEEK_SET);
|
||
read_size = wpnt->ne->ne_header->nrname_tab_length;
|
||
wpnt->ne->nrname_table = (char *) malloc(read_size);
|
||
if (read(wpnt->fd, wpnt->ne->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->ne_header->rname_tab_offset, SEEK_SET);
|
||
read_size = wpnt->ne->ne_header->moduleref_tab_offset -
|
||
wpnt->ne->ne_header->rname_tab_offset;
|
||
wpnt->ne->rname_table = (char *) malloc(read_size);
|
||
if (read(wpnt->fd, wpnt->ne->rname_table, read_size) != read_size)
|
||
myerror("Unable to read rname table header from file");
|
||
|
||
/*
|
||
* Now load any DLLs that this module refers to.
|
||
*/
|
||
for(i=0; i<wpnt->ne->ne_header->n_mod_ref_tab; i++)
|
||
{
|
||
char buff[14];
|
||
NE_GetModuleName(wpnt, i + 1, buff);
|
||
|
||
if (strcasecmp(buff, wpnt->name) != 0 )
|
||
LoadImage(buff, DLL, 0);
|
||
}
|
||
#ifndef WINELIB
|
||
/* fixup references */
|
||
|
||
for (segment = 0; segment < wpnt->ne->ne_header->n_segment_tab; segment++)
|
||
if (NE_FixupSegment(wpnt, segment) < 0)
|
||
myerror("fixup failed.");
|
||
|
||
FixupFunctionPrologs(wpnt);
|
||
InitializeLoadedDLLs(wpnt);
|
||
#endif
|
||
return(wpnt->hinstance);
|
||
}
|
||
|
||
#endif /* !WINELIB */
|