2000-10-28 02:53:56 +08:00
|
|
|
/* autoload.cc: all dynamic load stuff.
|
|
|
|
|
|
|
|
This file is part of Cygwin.
|
|
|
|
|
|
|
|
This software is a copyrighted work licensed under the terms of the
|
|
|
|
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
#include "winsup.h"
|
2008-04-08 00:15:45 +08:00
|
|
|
#include "miscfuncs.h"
|
winsup/cygwin/ChangeLog:
* Makefile.in (DLL_OFILES): Add new fenv.o module.
(fenv_CFLAGS): New flags definition for fenv.o compile.
* autoload.cc (std_dll_init): Use fenv.h functions instead of direct
manipulation of x87 FPU registers.
* crt0.c (mainCRTStartup): Likewise.
* cygwin.din (feclearexcept, fegetexceptflag, feraiseexcept,
fesetexceptflag, fetestexcept, fegetround, fesetround, fegetenv,
feholdexcept, fesetenv, feupdateenv, fegetprec, fesetprec,
feenableexcept, fedisableexcept, fegetexcept, _feinitialise,
_fe_dfl_env, _fe_nomask_env): Export new functions and data items.
* fenv.cc: New file.
* posix.sgml: Update status of newly-implemented APIs.
* include/fenv.h: Likewise related header.
* include/cygwin/version.h (CYGWIN_VERSION_API_MINOR): Bump.
2010-09-11 14:53:28 +08:00
|
|
|
#include "fenv.h"
|
2001-05-16 13:07:51 +08:00
|
|
|
#define USE_SYS_TYPES_FD_SET
|
|
|
|
#include <winsock2.h>
|
|
|
|
|
2008-04-08 00:15:45 +08:00
|
|
|
bool NO_COPY wsock_started;
|
|
|
|
|
2001-05-16 13:07:51 +08:00
|
|
|
/* Macro for defining "auto-load" functions.
|
|
|
|
* Note that this is self-modifying code *gasp*.
|
|
|
|
* The first invocation of a routine will trigger the loading of
|
|
|
|
* the DLL. This will then be followed by the discovery of
|
|
|
|
* the procedure's entry point, which is placed into the location
|
|
|
|
* pointed to by the stack pointer. This code then changes
|
|
|
|
* the "call" operand which invoked it to a "jmp" which will
|
|
|
|
* transfer directly to the DLL function on the next invocation.
|
|
|
|
*
|
|
|
|
* Subsequent calls to routines whose transfer address has not been
|
|
|
|
* determined will skip the "load the dll" step, starting at the
|
|
|
|
* "discovery of the entry point" step.
|
|
|
|
*
|
|
|
|
* So, immediately following the the call to one of the above routines
|
|
|
|
* we have:
|
2013-04-23 17:44:36 +08:00
|
|
|
* DLL info (4/8 bytes) Pointer to a block of information concerning
|
2001-05-16 13:07:51 +08:00
|
|
|
* the DLL (see below).
|
|
|
|
* DLL args (4 bytes) The number of arguments pushed on the stack by
|
|
|
|
* the call. If this is an odd value then this
|
|
|
|
* is a flag that non-existence of this function
|
|
|
|
* is not a fatal error
|
2013-04-23 17:44:36 +08:00
|
|
|
* func addr (8 bytes) (64 bit ONLY!)
|
|
|
|
* Address of the actual Win32 function. For the
|
|
|
|
* reason why this is necessary, see the below
|
|
|
|
* description of the load_state.
|
2001-05-16 13:07:51 +08:00
|
|
|
* func name (n bytes) asciz string containing the name of the function
|
|
|
|
* to be loaded.
|
|
|
|
*
|
|
|
|
* The DLL info block consists of the following
|
2013-04-23 17:44:36 +08:00
|
|
|
* load_state (4/8 bytes) Pointer to a word containing the routine used
|
2001-05-16 13:07:51 +08:00
|
|
|
* to eventually invoke the function. Initially
|
2013-04-23 17:44:36 +08:00
|
|
|
* points to an init function which loads the DLL,
|
|
|
|
* gets the process's load address, changes the contents
|
|
|
|
* here to point to the function address, and changes
|
|
|
|
* the address argument of the initial jmp call.
|
|
|
|
* On 64 bit, the jmp is not tweaked directly. Rather,
|
|
|
|
* the address of the Win32 function is stored in the
|
|
|
|
* aforementioned Win32 function address slot and fetched
|
|
|
|
* there for a jmp *%rax call. This indirection is
|
|
|
|
* necessary to workaround the lack of a jmp opcode with
|
|
|
|
* offset values > 32 bit. If the initialization has
|
|
|
|
* been done, only the load part is done.
|
|
|
|
* DLL handle (4/8 bytes) The handle to use when loading the DLL.
|
2001-05-16 13:07:51 +08:00
|
|
|
* DLL locker (4 bytes) Word to use to avoid multi-thread access during
|
|
|
|
* initialization.
|
2013-04-23 17:44:36 +08:00
|
|
|
* extra init (4/8 bytes) Extra initialization function.
|
2001-05-16 13:07:51 +08:00
|
|
|
* DLL name (n bytes) asciz string containing the name of the DLL.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* LoadDLLprime is used to prime the DLL info information, providing an
|
|
|
|
additional initialization routine to call prior to calling the first
|
|
|
|
function. */
|
2013-04-23 17:44:36 +08:00
|
|
|
#ifdef __x86_64__
|
|
|
|
#define LoadDLLprime(dllname, init_also, no_resolve_on_fork) __asm__ (" \n\
|
|
|
|
.ifndef " #dllname "_primed \n\
|
|
|
|
.section .data_cygwin_nocopy,\"w\" \n\
|
|
|
|
.align 8 \n\
|
|
|
|
."#dllname "_info: \n\
|
|
|
|
.quad _std_dll_init \n\
|
|
|
|
.quad " #no_resolve_on_fork " \n\
|
|
|
|
.long -1 \n\
|
|
|
|
.align 8 \n\
|
|
|
|
.quad " #init_also " \n\
|
|
|
|
.string16 \"" #dllname ".dll\" \n\
|
|
|
|
.text \n\
|
|
|
|
.set " #dllname "_primed, 1 \n\
|
|
|
|
.endif \n\
|
|
|
|
");
|
|
|
|
#else
|
2011-02-27 07:30:33 +08:00
|
|
|
#define LoadDLLprime(dllname, init_also, no_resolve_on_fork) __asm__ (" \n\
|
2005-03-10 22:41:50 +08:00
|
|
|
.ifndef " #dllname "_primed \n\
|
2005-03-11 00:59:55 +08:00
|
|
|
.section .data_cygwin_nocopy,\"w\" \n\
|
2005-03-10 22:41:50 +08:00
|
|
|
.align 4 \n\
|
|
|
|
."#dllname "_info: \n\
|
2003-04-18 04:05:15 +08:00
|
|
|
.long _std_dll_init \n\
|
2011-02-27 07:30:33 +08:00
|
|
|
.long " #no_resolve_on_fork " \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
.long -1 \n\
|
|
|
|
.long " #init_also " \n\
|
2010-08-28 16:51:21 +08:00
|
|
|
.string16 \"" #dllname ".dll\" \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
.text \n\
|
2005-03-10 22:41:50 +08:00
|
|
|
.set " #dllname "_primed, 1 \n\
|
|
|
|
.endif \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
");
|
2013-04-23 17:44:36 +08:00
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
|
|
|
|
/* Create a "decorated" name */
|
|
|
|
#define mangle(name, n) #name "@" #n
|
|
|
|
|
2004-04-18 01:53:29 +08:00
|
|
|
/* Standard DLL load macro. May invoke a fatal error if the function isn't
|
2001-05-16 13:07:51 +08:00
|
|
|
found. */
|
2005-04-29 07:59:44 +08:00
|
|
|
#define LoadDLLfunc(name, n, dllname) \
|
|
|
|
LoadDLLfuncEx (name, n, dllname, 0)
|
|
|
|
#define LoadDLLfuncEx(name, n, dllname, notimp) \
|
|
|
|
LoadDLLfuncEx2(name, n, dllname, notimp, 0)
|
|
|
|
#define LoadDLLfuncEx2(name, n, dllname, notimp, err) \
|
|
|
|
LoadDLLfuncEx3(name, n, dllname, notimp, err, 0)
|
2001-05-16 13:07:51 +08:00
|
|
|
|
|
|
|
/* Main DLL setup stuff. */
|
2013-04-23 17:44:36 +08:00
|
|
|
#ifdef __x86_64__
|
|
|
|
#define LoadDLLfuncEx3(name, n, dllname, notimp, err, no_resolve_on_fork) \
|
|
|
|
LoadDLLprime (dllname, dll_func_load, no_resolve_on_fork) \
|
|
|
|
__asm__ (" \n\
|
|
|
|
.section ." #dllname "_autoload_text,\"wx\" \n\
|
|
|
|
.global " #name " \n\
|
|
|
|
.global _win32_" #name " \n\
|
|
|
|
.align 16 \n\
|
|
|
|
" #name ": \n\
|
|
|
|
_win32_" #name ": \n\
|
|
|
|
movq 3f(%rip),%rax \n\
|
|
|
|
jmp *%rax \n\
|
|
|
|
1:movq 2f(%rip),%rax \n\
|
|
|
|
push %rbp # Keep 16 byte aligned \n\
|
|
|
|
push %r9 \n\
|
|
|
|
push %r8 \n\
|
|
|
|
push %rdx \n\
|
|
|
|
push %rcx \n\
|
|
|
|
call *(%rax) \n\
|
|
|
|
2:.quad ." #dllname "_info \n\
|
|
|
|
.long (" #n "+" #notimp ") | (((" #err ") & 0xff) <<16) \n\
|
|
|
|
3:.quad 1b \n\
|
|
|
|
.asciz \"" #name "\" \n\
|
|
|
|
.text \n\
|
|
|
|
");
|
|
|
|
#else
|
2011-02-27 07:30:33 +08:00
|
|
|
#define LoadDLLfuncEx3(name, n, dllname, notimp, err, no_resolve_on_fork) \
|
|
|
|
LoadDLLprime (dllname, dll_func_load, no_resolve_on_fork) \
|
2001-05-19 13:29:00 +08:00
|
|
|
__asm__ (" \n\
|
2005-07-03 10:28:14 +08:00
|
|
|
.section ." #dllname "_autoload_text,\"wx\" \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
.global _" mangle (name, n) " \n\
|
|
|
|
.global _win32_" mangle (name, n) " \n\
|
|
|
|
.align 8 \n\
|
|
|
|
_" mangle (name, n) ": \n\
|
|
|
|
_win32_" mangle (name, n) ": \n\
|
2002-07-20 12:59:15 +08:00
|
|
|
.byte 0xe9 \n\
|
|
|
|
.long -4 + 1f - . \n\
|
|
|
|
1:movl (2f),%eax \n\
|
2005-04-29 07:59:44 +08:00
|
|
|
call *(%eax) \n\
|
2002-07-20 12:59:15 +08:00
|
|
|
2:.long ." #dllname "_info \n\
|
2011-02-27 07:30:33 +08:00
|
|
|
.long (" #n "+" #notimp ") | (((" #err ") & 0xff) <<16) \n\
|
|
|
|
.asciz \"" #name "\" \n\
|
|
|
|
.text \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
");
|
2013-04-23 17:44:36 +08:00
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
|
|
|
|
/* DLL loader helper functions used during initialization. */
|
|
|
|
|
|
|
|
/* The function which finds the address, given the name and overwrites
|
|
|
|
the call so that future invocations go straight to the function in
|
|
|
|
the DLL. */
|
|
|
|
extern "C" void dll_func_load () __asm__ ("dll_func_load");
|
|
|
|
|
|
|
|
/* Called by the primary initialization function "init_std_dll" to
|
|
|
|
setup the stack and eliminate future calls to init_std_dll for other
|
|
|
|
functions from this DLL. */
|
|
|
|
extern "C" void dll_chain () __asm__ ("dll_chain");
|
|
|
|
|
2000-10-28 02:53:56 +08:00
|
|
|
extern "C" {
|
|
|
|
|
2013-04-23 17:44:36 +08:00
|
|
|
#ifdef __x86_64__
|
|
|
|
__asm__ (" \n\
|
|
|
|
.section .rdata,\"r\" \n\
|
|
|
|
msg1: \n\
|
|
|
|
.ascii \"couldn't dynamically determine load address for '%s' (handle %p), %E\\0\"\n\
|
|
|
|
\n\
|
|
|
|
.text \n\
|
|
|
|
.p2align 4,,15 \n\
|
|
|
|
noload: \n\
|
|
|
|
movq 40(%rsp),%rdx # Get the address of the information block\n\
|
|
|
|
movl 8(%rdx),%eax # Should we 'ignore' the lack \n\
|
|
|
|
test $1,%eax # of this function? \n\
|
|
|
|
jz 1f # Nope. \n\
|
|
|
|
andl $0xffff0000,%eax# upper word (== desired return value) \n\
|
|
|
|
movl %eax,32(%rsp) # Save for later (in shadow space) \n\
|
|
|
|
movl $127,%ecx # ERROR_PROC_NOT_FOUND \n\
|
|
|
|
call SetLastError # Set it \n\
|
|
|
|
movl 32(%rsp),%eax # Get back return value \n\
|
|
|
|
sarl $16,%eax # swap to low order word \n\
|
|
|
|
addq $40,%rsp # Revert stack \n\
|
|
|
|
pop %r10 # Drop pointer to 'return address' \n\
|
|
|
|
pop %rcx # Restore arg registers \n\
|
|
|
|
pop %rdx \n\
|
|
|
|
pop %r8 \n\
|
|
|
|
pop %r9 \n\
|
|
|
|
pop %rbp # ...and restore frame pointer \n\
|
|
|
|
ret # Return \n\
|
|
|
|
1: \n\
|
|
|
|
movq (%rdx),%rax # Handle value \n\
|
|
|
|
movq 8(%rax),%r8 \n\
|
|
|
|
lea 20(%rdx),%rdx # Location of name of function \n\
|
|
|
|
lea msg1(%rip),%rcx # The message \n\
|
|
|
|
call api_fatal # Print message. Never returns \n\
|
|
|
|
\n\
|
|
|
|
.globl dll_func_load \n\
|
|
|
|
dll_func_load: \n\
|
|
|
|
movq (%rsp),%rdx # 'Return address' contains load info \n\
|
|
|
|
movq (%rdx),%rcx # Where handle lives \n\
|
|
|
|
movq 8(%rcx),%rcx # Address of Handle to DLL \n\
|
|
|
|
addq $20,%rdx # Address of name of function to load \n\
|
|
|
|
subq $40,%rsp # Shadow space + 8 byte for alignment \n\
|
|
|
|
call GetProcAddress # Load it \n\
|
|
|
|
test %rax,%rax # Success? \n\
|
|
|
|
jne gotit # Yes \n\
|
|
|
|
jmp noload # Issue an error or return \n\
|
|
|
|
gotit: \n\
|
|
|
|
addq $40,%rsp # Revert stack \n\
|
|
|
|
pop %r10 # Pointer to 'return address' \n\
|
|
|
|
movq %rax,12(%r10) # Move absolute address to address slot \n\
|
|
|
|
subq $25,%r10 # Point to jmp \n\
|
|
|
|
pop %rcx # Restore arg registers \n\
|
|
|
|
pop %rdx \n\
|
|
|
|
pop %r8 \n\
|
|
|
|
pop %r9 \n\
|
|
|
|
pop %rbp # ...and restore frame pointer \n\
|
|
|
|
jmp *%r10 # Jump to actual function \n\
|
|
|
|
\n\
|
|
|
|
.global dll_chain \n\
|
|
|
|
dll_chain: \n\
|
|
|
|
push %rax # Restore 'return address' \n\
|
|
|
|
jmp *%rdx # Jump to next init function \n\
|
|
|
|
");
|
|
|
|
#else
|
2001-05-16 13:07:51 +08:00
|
|
|
__asm__ (" \n\
|
2008-04-19 02:20:30 +08:00
|
|
|
.text \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
msg1: \n\
|
|
|
|
.ascii \"couldn't dynamically determine load address for '%s' (handle %p), %E\\0\"\n\
|
|
|
|
\n\
|
|
|
|
.align 32 \n\
|
|
|
|
noload: \n\
|
|
|
|
popl %edx # Get the address of the information block\n\
|
|
|
|
movl 4(%edx),%eax # Should we 'ignore' the lack \n\
|
|
|
|
test $1,%eax # of this function? \n\
|
|
|
|
jz 1f # Nope. \n\
|
|
|
|
decl %eax # Yes. This is the # of bytes + 1 \n\
|
|
|
|
popl %edx # Caller's caller \n\
|
|
|
|
addl %eax,%esp # Pop off bytes \n\
|
2002-06-26 13:29:41 +08:00
|
|
|
andl $0xffff0000,%eax# upper word \n\
|
|
|
|
subl %eax,%esp # adjust for possible return value \n\
|
2015-08-26 04:15:22 +08:00
|
|
|
pushl %eax # Save return value for later \n\
|
|
|
|
pushl %edx # Save return address for later \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
movl $127,%eax # ERROR_PROC_NOT_FOUND \n\
|
|
|
|
pushl %eax # First argument \n\
|
|
|
|
call _SetLastError@4 # Set it \n\
|
2015-08-26 04:15:22 +08:00
|
|
|
popl %edx # Get back return address \n\
|
|
|
|
popl %eax # Get back return value \n\
|
2005-03-03 00:13:24 +08:00
|
|
|
sarl $16,%eax # return value in high order word \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
jmp *%edx # Return \n\
|
|
|
|
1: \n\
|
|
|
|
movl (%edx),%eax # Handle value \n\
|
|
|
|
pushl 4(%eax) \n\
|
|
|
|
leal 8(%edx),%eax # Location of name of function \n\
|
2010-07-19 04:14:51 +08:00
|
|
|
pushl %eax \n\
|
|
|
|
pushl $msg1 # The message \n\
|
2011-05-29 02:17:09 +08:00
|
|
|
call _api_fatal # Print message. Never returns \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
\n\
|
|
|
|
.globl dll_func_load \n\
|
|
|
|
dll_func_load: \n\
|
|
|
|
movl (%esp),%eax # 'Return address' contains load info \n\
|
|
|
|
addl $8,%eax # Address of name of function to load \n\
|
|
|
|
pushl %eax # Second argument \n\
|
|
|
|
movl -8(%eax),%eax # Where handle lives \n\
|
|
|
|
movl 4(%eax),%eax # Address of Handle to DLL \n\
|
|
|
|
pushl %eax # Handle to DLL \n\
|
|
|
|
call _GetProcAddress@8# Load it \n\
|
|
|
|
test %eax,%eax # Success? \n\
|
|
|
|
jne gotit # Yes \n\
|
|
|
|
jmp noload # Issue an error or return \n\
|
|
|
|
gotit: \n\
|
2002-07-20 12:59:15 +08:00
|
|
|
popl %edx # Pointer to 'return address' \n\
|
|
|
|
subl %edx,%eax # Make it relative \n\
|
|
|
|
addl $7,%eax # Tweak \n\
|
|
|
|
subl $12,%edx # Point to jmp \n\
|
|
|
|
movl %eax,1(%edx) # Move relative address after jump \n\
|
2001-05-16 13:07:51 +08:00
|
|
|
jmp *%edx # Jump to actual function \n\
|
|
|
|
\n\
|
|
|
|
.global dll_chain \n\
|
|
|
|
dll_chain: \n\
|
|
|
|
pushl %eax # Restore 'return address' \n\
|
|
|
|
jmp *%edx # Jump to next init function \n\
|
|
|
|
");
|
2013-04-23 17:44:36 +08:00
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
|
|
|
|
/* C representations of the two info blocks described above.
|
|
|
|
FIXME: These structures confuse gdb for some reason. GDB can print
|
|
|
|
the whole structure but has problems with the name field? */
|
|
|
|
struct dll_info
|
2000-10-28 02:53:56 +08:00
|
|
|
{
|
2013-04-23 17:44:36 +08:00
|
|
|
UINT_PTR load_state;
|
2001-05-16 13:07:51 +08:00
|
|
|
HANDLE handle;
|
|
|
|
LONG here;
|
|
|
|
void (*init) ();
|
2010-08-28 16:51:21 +08:00
|
|
|
WCHAR name[];
|
2000-10-28 02:53:56 +08:00
|
|
|
};
|
|
|
|
|
2001-05-16 13:07:51 +08:00
|
|
|
struct func_info
|
|
|
|
{
|
|
|
|
struct dll_info *dll;
|
|
|
|
LONG decoration;
|
2013-04-23 17:44:36 +08:00
|
|
|
#ifdef __x86_64__
|
|
|
|
UINT_PTR func_addr;
|
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
char name[];
|
|
|
|
};
|
2000-10-28 02:53:56 +08:00
|
|
|
|
2001-05-16 13:07:51 +08:00
|
|
|
/* Mechanism for setting up info for passing to dll_chain routines. */
|
2013-04-23 17:44:36 +08:00
|
|
|
#ifdef __x86_64__
|
|
|
|
typedef __uint128_t two_addr_t;
|
|
|
|
#else
|
|
|
|
typedef __uint64_t two_addr_t;
|
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
union retchain
|
|
|
|
{
|
2013-04-23 17:44:36 +08:00
|
|
|
struct {uintptr_t high; uintptr_t low;};
|
|
|
|
two_addr_t ll;
|
2001-05-16 13:07:51 +08:00
|
|
|
};
|
|
|
|
|
2016-01-12 22:23:14 +08:00
|
|
|
/* This function handles the problem described here:
|
|
|
|
|
|
|
|
http://www.microsoft.com/technet/security/advisory/2269637.mspx
|
2016-06-24 21:42:06 +08:00
|
|
|
https://msdn.microsoft.com/library/ff919712 */
|
2011-03-01 04:55:10 +08:00
|
|
|
static __inline bool
|
2016-01-12 22:23:14 +08:00
|
|
|
dll_load (HANDLE& handle, PWCHAR name)
|
2011-02-27 07:30:33 +08:00
|
|
|
{
|
2016-01-12 22:45:17 +08:00
|
|
|
HANDLE h = NULL;
|
2016-01-27 22:43:15 +08:00
|
|
|
WCHAR dll_path[MAX_PATH];
|
2016-01-12 22:23:14 +08:00
|
|
|
|
2016-06-24 21:42:06 +08:00
|
|
|
/* Try loading with full path, which sometimes fails for no good reason. */
|
2016-01-27 22:43:15 +08:00
|
|
|
wcpcpy (wcpcpy (dll_path, windows_system_directory), name);
|
|
|
|
h = LoadLibraryW (dll_path);
|
2016-06-24 21:42:06 +08:00
|
|
|
/* If it failed, try loading just by name. */
|
2016-01-12 22:23:14 +08:00
|
|
|
if (!h)
|
2016-01-27 22:43:15 +08:00
|
|
|
h = LoadLibraryW (name);
|
2011-03-01 04:55:10 +08:00
|
|
|
if (!h)
|
|
|
|
return false;
|
|
|
|
handle = h;
|
|
|
|
return true;
|
2011-02-27 07:30:33 +08:00
|
|
|
}
|
|
|
|
|
2010-12-10 00:50:36 +08:00
|
|
|
#define RETRY_COUNT 10
|
|
|
|
|
2001-05-16 13:07:51 +08:00
|
|
|
/* The standard DLL initialization routine. */
|
2013-04-23 17:44:36 +08:00
|
|
|
#ifdef __x86_64__
|
|
|
|
|
|
|
|
/* On x86_64, we need assembler wrappers for std_dll_init and wsock_init.
|
|
|
|
In the x86_64 ABI it's no safe bet that frame[1] (aka 8(%rbp)) contains
|
|
|
|
the return address. Consequentially, if we try to overwrite frame[1]
|
|
|
|
with the address of dll_chain, we end up with a scrambled stack, the
|
|
|
|
result depending on the optimization settings and the current frame of
|
|
|
|
mind of the compiler. So for x86_64, we disable overwriting the return
|
|
|
|
address in the real std_dll_init/wsock_init function, but rather do this
|
|
|
|
in the wrapper, after return from the function, when we exactly know
|
|
|
|
where the original return address is stored on the stack. */
|
|
|
|
|
|
|
|
#define INIT_WRAPPER(func) \
|
|
|
|
__asm__ (" \n\
|
|
|
|
.text \n\
|
|
|
|
.p2align 4,,15 \n\
|
|
|
|
.seh_proc _" #func " \n\
|
|
|
|
_" #func ": \n\
|
|
|
|
pushq %rbp \n\
|
|
|
|
.seh_pushreg %rbp \n\
|
|
|
|
movq %rsp,%rbp \n\
|
|
|
|
.seh_setframe %rbp,0 \n\
|
|
|
|
subq $0x20,%rsp \n\
|
|
|
|
.seh_stackalloc 32 \n\
|
|
|
|
.seh_endprologue \n\
|
|
|
|
movq 0x28(%rsp),%rcx # return address as parameter \n\
|
|
|
|
call " #func " \n\
|
|
|
|
movdqa %xmm0,0x10(%rsp) # 128 bit return value in xmm0 \n\
|
|
|
|
movq 0x10(%rsp),%rax # copy over to %rax and %rdx \n\
|
|
|
|
movq 0x18(%rsp),%rdx \n\
|
|
|
|
leaq dll_chain(%rip),%rcx # load address of dll_chain \n\
|
|
|
|
movq %rcx,0x28(%rsp) # and overwrite return address \n\
|
|
|
|
addq $0x20,%rsp \n\
|
|
|
|
popq %rbp \n\
|
|
|
|
ret \n\
|
|
|
|
.seh_endproc \n\
|
|
|
|
");
|
|
|
|
|
|
|
|
INIT_WRAPPER (std_dll_init)
|
|
|
|
|
|
|
|
__attribute__ ((used, noinline)) static two_addr_t
|
|
|
|
std_dll_init (struct func_info *func)
|
|
|
|
#else
|
|
|
|
__attribute__ ((used, noinline)) static two_addr_t
|
2001-05-16 13:07:51 +08:00
|
|
|
std_dll_init ()
|
2013-04-23 17:44:36 +08:00
|
|
|
#endif
|
2000-10-28 02:53:56 +08:00
|
|
|
{
|
2017-11-27 21:36:06 +08:00
|
|
|
#ifdef __i386__
|
2001-05-16 13:07:51 +08:00
|
|
|
struct func_info *func = (struct func_info *) __builtin_return_address (0);
|
2013-04-23 17:44:36 +08:00
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
struct dll_info *dll = func->dll;
|
|
|
|
retchain ret;
|
|
|
|
|
|
|
|
if (InterlockedIncrement (&dll->here))
|
|
|
|
do
|
|
|
|
{
|
|
|
|
InterlockedDecrement (&dll->here);
|
2010-03-13 07:13:48 +08:00
|
|
|
yield ();
|
2001-05-16 13:07:51 +08:00
|
|
|
}
|
|
|
|
while (InterlockedIncrement (&dll->here));
|
2011-02-27 07:30:33 +08:00
|
|
|
else if ((uintptr_t) dll->handle <= 1)
|
2001-05-16 13:07:51 +08:00
|
|
|
{
|
winsup/cygwin/ChangeLog:
* Makefile.in (DLL_OFILES): Add new fenv.o module.
(fenv_CFLAGS): New flags definition for fenv.o compile.
* autoload.cc (std_dll_init): Use fenv.h functions instead of direct
manipulation of x87 FPU registers.
* crt0.c (mainCRTStartup): Likewise.
* cygwin.din (feclearexcept, fegetexceptflag, feraiseexcept,
fesetexceptflag, fetestexcept, fegetround, fesetround, fegetenv,
feholdexcept, fesetenv, feupdateenv, fegetprec, fesetprec,
feenableexcept, fedisableexcept, fegetexcept, _feinitialise,
_fe_dfl_env, _fe_nomask_env): Export new functions and data items.
* fenv.cc: New file.
* posix.sgml: Update status of newly-implemented APIs.
* include/fenv.h: Likewise related header.
* include/cygwin/version.h (CYGWIN_VERSION_API_MINOR): Bump.
2010-09-11 14:53:28 +08:00
|
|
|
fenv_t fpuenv;
|
|
|
|
fegetenv (&fpuenv);
|
2010-12-10 00:50:36 +08:00
|
|
|
DWORD err = ERROR_SUCCESS;
|
2011-02-27 07:30:33 +08:00
|
|
|
int i;
|
2010-10-02 05:53:11 +08:00
|
|
|
/* MSDN seems to imply that LoadLibrary can fail mysteriously, so,
|
|
|
|
since there have been reports of this in the mailing list, retry
|
2010-12-10 00:50:36 +08:00
|
|
|
several times before giving up. */
|
2011-02-27 07:30:33 +08:00
|
|
|
for (i = 1; i <= RETRY_COUNT; i++)
|
2010-12-10 00:50:36 +08:00
|
|
|
{
|
|
|
|
/* If loading the library succeeds, just leave the loop. */
|
2016-01-12 22:23:14 +08:00
|
|
|
if (dll_load (dll->handle, dll->name))
|
2010-12-10 00:50:36 +08:00
|
|
|
break;
|
|
|
|
/* Otherwise check error code returned by LoadLibrary. If the
|
|
|
|
error code is neither NOACCESS nor DLL_INIT_FAILED, break out
|
|
|
|
of the loop. */
|
|
|
|
err = GetLastError ();
|
|
|
|
if (err != ERROR_NOACCESS && err != ERROR_DLL_INIT_FAILED)
|
|
|
|
break;
|
|
|
|
if (i < RETRY_COUNT)
|
|
|
|
yield ();
|
|
|
|
}
|
2011-02-27 07:30:33 +08:00
|
|
|
if ((uintptr_t) dll->handle <= 1)
|
2010-12-10 00:50:36 +08:00
|
|
|
{
|
2016-01-12 22:23:14 +08:00
|
|
|
if ((func->decoration & 1))
|
2010-12-10 00:50:36 +08:00
|
|
|
dll->handle = INVALID_HANDLE_VALUE;
|
|
|
|
else
|
2016-01-12 22:23:14 +08:00
|
|
|
api_fatal ("unable to load %W, %E", dll->name);
|
2010-12-10 00:50:36 +08:00
|
|
|
}
|
2010-09-26 03:57:34 +08:00
|
|
|
fesetenv (&fpuenv);
|
2001-05-16 13:07:51 +08:00
|
|
|
}
|
|
|
|
|
2010-10-02 05:53:11 +08:00
|
|
|
/* Set "arguments" for dll_chain. */
|
2013-04-23 17:44:36 +08:00
|
|
|
ret.low = (uintptr_t) dll->init;
|
|
|
|
ret.high = (uintptr_t) func;
|
2010-09-25 03:53:18 +08:00
|
|
|
|
2001-05-16 13:07:51 +08:00
|
|
|
InterlockedDecrement (&dll->here);
|
|
|
|
|
2017-11-27 21:36:06 +08:00
|
|
|
#ifdef __i386__
|
2001-05-16 13:07:51 +08:00
|
|
|
/* Kludge alert. Redirects the return address to dll_chain. */
|
2013-04-23 17:44:36 +08:00
|
|
|
uintptr_t *volatile frame = (uintptr_t *) __builtin_frame_address (0);
|
|
|
|
frame[1] = (uintptr_t) dll_chain;
|
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
|
|
|
|
return ret.ll;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialization function for winsock stuff. */
|
2013-04-23 17:44:36 +08:00
|
|
|
|
|
|
|
#ifdef __x86_64__
|
|
|
|
/* See above comment preceeding std_dll_init. */
|
|
|
|
INIT_WRAPPER (wsock_init)
|
|
|
|
|
|
|
|
__attribute__ ((used, noinline)) static two_addr_t
|
|
|
|
wsock_init (struct func_info *func)
|
|
|
|
#else
|
|
|
|
__attribute__ ((used, noinline)) static two_addr_t
|
2001-05-16 13:07:51 +08:00
|
|
|
wsock_init ()
|
2013-04-23 17:44:36 +08:00
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
{
|
2016-03-10 05:55:28 +08:00
|
|
|
/* CV 2016-03-09: Moved wsadata into wsock_init to workaround a problem
|
|
|
|
with the NO_COPY definition of wsadata and here starting with gcc-5.3.0.
|
|
|
|
See the git log for a description. */
|
|
|
|
static WSADATA NO_COPY wsadata;
|
2001-05-16 13:07:51 +08:00
|
|
|
static LONG NO_COPY here = -1L;
|
2017-11-27 21:36:06 +08:00
|
|
|
#ifdef __i386__
|
2001-05-16 13:07:51 +08:00
|
|
|
struct func_info *func = (struct func_info *) __builtin_return_address (0);
|
2013-04-23 17:44:36 +08:00
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
struct dll_info *dll = func->dll;
|
|
|
|
|
2001-05-05 05:02:15 +08:00
|
|
|
while (InterlockedIncrement (&here))
|
2000-10-28 02:53:56 +08:00
|
|
|
{
|
2001-05-05 05:02:15 +08:00
|
|
|
InterlockedDecrement (&here);
|
2010-03-13 07:13:48 +08:00
|
|
|
yield ();
|
2000-10-28 02:53:56 +08:00
|
|
|
}
|
|
|
|
|
2004-04-16 01:38:02 +08:00
|
|
|
if (!wsock_started)
|
2001-05-16 13:07:51 +08:00
|
|
|
{
|
2011-02-06 03:38:21 +08:00
|
|
|
int __stdcall (*wsastartup) (int, WSADATA *);
|
2000-10-28 02:53:56 +08:00
|
|
|
|
2005-03-03 23:12:50 +08:00
|
|
|
/* Don't use autoload to load WSAStartup to eliminate recursion. */
|
2011-02-06 03:38:21 +08:00
|
|
|
wsastartup = (int __stdcall (*)(int, WSADATA *))
|
2001-09-08 05:32:07 +08:00
|
|
|
GetProcAddress ((HMODULE) (dll->handle), "WSAStartup");
|
2001-05-16 13:07:51 +08:00
|
|
|
if (wsastartup)
|
2001-09-08 05:32:07 +08:00
|
|
|
{
|
2016-01-12 22:25:15 +08:00
|
|
|
int res = wsastartup (MAKEWORD (2, 2), &wsadata);
|
2000-10-28 02:53:56 +08:00
|
|
|
|
2001-05-16 13:07:51 +08:00
|
|
|
debug_printf ("res %d", res);
|
|
|
|
debug_printf ("wVersion %d", wsadata.wVersion);
|
|
|
|
debug_printf ("wHighVersion %d", wsadata.wHighVersion);
|
|
|
|
debug_printf ("szDescription %s", wsadata.szDescription);
|
|
|
|
debug_printf ("szSystemStatus %s", wsadata.szSystemStatus);
|
|
|
|
debug_printf ("iMaxSockets %d", wsadata.iMaxSockets);
|
|
|
|
debug_printf ("iMaxUdpDg %d", wsadata.iMaxUdpDg);
|
2001-03-22 11:42:08 +08:00
|
|
|
|
2001-05-16 13:07:51 +08:00
|
|
|
wsock_started = 1;
|
2001-09-08 05:32:07 +08:00
|
|
|
}
|
2001-05-16 13:07:51 +08:00
|
|
|
}
|
2001-03-22 11:42:08 +08:00
|
|
|
|
2017-11-27 21:36:06 +08:00
|
|
|
#ifdef __i386__
|
2004-04-16 01:38:02 +08:00
|
|
|
/* Kludge alert. Redirects the return address to dll_chain. */
|
2013-04-23 17:44:36 +08:00
|
|
|
uintptr_t *volatile frame = (uintptr_t *) __builtin_frame_address (0);
|
|
|
|
frame[1] = (uintptr_t) dll_chain;
|
|
|
|
#endif
|
2001-05-16 13:07:51 +08:00
|
|
|
|
2004-06-22 12:00:16 +08:00
|
|
|
InterlockedDecrement (&here);
|
|
|
|
|
2001-10-13 09:35:15 +08:00
|
|
|
volatile retchain ret;
|
2004-04-16 01:38:02 +08:00
|
|
|
/* Set "arguments for dll_chain. */
|
2013-04-23 17:44:36 +08:00
|
|
|
ret.low = (uintptr_t) dll_func_load;
|
|
|
|
ret.high = (uintptr_t) func;
|
2001-05-16 13:07:51 +08:00
|
|
|
return ret.ll;
|
2001-04-16 11:27:16 +08:00
|
|
|
}
|
|
|
|
|
2011-02-27 07:30:33 +08:00
|
|
|
LoadDLLprime (ws2_32, _wsock_init, 0)
|
2001-05-16 13:07:51 +08:00
|
|
|
|
2014-02-27 20:57:27 +08:00
|
|
|
LoadDLLfunc (CheckTokenMembership, 12, advapi32)
|
2011-04-01 20:46:09 +08:00
|
|
|
LoadDLLfunc (CreateProcessAsUserW, 44, advapi32)
|
|
|
|
LoadDLLfunc (DeregisterEventSource, 4, advapi32)
|
2018-12-26 06:38:52 +08:00
|
|
|
LoadDLLfunc (DecryptFileW, 8, advapi32)
|
|
|
|
LoadDLLfunc (EncryptFileW, 4, advapi32)
|
2011-04-01 20:46:09 +08:00
|
|
|
LoadDLLfunc (LogonUserW, 24, advapi32)
|
|
|
|
LoadDLLfunc (LookupAccountNameW, 28, advapi32)
|
|
|
|
LoadDLLfunc (LookupAccountSidW, 28, advapi32)
|
|
|
|
LoadDLLfunc (LsaClose, 4, advapi32)
|
|
|
|
LoadDLLfunc (LsaEnumerateAccountRights, 16, advapi32)
|
|
|
|
LoadDLLfunc (LsaFreeMemory, 4, advapi32)
|
2015-02-24 04:51:12 +08:00
|
|
|
LoadDLLfunc (LsaLookupSids, 20, advapi32)
|
2011-04-01 20:46:09 +08:00
|
|
|
LoadDLLfunc (LsaOpenPolicy, 16, advapi32)
|
|
|
|
LoadDLLfunc (LsaQueryInformationPolicy, 12, advapi32)
|
|
|
|
LoadDLLfunc (LsaRetrievePrivateData, 12, advapi32)
|
|
|
|
LoadDLLfunc (LsaStorePrivateData, 12, advapi32)
|
2012-02-02 21:58:20 +08:00
|
|
|
LoadDLLfunc (RegOpenUserClassesRoot, 16, advapi32)
|
|
|
|
LoadDLLfunc (RegOpenCurrentUser, 8, advapi32)
|
2011-04-01 20:46:09 +08:00
|
|
|
LoadDLLfunc (RegCloseKey, 4, advapi32)
|
|
|
|
LoadDLLfunc (RegCreateKeyExW, 36, advapi32)
|
|
|
|
LoadDLLfunc (RegEnumKeyExW, 32, advapi32)
|
|
|
|
LoadDLLfunc (RegEnumValueW, 32, advapi32)
|
|
|
|
LoadDLLfunc (RegGetKeySecurity, 16, advapi32)
|
|
|
|
LoadDLLfunc (RegOpenKeyExW, 20, advapi32)
|
|
|
|
LoadDLLfunc (RegQueryInfoKeyW, 48, advapi32)
|
|
|
|
LoadDLLfunc (RegQueryValueExW, 24, advapi32)
|
|
|
|
LoadDLLfunc (RegisterEventSourceW, 8, advapi32)
|
|
|
|
LoadDLLfunc (ReportEventW, 36, advapi32)
|
2013-10-15 22:00:37 +08:00
|
|
|
LoadDLLfunc (SystemFunction036, 8, advapi32) /* Aka "RtlGenRandom" */
|
2011-04-01 20:46:09 +08:00
|
|
|
|
2015-08-18 00:29:24 +08:00
|
|
|
LoadDLLfunc (AuthzAccessCheck, 36, authz)
|
|
|
|
LoadDLLfunc (AuthzFreeContext, 4, authz)
|
|
|
|
LoadDLLfunc (AuthzInitializeContextFromSid, 32, authz)
|
|
|
|
LoadDLLfunc (AuthzInitializeContextFromToken, 32, authz)
|
|
|
|
LoadDLLfunc (AuthzInitializeResourceManager, 24, authz)
|
|
|
|
|
Drop NT4 support.
* autoload.cc (DnsQuery_A): Fatal if not available.
(DnsRecordListFree): Ditto.
(DsGetDcNameW): Ditto.
(NetGetAnyDCName): Remove.
(NetGetDCName): Remove.
(EnumProcessModules): Fatal if not available.
(GetModuleFileNameExW): Ditto.
(GetModuleInformation): Ditto.
(GetProcessMemoryInfo): Ditto.
(QueryWorkingSet): Ditto.
(LsaRegisterLogonProcess): Ditto.
* fenv.cc (_feinitialise): Drop supports_sse condition.
* fhandler_disk_file.cc (path_conv::isgood_inode): Fix comment.
(fhandler_base::fstat_by_name): Drop has_fileid_dirinfo condition.
(fhandler_disk_file::opendir): Ditto.
* fhandler_netdrive.cc (fhandler_netdrive::readdir): Fix comment.
* fhandler_proc.cc (format_proc_partitions): Drop NT4-only code.
* fhandler_process.cc (get_process_state): Ditto.
* kernel32.cc (GetWindowsDirectoryW): Remove.
(GetWindowsDirectoryA): Remove.
* miscfuncs.cc (nice_to_winprio): Drop NT4-only code.
* mount.cc (fs_info::update): Fix comments.
* net.cc (get_2k_ifs): Drop NT4-only code.
* sec_auth.cc (get_logon_server): Ditto.
(lsaauth): Drop NT4-specific error handling.
* security.cc (alloc_sd): Set SE_DACL_PROTECTED unconditionally.
* select.cc (select_stuff::wait): Always use MWMO_INPUTAVAILABLE.
(peek_windows): Drop NT4-only condition in call to PeekMessage.
* syscalls.cc (gethostid): Remove NT4-only workaround.
* wincap.cc: Througout, drop has_dacl_protect,
has_broken_if_oper_status, has_process_io_counters,
has_terminal_services, has_extended_priority_class, has_guid_volumes,
has_fileid_dirinfo, has_mwmo_inputavailable and supports_sse from
wincaps.
(wincap_nt4sp4): Remove.
(wincap_minimal): Set to wincap_2000.
(wincapc::init): Rely on availability of OSVERSIONINFOEX structure.
Treat error from GetVersionEx as fatal. Treat NT4 as fatal.
* wincap.h (struct wincaps): Drop has_dacl_protect,
has_broken_if_oper_status, has_process_io_counters,
has_terminal_services, has_extended_priority_class, has_guid_volumes,
has_fileid_dirinfo, has_mwmo_inputavailable and supports_sse flags
and methods.
* winlean.h (GetWindowsDirectoryW) Define as GetSystemWindowsDirectoryW.
(GetWindowsDirectoryA): Define as GetSystemWindowsDirectoryA.
2011-04-04 20:23:36 +08:00
|
|
|
LoadDLLfunc (DnsQuery_A, 24, dnsapi)
|
2016-03-22 04:41:25 +08:00
|
|
|
LoadDLLfunc (DnsFree, 8, dnsapi)
|
2010-09-26 03:57:34 +08:00
|
|
|
|
2013-04-23 17:44:36 +08:00
|
|
|
LoadDLLfunc (GetAdaptersAddresses, 20, iphlpapi)
|
2010-09-26 03:57:34 +08:00
|
|
|
LoadDLLfunc (GetIfEntry, 4, iphlpapi)
|
|
|
|
LoadDLLfunc (GetIpAddrTable, 12, iphlpapi)
|
|
|
|
LoadDLLfunc (GetIpForwardTable, 12, iphlpapi)
|
|
|
|
LoadDLLfunc (GetNetworkParams, 8, iphlpapi)
|
2018-02-06 04:05:09 +08:00
|
|
|
LoadDLLfunc (GetTcpTable, 12, iphlpapi)
|
|
|
|
LoadDLLfunc (GetTcp6Table, 12, iphlpapi)
|
2011-03-29 19:07:23 +08:00
|
|
|
LoadDLLfunc (GetUdpTable, 12, iphlpapi)
|
2015-12-15 22:50:13 +08:00
|
|
|
LoadDLLfunc (if_indextoname, 8, iphlpapi)
|
|
|
|
LoadDLLfunc (if_nametoindex, 4, iphlpapi)
|
2010-09-26 03:57:34 +08:00
|
|
|
|
2015-08-27 19:34:10 +08:00
|
|
|
LoadDLLfuncEx2 (DiscardVirtualMemory, 8, kernel32, 1, 127)
|
2018-08-07 20:51:10 +08:00
|
|
|
LoadDLLfunc (GetCurrentProcessorNumberEx, 4, kernel32)
|
2015-08-13 23:59:47 +08:00
|
|
|
LoadDLLfuncEx (GetLogicalProcessorInformationEx, 12, kernel32, 1)
|
2019-12-17 23:44:54 +08:00
|
|
|
LoadDLLfuncEx (GetProcessGroupAffinity, 12, kernel32, 1)
|
2013-06-14 23:41:17 +08:00
|
|
|
LoadDLLfunc (GetSystemTimePreciseAsFileTime, 4, kernel32)
|
2019-12-17 23:44:54 +08:00
|
|
|
LoadDLLfuncEx (GetThreadGroupAffinity, 8, kernel32, 1)
|
2015-08-27 19:34:10 +08:00
|
|
|
LoadDLLfuncEx (PrefetchVirtualMemory, 16, kernel32, 1)
|
2015-08-13 23:59:47 +08:00
|
|
|
LoadDLLfunc (SetThreadGroupAffinity, 12, kernel32)
|
2010-09-26 03:57:34 +08:00
|
|
|
|
2018-11-27 20:47:02 +08:00
|
|
|
/* MSDN claims these are exported by kernel32.dll, but only
|
|
|
|
QueryUnbiasedInterruptTime actually is. The others are only
|
|
|
|
available via KernelBase.dll. */
|
|
|
|
LoadDLLfunc (QueryInterruptTime, 4, KernelBase)
|
|
|
|
LoadDLLfunc (QueryInterruptTimePrecise, 4, KernelBase)
|
|
|
|
LoadDLLfunc (QueryUnbiasedInterruptTime, 4, KernelBase)
|
|
|
|
LoadDLLfunc (QueryUnbiasedInterruptTimePrecise, 4, KernelBase)
|
|
|
|
|
2014-02-10 03:44:56 +08:00
|
|
|
/* ldap functions are cdecl! */
|
|
|
|
#pragma push_macro ("mangle")
|
|
|
|
#undef mangle
|
|
|
|
#define mangle(name, n) #name
|
|
|
|
LoadDLLfunc (ldap_bind_s, 0, wldap32)
|
2014-05-23 04:07:25 +08:00
|
|
|
LoadDLLfunc (ldap_count_entries, 0, wldap32)
|
2014-02-10 03:44:56 +08:00
|
|
|
LoadDLLfunc (ldap_count_valuesW, 0, wldap32)
|
|
|
|
LoadDLLfunc (ldap_first_entry, 0, wldap32)
|
2014-05-23 04:07:25 +08:00
|
|
|
LoadDLLfunc (ldap_get_next_page_s, 0, wldap32)
|
2014-02-10 03:44:56 +08:00
|
|
|
LoadDLLfunc (ldap_get_valuesW, 0, wldap32)
|
|
|
|
LoadDLLfunc (ldap_get_values_lenW, 0, wldap32)
|
|
|
|
LoadDLLfunc (ldap_initW, 0, wldap32)
|
2014-05-23 18:29:11 +08:00
|
|
|
LoadDLLfunc (ldap_msgfree, 0, wldap32)
|
2014-02-10 03:44:56 +08:00
|
|
|
LoadDLLfunc (ldap_next_entry, 0, wldap32)
|
2014-05-23 04:07:25 +08:00
|
|
|
LoadDLLfunc (ldap_search_abandon_page, 0, wldap32)
|
|
|
|
LoadDLLfunc (ldap_search_init_pageW, 0, wldap32)
|
2014-06-25 17:10:50 +08:00
|
|
|
LoadDLLfunc (ldap_search_sW, 0, wldap32)
|
2014-02-10 03:44:56 +08:00
|
|
|
LoadDLLfunc (ldap_set_option, 0, wldap32)
|
|
|
|
LoadDLLfunc (ldap_sslinitW, 0, wldap32)
|
|
|
|
LoadDLLfunc (ldap_unbind, 0, wldap32)
|
|
|
|
LoadDLLfunc (ldap_value_freeW, 0, wldap32)
|
|
|
|
LoadDLLfunc (ldap_value_free_len, 0, wldap32)
|
|
|
|
LoadDLLfunc (LdapGetLastError, 0, wldap32)
|
2014-06-25 17:10:50 +08:00
|
|
|
LoadDLLfunc (LdapMapErrorToWin32, 0, wldap32)
|
2014-02-10 03:44:56 +08:00
|
|
|
#pragma pop_macro ("mangle")
|
|
|
|
|
2010-09-26 03:57:34 +08:00
|
|
|
LoadDLLfunc (WNetCloseEnum, 4, mpr)
|
2014-08-19 16:41:40 +08:00
|
|
|
LoadDLLfunc (WNetEnumResourceW, 16, mpr)
|
|
|
|
LoadDLLfunc (WNetGetProviderNameW, 12, mpr)
|
|
|
|
LoadDLLfunc (WNetGetResourceInformationW, 16, mpr)
|
|
|
|
LoadDLLfunc (WNetOpenEnumW, 20, mpr)
|
2010-09-26 03:57:34 +08:00
|
|
|
|
2014-02-10 03:44:56 +08:00
|
|
|
LoadDLLfunc (DsEnumerateDomainTrustsW, 16, netapi32)
|
Drop NT4 support.
* autoload.cc (DnsQuery_A): Fatal if not available.
(DnsRecordListFree): Ditto.
(DsGetDcNameW): Ditto.
(NetGetAnyDCName): Remove.
(NetGetDCName): Remove.
(EnumProcessModules): Fatal if not available.
(GetModuleFileNameExW): Ditto.
(GetModuleInformation): Ditto.
(GetProcessMemoryInfo): Ditto.
(QueryWorkingSet): Ditto.
(LsaRegisterLogonProcess): Ditto.
* fenv.cc (_feinitialise): Drop supports_sse condition.
* fhandler_disk_file.cc (path_conv::isgood_inode): Fix comment.
(fhandler_base::fstat_by_name): Drop has_fileid_dirinfo condition.
(fhandler_disk_file::opendir): Ditto.
* fhandler_netdrive.cc (fhandler_netdrive::readdir): Fix comment.
* fhandler_proc.cc (format_proc_partitions): Drop NT4-only code.
* fhandler_process.cc (get_process_state): Ditto.
* kernel32.cc (GetWindowsDirectoryW): Remove.
(GetWindowsDirectoryA): Remove.
* miscfuncs.cc (nice_to_winprio): Drop NT4-only code.
* mount.cc (fs_info::update): Fix comments.
* net.cc (get_2k_ifs): Drop NT4-only code.
* sec_auth.cc (get_logon_server): Ditto.
(lsaauth): Drop NT4-specific error handling.
* security.cc (alloc_sd): Set SE_DACL_PROTECTED unconditionally.
* select.cc (select_stuff::wait): Always use MWMO_INPUTAVAILABLE.
(peek_windows): Drop NT4-only condition in call to PeekMessage.
* syscalls.cc (gethostid): Remove NT4-only workaround.
* wincap.cc: Througout, drop has_dacl_protect,
has_broken_if_oper_status, has_process_io_counters,
has_terminal_services, has_extended_priority_class, has_guid_volumes,
has_fileid_dirinfo, has_mwmo_inputavailable and supports_sse from
wincaps.
(wincap_nt4sp4): Remove.
(wincap_minimal): Set to wincap_2000.
(wincapc::init): Rely on availability of OSVERSIONINFOEX structure.
Treat error from GetVersionEx as fatal. Treat NT4 as fatal.
* wincap.h (struct wincaps): Drop has_dacl_protect,
has_broken_if_oper_status, has_process_io_counters,
has_terminal_services, has_extended_priority_class, has_guid_volumes,
has_fileid_dirinfo, has_mwmo_inputavailable and supports_sse flags
and methods.
* winlean.h (GetWindowsDirectoryW) Define as GetSystemWindowsDirectoryW.
(GetWindowsDirectoryA): Define as GetSystemWindowsDirectoryA.
2011-04-04 20:23:36 +08:00
|
|
|
LoadDLLfunc (DsGetDcNameW, 24, netapi32)
|
2000-10-28 02:53:56 +08:00
|
|
|
LoadDLLfunc (NetApiBufferFree, 4, netapi32)
|
2014-02-17 23:36:33 +08:00
|
|
|
LoadDLLfunc (NetGroupEnum, 28, netapi32)
|
|
|
|
LoadDLLfunc (NetLocalGroupEnum, 28, netapi32)
|
2014-02-11 19:51:29 +08:00
|
|
|
LoadDLLfunc (NetLocalGroupGetInfo, 16, netapi32)
|
2012-02-16 19:02:05 +08:00
|
|
|
LoadDLLfunc (NetUseGetInfo, 16, netapi32)
|
2014-02-17 23:36:33 +08:00
|
|
|
LoadDLLfunc (NetUserEnum, 32, netapi32)
|
* autoload.cc: Add load statements for `LookupAccountNameW',
`LsaClose', `LsaEnumerateAccountRights', `LsaFreeMemory',
`LsaOpenPolicy', `LsaQueryInformationPolicy', `NetLocalGroupEnum',
`NetLocalGroupGetMembers', `NetServerEnum', `NetUserGetGroups' and
`NtCreateToken'.
* ntdll.h: Add declaration for `NtCreateToken'.
* sec_helper.cc: Add `well_known_local_sid', `well_known_dialup_sid',
`well_known_network_sid', `well_known_batch_sid',
`well_known_interactive_sid', `well_known_service_sid' and
`well_known_authenticated_users_sid'.
(cygsid::string): Define as const method.
(cygsid::get_sid): Set psid to NO_SID on error.
(cygsid::getfromstr): Ditto.
(cygsid::getfrompw): Simplify.
(cygsid::getfromgr): Check for gr == NULL.
(legal_sid_type): Move to security.h.
(set_process_privilege): Return -1 on error, otherwise 0 or 1 related
to previous privilege setting.
* security.cc (extract_nt_dom_user): Remove `static'.
(lsa2wchar): New function.
(open_local_policy): Ditto.
(close_local_policy): Ditto.
(get_lsa_srv_inf): Ditto.
(get_logon_server): Ditto.
(get_logon_server_and_user_domain): Ditto.
(get_user_groups): Ditto.
(is_group_member): Ditto.
(get_user_local_groups): Ditto.
(sid_in_token_groups): Ditto.
(get_user_primary_group): Ditto.
(get_group_sidlist): Ditto.
(get_system_priv_list): Ditto.
(get_priv_list): Ditto.
(get_dacl): Ditto.
(create_token): Ditto.
(subauth): Return immediately if SE_TCB_NAME can't be assigned.
Change all return statements in case of error to jumps to `out'
label. Add `out' label to support cleanup.
* security.h: Add extern declarations for `well_known_local_sid',
`well_known_dialup_sid', `well_known_network_sid',
`well_known_batch_sid', `well_known_interactive_sid',
`well_known_service_sid' and `well_known_authenticated_users_sid'.
Add extern declarations for functions `create_token',
`extract_nt_dom_user' and `get_logon_server_and_user_domain'.
(class cygsid): Add method `assign'. Change operator= to call new
`assign' method. Add `debug_print' method.
(class cygsidlist): New class.
(legal_sid_type): Moved from sec_helper.cc to here.
* spawn.cc (spawn_guts) Revert reversion of previous patch.
Call `RevertToSelf' and `ImpersonateLoggedOnUser' instead of `seteuid'
again.
* syscalls.cc (seteuid): Rearranged. Call `create_token' now when
needed. Call `subauth' if `create_token' fails. Try setting token
owner and primary group only if token was not explicitely created
by `create_token'.
* uinfo.cc (internal_getlogin): Try harder to generate correct user
information. Especially don't trust return value of `GetUserName'.
2001-05-20 16:10:47 +08:00
|
|
|
LoadDLLfunc (NetUserGetGroups, 28, netapi32)
|
|
|
|
LoadDLLfunc (NetUserGetInfo, 16, netapi32)
|
2010-09-26 03:57:34 +08:00
|
|
|
LoadDLLfunc (NetUserGetLocalGroups, 32, netapi32)
|
2000-10-28 02:53:56 +08:00
|
|
|
|
2010-09-26 03:57:34 +08:00
|
|
|
LoadDLLfunc (CoTaskMemFree, 4, ole32)
|
|
|
|
|
2019-02-23 01:27:38 +08:00
|
|
|
LoadDLLfunc (LsaConnectUntrusted, 4, secur32)
|
2011-02-27 04:47:56 +08:00
|
|
|
LoadDLLfunc (LsaDeregisterLogonProcess, 4, secur32)
|
|
|
|
LoadDLLfunc (LsaFreeReturnBuffer, 4, secur32)
|
|
|
|
LoadDLLfunc (LsaLogonUser, 56, secur32)
|
|
|
|
LoadDLLfunc (LsaLookupAuthenticationPackage, 12, secur32)
|
Drop NT4 support.
* autoload.cc (DnsQuery_A): Fatal if not available.
(DnsRecordListFree): Ditto.
(DsGetDcNameW): Ditto.
(NetGetAnyDCName): Remove.
(NetGetDCName): Remove.
(EnumProcessModules): Fatal if not available.
(GetModuleFileNameExW): Ditto.
(GetModuleInformation): Ditto.
(GetProcessMemoryInfo): Ditto.
(QueryWorkingSet): Ditto.
(LsaRegisterLogonProcess): Ditto.
* fenv.cc (_feinitialise): Drop supports_sse condition.
* fhandler_disk_file.cc (path_conv::isgood_inode): Fix comment.
(fhandler_base::fstat_by_name): Drop has_fileid_dirinfo condition.
(fhandler_disk_file::opendir): Ditto.
* fhandler_netdrive.cc (fhandler_netdrive::readdir): Fix comment.
* fhandler_proc.cc (format_proc_partitions): Drop NT4-only code.
* fhandler_process.cc (get_process_state): Ditto.
* kernel32.cc (GetWindowsDirectoryW): Remove.
(GetWindowsDirectoryA): Remove.
* miscfuncs.cc (nice_to_winprio): Drop NT4-only code.
* mount.cc (fs_info::update): Fix comments.
* net.cc (get_2k_ifs): Drop NT4-only code.
* sec_auth.cc (get_logon_server): Ditto.
(lsaauth): Drop NT4-specific error handling.
* security.cc (alloc_sd): Set SE_DACL_PROTECTED unconditionally.
* select.cc (select_stuff::wait): Always use MWMO_INPUTAVAILABLE.
(peek_windows): Drop NT4-only condition in call to PeekMessage.
* syscalls.cc (gethostid): Remove NT4-only workaround.
* wincap.cc: Througout, drop has_dacl_protect,
has_broken_if_oper_status, has_process_io_counters,
has_terminal_services, has_extended_priority_class, has_guid_volumes,
has_fileid_dirinfo, has_mwmo_inputavailable and supports_sse from
wincaps.
(wincap_nt4sp4): Remove.
(wincap_minimal): Set to wincap_2000.
(wincapc::init): Rely on availability of OSVERSIONINFOEX structure.
Treat error from GetVersionEx as fatal. Treat NT4 as fatal.
* wincap.h (struct wincaps): Drop has_dacl_protect,
has_broken_if_oper_status, has_process_io_counters,
has_terminal_services, has_extended_priority_class, has_guid_volumes,
has_fileid_dirinfo, has_mwmo_inputavailable and supports_sse flags
and methods.
* winlean.h (GetWindowsDirectoryW) Define as GetSystemWindowsDirectoryW.
(GetWindowsDirectoryA): Define as GetSystemWindowsDirectoryA.
2011-04-04 20:23:36 +08:00
|
|
|
LoadDLLfunc (LsaRegisterLogonProcess, 12, secur32)
|
2019-01-27 01:33:41 +08:00
|
|
|
LoadDLLfunc (TranslateNameW, 20, secur32)
|
2010-09-26 03:57:34 +08:00
|
|
|
|
|
|
|
LoadDLLfunc (SHGetDesktopFolder, 4, shell32)
|
2001-05-01 05:19:42 +08:00
|
|
|
|
2018-09-06 05:39:25 +08:00
|
|
|
LoadDLLfunc (CreateFontW, 56, gdi32)
|
|
|
|
LoadDLLfunc (DeleteObject, 4, gdi32)
|
|
|
|
LoadDLLfunc (EnumFontFamiliesExW, 20, gdi32)
|
|
|
|
LoadDLLfunc (GetGlyphIndicesW, 20, gdi32)
|
|
|
|
LoadDLLfunc (SelectObject, 8, gdi32)
|
|
|
|
|
2000-10-28 02:53:56 +08:00
|
|
|
LoadDLLfunc (CloseClipboard, 0, user32)
|
2007-08-18 03:58:57 +08:00
|
|
|
LoadDLLfunc (CloseDesktop, 4, user32)
|
2005-11-24 10:34:31 +08:00
|
|
|
LoadDLLfunc (CloseWindowStation, 4, user32)
|
2008-03-07 19:24:51 +08:00
|
|
|
LoadDLLfunc (CreateDesktopW, 24, user32)
|
2011-05-01 22:35:12 +08:00
|
|
|
LoadDLLfunc (CreateWindowExW, 48, user32)
|
2008-03-07 19:24:51 +08:00
|
|
|
LoadDLLfunc (CreateWindowStationW, 16, user32)
|
2011-05-01 22:35:12 +08:00
|
|
|
LoadDLLfunc (DefWindowProcW, 16, user32)
|
2019-01-21 05:47:52 +08:00
|
|
|
LoadDLLfunc (DestroyWindow, 4, user32)
|
2011-05-01 22:35:12 +08:00
|
|
|
LoadDLLfunc (DispatchMessageW, 4, user32)
|
2001-03-23 06:11:34 +08:00
|
|
|
LoadDLLfunc (EmptyClipboard, 0, user32)
|
2011-05-29 02:17:09 +08:00
|
|
|
LoadDLLfunc (EnumWindows, 8, user32)
|
2000-10-28 02:53:56 +08:00
|
|
|
LoadDLLfunc (GetClipboardData, 4, user32)
|
2018-09-06 05:39:25 +08:00
|
|
|
LoadDLLfunc (GetDC, 4, user32)
|
2003-09-27 11:44:31 +08:00
|
|
|
LoadDLLfunc (GetForegroundWindow, 0, user32)
|
2001-03-06 20:05:45 +08:00
|
|
|
LoadDLLfunc (GetKeyboardLayout, 4, user32)
|
2011-05-01 22:35:12 +08:00
|
|
|
LoadDLLfunc (GetMessageW, 16, user32)
|
2001-03-23 06:11:34 +08:00
|
|
|
LoadDLLfunc (GetPriorityClipboardFormat, 8, user32)
|
2000-10-28 02:53:56 +08:00
|
|
|
LoadDLLfunc (GetProcessWindowStation, 0, user32)
|
|
|
|
LoadDLLfunc (GetThreadDesktop, 4, user32)
|
2008-03-07 19:24:51 +08:00
|
|
|
LoadDLLfunc (GetUserObjectInformationW, 20, user32)
|
2010-09-26 03:57:34 +08:00
|
|
|
LoadDLLfunc (GetWindowThreadProcessId, 8, user32)
|
2003-02-14 07:51:41 +08:00
|
|
|
LoadDLLfunc (MessageBeep, 4, user32)
|
2011-05-01 22:35:12 +08:00
|
|
|
LoadDLLfunc (MessageBoxW, 16, user32)
|
2010-08-30 18:39:43 +08:00
|
|
|
LoadDLLfunc (MsgWaitForMultipleObjectsEx, 20, user32)
|
2000-10-28 02:53:56 +08:00
|
|
|
LoadDLLfunc (OpenClipboard, 4, user32)
|
2011-05-01 22:35:12 +08:00
|
|
|
LoadDLLfunc (PeekMessageW, 20, user32)
|
|
|
|
LoadDLLfunc (PostMessageW, 16, user32)
|
2000-10-28 02:53:56 +08:00
|
|
|
LoadDLLfunc (PostQuitMessage, 4, user32)
|
2011-05-01 22:35:12 +08:00
|
|
|
LoadDLLfunc (RegisterClassW, 4, user32)
|
|
|
|
LoadDLLfunc (RegisterClipboardFormatW, 4, user32)
|
|
|
|
LoadDLLfunc (SendNotifyMessageW, 16, user32)
|
2001-03-23 06:11:34 +08:00
|
|
|
LoadDLLfunc (SetClipboardData, 8, user32)
|
2009-07-05 07:51:10 +08:00
|
|
|
LoadDLLfunc (SetParent, 8, user32)
|
2003-04-20 16:56:42 +08:00
|
|
|
LoadDLLfunc (SetProcessWindowStation, 4, user32)
|
2010-09-26 03:57:34 +08:00
|
|
|
LoadDLLfunc (SetThreadDesktop, 4, user32)
|
2019-01-21 05:47:52 +08:00
|
|
|
LoadDLLfunc (UnregisterClassW, 8, user32)
|
2010-09-26 03:57:34 +08:00
|
|
|
|
2015-03-11 19:56:53 +08:00
|
|
|
LoadDLLfuncEx (CreateEnvironmentBlock, 12, userenv, 1)
|
2014-12-02 18:49:47 +08:00
|
|
|
LoadDLLfuncEx2 (CreateProfile, 16, userenv, 1, 1)
|
2014-12-02 18:16:03 +08:00
|
|
|
LoadDLLfunc (DestroyEnvironmentBlock, 4, userenv)
|
2014-12-02 18:49:47 +08:00
|
|
|
LoadDLLfunc (LoadUserProfileW, 8, userenv)
|
2014-12-02 18:16:03 +08:00
|
|
|
|
2011-03-29 18:21:30 +08:00
|
|
|
LoadDLLfuncEx3 (waveInAddBuffer, 12, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveInClose, 4, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveInGetNumDevs, 0, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveInOpen, 24, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveInPrepareHeader, 12, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveInReset, 4, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveInStart, 4, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveInUnprepareHeader, 12, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveOutClose, 4, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveOutGetNumDevs, 0, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveOutGetVolume, 8, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveOutOpen, 24, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveOutPrepareHeader, 12, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveOutReset, 4, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveOutSetVolume, 8, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveOutUnprepareHeader, 12, winmm, 1, 0, 1)
|
|
|
|
LoadDLLfuncEx3 (waveOutWrite, 12, winmm, 1, 0, 1)
|
2000-10-28 02:53:56 +08:00
|
|
|
|
2005-10-18 05:22:18 +08:00
|
|
|
LoadDLLfunc (accept, 12, ws2_32)
|
|
|
|
LoadDLLfunc (bind, 12, ws2_32)
|
|
|
|
LoadDLLfunc (closesocket, 4, ws2_32)
|
|
|
|
LoadDLLfunc (connect, 12, ws2_32)
|
2013-11-19 21:29:37 +08:00
|
|
|
LoadDLLfunc (FreeAddrInfoW, 4, ws2_32)
|
|
|
|
LoadDLLfunc (GetAddrInfoW, 16, ws2_32)
|
|
|
|
LoadDLLfunc (GetNameInfoW, 28, ws2_32)
|
2005-10-18 05:22:18 +08:00
|
|
|
LoadDLLfunc (gethostbyaddr, 12, ws2_32)
|
|
|
|
LoadDLLfunc (gethostbyname, 4, ws2_32)
|
2010-09-26 03:57:34 +08:00
|
|
|
LoadDLLfunc (gethostname, 8, ws2_32)
|
2005-10-18 05:22:18 +08:00
|
|
|
LoadDLLfunc (getpeername, 12, ws2_32)
|
|
|
|
LoadDLLfunc (getprotobyname, 4, ws2_32)
|
|
|
|
LoadDLLfunc (getprotobynumber, 4, ws2_32)
|
|
|
|
LoadDLLfunc (getservbyname, 8, ws2_32)
|
|
|
|
LoadDLLfunc (getservbyport, 8, ws2_32)
|
|
|
|
LoadDLLfunc (getsockname, 12, ws2_32)
|
|
|
|
LoadDLLfunc (getsockopt, 20, ws2_32)
|
|
|
|
LoadDLLfunc (ioctlsocket, 12, ws2_32)
|
|
|
|
LoadDLLfunc (listen, 8, ws2_32)
|
|
|
|
LoadDLLfunc (setsockopt, 20, ws2_32)
|
|
|
|
LoadDLLfunc (shutdown, 8, ws2_32)
|
|
|
|
LoadDLLfunc (socket, 12, ws2_32)
|
|
|
|
LoadDLLfunc (WSAAsyncSelect, 16, ws2_32)
|
2009-11-17 18:43:01 +08:00
|
|
|
LoadDLLfunc (WSADuplicateSocketW, 12, ws2_32)
|
2006-07-31 16:59:56 +08:00
|
|
|
LoadDLLfunc (WSAEnumNetworkEvents, 12, ws2_32)
|
|
|
|
LoadDLLfunc (WSAEventSelect, 12, ws2_32)
|
2005-10-18 05:22:18 +08:00
|
|
|
LoadDLLfunc (WSAGetLastError, 0, ws2_32)
|
2009-01-20 19:16:59 +08:00
|
|
|
LoadDLLfunc (WSAIoctl, 36, ws2_32)
|
2009-09-22 22:27:57 +08:00
|
|
|
LoadDLLfunc (WSARecv, 28, ws2_32)
|
2005-10-18 05:22:18 +08:00
|
|
|
LoadDLLfunc (WSARecvFrom, 36, ws2_32)
|
2009-01-20 19:16:59 +08:00
|
|
|
LoadDLLfunc (WSASendMsg, 24, ws2_32)
|
2005-10-18 05:22:18 +08:00
|
|
|
LoadDLLfunc (WSASendTo, 36, ws2_32)
|
|
|
|
LoadDLLfunc (WSASetLastError, 4, ws2_32)
|
2009-11-17 18:43:01 +08:00
|
|
|
LoadDLLfunc (WSASocketW, 24, ws2_32)
|
2005-10-18 05:22:18 +08:00
|
|
|
// LoadDLLfunc (WSAStartup, 8, ws2_32)
|
|
|
|
LoadDLLfunc (WSAWaitForMultipleEvents, 20, ws2_32)
|
2017-03-22 03:17:42 +08:00
|
|
|
|
2019-03-13 00:09:42 +08:00
|
|
|
LoadDLLfunc (PdhAddEnglishCounterW, 16, pdh)
|
2017-03-22 03:17:42 +08:00
|
|
|
LoadDLLfunc (PdhCollectQueryData, 4, pdh)
|
|
|
|
LoadDLLfunc (PdhGetFormattedCounterValue, 16, pdh)
|
2019-03-13 00:09:42 +08:00
|
|
|
LoadDLLfunc (PdhOpenQueryW, 12, pdh)
|
2019-09-15 18:55:44 +08:00
|
|
|
LoadDLLfuncEx (CreatePseudoConsole, 20, kernel32, 1)
|
|
|
|
LoadDLLfuncEx (ResizePseudoConsole, 8, kernel32, 1)
|
|
|
|
LoadDLLfuncEx (ClosePseudoConsole, 4, kernel32, 1)
|
2000-10-28 02:53:56 +08:00
|
|
|
}
|