Implemented DOS INT21 AH=4B (EXEC).

diff --git a/include/dosexe.h b/include/dosexe.h
index 6188c66..d35d008 100644
--- a/include/dosexe.h
+++ b/include/dosexe.h
@@ -14,7 +14,7 @@
 struct _DOSEVENT;
 
 typedef struct _DOSTASK {
- WORD psp_seg;
+ WORD psp_seg, retval;
  WORD dpmi_flag;
 } DOSTASK, *LPDOSTASK;
 
@@ -31,6 +31,8 @@
 #define V86_FLAG 0x00020000
 
 extern BOOL MZ_LoadImage( HMODULE module, HANDLE hFile, LPCSTR filename );
+extern BOOL MZ_Exec( CONTEXT86 *context, LPCSTR filename, BYTE func, LPVOID paramblk );
+extern void MZ_Exit( CONTEXT86 *context, BOOL cs_psp, WORD retval );
 extern LPDOSTASK MZ_Current( void );
 extern LPDOSTASK MZ_AllocDPMITask( void );
 extern int DOSVM_Enter( CONTEXT86 *context );
diff --git a/include/task.h b/include/task.h
index 8750a32..cb666eb 100644
--- a/include/task.h
+++ b/include/task.h
@@ -26,7 +26,7 @@
     WORD      parentPSP;               /* 16 Selector of parent PSP */
     BYTE      fileHandles[20];         /* 18 Open file handles */
     HANDLE16  environment;             /* 2c Selector of environment */
-    WORD      reserved2[2];
+    DWORD     saveStack;               /* 2e SS:SP on last int21 call */
     WORD      nbFiles;                 /* 32 Number of file handles */
     SEGPTR    fileHandlesPtr;          /* 34 Pointer to file handle table */
     HANDLE16  hFileHandles;            /* 38 Handle to fileHandlesPtr */
diff --git a/loader/dos/module.c b/loader/dos/module.c
index 3636cb4..3e55afa 100644
--- a/loader/dos/module.c
+++ b/loader/dos/module.c
@@ -55,6 +55,26 @@
 #define SEG16(ptr,seg) ((LPVOID)((BYTE*)ptr+((DWORD)(seg)<<4)))
 #define SEGPTR16(ptr,segptr) ((LPVOID)((BYTE*)ptr+((DWORD)SELECTOROF(segptr)<<4)+OFFSETOF(segptr)))
 
+/* structures for EXEC */
+
+typedef struct {
+  WORD env_seg;
+  DWORD cmdline WINE_PACKED;
+  DWORD fcb1 WINE_PACKED;
+  DWORD fcb2 WINE_PACKED;
+  WORD init_sp;
+  WORD init_ss;
+  WORD init_ip;
+  WORD init_cs;
+} ExecBlock;
+
+typedef struct {
+  WORD load_seg;
+  WORD rel_seg;
+} OverlayBlock;
+
+/* global variables */
+
 static WORD init_cs,init_ip,init_ss,init_sp;
 static char mm_name[128];
 
@@ -66,15 +86,20 @@
 static BOOL MZ_InitTask(void);
 static void MZ_KillTask(void);
 
-static void MZ_CreatePSP( LPVOID lpPSP, WORD env )
+static void MZ_CreatePSP( LPVOID lpPSP, WORD env, WORD par )
 {
- PDB16*psp=lpPSP;
+  PDB16*psp=lpPSP;
 
- psp->int20=0x20CD; /* int 20 */
- /* some programs use this to calculate how much memory they need */
- psp->nextParagraph=0x9FFF;
- psp->environment=env;
- /* FIXME: more PSP stuff */
+  psp->int20=0x20CD; /* int 20 */
+  /* some programs use this to calculate how much memory they need */
+  psp->nextParagraph=0x9FFF; /* FIXME: use a real value */
+  /* FIXME: dispatcher */
+  psp->savedint22 = INT_GetRMHandler(0x22);
+  psp->savedint23 = INT_GetRMHandler(0x23);
+  psp->savedint24 = INT_GetRMHandler(0x24);
+  psp->parentPSP=par;
+  psp->environment=env;
+  /* FIXME: more PSP stuff */
 }
 
 static void MZ_FillPSP( LPVOID lpPSP, LPCSTR cmdline )
@@ -178,25 +203,30 @@
     return TRUE;
 }
 
-BOOL MZ_LoadImage( HMODULE module, HANDLE hFile, LPCSTR filename )
+BOOL MZ_DoLoadImage( HMODULE module, HANDLE hFile, LPCSTR filename, OverlayBlock *oblk )
 {
   LPDOSTASK lpDosTask = dos_current;
-  IMAGE_NT_HEADERS *win_hdr = PE_HEADER(module);
   IMAGE_DOS_HEADER mz_header;
   DWORD image_start,image_size,min_size,max_size,avail;
-  BYTE*psp_start,*load_start;
-  int x, old_com=0, alloc=0;
+  BYTE*psp_start,*load_start,*oldenv;
+  int x, old_com=0, alloc;
   SEGPTR reloc;
-  WORD env_seg, load_seg;
+  WORD env_seg, load_seg, rel_seg, oldpsp_seg;
   DWORD len;
 
-  win_hdr->OptionalHeader.Subsystem = IMAGE_SUBSYSTEM_WINDOWS_CUI;
-  win_hdr->OptionalHeader.AddressOfEntryPoint = (LPBYTE)MZ_Launch - (LPBYTE)module;
-
-  if (!lpDosTask) {
+  if (lpDosTask) {
+    /* DOS process already running, inherit from it */
+    PDB16* par_psp = (PDB16*)((DWORD)lpDosTask->psp_seg << 4);
+    alloc=0;
+    oldenv = (LPBYTE)((DWORD)par_psp->environment << 4);
+    oldpsp_seg = lpDosTask->psp_seg;
+  } else {
+    /* allocate new DOS process, inheriting from Wine environment */
     alloc=1;
     lpDosTask = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(DOSTASK));
     dos_current = lpDosTask;
+    oldenv = GetEnvironmentStringsA();
+    oldpsp_seg = 0;
   }
 
  SetFilePointer(hFile,0,NULL,FILE_BEGIN);
@@ -220,29 +250,37 @@
   max_size=image_size+((DWORD)mz_header.e_maxalloc<<4)+(PSP_SIZE<<4);
  }
 
- MZ_InitMemory();
+  if (alloc) MZ_InitMemory();
 
- /* allocate environment block */
- env_seg=MZ_InitEnvironment(GetEnvironmentStringsA(),filename);
+  if (oblk) {
+    /* load overlay into preallocated memory */
+    load_seg=oblk->load_seg;
+    rel_seg=oblk->rel_seg;
+    load_start=(LPBYTE)((DWORD)load_seg<<4);
+  } else {
+    /* allocate environment block */
+    env_seg=MZ_InitEnvironment(oldenv, filename);
 
- /* allocate memory for the executable */
- TRACE("Allocating DOS memory (min=%ld, max=%ld)\n",min_size,max_size);
- avail=DOSMEM_Available();
- if (avail<min_size) {
-  ERR("insufficient DOS memory\n");
-  SetLastError(ERROR_NOT_ENOUGH_MEMORY);
-  goto load_error;
- }
- if (avail>max_size) avail=max_size;
- psp_start=DOSMEM_GetBlock(avail,&lpDosTask->psp_seg);
- if (!psp_start) {
-  ERR("error allocating DOS memory\n");
-  SetLastError(ERROR_NOT_ENOUGH_MEMORY);
-  goto load_error;
- }
- load_seg=lpDosTask->psp_seg+(old_com?0:PSP_SIZE);
- load_start=psp_start+(PSP_SIZE<<4);
- MZ_CreatePSP(psp_start, env_seg);
+    /* allocate memory for the executable */
+    TRACE("Allocating DOS memory (min=%ld, max=%ld)\n",min_size,max_size);
+    avail=DOSMEM_Available();
+    if (avail<min_size) {
+      ERR("insufficient DOS memory\n");
+      SetLastError(ERROR_NOT_ENOUGH_MEMORY);
+      goto load_error;
+    }
+    if (avail>max_size) avail=max_size;
+    psp_start=DOSMEM_GetBlock(avail,&lpDosTask->psp_seg);
+    if (!psp_start) {
+      ERR("error allocating DOS memory\n");
+      SetLastError(ERROR_NOT_ENOUGH_MEMORY);
+      goto load_error;
+    }
+    load_seg=lpDosTask->psp_seg+(old_com?0:PSP_SIZE);
+    rel_seg=load_seg;
+    load_start=psp_start+(PSP_SIZE<<4);
+    MZ_CreatePSP(psp_start, env_seg, oldpsp_seg);
+  }
 
  /* load executable image */
  TRACE("loading DOS %s image, %08lx bytes\n",old_com?"COM":"EXE",image_size);
@@ -262,18 +300,20 @@
     SetLastError(ERROR_BAD_FORMAT);
     goto load_error;
    }
-   *(WORD*)SEGPTR16(load_start,reloc)+=load_seg;
+   *(WORD*)SEGPTR16(load_start,reloc)+=rel_seg;
   }
  }
 
- init_cs = load_seg+mz_header.e_cs;
- init_ip = mz_header.e_ip;
- init_ss = load_seg+mz_header.e_ss;
- init_sp = mz_header.e_sp;
+  if (!oblk) {
+    init_cs = load_seg+mz_header.e_cs;
+    init_ip = mz_header.e_ip;
+    init_ss = load_seg+mz_header.e_ss;
+    init_sp = mz_header.e_sp;
 
-  TRACE("entry point: %04x:%04x\n",init_cs,init_ip);
+    TRACE("entry point: %04x:%04x\n",init_cs,init_ip);
+  }
 
-  if (!MZ_InitTask()) {
+  if (alloc && !MZ_InitTask()) {
     MZ_KillTask();
     SetLastError(ERROR_GEN_FAILURE);
     return FALSE;
@@ -282,6 +322,7 @@
   return TRUE;
 
 load_error:
+  lpDosTask->psp_seg = oldpsp_seg;
   if (alloc) {
     dos_current = NULL;
     if (mm_name[0]!=0) unlink(mm_name);
@@ -290,6 +331,80 @@
   return FALSE;
 }
 
+BOOL MZ_LoadImage( HMODULE module, HANDLE hFile, LPCSTR filename )
+{
+  IMAGE_NT_HEADERS *win_hdr = PE_HEADER(module);
+
+  if (win_hdr) {
+    win_hdr->OptionalHeader.Subsystem = IMAGE_SUBSYSTEM_WINDOWS_CUI;
+    win_hdr->OptionalHeader.AddressOfEntryPoint = (LPBYTE)MZ_Launch - (LPBYTE)module;
+  }
+
+  return MZ_DoLoadImage( module, hFile, filename, NULL );
+}
+
+BOOL MZ_Exec( CONTEXT86 *context, LPCSTR filename, BYTE func, LPVOID paramblk )
+{
+  /* this may only be called from existing DOS processes
+   * (i.e. one DOS app spawning another) */
+  /* FIXME: do we want to check binary type first, to check
+   * whether it's a NE/PE executable? */
+  LPDOSTASK lpDosTask = MZ_Current();
+  HFILE hFile = CreateFileA( filename, GENERIC_READ, FILE_SHARE_READ,
+			     NULL, OPEN_EXISTING, 0, -1);
+  BOOL ret = FALSE;
+  if (hFile == INVALID_HANDLE_VALUE) return FALSE;
+  switch (func) {
+  case 0: /* load and execute */
+  case 1: /* load but don't execute */
+    {
+      /* save current process's return SS:SP now */
+      LPBYTE psp_start = (LPBYTE)((DWORD)lpDosTask->psp_seg << 4);
+      PDB16 *psp = (PDB16 *)psp_start;
+      psp->saveStack = (DWORD)PTR_SEG_OFF_TO_SEGPTR(context->SegSs, LOWORD(context->Esp));
+    }
+    ret = MZ_DoLoadImage( NULL, hFile, filename, NULL );
+    if (ret) {
+      /* MZ_LoadImage created a new PSP and loaded new values into lpDosTask,
+       * let's work on the new values now */
+      LPBYTE psp_start = (LPBYTE)((DWORD)lpDosTask->psp_seg << 4);
+      ExecBlock *blk = (ExecBlock *)paramblk;
+      MZ_FillPSP(psp_start, DOSMEM_MapRealToLinear(blk->cmdline));
+      /* the lame MS-DOS engineers decided that the return address should be in int22 */
+      INT_SetRMHandler(0x22, (FARPROC16)PTR_SEG_OFF_TO_SEGPTR(context->SegCs, LOWORD(context->Eip)));
+      if (func) {
+	/* don't execute, just return startup state */
+	blk->init_cs = init_cs;
+	blk->init_ip = init_ip;
+	blk->init_ss = init_ss;
+	blk->init_sp = init_sp;
+      } else {
+	/* execute by making us return to new process */
+	context->SegCs = init_cs;
+	context->Eip   = init_ip;
+	context->SegSs = init_ss;
+	context->Esp   = init_sp;
+	context->SegDs = lpDosTask->psp_seg;
+	context->SegEs = lpDosTask->psp_seg;
+	context->Eax   = 0;
+      }
+    }
+    break;
+  case 3: /* load overlay */
+    {
+      OverlayBlock *blk = (OverlayBlock *)paramblk;
+      ret = MZ_DoLoadImage( NULL, hFile, filename, blk );
+    }
+    break;
+  default:
+    FIXME("EXEC load type %d not implemented\n", func);
+    SetLastError(ERROR_INVALID_FUNCTION);
+    break;
+  }
+  CloseHandle(hFile);
+  return ret;
+}
+
 LPDOSTASK MZ_AllocDPMITask( void )
 {
   LPDOSTASK lpDosTask = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(DOSTASK));
@@ -435,21 +550,70 @@
   kill(dosmod_pid,SIGTERM);
 }
 
+void MZ_Exit( CONTEXT86 *context, BOOL cs_psp, WORD retval )
+{
+  LPDOSTASK lpDosTask = MZ_Current();
+  if (lpDosTask) {
+    WORD psp_seg = cs_psp ? context->SegCs : lpDosTask->psp_seg;
+    LPBYTE psp_start = (LPBYTE)((DWORD)psp_seg << 4);
+    PDB16 *psp = (PDB16 *)psp_start;
+    WORD parpsp = psp->parentPSP; /* check for parent DOS process */
+    if (parpsp) {
+      /* retrieve parent's return address */
+      FARPROC16 retaddr = INT_GetRMHandler(0x22);
+      /* restore interrupts */
+      INT_SetRMHandler(0x22, psp->savedint22);
+      INT_SetRMHandler(0x23, psp->savedint23);
+      INT_SetRMHandler(0x24, psp->savedint24);
+      /* FIXME: deallocate file handles etc */
+      /* free process's associated memory
+       * FIXME: walk memory and deallocate all blocks owned by process */
+      DOSMEM_FreeBlock(DOSMEM_MapRealToLinear(MAKELONG(0,psp->environment)));
+      DOSMEM_FreeBlock(DOSMEM_MapRealToLinear(MAKELONG(0,lpDosTask->psp_seg)));
+      /* switch to parent's PSP */
+      lpDosTask->psp_seg = parpsp;
+      psp_start = (LPBYTE)((DWORD)parpsp << 4);
+      psp = (PDB16 *)psp_start;
+      /* now return to parent */
+      lpDosTask->retval = retval;
+      context->SegCs = SELECTOROF(retaddr);
+      context->Eip   = OFFSETOF(retaddr);
+      context->SegSs = SELECTOROF(psp->saveStack);
+      context->Esp   = OFFSETOF(psp->saveStack);
+      return;
+    } else
+      MZ_KillTask();
+  }
+  ExitThread( retval );
+}
+
 #else /* !MZ_SUPPORTED */
 
 BOOL MZ_LoadImage( HMODULE module, HANDLE hFile, LPCSTR filename )
 {
- WARN("DOS executables not supported on this architecture\n");
- SetLastError(ERROR_BAD_FORMAT);
- return FALSE;
+  WARN("DOS executables not supported on this platform\n");
+  SetLastError(ERROR_BAD_FORMAT);
+  return FALSE;
+}
+
+BOOL MZ_Exec( CONTEXT86 *context, LPCSTR filename, BYTE func, LPVOID paramblk )
+{
+  /* can't happen */
+  SetLastError(ERROR_BAD_FORMAT);
+  return FALSE;
 }
 
 LPDOSTASK MZ_AllocDPMITask( void )
 {
-    ERR("Actual real-mode calls not supported on this architecture!\n");
+    ERR("Actual real-mode calls not supported on this platform!\n");
     return NULL;
 }
 
+void MZ_Exit( CONTEXT86 *context, BOOL cs_psp, WORD retval )
+{
+  ExitThread( retval );
+}
+
 #endif /* !MZ_SUPPORTED */
 
 LPDOSTASK MZ_Current( void )
diff --git a/msdos/int20.c b/msdos/int20.c
index 232ad65..9cc7e63 100644
--- a/msdos/int20.c
+++ b/msdos/int20.c
@@ -8,6 +8,7 @@
 /* #define DEBUG_INT */
 #include "debugtools.h"
 #include "task.h"
+#include "dosexe.h"
 
 /**********************************************************************
  *	    INT_Int20Handler
@@ -16,5 +17,5 @@
  */
 void WINAPI INT_Int20Handler( CONTEXT86 *context )
 {
-        ExitThread( 0 );
+        MZ_Exit( context, TRUE, 0 );
 }
diff --git a/msdos/int21.c b/msdos/int21.c
index acfbce0..b1a7d73 100644
--- a/msdos/int21.c
+++ b/msdos/int21.c
@@ -954,7 +954,7 @@
         ERR("Cannot change PSP for non-DOS task!\n");
 }
     
-static WORD INT21_GetCurrentPSP()
+static WORD INT21_GetCurrentPSP(void)
 {
     LPDOSTASK lpDosTask = MZ_Current();
     if (lpDosTask)
@@ -963,6 +963,15 @@
         return GetCurrentPDB16();
 }
 
+static WORD INT21_GetReturnCode(void)
+{
+    LPDOSTASK lpDosTask = MZ_Current();
+    if (lpDosTask) {
+        WORD ret = lpDosTask->retval;
+        lpDosTask->retval = 0;
+        return ret;
+    } else return 0;
+}
 
 /***********************************************************************
  *           INT21_GetExtendedError
@@ -1136,7 +1145,7 @@
 
     case 0x00: /* TERMINATE PROGRAM */
         TRACE("TERMINATE PROGRAM\n");
-        ExitThread( 0 );
+        MZ_Exit( context, FALSE, 0 );
         break;
 
     case 0x01: /* READ CHARACTER FROM STANDARD INPUT, WITH ECHO */
@@ -1835,21 +1844,27 @@
 
     case 0x4b: /* "EXEC" - LOAD AND/OR EXECUTE PROGRAM */
         TRACE("EXEC %s\n",
-	      (LPCSTR)CTX_SEG_OFF_TO_LIN(context,  context->SegDs,context->Edx ));
-        AX_reg(context) = WinExec16( CTX_SEG_OFF_TO_LIN(context,  context->SegDs,
-                                                         context->Edx ),
-                                     SW_NORMAL );
-        if (AX_reg(context) < 32) SET_CFLAG(context);
+	      (LPCSTR)CTX_SEG_OFF_TO_LIN(context, context->SegDs, context->Edx ));
+	if (ISV86(context)) {
+            if (!MZ_Exec( context, CTX_SEG_OFF_TO_LIN(context, context->SegDs, context->Edx),
+                         AL_reg(context), CTX_SEG_OFF_TO_LIN(context, context->SegEs, context->Ebx) ))
+                bSetDOSExtendedError = TRUE;
+	} else {
+            AX_reg(context) = WinExec16( CTX_SEG_OFF_TO_LIN(context,  context->SegDs,
+                                                             context->Edx ),
+                                         SW_NORMAL );
+            if (AX_reg(context) < 32) SET_CFLAG(context);
+        }
         break;		
 	
     case 0x4c: /* "EXIT" - TERMINATE WITH RETURN CODE */
         TRACE("EXIT with return code %d\n",AL_reg(context));
-        ExitThread( AL_reg(context) );
+        MZ_Exit( context, FALSE, AL_reg(context) );
         break;
 
     case 0x4d: /* GET RETURN CODE */
         TRACE("GET RETURN CODE (ERRORLEVEL)\n");
-        AX_reg(context) = 0; /* normal exit */
+        AX_reg(context) = INT21_GetReturnCode();
         break;
 
     case 0x4e: /* "FINDFIRST" - FIND FIRST MATCHING FILE */