Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 1 | /* |
| 2 | * VxD emulation |
| 3 | * |
| 4 | * Copyright 1995 Anand Kumria |
Alexandre Julliard | 0799c1a | 2002-03-09 23:29:33 +0000 | [diff] [blame] | 5 | * |
| 6 | * This library is free software; you can redistribute it and/or |
| 7 | * modify it under the terms of the GNU Lesser General Public |
| 8 | * License as published by the Free Software Foundation; either |
| 9 | * version 2.1 of the License, or (at your option) any later version. |
| 10 | * |
| 11 | * This library is distributed in the hope that it will be useful, |
| 12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 14 | * Lesser General Public License for more details. |
| 15 | * |
| 16 | * You should have received a copy of the GNU Lesser General Public |
| 17 | * License along with this library; if not, write to the Free Software |
| 18 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 19 | */ |
| 20 | |
Patrik Stridvall | d016f81 | 2002-08-17 00:43:16 +0000 | [diff] [blame] | 21 | #include "config.h" |
Patrik Stridvall | 51e6c0c | 2002-08-31 19:04:14 +0000 | [diff] [blame] | 22 | #include "wine/port.h" |
Patrik Stridvall | d016f81 | 2002-08-17 00:43:16 +0000 | [diff] [blame] | 23 | |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 24 | #include <fcntl.h> |
| 25 | #include <memory.h> |
Patrik Stridvall | 51e6c0c | 2002-08-31 19:04:14 +0000 | [diff] [blame] | 26 | #include <stdio.h> |
Jeremy White | d3e22d9 | 2000-02-10 19:03:02 +0000 | [diff] [blame] | 27 | #include <sys/types.h> |
Patrik Stridvall | d016f81 | 2002-08-17 00:43:16 +0000 | [diff] [blame] | 28 | #ifdef HAVE_UNISTD_H |
| 29 | # include <unistd.h> |
| 30 | #endif |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 31 | #include "winbase.h" |
Jeremy White | d3e22d9 | 2000-02-10 19:03:02 +0000 | [diff] [blame] | 32 | #include "windef.h" |
Patrik Stridvall | 9c1de6d | 2002-09-12 22:07:02 +0000 | [diff] [blame] | 33 | #include "winternl.h" |
Jeremy White | d3e22d9 | 2000-02-10 19:03:02 +0000 | [diff] [blame] | 34 | #include "wingdi.h" |
Marcus Meissner | 219cfd8 | 1999-02-24 13:05:13 +0000 | [diff] [blame] | 35 | #include "winuser.h" |
| 36 | #include "wine/winbase16.h" |
| 37 | #include "wine/winuser16.h" |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 38 | #include "msdos.h" |
| 39 | #include "miscemu.h" |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 40 | #include "module.h" |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 41 | #include "selectors.h" |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 42 | #include "task.h" |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 43 | #include "file.h" |
Alexandre Julliard | 0799c1a | 2002-03-09 23:29:33 +0000 | [diff] [blame] | 44 | #include "wine/debug.h" |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 45 | |
Alexandre Julliard | 0799c1a | 2002-03-09 23:29:33 +0000 | [diff] [blame] | 46 | WINE_DEFAULT_DEBUG_CHANNEL(vxd); |
Patrik Stridvall | b4b9fae | 1999-04-19 14:56:29 +0000 | [diff] [blame] | 47 | |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 48 | |
Alexandre Julliard | 139a4b1 | 1996-11-02 14:24:07 +0000 | [diff] [blame] | 49 | #define VXD_BARF(context,name) \ |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 50 | DPRINTF( "vxd %s: unknown/not implemented parameters:\n" \ |
Alexandre Julliard | 139a4b1 | 1996-11-02 14:24:07 +0000 | [diff] [blame] | 51 | "vxd %s: AX %04x, BX %04x, CX %04x, DX %04x, " \ |
| 52 | "SI %04x, DI %04x, DS %04x, ES %04x\n", \ |
| 53 | (name), (name), AX_reg(context), BX_reg(context), \ |
| 54 | CX_reg(context), DX_reg(context), SI_reg(context), \ |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 55 | DI_reg(context), (WORD)context->SegDs, (WORD)context->SegEs ) |
Alexandre Julliard | 139a4b1 | 1996-11-02 14:24:07 +0000 | [diff] [blame] | 56 | |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 57 | UINT W32S_offset = 0; |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 58 | |
| 59 | static WORD VXD_WinVersion(void) |
| 60 | { |
Alexandre Julliard | 3db94ef | 1997-09-28 17:43:24 +0000 | [diff] [blame] | 61 | WORD version = LOWORD(GetVersion16()); |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 62 | return (version >> 8) | (version << 8); |
| 63 | } |
| 64 | |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 65 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 66 | * VXD_VMM (WPROCS.401) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 67 | */ |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 68 | void WINAPI VXD_VMM ( CONTEXT86 *context ) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 69 | { |
| 70 | unsigned service = AX_reg(context); |
| 71 | |
Alexandre Julliard | 908464d | 2000-11-01 03:11:12 +0000 | [diff] [blame] | 72 | TRACE("[%04x] VMM\n", (UINT16)service); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 73 | |
| 74 | switch(service) |
| 75 | { |
| 76 | case 0x0000: /* version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 77 | SET_AX( context, VXD_WinVersion() ); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 78 | RESET_CFLAG(context); |
| 79 | break; |
| 80 | |
Ulrich Weigand | ff1c569 | 1998-10-14 18:01:08 +0000 | [diff] [blame] | 81 | case 0x026d: /* Get_Debug_Flag '/m' */ |
| 82 | case 0x026e: /* Get_Debug_Flag '/n' */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 83 | SET_AL( context, 0 ); |
Ulrich Weigand | ff1c569 | 1998-10-14 18:01:08 +0000 | [diff] [blame] | 84 | RESET_CFLAG(context); |
| 85 | break; |
| 86 | |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 87 | default: |
| 88 | VXD_BARF( context, "VMM" ); |
| 89 | } |
| 90 | } |
| 91 | |
| 92 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 93 | * VXD_PageFile (WPROCS.433) |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 94 | */ |
Alexandre Julliard | 617955d | 1999-06-26 18:40:24 +0000 | [diff] [blame] | 95 | void WINAPI VXD_PageFile( CONTEXT86 *context ) |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 96 | { |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 97 | unsigned service = AX_reg(context); |
| 98 | |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 99 | /* taken from Ralf Brown's Interrupt List */ |
| 100 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 101 | TRACE("[%04x] PageFile\n", (UINT16)service ); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 102 | |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 103 | switch(service) |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 104 | { |
| 105 | case 0x00: /* get version, is this windows version? */ |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 106 | TRACE("returning version\n"); |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 107 | SET_AX( context, VXD_WinVersion() ); |
Alexandre Julliard | ca22b33 | 1996-07-12 19:02:39 +0000 | [diff] [blame] | 108 | RESET_CFLAG(context); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 109 | break; |
| 110 | |
| 111 | case 0x01: /* get swap file info */ |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 112 | TRACE("VxD PageFile: returning swap file info\n"); |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 113 | SET_AX( context, 0x00 ); /* paging disabled */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 114 | context->Ecx = 0; /* maximum size of paging file */ |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 115 | /* FIXME: do I touch DS:SI or DS:DI? */ |
Alexandre Julliard | ca22b33 | 1996-07-12 19:02:39 +0000 | [diff] [blame] | 116 | RESET_CFLAG(context); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 117 | break; |
| 118 | |
| 119 | case 0x02: /* delete permanent swap on exit */ |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 120 | TRACE("VxD PageFile: supposed to delete swap\n"); |
Alexandre Julliard | ca22b33 | 1996-07-12 19:02:39 +0000 | [diff] [blame] | 121 | RESET_CFLAG(context); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 122 | break; |
| 123 | |
| 124 | case 0x03: /* current temporary swap file size */ |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 125 | TRACE("VxD PageFile: what is current temp. swap size\n"); |
Alexandre Julliard | ca22b33 | 1996-07-12 19:02:39 +0000 | [diff] [blame] | 126 | RESET_CFLAG(context); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 127 | break; |
| 128 | |
| 129 | case 0x04: /* read or write?? INTERRUP.D */ |
| 130 | case 0x05: /* cancel?? INTERRUP.D */ |
| 131 | case 0x06: /* test I/O valid INTERRUP.D */ |
| 132 | default: |
Alexandre Julliard | 139a4b1 | 1996-11-02 14:24:07 +0000 | [diff] [blame] | 133 | VXD_BARF( context, "pagefile" ); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 134 | break; |
| 135 | } |
| 136 | } |
| 137 | |
Ulrich Weigand | 0b597bc | 1998-10-11 19:01:33 +0000 | [diff] [blame] | 138 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 139 | * VXD_Reboot (WPROCS.409) |
Ulrich Weigand | 0b597bc | 1998-10-11 19:01:33 +0000 | [diff] [blame] | 140 | */ |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 141 | void WINAPI VXD_Reboot ( CONTEXT86 *context ) |
Ulrich Weigand | 0b597bc | 1998-10-11 19:01:33 +0000 | [diff] [blame] | 142 | { |
| 143 | unsigned service = AX_reg(context); |
| 144 | |
Alexandre Julliard | 908464d | 2000-11-01 03:11:12 +0000 | [diff] [blame] | 145 | TRACE("[%04x] Reboot\n", (UINT16)service); |
Ulrich Weigand | 0b597bc | 1998-10-11 19:01:33 +0000 | [diff] [blame] | 146 | |
| 147 | switch(service) |
| 148 | { |
| 149 | case 0x0000: /* version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 150 | SET_AX( context, VXD_WinVersion() ); |
Ulrich Weigand | 0b597bc | 1998-10-11 19:01:33 +0000 | [diff] [blame] | 151 | RESET_CFLAG(context); |
| 152 | break; |
| 153 | |
| 154 | default: |
| 155 | VXD_BARF( context, "REBOOT" ); |
| 156 | } |
| 157 | } |
| 158 | |
| 159 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 160 | * VXD_VDD (WPROCS.410) |
Ulrich Weigand | 0b597bc | 1998-10-11 19:01:33 +0000 | [diff] [blame] | 161 | */ |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 162 | void WINAPI VXD_VDD ( CONTEXT86 *context ) |
Ulrich Weigand | 0b597bc | 1998-10-11 19:01:33 +0000 | [diff] [blame] | 163 | { |
| 164 | unsigned service = AX_reg(context); |
| 165 | |
Alexandre Julliard | 908464d | 2000-11-01 03:11:12 +0000 | [diff] [blame] | 166 | TRACE("[%04x] VDD\n", (UINT16)service); |
Ulrich Weigand | 0b597bc | 1998-10-11 19:01:33 +0000 | [diff] [blame] | 167 | |
| 168 | switch(service) |
| 169 | { |
| 170 | case 0x0000: /* version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 171 | SET_AX( context, VXD_WinVersion() ); |
Ulrich Weigand | 0b597bc | 1998-10-11 19:01:33 +0000 | [diff] [blame] | 172 | RESET_CFLAG(context); |
| 173 | break; |
| 174 | |
| 175 | default: |
| 176 | VXD_BARF( context, "VDD" ); |
| 177 | } |
| 178 | } |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 179 | |
| 180 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 181 | * VXD_VMD (WPROCS.412) |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 182 | */ |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 183 | void WINAPI VXD_VMD ( CONTEXT86 *context ) |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 184 | { |
| 185 | unsigned service = AX_reg(context); |
| 186 | |
Alexandre Julliard | 908464d | 2000-11-01 03:11:12 +0000 | [diff] [blame] | 187 | TRACE("[%04x] VMD\n", (UINT16)service); |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 188 | |
| 189 | switch(service) |
| 190 | { |
| 191 | case 0x0000: /* version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 192 | SET_AX( context, VXD_WinVersion() ); |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 193 | RESET_CFLAG(context); |
| 194 | break; |
| 195 | |
| 196 | default: |
| 197 | VXD_BARF( context, "VMD" ); |
| 198 | } |
| 199 | } |
| 200 | |
| 201 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 202 | * VXD_VXDLoader (WPROCS.439) |
James Abbatiello | 5f150ba | 2000-08-28 19:29:26 +0000 | [diff] [blame] | 203 | */ |
| 204 | void WINAPI VXD_VXDLoader( CONTEXT86 *context ) |
| 205 | { |
| 206 | unsigned service = AX_reg(context); |
| 207 | |
| 208 | TRACE("[%04x] VXDLoader\n", (UINT16)service); |
| 209 | |
| 210 | switch (service) |
| 211 | { |
| 212 | case 0x0000: /* get version */ |
| 213 | TRACE("returning version\n"); |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 214 | SET_AX( context, 0x0000 ); |
| 215 | SET_DX( context, VXD_WinVersion() ); |
James Abbatiello | 5f150ba | 2000-08-28 19:29:26 +0000 | [diff] [blame] | 216 | RESET_CFLAG(context); |
| 217 | break; |
| 218 | |
| 219 | case 0x0001: /* load device */ |
| 220 | FIXME("load device %04lx:%04x (%s)\n", |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 221 | context->SegDs, DX_reg(context), |
Alexandre Julliard | 982a223 | 2000-12-13 20:20:09 +0000 | [diff] [blame] | 222 | debugstr_a(MapSL(MAKESEGPTR(context->SegDs, DX_reg(context))))); |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 223 | SET_AX( context, 0x0000 ); |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 224 | context->SegEs = 0x0000; |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 225 | SET_DI( context, 0x0000 ); |
James Abbatiello | 5f150ba | 2000-08-28 19:29:26 +0000 | [diff] [blame] | 226 | RESET_CFLAG(context); |
| 227 | break; |
| 228 | |
| 229 | case 0x0002: /* unload device */ |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 230 | FIXME("unload device (%08lx)\n", context->Ebx); |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 231 | SET_AX( context, 0x0000 ); |
James Abbatiello | 5f150ba | 2000-08-28 19:29:26 +0000 | [diff] [blame] | 232 | RESET_CFLAG(context); |
| 233 | break; |
| 234 | |
| 235 | default: |
| 236 | VXD_BARF( context, "VXDLDR" ); |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 237 | SET_AX( context, 0x000B ); /* invalid function number */ |
James Abbatiello | 5f150ba | 2000-08-28 19:29:26 +0000 | [diff] [blame] | 238 | SET_CFLAG(context); |
| 239 | break; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 240 | } |
James Abbatiello | 5f150ba | 2000-08-28 19:29:26 +0000 | [diff] [blame] | 241 | } |
| 242 | |
| 243 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 244 | * VXD_Shell (WPROCS.423) |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 245 | */ |
Alexandre Julliard | 617955d | 1999-06-26 18:40:24 +0000 | [diff] [blame] | 246 | void WINAPI VXD_Shell( CONTEXT86 *context ) |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 247 | { |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 248 | unsigned service = DX_reg(context); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 249 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 250 | TRACE("[%04x] Shell\n", (UINT16)service); |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 251 | |
| 252 | switch (service) /* Ralf Brown says EDX, but I use DX instead */ |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 253 | { |
| 254 | case 0x0000: |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 255 | TRACE("returning version\n"); |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 256 | SET_AX( context, VXD_WinVersion() ); |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 257 | context->Ebx = 1; /* system VM Handle */ |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 258 | break; |
| 259 | |
| 260 | case 0x0001: |
| 261 | case 0x0002: |
| 262 | case 0x0003: |
Juergen Schmied | f08b819 | 1999-05-22 10:33:50 +0000 | [diff] [blame] | 263 | /* SHELL_SYSMODAL_Message |
| 264 | ebx virtual maschine handle |
| 265 | eax message box flags |
| 266 | ecx address of message |
| 267 | edi address of caption |
| 268 | return response in eax |
| 269 | */ |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 270 | case 0x0004: |
Juergen Schmied | f08b819 | 1999-05-22 10:33:50 +0000 | [diff] [blame] | 271 | /* SHELL_Message |
| 272 | ebx virtual maschine handle |
| 273 | eax message box flags |
| 274 | ecx address of message |
| 275 | edi address of caption |
| 276 | esi address callback |
| 277 | edx reference data for callback |
| 278 | return response in eax |
| 279 | */ |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 280 | case 0x0005: |
Alexandre Julliard | 139a4b1 | 1996-11-02 14:24:07 +0000 | [diff] [blame] | 281 | VXD_BARF( context, "shell" ); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 282 | break; |
| 283 | |
| 284 | case 0x0006: /* SHELL_Get_VM_State */ |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 285 | TRACE("VxD Shell: returning VM state\n"); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 286 | /* Actually we don't, not yet. We have to return a structure |
| 287 | * and I am not to sure how to set it up and return it yet, |
| 288 | * so for now let's do nothing. I can (hopefully) get this |
| 289 | * by the next release |
| 290 | */ |
Alexandre Julliard | ca22b33 | 1996-07-12 19:02:39 +0000 | [diff] [blame] | 291 | /* RESET_CFLAG(context); */ |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 292 | break; |
| 293 | |
| 294 | case 0x0007: |
| 295 | case 0x0008: |
| 296 | case 0x0009: |
| 297 | case 0x000A: |
| 298 | case 0x000B: |
| 299 | case 0x000C: |
| 300 | case 0x000D: |
| 301 | case 0x000E: |
| 302 | case 0x000F: |
| 303 | case 0x0010: |
| 304 | case 0x0011: |
| 305 | case 0x0012: |
| 306 | case 0x0013: |
| 307 | case 0x0014: |
| 308 | case 0x0015: |
| 309 | case 0x0016: |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 310 | VXD_BARF( context, "SHELL" ); |
| 311 | break; |
| 312 | |
| 313 | /* the new Win95 shell API */ |
| 314 | case 0x0100: /* get version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 315 | SET_AX( context, VXD_WinVersion() ); |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 316 | break; |
| 317 | |
| 318 | case 0x0104: /* retrieve Hook_Properties list */ |
| 319 | case 0x0105: /* call Hook_Properties callbacks */ |
| 320 | VXD_BARF( context, "SHELL" ); |
| 321 | break; |
| 322 | |
| 323 | case 0x0106: /* install timeout callback */ |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 324 | TRACE("VxD Shell: ignoring shell callback (%ld sec.)\n", context->Ebx); |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 325 | SET_CFLAG(context); |
| 326 | break; |
| 327 | |
| 328 | case 0x0107: /* get version of any VxD */ |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 329 | default: |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 330 | VXD_BARF( context, "SHELL" ); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 331 | break; |
| 332 | } |
| 333 | } |
| 334 | |
| 335 | |
| 336 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 337 | * VXD_Comm (WPROCS.414) |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 338 | */ |
Alexandre Julliard | 617955d | 1999-06-26 18:40:24 +0000 | [diff] [blame] | 339 | void WINAPI VXD_Comm( CONTEXT86 *context ) |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 340 | { |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 341 | unsigned service = AX_reg(context); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 342 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 343 | TRACE("[%04x] Comm\n", (UINT16)service); |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 344 | |
| 345 | switch (service) |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 346 | { |
| 347 | case 0x0000: /* get version */ |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 348 | TRACE("returning version\n"); |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 349 | SET_AX( context, VXD_WinVersion() ); |
Alexandre Julliard | ca22b33 | 1996-07-12 19:02:39 +0000 | [diff] [blame] | 350 | RESET_CFLAG(context); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 351 | break; |
| 352 | |
| 353 | case 0x0001: /* set port global */ |
| 354 | case 0x0002: /* get focus */ |
| 355 | case 0x0003: /* virtualise port */ |
| 356 | default: |
Alexandre Julliard | 139a4b1 | 1996-11-02 14:24:07 +0000 | [diff] [blame] | 357 | VXD_BARF( context, "comm" ); |
Alexandre Julliard | af0bae5 | 1995-10-03 17:06:08 +0000 | [diff] [blame] | 358 | } |
| 359 | } |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 360 | |
| 361 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 362 | * VXD_Timer (WPROCS.405) |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 363 | */ |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 364 | void WINAPI VXD_Timer( CONTEXT86 *context ) |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 365 | { |
| 366 | unsigned service = AX_reg(context); |
| 367 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 368 | TRACE("[%04x] Virtual Timer\n", (UINT16)service); |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 369 | |
| 370 | switch(service) |
| 371 | { |
| 372 | case 0x0000: /* version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 373 | SET_AX( context, VXD_WinVersion() ); |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 374 | RESET_CFLAG(context); |
| 375 | break; |
| 376 | |
| 377 | case 0x0100: /* clock tick time, in 840nsecs */ |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 378 | context->Eax = GetTickCount(); |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 379 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 380 | context->Edx = context->Eax >> 22; |
| 381 | context->Eax <<= 10; /* not very precise */ |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 382 | break; |
| 383 | |
| 384 | case 0x0101: /* current Windows time, msecs */ |
| 385 | case 0x0102: /* current VM time, msecs */ |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 386 | context->Eax = GetTickCount(); |
Alexandre Julliard | 670cdc4 | 1997-08-24 16:00:30 +0000 | [diff] [blame] | 387 | break; |
| 388 | |
| 389 | default: |
| 390 | VXD_BARF( context, "VTD" ); |
| 391 | } |
| 392 | } |
| 393 | |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 394 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 395 | * VXD_TimerAPI (WPROCS.1490) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 396 | */ |
Ulrich Weigand | 1cd0794 | 1998-12-10 11:01:09 +0000 | [diff] [blame] | 397 | static DWORD System_Time = 0; |
| 398 | static WORD System_Time_Selector = 0; |
| 399 | static void System_Time_Tick( WORD timer ) { System_Time += 55; } |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 400 | void WINAPI VXD_TimerAPI ( CONTEXT86 *context ) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 401 | { |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 402 | unsigned service = AX_reg(context); |
| 403 | |
Alexandre Julliard | 908464d | 2000-11-01 03:11:12 +0000 | [diff] [blame] | 404 | TRACE("[%04x] TimerAPI\n", (UINT16)service); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 405 | |
| 406 | switch(service) |
| 407 | { |
| 408 | case 0x0000: /* version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 409 | SET_AX( context, VXD_WinVersion() ); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 410 | RESET_CFLAG(context); |
| 411 | break; |
| 412 | |
| 413 | case 0x0009: /* get system time selector */ |
Ulrich Weigand | 1cd0794 | 1998-12-10 11:01:09 +0000 | [diff] [blame] | 414 | if ( !System_Time_Selector ) |
| 415 | { |
Alexandre Julliard | 914406f | 2000-11-14 01:54:49 +0000 | [diff] [blame] | 416 | System_Time_Selector = SELECTOR_AllocBlock( &System_Time, sizeof(DWORD), WINE_LDT_FLAGS_DATA ); |
Ulrich Weigand | 1cd0794 | 1998-12-10 11:01:09 +0000 | [diff] [blame] | 417 | CreateSystemTimer( 55, System_Time_Tick ); |
| 418 | } |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 419 | |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 420 | SET_AX( context, System_Time_Selector ); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 421 | RESET_CFLAG(context); |
| 422 | break; |
| 423 | |
| 424 | default: |
| 425 | VXD_BARF( context, "VTDAPI" ); |
| 426 | } |
| 427 | } |
| 428 | |
| 429 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 430 | * VXD_ConfigMG (WPROCS.451) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 431 | */ |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 432 | void WINAPI VXD_ConfigMG ( CONTEXT86 *context ) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 433 | { |
| 434 | unsigned service = AX_reg(context); |
| 435 | |
Alexandre Julliard | 908464d | 2000-11-01 03:11:12 +0000 | [diff] [blame] | 436 | TRACE("[%04x] ConfigMG\n", (UINT16)service); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 437 | |
| 438 | switch(service) |
| 439 | { |
| 440 | case 0x0000: /* version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 441 | SET_AX( context, VXD_WinVersion() ); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 442 | RESET_CFLAG(context); |
| 443 | break; |
| 444 | |
| 445 | default: |
| 446 | VXD_BARF( context, "CONFIGMG" ); |
| 447 | } |
| 448 | } |
| 449 | |
| 450 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 451 | * VXD_Enable (WPROCS.455) |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 452 | */ |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 453 | void WINAPI VXD_Enable ( CONTEXT86 *context ) |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 454 | { |
| 455 | unsigned service = AX_reg(context); |
| 456 | |
Alexandre Julliard | 908464d | 2000-11-01 03:11:12 +0000 | [diff] [blame] | 457 | TRACE("[%04x] Enable\n", (UINT16)service); |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 458 | |
| 459 | switch(service) |
| 460 | { |
| 461 | case 0x0000: /* version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 462 | SET_AX( context, VXD_WinVersion() ); |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 463 | RESET_CFLAG(context); |
| 464 | break; |
| 465 | |
| 466 | default: |
| 467 | VXD_BARF( context, "ENABLE" ); |
| 468 | } |
| 469 | } |
| 470 | |
| 471 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 472 | * VXD_APM (WPROCS.438) |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 473 | */ |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 474 | void WINAPI VXD_APM ( CONTEXT86 *context ) |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 475 | { |
| 476 | unsigned service = AX_reg(context); |
| 477 | |
Alexandre Julliard | 908464d | 2000-11-01 03:11:12 +0000 | [diff] [blame] | 478 | TRACE("[%04x] APM\n", (UINT16)service); |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 479 | |
| 480 | switch(service) |
| 481 | { |
| 482 | case 0x0000: /* version */ |
Alexandre Julliard | 3fa613c | 2002-08-31 18:47:00 +0000 | [diff] [blame] | 483 | SET_AX( context, VXD_WinVersion() ); |
Ulrich Weigand | ab635b2 | 1998-11-14 18:33:34 +0000 | [diff] [blame] | 484 | RESET_CFLAG(context); |
| 485 | break; |
| 486 | |
| 487 | default: |
| 488 | VXD_BARF( context, "APM" ); |
| 489 | } |
| 490 | } |
| 491 | |
| 492 | /*********************************************************************** |
Patrik Stridvall | 3ca9823 | 2001-06-20 23:03:14 +0000 | [diff] [blame] | 493 | * VXD_Win32s (WPROCS.445) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 494 | * |
| 495 | * This is an implementation of the services of the Win32s VxD. |
| 496 | * Since official documentation of these does not seem to be available, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 497 | * certain arguments of some of the services remain unclear. |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 498 | * |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 499 | * FIXME: The following services are currently unimplemented: |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 500 | * Exception handling (0x01, 0x1C) |
| 501 | * Debugger support (0x0C, 0x14, 0x17) |
| 502 | * Low-level memory access (0x02, 0x03, 0x0A, 0x0B) |
| 503 | * Memory Statistics (0x1B) |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 504 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 505 | * |
| 506 | * We have a specific problem running Win32s on Linux (and probably also |
| 507 | * the other x86 unixes), since Win32s tries to allocate its main 'flat |
| 508 | * code/data segment' selectors with a base of 0xffff0000 (and limit 4GB). |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 509 | * The rationale for this seems to be that they want one the one hand to |
| 510 | * be able to leave the Win 3.1 memory (starting with the main DOS memory) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 511 | * at linear address 0, but want at other hand to have offset 0 of the |
| 512 | * flat data/code segment point to an unmapped page (to catch NULL pointer |
| 513 | * accesses). Hence they allocate the flat segments with a base of 0xffff0000 |
| 514 | * so that the Win 3.1 memory area at linear address zero shows up in the |
| 515 | * flat segments at offset 0x10000 (since linear addresses wrap around at |
| 516 | * 4GB). To compensate for that discrepancy between flat segment offsets |
| 517 | * and plain linear addresses, all flat pointers passed between the 32-bit |
| 518 | * and the 16-bit parts of Win32s are shifted by 0x10000 in the appropriate |
| 519 | * direction by the glue code (mainly) in W32SKRNL and WIN32S16. |
| 520 | * |
| 521 | * The problem for us is now that Linux does not allow a LDT selector with |
| 522 | * base 0xffff0000 to be created, since it would 'see' a part of the kernel |
| 523 | * address space. To address this problem we introduce *another* offset: |
| 524 | * We add 0x10000 to every linear address we get as an argument from Win32s. |
| 525 | * This means especially that the flat code/data selectors get actually |
| 526 | * allocated with base 0x0, so that flat offsets and (real) linear addresses |
| 527 | * do again agree! In fact, every call e.g. of a Win32s VxD service now |
| 528 | * has all pointer arguments (which are offsets in the flat data segement) |
| 529 | * first reduced by 0x10000 by the W32SKRNL glue code, and then again |
| 530 | * increased by 0x10000 by *our* code. |
| 531 | * |
| 532 | * Note that to keep everything consistent, this offset has to be applied by |
| 533 | * every Wine function that operates on 'linear addresses' passed to it by |
| 534 | * Win32s. Fortunately, since Win32s does not directly call any Wine 32-bit |
| 535 | * API routines, this affects only two locations: this VxD and the DPMI |
| 536 | * handler. (NOTE: Should any Win32s application pass a linear address to |
| 537 | * any routine apart from those, e.g. some other VxD handler, that code |
| 538 | * would have to take the offset into account as well!) |
| 539 | * |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 540 | * The offset is set the first time any application calls the GetVersion() |
| 541 | * service of the Win32s VxD. (Note that the offset is never reset.) |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 542 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 543 | */ |
| 544 | |
Ulrich Weigand | ff9c44f | 1999-06-27 15:28:51 +0000 | [diff] [blame] | 545 | void WINAPI VXD_Win32s( CONTEXT86 *context ) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 546 | { |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 547 | switch (AX_reg(context)) |
| 548 | { |
| 549 | case 0x0000: /* Get Version */ |
| 550 | /* |
| 551 | * Input: None |
| 552 | * |
| 553 | * Output: EAX: LoWord: Win32s Version (1.30) |
| 554 | * HiWord: VxD Version (200) |
| 555 | * |
| 556 | * EBX: Build (172) |
| 557 | * |
| 558 | * ECX: ??? (1) |
| 559 | * |
| 560 | * EDX: Debugging Flags |
| 561 | * |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 562 | * EDI: Error Flag |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 563 | * 0 if OK, |
| 564 | * 1 if VMCPD VxD not found |
| 565 | */ |
| 566 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 567 | TRACE("GetVersion()\n"); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 568 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 569 | context->Eax = VXD_WinVersion() | (200 << 16); |
| 570 | context->Ebx = 0; |
| 571 | context->Ecx = 0; |
| 572 | context->Edx = 0; |
| 573 | context->Edi = 0; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 574 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 575 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 576 | * If this is the first time we are called for this process, |
| 577 | * hack the memory image of WIN32S16 so that it doesn't try |
| 578 | * to access the GDT directly ... |
| 579 | * |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 580 | * The first code segment of WIN32S16 (version 1.30) contains |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 581 | * an unexported function somewhere between the exported functions |
| 582 | * SetFS and StackLinearToSegmented that tries to find a selector |
| 583 | * in the LDT that maps to the memory image of the LDT itself. |
| 584 | * If it succeeds, it stores this selector into a global variable |
| 585 | * which will be used to speed up execution by using this selector |
| 586 | * to modify the LDT directly instead of using the DPMI calls. |
| 587 | * |
| 588 | * To perform this search of the LDT, this function uses the |
| 589 | * sgdt and sldt instructions to find the linear address of |
| 590 | * the (GDT and then) LDT. While those instructions themselves |
| 591 | * execute without problem, the linear address that sgdt returns |
| 592 | * points (at least under Linux) to the kernel address space, so |
| 593 | * that any subsequent access leads to a segfault. |
| 594 | * |
| 595 | * Fortunately, WIN32S16 still contains as a fallback option the |
| 596 | * mechanism of using DPMI calls to modify LDT selectors instead |
| 597 | * of direct writes to the LDT. Thus we can circumvent the problem |
| 598 | * by simply replacing the first byte of the offending function |
| 599 | * with an 'retf' instruction. This means that the global variable |
| 600 | * supposed to contain the LDT alias selector will remain zero, |
| 601 | * and hence WIN32S16 will fall back to using DPMI calls. |
| 602 | * |
| 603 | * The heuristic we employ to _find_ that function is as follows: |
| 604 | * We search between the addresses of the exported symbols SetFS |
| 605 | * and StackLinearToSegmented for the byte sequence '0F 01 04' |
| 606 | * (this is the opcode of 'sgdt [si]'). We then search backwards |
Andreas Mohr | 8bc7f16 | 2002-02-27 01:34:08 +0000 | [diff] [blame] | 607 | * from this address for the last occurrence of 'CB' (retf) that marks |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 608 | * the end of the preceeding function. The following byte (which |
| 609 | * should now be the first byte of the function we are looking for) |
| 610 | * will be replaced by 'CB' (retf). |
| 611 | * |
| 612 | * This heuristic works for the retail as well as the debug version |
| 613 | * of Win32s version 1.30. For versions earlier than that this |
| 614 | * hack should not be necessary at all, since the whole mechanism |
| 615 | * ('PERF130') was introduced only in 1.30 to improve the overall |
| 616 | * performance of Win32s. |
| 617 | */ |
| 618 | |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 619 | if (!W32S_offset) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 620 | { |
| 621 | HMODULE16 hModule = GetModuleHandle16("win32s16"); |
Alexandre Julliard | 5ce902b | 2000-11-27 21:59:08 +0000 | [diff] [blame] | 622 | SEGPTR func1 = (SEGPTR)GetProcAddress16(hModule, "SetFS"); |
| 623 | SEGPTR func2 = (SEGPTR)GetProcAddress16(hModule, "StackLinearToSegmented"); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 624 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 625 | if ( hModule && func1 && func2 |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 626 | && SELECTOROF(func1) == SELECTOROF(func2)) |
| 627 | { |
Alexandre Julliard | 982a223 | 2000-12-13 20:20:09 +0000 | [diff] [blame] | 628 | BYTE *start = MapSL(func1); |
| 629 | BYTE *end = MapSL(func2); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 630 | BYTE *p, *retv = NULL; |
| 631 | int found = 0; |
| 632 | |
| 633 | for (p = start; p < end; p++) |
| 634 | if (*p == 0xCB) found = 0, retv = p; |
| 635 | else if (*p == 0x0F) found = 1; |
| 636 | else if (*p == 0x01 && found == 1) found = 2; |
| 637 | else if (*p == 0x04 && found == 2) { found = 3; break; } |
| 638 | else found = 0; |
| 639 | |
| 640 | if (found == 3 && retv) |
| 641 | { |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 642 | TRACE("PERF130 hack: " |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 643 | "Replacing byte %02X at offset %04X:%04X\n", |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 644 | *(retv+1), SELECTOROF(func1), |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 645 | OFFSETOF(func1) + retv+1-start); |
| 646 | |
| 647 | *(retv+1) = (BYTE)0xCB; |
| 648 | } |
| 649 | } |
| 650 | } |
| 651 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 652 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 653 | * Mark process as Win32s, so that subsequent DPMI calls |
Andreas Mohr | a00b49f | 1998-12-07 10:48:09 +0000 | [diff] [blame] | 654 | * will perform the W32S_APP2WINE/W32S_WINE2APP address shift. |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 655 | */ |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 656 | W32S_offset = 0x10000; |
| 657 | break; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 658 | |
| 659 | |
| 660 | case 0x0001: /* Install Exception Handling */ |
| 661 | /* |
| 662 | * Input: EBX: Flat address of W32SKRNL Exception Data |
| 663 | * |
| 664 | * ECX: LoWord: Flat Code Selector |
| 665 | * HiWord: Flat Data Selector |
| 666 | * |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 667 | * EDX: Flat address of W32SKRNL Exception Handler |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 668 | * (this is equal to W32S_BackTo32 + 0x40) |
| 669 | * |
| 670 | * ESI: SEGPTR KERNEL.HASGPHANDLER |
| 671 | * |
| 672 | * EDI: SEGPTR phCurrentTask (KERNEL.THHOOK + 0x10) |
| 673 | * |
| 674 | * Output: EAX: 0 if OK |
| 675 | */ |
| 676 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 677 | TRACE("[0001] EBX=%lx ECX=%lx EDX=%lx ESI=%lx EDI=%lx\n", |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 678 | context->Ebx, context->Ecx, context->Edx, |
| 679 | context->Esi, context->Edi); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 680 | |
| 681 | /* FIXME */ |
| 682 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 683 | context->Eax = 0; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 684 | break; |
| 685 | |
| 686 | |
| 687 | case 0x0002: /* Set Page Access Flags */ |
| 688 | /* |
| 689 | * Input: EBX: New access flags |
| 690 | * Bit 2: User Page if set, Supervisor Page if clear |
| 691 | * Bit 1: Read-Write if set, Read-Only if clear |
| 692 | * |
| 693 | * ECX: Size of memory area to change |
| 694 | * |
| 695 | * EDX: Flat start address of memory area |
| 696 | * |
| 697 | * Output: EAX: Size of area changed |
| 698 | */ |
| 699 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 700 | TRACE("[0002] EBX=%lx ECX=%lx EDX=%lx\n", |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 701 | context->Ebx, context->Ecx, context->Edx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 702 | |
| 703 | /* FIXME */ |
| 704 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 705 | context->Eax = context->Ecx; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 706 | break; |
| 707 | |
| 708 | |
| 709 | case 0x0003: /* Get Page Access Flags */ |
| 710 | /* |
| 711 | * Input: EDX: Flat address of page to query |
| 712 | * |
| 713 | * Output: EAX: Page access flags |
| 714 | * Bit 2: User Page if set, Supervisor Page if clear |
| 715 | * Bit 1: Read-Write if set, Read-Only if clear |
| 716 | */ |
| 717 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 718 | TRACE("[0003] EDX=%lx\n", context->Edx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 719 | |
| 720 | /* FIXME */ |
| 721 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 722 | context->Eax = 6; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 723 | break; |
| 724 | |
| 725 | |
| 726 | case 0x0004: /* Map Module */ |
| 727 | /* |
| 728 | * Input: ECX: IMTE (offset in Module Table) of new module |
| 729 | * |
| 730 | * EDX: Flat address of Win32s Module Table |
| 731 | * |
| 732 | * Output: EAX: 0 if OK |
| 733 | */ |
| 734 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 735 | if (!context->Edx || CX_reg(context) == 0xFFFF) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 736 | { |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 737 | TRACE("MapModule: Initialization call\n"); |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 738 | context->Eax = 0; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 739 | } |
| 740 | else |
| 741 | { |
| 742 | /* |
| 743 | * Structure of a Win32s Module Table Entry: |
| 744 | */ |
| 745 | struct Win32sModule |
| 746 | { |
| 747 | DWORD flags; |
| 748 | DWORD flatBaseAddr; |
| 749 | LPCSTR moduleName; |
| 750 | LPCSTR pathName; |
| 751 | LPCSTR unknown; |
| 752 | LPBYTE baseAddr; |
| 753 | DWORD hModule; |
| 754 | DWORD relocDelta; |
| 755 | }; |
| 756 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 757 | /* |
| 758 | * Note: This function should set up a demand-paged memory image |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 759 | * of the given module. Since mmap does not allow file offsets |
| 760 | * not aligned at 1024 bytes, we simply load the image fully |
| 761 | * into memory. |
| 762 | */ |
| 763 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 764 | struct Win32sModule *moduleTable = |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 765 | (struct Win32sModule *)W32S_APP2WINE(context->Edx); |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 766 | struct Win32sModule *module = moduleTable + context->Ecx; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 767 | |
Alexandre Julliard | a5dea21 | 2002-08-09 19:57:38 +0000 | [diff] [blame] | 768 | IMAGE_NT_HEADERS *nt_header = RtlImageNtHeader( (HMODULE)module->baseAddr ); |
| 769 | IMAGE_SECTION_HEADER *pe_seg = (IMAGE_SECTION_HEADER*)((char *)&nt_header->OptionalHeader + |
| 770 | nt_header->FileHeader.SizeOfOptionalHeader); |
| 771 | |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 772 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 773 | HFILE image = _lopen(module->pathName, OF_READ); |
Alexandre Julliard | 908464d | 2000-11-01 03:11:12 +0000 | [diff] [blame] | 774 | BOOL error = (image == HFILE_ERROR); |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 775 | UINT i; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 776 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 777 | TRACE("MapModule: Loading %s\n", module->pathName); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 778 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 779 | for (i = 0; |
| 780 | !error && i < nt_header->FileHeader.NumberOfSections; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 781 | i++, pe_seg++) |
| 782 | if(!(pe_seg->Characteristics & IMAGE_SCN_CNT_UNINITIALIZED_DATA)) |
| 783 | { |
| 784 | DWORD off = pe_seg->PointerToRawData; |
| 785 | DWORD len = pe_seg->SizeOfRawData; |
| 786 | LPBYTE addr = module->baseAddr + pe_seg->VirtualAddress; |
| 787 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 788 | TRACE("MapModule: " |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 789 | "Section %d at %08lx from %08lx len %08lx\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 790 | i, (DWORD)addr, off, len); |
| 791 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 792 | if ( _llseek(image, off, SEEK_SET) != off |
| 793 | || _lread(image, addr, len) != len) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 794 | error = TRUE; |
| 795 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 796 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 797 | _lclose(image); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 798 | |
| 799 | if (error) |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 800 | ERR("MapModule: Unable to load %s\n", module->pathName); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 801 | |
| 802 | else if (module->relocDelta != 0) |
| 803 | { |
| 804 | IMAGE_DATA_DIRECTORY *dir = nt_header->OptionalHeader.DataDirectory |
| 805 | + IMAGE_DIRECTORY_ENTRY_BASERELOC; |
| 806 | IMAGE_BASE_RELOCATION *r = (IMAGE_BASE_RELOCATION *) |
| 807 | (dir->Size? module->baseAddr + dir->VirtualAddress : 0); |
| 808 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 809 | TRACE("MapModule: Reloc delta %08lx\n", module->relocDelta); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 810 | |
| 811 | while (r && r->VirtualAddress) |
| 812 | { |
| 813 | LPBYTE page = module->baseAddr + r->VirtualAddress; |
Dmitry Timoshkov | 6b6596a | 2001-11-23 18:44:43 +0000 | [diff] [blame] | 814 | WORD *TypeOffset = (WORD *)(r + 1); |
| 815 | int count = (r->SizeOfBlock - sizeof(*r)) / sizeof(*TypeOffset); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 816 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 817 | TRACE("MapModule: %d relocations for page %08lx\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 818 | count, (DWORD)page); |
| 819 | |
| 820 | for(i = 0; i < count; i++) |
| 821 | { |
Dmitry Timoshkov | 6b6596a | 2001-11-23 18:44:43 +0000 | [diff] [blame] | 822 | int offset = TypeOffset[i] & 0xFFF; |
| 823 | int type = TypeOffset[i] >> 12; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 824 | switch(type) |
| 825 | { |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 826 | case IMAGE_REL_BASED_ABSOLUTE: |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 827 | break; |
| 828 | case IMAGE_REL_BASED_HIGH: |
| 829 | *(WORD *)(page+offset) += HIWORD(module->relocDelta); |
| 830 | break; |
| 831 | case IMAGE_REL_BASED_LOW: |
| 832 | *(WORD *)(page+offset) += LOWORD(module->relocDelta); |
| 833 | break; |
| 834 | case IMAGE_REL_BASED_HIGHLOW: |
| 835 | *(DWORD*)(page+offset) += module->relocDelta; |
| 836 | break; |
| 837 | default: |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 838 | WARN("MapModule: Unsupported fixup type\n"); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 839 | break; |
| 840 | } |
| 841 | } |
| 842 | |
| 843 | r = (IMAGE_BASE_RELOCATION *)((LPBYTE)r + r->SizeOfBlock); |
| 844 | } |
| 845 | } |
| 846 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 847 | context->Eax = 0; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 848 | RESET_CFLAG(context); |
| 849 | } |
| 850 | break; |
| 851 | |
| 852 | |
| 853 | case 0x0005: /* UnMap Module */ |
| 854 | /* |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 855 | * Input: EDX: Flat address of module image |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 856 | * |
| 857 | * Output: EAX: 1 if OK |
| 858 | */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 859 | |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 860 | TRACE("UnMapModule: %lx\n", (DWORD)W32S_APP2WINE(context->Edx)); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 861 | |
| 862 | /* As we didn't map anything, there's nothing to unmap ... */ |
| 863 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 864 | context->Eax = 1; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 865 | break; |
| 866 | |
| 867 | |
| 868 | case 0x0006: /* VirtualAlloc */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 869 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 870 | * Input: ECX: Current Process |
| 871 | * |
| 872 | * EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 873 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 874 | * DWORD *retv [out] Flat base address of allocated region |
| 875 | * LPVOID base [in] Flat address of region to reserve/commit |
| 876 | * DWORD size [in] Size of region |
| 877 | * DWORD type [in] Type of allocation |
| 878 | * DWORD prot [in] Type of access protection |
| 879 | * |
| 880 | * Output: EAX: NtStatus |
| 881 | */ |
| 882 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 883 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
| 884 | DWORD *retv = (DWORD *)W32S_APP2WINE(stack[0]); |
| 885 | LPVOID base = (LPVOID) W32S_APP2WINE(stack[1]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 886 | DWORD size = stack[2]; |
| 887 | DWORD type = stack[3]; |
| 888 | DWORD prot = stack[4]; |
| 889 | DWORD result; |
| 890 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 891 | TRACE("VirtualAlloc(%lx, %lx, %lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 892 | (DWORD)retv, (DWORD)base, size, type, prot); |
| 893 | |
| 894 | if (type & 0x80000000) |
| 895 | { |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 896 | WARN("VirtualAlloc: strange type %lx\n", type); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 897 | type &= 0x7fffffff; |
| 898 | } |
| 899 | |
| 900 | if (!base && (type & MEM_COMMIT) && prot == PAGE_READONLY) |
| 901 | { |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 902 | WARN("VirtualAlloc: NLS hack, allowing write access!\n"); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 903 | prot = PAGE_READWRITE; |
| 904 | } |
| 905 | |
| 906 | result = (DWORD)VirtualAlloc(base, size, type, prot); |
| 907 | |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 908 | if (W32S_WINE2APP(result)) |
| 909 | *retv = W32S_WINE2APP(result), |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 910 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 911 | else |
| 912 | *retv = 0, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 913 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 914 | } |
| 915 | break; |
| 916 | |
| 917 | |
| 918 | case 0x0007: /* VirtualFree */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 919 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 920 | * Input: ECX: Current Process |
| 921 | * |
| 922 | * EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 923 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 924 | * DWORD *retv [out] TRUE if success, FALSE if failure |
| 925 | * LPVOID base [in] Flat address of region |
| 926 | * DWORD size [in] Size of region |
| 927 | * DWORD type [in] Type of operation |
| 928 | * |
| 929 | * Output: EAX: NtStatus |
| 930 | */ |
| 931 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 932 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
| 933 | DWORD *retv = (DWORD *)W32S_APP2WINE(stack[0]); |
| 934 | LPVOID base = (LPVOID) W32S_APP2WINE(stack[1]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 935 | DWORD size = stack[2]; |
| 936 | DWORD type = stack[3]; |
| 937 | DWORD result; |
| 938 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 939 | TRACE("VirtualFree(%lx, %lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 940 | (DWORD)retv, (DWORD)base, size, type); |
| 941 | |
| 942 | result = VirtualFree(base, size, type); |
| 943 | |
| 944 | if (result) |
| 945 | *retv = TRUE, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 946 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 947 | else |
| 948 | *retv = FALSE, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 949 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 950 | } |
| 951 | break; |
| 952 | |
| 953 | |
| 954 | case 0x0008: /* VirtualProtect */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 955 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 956 | * Input: ECX: Current Process |
| 957 | * |
| 958 | * EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 959 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 960 | * DWORD *retv [out] TRUE if success, FALSE if failure |
| 961 | * LPVOID base [in] Flat address of region |
| 962 | * DWORD size [in] Size of region |
| 963 | * DWORD new_prot [in] Desired access protection |
| 964 | * DWORD *old_prot [out] Previous access protection |
| 965 | * |
| 966 | * Output: EAX: NtStatus |
| 967 | */ |
| 968 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 969 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
| 970 | DWORD *retv = (DWORD *)W32S_APP2WINE(stack[0]); |
| 971 | LPVOID base = (LPVOID) W32S_APP2WINE(stack[1]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 972 | DWORD size = stack[2]; |
| 973 | DWORD new_prot = stack[3]; |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 974 | DWORD *old_prot = (DWORD *)W32S_APP2WINE(stack[4]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 975 | DWORD result; |
| 976 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 977 | TRACE("VirtualProtect(%lx, %lx, %lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 978 | (DWORD)retv, (DWORD)base, size, new_prot, (DWORD)old_prot); |
| 979 | |
| 980 | result = VirtualProtect(base, size, new_prot, old_prot); |
| 981 | |
| 982 | if (result) |
| 983 | *retv = TRUE, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 984 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 985 | else |
| 986 | *retv = FALSE, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 987 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 988 | } |
| 989 | break; |
| 990 | |
| 991 | |
| 992 | case 0x0009: /* VirtualQuery */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 993 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 994 | * Input: ECX: Current Process |
| 995 | * |
| 996 | * EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 997 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 998 | * DWORD *retv [out] Nr. bytes returned |
| 999 | * LPVOID base [in] Flat address of region |
| 1000 | * LPMEMORY_BASIC_INFORMATION info [out] Info buffer |
| 1001 | * DWORD len [in] Size of buffer |
| 1002 | * |
| 1003 | * Output: EAX: NtStatus |
| 1004 | */ |
| 1005 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1006 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
| 1007 | DWORD *retv = (DWORD *)W32S_APP2WINE(stack[0]); |
| 1008 | LPVOID base = (LPVOID) W32S_APP2WINE(stack[1]); |
Patrik Stridvall | ba9dc78 | 2002-11-04 22:43:24 +0000 | [diff] [blame^] | 1009 | PMEMORY_BASIC_INFORMATION info = |
| 1010 | (PMEMORY_BASIC_INFORMATION)W32S_APP2WINE(stack[2]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1011 | DWORD len = stack[3]; |
| 1012 | DWORD result; |
| 1013 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1014 | TRACE("VirtualQuery(%lx, %lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1015 | (DWORD)retv, (DWORD)base, (DWORD)info, len); |
| 1016 | |
| 1017 | result = VirtualQuery(base, info, len); |
| 1018 | |
| 1019 | *retv = result; |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1020 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1021 | } |
| 1022 | break; |
| 1023 | |
| 1024 | |
| 1025 | case 0x000A: /* SetVirtMemProcess */ |
| 1026 | /* |
| 1027 | * Input: ECX: Process Handle |
| 1028 | * |
| 1029 | * EDX: Flat address of region |
| 1030 | * |
| 1031 | * Output: EAX: NtStatus |
| 1032 | */ |
| 1033 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1034 | TRACE("[000a] ECX=%lx EDX=%lx\n", |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1035 | context->Ecx, context->Edx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1036 | |
| 1037 | /* FIXME */ |
| 1038 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1039 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1040 | break; |
| 1041 | |
| 1042 | |
| 1043 | case 0x000B: /* ??? some kind of cleanup */ |
| 1044 | /* |
| 1045 | * Input: ECX: Process Handle |
| 1046 | * |
| 1047 | * Output: EAX: NtStatus |
| 1048 | */ |
| 1049 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1050 | TRACE("[000b] ECX=%lx\n", context->Ecx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1051 | |
| 1052 | /* FIXME */ |
| 1053 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1054 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1055 | break; |
| 1056 | |
| 1057 | |
| 1058 | case 0x000C: /* Set Debug Flags */ |
| 1059 | /* |
| 1060 | * Input: EDX: Debug Flags |
| 1061 | * |
| 1062 | * Output: EDX: Previous Debug Flags |
| 1063 | */ |
| 1064 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1065 | FIXME("[000c] EDX=%lx\n", context->Edx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1066 | |
| 1067 | /* FIXME */ |
| 1068 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1069 | context->Edx = 0; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1070 | break; |
| 1071 | |
| 1072 | |
| 1073 | case 0x000D: /* NtCreateSection */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1074 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1075 | * Input: EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1076 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1077 | * HANDLE32 *retv [out] Handle of Section created |
| 1078 | * DWORD flags1 [in] (?? unknown ??) |
| 1079 | * DWORD atom [in] Name of Section to create |
| 1080 | * LARGE_INTEGER *size [in] Size of Section |
| 1081 | * DWORD protect [in] Access protection |
| 1082 | * DWORD flags2 [in] (?? unknown ??) |
| 1083 | * HFILE32 hFile [in] Handle of file to map |
| 1084 | * DWORD psp [in] (Win32s: PSP that hFile belongs to) |
| 1085 | * |
| 1086 | * Output: EAX: NtStatus |
| 1087 | */ |
| 1088 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1089 | DWORD *stack = (DWORD *) W32S_APP2WINE(context->Edx); |
| 1090 | HANDLE *retv = (HANDLE *)W32S_APP2WINE(stack[0]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1091 | DWORD flags1 = stack[1]; |
| 1092 | DWORD atom = stack[2]; |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1093 | LARGE_INTEGER *size = (LARGE_INTEGER *)W32S_APP2WINE(stack[3]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1094 | DWORD protect = stack[4]; |
| 1095 | DWORD flags2 = stack[5]; |
Alexandre Julliard | 5ce902b | 2000-11-27 21:59:08 +0000 | [diff] [blame] | 1096 | HANDLE hFile = DosFileHandleToWin32Handle(stack[6]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1097 | DWORD psp = stack[7]; |
| 1098 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1099 | HANDLE result = INVALID_HANDLE_VALUE; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1100 | char name[128]; |
| 1101 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1102 | TRACE("NtCreateSection(%lx, %lx, %lx, %lx, %lx, %lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1103 | (DWORD)retv, flags1, atom, (DWORD)size, protect, flags2, |
| 1104 | (DWORD)hFile, psp); |
| 1105 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1106 | if (!atom || GlobalGetAtomNameA(atom, name, sizeof(name))) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1107 | { |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1108 | TRACE("NtCreateSection: name=%s\n", atom? name : NULL); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1109 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1110 | result = CreateFileMappingA(hFile, NULL, protect, |
| 1111 | size? size->s.HighPart : 0, |
| 1112 | size? size->s.LowPart : 0, |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1113 | atom? name : NULL); |
| 1114 | } |
| 1115 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1116 | if (result == INVALID_HANDLE_VALUE) |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1117 | WARN("NtCreateSection: failed!\n"); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1118 | else |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1119 | TRACE("NtCreateSection: returned %lx\n", (DWORD)result); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1120 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1121 | if (result != INVALID_HANDLE_VALUE) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1122 | *retv = result, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1123 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1124 | else |
| 1125 | *retv = result, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1126 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1127 | } |
| 1128 | break; |
| 1129 | |
| 1130 | |
| 1131 | case 0x000E: /* NtOpenSection */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1132 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1133 | * Input: EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1134 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1135 | * HANDLE32 *retv [out] Handle of Section opened |
| 1136 | * DWORD protect [in] Access protection |
| 1137 | * DWORD atom [in] Name of Section to create |
| 1138 | * |
| 1139 | * Output: EAX: NtStatus |
| 1140 | */ |
| 1141 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1142 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
| 1143 | HANDLE *retv = (HANDLE *)W32S_APP2WINE(stack[0]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1144 | DWORD protect = stack[1]; |
| 1145 | DWORD atom = stack[2]; |
| 1146 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1147 | HANDLE result = INVALID_HANDLE_VALUE; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1148 | char name[128]; |
| 1149 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1150 | TRACE("NtOpenSection(%lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1151 | (DWORD)retv, protect, atom); |
| 1152 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1153 | if (atom && GlobalGetAtomNameA(atom, name, sizeof(name))) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1154 | { |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1155 | TRACE("NtOpenSection: name=%s\n", name); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1156 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1157 | result = OpenFileMappingA(protect, FALSE, name); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1158 | } |
| 1159 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1160 | if (result == INVALID_HANDLE_VALUE) |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1161 | WARN("NtOpenSection: failed!\n"); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1162 | else |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1163 | TRACE("NtOpenSection: returned %lx\n", (DWORD)result); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1164 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1165 | if (result != INVALID_HANDLE_VALUE) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1166 | *retv = result, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1167 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1168 | else |
| 1169 | *retv = result, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1170 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1171 | } |
| 1172 | break; |
| 1173 | |
| 1174 | |
| 1175 | case 0x000F: /* NtCloseSection */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1176 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1177 | * Input: EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1178 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1179 | * HANDLE32 handle [in] Handle of Section to close |
| 1180 | * DWORD *id [out] Unique ID (?? unclear ??) |
| 1181 | * |
| 1182 | * Output: EAX: NtStatus |
| 1183 | */ |
| 1184 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1185 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
Alexandre Julliard | 7375597 | 2002-07-31 19:26:03 +0000 | [diff] [blame] | 1186 | HANDLE handle = (HANDLE)stack[0]; |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1187 | DWORD *id = (DWORD *)W32S_APP2WINE(stack[1]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1188 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1189 | TRACE("NtCloseSection(%lx, %lx)\n", (DWORD)handle, (DWORD)id); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1190 | |
| 1191 | CloseHandle(handle); |
| 1192 | if (id) *id = 0; /* FIXME */ |
| 1193 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1194 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1195 | } |
| 1196 | break; |
| 1197 | |
| 1198 | |
| 1199 | case 0x0010: /* NtDupSection */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1200 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1201 | * Input: EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1202 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1203 | * HANDLE32 handle [in] Handle of Section to duplicate |
| 1204 | * |
| 1205 | * Output: EAX: NtStatus |
| 1206 | */ |
| 1207 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1208 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
Alexandre Julliard | 7375597 | 2002-07-31 19:26:03 +0000 | [diff] [blame] | 1209 | HANDLE handle = (HANDLE)stack[0]; |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 1210 | HANDLE new_handle; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1211 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1212 | TRACE("NtDupSection(%lx)\n", (DWORD)handle); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1213 | |
Alexandre Julliard | 62a8b43 | 1999-01-19 17:48:23 +0000 | [diff] [blame] | 1214 | DuplicateHandle( GetCurrentProcess(), handle, |
| 1215 | GetCurrentProcess(), &new_handle, |
| 1216 | 0, FALSE, DUPLICATE_SAME_ACCESS ); |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1217 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1218 | } |
| 1219 | break; |
| 1220 | |
| 1221 | |
| 1222 | case 0x0011: /* NtMapViewOfSection */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1223 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1224 | * Input: EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1225 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1226 | * HANDLE32 SectionHandle [in] Section to be mapped |
| 1227 | * DWORD ProcessHandle [in] Process to be mapped into |
| 1228 | * DWORD * BaseAddress [in/out] Address to be mapped at |
| 1229 | * DWORD ZeroBits [in] (?? unclear ??) |
| 1230 | * DWORD CommitSize [in] (?? unclear ??) |
| 1231 | * LARGE_INTEGER *SectionOffset [in] Offset within section |
| 1232 | * DWORD * ViewSize [in] Size of view |
| 1233 | * DWORD InheritDisposition [in] (?? unclear ??) |
| 1234 | * DWORD AllocationType [in] (?? unclear ??) |
| 1235 | * DWORD Protect [in] Access protection |
| 1236 | * |
| 1237 | * Output: EAX: NtStatus |
| 1238 | */ |
| 1239 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1240 | DWORD * stack = (DWORD *)W32S_APP2WINE(context->Edx); |
Alexandre Julliard | 7375597 | 2002-07-31 19:26:03 +0000 | [diff] [blame] | 1241 | HANDLE SectionHandle = (HANDLE)stack[0]; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1242 | DWORD ProcessHandle = stack[1]; /* ignored */ |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1243 | DWORD * BaseAddress = (DWORD *)W32S_APP2WINE(stack[2]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1244 | DWORD ZeroBits = stack[3]; |
| 1245 | DWORD CommitSize = stack[4]; |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1246 | LARGE_INTEGER *SectionOffset = (LARGE_INTEGER *)W32S_APP2WINE(stack[5]); |
| 1247 | DWORD * ViewSize = (DWORD *)W32S_APP2WINE(stack[6]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1248 | DWORD InheritDisposition = stack[7]; |
| 1249 | DWORD AllocationType = stack[8]; |
| 1250 | DWORD Protect = stack[9]; |
| 1251 | |
Andreas Mohr | a00b49f | 1998-12-07 10:48:09 +0000 | [diff] [blame] | 1252 | LPBYTE address = (LPBYTE)(BaseAddress? |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1253 | W32S_APP2WINE(*BaseAddress) : 0); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1254 | DWORD access = 0, result; |
| 1255 | |
| 1256 | switch (Protect & ~(PAGE_GUARD|PAGE_NOCACHE)) |
| 1257 | { |
| 1258 | case PAGE_READONLY: access = FILE_MAP_READ; break; |
| 1259 | case PAGE_READWRITE: access = FILE_MAP_WRITE; break; |
| 1260 | case PAGE_WRITECOPY: access = FILE_MAP_COPY; break; |
| 1261 | |
| 1262 | case PAGE_EXECUTE_READ: access = FILE_MAP_READ; break; |
| 1263 | case PAGE_EXECUTE_READWRITE: access = FILE_MAP_WRITE; break; |
| 1264 | case PAGE_EXECUTE_WRITECOPY: access = FILE_MAP_COPY; break; |
| 1265 | } |
| 1266 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1267 | TRACE("NtMapViewOfSection" |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1268 | "(%lx, %lx, %lx, %lx, %lx, %lx, %lx, %lx, %lx, %lx)\n", |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1269 | (DWORD)SectionHandle, ProcessHandle, (DWORD)BaseAddress, |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1270 | ZeroBits, CommitSize, (DWORD)SectionOffset, (DWORD)ViewSize, |
| 1271 | InheritDisposition, AllocationType, Protect); |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1272 | TRACE("NtMapViewOfSection: " |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1273 | "base=%lx, offset=%lx, size=%lx, access=%lx\n", |
| 1274 | (DWORD)address, SectionOffset? SectionOffset->s.LowPart : 0, |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1275 | ViewSize? *ViewSize : 0, access); |
| 1276 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1277 | result = (DWORD)MapViewOfFileEx(SectionHandle, access, |
| 1278 | SectionOffset? SectionOffset->s.HighPart : 0, |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 1279 | SectionOffset? SectionOffset->s.LowPart : 0, |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1280 | ViewSize? *ViewSize : 0, address); |
| 1281 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1282 | TRACE("NtMapViewOfSection: result=%lx\n", result); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1283 | |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1284 | if (W32S_WINE2APP(result)) |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1285 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1286 | if (BaseAddress) *BaseAddress = W32S_WINE2APP(result); |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1287 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1288 | } |
| 1289 | else |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1290 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1291 | } |
| 1292 | break; |
| 1293 | |
| 1294 | |
| 1295 | case 0x0012: /* NtUnmapViewOfSection */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1296 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1297 | * Input: EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1298 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1299 | * DWORD ProcessHandle [in] Process (defining address space) |
| 1300 | * LPBYTE BaseAddress [in] Base address of view to be unmapped |
| 1301 | * |
| 1302 | * Output: EAX: NtStatus |
| 1303 | */ |
| 1304 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1305 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1306 | DWORD ProcessHandle = stack[0]; /* ignored */ |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1307 | LPBYTE BaseAddress = (LPBYTE)W32S_APP2WINE(stack[1]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1308 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1309 | TRACE("NtUnmapViewOfSection(%lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1310 | ProcessHandle, (DWORD)BaseAddress); |
| 1311 | |
| 1312 | UnmapViewOfFile(BaseAddress); |
| 1313 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1314 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1315 | } |
| 1316 | break; |
| 1317 | |
| 1318 | |
| 1319 | case 0x0013: /* NtFlushVirtualMemory */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1320 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1321 | * Input: EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1322 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1323 | * DWORD ProcessHandle [in] Process (defining address space) |
| 1324 | * LPBYTE *BaseAddress [in?] Base address of range to be flushed |
| 1325 | * DWORD *ViewSize [in?] Number of bytes to be flushed |
| 1326 | * DWORD *unknown [???] (?? unknown ??) |
| 1327 | * |
| 1328 | * Output: EAX: NtStatus |
| 1329 | */ |
| 1330 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1331 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1332 | DWORD ProcessHandle = stack[0]; /* ignored */ |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1333 | DWORD *BaseAddress = (DWORD *)W32S_APP2WINE(stack[1]); |
| 1334 | DWORD *ViewSize = (DWORD *)W32S_APP2WINE(stack[2]); |
| 1335 | DWORD *unknown = (DWORD *)W32S_APP2WINE(stack[3]); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1336 | |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1337 | LPBYTE address = (LPBYTE)(BaseAddress? W32S_APP2WINE(*BaseAddress) : 0); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1338 | DWORD size = ViewSize? *ViewSize : 0; |
| 1339 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1340 | TRACE("NtFlushVirtualMemory(%lx, %lx, %lx, %lx)\n", |
| 1341 | ProcessHandle, (DWORD)BaseAddress, (DWORD)ViewSize, |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1342 | (DWORD)unknown); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1343 | TRACE("NtFlushVirtualMemory: base=%lx, size=%lx\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1344 | (DWORD)address, size); |
| 1345 | |
| 1346 | FlushViewOfFile(address, size); |
| 1347 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1348 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1349 | } |
| 1350 | break; |
| 1351 | |
| 1352 | |
| 1353 | case 0x0014: /* Get/Set Debug Registers */ |
| 1354 | /* |
| 1355 | * Input: ECX: 0 if Get, 1 if Set |
| 1356 | * |
| 1357 | * EDX: Get: Flat address of buffer to receive values of |
| 1358 | * debug registers DR0 .. DR7 |
| 1359 | * Set: Flat address of buffer containing values of |
| 1360 | * debug registers DR0 .. DR7 to be set |
| 1361 | * Output: None |
| 1362 | */ |
| 1363 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1364 | FIXME("[0014] ECX=%lx EDX=%lx\n", |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1365 | context->Ecx, context->Edx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1366 | |
| 1367 | /* FIXME */ |
| 1368 | break; |
| 1369 | |
| 1370 | |
| 1371 | case 0x0015: /* Set Coprocessor Emulation Flag */ |
| 1372 | /* |
| 1373 | * Input: EDX: 0 to deactivate, 1 to activate coprocessor emulation |
| 1374 | * |
| 1375 | * Output: None |
| 1376 | */ |
| 1377 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1378 | TRACE("[0015] EDX=%lx\n", context->Edx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1379 | |
| 1380 | /* We don't care, as we always have a coprocessor anyway */ |
| 1381 | break; |
| 1382 | |
| 1383 | |
| 1384 | case 0x0016: /* Init Win32S VxD PSP */ |
| 1385 | /* |
| 1386 | * If called to query required PSP size: |
| 1387 | * |
| 1388 | * Input: EBX: 0 |
| 1389 | * Output: EDX: Required size of Win32s VxD PSP |
| 1390 | * |
| 1391 | * If called to initialize allocated PSP: |
| 1392 | * |
| 1393 | * Input: EBX: LoWord: Selector of Win32s VxD PSP |
| 1394 | * HiWord: Paragraph of Win32s VxD PSP (DOSMEM) |
| 1395 | * Output: None |
| 1396 | */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1397 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1398 | if (context->Ebx == 0) |
| 1399 | context->Edx = 0x80; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1400 | else |
| 1401 | { |
Alexandre Julliard | 982a223 | 2000-12-13 20:20:09 +0000 | [diff] [blame] | 1402 | PDB16 *psp = MapSL( MAKESEGPTR( BX_reg(context), 0 )); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1403 | psp->nbFiles = 32; |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1404 | psp->fileHandlesPtr = MAKELONG(HIWORD(context->Ebx), 0x5c); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1405 | memset((LPBYTE)psp + 0x5c, '\xFF', 32); |
| 1406 | } |
| 1407 | break; |
| 1408 | |
| 1409 | |
| 1410 | case 0x0017: /* Set Break Point */ |
| 1411 | /* |
| 1412 | * Input: EBX: Offset of Break Point |
| 1413 | * CX: Selector of Break Point |
| 1414 | * |
| 1415 | * Output: None |
| 1416 | */ |
| 1417 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1418 | FIXME("[0017] EBX=%lx CX=%x\n", |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1419 | context->Ebx, CX_reg(context)); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1420 | |
| 1421 | /* FIXME */ |
| 1422 | break; |
| 1423 | |
| 1424 | |
| 1425 | case 0x0018: /* VirtualLock */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1426 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1427 | * Input: ECX: Current Process |
| 1428 | * |
| 1429 | * EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1430 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1431 | * DWORD *retv [out] TRUE if success, FALSE if failure |
| 1432 | * LPVOID base [in] Flat address of range to lock |
| 1433 | * DWORD size [in] Size of range |
| 1434 | * |
| 1435 | * Output: EAX: NtStatus |
| 1436 | */ |
| 1437 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1438 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
| 1439 | DWORD *retv = (DWORD *)W32S_APP2WINE(stack[0]); |
| 1440 | LPVOID base = (LPVOID) W32S_APP2WINE(stack[1]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1441 | DWORD size = stack[2]; |
| 1442 | DWORD result; |
| 1443 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1444 | TRACE("VirtualLock(%lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1445 | (DWORD)retv, (DWORD)base, size); |
| 1446 | |
| 1447 | result = VirtualLock(base, size); |
| 1448 | |
| 1449 | if (result) |
| 1450 | *retv = TRUE, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1451 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1452 | else |
| 1453 | *retv = FALSE, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1454 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1455 | } |
| 1456 | break; |
| 1457 | |
| 1458 | |
| 1459 | case 0x0019: /* VirtualUnlock */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1460 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1461 | * Input: ECX: Current Process |
| 1462 | * |
| 1463 | * EDX: Flat address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1464 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1465 | * DWORD *retv [out] TRUE if success, FALSE if failure |
| 1466 | * LPVOID base [in] Flat address of range to unlock |
| 1467 | * DWORD size [in] Size of range |
| 1468 | * |
| 1469 | * Output: EAX: NtStatus |
| 1470 | */ |
| 1471 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1472 | DWORD *stack = (DWORD *)W32S_APP2WINE(context->Edx); |
| 1473 | DWORD *retv = (DWORD *)W32S_APP2WINE(stack[0]); |
| 1474 | LPVOID base = (LPVOID) W32S_APP2WINE(stack[1]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1475 | DWORD size = stack[2]; |
| 1476 | DWORD result; |
| 1477 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1478 | TRACE("VirtualUnlock(%lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1479 | (DWORD)retv, (DWORD)base, size); |
| 1480 | |
| 1481 | result = VirtualUnlock(base, size); |
| 1482 | |
| 1483 | if (result) |
| 1484 | *retv = TRUE, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1485 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1486 | else |
| 1487 | *retv = FALSE, |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1488 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1489 | } |
| 1490 | break; |
| 1491 | |
| 1492 | |
| 1493 | case 0x001A: /* KGetSystemInfo */ |
| 1494 | /* |
| 1495 | * Input: None |
| 1496 | * |
| 1497 | * Output: ECX: Start of sparse memory arena |
| 1498 | * EDX: End of sparse memory arena |
| 1499 | */ |
| 1500 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1501 | TRACE("KGetSystemInfo()\n"); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1502 | |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1503 | /* |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1504 | * Note: Win32s reserves 0GB - 2GB for Win 3.1 and uses 2GB - 4GB as |
| 1505 | * sparse memory arena. We do it the other way around, since |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1506 | * we have to reserve 3GB - 4GB for Linux, and thus use |
| 1507 | * 0GB - 3GB as sparse memory arena. |
| 1508 | * |
| 1509 | * FIXME: What about other OSes ? |
| 1510 | */ |
| 1511 | |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1512 | context->Ecx = W32S_WINE2APP(0x00000000); |
| 1513 | context->Edx = W32S_WINE2APP(0xbfffffff); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1514 | break; |
| 1515 | |
| 1516 | |
| 1517 | case 0x001B: /* KGlobalMemStat */ |
| 1518 | /* |
| 1519 | * Input: ESI: Flat address of buffer to receive memory info |
| 1520 | * |
| 1521 | * Output: None |
| 1522 | */ |
| 1523 | { |
| 1524 | struct Win32sMemoryInfo |
| 1525 | { |
| 1526 | DWORD DIPhys_Count; /* Total physical pages */ |
| 1527 | DWORD DIFree_Count; /* Free physical pages */ |
| 1528 | DWORD DILin_Total_Count; /* Total virtual pages (private arena) */ |
| 1529 | DWORD DILin_Total_Free; /* Free virtual pages (private arena) */ |
| 1530 | |
| 1531 | DWORD SparseTotal; /* Total size of sparse arena (bytes ?) */ |
| 1532 | DWORD SparseFree; /* Free size of sparse arena (bytes ?) */ |
| 1533 | }; |
| 1534 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1535 | struct Win32sMemoryInfo *info = |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1536 | (struct Win32sMemoryInfo *)W32S_APP2WINE(context->Esi); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1537 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1538 | FIXME("KGlobalMemStat(%lx)\n", (DWORD)info); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1539 | |
| 1540 | /* FIXME */ |
| 1541 | } |
| 1542 | break; |
| 1543 | |
| 1544 | |
| 1545 | case 0x001C: /* Enable/Disable Exceptions */ |
| 1546 | /* |
| 1547 | * Input: ECX: 0 to disable, 1 to enable exception handling |
| 1548 | * |
| 1549 | * Output: None |
| 1550 | */ |
| 1551 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1552 | TRACE("[001c] ECX=%lx\n", context->Ecx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1553 | |
| 1554 | /* FIXME */ |
| 1555 | break; |
| 1556 | |
| 1557 | |
| 1558 | case 0x001D: /* VirtualAlloc called from 16-bit code */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1559 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1560 | * Input: EDX: Segmented address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1561 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1562 | * LPVOID base [in] Flat address of region to reserve/commit |
| 1563 | * DWORD size [in] Size of region |
| 1564 | * DWORD type [in] Type of allocation |
| 1565 | * DWORD prot [in] Type of access protection |
| 1566 | * |
| 1567 | * Output: EAX: NtStatus |
| 1568 | * EDX: Flat base address of allocated region |
| 1569 | */ |
| 1570 | { |
Alexandre Julliard | 982a223 | 2000-12-13 20:20:09 +0000 | [diff] [blame] | 1571 | DWORD *stack = MapSL( MAKESEGPTR( LOWORD(context->Edx), HIWORD(context->Edx) )); |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1572 | LPVOID base = (LPVOID)W32S_APP2WINE(stack[0]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1573 | DWORD size = stack[1]; |
| 1574 | DWORD type = stack[2]; |
| 1575 | DWORD prot = stack[3]; |
| 1576 | DWORD result; |
| 1577 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1578 | TRACE("VirtualAlloc16(%lx, %lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1579 | (DWORD)base, size, type, prot); |
| 1580 | |
| 1581 | if (type & 0x80000000) |
| 1582 | { |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1583 | WARN("VirtualAlloc16: strange type %lx\n", type); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1584 | type &= 0x7fffffff; |
| 1585 | } |
| 1586 | |
| 1587 | result = (DWORD)VirtualAlloc(base, size, type, prot); |
| 1588 | |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1589 | if (W32S_WINE2APP(result)) |
| 1590 | context->Edx = W32S_WINE2APP(result), |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1591 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1592 | else |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1593 | context->Edx = 0, |
| 1594 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
| 1595 | TRACE("VirtualAlloc16: returning base %lx\n", context->Edx); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1596 | } |
| 1597 | break; |
| 1598 | |
| 1599 | |
| 1600 | case 0x001E: /* VirtualFree called from 16-bit code */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1601 | /* |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1602 | * Input: EDX: Segmented address of arguments on stack |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1603 | * |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1604 | * LPVOID base [in] Flat address of region |
| 1605 | * DWORD size [in] Size of region |
| 1606 | * DWORD type [in] Type of operation |
| 1607 | * |
| 1608 | * Output: EAX: NtStatus |
| 1609 | * EDX: TRUE if success, FALSE if failure |
| 1610 | */ |
| 1611 | { |
Alexandre Julliard | 982a223 | 2000-12-13 20:20:09 +0000 | [diff] [blame] | 1612 | DWORD *stack = MapSL( MAKESEGPTR( LOWORD(context->Edx), HIWORD(context->Edx) )); |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1613 | LPVOID base = (LPVOID)W32S_APP2WINE(stack[0]); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1614 | DWORD size = stack[1]; |
| 1615 | DWORD type = stack[2]; |
| 1616 | DWORD result; |
| 1617 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1618 | TRACE("VirtualFree16(%lx, %lx, %lx)\n", |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1619 | (DWORD)base, size, type); |
| 1620 | |
| 1621 | result = VirtualFree(base, size, type); |
| 1622 | |
| 1623 | if (result) |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1624 | context->Edx = TRUE, |
| 1625 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1626 | else |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1627 | context->Edx = FALSE, |
| 1628 | context->Eax = STATUS_NO_MEMORY; /* FIXME */ |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1629 | } |
| 1630 | break; |
| 1631 | |
| 1632 | |
| 1633 | case 0x001F: /* FWorkingSetSize */ |
| 1634 | /* |
| 1635 | * Input: EDX: 0 if Get, 1 if Set |
| 1636 | * |
| 1637 | * ECX: Get: Buffer to receive Working Set Size |
| 1638 | * Set: Buffer containing Working Set Size |
| 1639 | * |
| 1640 | * Output: NtStatus |
| 1641 | */ |
| 1642 | { |
Alexandre Julliard | becb9a3 | 2000-12-11 03:48:15 +0000 | [diff] [blame] | 1643 | DWORD *ptr = (DWORD *)W32S_APP2WINE(context->Ecx); |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1644 | BOOL set = context->Edx; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 1645 | |
Alexandre Julliard | 61fece0 | 1999-06-26 19:09:08 +0000 | [diff] [blame] | 1646 | TRACE("FWorkingSetSize(%lx, %lx)\n", (DWORD)ptr, (DWORD)set); |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1647 | |
| 1648 | if (set) |
| 1649 | /* We do it differently ... */; |
| 1650 | else |
| 1651 | *ptr = 0x100; |
| 1652 | |
Alexandre Julliard | d8fab2e | 2000-09-25 23:53:07 +0000 | [diff] [blame] | 1653 | context->Eax = STATUS_SUCCESS; |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1654 | } |
| 1655 | break; |
| 1656 | |
| 1657 | |
| 1658 | default: |
| 1659 | VXD_BARF( context, "W32S" ); |
| 1660 | } |
| 1661 | |
Alexandre Julliard | 642d313 | 1998-07-12 19:29:36 +0000 | [diff] [blame] | 1662 | } |