- Using uniform DocBook tags for DLLs, commands, environment vars...

- Moved detailed memory management from archi to kernel(s) part.
- Enhanced process handling description.
- Added section about 16 bit processes' support.
- Added section about DOS support.
This commit is contained in:
Eric Pouech 2005-04-18 09:53:39 +00:00 committed by Alexandre Julliard
parent 81857b7373
commit 3bebb0fed9
2 changed files with 620 additions and 524 deletions

View File

@ -81,7 +81,7 @@
</listitem> </listitem>
<listitem> <listitem>
<para> <para>
WineLib executable. These are applications, written Winelib executable. These are applications, written
using the Windows API, but compiled as a Unix using the Windows API, but compiled as a Unix
executable. Wine provides the tools to create such executable. Wine provides the tools to create such
executables. executables.
@ -101,7 +101,7 @@
<entry>DOS (.COM or .EXE)</entry> <entry>DOS (.COM or .EXE)</entry>
<entry>Win16 (NE)</entry> <entry>Win16 (NE)</entry>
<entry>Win32 (PE)</entry> <entry>Win32 (PE)</entry>
<entry>WineLib</entry> <entry>Winelib</entry>
</row> </row>
</thead> </thead>
<tbody> <tbody>
@ -188,14 +188,12 @@
</para> </para>
<para> <para>
Wine deals with this issue by launching a separate Wine Wine deals with this issue by launching a separate Wine process (which
process (which is in fact a Unix process) for each Win32 is in fact a Unix process) for each Win32 process, but not for Win16
process, but not for Win16 tasks. Win16 tasks (as well as tasks. Win16 tasks are run as different intersynchronized Unix-threads
DOS programs) are run as different intersynchronized in the same dedicated Wine process; this Wine process is commonly
Unix-threads in the same dedicated Wine process; this Wine known as a <firstterm>WOW</firstterm> process (Windows on Windows),
process is commonly known as a <firstterm>WOW</firstterm> referring to a similar mechanism used by Windows NT.
process (Windows on Windows), referring to a similar
mechanism used by Windows NT.
</para> </para>
<para> <para>
Synchronization between the Win16 tasks running in the WOW Synchronization between the Win16 tasks running in the WOW
@ -206,6 +204,15 @@
one of the waiting threads will then acquire it and let its one of the waiting threads will then acquire it and let its
task run. task run.
</para> </para>
<para>
<command>winevdm</command> is the Wine process dedicated to running the
Win16 processes. Note that several instances of this process could
exist, has Windows has support for different VDM (Virtual Dos
Machines) in order to have Win16 processes running in different
address spaces. Wine also uses the same architecture to run DOS
programs (in this case, the DOS emulation is provided by a Wine only
DLL called <filename>winedos</filename>.
</para>
</sect2> </sect2>
</sect1> </sect1>
@ -251,9 +258,12 @@
<para> <para>
The windows architecture (Windows NT way) looks like the The windows architecture (Windows NT way) looks like the
following drawing. Note the new DLL (NTDLL) which allows following drawing. Note the new DLL
implementing different subsystems (as win32); kernel32 in NT (<filename>NTDLL</filename>) which allows implementing
architecture implements the Win32 subsystem on top of NTDLL. different subsystems (as win32);
<filename>kernel32</filename> in NT architecture
implements the Win32 subsystem on top of
<filename>NTDLL</filename>.
<screen> <screen>
+---------------------+ \ +---------------------+ \
| Windows EXE | } application | Windows EXE | } application
@ -371,12 +381,16 @@
<para> <para>
Wine must at least completely replace the "Big Three" DLLs Wine must at least completely replace the "Big Three" DLLs
(KERNEL/KERNEL32, GDI/GDI32, and USER/USER32), which all (<filename>KERNEL</filename>/<filename>KERNEL32</filename>,
other DLLs are layered on top of. But since Wine is (for <filename>GDI</filename>/<filename>GDI32</filename>, and
various reasons) leaning towards the NT way of implementing <filename>USER</filename>/<filename>USER32</filename>),
things, the NTDLL is another core DLL to be implemented in which all other DLLs are layered on top of. But since Wine
Wine, and many KERNEL32 and ADVAPI32 features will be is (for various reasons) leaning towards the NT way of
implemented through the NTDLL. implementing things, the <filename>NTDLL</filename> is
another core DLL to be implemented in Wine, and many
<filename>KERNEL32</filename> and
<filename>ADVAPI32</filename> features will be
implemented through the <filename>NTDLL</filename>.
</para> </para>
<para> <para>
As of today, no real subsystem (apart the Win32 one) has As of today, no real subsystem (apart the Win32 one) has
@ -421,30 +435,30 @@
Wine server launches, it creates a Unix socket for the Wine server launches, it creates a Unix socket for the
current host based on (see below) your home directory's current host based on (see below) your home directory's
<filename>.wine</filename> subdirectory (or wherever the <filename>.wine</filename> subdirectory (or wherever the
<constant>WINEPREFIX</constant> environment variable <envar>WINEPREFIX</envar> environment variable points to)
points to) - all Wine processes launched later connects to - all Wine processes launched later connects to the Wine
the Wine server using this socket. (If a Wine server was server using this socket. If a Wine server was not
not already running, the first Wine process will start up already running, the first Wine process will start up the
the Wine server in auto-terminate mode (i.e. the Wine Wine server in auto-terminate mode (i.e. the Wine server
server will then terminate itself once the last Wine will then terminate itself once the last Wine process has
process has terminated).) terminated).
</para> </para>
<para> <para>
In earlier versions of Wine the master socket mentioned In earlier versions of Wine the master socket mentioned
above was actually created in the configuration directory; above was actually created in the configuration directory;
either your home directory's <filename>/wine</filename> either your home directory's <filename>/wine</filename>
subdirectory or wherever the subdirectory or wherever the
<constant>WINEPREFIX</constant> environment variable <envar>WINEPREFIX</envar> environment variable points>.
points>. Since that might not be possible the socket is Since that might not be possible the socket is actually
actually created within the <filename>/tmp</filename> created within the <filename>/tmp</filename> directory
directory with a name that reflects the configuration with a name that reflects the configuration directory.
directory. This means that there can actually be several This means that there can actually be several separate
separate copies of the Wine server running; one per copies of the Wine server running; one per combination of
combination of user and configuration directory. Note that user and configuration directory. Note that you should
you should not have several users using the same not have several users using the same configuration
configuration directory at the same time; they will have directory at the same time; they will have different
different copies of the Wine server running and this could copies of the Wine server running and this could well
well lead to problems with the registry information that lead to problems with the registry information that
they are sharing. they are sharing.
</para> </para>
<para> <para>
@ -544,9 +558,9 @@
automatically registered when the shared library is automatically registered when the shared library is
loaded - remember, registration call is put inside a loaded - remember, registration call is put inside a
shared library constructor - using the shared library constructor - using the
<constant>PRELOAD</constant> environment variable <envar>PRELOAD</envar> environment variable when
when running a Wine process can force the running a Wine process can force the registration of
registration of some DLL descriptors. some DLL descriptors.
</para> </para>
</listitem> </listitem>
<listitem> <listitem>
@ -554,8 +568,7 @@
If it's not registered, Wine will look for it on If it's not registered, Wine will look for it on
disk, building the shared library name from the DLL disk, building the shared library name from the DLL
module name. Directory searched for are specified by module name. Directory searched for are specified by
the <constant>WINEDLLPATH</constant> environment the <envar>WINEDLLPATH</envar> environment variable.
variable.
</para> </para>
</listitem> </listitem>
<listitem> <listitem>
@ -647,15 +660,19 @@
A DLL descriptor is also created for every 16 bit A DLL descriptor is also created for every 16 bit
DLL. However, this DLL normally paired with a 32 bit DLL. However, this DLL normally paired with a 32 bit
DLL. Either, it's the 16 bit counterpart of the 16 bit DLL DLL. Either, it's the 16 bit counterpart of the 16 bit DLL
(KRNL386.EXE for KERNEL32, USER for USER32...), or a 16 (<filename>KRNL386.EXE</filename> for
bit DLL directly linked to a 32 bit DLL (like SYSTEM for <filename>KERNEL32</filename>, <filename>USER</filename>
KERNEL32, or DDEML for USER32). In those cases, the 16 bit for <filename>USER32</filename>...), or a 16
bit DLL directly linked to a 32 bit DLL (like
<filename>SYSTEM</filename> for <filename>KERNEL32</filename>,
or <filename>DDEML</filename> for
<filename>USER32</filename>). In those cases, the 16 bit
descriptor(s) is (are) inserted in the same shared library descriptor(s) is (are) inserted in the same shared library
as the the corresponding 32 bit DLL. Wine will also create as the the corresponding 32 bit DLL. Wine will also create
symbolic links between kernel32.dll.so and system.dll.so symbolic links between kernel32.dll.so and system.dll.so
so that loading of either so that loading of either
<filename>kernel32.dll</filename> or <filename>KERNEL32.DLL</filename> or
<filename>system.dll</filename> will end up on the same <filename>SYSTEM.DLL</filename> will end up on the same
shared library. shared library.
</para> </para>
</sect2> </sect2>
@ -675,24 +692,26 @@
<para> <para>
Native DLLs of course guarantee 100% compatibility for Native DLLs of course guarantee 100% compatibility for
routines they implement. For example, using the native USER routines they implement. For example, using the native
DLL would maintain a virtually perfect and Windows 95-like <filename>USER</filename> DLL would maintain a virtually
look for window borders, dialog controls, and so on. Using perfect and Windows 95-like look for window borders,
the built-in Wine version of this library, on the other dialog controls, and so on. Using the built-in Wine
hand, would produce a display that does not precisely mimic version of this library, on the other hand, would produce
that of Windows 95. Such subtle differences can be a display that does not precisely mimic that of Windows
engendered in other important DLLs, such as the common 95. Such subtle differences can be engendered in other
controls library COMMCTRL or the common dialogs library important DLLs, such as the common controls library
COMMDLG, when built-in Wine DLLs outrank other types in load <filename>COMMCTRL</filename> or the common dialogs library
order. <filename>COMMDLG</filename>, when built-in Wine DLLs
outrank other types in load order.
</para> </para>
<para> <para>
More significant, less aesthetically-oriented problems can More significant, less aesthetically-oriented problems can
result if the built-in Wine version of the SHELL DLL is result if the built-in Wine version of the
loaded before the native version of this library. SHELL <filename>SHELL</filename> DLL is loaded before the native
version of this library. <filename>SHELL</filename>
contains routines such as those used by installer utilities contains routines such as those used by installer utilities
to create desktop shortcuts. Some installers might fail when to create desktop shortcuts. Some installers might fail when
using Wine's built-in SHELL. using Wine's built-in <filename>SHELL</filename>.
</para> </para>
</sect3> </sect3>
@ -704,16 +723,17 @@
a library tries to access features of the rest of the system a library tries to access features of the rest of the system
that are not fully implemented in Wine, the native DLL might that are not fully implemented in Wine, the native DLL might
work much worse than the corresponding built-in one, if at work much worse than the corresponding built-in one, if at
all. For example, the native Windows GDI library must be all. For example, the native Windows <filename>GDI</filename>
paired with a Windows display driver, which of course is not library must be paired with a Windows display driver, which
present under Intel Unix and Wine. of course is not present under Intel Unix and Wine.
</para> </para>
<para> <para>
Finally, occasionally built-in Wine DLLs implement more Finally, occasionally built-in Wine DLLs implement more
features than the corresponding native Windows DLLs. features than the corresponding native Windows DLLs.
Probably the most important example of such behavior is the Probably the most important example of such behavior is the
integration of Wine with X provided by Wine's built-in USER integration of Wine with X provided by Wine's built-in
DLL. Should the native Windows USER library take load-order <filename>USER</filename> DLL. Should the native Windows
<filename>USER</filename> library take load-order
precedence, such features as the ability to use the precedence, such features as the ability to use the
clipboard or drag-and-drop between Wine windows and X clipboard or drag-and-drop between Wine windows and X
windows will be lost. windows will be lost.
@ -820,10 +840,11 @@
</para> </para>
<para> <para>
The EXE file itself is usually mapped at address 0x400000 The EXE file itself is usually mapped at address
and up: indeed, most EXEs have their relocation records <constant>0x400000</constant> and up: indeed, most EXEs have
stripped which means they must be loaded at their base their relocation records stripped which means they must be
address and cannot be loaded at any other address. loaded at their base address and cannot be loaded at any
other address.
</para> </para>
<para> <para>
@ -836,7 +857,8 @@
be loaded at one address in one process, and a totally be loaded at one address in one process, and a totally
different one in another. Ensuring all the functions different one in another. Ensuring all the functions
loaded into memory can find each other is the job of the loaded into memory can find each other is the job of the
Windows dynamic linker, which is a part of NTDLL. Windows dynamic linker, which is a part of
<filename>NTDLL</filename>.
</para> </para>
<para> <para>
So, we have the EXE and its DLLs mapped into memory. Two So, we have the EXE and its DLLs mapped into memory. Two
@ -864,16 +886,16 @@
available for the application. In fact that's not so: only available for the application. In fact that's not so: only
the lower 2 gigs are available, the upper 2 gigs are on the lower 2 gigs are available, the upper 2 gigs are on
Windows NT used by the operating system and hold the Windows NT used by the operating system and hold the
kernel (from 0x80000000). Why is the kernel mapped into kernel (from <constant>0x80000000</constant>). Why is the
every address space? Mostly for performance: while it's kernel mapped into every address space? Mostly for
possible to give the kernel its own address space too - performance: while it's possible to give the kernel its own
this is what Ingo Molnars 4G/4G VM split patch does for address space too - this is what Ingo Molnars 4G/4G VM
Linux - it requires that every system call into the kernel split patch does for Linux - it requires that every system
switches address space. As that is a fairly expensive call into the kernel switches address space. As that is a
operation (requires flushing the translation lookaside fairly expensive operation (requires flushing the
buffers etc) and syscalls are made frequently it's best translation lookaside buffers etc) and syscalls are made
avoided by keeping the kernel mapped at a constant frequently it's best avoided by keeping the kernel mapped
position in every processes address space. at a constant position in every processes address space.
</para> </para>
<para> <para>
@ -923,403 +945,8 @@
memory allocation and stack. memory allocation and stack.
</para> </para>
</sect3> </sect3>
<sect3>
<title>Implementation</title>
<para>
Wine (with a bit of black magic) is able to map all items
at the correct locations as depicted above.
</para>
<para>
Wine also implements the shared heap so native win9x DLLs
can be used. This heap is always created at the
<constant>SYSTEM_HEAP_BASE</constant> address or
<constant>0x80000000</constant> and defaults to 16
megabytes in size.
</para>
<para>
There are a few other magic locations. The bottom 64k of
memory is deliberately left unmapped to catch null pointer
dereferences. The region from 64k to 1mb+64k are reserved
for DOS compatibility and contain various DOS data
structures. Finally, the address space also contains
mappings for the Wine binary itself, any native libaries
Wine is using, the glibc malloc arena and so on.
</para>
</sect3>
<sect3 id="address-space">
<title>Laying out the address space</title>
<para>
Up until about the start of 2004, the Linux address space
very much resembled the Windows 9x layout: the kernel sat
in the top gigabyte, the bottom pages were unmapped to
catch null pointer dereferences, and the rest was
free. The kernels mmap algorithm was predictable: it would
start by mapping files at low addresses and work up from
there.
</para>
<para>
The development of a series of new low level patches
violated many of these assumptions, and resulted in Wine
needing to force the Win32 address space layout upon the
system. This section looks at why and how this is done.
</para>
<para>
The exec-shield patch increases security by randomizing
the kernels mmap algorithms. Rather than consistently
choosing the same addresses given the same sequence of
requests, the kernel will now choose randomized
addresses. Because the Linux dynamic linker
(ld-linux.so.2) loads DSOs into memory by using mmap, this
means that DSOs are no longer loaded at predictable
addresses, so making it harder to attack software by using
buffer overflows. It also attempts to relocate certain
binaries into a special low area of memory known as the
ASCII armor so making it harder to jump into them when
using string based attacks.
</para>
<para>
Prelink is a technology that enhances startup times by
precalculating ELF global offset tables then saving the
results inside the native binaries themselves. By grid
fitting each DSO into the address space, the dynamic
linker does not have to perform as many relocations so
allowing applications that heavily rely on dynamic linkage
to be loaded into memory much quicker. Complex C++
applications such as Mozilla, OpenOffice and KDE can
especially benefit from this technique.
</para>
<para>
The 4G VM split patch was developed by Ingo Molnar. It
gives the Linux kernel its own address space, thereby
allowing processes to access the maximum addressable
amount of memory on a 32-bit machine: 4 gigabytes. It
allows people with lots of RAM to fully utilise that in
any given process at the cost of performance: the reason
behind giving the kernel a part of each processes address
space was to avoid the overhead of switching on each
syscall.
</para>
<para>
Each of these changes alter the address space in a way
incompatible with Windows. Prelink and exec-shield mean
that the libraries Wine uses can be placed at any point in
the address space: typically this meant that a library was
sitting in the region that the EXE you wanted to run had
to be loaded (remember that unlike DLLs, EXE files cannot
be moved around in memory). The 4G VM split means that
programs could receive pointers to the top gigabyte of
address space which some are not prepared for (they may
store extra information in the high bits of a pointer, for
instance). In particular, in combination with exec-shield
this one is especially deadly as it's possible the process
heap could be allocated beyond ADDRESS_SPACE_LIMIT which
causes Wine initialization to fail.
</para>
<para>
The solution to these problems is for Wine to reserve
particular parts of the address space so that areas that
we don't want the system to use will be avoided. We later
on (re/de)allocate those areas as needed. One problem is
that some of these mappings are put in place automatically
by the dynamic linker: for instance any libraries that
Wine is linked to (like libc, libwine, libpthread etc)
will be mapped into memory before Wine even gets
control. In order to solve that, Wine overrides the
default ELF initialization sequence at a low level and
reserves the needed areas by using direct syscalls into
the kernel (ie without linking against any other code to
do it) before restarting the standard initialization and
letting the dynamic linker continue. This is referred to
as the preloader and is found in loader/preloader.c.
</para>
<para>
Once the usual ELF boot sequence has been completed, some
native libraries may well have been mapped above the 3gig
limit: however, this doesn't matter as 3G is a Windows
limit, not a Linux limit. We still have to prevent the
system from allocating anything else above there (like the
heap or other DLLs) though so Wine performs a binary
search over the upper gig of address space in order to
iteratively fill in the holes with MAP_NORESERVE mappings
so the address space is allocated but the memory to
actually back it is not. This code can be found in libs/wine/mmap.c:reserve_area.
</para>
</sect3>
</sect2> </sect2>
<sect2>
<title>Processes</title>
<para>
Let's take a closer look at the way Wine loads and run
processes in memory.
</para>
<sect3>
<title>Starting a process from command line</title>
<para>
When starting a Wine process from command line (we'll get
later on to the differences between NE, PE and Winelib
executables), there are a couple of things Wine need to do
first. A first executable is run to check the threading
model of the underlying OS (see <xref linkend="threading">
for the details) and will start the real Wine loader
corresponding to the choosen threading model.
</para>
<para>
Then Wine graps a few elements from the Unix world: the
environment, the program arguments. Then the
<filename>ntdll.dll.so</filename> is loaded into memory
using the standard shared library dynamic loader. When
loaded, NTDLL will mainly first create a decent Windows
environment:
<itemizedlist>
<listitem>
<para>create a PEB and a TEB</para>
</listitem>
<listitem>
<para>
set up the connection to the Wine server - and
eventually launching the Wine server if none runs
</para>
</listitem>
<listitem>
<para>create the process heap</para>
</listitem>
</itemizedlist>
</para>
<para>
Then <filename>Kernel32</filename> is loaded (but now
using the Windows dynamic loading capabilities) and a Wine
specific entry point is called
<function>__wine_kernel_init</function>. This function
will actually handle all the logic of the process loading
and execution, and will never return from it's call.
</para>
<para>
<function>__wine_kernel_init</function> will undergo the
following tasks:
<itemizedlist>
<listitem>
<para>
initialization of program arguments from Unix
program arguments
</para>
</listitem>
<listitem>
<para>
lookup of executable in the file system
</para>
</listitem>
<listitem>
<para>
If the file is not found, then an error is printed
and the Wine loader stops.
</para>
</listitem>
<listitem>
<para>
We'll cover the non-PE file type later on, so assume
for now it's a PE file. The PE module is loaded in
memory using the Windows shared library
mechanism. Note that the dependencies on the module
are not resolved at this point.
</para>
</listitem>
<listitem>
<para>
A new stack is created, which size is given in the
PE header, and this stack is made the one of the
running thread (which is still the only one in the
process). The stack used at startup will no longer
be used.
</para>
</listitem>
<listitem>
<para>
Which this new stack,
<function>ntdll.LdrInitializeThunk</function> is
called which performs the remaining initialization
parts, including resolving all the DLL imports on
the PE module, and doing the init of the TLS slots.
</para>
</listitem>
<listitem>
<para>
Control can now be passed to the
<function>EntryPoint</function> of the PE module,
which will let the executable run.
</para>
</listitem>
</itemizedlist>
</para>
</sect3>
<sect3>
<title>Creating a child process from a running process</title>
<para>
The steps used are closely link to what is done in the
previous case.
</para>
<para>
There are however a few points to look at a bit more
closely. The inner implementation creates the child
process using the <function>fork()</function> and
<function>exec()</function> calls. This means that we
don't need to check again for the threading model, we can
use what the parent (or the grand-parent process...)
started from command line has found.
</para>
<para>
The Win32 process creation allows to pass a lot of
information between the parent and the child. This
includes object handles, windows title, console
parameters, environment strings... Wine makes use of both
the standard Unix inheritance mechanisms (for environment
for example) and the Wine server (to pass from parent to
child a chunk of data containing the relevant information).
</para>
<para>
The previously described loading mechanism will check in
the Wine server if such a chunk exists, and, if so, will
perform the relevant initialization.
</para>
<para>
Some further synchronization is also put in place: a
parent will wait until the child has started, or has
failed. The Wine server is also used to perform those
tasks.
</para>
</sect3>
<sect3>
<title>Starting a Winelib process</title>
<para>
Before going into the gory details, let's first go back to
what a Winelib application is. It can be either a regular
Unix executable, or a more specific Wine beast. This later
form in fact creates two files for a given executable (say
<filename>foo.exe</filename>). The first one, named
<filename>foo</filename> will be a symbolic link to the
Wine loader (<filename>wine</filename>). The second one,
named <filename>foo.exe.so</filename>, is the equivalent
of the <filename>.dll.so</filename> files we've already
described for DLLs. As in Windows, an executable is, among
other things, a module with its import and export
information, as any DLL, it makes sense Wine uses the same
mechanisms for loading native executables and DLLs.
</para>
<para>
When starting a Winelib application from the command line
(say with <command>foo arg1 arg2</command>), the Unix
shell will execute <command>foo</command> as a Unix
executable. Since this is in fact the Wine loader, Wine
will fire up. However, will notice that it hasn't been
started as <command>wine</command> but as
<command>foo</command>, and hence, will try to load (using
Unix shared library mechanism) the second file
<filename>foo.exe.so</filename>. Wine will recognize a 32
bit module (with its descriptor) embedded in the shared
library, and once the shared library loaded, it will
proceed the same path as when loading a standard native PE
executable.
</para>
<para>
Wine needs to implement this second form of executable in
order to maintain the order of initialization of some
elements in the executable. One particular issue is when
dealing with global C++ objects. In standard Unix
executable, the call of the constructor to such objects is
stored in the specific section of the executable
(<function>.init</function> not to name it). All
constructors in this section are called before the
<function>main</function> function is called. Creating a
Wine executable using the first form mentionned above will
let those constructors being called before Wine gets a
chance to initialize itself. So, any constructor using a
Windows API will fail, because Wine infrastructure isn't
in place. The use of the second form for Winelib
executables ensures that we do the initialization using
the following steps:
<itemizedlist>
<listitem>
<para>
initialize the Wine infrastructure
</para>
</listitem>
<listitem>
<para>
load the executable into memory
</para>
</listitem>
<listitem>
<para>
handle the import sections for the executable
</para>
</listitem>
<listitem>
<para>
call the global object constructors (if any). They
now can properly call the Windows APIs
</para>
</listitem>
<listitem>
<para>
call the executable entry point
</para>
</listitem>
</itemizedlist>
</para>
<para>
The attentive reader would have noted that the resolution
of imports for the executable is done, as for a DLL, when
the executable/DLL descriptor is registered. However, this
is done also by adding a specific constructor in the
<function>.init</function> section. For the above describe
scheme to function properly, this constructor must be the
first constructor to be called, before all the other
constructors, generated by the executable itself. The Wine
build chain takes care of that, and also generating the
executable/DLL descriptor for the Winelib executable.
</para>
</sect3>
<sect3>
<title>Starting a NE (Win16) process</title>
<para>
When Wine is requested to run a NE (Win 16 process), it
will in fact hand over the execution of it to a specific
executable <filename>winevdm</filename>. VDM stands for
Virtual DOS Machine. This <filename>winevdm</filename>
will in fact set up the correct 16 bit environment to run
the executable. Any new 16 bit process created by this
executable (or its children) will run into the same
<filename>winevdm</filename> instance. Among one instance,
several functionalities will be provided to those 16 bit
processes, including the cooperative multitasking, sharing
the same address space, managing the selectors for the 16
bit segments needed for code, data and stack.
</para>
<para>
Note that several <filename>winevdm</filename> instances
can run in the same Wine session, but the functionalities
described above are only shared among a given instance,
not among all the instances. <filename>winevdm</filename>
is built as Winelib application, and hence has access to
any facility a 32 bit application has.
</para>
<para>
The behaviour we just described also applies to DOS
executables, which are handled the same way by
<filename>winevdm</filename>.
</para>
</sect3>
</sect2>
<sect2> <sect2>
<title>Wine drivers</title> <title>Wine drivers</title>
<para> <para>
@ -1327,11 +954,13 @@
Unix. This comes mainly because (look at the generic Unix. This comes mainly because (look at the generic
architecture schemas) Wine doesn't implement the kernel architecture schemas) Wine doesn't implement the kernel
features of Windows (kernel here really means the kernel, features of Windows (kernel here really means the kernel,
not the KERNEL32 DLL), but rather sets up a proxy layer on not the <filename>KERNEL32</filename> DLL), but rather
top of the Unix kernel to provide the NTDLL and KERNEL32 sets up a proxy layer on top of the Unix kernel to provide
features. This means that Wine doesn't provide the inner the <filename>NTDLL</filename> and
infrastructure to run native drivers, either from the Win9x <filename>KERNEL32</filename> features. This means that
family or from the NT family. Wine doesn't provide the inner infrastructure to run
native drivers, either from the Win9x family or from the
NT family.
</para> </para>
<para> <para>
In other words, Wine will only be able to provide access to In other words, Wine will only be able to provide access to

View File

@ -202,6 +202,361 @@ ExitProcess( entry( peb ) );
</sect2> </sect2>
</sect1> </sect1>
<sect1>
<title>Detailed memory management</title>
<para>
As already explained in previous chapter (see <xref linkend="arch-mem">
for the details), Wine creates every 32-bit Windows process in its own
32 address space. Wine also tries to map at the relevant addresses what
Windows would do. There are however a few nasty bits to look at.
</para>
<sect2>
<title>Implementation</title>
<para>
Wine (with a bit of black magic) is able to map the main module at
it's desired address (likely <constant>0x400000</constant>), to create
the process heap, its stack (as a Windows executable can ask for a
specific stack size), Wine simply use the initial stack of the ELF
executable for its initialisation, but creates a new stack (as a Win32
one) for the main thread of the executable. Wine also tries to map all
native DLLs at their desired address, so that no relocation has to be
performed.
</para>
<para>
Wine also implements the shared heap so native win9x DLLs can be
used. This heap is always created at the
<constant>SYSTEM_HEAP_BASE</constant> address or
<constant>0x80000000</constant> and defaults to 16 megabytes in size.
</para>
<para>
There are a few other magic locations. The bottom 64k of memory is
deliberately left unmapped to catch null pointer dereferences. The
region from 64k to 1mb+64k are reserved for DOS compatibility and
contain various DOS data structures. Finally, the address space also
contains mappings for the Wine binary itself, any native libaries
Wine is using, the glibc malloc arena and so on.
</para>
</sect2>
<sect2 id="address-space">
<title>Laying out the address space</title>
<para>
Up until about the start of 2004, the Linux address space very much
resembled the Windows 9x layout: the kernel sat in the top gigabyte,
the bottom pages were unmapped to catch null pointer dereferences, and
the rest was free. The kernels mmap algorithm was predictable: it
would start by mapping files at low addresses and work up from there.
</para>
<para>
The development of a series of new low level patches violated many of
these assumptions, and resulted in Wine needing to force the Win32
address space layout upon the system. This section looks at why and
how this is done.
</para>
<para>
The exec-shield patch increases security by randomizing the kernels
mmap algorithms. Rather than consistently choosing the same addresses
given the same sequence of requests, the kernel will now choose
randomized addresses. Because the Linux dynamic linker
(<filename>ld-linux.so.2</filename>) loads DSOs into memory by using
mmap, this means that DSOs are no longer loaded at predictable
addresses, so making it harder to attack software by using buffer
overflows. It also attempts to relocate certain binaries into a
special low area of memory known as the ASCII armor so making it
harder to jump into them when using string based attacks.
</para>
<para>
Prelink is a technology that enhances startup times by precalculating
ELF global offset tables then saving the results inside the native
binaries themselves. By grid fitting each DSO into the address space,
the dynamic linker does not have to perform as many relocations so
allowing applications that heavily rely on dynamic linkage to be
loaded into memory much quicker. Complex C++ applications such as
Mozilla, OpenOffice and KDE can especially benefit from this
technique.
</para>
<para>
The 4G VM split patch was developed by Ingo Molnar. It gives the Linux
kernel its own address space, thereby allowing processes to access the
maximum addressable amount of memory on a 32-bit machine: 4
gigabytes. It allows people with lots of RAM to fully utilise that in
any given process at the cost of performance: the reason behind giving
the kernel a part of each processes address space was to avoid the
overhead of switching on each syscall.
</para>
<para>
Each of these changes alter the address space in a way incompatible
with Windows. Prelink and exec-shield mean that the libraries Wine
uses can be placed at any point in the address space: typically this
meant that a library was sitting in the region that the EXE you wanted
to run had to be loaded (remember that unlike DLLs, EXE files cannot
be moved around in memory). The 4G VM split means that programs could
receive pointers to the top gigabyte of address space which some are
not prepared for (they may store extra information in the high bits of
a pointer, for instance). In particular, in combination with
exec-shield this one is especially deadly as it's possible the process
heap could be allocated beyond
<constant>ADDRESS_SPACE_LIMIT</constant> which causes Wine
initialization to fail.
</para>
<para>
The solution to these problems is for Wine to reserve particular parts
of the address space so that areas that we don't want the system to
use will be avoided. We later on (re/de)allocate those areas as
needed. One problem is that some of these mappings are put in place
automatically by the dynamic linker: for instance any libraries that
Wine is linked to (like <filename>libc</filename>,
<filename>libwine</filename>, <filename>libpthread</filename> etc)
will be mapped into memory before Wine even gets control. In order to
solve that, Wine overrides the default ELF initialization sequence at
a low level and reserves the needed areas by using direct syscalls
into the kernel (ie without linking against any other code to do it)
before restarting the standard initialization and letting the dynamic
linker continue. This is referred to as the preloader and is found in
<filename>loader/preloader.c</filename>.
</para>
<para>
Once the usual ELF boot sequence has been completed, some native
libraries may well have been mapped above the 3gig limit: however,
this doesn't matter as 3G is a Windows limit, not a Linux limit. We
still have to prevent the system from allocating anything else above
there (like the heap or other DLLs) though so Wine performs a binary
search over the upper gig of address space in order to iteratively
fill in the holes with <constant>MAP_NORESERVE</constant> mappings so
the address space is allocated but the memory to actually back it is
not. This code can be found in
<filename>libs/wine/mmap.c</filename>:<function>reserve_area</function>.
</para>
</sect2>
</sect1>
<sect1>
<title>Multi-processing in Wine</title>
<para>
Let's take a closer look at the way Wine loads and run processes in memory.
</para>
<sect2>
<title>Starting a process from command line</title>
<para>
When starting a Wine process from command line (we'll get later on to
the differences between NE, PE and Winelib executables), there are a
couple of things Wine need to do first. A first executable is run to
check the threading model of the underlying OS (see
<xref linkend="threading"> for the details) and will start the real
Wine loader corresponding to the choosen threading model.
</para>
<para>
Then Wine graps a few elements from the Unix world: the environment,
the program arguments. Then the <filename>ntdll.dll.so</filename> is
loaded into memory using the standard shared library dynamic
loader. When loaded, <filename>NTDLL</filename> will mainly first
create a decent Windows environment:
<itemizedlist>
<listitem>
<para>
create a PEB (Process Environment Block) and a TEB (Thread
Environment Block).
</para>
</listitem>
<listitem>
<para>
set up the connection to the Wine server - and eventually
launching the Wine server if none runs
</para>
</listitem>
<listitem>
<para>create the process heap</para>
</listitem>
</itemizedlist>
</para>
<para>
Then <filename>Kernel32</filename> is loaded (but now using the
Windows dynamic loading capabilities) and a Wine specific entry point
is called <function>__wine_kernel_init</function>. This function will
actually handle all the logic of the process loading and execution,
and will never return from it's call.
</para>
<para>
<function>__wine_kernel_init</function> will undergo the following
tasks:
<itemizedlist>
<listitem>
<para>
initialization of program arguments from Unix program arguments
</para>
</listitem>
<listitem>
<para>lookup of executable in the file system</para>
</listitem>
<listitem>
<para>
If the file is not found, then an error is printed and the Wine
loader stops.
</para>
</listitem>
<listitem>
<para>
We'll cover the non-PE file type later on, so assume for now
it's a PE file. The PE module is loaded in memory using the
same mechanisms as for a native DLLs (mainly mapping the file
data and code sections into memory, and handling relocation if
needed). Note that the dependencies on the module are not
resolved at this point.
</para>
</listitem>
<listitem>
<para>
A new stack is created, which size is given in the PE header,
and this stack is made the one of the running thread (which is
still the only one in the process). The stack used at startup
will no longer be used.
</para>
</listitem>
<listitem>
<para>
Which this new stack,
<function>ntdll.LdrInitializeThunk</function> is called which
performs the remaining initialization parts, including resolving
all the DLL imports on the PE module, and doing the init of the
TLS slots.
</para>
</listitem>
<listitem>
<para>
Control can now be passed to the <function>EntryPoint</function>
of the PE module, which will let the executable run.
</para>
</listitem>
</itemizedlist>
</para>
</sect2>
<sect2>
<title>Creating a child process from a running process</title>
<para>
The steps used are closely link to what is done in the previous case.
</para>
<para>
There are however a few points to look at a bit more closely. The
inner implementation creates the child process using the
<function>fork()</function> and <function>exec()</function>
calls. This means that we don't need to check again for the threading
model, we can use what the parent (or the grand-parent process...)
started from command line has found.
</para>
<para>
The Win32 process creation allows to pass a lot of information between
the parent and the child. This includes object handles, windows title,
console parameters, environment strings... Wine makes use of both the
standard Unix inheritance mechanisms (for environment for example) and
the Wine server (to pass from parent to child a chunk of data
containing the relevant information).
</para>
<para>
The previously described loading mechanism will check in the Wine
server if such a chunk exists, and, if so, will perform the relevant
initialization.
</para>
<para>
Some further synchronization is also put in place: a parent will wait
until the child has started, or has failed. The Wine server is also
used to perform those tasks.
</para>
</sect2>
<sect2>
<title>Starting a Winelib process</title>
<para>
Before going into the gory details, let's first go back to what a
Winelib application is. It can be either a regular Unix executable, or
a more specific Wine beast. This later form in fact creates two files
for a given executable (say <filename>foo.exe</filename>). The first
one, named <filename>foo</filename> will be a symbolic link to the
Wine loader (<filename>wine</filename>). The second one, named
<filename>foo.exe.so</filename>, is the equivalent of the
<filename>.dll.so</filename> files we've already described for
DLLs. As in Windows, an executable is, among other things, a module
with its import and export information, as any DLL, it makes sense
Wine uses the same mechanisms for loading native executables and
DLLs.
</para>
<para>
When starting a Winelib application from the command line (say with
<command>foo arg1 arg2</command>), the Unix shell will execute
<command>foo</command> as a Unix executable. Since this is in fact the
Wine loader, Wine will fire up. However, will notice that it hasn't
been started as <command>wine</command> but as <command>foo</command>,
and hence, will try to load (using Unix shared library mechanism) the
second file <filename>foo.exe.so</filename>. Wine will recognize a 32
bit module (with its descriptor) embedded in the shared library, and
once the shared library loaded, it will proceed the same path as when
loading a standard native PE executable.
</para>
<para>
Wine needs to implement this second form of executable in order to
maintain the order of initialization of some elements in the
executable. One particular issue is when dealing with global C++
objects. In standard Unix executable, the call of the constructor to
such objects is stored in the specific section of the executable
(<function>.init</function> not to name it). All constructors in this
section are called before the <function>main()</function> or
<function>WinMain</function> function is called. Creating a Wine
executable using the first form mentionned above will let those
constructors being called before Wine gets a chance to initialize
itself. So, any constructor using a Windows API will fail, because
Wine infrastructure isn't in place. The use of the second form for
Winelib executables ensures that we do the initialization using the
following steps:
<itemizedlist>
<listitem>
<para>
initialize the Wine infrastructure
</para>
</listitem>
<listitem>
<para>
load the executable into memory
</para>
</listitem>
<listitem>
<para>
handle the import sections for the executable
</para>
</listitem>
<listitem>
<para>
call the global object constructors (if any). They now can
properly call the Windows APIs
</para>
</listitem>
<listitem>
<para>
call the executable entry point
</para>
</listitem>
</itemizedlist>
</para>
<para>
The attentive reader would have noted that the resolution of imports
for the executable is done, as for a DLL, when the executable/DLL
descriptor is registered. However, this is done also by adding a
specific constructor in the <function>.init</function> section. For
the above describe scheme to function properly, this constructor must
be the first constructor to be called, before all the other
constructors, generated by the executable itself. The Wine build chain
takes care of that, and also generating the executable/DLL descriptor
for the Winelib executable.
</para>
</sect2>
</sect1>
<sect1 id="threading"> <sect1 id="threading">
<title>Multi-threading in Wine</title> <title>Multi-threading in Wine</title>
@ -313,40 +668,6 @@ if (res != ERROR_SUCCESS) return res;
</para> </para>
</sect2> </sect2>
<sect2>
<title>SysLevels</title>
<para>
SysLevels are an undocumented Windows-internal thread-safety
system. They are basically critical sections which must be taken in a
particular order. The mechanism is generic but there are always three
syslevels: level 1 is the Win16 mutex, level 2 is the
<filename>USER</filename> mutex and level 3 is the
<filename>GDI</filename> mutex.
</para>
<para>
When entering a syslevel, the code (in
<filename>dlls/kernel/syslevel.c</filename>) will check that a higher
syslevel is not already held and produce an error if so. This is
because it's not legal to enter level 2 while holding level 3 - first,
you must leave level 3.
</para>
<para>
Throughout the code you may see calls to
<function>_ConfirmSysLevel()</function> and
<function>_CheckNotSysLevel()</function>. These functions are
essentially assertions about the syslevel states and can be used to
check that the rules have not been accidentally violated. In
particular, <function>_CheckNotSysLevel()</function> will break
probably into the debugger) if the check fails. If this happens the
solution is to get a backtrace and find out, by reading the source of
the wine functions called along the way, how Wine got into the invalid
state.
</para>
</sect2>
<sect2> <sect2>
<title>POSIX threading vs. kernel threading</title> <title>POSIX threading vs. kernel threading</title>
@ -393,7 +714,7 @@ if (res != ERROR_SUCCESS) return res;
glibc, for instance. Worse, pthreads and pure kernel threads had glibc, for instance. Worse, pthreads and pure kernel threads had
strange interactions when run in the same process yet some libraries strange interactions when run in the same process yet some libraries
used by Wine used pthreads internally. Throw in source code porting used by Wine used pthreads internally. Throw in source code porting
using WineLib - where you have both UNIX and Win32 code in the same using Winelib - where you have both UNIX and Win32 code in the same
process - and chaos was the result. process - and chaos was the result.
</para> </para>
@ -651,15 +972,16 @@ if (res != ERROR_SUCCESS) return res;
<para> <para>
Windows exceptions typically contain more information than the Unix Windows exceptions typically contain more information than the Unix
standard APIs provide. For instance, a standard APIs provide. For instance, a
<constant>STATUS_ACCESS_VIOLATION</constant> exception (0xC0000005) <constant>STATUS_ACCESS_VIOLATION</constant> exception
structure contains the faulting address, whereas a standard Unix (<constant>0xC0000005</constant>) structure contains the faulting
SIGSEGV just tells the app that it crashed. Usually this information address, whereas a standard Unix <constant>SIGSEGV</constant> just
is passed as an extra parameter to the signal handler, however its tells the app that it crashed. Usually this information is passed as
location and contents vary between kernels (BSD, Solaris, an extra parameter to the signal handler, however its location and
etc). This data is provided in a <structname>SIGCONTEXT</structname> contents vary between kernels (BSD, Solaris, etc). This data is
structure, and on entry to the signal handler it contains the register provided in a <structname>SIGCONTEXT</structname> structure, and on
state of the CPU before the signal was sent. Modifying it will cause entry to the signal handler it contains the register state of the CPU
the kernel to adjust the context before restarting the thread. before the signal was sent. Modifying it will cause the kernel to
adjust the context before restarting the thread.
</para> </para>
</sect2> </sect2>
</sect1> </sect1>
@ -1195,6 +1517,7 @@ if (res != ERROR_SUCCESS) return res;
</para> </para>
<para> <para>
<note> <note>
<title>Case sensitivity vs. preservation</title>
<para> <para>
When we say that most systems in NT are case insensitive, this When we say that most systems in NT are case insensitive, this
has to be understood for looking up for a file, where the has to be understood for looking up for a file, where the
@ -2125,6 +2448,150 @@ if (res != ERROR_SUCCESS) return res;
</para> </para>
</sect3> </sect3>
</sect2> </sect2>
<sect2>
<title>Win16 processes support</title>
<sect3>
<title>Starting a NE (Win16) process</title>
<para>
Wine is also able to run 16 bit processes, but this feature is only
supported on Intel IA-32 architectures.
</para>
<para>
When Wine is requested to run a NE (Win 16 process), it will in fact
hand over the execution of it to a specific executable
<command>winevdm</command>. VDM stands for Virtual DOS
Machine. This <command>winevdm</command> is a Winelib application,
but will in fact set up the correct 16 bit environment to run the
executable. We will get back later on in details to what this means.
</para>
<para>
Any new 16 bit process created by this executable (or its children)
will run into the same <command>winevdm</command> instance. Among
one instance, several functionalities will be provided to those 16
bit processes, including the cooperative multitasking, sharing the
same address space, managing the selectors for the 16 bit segments
needed for code, data and stack.
</para>
<para>
Note that several <command>winevdm</command> instances can run in
the same Wine session, but the functionalities described above are
only shared among a given instance, not among all the
instances. <command>winevdm</command> is built as Winelib
application, and hence has access to any facility a 32 bit
application has.
</para>
<para>
Each Win16 application is implemented in
<command>winevdm</command> as a Win32
thread. <command>winevdm</command> then implements its own
scheduling facilities (in fact, the code for this feature is in the
<filename>krnl386.exe</filename> DLL). Since the required Win16
scheduling is non pre-emptive, this doesn't require any underlying
OS kernel support.
</para>
</sect3>
<sect3>
<title>SysLevels</title>
<para>
SysLevels are an undocumented Windows-internal thread-safety system
dedicated to 16 bit applications (or 32 bit applications that call -
directly or indirectly - 16 bit code). They are basically critical
sections which must be taken in a particular order. The mechanism is
generic but there are always three syslevels:
<itemizedlist>
<listitem>
<para>level 1 is the Win16 mutex,</para>
</listitem>
<listitem>
<para>level 2 is the <filename>USER</filename> mutex,</para>
</listitem>
<listitem>
<para>level 3 is the <filename>GDI</filename> mutex.</para>
</listitem>
</itemizedlist>
</para>
<para>
When entering a syslevel, the code (in
<filename>dlls/kernel/syslevel.c</filename>) will check that a
higher syslevel is not already held and produce an error if so. This
is because it's not legal to enter level 2 while holding level 3 -
first, you must leave level 3.
</para>
<para>
Throughout the code you may see calls to
<function>_ConfirmSysLevel()</function> and
<function>_CheckNotSysLevel()</function>. These functions are
essentially assertions about the syslevel states and can be used to
check that the rules have not been accidentally violated. In
particular, <function>_CheckNotSysLevel()</function> will break
probably into the debugger) if the check fails. If this happens the
solution is to get a backtrace and find out, by reading the source
of the wine functions called along the way, how Wine got into the
invalid state.
</para>
</sect3>
<sect3>
<title>Memory management</title>
<para>
Every Win16 address is expressed in the form of selector:offset. The
selector is an entry in the LDT, but a 16 bit entry, limiting each
offset to 64 KB. Hence, the maximum available memory to a Win16
process is 512 MB. Note, that the processor runs in protected mode,
but using 16 bit selectors.
</para>
<para>
Windows, for a 16 bit process, defines a few selectors to access the
"real" memory (the one provided) by DOS. Basically, Wine also
provides this area of memory.
</para>
</sect3>
</sect2>
<sect2>
<title>DOS processes support</title>
<para>
The behaviour we just described also applies to DOS executables, which
are handled the same way by <command>winevdm</command>. This is only
supported on Intel IA-32 architectures.
</para>
<para>
Wine implements also most of the DOS support in a Wine specific DLL
(<filename>winedos</filename>). This DLL is called under certain
conditions, like:
<itemizedlist>
<listitem>
<para>
In <command>winevdm</command>, when trying to launch a DOS
application (<filename>.EXE</filename> or
<filename>.COM</filename>, <filename>.PIF</filename>).
</para>
</listitem>
<listitem>
<para>
In <filename>kernel32</filename>, when an attempt is made in the
binary code to call some DOS or BIOS interrupts (like Int 21h
for example).
</para>
</listitem>
</itemizedlist>
</para>
<para>
When <command>winevdm</command> runs a DOS program, this one runs in
real mode (in fact in V86 mode from the IA-32 point of view).
</para>
<para>
Wine also supports part of the DPMI (DOS Protected Mode Interface).
</para>
<para>
Wine, when running a DOS programs, needs to map the 1 MB of virtual
memory to the real memory (as seen by the DOS program). When this is
not possible (like when someone else is already using this area), the
DOS support is not possible. Not also that by doing so, access to
linear address 0 is enabled (as it's also real mode address 0 which is
valid). Hence, NULL pointer derefence faults are no longer catched.
</para>
</sect2>
</sect1> </sect1>
</chapter> </chapter>