Tue Sep 28 19:59:21 1993 David Metcalfe * [windows/win.c] Implemented support for windows with no borders. Added GetParent(), GetDlgCtrlID(), GetWindowText() and GetWindowTextLength() functions. * [misc/xt.c] Added processing of WM_GETTEXT and WM_GETTEXTLENGTH messages to DefWindowProc and Implemented MessageBeep(). * [windows/syscolor.c] Added preliminary system color support. * [controls/button1.c] Mods to new button control and integration with Wine. Tue Sep 28 19:59:21 1993 Johannes Ruscheinski * [controls/button1.c] New button control using GDI functions. Tue Sep 28 19:59:21 1993 Eric Youngdale * [debugger/*] Added debugging capabilities to Wine Sat Sep 25 13:22:50 1993 Alexandre Julliard (julliard@di.epfl.ch) * [objects/region.c] Bug fix Fri Sep 24 07:35:11 1993 Bob Amstadt (bob at pooh) * [tools/build.c] Changed the entry point code to reduce the standard entry point size from 22 bytes to 10 bytes. This leaves about 4000 free entry points instead of the 800 in version 0.4.2. * [loader/resource.c] Rewrote functions to allow loading of resources from any DLL. * [loader/wine.c] [include/wine.h] Added functions GetFilenameFromInstance() and GetFileInfo() to search for a loaded file based on its instance handle. Added a field in struct w_files to make searching by an instance handle faster. Tue Sep 21 09:57:01 1993 miguel@roxanne.nuclecu.unam.mx (Miguel de Icaza) * [misc/profile.c] Implementation of .INI file handling Mon Sep 20 10:54:32 1993 David Metcalfe * [misc/profile.c.old] Implementation of .INI file handling Mon Sep 20 10:54:32 1993 John Brezak * [controls/WinButton.c] Bug fix with call to XtVaSetValues. Mon Sep 20 10:54:32 1993 Alexandre Julliard * [windows/win.c] Quick patch to get colormaps to work with button widget. Mon Sep 20 02:42:54 1993 (yngvi@hafro.is) * misc/keyboard.c: Ifdefed out some bogus Ansi<->Oem conversion functions * misc/lstr.c: New file with string functions like lstr* IsChar* *Ansi* Wed Sep 15 20:35:10 1993 John Brezak * [loader/signal.c] Additional changes to support NetBSD. Wed Sep 15 22:19:22 1993 Martin Ayotte * [windows/graphics.c] Added FrameRect function Tue Sep 14 13:54:45 1993 Alexandre Julliard * [objects/color.c] [objects/palette.c] Preliminary support for private color map. * [windows/class.c] Implemented CS_CLASSDC style. * [windows/dce.c] Moved DCEs to USER heap. Implemented class and window DCs. * [windows/event.c] Implemented CS_DBLCLKS style. * [windows/graphics.c] Bug fix in SetPixel(). * [windows/win.c] Implemented CS_OWNDC style. Implemented Get/SetWindowLong(). * [controls/menu.c] [windows/class.c] [windows/clipping.c] [windows/dce.c] [windows/message.c] [windows/win.c] Moved windows from global heap to USER heap.
648 lines
16 KiB
C
648 lines
16 KiB
C
static char RCSId[] = "$Id: selector.c,v 1.3 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>
|
||
#ifdef linux
|
||
#include <linux/unistd.h>
|
||
#include <linux/head.h>
|
||
#include <linux/mman.h>
|
||
#include <linux/a.out.h>
|
||
#include <linux/ldt.h>
|
||
#endif
|
||
#ifdef __NetBSD__
|
||
#include <sys/mman.h>
|
||
#endif
|
||
#include <errno.h>
|
||
#include "neexe.h"
|
||
#include "segmem.h"
|
||
#include "prototypes.h"
|
||
#include "wine.h"
|
||
|
||
|
||
#ifdef linux
|
||
#define DEV_ZERO
|
||
#define UTEXTSEL 0x23
|
||
#endif
|
||
|
||
#ifdef __NetBSD__
|
||
#include <machine/segments.h>
|
||
#define PAGE_SIZE getpagesize()
|
||
#define MODIFY_LDT_CONTENTS_DATA 0
|
||
#define MODIFY_LDT_CONTENTS_STACK 1
|
||
#define MODIFY_LDT_CONTENTS_CODE 2
|
||
#define UTEXTSEL 0x1f
|
||
#endif
|
||
|
||
#define MAX_SELECTORS 512
|
||
|
||
static struct segment_descriptor_s * EnvironmentSelector = NULL;
|
||
static struct segment_descriptor_s * PSP_Selector = NULL;
|
||
struct segment_descriptor_s * MakeProcThunks = NULL;
|
||
unsigned short PSPSelector;
|
||
unsigned char ran_out = 0;
|
||
unsigned short SelectorOwners[MAX_SELECTORS];
|
||
|
||
static int next_unused_selector = FIRST_SELECTOR;
|
||
extern void KERNEL_Ordinal_102();
|
||
extern void UNIXLIB_Ordinal_0();
|
||
|
||
extern char **Argv;
|
||
extern int Argc;
|
||
|
||
/**********************************************************************
|
||
* GetNextSegment
|
||
*/
|
||
struct segment_descriptor_s *
|
||
GetNextSegment(unsigned int flags, unsigned int limit)
|
||
{
|
||
struct segment_descriptor_s *selectors, *s;
|
||
int sel_idx;
|
||
#ifdef DEV_ZERO
|
||
FILE *zfile;
|
||
#endif
|
||
|
||
sel_idx = next_unused_selector++;
|
||
|
||
/*
|
||
* Fill in selector info.
|
||
*/
|
||
s = malloc(sizeof(*s));
|
||
s->flags = NE_SEGFLAGS_DATA;
|
||
s->selector = (sel_idx << 3) | 0x0007;
|
||
s->length = limit;
|
||
#ifdef DEV_ZERO
|
||
zfile = fopen("/dev/zero","r");
|
||
s->base_addr = (void *) mmap((char *) (s->selector << 16),
|
||
((s->length + PAGE_SIZE - 1) &
|
||
~(PAGE_SIZE - 1)),
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE, fileno(zfile), 0);
|
||
|
||
fclose(zfile);
|
||
#else
|
||
s->base_addr = (void *) mmap((char *) (s->selector << 16),
|
||
((s->length + PAGE_SIZE - 1) &
|
||
~(PAGE_SIZE - 1)),
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE | MAP_ANON, -1, 0);
|
||
|
||
#endif
|
||
|
||
if (set_ldt_entry(sel_idx, (unsigned long) s->base_addr,
|
||
(s->length - 1) & 0xffff, 0,
|
||
MODIFY_LDT_CONTENTS_DATA, 0, 0) < 0)
|
||
{
|
||
next_unused_selector--;
|
||
free(s);
|
||
return NULL;
|
||
}
|
||
|
||
return s;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* GetEntryPointFromOrdinal
|
||
*/
|
||
union lookup{
|
||
struct entry_tab_header_s *eth;
|
||
struct entry_tab_movable_s *etm;
|
||
struct entry_tab_fixed_s *etf;
|
||
char * cpnt;
|
||
};
|
||
|
||
unsigned int GetEntryDLLName(char * dll_name, char * function, int * sel,
|
||
int * addr)
|
||
{
|
||
struct dll_table_entry_s *dll_table;
|
||
struct w_files * wpnt;
|
||
char * cpnt;
|
||
int ordinal, j, len;
|
||
|
||
dll_table = FindDLLTable(dll_name);
|
||
|
||
if(dll_table) {
|
||
ordinal = FindOrdinalFromName(dll_table, function);
|
||
*sel = dll_table[ordinal].selector;
|
||
*addr = (unsigned int) dll_table[ordinal].address;
|
||
#ifdef WINESTAT
|
||
dll_table[ordinal].used++;
|
||
#endif
|
||
return 0;
|
||
};
|
||
|
||
/* We need a means of determining the ordinal for the function. */
|
||
/* Not a builtin symbol, look to see what the file has for us */
|
||
for(wpnt = wine_files; wpnt; wpnt = wpnt->next){
|
||
if(strcmp(wpnt->name, dll_name)) continue;
|
||
cpnt = wpnt->nrname_table;
|
||
while(1==1){
|
||
if( ((int) cpnt) - ((int)wpnt->nrname_table) >
|
||
wpnt->ne_header->nrname_tab_length) return 1;
|
||
len = *cpnt++;
|
||
if(strncmp(cpnt, function, len) == 0) break;
|
||
cpnt += len + 2;
|
||
};
|
||
ordinal = *((unsigned short *) (cpnt + len));
|
||
j = GetEntryPointFromOrdinal(wpnt, ordinal);
|
||
*addr = j & 0xffff;
|
||
j = j >> 16;
|
||
*sel = wpnt->selector_table[j].selector;
|
||
return 0;
|
||
};
|
||
return 1;
|
||
}
|
||
|
||
unsigned int GetEntryDLLOrdinal(char * dll_name, int ordinal, int * sel,
|
||
int * addr)
|
||
{
|
||
struct dll_table_entry_s *dll_table;
|
||
struct w_files * wpnt;
|
||
int j;
|
||
|
||
dll_table = FindDLLTable(dll_name);
|
||
|
||
if(dll_table) {
|
||
*sel = dll_table[ordinal].selector;
|
||
*addr = (unsigned int) dll_table[ordinal].address;
|
||
#ifdef WINESTAT
|
||
dll_table[ordinal].used++;
|
||
#endif
|
||
return 0;
|
||
};
|
||
|
||
/* Not a builtin symbol, look to see what the file has for us */
|
||
for(wpnt = wine_files; wpnt; wpnt = wpnt->next){
|
||
if(strcmp(wpnt->name, dll_name)) continue;
|
||
j = GetEntryPointFromOrdinal(wpnt, ordinal);
|
||
*addr = j & 0xffff;
|
||
j = j >> 16;
|
||
#if 0
|
||
/* This seems like it would never work */
|
||
*sel = wpnt->selector_table[j].selector;
|
||
#else
|
||
*sel = j; /* Is there any reason this will ever fail?? */
|
||
#endif
|
||
return 0;
|
||
};
|
||
return 1;
|
||
}
|
||
|
||
unsigned int
|
||
GetEntryPointFromOrdinal(struct w_files * wpnt, int ordinal)
|
||
{
|
||
int fd = wpnt->fd;
|
||
struct mz_header_s *mz_header = wpnt->mz_header;
|
||
struct ne_header_s *ne_header = wpnt->ne_header;
|
||
|
||
|
||
union lookup entry_tab_pointer;
|
||
struct entry_tab_header_s *eth;
|
||
struct entry_tab_movable_s *etm;
|
||
struct entry_tab_fixed_s *etf;
|
||
int current_ordinal;
|
||
int i;
|
||
|
||
|
||
entry_tab_pointer.cpnt = wpnt->lookup_table;
|
||
/*
|
||
* Let's walk through the table until we get to our entry.
|
||
*/
|
||
current_ordinal = 1;
|
||
while (1)
|
||
{
|
||
/*
|
||
* Read header for this bundle.
|
||
*/
|
||
eth = entry_tab_pointer.eth++;
|
||
|
||
if (eth->n_entries == 0)
|
||
return 0xffffffff; /* Yikes - we went off the end of the table */
|
||
|
||
if (eth->seg_number == 0)
|
||
{
|
||
current_ordinal += eth->n_entries;
|
||
if(current_ordinal > ordinal) return 0;
|
||
continue;
|
||
}
|
||
|
||
/*
|
||
* Read each of the bundle entries.
|
||
*/
|
||
for (i = 0; i < eth->n_entries; i++, current_ordinal++)
|
||
{
|
||
if (eth->seg_number >= 0xfe)
|
||
{
|
||
etm = entry_tab_pointer.etm++;
|
||
|
||
if (current_ordinal == ordinal)
|
||
{
|
||
return ((unsigned int)
|
||
(wpnt->selector_table[etm->seg_number - 1].base_addr +
|
||
etm->offset));
|
||
}
|
||
}
|
||
else
|
||
{
|
||
etf = entry_tab_pointer.etf++;
|
||
|
||
if (current_ordinal == ordinal)
|
||
{
|
||
return ((unsigned int)
|
||
(wpnt->selector_table[eth->seg_number - 1].base_addr +
|
||
(int) etf->offset[0] +
|
||
((int) etf->offset[1] << 8)));
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/**********************************************************************
|
||
* GetDOSEnvironment
|
||
*/
|
||
void *
|
||
GetDOSEnvironment()
|
||
{
|
||
return EnvironmentSelector->base_addr;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* CreateEnvironment
|
||
*/
|
||
static struct segment_descriptor_s *
|
||
#ifdef DEV_ZERO
|
||
CreateEnvironment(FILE *zfile)
|
||
#else
|
||
CreateEnvironment(void)
|
||
#endif
|
||
{
|
||
char *p;
|
||
int sel_idx;
|
||
struct segment_descriptor_s * s;
|
||
|
||
s = (struct segment_descriptor_s *)
|
||
malloc(sizeof(struct segment_descriptor_s));
|
||
|
||
sel_idx = next_unused_selector++;
|
||
/*
|
||
* Create memory to hold environment.
|
||
*/
|
||
s->flags = NE_SEGFLAGS_DATA;
|
||
s->selector = (sel_idx << 3) | 0x0007;
|
||
s->length = PAGE_SIZE;
|
||
#ifdef DEV_ZERO
|
||
s->base_addr = (void *) mmap((char *) (s->selector << 16),
|
||
PAGE_SIZE,
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE, fileno(zfile), 0);
|
||
#else
|
||
s->base_addr = (void *) mmap((char *) (s->selector << 16),
|
||
PAGE_SIZE,
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE | MAP_ANON, -1, 0);
|
||
#endif
|
||
|
||
/*
|
||
* Fill environment with meaningless babble.
|
||
*/
|
||
p = (char *) s->base_addr;
|
||
strcpy(p, "PATH=C:\\WINDOWS");
|
||
p += strlen(p) + 1;
|
||
*p++ = '\0';
|
||
*p++ = 11;
|
||
*p++ = 0;
|
||
strcpy(p, "C:\\TEST.EXE");
|
||
|
||
/*
|
||
* Create entry in LDT for this segment.
|
||
*/
|
||
if (set_ldt_entry(sel_idx, (unsigned long) s->base_addr,
|
||
(s->length - 1) & 0xffff, 0,
|
||
MODIFY_LDT_CONTENTS_DATA, 0, 0) < 0)
|
||
{
|
||
myerror("Could not create LDT entry for environment");
|
||
}
|
||
return s;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* CreateThunks
|
||
*/
|
||
static struct segment_descriptor_s *
|
||
#ifdef DEV_ZERO
|
||
CreateThunks(FILE *zfile)
|
||
#else
|
||
CreateThunks(void)
|
||
#endif
|
||
{
|
||
int sel_idx;
|
||
struct segment_descriptor_s * s;
|
||
|
||
s = (struct segment_descriptor_s *)
|
||
malloc(sizeof(struct segment_descriptor_s));
|
||
|
||
sel_idx = next_unused_selector++;
|
||
/*
|
||
* Create memory to hold environment.
|
||
*/
|
||
s->flags = 0;
|
||
s->selector = (sel_idx << 3) | 0x0007;
|
||
s->length = 0x10000;
|
||
#ifdef DEV_ZERO
|
||
s->base_addr = (void *) mmap((char *) (s->selector << 16),
|
||
s->length,
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE, fileno(zfile), 0);
|
||
#else
|
||
s->base_addr = (void *) mmap((char *) (s->selector << 16),
|
||
s->length,
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE | MAP_ANON, -1, 0);
|
||
#endif
|
||
|
||
|
||
/*
|
||
* Create entry in LDT for this segment.
|
||
*/
|
||
if (set_ldt_entry(sel_idx, (unsigned long) s->base_addr,
|
||
(s->length - 1) & 0xffff, 0,
|
||
MODIFY_LDT_CONTENTS_CODE, 0, 0) < 0)
|
||
{
|
||
myerror("Could not create LDT entry for thunks");
|
||
}
|
||
return s;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* CreatePSP
|
||
*/
|
||
static struct segment_descriptor_s *
|
||
#ifdef DEV_ZERO
|
||
CreatePSP(FILE *zfile)
|
||
#else
|
||
CreatePSP(void)
|
||
#endif
|
||
{
|
||
struct dos_psp_s *psp;
|
||
unsigned short *usp;
|
||
int sel_idx;
|
||
struct segment_descriptor_s * s;
|
||
char *p1, *p2;
|
||
int i;
|
||
|
||
s = (struct segment_descriptor_s *)
|
||
malloc(sizeof(struct segment_descriptor_s));
|
||
|
||
sel_idx = next_unused_selector++;
|
||
/*
|
||
* Create memory to hold PSP.
|
||
*/
|
||
s->flags = NE_SEGFLAGS_DATA;
|
||
s->selector = (sel_idx << 3) | 0x0007;
|
||
s->length = PAGE_SIZE;
|
||
#ifdef DEV_ZERO
|
||
s->base_addr = (void *) mmap((char *) (s->selector << 16),
|
||
PAGE_SIZE,
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE, fileno(zfile), 0);
|
||
#else
|
||
s->base_addr = (void *) mmap((char *) (s->selector << 16),
|
||
PAGE_SIZE,
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE | MAP_ANON, -1, 0);
|
||
#endif
|
||
|
||
/*
|
||
* Fill PSP
|
||
*/
|
||
PSPSelector = s->selector;
|
||
psp = (struct dos_psp_s *) s->base_addr;
|
||
psp->pspInt20 = 0x20cd;
|
||
psp->pspDispatcher[0] = 0x9a;
|
||
usp = (unsigned short *) &psp->pspDispatcher[1];
|
||
*usp = (unsigned short) KERNEL_Ordinal_102;
|
||
*(usp + 1) = UTEXTSEL;
|
||
psp->pspTerminateVector[0] = (unsigned short) UNIXLIB_Ordinal_0;
|
||
psp->pspTerminateVector[1] = UTEXTSEL;
|
||
psp->pspControlCVector[0] = (unsigned short) UNIXLIB_Ordinal_0;
|
||
psp->pspControlCVector[1] = UTEXTSEL;
|
||
psp->pspCritErrorVector[0] = (unsigned short) UNIXLIB_Ordinal_0;
|
||
psp->pspCritErrorVector[1] = UTEXTSEL;
|
||
psp->pspEnvironment = EnvironmentSelector->selector;
|
||
|
||
p1 = psp->pspCommandTail;
|
||
for (i = 1; i < Argc; i++)
|
||
{
|
||
if ((int) ((int) p1 - (int) psp->pspCommandTail) +
|
||
strlen(Argv[i]) > 124)
|
||
break;
|
||
|
||
for (p2 = Argv[i]; *p2 != '\0'; )
|
||
*p1++ = *p2++;
|
||
|
||
*p1++ = ' ';
|
||
}
|
||
*p1++ = '\r';
|
||
*p1 = '\0';
|
||
psp->pspCommandTailCount = strlen(psp->pspCommandTail);
|
||
|
||
/*
|
||
* Create entry in LDT for this segment.
|
||
*/
|
||
if (set_ldt_entry(sel_idx, (unsigned long) s->base_addr,
|
||
(s->length - 1) & 0xffff, 0,
|
||
MODIFY_LDT_CONTENTS_DATA, 0, 0) < 0)
|
||
{
|
||
myerror("Could not create LDT entry for PSP");
|
||
}
|
||
return s;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* CreateSelectors
|
||
*/
|
||
struct segment_descriptor_s *
|
||
CreateSelectors(struct w_files * wpnt)
|
||
{
|
||
int fd = wpnt->fd;
|
||
struct ne_segment_table_entry_s *seg_table = wpnt->seg_table;
|
||
struct ne_header_s *ne_header = wpnt->ne_header;
|
||
struct segment_descriptor_s *selectors, *s;
|
||
unsigned short *sp;
|
||
int contents, read_only;
|
||
int SelectorTableLength;
|
||
int i;
|
||
int status;
|
||
#ifdef DEV_ZERO
|
||
FILE * zfile;
|
||
#endif
|
||
int old_length, file_image_length;
|
||
|
||
/*
|
||
* Allocate memory for the table to keep track of all selectors.
|
||
*/
|
||
SelectorTableLength = ne_header->n_segment_tab;
|
||
selectors = malloc(SelectorTableLength * sizeof(*selectors));
|
||
if (selectors == NULL)
|
||
return NULL;
|
||
|
||
/*
|
||
* Step through the segment table in the exe header.
|
||
*/
|
||
s = selectors;
|
||
#ifdef DEV_ZERO
|
||
zfile = fopen("/dev/zero","r");
|
||
#endif
|
||
for (i = 0; i < ne_header->n_segment_tab; i++, s++)
|
||
{
|
||
#ifdef DEBUG_SEGMENT
|
||
printf(" %2d: OFFSET %04.4x, LENGTH %04.4x, ",
|
||
i + 1, seg_table[i].seg_data_offset,
|
||
seg_table[i].seg_data_length);
|
||
printf("FLAGS %04.4x, MIN ALLOC %04.4x\n",
|
||
seg_table[i].seg_flags, seg_table[i].min_alloc);
|
||
#endif
|
||
|
||
/*
|
||
* Store the flags in our table.
|
||
*/
|
||
s->flags = seg_table[i].seg_flags;
|
||
s->selector = ((next_unused_selector + i) << 3) | 0x0007;
|
||
|
||
/*
|
||
* Is there an image for this segment in the file?
|
||
*/
|
||
if (seg_table[i].seg_data_offset == 0)
|
||
{
|
||
/*
|
||
* No image in exe file, let's allocate some memory for it.
|
||
*/
|
||
s->length = seg_table[i].min_alloc;
|
||
}
|
||
else
|
||
{
|
||
/*
|
||
* Image in file, let's just point to the image in memory.
|
||
*/
|
||
s->length = seg_table[i].min_alloc;
|
||
file_image_length = seg_table[i].seg_data_length;
|
||
if (file_image_length == 0) file_image_length = 0x10000;
|
||
}
|
||
|
||
if (s->length == 0)
|
||
s->length = 0x10000;
|
||
old_length = s->length;
|
||
|
||
/*
|
||
* If this is the automatic data segment, its size must be adjusted.
|
||
* First we need to check for local heap. Second we nee to see if
|
||
* this is also the stack segment.
|
||
*/
|
||
if (i + 1 == ne_header->auto_data_seg)
|
||
{
|
||
s->length += ne_header->local_heap_length;
|
||
|
||
if (i + 1 == ne_header->ss)
|
||
{
|
||
s->length += ne_header->stack_length;
|
||
ne_header->sp = s->length;
|
||
}
|
||
}
|
||
|
||
/*
|
||
* Is this a DATA or CODE segment?
|
||
*/
|
||
read_only = 0;
|
||
if (s->flags & NE_SEGFLAGS_DATA)
|
||
{
|
||
contents = MODIFY_LDT_CONTENTS_DATA;
|
||
if (s->flags & NE_SEGFLAGS_READONLY)
|
||
read_only = 1;
|
||
}
|
||
else
|
||
{
|
||
contents = MODIFY_LDT_CONTENTS_CODE;
|
||
if (s->flags & NE_SEGFLAGS_EXECUTEONLY)
|
||
read_only = 1;
|
||
}
|
||
#ifdef DEV_ZERO
|
||
s->base_addr =
|
||
(void *) mmap((char *) (s->selector << 16),
|
||
(s->length + PAGE_SIZE - 1) & ~(PAGE_SIZE - 1),
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE, fileno(zfile), 0);
|
||
#else
|
||
s->base_addr =
|
||
(void *) mmap((char *) (s->selector << 16),
|
||
(s->length + PAGE_SIZE - 1) & ~(PAGE_SIZE - 1),
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE | MAP_ANON, -1, 0);
|
||
#endif
|
||
if (seg_table[i].seg_data_offset != 0)
|
||
{
|
||
/*
|
||
* Image in file.
|
||
*/
|
||
status = lseek(fd, seg_table[i].seg_data_offset *
|
||
(1 << ne_header->align_shift_count), SEEK_SET);
|
||
if(read(fd, s->base_addr, file_image_length) != file_image_length)
|
||
myerror("Unable to read segment from file");
|
||
}
|
||
/*
|
||
* Create entry in LDT for this segment.
|
||
*/
|
||
if (set_ldt_entry(s->selector >> 3, (unsigned long) s->base_addr,
|
||
(s->length - 1) & 0xffff, 0,
|
||
contents, read_only, 0) < 0)
|
||
{
|
||
free(selectors);
|
||
fprintf(stderr,"Ran out of ldt entries.\n");
|
||
ran_out++;
|
||
return NULL;
|
||
}
|
||
#ifdef DEBUG_SEGMENT
|
||
printf(" SELECTOR %04.4x, %s\n",
|
||
s->selector,
|
||
contents == MODIFY_LDT_CONTENTS_CODE ? "CODE" : "DATA");
|
||
#endif
|
||
/*
|
||
* If this is the automatic data segment, then we must initialize
|
||
* the local heap.
|
||
*/
|
||
if (i + 1 == ne_header->auto_data_seg)
|
||
{
|
||
HEAP_LocalInit(s->base_addr + old_length,
|
||
ne_header->local_heap_length);
|
||
}
|
||
}
|
||
|
||
sp = &SelectorOwners[next_unused_selector];
|
||
for (i = 0; i < ne_header->n_segment_tab; i++)
|
||
*sp++ = (((next_unused_selector + ne_header->auto_data_seg - 1) << 3)
|
||
| 0x0007);
|
||
|
||
next_unused_selector += ne_header->n_segment_tab;
|
||
|
||
if(!EnvironmentSelector) {
|
||
#ifdef DEV_ZERO
|
||
EnvironmentSelector = CreateEnvironment(zfile);
|
||
PSP_Selector = CreatePSP(zfile);
|
||
MakeProcThunks = CreateThunks(zfile);
|
||
#else
|
||
EnvironmentSelector = CreateEnvironment();
|
||
PSP_Selector = CreatePSP();
|
||
MakeProcThunks = CreateThunks();
|
||
#endif
|
||
};
|
||
|
||
#ifdef DEV_ZERO
|
||
fclose(zfile);
|
||
#endif
|
||
|
||
return selectors;
|
||
}
|