blob: 2308d60fe2a5dc4f9cf555b5817457a2dd486569 [file] [log] [blame]
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +00001/*
2 * Kernel initialization code
Alexandre Julliard0799c1a2002-03-09 23:29:33 +00003 *
4 * Copyright 2000 Alexandre Julliard
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
Jonathan Ernst360a3f92006-05-18 14:49:52 +020018 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +000019 */
20
Patrik Stridvall33929be2001-07-18 21:04:23 +000021#include "config.h"
Patrik Stridvall9aab47e2002-08-28 23:42:34 +000022#include "wine/port.h"
Patrik Stridvall33929be2001-07-18 21:04:23 +000023
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +000024#include <assert.h>
Andreas Mohr04a8eda2000-10-17 00:27:47 +000025#include <ctype.h>
Alexandre Julliarde37c6e12003-09-05 23:08:26 +000026#include <stdarg.h>
James Juranf4d5fef2001-01-26 20:43:40 +000027#include <string.h>
Eric Pouech3d4d7e02002-07-31 18:46:09 +000028#include <signal.h>
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +000029
Alexandre Julliarde37c6e12003-09-05 23:08:26 +000030#include "windef.h"
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +000031#include "winbase.h"
Alexandre Julliard99d8b7f2002-07-31 19:19:36 +000032#include "wincon.h"
Patrik Stridvall9c1de6d2002-09-12 22:07:02 +000033#include "winternl.h"
Alexandre Julliard628939d2005-08-10 10:59:19 +000034#include "wownt32.h"
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +000035
Alexandre Julliard59008672002-05-16 20:32:16 +000036#include "wine/library.h"
Alexandre Julliard3be5d622003-08-25 00:56:37 +000037#include "kernel_private.h"
Eric Pouechd7d98362002-09-04 18:41:03 +000038#include "console_private.h"
Alexandre Julliard3baf4db2007-05-07 17:10:18 +020039#include "wine/debug.h"
40
41WINE_DEFAULT_DEBUG_CHANNEL(process);
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +000042
Maarten Lankhorst768160e2008-12-16 16:37:46 +010043extern int CDECL __wine_set_signal_handler(unsigned, int (*)(unsigned));
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +000044
Alexandre Julliard40fb7ca2007-07-11 14:04:54 +020045static ULONGLONG server_start_time;
46
Alexandre Julliarde15aadd2003-05-13 00:49:49 +000047/***********************************************************************
Alexandre Julliard3baf4db2007-05-07 17:10:18 +020048 * set_entry_point
49 */
50static void set_entry_point( HMODULE module, const char *name, DWORD rva )
51{
52 IMAGE_EXPORT_DIRECTORY *exports;
53 DWORD exp_size;
54
55 if ((exports = RtlImageDirectoryEntryToData( module, TRUE,
56 IMAGE_DIRECTORY_ENTRY_EXPORT, &exp_size )))
57 {
58 DWORD *functions = (DWORD *)((char *)module + exports->AddressOfFunctions);
59 const WORD *ordinals = (const WORD *)((const char *)module + exports->AddressOfNameOrdinals);
60 const DWORD *names = (const DWORD *)((const char *)module + exports->AddressOfNames);
61 int min = 0, max = exports->NumberOfNames - 1;
62
63 while (min <= max)
64 {
65 int res, pos = (min + max) / 2;
66 const char *ename = (const char *)module + names[pos];
67 if (!(res = strcmp( ename, name )))
68 {
69 WORD ordinal = ordinals[pos];
70 assert( ordinal < exports->NumberOfFunctions );
71 TRACE( "setting %s at %p to %08x\n", name, &functions[ordinal], rva );
72 functions[ordinal] = rva;
73 return;
74 }
75 if (res > 0) max = pos - 1;
76 else min = pos + 1;
77 }
78 }
79}
80
81
82/***********************************************************************
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +000083 * KERNEL process initialisation routine
84 */
Alexandre Julliard3baf4db2007-05-07 17:10:18 +020085static BOOL process_attach( HMODULE module )
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +000086{
Alexandre Julliard40fb7ca2007-07-11 14:04:54 +020087 SYSTEM_TIMEOFDAY_INFORMATION ti;
Alexandre Julliard38440f52006-07-13 14:04:40 +020088 RTL_USER_PROCESS_PARAMETERS *params = NtCurrentTeb()->Peb->ProcessParameters;
Alexandre Julliardcc461322004-06-15 00:52:03 +000089
Alexandre Julliard40fb7ca2007-07-11 14:04:54 +020090 NtQuerySystemInformation( SystemTimeOfDayInformation, &ti, sizeof(ti), NULL );
91 server_start_time = ti.liKeBootTime.QuadPart;
92
Alexandre Julliarda3c45682003-10-14 05:32:30 +000093 /* Setup registry locale information */
94 LOCALE_InitRegistry();
Alexandre Julliard6ce25702000-07-11 22:08:43 +000095
Martin Wilck6d886a52002-11-15 01:01:47 +000096 /* Setup computer name */
97 COMPUTERNAME_Init();
Alexandre Julliard1de20ae2004-03-20 02:28:51 +000098
Alexandre Julliard38440f52006-07-13 14:04:40 +020099 /* convert value from server:
100 * + 0 => INVALID_HANDLE_VALUE
101 * + console handle needs to be mapped
102 */
103 if (!params->hStdInput)
104 params->hStdInput = INVALID_HANDLE_VALUE;
105 else if (VerifyConsoleIoHandle(console_handle_map(params->hStdInput)))
106 params->hStdInput = console_handle_map(params->hStdInput);
107
108 if (!params->hStdOutput)
109 params->hStdOutput = INVALID_HANDLE_VALUE;
110 else if (VerifyConsoleIoHandle(console_handle_map(params->hStdOutput)))
111 params->hStdOutput = console_handle_map(params->hStdOutput);
112
113 if (!params->hStdError)
114 params->hStdError = INVALID_HANDLE_VALUE;
115 else if (VerifyConsoleIoHandle(console_handle_map(params->hStdError)))
116 params->hStdError = console_handle_map(params->hStdError);
117
Eric Pouechb53b5bc2003-06-18 03:23:22 +0000118 /* copy process information from ntdll */
119 ENV_CopyStartupInformation();
Martin Wilck6d886a52002-11-15 01:01:47 +0000120
Alexandre Julliard3baf4db2007-05-07 17:10:18 +0200121 if (!(GetVersion() & 0x80000000))
122 {
123 /* Securom checks for this one when version is NT */
124 set_entry_point( module, "FT_Thunk", 0 );
125 }
Alexandre Julliard00ac7072010-03-01 11:34:34 +0100126 else LoadLibraryA( "krnl386.exe16" );
Alexandre Julliard389b0392001-03-04 01:06:07 +0000127
Eric Pouechd7d98362002-09-04 18:41:03 +0000128 /* finish the process initialisation for console bits, if needed */
Eric Pouech3d4d7e02002-07-31 18:46:09 +0000129 __wine_set_signal_handler(SIGINT, CONSOLE_HandleCtrlC);
130
Alexandre Julliard38440f52006-07-13 14:04:40 +0200131 if (params->ConsoleHandle == (HANDLE)1) /* FIXME */
Alexandre Julliard99d8b7f2002-07-31 19:19:36 +0000132 {
133 HMODULE mod = GetModuleHandleA(0);
134 if (RtlImageNtHeader(mod)->OptionalHeader.Subsystem == IMAGE_SUBSYSTEM_WINDOWS_CUI)
135 AllocConsole();
136 }
Alexandre Julliard626669e2006-02-05 12:24:43 +0100137 /* else TODO for DETACHED_PROCESS:
138 * 1/ inherit console + handles
139 * 2/ create std handles, if handles are not inherited
140 * TBD when not using wineserver handles for console handles
141 */
Eric Pouechb53b5bc2003-06-18 03:23:22 +0000142
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +0000143 return TRUE;
144}
145
146/***********************************************************************
147 * KERNEL initialisation routine
148 */
Alexandre Julliard71967242003-10-10 00:50:56 +0000149BOOL WINAPI DllMain( HINSTANCE hinst, DWORD reason, LPVOID reserved )
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +0000150{
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +0000151 switch(reason)
152 {
153 case DLL_PROCESS_ATTACH:
Alexandre Julliard475b7d22009-12-29 16:24:00 +0100154 DisableThreadLibraryCalls( hinst );
Alexandre Julliard3baf4db2007-05-07 17:10:18 +0200155 return process_attach( hinst );
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +0000156 case DLL_PROCESS_DETACH:
Alexandre Julliard86c60212009-10-08 19:11:58 +0200157 WritePrivateProfileSectionW( NULL, NULL, NULL );
Alexandre Julliard7fe09bc2000-06-03 04:49:40 +0000158 break;
159 }
160 return TRUE;
161}
Alexandre Julliard3850c1a2000-08-06 02:42:46 +0000162
163/***********************************************************************
Alexandre Julliardc349d7d2004-04-28 03:53:19 +0000164 * MulDiv (KERNEL32.@)
165 * RETURNS
166 * Result of multiplication and division
167 * -1: Overflow occurred or Divisor was 0
168 */
169INT WINAPI MulDiv( INT nMultiplicand, INT nMultiplier, INT nDivisor)
170{
171 LONGLONG ret;
172
173 if (!nDivisor) return -1;
174
175 /* We want to deal with a positive divisor to simplify the logic. */
176 if (nDivisor < 0)
177 {
178 nMultiplicand = - nMultiplicand;
179 nDivisor = -nDivisor;
180 }
181
182 /* If the result is positive, we "add" to round. else, we subtract to round. */
183 if ( ( (nMultiplicand < 0) && (nMultiplier < 0) ) ||
184 ( (nMultiplicand >= 0) && (nMultiplier >= 0) ) )
185 ret = (((LONGLONG)nMultiplicand * nMultiplier) + (nDivisor/2)) / nDivisor;
186 else
187 ret = (((LONGLONG)nMultiplicand * nMultiplier) - (nDivisor/2)) / nDivisor;
188
189 if ((ret > 2147483647) || (ret < -2147483647)) return -1;
190 return ret;
191}
Alexandre Julliard9ad56282005-07-14 10:32:46 +0000192
193
Alexandre Julliard40fb7ca2007-07-11 14:04:54 +0200194/******************************************************************************
195 * GetTickCount64 (KERNEL32.@)
196 */
197ULONGLONG WINAPI GetTickCount64(void)
198{
199 LARGE_INTEGER now;
200
201 NtQuerySystemTime( &now );
202 return (now.QuadPart - server_start_time) / 10000;
203}
204
205
Alexandre Julliard9ad56282005-07-14 10:32:46 +0000206/***********************************************************************
Alexandre Julliard9ad56282005-07-14 10:32:46 +0000207 * GetTickCount (KERNEL32.@)
208 *
209 * Get the number of milliseconds the system has been running.
210 *
211 * PARAMS
212 * None.
213 *
214 * RETURNS
215 * The current tick count.
216 *
217 * NOTES
Austin Englishd5de9972008-03-17 15:51:39 -0500218 * The value returned will wrap around every 2^32 milliseconds.
219 * Under Windows, tick 0 is the moment at which the system is rebooted.
220 * Under Wine, tick 0 begins at the moment the wineserver process is started.
Alexandre Julliard9ad56282005-07-14 10:32:46 +0000221 */
222DWORD WINAPI GetTickCount(void)
223{
Alexandre Julliard40fb7ca2007-07-11 14:04:54 +0200224 return GetTickCount64();
Alexandre Julliard9ad56282005-07-14 10:32:46 +0000225}