Sun Oct 16 13:29:07 1994 Alexandre Julliard (julliard@lamisun.epfl.ch) * [controls/button.c] Implemented WM_GETDLGCODE, WM_SETFONT, BM_SETSTYLE, BM_SETCHECK, BM_SETSTATE. Implemented default button painting. Optimised redrawing. Fixed owner-draw buttons. * [controls/static.c] Implemented WM_SETFONT. A few optimisations in painting code. Bug fix for SS_SIMPLE controls. * [if1632/callback.c] Preliminary GetCodeHandle(). * [if1632/gdi.spec] Changed 'pascal' to 'pascal16' everywhere it's needed. * [include/windows.h] Fixed a few data structures. * [memory/heap.c] Bug fix in HEAP_ReAlloc(). * [misc/cursor.c] Fixed SetCursor(). * [objects/bitblt.c] Fixed PatBlt() when using BLACKNESS or WHITENESS codes. * [objects/font.c] Better font mapping code. Implemented GetTextFace(). * [objects/region.c] Bug fix in REGION_MakePixmap(). Faster region copying. * [objects/text.c] Implemented ExtTextOut(). Implemented DT_NOCLIP style for DrawText(). * [windows/dc.c] Free the bitmap when deleting a memory DC. * [windows/dce.c] Added support for windows that have no associated X window. Implemented DCX_CLIPSIBLINGS and DCX_PARENTCLIP in GetDCEx(). * [windows/defdlg.c] Implemented default push button handling and DM_SETDEFID. Implemented WM_NEXTDLGCTL. * [windows/dialog.c] Implemented default push button handling. Beginning of a keyboard interface in dialogs (does not really work yet). Fixed dialogs that use a special font. * [windows/event.c] [windows/focus.c] Added support for non-X windows. * [windows/graphics.c] Rewritten FloodFill() and implemented ExtFloodFill(). * [windows/message.c] Cleaner hardware messages and X events handling. * [windows/defwnd.c] [windows/painting.c] Implemented WM_SETREDRAW. * [windows/win.c] Only create an X window for top-level windows, or for the desktop. Child windows now use their parent's drawable. * [windows/winpos.c] Beginning of support for non-X windows (still somewhat broken). Implemented *DeferWindowPos(). * [*/Imakefile] Cleaned up some Imakefiles. Moved dc.c from windows/ to objects/. Moved cursor.c from misc/ to windows/. Sun Oct 16 12:21:52 1994 Michael Patra <micky@marie.physik.tu-berlin.d400.de> * [include/debug.h] [include/stddebug.h] [*/*.c] Rewritten all the calls to printf for displaying debug-information (messages like "LoadImage: loading SOL (SOL.EXE)" etc.) Added option "-debugmsg" to Wine. Example: "-debugmsg +all" will turn all these messages on, "-debugmsg -dll" will turn all messages concerning DLLs off. * [controls/combo.c] Added some handling for combo controls with ownerdraw-styles (just creating and passing the necessary messages to the corresponding listbox control; the edit-control needs to be replaced with something else). * [controls/edit.c] [windows/dialog.c] Added support for use of global heap memory in dialogs with edit controls. * [controls/listbox.c] Added support for item data. ListBoxInsertString(): Fixed bug for elements which are not inserted after the currently last element. * [misc/dos_fs.c] [miscemu/int21.c] DOS_ValidDirectory(): Checks whether a given string is in fact the valid name of a directory. Sat Oct 15 17:35:00 PDT 1994 <martin@cs.csufresno.edu> * [Imakefile] generate wine.sym after creating wine * [debugger/dbg.y] load "wine.sym" when entering debugger * [debugger/info.c] symbolic backtrace for 32-bit stack. Breaks 16-bit bt. Sun Sep 25 12:00:00 PDT 1994 <martin@osiris.cs.csufresno.edu> * [rc/rc.y] [rc/rc.h] [rc/rc.l] [rc/winerc.c] Files created Sun Sep 18 11:04:45 MET DST 1994 Dag Asheim (dash@ifi.uio.no) * [misc/spy.c] Exclude and Include no longer requires a terminating ';' to register the last component. Thu Sep 15 23:10:09 MET DST 1994 Dag Asheim (dash@ifi.uio.no) * [Configure] Rewrote much of it. Added capability to generate a wine.ini file. Commented out the processor emulator options until that becomes interesting. Gives a warning if it sees any *.rej files. I hope I haven't assumed to much about the shell/OS so that it breaks under *BSD. * [misc/dos_fs.c] Removed/changed calls to ToUnix() (which calls tolower()) so that the part of the pathname which correspond with the drive letter on DOS no longer will be mapped to lowercase. This means that it should be possible to have uppercase letters in the [drives] section of wine.ini. * [LICENSE] Cosmetic changes so that it displays better in the window you get from pressing "Credit_License" in the "About WINE" window. Sun Aug 21 21:12:06 MET DST 1994 <erik@xs4all.nl> * [controls/menu.c] LoadMenu() moved to loader/resource.c. * [misc/main.c] Added stub for FileCDR(). * [include/peexe.h] Added, from Eric's pe-test. * [include/resource.h] Added. * [loader/resources.c] Removed duplicated code in *Resource() functions. Moved NE-program specific functions into ne_image.c. Moved NE-fileformat functions to ne_resource.c. * [pe_image.c] [pe_resource.c] Added. Nothing implemented to run PE-executables, resource loading only. * [misc/file.c] Changed OpenFile() to use macros. * [misc/shell.c] Added NULL-ptr checks to ShellAbout(). * [miscemu/int21.c] Fixed a few typos. * [miscemu/kernel.c] Added _DI = _DS, to put the caller's instance in DI. Doesn't work properly if caller changed DS :-(
1092 lines
25 KiB
C
1092 lines
25 KiB
C
#ifndef WINELIB
|
||
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 <string.h>
|
||
#include <sys/types.h>
|
||
#include <sys/stat.h>
|
||
#include <fcntl.h>
|
||
#include <unistd.h>
|
||
#include <errno.h>
|
||
|
||
#ifdef __linux__
|
||
#include <sys/mman.h>
|
||
#include <linux/unistd.h>
|
||
#include <linux/head.h>
|
||
#include <linux/mman.h>
|
||
#include <linux/a.out.h>
|
||
#include <linux/ldt.h>
|
||
#endif
|
||
#if defined(__NetBSD__) || defined(__FreeBSD__)
|
||
#include <sys/mman.h>
|
||
#include <machine/segments.h>
|
||
#endif
|
||
#include "dlls.h"
|
||
#include "neexe.h"
|
||
#include "segmem.h"
|
||
#include "wine.h"
|
||
#include "windows.h"
|
||
#include "prototypes.h"
|
||
#include "debug.h"
|
||
|
||
/* #define DEBUG_SELECTORS /* */
|
||
/* #undef DEBUG_SELECTORS /* */
|
||
|
||
#ifdef linux
|
||
#define DEV_ZERO
|
||
#define UTEXTSEL 0x23
|
||
#endif
|
||
|
||
#if defined(__NetBSD__) || defined(__FreeBSD__)
|
||
#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
|
||
|
||
static SEGDESC * EnvironmentSelector = NULL;
|
||
static SEGDESC * PSP_Selector = NULL;
|
||
SEGDESC * MakeProcThunks = NULL;
|
||
unsigned short PSPSelector;
|
||
unsigned char ran_out = 0;
|
||
int LastUsedSelector = FIRST_SELECTOR - 1;
|
||
|
||
unsigned short SelectorMap[MAX_SELECTORS];
|
||
SEGDESC Segments[MAX_SELECTORS];
|
||
|
||
#ifdef DEV_ZERO
|
||
static FILE *zfile = NULL;
|
||
#endif
|
||
|
||
extern void KERNEL_Ordinal_102();
|
||
extern void UNIXLIB_Ordinal_0();
|
||
extern char WindowsPath[256];
|
||
|
||
extern char **Argv;
|
||
extern int Argc;
|
||
extern char **environ;
|
||
|
||
/**********************************************************************
|
||
* FindUnusedSelectors
|
||
*/
|
||
int
|
||
FindUnusedSelectors(int n_selectors)
|
||
{
|
||
int i;
|
||
int n_found;
|
||
|
||
n_found = 0;
|
||
for (i = LastUsedSelector + 1; i != LastUsedSelector; i++)
|
||
{
|
||
if (i >= MAX_SELECTORS)
|
||
{
|
||
n_found = 0;
|
||
i = FIRST_SELECTOR;
|
||
}
|
||
|
||
if (!SelectorMap[i] && ++n_found == n_selectors)
|
||
break;
|
||
}
|
||
|
||
if (i == LastUsedSelector)
|
||
return 0;
|
||
|
||
LastUsedSelector = i;
|
||
return i - n_selectors + 1;
|
||
}
|
||
|
||
#ifdef HAVE_IPC
|
||
/**********************************************************************
|
||
* IPCCopySelector
|
||
*
|
||
* Created a shared memory copy of a segment:
|
||
*
|
||
* - at a new selector location (if "new" is a 16-bit value)
|
||
* - at an arbitrary memory location (if "new" is a 32-bit value)
|
||
*/
|
||
int
|
||
IPCCopySelector(int i_old, unsigned long new, int swap_type)
|
||
{
|
||
SEGDESC *s_new, *s_old;
|
||
int i_new;
|
||
void *base_addr;
|
||
|
||
s_old = &Segments[i_old];
|
||
|
||
if (new & 0xffff0000)
|
||
{
|
||
/**************************************************************
|
||
* Let's set the address parameter for no segment.
|
||
*/
|
||
i_new = -1;
|
||
s_new = NULL;
|
||
base_addr = (void *) new;
|
||
}
|
||
else
|
||
{
|
||
/***************************************************************
|
||
* We need to fill in the segment descriptor for segment "new".
|
||
*/
|
||
i_new = new;
|
||
s_new = &Segments[i_new];
|
||
|
||
SelectorMap[i_new] = i_new;
|
||
|
||
s_new->selector = (i_new << 3) | 0x0007;
|
||
s_new->base_addr = (void *) ((long) s_new->selector << 16);
|
||
s_new->length = s_old->length;
|
||
s_new->flags = s_old->flags;
|
||
s_new->owner = s_old->owner;
|
||
if (swap_type)
|
||
{
|
||
if (s_old->type == MODIFY_LDT_CONTENTS_DATA)
|
||
s_new->type = MODIFY_LDT_CONTENTS_CODE;
|
||
else
|
||
s_new->type = MODIFY_LDT_CONTENTS_DATA;
|
||
}
|
||
else
|
||
s_new->type = s_old->type;
|
||
|
||
base_addr = s_new->base_addr;
|
||
}
|
||
|
||
/******************************************************************
|
||
* If we don't have a shared memory key for s_old, then we need
|
||
* to get one. In this case, we'll also have to copy the data
|
||
* to protect it.
|
||
*/
|
||
if (s_old->shm_key == -1)
|
||
{
|
||
s_old->shm_key = shmget(IPC_PRIVATE, s_old->length, IPC_CREAT);
|
||
if (s_old->shm_key == -1)
|
||
{
|
||
if (s_new) {
|
||
memset(s_new, 0, sizeof(*s_new));
|
||
s_new->shm_key = -1;
|
||
}
|
||
return -1;
|
||
}
|
||
if (shmat(s_old->shm_key, base_addr, 0) == (char *) -1)
|
||
{
|
||
if (s_new) {
|
||
memset(s_new, 0, sizeof(*s_new));
|
||
s_new->shm_key = -1;
|
||
}
|
||
shmctl(s_old->shm_key, IPC_RMID, NULL);
|
||
return -1;
|
||
}
|
||
memcpy(base_addr, s_old->base_addr, s_old->length);
|
||
munmap(s_old->base_addr,
|
||
((s_old->length + PAGE_SIZE) & ~(PAGE_SIZE - 1)));
|
||
shmat(s_old->shm_key, s_old->base_addr, 0);
|
||
}
|
||
/******************************************************************
|
||
* If have shared memory key s_old, then just attach the new
|
||
* address.
|
||
*/
|
||
else
|
||
{
|
||
if (shmat(s_old->shm_key, base_addr, 0) == (char *) -1)
|
||
{
|
||
if (s_new) {
|
||
memset(s_new, 0, sizeof(*s_new));
|
||
s_new->shm_key = -1;
|
||
}
|
||
return -1;
|
||
}
|
||
}
|
||
|
||
/******************************************************************
|
||
* If we are creating a new segment, then we also need to update
|
||
* the LDT to include the new selector. In this return the
|
||
* new selector.
|
||
*/
|
||
if (s_new)
|
||
{
|
||
s_new->shm_key = s_old->shm_key;
|
||
|
||
if (set_ldt_entry(i_new, (unsigned long) base_addr,
|
||
s_old->length - 1, 0, s_new->type, 0, 0) < 0)
|
||
{
|
||
return -1;
|
||
}
|
||
|
||
return s_new->selector;
|
||
}
|
||
/******************************************************************
|
||
* No new segment. So, just return the shared memory key.
|
||
*/
|
||
else
|
||
return s_old->shm_key;
|
||
}
|
||
#endif
|
||
|
||
/**********************************************************************
|
||
* AllocSelector
|
||
*
|
||
* This is very bad!!! This function is implemented for Windows
|
||
* compatibility only. Do not call this from the emulation library.
|
||
*/
|
||
WORD AllocSelector(WORD old_selector)
|
||
{
|
||
SEGDESC *s_new, *s_old;
|
||
int i_new, i_old;
|
||
int selector;
|
||
|
||
i_new = FindUnusedSelectors(1);
|
||
s_new = &Segments[i_new];
|
||
|
||
if (old_selector)
|
||
{
|
||
i_old = (old_selector >> 3);
|
||
#ifdef HAVE_IPC
|
||
selector = IPCCopySelector(i_old, i_new, 0);
|
||
if (selector < 0)
|
||
return 0;
|
||
else
|
||
return selector;
|
||
#else
|
||
s_old = &Segments[i_old];
|
||
s_new->selector = (i_new << 3) | 0x0007;
|
||
*s_new = *s_old;
|
||
SelectorMap[i_new] = SelectorMap[i_old];
|
||
|
||
if (set_ldt_entry(i_new, s_new->base_addr,
|
||
s_new->length - 1, 0,
|
||
s_new->type, 0, 0) < 0)
|
||
{
|
||
return 0;
|
||
}
|
||
#endif
|
||
}
|
||
else
|
||
{
|
||
memset(s_new, 0, sizeof(*s_new));
|
||
#ifdef HAVE_IPC
|
||
s_new->shm_key = -1;
|
||
#endif
|
||
SelectorMap[i_new] = i_new;
|
||
}
|
||
|
||
return (i_new << 3) | 0x0007;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* PrestoChangoSelector
|
||
*
|
||
* This is very bad!!! This function is implemented for Windows
|
||
* compatibility only. Do not call this from the emulation library.
|
||
*/
|
||
unsigned int PrestoChangoSelector(unsigned src_selector, unsigned dst_selector)
|
||
{
|
||
#ifdef HAVE_IPC
|
||
SEGDESC *src_s;
|
||
int src_idx, dst_idx;
|
||
|
||
src_idx = src_selector >> 3;
|
||
dst_idx = dst_selector >> 3;
|
||
|
||
if (src_idx == dst_idx)
|
||
{
|
||
src_s = &Segments[src_idx];
|
||
|
||
if (src_s->type == MODIFY_LDT_CONTENTS_DATA)
|
||
src_s->type = MODIFY_LDT_CONTENTS_CODE;
|
||
else
|
||
src_s->type = MODIFY_LDT_CONTENTS_DATA;
|
||
|
||
if (set_ldt_entry(src_idx, (long) src_s->base_addr,
|
||
src_s->length - 1, 0, src_s->type, 0, 0) < 0)
|
||
{
|
||
return 0;
|
||
}
|
||
|
||
return src_s->selector;
|
||
}
|
||
else
|
||
{
|
||
return IPCCopySelector(src_idx, dst_idx, 1);
|
||
}
|
||
#else /* HAVE_IPC */
|
||
SEGDESC *src_s, *dst_s;
|
||
char *p;
|
||
int src_idx, dst_idx;
|
||
int alias_count;
|
||
int i;
|
||
|
||
src_idx = (SelectorMap[src_selector >> 3]);
|
||
dst_idx = dst_selector >> 3;
|
||
src_s = &Segments[src_idx];
|
||
dst_s = &Segments[dst_idx];
|
||
|
||
alias_count = 0;
|
||
for (i = FIRST_SELECTOR; i < MAX_SELECTORS; i++)
|
||
if (SelectorMap[i] == src_idx)
|
||
alias_count++;
|
||
|
||
if (src_s->type == MODIFY_LDT_CONTENTS_DATA
|
||
|| alias_count > 1 || src_idx == dst_idx)
|
||
{
|
||
*dst_s = *src_s;
|
||
|
||
if (src_s->type == MODIFY_LDT_CONTENTS_DATA)
|
||
dst_s->type = MODIFY_LDT_CONTENTS_CODE;
|
||
else
|
||
dst_s->type = MODIFY_LDT_CONTENTS_DATA;
|
||
|
||
SelectorMap[dst_idx] = SelectorMap[src_idx];
|
||
if (set_ldt_entry(dst_idx, (long) dst_s->base_addr,
|
||
dst_s->length - 1, 0, dst_s->type, 0, 0) < 0)
|
||
{
|
||
return 0;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
/*
|
||
* We're changing an unaliased code segment into a data
|
||
* segment. The SAFEST (but ugliest) way to deal with
|
||
* this is to map the new segment and copy all the contents.
|
||
*/
|
||
SelectorMap[dst_idx] = dst_idx;
|
||
*dst_s = *src_s;
|
||
dst_s->selector = (dst_idx << 3) | 0x0007;
|
||
dst_s->base_addr = (void *) ((unsigned int) dst_s->selector << 16);
|
||
dst_s->type = MODIFY_LDT_CONTENTS_DATA;
|
||
#ifdef DEV_ZERO
|
||
if (zfile == NULL)
|
||
zfile = fopen("/dev/zero","r");
|
||
p = (void *) mmap((char *) dst_s->base_addr,
|
||
((dst_s->length + PAGE_SIZE)
|
||
& ~(PAGE_SIZE - 1)),
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE, fileno(zfile), 0);
|
||
#else
|
||
p = (void *) mmap((char *) dst_s->base_addr,
|
||
((dst_s->length + PAGE_SIZE)
|
||
& ~(PAGE_SIZE - 1)),
|
||
PROT_EXEC | PROT_READ | PROT_WRITE,
|
||
MAP_FIXED | MAP_PRIVATE | MAP_ANON, -1, 0);
|
||
#endif
|
||
if (p == NULL)
|
||
return 0;
|
||
|
||
memcpy((void *) dst_s->base_addr, (void *) src_s->base_addr,
|
||
dst_s->length);
|
||
if (set_ldt_entry(src_idx, dst_s->base_addr,
|
||
dst_s->length - 1, 0, dst_s->type, 0, 0) < 0)
|
||
{
|
||
return 0;
|
||
}
|
||
if (set_ldt_entry(dst_idx, dst_s->base_addr,
|
||
dst_s->length - 1, 0, dst_s->type, 0, 0) < 0)
|
||
{
|
||
return 0;
|
||
}
|
||
|
||
munmap(src_s->base_addr,
|
||
(src_s->length + PAGE_SIZE) & ~(PAGE_SIZE - 1));
|
||
SelectorMap[src_idx] = dst_idx;
|
||
src_s->base_addr = dst_s->base_addr;
|
||
}
|
||
|
||
return dst_s->selector;
|
||
#endif /* HAVE_IPC */
|
||
}
|
||
|
||
/**********************************************************************
|
||
* AllocCStoDSAlias
|
||
*/
|
||
WORD AllocDStoCSAlias(WORD ds_selector)
|
||
{
|
||
unsigned int cs_selector;
|
||
|
||
if (ds_selector == 0)
|
||
return 0;
|
||
|
||
cs_selector = AllocSelector(0);
|
||
return PrestoChangoSelector(ds_selector, cs_selector);
|
||
}
|
||
|
||
/**********************************************************************
|
||
* CleanupSelectors
|
||
*/
|
||
|
||
void CleanupSelectors(void)
|
||
{
|
||
int sel_idx;
|
||
|
||
for (sel_idx = FIRST_SELECTOR; sel_idx < MAX_SELECTORS; sel_idx++)
|
||
if (SelectorMap[sel_idx])
|
||
FreeSelector((sel_idx << 3) | 7);
|
||
}
|
||
|
||
/**********************************************************************
|
||
* FreeSelector
|
||
*/
|
||
WORD FreeSelector(WORD sel)
|
||
{
|
||
SEGDESC *s;
|
||
int sel_idx;
|
||
int alias_count;
|
||
int i;
|
||
|
||
#ifdef HAVE_IPC
|
||
sel_idx = sel >> 3;
|
||
|
||
if (sel_idx < FIRST_SELECTOR || sel_idx >= MAX_SELECTORS)
|
||
return 0;
|
||
|
||
s = &Segments[sel_idx];
|
||
if (s->shm_key == -1)
|
||
{
|
||
munmap(s->base_addr, ((s->length + PAGE_SIZE) & ~(PAGE_SIZE - 1)));
|
||
memset(s, 0, sizeof(*s));
|
||
s->shm_key = -1;
|
||
SelectorMap[sel_idx] = 0;
|
||
}
|
||
else
|
||
{
|
||
shmdt(s->base_addr);
|
||
|
||
alias_count = 0;
|
||
for (i = FIRST_SELECTOR; i < MAX_SELECTORS; i++)
|
||
if (SelectorMap[i] && Segments[i].shm_key == s->shm_key)
|
||
alias_count++;
|
||
|
||
if (alias_count == 1)
|
||
shmctl(s->shm_key, IPC_RMID, NULL);
|
||
|
||
memset(s, 0, sizeof(*s));
|
||
s->shm_key = -1;
|
||
SelectorMap[sel_idx] = 0;
|
||
}
|
||
|
||
#else /* HAVE_IPC */
|
||
sel_idx = SelectorMap[sel >> 3];
|
||
|
||
if (sel_idx < FIRST_SELECTOR || sel_idx >= MAX_SELECTORS)
|
||
return 0;
|
||
|
||
if (sel_idx != (sel >> 3))
|
||
{
|
||
SelectorMap[sel >> 3] = 0;
|
||
return 0;
|
||
}
|
||
|
||
alias_count = 0;
|
||
for (i = FIRST_SELECTOR; i < MAX_SELECTORS; i++)
|
||
if (SelectorMap[i] == sel_idx)
|
||
alias_count++;
|
||
|
||
if (alias_count == 1)
|
||
{
|
||
s = &Segments[sel_idx];
|
||
munmap(s->base_addr, ((s->length + PAGE_SIZE) & ~(PAGE_SIZE - 1)));
|
||
memset(s, 0, sizeof(*s));
|
||
SelectorMap[sel >> 3] = 0;
|
||
}
|
||
#endif /* HAVE_IPC */
|
||
|
||
return 0;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* CreateNewSegments
|
||
*/
|
||
SEGDESC *
|
||
CreateNewSegments(int code_flag, int read_only, int length, int n_segments)
|
||
{
|
||
SEGDESC *s, *first_segment;
|
||
int contents;
|
||
int i, last_i;
|
||
|
||
i = FindUnusedSelectors(n_segments);
|
||
|
||
#ifdef DEBUG_SELECTORS
|
||
fprintf(stddeb, "Using %d segments starting at index %d.\n", n_segments, i);
|
||
#endif
|
||
|
||
/*
|
||
* Fill in selector info.
|
||
*/
|
||
first_segment = s = &Segments[i];
|
||
for (last_i = i + n_segments; i < last_i; i++, s++)
|
||
{
|
||
if (code_flag)
|
||
{
|
||
contents = MODIFY_LDT_CONTENTS_CODE;
|
||
s->flags = 0;
|
||
}
|
||
else
|
||
{
|
||
contents = MODIFY_LDT_CONTENTS_DATA;
|
||
s->flags = NE_SEGFLAGS_DATA;
|
||
}
|
||
|
||
s->selector = (i << 3) | 0x0007;
|
||
s->length = length;
|
||
#ifdef DEV_ZERO
|
||
if (zfile == NULL)
|
||
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);
|
||
#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
|
||
#ifdef HAVE_IPC
|
||
s->shm_key = -1;
|
||
#endif
|
||
if (set_ldt_entry(i, (unsigned long) s->base_addr,
|
||
(s->length - 1) & 0xffff, 0,
|
||
contents, read_only, 0) < 0)
|
||
{
|
||
memset(s, 0, sizeof(*s));
|
||
#ifdef HAVE_IPC
|
||
s->shm_key = -1;
|
||
#endif
|
||
return NULL;
|
||
}
|
||
|
||
SelectorMap[i] = (unsigned short) i;
|
||
s->type = contents;
|
||
}
|
||
|
||
return first_segment;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* GetNextSegment
|
||
*/
|
||
SEGDESC *
|
||
GetNextSegment(unsigned int flags, unsigned int limit)
|
||
{
|
||
return CreateNewSegments(0, 0, limit, 1);
|
||
}
|
||
|
||
/**********************************************************************
|
||
* 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(strcasecmp(wpnt->name, dll_name)) continue;
|
||
cpnt = wpnt->ne->nrname_table;
|
||
while(1==1){
|
||
if( ((int) cpnt) - ((int)wpnt->ne->nrname_table) >
|
||
wpnt->ne->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 = j;
|
||
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(strcasecmp(wpnt->name, dll_name)) continue;
|
||
j = GetEntryPointFromOrdinal(wpnt, ordinal);
|
||
*addr = j & 0xffff;
|
||
j = j >> 16;
|
||
*sel = j;
|
||
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->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->ne->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->ne->selector_table[etm->seg_number - 1].base_addr +
|
||
etm->offset));
|
||
}
|
||
}
|
||
else
|
||
{
|
||
etf = entry_tab_pointer.etf++;
|
||
|
||
if (current_ordinal == ordinal)
|
||
{
|
||
return ((unsigned int)
|
||
(wpnt->ne->selector_table[eth->seg_number - 1].base_addr +
|
||
(int) etf->offset[0] +
|
||
((int) etf->offset[1] << 8)));
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/**********************************************************************
|
||
* GetDOSEnvironment
|
||
*/
|
||
LPSTR GetDOSEnvironment(void)
|
||
{
|
||
return (LPSTR) EnvironmentSelector->base_addr;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* CreateEnvironment
|
||
*/
|
||
static SEGDESC *
|
||
CreateEnvironment(void)
|
||
{
|
||
char **e;
|
||
char *p;
|
||
SEGDESC * s;
|
||
|
||
s = CreateNewSegments(0, 0, PAGE_SIZE, 1);
|
||
if (s == NULL)
|
||
return NULL;
|
||
|
||
/*
|
||
* Fill environment with Windows path, the Unix environment,
|
||
* and program name.
|
||
*/
|
||
p = (char *) s->base_addr;
|
||
strcpy(p, "PATH=");
|
||
strcat(p, WindowsPath);
|
||
p += strlen(p) + 1;
|
||
|
||
for (e = environ; *e; e++)
|
||
{
|
||
if (strncasecmp(*e, "path", 4))
|
||
{
|
||
strcpy(p, *e);
|
||
p += strlen(p) + 1;
|
||
}
|
||
}
|
||
|
||
*p++ = '\0';
|
||
|
||
/*
|
||
* Display environment
|
||
*/
|
||
#ifdef DEBUG_SELECTORS
|
||
fprintf(stddeb, "Environment at %08.8x\n", s->base_addr);
|
||
for (p = s->base_addr; *p; p += strlen(p) + 1)
|
||
fprintf(stddeb, " %s\n", p);
|
||
#endif
|
||
|
||
return s;
|
||
}
|
||
|
||
/**********************************************************************
|
||
*/
|
||
WORD GetCurrentPDB()
|
||
{
|
||
return PSPSelector;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* CreatePSP
|
||
*/
|
||
static SEGDESC *
|
||
CreatePSP(void)
|
||
{
|
||
struct dos_psp_s *psp;
|
||
unsigned short *usp;
|
||
SEGDESC * s;
|
||
char *p1, *p2;
|
||
int i;
|
||
|
||
s = CreateNewSegments(0, 0, PAGE_SIZE, 1);
|
||
|
||
/*
|
||
* 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;
|
||
|
||
if (i != 1)
|
||
*p1++ = ' ';
|
||
|
||
for (p2 = Argv[i]; *p2 != '\0'; )
|
||
*p1++ = *p2++;
|
||
|
||
}
|
||
*p1 = '\0';
|
||
psp->pspCommandTailCount = strlen(psp->pspCommandTail);
|
||
|
||
return s;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* CreateSelectors
|
||
*/
|
||
SEGDESC *
|
||
CreateSelectors(struct w_files * wpnt)
|
||
{
|
||
int fd = wpnt->fd;
|
||
struct ne_segment_table_entry_s *seg_table = wpnt->ne->seg_table;
|
||
struct ne_header_s *ne_header = wpnt->ne->ne_header;
|
||
SEGDESC *selectors, *s, *stmp;
|
||
unsigned short auto_data_sel;
|
||
int contents, read_only;
|
||
int SelectorTableLength;
|
||
int i;
|
||
int status;
|
||
int old_length, file_image_length;
|
||
int saved_old_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;
|
||
for (i = 0; i < ne_header->n_segment_tab; i++, s++)
|
||
{
|
||
/*
|
||
* Store the flags in our table.
|
||
*/
|
||
s->flags = seg_table[i].seg_flags;
|
||
|
||
/*
|
||
* 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 || i + 1 == ne_header->ss)
|
||
{
|
||
s->length = 0x10000;
|
||
ne_header->sp = s->length - 2;
|
||
}
|
||
|
||
/*
|
||
* 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;
|
||
}
|
||
|
||
#if 0
|
||
stmp = CreateNewSegments(!(s->flags & NE_SEGFLAGS_DATA), read_only,
|
||
s->length, 1);
|
||
s->base_addr = stmp->base_addr;
|
||
s->selector = stmp->selector;
|
||
#endif
|
||
s->selector = GlobalAlloc(GMEM_FIXED, s->length);
|
||
if (s->selector == 0)
|
||
myerror("CreateSelectors: GlobalAlloc() failed");
|
||
|
||
s->base_addr = (void *) ((LONG) s->selector << 16);
|
||
#ifdef HAVE_IPC
|
||
s->shm_key = -1;
|
||
#endif
|
||
if (!(s->flags & NE_SEGFLAGS_DATA))
|
||
PrestoChangoSelector(s->selector, s->selector);
|
||
else
|
||
memset(s->base_addr, 0, s->length);
|
||
|
||
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");
|
||
}
|
||
|
||
/*
|
||
* If this is the automatic data segment, then we must initialize
|
||
* the local heap.
|
||
*/
|
||
if (i + 1 == ne_header->auto_data_seg)
|
||
{
|
||
auto_data_sel = s->selector;
|
||
saved_old_length = old_length;
|
||
}
|
||
}
|
||
|
||
s = selectors;
|
||
for (i = 0; i < ne_header->n_segment_tab; i++, s++)
|
||
{
|
||
Segments[s->selector >> 3].owner = auto_data_sel;
|
||
if (s->selector == auto_data_sel)
|
||
HEAP_LocalInit(auto_data_sel, s->base_addr + saved_old_length,
|
||
0x10000 - 2 - saved_old_length
|
||
- ne_header->stack_length);
|
||
}
|
||
|
||
if(!EnvironmentSelector) {
|
||
EnvironmentSelector = CreateEnvironment();
|
||
PSP_Selector = CreatePSP();
|
||
MakeProcThunks = CreateNewSegments(1, 0, 0x10000, 1);
|
||
};
|
||
|
||
return selectors;
|
||
}
|
||
/**********************************************************************
|
||
*/
|
||
void
|
||
FixupFunctionPrologs(struct w_files * wpnt)
|
||
{
|
||
struct mz_header_s *mz_header = wpnt->mz_header;
|
||
struct ne_header_s *ne_header = wpnt->ne->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;
|
||
unsigned char *fixup_ptr;
|
||
int i;
|
||
|
||
if (!(ne_header->format_flags & 0x0001))
|
||
return;
|
||
|
||
entry_tab_pointer.cpnt = wpnt->ne->lookup_table;
|
||
/*
|
||
* Let's walk through the table and fixup prologs as we go.
|
||
*/
|
||
while (1)
|
||
{
|
||
/* Get bundle header */
|
||
eth = entry_tab_pointer.eth++;
|
||
|
||
/* Check for end of table */
|
||
if (eth->n_entries == 0)
|
||
return;
|
||
|
||
/* Check for empty bundle */
|
||
if (eth->seg_number == 0)
|
||
continue;
|
||
|
||
/* Examine each bundle */
|
||
for (i = 0; i < eth->n_entries; i++)
|
||
{
|
||
/* Moveable segment */
|
||
if (eth->seg_number >= 0xfe)
|
||
{
|
||
etm = entry_tab_pointer.etm++;
|
||
fixup_ptr = (wpnt->ne->selector_table[etm->seg_number-1].base_addr
|
||
+ etm->offset);
|
||
}
|
||
else
|
||
{
|
||
etf = entry_tab_pointer.etf++;
|
||
fixup_ptr = (wpnt->ne->selector_table[eth->seg_number-1].base_addr
|
||
+ (int) etf->offset[0]
|
||
+ ((int) etf->offset[1] << 8));
|
||
|
||
}
|
||
|
||
/* Verify the signature */
|
||
if (((fixup_ptr[0] == 0x1e && fixup_ptr[1] == 0x58)
|
||
|| (fixup_ptr[0] == 0x8c && fixup_ptr[1] == 0xd8))
|
||
&& fixup_ptr[2] == 0x90)
|
||
{
|
||
fixup_ptr[0] = 0xb8; /* MOV AX, */
|
||
fixup_ptr[1] = wpnt->hinstance;
|
||
fixup_ptr[2] = (wpnt->hinstance >> 8);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/***********************************************************************
|
||
* GetSelectorBase (KERNEL.186)
|
||
*/
|
||
DWORD GetSelectorBase(WORD wSelector)
|
||
{
|
||
fprintf(stdnimp, "GetSelectorBase(selector %4X) stub!\n", wSelector);
|
||
}
|
||
|
||
/***********************************************************************
|
||
* SetSelectorBase (KERNEL.187)
|
||
*/
|
||
void SetSelectorBase(WORD wSelector, DWORD dwBase)
|
||
{
|
||
fprintf(stdnimp, "SetSelectorBase(selector %4X, base %8X) stub!\n",
|
||
wSelector, dwBase);
|
||
}
|
||
|
||
/***********************************************************************
|
||
* GetSelectorLimit (KERNEL.188)
|
||
*/
|
||
DWORD GetSelectorLimit(WORD wSelector)
|
||
{
|
||
fprintf(stdnimp, "GetSelectorLimit(selector %4X) stub!\n", wSelector);
|
||
|
||
return 0xffff;
|
||
}
|
||
|
||
/***********************************************************************
|
||
* SetSelectorLimit (KERNEL.189)
|
||
*/
|
||
void SetSelectorLimit(WORD wSelector, DWORD dwLimit)
|
||
{
|
||
fprintf(stdnimp, "SetSelectorLimit(selector %4X, base %8X) stub!\n",
|
||
wSelector, dwLimit);
|
||
}
|
||
|
||
#endif /* ifndef WINELIB */
|