1997-01-20 00:38:22 +00:00
|
|
|
|
/* Utility and Unix shadow routines for GNU Emacs on the Microsoft W32 API.
|
2001-04-10 12:15:25 +00:00
|
|
|
|
Copyright (C) 1994, 1995, 2000, 2001 Free Software Foundation, Inc.
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
1996-01-15 09:18:04 +00:00
|
|
|
|
This file is part of GNU Emacs.
|
|
|
|
|
|
|
|
|
|
GNU Emacs is free software; you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
|
the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
|
|
GNU Emacs is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with GNU Emacs; see the file COPYING. If not, write to
|
|
|
|
|
the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
|
Boston, MA 02111-1307, USA.
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
|
|
|
|
Geoff Voelker (voelker@cs.washington.edu) 7-29-94
|
|
|
|
|
*/
|
|
|
|
|
|
1996-01-21 00:31:34 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
#include <stddef.h> /* for offsetof */
|
1994-11-01 10:27:02 +00:00
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <io.h>
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#include <errno.h>
|
1994-11-01 10:27:02 +00:00
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
#include <ctype.h>
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#include <signal.h>
|
1998-06-05 16:08:32 +00:00
|
|
|
|
#include <sys/file.h>
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#include <sys/time.h>
|
1998-04-23 23:41:12 +00:00
|
|
|
|
#include <sys/utime.h>
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
/* must include CRT headers *before* config.h */
|
2002-01-01 19:15:26 +00:00
|
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#undef access
|
|
|
|
|
#undef chdir
|
|
|
|
|
#undef chmod
|
|
|
|
|
#undef creat
|
|
|
|
|
#undef ctime
|
|
|
|
|
#undef fopen
|
|
|
|
|
#undef link
|
|
|
|
|
#undef mkdir
|
|
|
|
|
#undef mktemp
|
|
|
|
|
#undef open
|
|
|
|
|
#undef rename
|
|
|
|
|
#undef rmdir
|
|
|
|
|
#undef unlink
|
|
|
|
|
|
|
|
|
|
#undef close
|
|
|
|
|
#undef dup
|
|
|
|
|
#undef dup2
|
|
|
|
|
#undef pipe
|
|
|
|
|
#undef read
|
|
|
|
|
#undef write
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
1999-06-16 20:00:19 +00:00
|
|
|
|
#undef strerror
|
|
|
|
|
|
1994-11-01 10:27:02 +00:00
|
|
|
|
#include "lisp.h"
|
|
|
|
|
|
|
|
|
|
#include <pwd.h>
|
2003-11-22 23:01:18 +00:00
|
|
|
|
#include <grp.h>
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
2000-09-03 17:41:20 +00:00
|
|
|
|
#ifdef __GNUC__
|
|
|
|
|
#define _ANONYMOUS_UNION
|
|
|
|
|
#define _ANONYMOUS_STRUCT
|
|
|
|
|
#endif
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#include <windows.h>
|
1996-01-21 00:31:34 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#ifdef HAVE_SOCKETS /* TCP connection support, if kernel can do it */
|
|
|
|
|
#include <sys/socket.h>
|
|
|
|
|
#undef socket
|
|
|
|
|
#undef bind
|
|
|
|
|
#undef connect
|
|
|
|
|
#undef htons
|
|
|
|
|
#undef ntohs
|
|
|
|
|
#undef inet_addr
|
|
|
|
|
#undef gethostname
|
|
|
|
|
#undef gethostbyname
|
|
|
|
|
#undef getservbyname
|
2002-05-03 20:40:03 +00:00
|
|
|
|
#undef getpeername
|
1997-07-10 20:10:09 +00:00
|
|
|
|
#undef shutdown
|
2002-03-20 20:38:44 +00:00
|
|
|
|
#undef setsockopt
|
|
|
|
|
#undef listen
|
|
|
|
|
#undef getsockname
|
|
|
|
|
#undef accept
|
|
|
|
|
#undef recvfrom
|
|
|
|
|
#undef sendto
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#endif
|
1996-01-21 00:31:34 +00:00
|
|
|
|
|
1996-11-19 07:04:49 +00:00
|
|
|
|
#include "w32.h"
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#include "ndir.h"
|
1996-11-19 07:04:49 +00:00
|
|
|
|
#include "w32heap.h"
|
2001-01-13 00:00:26 +00:00
|
|
|
|
#include "systime.h"
|
|
|
|
|
|
2002-12-18 06:16:28 +00:00
|
|
|
|
void globals_of_w32 ();
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
extern Lisp_Object Vw32_downcase_file_names;
|
|
|
|
|
extern Lisp_Object Vw32_generate_fake_inodes;
|
|
|
|
|
extern Lisp_Object Vw32_get_true_file_attributes;
|
2004-05-17 21:33:16 +00:00
|
|
|
|
extern int w32_num_mouse_buttons;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
|
2002-11-17 22:35:26 +00:00
|
|
|
|
|
2002-12-18 06:16:28 +00:00
|
|
|
|
/*
|
|
|
|
|
Initialization states
|
|
|
|
|
*/
|
|
|
|
|
static BOOL g_b_init_is_windows_9x;
|
|
|
|
|
static BOOL g_b_init_open_process_token;
|
|
|
|
|
static BOOL g_b_init_get_token_information;
|
|
|
|
|
static BOOL g_b_init_lookup_account_sid;
|
|
|
|
|
static BOOL g_b_init_get_sid_identifier_authority;
|
|
|
|
|
|
2002-11-17 22:35:26 +00:00
|
|
|
|
/*
|
|
|
|
|
BEGIN: Wrapper functions around OpenProcessToken
|
|
|
|
|
and other functions in advapi32.dll that are only
|
|
|
|
|
supported in Windows NT / 2k / XP
|
|
|
|
|
*/
|
|
|
|
|
/* ** Function pointer typedefs ** */
|
|
|
|
|
typedef BOOL (WINAPI * OpenProcessToken_Proc) (
|
|
|
|
|
HANDLE ProcessHandle,
|
|
|
|
|
DWORD DesiredAccess,
|
|
|
|
|
PHANDLE TokenHandle);
|
|
|
|
|
typedef BOOL (WINAPI * GetTokenInformation_Proc) (
|
|
|
|
|
HANDLE TokenHandle,
|
|
|
|
|
TOKEN_INFORMATION_CLASS TokenInformationClass,
|
|
|
|
|
LPVOID TokenInformation,
|
|
|
|
|
DWORD TokenInformationLength,
|
|
|
|
|
PDWORD ReturnLength);
|
|
|
|
|
#ifdef _UNICODE
|
|
|
|
|
const char * const LookupAccountSid_Name = "LookupAccountSidW";
|
|
|
|
|
#else
|
|
|
|
|
const char * const LookupAccountSid_Name = "LookupAccountSidA";
|
|
|
|
|
#endif
|
|
|
|
|
typedef BOOL (WINAPI * LookupAccountSid_Proc) (
|
|
|
|
|
LPCTSTR lpSystemName,
|
|
|
|
|
PSID Sid,
|
|
|
|
|
LPTSTR Name,
|
|
|
|
|
LPDWORD cbName,
|
|
|
|
|
LPTSTR DomainName,
|
|
|
|
|
LPDWORD cbDomainName,
|
|
|
|
|
PSID_NAME_USE peUse);
|
|
|
|
|
typedef PSID_IDENTIFIER_AUTHORITY (WINAPI * GetSidIdentifierAuthority_Proc) (
|
|
|
|
|
PSID pSid);
|
|
|
|
|
|
|
|
|
|
/* ** A utility function ** */
|
|
|
|
|
static BOOL is_windows_9x ()
|
|
|
|
|
{
|
2002-12-18 06:16:28 +00:00
|
|
|
|
static BOOL s_b_ret=0;
|
2002-11-17 22:35:26 +00:00
|
|
|
|
OSVERSIONINFO os_ver;
|
2002-12-18 06:16:28 +00:00
|
|
|
|
if (g_b_init_is_windows_9x == 0)
|
2002-11-17 22:35:26 +00:00
|
|
|
|
{
|
2002-12-18 06:16:28 +00:00
|
|
|
|
g_b_init_is_windows_9x = 1;
|
|
|
|
|
ZeroMemory(&os_ver, sizeof(OSVERSIONINFO));
|
|
|
|
|
os_ver.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
|
|
|
|
|
if (GetVersionEx (&os_ver))
|
|
|
|
|
{
|
|
|
|
|
s_b_ret = (os_ver.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS);
|
|
|
|
|
}
|
2002-11-17 22:35:26 +00:00
|
|
|
|
}
|
2002-12-18 06:16:28 +00:00
|
|
|
|
return s_b_ret;
|
2002-11-17 22:35:26 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ** The wrapper functions ** */
|
|
|
|
|
|
|
|
|
|
BOOL WINAPI open_process_token (
|
|
|
|
|
HANDLE ProcessHandle,
|
|
|
|
|
DWORD DesiredAccess,
|
|
|
|
|
PHANDLE TokenHandle)
|
|
|
|
|
{
|
2002-12-18 06:16:28 +00:00
|
|
|
|
static OpenProcessToken_Proc s_pfn_Open_Process_Token = NULL;
|
2002-11-17 22:35:26 +00:00
|
|
|
|
HMODULE hm_advapi32 = NULL;
|
|
|
|
|
if (is_windows_9x () == TRUE)
|
|
|
|
|
{
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
2002-12-18 06:16:28 +00:00
|
|
|
|
if (g_b_init_open_process_token == 0)
|
|
|
|
|
{
|
|
|
|
|
g_b_init_open_process_token = 1;
|
|
|
|
|
hm_advapi32 = LoadLibrary ("Advapi32.dll");
|
|
|
|
|
s_pfn_Open_Process_Token =
|
|
|
|
|
(OpenProcessToken_Proc) GetProcAddress (hm_advapi32, "OpenProcessToken");
|
|
|
|
|
}
|
|
|
|
|
if (s_pfn_Open_Process_Token == NULL)
|
2002-11-17 22:35:26 +00:00
|
|
|
|
{
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
return (
|
2002-12-18 06:16:28 +00:00
|
|
|
|
s_pfn_Open_Process_Token (
|
2002-11-17 22:35:26 +00:00
|
|
|
|
ProcessHandle,
|
|
|
|
|
DesiredAccess,
|
|
|
|
|
TokenHandle)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BOOL WINAPI get_token_information (
|
|
|
|
|
HANDLE TokenHandle,
|
|
|
|
|
TOKEN_INFORMATION_CLASS TokenInformationClass,
|
|
|
|
|
LPVOID TokenInformation,
|
|
|
|
|
DWORD TokenInformationLength,
|
|
|
|
|
PDWORD ReturnLength)
|
|
|
|
|
{
|
2002-12-18 06:16:28 +00:00
|
|
|
|
static GetTokenInformation_Proc s_pfn_Get_Token_Information = NULL;
|
2002-11-17 22:35:26 +00:00
|
|
|
|
HMODULE hm_advapi32 = NULL;
|
|
|
|
|
if (is_windows_9x () == TRUE)
|
|
|
|
|
{
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
2002-12-18 06:16:28 +00:00
|
|
|
|
if (g_b_init_get_token_information == 0)
|
|
|
|
|
{
|
|
|
|
|
g_b_init_get_token_information = 1;
|
|
|
|
|
hm_advapi32 = LoadLibrary ("Advapi32.dll");
|
|
|
|
|
s_pfn_Get_Token_Information =
|
|
|
|
|
(GetTokenInformation_Proc) GetProcAddress (hm_advapi32, "GetTokenInformation");
|
|
|
|
|
}
|
|
|
|
|
if (s_pfn_Get_Token_Information == NULL)
|
2002-11-17 22:35:26 +00:00
|
|
|
|
{
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
return (
|
2002-12-18 06:16:28 +00:00
|
|
|
|
s_pfn_Get_Token_Information (
|
2002-11-17 22:35:26 +00:00
|
|
|
|
TokenHandle,
|
|
|
|
|
TokenInformationClass,
|
|
|
|
|
TokenInformation,
|
|
|
|
|
TokenInformationLength,
|
|
|
|
|
ReturnLength)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BOOL WINAPI lookup_account_sid (
|
|
|
|
|
LPCTSTR lpSystemName,
|
|
|
|
|
PSID Sid,
|
|
|
|
|
LPTSTR Name,
|
|
|
|
|
LPDWORD cbName,
|
|
|
|
|
LPTSTR DomainName,
|
|
|
|
|
LPDWORD cbDomainName,
|
|
|
|
|
PSID_NAME_USE peUse)
|
|
|
|
|
{
|
2002-12-18 06:16:28 +00:00
|
|
|
|
static LookupAccountSid_Proc s_pfn_Lookup_Account_Sid = NULL;
|
2002-11-17 22:35:26 +00:00
|
|
|
|
HMODULE hm_advapi32 = NULL;
|
|
|
|
|
if (is_windows_9x () == TRUE)
|
|
|
|
|
{
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
2002-12-18 06:16:28 +00:00
|
|
|
|
if (g_b_init_lookup_account_sid == 0)
|
|
|
|
|
{
|
|
|
|
|
g_b_init_lookup_account_sid = 1;
|
|
|
|
|
hm_advapi32 = LoadLibrary ("Advapi32.dll");
|
|
|
|
|
s_pfn_Lookup_Account_Sid =
|
|
|
|
|
(LookupAccountSid_Proc) GetProcAddress (hm_advapi32, LookupAccountSid_Name);
|
|
|
|
|
}
|
|
|
|
|
if (s_pfn_Lookup_Account_Sid == NULL)
|
2002-11-17 22:35:26 +00:00
|
|
|
|
{
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
return (
|
2002-12-18 06:16:28 +00:00
|
|
|
|
s_pfn_Lookup_Account_Sid (
|
2002-11-17 22:35:26 +00:00
|
|
|
|
lpSystemName,
|
|
|
|
|
Sid,
|
|
|
|
|
Name,
|
|
|
|
|
cbName,
|
|
|
|
|
DomainName,
|
|
|
|
|
cbDomainName,
|
|
|
|
|
peUse)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PSID_IDENTIFIER_AUTHORITY WINAPI get_sid_identifier_authority (
|
|
|
|
|
PSID pSid)
|
|
|
|
|
{
|
2002-12-18 06:16:28 +00:00
|
|
|
|
static GetSidIdentifierAuthority_Proc s_pfn_Get_Sid_Identifier_Authority = NULL;
|
2002-11-17 22:35:26 +00:00
|
|
|
|
HMODULE hm_advapi32 = NULL;
|
|
|
|
|
if (is_windows_9x () == TRUE)
|
|
|
|
|
{
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2002-12-18 06:16:28 +00:00
|
|
|
|
if (g_b_init_get_sid_identifier_authority == 0)
|
|
|
|
|
{
|
|
|
|
|
g_b_init_get_sid_identifier_authority = 1;
|
|
|
|
|
hm_advapi32 = LoadLibrary ("Advapi32.dll");
|
|
|
|
|
s_pfn_Get_Sid_Identifier_Authority =
|
|
|
|
|
(GetSidIdentifierAuthority_Proc) GetProcAddress (
|
|
|
|
|
hm_advapi32, "GetSidIdentifierAuthority");
|
|
|
|
|
}
|
|
|
|
|
if (s_pfn_Get_Sid_Identifier_Authority == NULL)
|
2002-11-17 22:35:26 +00:00
|
|
|
|
{
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2002-12-18 06:16:28 +00:00
|
|
|
|
return (s_pfn_Get_Sid_Identifier_Authority (pSid));
|
2002-11-17 22:35:26 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
END: Wrapper functions around OpenProcessToken
|
|
|
|
|
and other functions in advapi32.dll that are only
|
|
|
|
|
supported in Windows NT / 2k / XP
|
|
|
|
|
*/
|
|
|
|
|
|
2000-12-18 23:50:26 +00:00
|
|
|
|
|
|
|
|
|
/* Equivalent of strerror for W32 error codes. */
|
|
|
|
|
char *
|
|
|
|
|
w32_strerror (int error_no)
|
|
|
|
|
{
|
|
|
|
|
static char buf[500];
|
|
|
|
|
|
|
|
|
|
if (error_no == 0)
|
|
|
|
|
error_no = GetLastError ();
|
|
|
|
|
|
|
|
|
|
buf[0] = '\0';
|
|
|
|
|
if (!FormatMessage (FORMAT_MESSAGE_FROM_SYSTEM, NULL,
|
|
|
|
|
error_no,
|
|
|
|
|
0, /* choose most suitable language */
|
|
|
|
|
buf, sizeof (buf), NULL))
|
|
|
|
|
sprintf (buf, "w32 error %u", error_no);
|
|
|
|
|
return buf;
|
|
|
|
|
}
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
static char startup_dir[MAXPATHLEN];
|
1996-01-21 00:31:34 +00:00
|
|
|
|
|
1994-11-01 10:27:02 +00:00
|
|
|
|
/* Get the current working directory. */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
char *
|
1994-11-01 10:27:02 +00:00
|
|
|
|
getwd (char *dir)
|
|
|
|
|
{
|
1997-09-03 00:43:20 +00:00
|
|
|
|
#if 0
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (GetCurrentDirectory (MAXPATHLEN, dir) > 0)
|
|
|
|
|
return dir;
|
|
|
|
|
return NULL;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
#else
|
|
|
|
|
/* Emacs doesn't actually change directory itself, and we want to
|
|
|
|
|
force our real wd to be where emacs.exe is to avoid unnecessary
|
|
|
|
|
conflicts when trying to rename or delete directories. */
|
|
|
|
|
strcpy (dir, startup_dir);
|
|
|
|
|
return dir;
|
|
|
|
|
#endif
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#ifndef HAVE_SOCKETS
|
1994-11-01 10:27:02 +00:00
|
|
|
|
/* Emulate gethostname. */
|
|
|
|
|
int
|
|
|
|
|
gethostname (char *buffer, int size)
|
|
|
|
|
{
|
2003-02-04 14:56:31 +00:00
|
|
|
|
/* NT only allows small host names, so the buffer is
|
1994-11-01 10:27:02 +00:00
|
|
|
|
certainly large enough. */
|
|
|
|
|
return !GetComputerName (buffer, &size);
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#endif /* HAVE_SOCKETS */
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
|
|
|
|
/* Emulate getloadavg. */
|
|
|
|
|
int
|
|
|
|
|
getloadavg (double loadavg[], int nelem)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* A faithful emulation is going to have to be saved for a rainy day. */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
for (i = 0; i < nelem; i++)
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
|
|
|
|
loadavg[i] = 0.0;
|
|
|
|
|
}
|
|
|
|
|
return i;
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Emulate getpwuid, getpwnam and others. */
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
1995-06-30 21:12:37 +00:00
|
|
|
|
#define PASSWD_FIELD_SIZE 256
|
|
|
|
|
|
|
|
|
|
static char the_passwd_name[PASSWD_FIELD_SIZE];
|
|
|
|
|
static char the_passwd_passwd[PASSWD_FIELD_SIZE];
|
|
|
|
|
static char the_passwd_gecos[PASSWD_FIELD_SIZE];
|
|
|
|
|
static char the_passwd_dir[PASSWD_FIELD_SIZE];
|
|
|
|
|
static char the_passwd_shell[PASSWD_FIELD_SIZE];
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
static struct passwd the_passwd =
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
|
|
|
|
the_passwd_name,
|
|
|
|
|
the_passwd_passwd,
|
|
|
|
|
0,
|
|
|
|
|
0,
|
|
|
|
|
0,
|
|
|
|
|
the_passwd_gecos,
|
|
|
|
|
the_passwd_dir,
|
|
|
|
|
the_passwd_shell,
|
|
|
|
|
};
|
|
|
|
|
|
2003-11-22 23:01:18 +00:00
|
|
|
|
static struct group the_group =
|
|
|
|
|
{
|
|
|
|
|
/* There are no groups on NT, so we just return "root" as the
|
|
|
|
|
group name. */
|
|
|
|
|
"root",
|
|
|
|
|
};
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
getuid ()
|
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
return the_passwd.pw_uid;
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
geteuid ()
|
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* I could imagine arguing for checking to see whether the user is
|
|
|
|
|
in the Administrators group and returning a UID of 0 for that
|
|
|
|
|
case, but I don't know how wise that would be in the long run. */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
return getuid ();
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
getgid ()
|
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
return the_passwd.pw_gid;
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
getegid ()
|
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
return getgid ();
|
|
|
|
|
}
|
|
|
|
|
|
1994-11-01 10:27:02 +00:00
|
|
|
|
struct passwd *
|
|
|
|
|
getpwuid (int uid)
|
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (uid == the_passwd.pw_uid)
|
|
|
|
|
return &the_passwd;
|
|
|
|
|
return NULL;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
2003-11-22 23:01:18 +00:00
|
|
|
|
struct group *
|
|
|
|
|
getgrgid (gid_t gid)
|
|
|
|
|
{
|
|
|
|
|
return &the_group;
|
|
|
|
|
}
|
|
|
|
|
|
1994-11-01 10:27:02 +00:00
|
|
|
|
struct passwd *
|
|
|
|
|
getpwnam (char *name)
|
|
|
|
|
{
|
|
|
|
|
struct passwd *pw;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1994-11-01 10:27:02 +00:00
|
|
|
|
pw = getpwuid (getuid ());
|
|
|
|
|
if (!pw)
|
|
|
|
|
return pw;
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (stricmp (name, pw->pw_name))
|
1994-11-01 10:27:02 +00:00
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
return pw;
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
void
|
|
|
|
|
init_user_info ()
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Find the user's real name by opening the process token and
|
|
|
|
|
looking up the name associated with the user-sid in that token.
|
|
|
|
|
|
|
|
|
|
Use the relative portion of the identifier authority value from
|
|
|
|
|
the user-sid as the user id value (same for group id using the
|
|
|
|
|
primary group sid from the process token). */
|
|
|
|
|
|
|
|
|
|
char user_sid[256], name[256], domain[256];
|
|
|
|
|
DWORD length = sizeof (name), dlength = sizeof (domain), trash;
|
|
|
|
|
HANDLE token = NULL;
|
|
|
|
|
SID_NAME_USE user_type;
|
|
|
|
|
|
2002-11-17 22:35:26 +00:00
|
|
|
|
if (
|
|
|
|
|
open_process_token (GetCurrentProcess (), TOKEN_QUERY, &token)
|
|
|
|
|
&& get_token_information (
|
|
|
|
|
token, TokenUser,
|
1996-05-03 18:32:52 +00:00
|
|
|
|
(PVOID) user_sid, sizeof (user_sid), &trash)
|
2002-11-17 22:35:26 +00:00
|
|
|
|
&& lookup_account_sid (
|
|
|
|
|
NULL, *((PSID *) user_sid), name, &length,
|
|
|
|
|
domain, &dlength, &user_type)
|
|
|
|
|
)
|
1995-10-08 19:38:53 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
strcpy (the_passwd.pw_name, name);
|
|
|
|
|
/* Determine a reasonable uid value. */
|
|
|
|
|
if (stricmp ("administrator", name) == 0)
|
|
|
|
|
{
|
|
|
|
|
the_passwd.pw_uid = 0;
|
|
|
|
|
the_passwd.pw_gid = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
SID_IDENTIFIER_AUTHORITY * pSIA;
|
|
|
|
|
|
2002-11-17 22:35:26 +00:00
|
|
|
|
pSIA = get_sid_identifier_authority (*((PSID *) user_sid));
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* I believe the relative portion is the last 4 bytes (of 6)
|
|
|
|
|
with msb first. */
|
|
|
|
|
the_passwd.pw_uid = ((pSIA->Value[2] << 24) +
|
|
|
|
|
(pSIA->Value[3] << 16) +
|
|
|
|
|
(pSIA->Value[4] << 8) +
|
|
|
|
|
(pSIA->Value[5] << 0));
|
|
|
|
|
/* restrict to conventional uid range for normal users */
|
|
|
|
|
the_passwd.pw_uid = the_passwd.pw_uid % 60001;
|
|
|
|
|
|
|
|
|
|
/* Get group id */
|
2002-11-17 22:35:26 +00:00
|
|
|
|
if (get_token_information (token, TokenPrimaryGroup,
|
1996-05-03 18:32:52 +00:00
|
|
|
|
(PVOID) user_sid, sizeof (user_sid), &trash))
|
|
|
|
|
{
|
|
|
|
|
SID_IDENTIFIER_AUTHORITY * pSIA;
|
|
|
|
|
|
2002-11-17 22:35:26 +00:00
|
|
|
|
pSIA = get_sid_identifier_authority (*((PSID *) user_sid));
|
1996-05-03 18:32:52 +00:00
|
|
|
|
the_passwd.pw_gid = ((pSIA->Value[2] << 24) +
|
|
|
|
|
(pSIA->Value[3] << 16) +
|
|
|
|
|
(pSIA->Value[4] << 8) +
|
|
|
|
|
(pSIA->Value[5] << 0));
|
|
|
|
|
/* I don't know if this is necessary, but for safety... */
|
|
|
|
|
the_passwd.pw_gid = the_passwd.pw_gid % 60001;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
the_passwd.pw_gid = the_passwd.pw_uid;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* If security calls are not supported (presumably because we
|
|
|
|
|
are running under Windows 95), fallback to this. */
|
|
|
|
|
else if (GetUserName (name, &length))
|
|
|
|
|
{
|
|
|
|
|
strcpy (the_passwd.pw_name, name);
|
|
|
|
|
if (stricmp ("administrator", name) == 0)
|
|
|
|
|
the_passwd.pw_uid = 0;
|
|
|
|
|
else
|
|
|
|
|
the_passwd.pw_uid = 123;
|
|
|
|
|
the_passwd.pw_gid = the_passwd.pw_uid;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
strcpy (the_passwd.pw_name, "unknown");
|
|
|
|
|
the_passwd.pw_uid = 123;
|
|
|
|
|
the_passwd.pw_gid = 123;
|
1995-10-08 19:38:53 +00:00
|
|
|
|
}
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Ensure HOME and SHELL are defined. */
|
|
|
|
|
if (getenv ("HOME") == NULL)
|
2000-07-05 16:00:13 +00:00
|
|
|
|
abort ();
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (getenv ("SHELL") == NULL)
|
2000-07-05 16:00:13 +00:00
|
|
|
|
abort ();
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Set dir and shell from environment variables. */
|
|
|
|
|
strcpy (the_passwd.pw_dir, getenv ("HOME"));
|
|
|
|
|
strcpy (the_passwd.pw_shell, getenv ("SHELL"));
|
1995-05-26 05:26:17 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (token)
|
|
|
|
|
CloseHandle (token);
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
1996-05-03 18:32:52 +00:00
|
|
|
|
random ()
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* rand () on NT gives us 15 random bits...hack together 30 bits. */
|
|
|
|
|
return ((rand () << 15) | rand ());
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
1996-05-03 18:32:52 +00:00
|
|
|
|
srandom (int seed)
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
srand (seed);
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
|
1996-06-03 21:18:55 +00:00
|
|
|
|
/* Normalize filename by converting all path separators to
|
|
|
|
|
the specified separator. Also conditionally convert upper
|
|
|
|
|
case path name components to lower case. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
normalize_filename (fp, path_sep)
|
|
|
|
|
register char *fp;
|
|
|
|
|
char path_sep;
|
|
|
|
|
{
|
|
|
|
|
char sep;
|
|
|
|
|
char *elem;
|
|
|
|
|
|
1996-07-07 01:59:24 +00:00
|
|
|
|
/* Always lower-case drive letters a-z, even if the filesystem
|
|
|
|
|
preserves case in filenames.
|
|
|
|
|
This is so filenames can be compared by string comparison
|
|
|
|
|
functions that are case-sensitive. Even case-preserving filesystems
|
|
|
|
|
do not distinguish case in drive letters. */
|
|
|
|
|
if (fp[1] == ':' && *fp >= 'A' && *fp <= 'Z')
|
|
|
|
|
{
|
|
|
|
|
*fp += 'a' - 'A';
|
|
|
|
|
fp += 2;
|
|
|
|
|
}
|
|
|
|
|
|
1996-11-19 04:59:23 +00:00
|
|
|
|
if (NILP (Vw32_downcase_file_names))
|
1996-06-03 21:18:55 +00:00
|
|
|
|
{
|
|
|
|
|
while (*fp)
|
|
|
|
|
{
|
|
|
|
|
if (*fp == '/' || *fp == '\\')
|
|
|
|
|
*fp = path_sep;
|
|
|
|
|
fp++;
|
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sep = path_sep; /* convert to this path separator */
|
|
|
|
|
elem = fp; /* start of current path element */
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
if (*fp >= 'a' && *fp <= 'z')
|
|
|
|
|
elem = 0; /* don't convert this element */
|
|
|
|
|
|
|
|
|
|
if (*fp == 0 || *fp == ':')
|
|
|
|
|
{
|
|
|
|
|
sep = *fp; /* restore current separator (or 0) */
|
|
|
|
|
*fp = '/'; /* after conversion of this element */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (*fp == '/' || *fp == '\\')
|
|
|
|
|
{
|
|
|
|
|
if (elem && elem != fp)
|
|
|
|
|
{
|
|
|
|
|
*fp = 0; /* temporary end of string */
|
|
|
|
|
_strlwr (elem); /* while we convert to lower case */
|
|
|
|
|
}
|
|
|
|
|
*fp = sep; /* convert (or restore) path separator */
|
|
|
|
|
elem = fp + 1; /* next element starts after separator */
|
|
|
|
|
sep = path_sep;
|
|
|
|
|
}
|
|
|
|
|
} while (*fp++);
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Destructively turn backslashes into slashes. */
|
1994-11-01 10:27:02 +00:00
|
|
|
|
void
|
1996-05-03 18:32:52 +00:00
|
|
|
|
dostounix_filename (p)
|
|
|
|
|
register char *p;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1996-06-03 21:18:55 +00:00
|
|
|
|
normalize_filename (p, '/');
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Destructively turn slashes into backslashes. */
|
1994-11-01 10:27:02 +00:00
|
|
|
|
void
|
1996-05-03 18:32:52 +00:00
|
|
|
|
unixtodos_filename (p)
|
|
|
|
|
register char *p;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1996-06-03 21:18:55 +00:00
|
|
|
|
normalize_filename (p, '\\');
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Remove all CR's that are followed by a LF.
|
|
|
|
|
(From msdos.c...probably should figure out a way to share it,
|
|
|
|
|
although this code isn't going to ever change.) */
|
1995-04-12 02:10:10 +00:00
|
|
|
|
int
|
1996-05-03 18:32:52 +00:00
|
|
|
|
crlf_to_lf (n, buf)
|
|
|
|
|
register int n;
|
|
|
|
|
register unsigned char *buf;
|
1995-04-12 02:10:10 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
unsigned char *np = buf;
|
|
|
|
|
unsigned char *startp = buf;
|
|
|
|
|
unsigned char *endp = buf + n;
|
1995-04-12 02:10:10 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (n == 0)
|
|
|
|
|
return n;
|
|
|
|
|
while (buf < endp - 1)
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (*buf == 0x0d)
|
|
|
|
|
{
|
|
|
|
|
if (*(++buf) != 0x0a)
|
|
|
|
|
*np++ = 0x0d;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
*np++ = *buf++;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (buf < endp)
|
|
|
|
|
*np++ = *buf++;
|
|
|
|
|
return np - startp;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* Parse the root part of file name, if present. Return length and
|
|
|
|
|
optionally store pointer to char after root. */
|
|
|
|
|
static int
|
|
|
|
|
parse_root (char * name, char ** pPath)
|
|
|
|
|
{
|
|
|
|
|
char * start = name;
|
|
|
|
|
|
|
|
|
|
if (name == NULL)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
/* find the root name of the volume if given */
|
|
|
|
|
if (isalpha (name[0]) && name[1] == ':')
|
|
|
|
|
{
|
|
|
|
|
/* skip past drive specifier */
|
|
|
|
|
name += 2;
|
|
|
|
|
if (IS_DIRECTORY_SEP (name[0]))
|
|
|
|
|
name++;
|
|
|
|
|
}
|
|
|
|
|
else if (IS_DIRECTORY_SEP (name[0]) && IS_DIRECTORY_SEP (name[1]))
|
|
|
|
|
{
|
|
|
|
|
int slashes = 2;
|
|
|
|
|
name += 2;
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
if (IS_DIRECTORY_SEP (*name) && --slashes == 0)
|
|
|
|
|
break;
|
|
|
|
|
name++;
|
|
|
|
|
}
|
|
|
|
|
while ( *name );
|
|
|
|
|
if (IS_DIRECTORY_SEP (name[0]))
|
|
|
|
|
name++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pPath)
|
|
|
|
|
*pPath = name;
|
|
|
|
|
|
|
|
|
|
return name - start;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Get long base name for name; name is assumed to be absolute. */
|
|
|
|
|
static int
|
|
|
|
|
get_long_basename (char * name, char * buf, int size)
|
|
|
|
|
{
|
|
|
|
|
WIN32_FIND_DATA find_data;
|
|
|
|
|
HANDLE dir_handle;
|
|
|
|
|
int len = 0;
|
|
|
|
|
|
2000-05-16 14:29:08 +00:00
|
|
|
|
/* must be valid filename, no wild cards or other invalid characters */
|
1998-05-12 21:33:44 +00:00
|
|
|
|
if (strpbrk (name, "*?|<>\""))
|
|
|
|
|
return 0;
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
dir_handle = FindFirstFile (name, &find_data);
|
|
|
|
|
if (dir_handle != INVALID_HANDLE_VALUE)
|
|
|
|
|
{
|
|
|
|
|
if ((len = strlen (find_data.cFileName)) < size)
|
|
|
|
|
memcpy (buf, find_data.cFileName, len + 1);
|
|
|
|
|
else
|
|
|
|
|
len = 0;
|
|
|
|
|
FindClose (dir_handle);
|
|
|
|
|
}
|
|
|
|
|
return len;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Get long name for file, if possible (assumed to be absolute). */
|
|
|
|
|
BOOL
|
|
|
|
|
w32_get_long_filename (char * name, char * buf, int size)
|
|
|
|
|
{
|
|
|
|
|
char * o = buf;
|
|
|
|
|
char * p;
|
|
|
|
|
char * q;
|
|
|
|
|
char full[ MAX_PATH ];
|
|
|
|
|
int len;
|
|
|
|
|
|
|
|
|
|
len = strlen (name);
|
|
|
|
|
if (len >= MAX_PATH)
|
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
|
|
/* Use local copy for destructive modification. */
|
|
|
|
|
memcpy (full, name, len+1);
|
|
|
|
|
unixtodos_filename (full);
|
|
|
|
|
|
|
|
|
|
/* Copy root part verbatim. */
|
|
|
|
|
len = parse_root (full, &p);
|
|
|
|
|
memcpy (o, full, len);
|
|
|
|
|
o += len;
|
1998-12-10 05:43:07 +00:00
|
|
|
|
*o = '\0';
|
1997-09-03 00:43:20 +00:00
|
|
|
|
size -= len;
|
|
|
|
|
|
1998-12-10 05:43:07 +00:00
|
|
|
|
while (p != NULL && *p)
|
1997-09-03 00:43:20 +00:00
|
|
|
|
{
|
|
|
|
|
q = p;
|
|
|
|
|
p = strchr (q, '\\');
|
|
|
|
|
if (p) *p = '\0';
|
|
|
|
|
len = get_long_basename (full, o, size);
|
|
|
|
|
if (len > 0)
|
|
|
|
|
{
|
|
|
|
|
o += len;
|
|
|
|
|
size -= len;
|
|
|
|
|
if (p != NULL)
|
|
|
|
|
{
|
|
|
|
|
*p++ = '\\';
|
|
|
|
|
if (size < 2)
|
|
|
|
|
return FALSE;
|
|
|
|
|
*o++ = '\\';
|
|
|
|
|
size--;
|
|
|
|
|
*o = '\0';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
1998-06-17 04:22:58 +00:00
|
|
|
|
int
|
|
|
|
|
is_unc_volume (const char *filename)
|
|
|
|
|
{
|
|
|
|
|
const char *ptr = filename;
|
|
|
|
|
|
|
|
|
|
if (!IS_DIRECTORY_SEP (ptr[0]) || !IS_DIRECTORY_SEP (ptr[1]) || !ptr[2])
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
if (strpbrk (ptr + 2, "*?|<>\"\\/"))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
1997-09-03 00:43:20 +00:00
|
|
|
|
|
1994-11-01 10:27:02 +00:00
|
|
|
|
/* Routines that are no-ops on NT but are defined to get Emacs to compile. */
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
sigsetmask (int signal_mask)
|
|
|
|
|
{
|
1994-11-01 10:27:02 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
sigmask (int sig)
|
|
|
|
|
{
|
2000-02-06 23:37:46 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
sigblock (int sig)
|
|
|
|
|
{
|
1994-11-01 10:27:02 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
sigunblock (int sig)
|
|
|
|
|
{
|
2000-02-06 23:37:46 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
setpgrp (int pid, int gid)
|
|
|
|
|
{
|
1994-11-01 10:27:02 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
int
|
|
|
|
|
alarm (int seconds)
|
|
|
|
|
{
|
1994-11-01 10:27:02 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
void
|
|
|
|
|
unrequest_sigio (void)
|
|
|
|
|
{
|
1998-04-09 17:06:31 +00:00
|
|
|
|
return;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-04-09 17:06:31 +00:00
|
|
|
|
void
|
2003-02-04 14:56:31 +00:00
|
|
|
|
request_sigio (void)
|
|
|
|
|
{
|
1998-04-09 17:06:31 +00:00
|
|
|
|
return;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#define REG_ROOT "SOFTWARE\\GNU\\Emacs"
|
1995-11-07 07:36:09 +00:00
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
LPBYTE
|
1996-11-19 04:59:23 +00:00
|
|
|
|
w32_get_resource (key, lpdwtype)
|
1995-11-07 07:36:09 +00:00
|
|
|
|
char *key;
|
|
|
|
|
LPDWORD lpdwtype;
|
|
|
|
|
{
|
|
|
|
|
LPBYTE lpvalue;
|
|
|
|
|
HKEY hrootkey = NULL;
|
|
|
|
|
DWORD cbData;
|
|
|
|
|
BOOL ok = FALSE;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
|
|
|
|
/* Check both the current user and the local machine to see if
|
1995-11-07 07:36:09 +00:00
|
|
|
|
we have any resources. */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1995-11-07 07:36:09 +00:00
|
|
|
|
if (RegOpenKeyEx (HKEY_CURRENT_USER, REG_ROOT, 0, KEY_READ, &hrootkey) == ERROR_SUCCESS)
|
|
|
|
|
{
|
|
|
|
|
lpvalue = NULL;
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
if (RegQueryValueEx (hrootkey, key, NULL, NULL, NULL, &cbData) == ERROR_SUCCESS
|
|
|
|
|
&& (lpvalue = (LPBYTE) xmalloc (cbData)) != NULL
|
1995-11-07 07:36:09 +00:00
|
|
|
|
&& RegQueryValueEx (hrootkey, key, NULL, lpdwtype, lpvalue, &cbData) == ERROR_SUCCESS)
|
|
|
|
|
{
|
|
|
|
|
return (lpvalue);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (lpvalue) xfree (lpvalue);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1995-11-07 07:36:09 +00:00
|
|
|
|
RegCloseKey (hrootkey);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
1995-11-07 07:36:09 +00:00
|
|
|
|
if (RegOpenKeyEx (HKEY_LOCAL_MACHINE, REG_ROOT, 0, KEY_READ, &hrootkey) == ERROR_SUCCESS)
|
|
|
|
|
{
|
|
|
|
|
lpvalue = NULL;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
if (RegQueryValueEx (hrootkey, key, NULL, NULL, NULL, &cbData) == ERROR_SUCCESS
|
|
|
|
|
&& (lpvalue = (LPBYTE) xmalloc (cbData)) != NULL
|
|
|
|
|
&& RegQueryValueEx (hrootkey, key, NULL, lpdwtype, lpvalue, &cbData) == ERROR_SUCCESS)
|
1995-11-07 07:36:09 +00:00
|
|
|
|
{
|
|
|
|
|
return (lpvalue);
|
|
|
|
|
}
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1995-11-07 07:36:09 +00:00
|
|
|
|
if (lpvalue) xfree (lpvalue);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1995-11-07 07:36:09 +00:00
|
|
|
|
RegCloseKey (hrootkey);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
1995-11-07 07:36:09 +00:00
|
|
|
|
return (NULL);
|
|
|
|
|
}
|
|
|
|
|
|
1997-01-20 01:52:48 +00:00
|
|
|
|
char *get_emacs_configuration (void);
|
|
|
|
|
extern Lisp_Object Vsystem_configuration;
|
|
|
|
|
|
1995-11-07 07:36:09 +00:00
|
|
|
|
void
|
1999-01-30 18:48:57 +00:00
|
|
|
|
init_environment (char ** argv)
|
1995-11-07 07:36:09 +00:00
|
|
|
|
{
|
1998-06-05 16:08:32 +00:00
|
|
|
|
static const char * const tempdirs[] = {
|
|
|
|
|
"$TMPDIR", "$TEMP", "$TMP", "c:/"
|
|
|
|
|
};
|
|
|
|
|
int i;
|
|
|
|
|
const int imax = sizeof (tempdirs) / sizeof (tempdirs[0]);
|
|
|
|
|
|
|
|
|
|
/* Make sure they have a usable $TMPDIR. Many Emacs functions use
|
|
|
|
|
temporary files and assume "/tmp" if $TMPDIR is unset, which
|
|
|
|
|
will break on DOS/Windows. Refuse to work if we cannot find
|
|
|
|
|
a directory, not even "c:/", usable for that purpose. */
|
|
|
|
|
for (i = 0; i < imax ; i++)
|
|
|
|
|
{
|
|
|
|
|
const char *tmp = tempdirs[i];
|
|
|
|
|
|
|
|
|
|
if (*tmp == '$')
|
|
|
|
|
tmp = getenv (tmp + 1);
|
|
|
|
|
/* Note that `access' can lie to us if the directory resides on a
|
|
|
|
|
read-only filesystem, like CD-ROM or a write-protected floppy.
|
|
|
|
|
The only way to be really sure is to actually create a file and
|
|
|
|
|
see if it succeeds. But I think that's too much to ask. */
|
2000-08-22 22:49:27 +00:00
|
|
|
|
if (tmp && _access (tmp, D_OK) == 0)
|
1998-06-05 16:08:32 +00:00
|
|
|
|
{
|
|
|
|
|
char * var = alloca (strlen (tmp) + 8);
|
|
|
|
|
sprintf (var, "TMPDIR=%s", tmp);
|
2001-02-09 09:57:04 +00:00
|
|
|
|
_putenv (strdup (var));
|
1998-06-05 16:08:32 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (i >= imax)
|
|
|
|
|
cmd_error_internal
|
|
|
|
|
(Fcons (Qerror,
|
|
|
|
|
Fcons (build_string ("no usable temporary directories found!!"),
|
|
|
|
|
Qnil)),
|
|
|
|
|
"While setting TMPDIR: ");
|
|
|
|
|
|
2000-07-05 16:00:13 +00:00
|
|
|
|
/* Check for environment variables and use registry settings if they
|
|
|
|
|
don't exist. Fallback on default values where applicable. */
|
1995-11-07 07:36:09 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
int i;
|
|
|
|
|
LPBYTE lpval;
|
|
|
|
|
DWORD dwType;
|
2000-11-24 21:30:06 +00:00
|
|
|
|
char locale_name[32];
|
1995-11-07 07:36:09 +00:00
|
|
|
|
|
2000-07-05 16:00:13 +00:00
|
|
|
|
static struct env_entry
|
|
|
|
|
{
|
|
|
|
|
char * name;
|
|
|
|
|
char * def_value;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
} env_vars[] =
|
2000-07-05 16:00:13 +00:00
|
|
|
|
{
|
|
|
|
|
{"HOME", "C:/"},
|
|
|
|
|
{"PRELOAD_WINSOCK", NULL},
|
|
|
|
|
{"emacs_dir", "C:/emacs"},
|
2001-01-19 14:56:06 +00:00
|
|
|
|
{"EMACSLOADPATH", "%emacs_dir%/site-lisp;%emacs_dir%/../site-lisp;%emacs_dir%/lisp;%emacs_dir%/leim"},
|
2000-07-05 16:00:13 +00:00
|
|
|
|
{"SHELL", "%emacs_dir%/bin/cmdproxy.exe"},
|
|
|
|
|
{"EMACSDATA", "%emacs_dir%/etc"},
|
|
|
|
|
{"EMACSPATH", "%emacs_dir%/bin"},
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* We no longer set INFOPATH because Info-default-directory-list
|
2000-07-05 16:00:13 +00:00
|
|
|
|
is then ignored. */
|
|
|
|
|
/* {"INFOPATH", "%emacs_dir%/info"}, */
|
|
|
|
|
{"EMACSDOC", "%emacs_dir%/etc"},
|
2000-11-24 21:30:06 +00:00
|
|
|
|
{"TERM", "cmd"},
|
|
|
|
|
{"LANG", NULL},
|
1996-05-03 18:32:52 +00:00
|
|
|
|
};
|
|
|
|
|
|
2000-11-24 21:30:06 +00:00
|
|
|
|
/* Get default locale info and use it for LANG. */
|
|
|
|
|
if (GetLocaleInfo (LOCALE_USER_DEFAULT,
|
|
|
|
|
LOCALE_SABBREVLANGNAME | LOCALE_USE_CP_ACP,
|
|
|
|
|
locale_name, sizeof (locale_name)))
|
|
|
|
|
{
|
|
|
|
|
for (i = 0; i < (sizeof (env_vars) / sizeof (env_vars[0])); i++)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp (env_vars[i].name, "LANG") == 0)
|
|
|
|
|
{
|
|
|
|
|
env_vars[i].def_value = locale_name;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2000-07-05 16:00:13 +00:00
|
|
|
|
#define SET_ENV_BUF_SIZE (4 * MAX_PATH) /* to cover EMACSLOADPATH */
|
|
|
|
|
|
|
|
|
|
/* Treat emacs_dir specially: set it unconditionally based on our
|
|
|
|
|
location, if it appears that we are running from the bin subdir
|
|
|
|
|
of a standard installation. */
|
|
|
|
|
{
|
|
|
|
|
char *p;
|
|
|
|
|
char modname[MAX_PATH];
|
|
|
|
|
|
|
|
|
|
if (!GetModuleFileName (NULL, modname, MAX_PATH))
|
|
|
|
|
abort ();
|
|
|
|
|
if ((p = strrchr (modname, '\\')) == NULL)
|
|
|
|
|
abort ();
|
|
|
|
|
*p = 0;
|
|
|
|
|
|
|
|
|
|
if ((p = strrchr (modname, '\\')) && stricmp (p, "\\bin") == 0)
|
|
|
|
|
{
|
|
|
|
|
char buf[SET_ENV_BUF_SIZE];
|
|
|
|
|
|
|
|
|
|
*p = 0;
|
|
|
|
|
for (p = modname; *p; p++)
|
|
|
|
|
if (*p == '\\') *p = '/';
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
2000-07-05 16:00:13 +00:00
|
|
|
|
_snprintf (buf, sizeof(buf)-1, "emacs_dir=%s", modname);
|
2000-08-22 22:49:27 +00:00
|
|
|
|
_putenv (strdup (buf));
|
2000-07-05 16:00:13 +00:00
|
|
|
|
}
|
2004-10-19 19:09:37 +00:00
|
|
|
|
/* Handle running emacs from the build directory: src/oo-spd/i386/ */
|
|
|
|
|
|
|
|
|
|
/* FIXME: should use substring of get_emacs_configuration ().
|
|
|
|
|
But I don't think the Windows build supports alpha, mips etc
|
|
|
|
|
anymore, so have taken the easy option for now. */
|
|
|
|
|
else if (p && stricmp (p, "\\i386") == 0)
|
|
|
|
|
{
|
|
|
|
|
*p = 0;
|
|
|
|
|
p = strrchr (modname, '\\');
|
|
|
|
|
if (p != NULL)
|
|
|
|
|
{
|
|
|
|
|
*p = 0;
|
|
|
|
|
p = strrchr (modname, '\\');
|
|
|
|
|
if (p && stricmp (p, "\\src") == 0)
|
|
|
|
|
{
|
|
|
|
|
char buf[SET_ENV_BUF_SIZE];
|
|
|
|
|
|
|
|
|
|
*p = 0;
|
|
|
|
|
for (p = modname; *p; p++)
|
|
|
|
|
if (*p == '\\') *p = '/';
|
|
|
|
|
|
|
|
|
|
_snprintf (buf, sizeof(buf)-1, "emacs_dir=%s", modname);
|
|
|
|
|
_putenv (strdup (buf));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2000-07-05 16:00:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-11-24 21:30:06 +00:00
|
|
|
|
for (i = 0; i < (sizeof (env_vars) / sizeof (env_vars[0])); i++)
|
1995-11-07 07:36:09 +00:00
|
|
|
|
{
|
2000-07-05 16:00:13 +00:00
|
|
|
|
if (!getenv (env_vars[i].name))
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
2000-07-05 16:00:13 +00:00
|
|
|
|
int dont_free = 0;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
2000-07-05 16:00:13 +00:00
|
|
|
|
if ((lpval = w32_get_resource (env_vars[i].name, &dwType)) == NULL)
|
|
|
|
|
{
|
|
|
|
|
lpval = env_vars[i].def_value;
|
|
|
|
|
dwType = REG_EXPAND_SZ;
|
|
|
|
|
dont_free = 1;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
2000-07-05 16:00:13 +00:00
|
|
|
|
|
|
|
|
|
if (lpval)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
2000-07-05 16:00:13 +00:00
|
|
|
|
if (dwType == REG_EXPAND_SZ)
|
|
|
|
|
{
|
|
|
|
|
char buf1[SET_ENV_BUF_SIZE], buf2[SET_ENV_BUF_SIZE];
|
|
|
|
|
|
|
|
|
|
ExpandEnvironmentStrings ((LPSTR) lpval, buf1, sizeof(buf1));
|
|
|
|
|
_snprintf (buf2, sizeof(buf2)-1, "%s=%s", env_vars[i].name, buf1);
|
2000-08-22 22:49:27 +00:00
|
|
|
|
_putenv (strdup (buf2));
|
2000-07-05 16:00:13 +00:00
|
|
|
|
}
|
|
|
|
|
else if (dwType == REG_SZ)
|
|
|
|
|
{
|
|
|
|
|
char buf[SET_ENV_BUF_SIZE];
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
2000-07-05 16:00:13 +00:00
|
|
|
|
_snprintf (buf, sizeof(buf)-1, "%s=%s", env_vars[i].name, lpval);
|
2000-08-22 22:49:27 +00:00
|
|
|
|
_putenv (strdup (buf));
|
2000-07-05 16:00:13 +00:00
|
|
|
|
}
|
1995-11-07 07:36:09 +00:00
|
|
|
|
|
2000-07-05 16:00:13 +00:00
|
|
|
|
if (!dont_free)
|
|
|
|
|
xfree (lpval);
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1997-01-20 01:52:48 +00:00
|
|
|
|
/* Rebuild system configuration to reflect invoking system. */
|
|
|
|
|
Vsystem_configuration = build_string (EMACS_CONFIGURATION);
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* Another special case: on NT, the PATH variable is actually named
|
|
|
|
|
"Path" although cmd.exe (perhaps NT itself) arranges for
|
|
|
|
|
environment variable lookup and setting to be case insensitive.
|
|
|
|
|
However, Emacs assumes a fully case sensitive environment, so we
|
|
|
|
|
need to change "Path" to "PATH" to match the expectations of
|
|
|
|
|
various elisp packages. We do this by the sneaky method of
|
|
|
|
|
modifying the string in the C runtime environ entry.
|
|
|
|
|
|
|
|
|
|
The same applies to COMSPEC. */
|
|
|
|
|
{
|
|
|
|
|
char ** envp;
|
|
|
|
|
|
|
|
|
|
for (envp = environ; *envp; envp++)
|
|
|
|
|
if (_strnicmp (*envp, "PATH=", 5) == 0)
|
|
|
|
|
memcpy (*envp, "PATH=", 5);
|
|
|
|
|
else if (_strnicmp (*envp, "COMSPEC=", 8) == 0)
|
|
|
|
|
memcpy (*envp, "COMSPEC=", 8);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Remember the initial working directory for getwd, then make the
|
|
|
|
|
real wd be the location of emacs.exe to avoid conflicts when
|
|
|
|
|
renaming or deleting directories. (We also don't call chdir when
|
|
|
|
|
running subprocesses for the same reason.) */
|
|
|
|
|
if (!GetCurrentDirectory (MAXPATHLEN, startup_dir))
|
|
|
|
|
abort ();
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
char *p;
|
1999-01-30 18:48:57 +00:00
|
|
|
|
static char modname[MAX_PATH];
|
1997-09-03 00:43:20 +00:00
|
|
|
|
|
|
|
|
|
if (!GetModuleFileName (NULL, modname, MAX_PATH))
|
|
|
|
|
abort ();
|
|
|
|
|
if ((p = strrchr (modname, '\\')) == NULL)
|
|
|
|
|
abort ();
|
|
|
|
|
*p = 0;
|
|
|
|
|
|
|
|
|
|
SetCurrentDirectory (modname);
|
1999-01-30 18:48:57 +00:00
|
|
|
|
|
|
|
|
|
/* Ensure argv[0] has the full path to Emacs. */
|
|
|
|
|
*p = '\\';
|
|
|
|
|
argv[0] = modname;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-02-27 21:28:38 +00:00
|
|
|
|
/* Determine if there is a middle mouse button, to allow parse_button
|
|
|
|
|
to decide whether right mouse events should be mouse-2 or
|
|
|
|
|
mouse-3. */
|
2004-05-17 21:33:16 +00:00
|
|
|
|
w32_num_mouse_buttons = GetSystemMetrics (SM_CMOUSEBUTTONS);
|
2000-02-27 21:28:38 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
init_user_info ();
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-19 19:00:44 +00:00
|
|
|
|
char *
|
|
|
|
|
emacs_root_dir (void)
|
|
|
|
|
{
|
|
|
|
|
static char root_dir[FILENAME_MAX];
|
|
|
|
|
const char *p;
|
|
|
|
|
|
|
|
|
|
p = getenv ("emacs_dir");
|
|
|
|
|
if (p == NULL)
|
|
|
|
|
abort ();
|
|
|
|
|
strcpy (root_dir, p);
|
|
|
|
|
root_dir[parse_root (root_dir, NULL)] = '\0';
|
|
|
|
|
dostounix_filename (root_dir);
|
|
|
|
|
return root_dir;
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* We don't have scripts to automatically determine the system configuration
|
|
|
|
|
for Emacs before it's compiled, and we don't want to have to make the
|
|
|
|
|
user enter it, so we define EMACS_CONFIGURATION to invoke this runtime
|
|
|
|
|
routine. */
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
get_emacs_configuration (void)
|
|
|
|
|
{
|
|
|
|
|
char *arch, *oem, *os;
|
1999-06-24 22:05:14 +00:00
|
|
|
|
int build_num;
|
2000-08-22 22:49:27 +00:00
|
|
|
|
static char configuration_buffer[32];
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
/* Determine the processor type. */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
switch (get_processor_type ())
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
#ifdef PROCESSOR_INTEL_386
|
|
|
|
|
case PROCESSOR_INTEL_386:
|
|
|
|
|
case PROCESSOR_INTEL_486:
|
|
|
|
|
case PROCESSOR_INTEL_PENTIUM:
|
|
|
|
|
arch = "i386";
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef PROCESSOR_INTEL_860
|
|
|
|
|
case PROCESSOR_INTEL_860:
|
|
|
|
|
arch = "i860";
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef PROCESSOR_MIPS_R2000
|
|
|
|
|
case PROCESSOR_MIPS_R2000:
|
|
|
|
|
case PROCESSOR_MIPS_R3000:
|
|
|
|
|
case PROCESSOR_MIPS_R4000:
|
|
|
|
|
arch = "mips";
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef PROCESSOR_ALPHA_21064
|
|
|
|
|
case PROCESSOR_ALPHA_21064:
|
|
|
|
|
arch = "alpha";
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
arch = "unknown";
|
|
|
|
|
break;
|
1995-11-07 07:36:09 +00:00
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
2000-08-22 22:49:27 +00:00
|
|
|
|
/* Use the OEM field to reflect the compiler/library combination. */
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
|
#define COMPILER_NAME "msvc"
|
|
|
|
|
#else
|
|
|
|
|
#ifdef __GNUC__
|
|
|
|
|
#define COMPILER_NAME "mingw"
|
|
|
|
|
#else
|
|
|
|
|
#define COMPILER_NAME "unknown"
|
|
|
|
|
#endif
|
|
|
|
|
#endif
|
|
|
|
|
oem = COMPILER_NAME;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
1999-06-24 22:05:14 +00:00
|
|
|
|
switch (osinfo_cache.dwPlatformId) {
|
|
|
|
|
case VER_PLATFORM_WIN32_NT:
|
|
|
|
|
os = "nt";
|
|
|
|
|
build_num = osinfo_cache.dwBuildNumber;
|
|
|
|
|
break;
|
|
|
|
|
case VER_PLATFORM_WIN32_WINDOWS:
|
|
|
|
|
if (osinfo_cache.dwMinorVersion == 0) {
|
|
|
|
|
os = "windows95";
|
|
|
|
|
} else {
|
|
|
|
|
os = "windows98";
|
|
|
|
|
}
|
|
|
|
|
build_num = LOWORD (osinfo_cache.dwBuildNumber);
|
|
|
|
|
break;
|
|
|
|
|
case VER_PLATFORM_WIN32s:
|
|
|
|
|
/* Not supported, should not happen. */
|
|
|
|
|
os = "windows32s";
|
|
|
|
|
build_num = LOWORD (osinfo_cache.dwBuildNumber);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
os = "unknown";
|
|
|
|
|
build_num = 0;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (osinfo_cache.dwPlatformId == VER_PLATFORM_WIN32_NT) {
|
|
|
|
|
sprintf (configuration_buffer, "%s-%s-%s%d.%d.%d", arch, oem, os,
|
|
|
|
|
get_w32_major_version (), get_w32_minor_version (), build_num);
|
|
|
|
|
} else {
|
|
|
|
|
sprintf (configuration_buffer, "%s-%s-%s.%d", arch, oem, os, build_num);
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
return configuration_buffer;
|
1995-11-07 07:36:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-08-22 22:49:27 +00:00
|
|
|
|
char *
|
|
|
|
|
get_emacs_configuration_options (void)
|
|
|
|
|
{
|
|
|
|
|
static char options_buffer[256];
|
|
|
|
|
|
|
|
|
|
/* Work out the effective configure options for this build. */
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
|
#define COMPILER_VERSION "--with-msvc (%d.%02d)", _MSC_VER / 100, _MSC_VER % 100
|
|
|
|
|
#else
|
|
|
|
|
#ifdef __GNUC__
|
|
|
|
|
#define COMPILER_VERSION "--with-gcc (%d.%d)", __GNUC__, __GNUC_MINOR__
|
|
|
|
|
#else
|
|
|
|
|
#define COMPILER_VERSION ""
|
|
|
|
|
#endif
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
sprintf (options_buffer, COMPILER_VERSION);
|
|
|
|
|
#ifdef EMACSDEBUG
|
|
|
|
|
strcat (options_buffer, " --no-opt");
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef USER_CFLAGS
|
|
|
|
|
strcat (options_buffer, " --cflags");
|
|
|
|
|
strcat (options_buffer, USER_CFLAGS);
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef USER_LDFLAGS
|
|
|
|
|
strcat (options_buffer, " --ldflags");
|
|
|
|
|
strcat (options_buffer, USER_LDFLAGS);
|
|
|
|
|
#endif
|
|
|
|
|
return options_buffer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1995-04-12 02:10:10 +00:00
|
|
|
|
#include <sys/timeb.h>
|
|
|
|
|
|
|
|
|
|
/* Emulate gettimeofday (Ulrich Leodolter, 1/11/95). */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
void
|
1995-04-12 02:10:10 +00:00
|
|
|
|
gettimeofday (struct timeval *tv, struct timezone *tz)
|
|
|
|
|
{
|
2005-06-04 20:39:15 +00:00
|
|
|
|
struct _timeb tb;
|
1995-04-12 02:10:10 +00:00
|
|
|
|
_ftime (&tb);
|
|
|
|
|
|
|
|
|
|
tv->tv_sec = tb.time;
|
|
|
|
|
tv->tv_usec = tb.millitm * 1000L;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
if (tz)
|
1995-04-12 02:10:10 +00:00
|
|
|
|
{
|
|
|
|
|
tz->tz_minuteswest = tb.timezone; /* minutes west of Greenwich */
|
|
|
|
|
tz->tz_dsttime = tb.dstflag; /* type of dst correction */
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* ------------------------------------------------------------------------- */
|
1996-11-19 04:59:23 +00:00
|
|
|
|
/* IO support and wrapper functions for W32 API. */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* ------------------------------------------------------------------------- */
|
1994-11-01 10:27:02 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Place a wrapper around the MSVC version of ctime. It returns NULL
|
2003-02-04 14:56:31 +00:00
|
|
|
|
on network directories, so we handle that case here.
|
1996-05-03 18:32:52 +00:00
|
|
|
|
(Ulrich Leodolter, 1/11/95). */
|
|
|
|
|
char *
|
|
|
|
|
sys_ctime (const time_t *t)
|
|
|
|
|
{
|
|
|
|
|
char *str = (char *) ctime (t);
|
|
|
|
|
return (str ? str : "Sun Jan 01 00:00:00 1970");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Emulate sleep...we could have done this with a define, but that
|
|
|
|
|
would necessitate including windows.h in the files that used it.
|
|
|
|
|
This is much easier. */
|
|
|
|
|
void
|
|
|
|
|
sys_sleep (int seconds)
|
|
|
|
|
{
|
|
|
|
|
Sleep (seconds * 1000);
|
|
|
|
|
}
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* Internal MSVC functions for low-level descriptor munging */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
extern int __cdecl _set_osfhnd (int fd, long h);
|
|
|
|
|
extern int __cdecl _free_osfhnd (int fd);
|
|
|
|
|
|
|
|
|
|
/* parallel array of private info on file handles */
|
|
|
|
|
filedesc fd_info [ MAXDESC ];
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
typedef struct volume_info_data {
|
|
|
|
|
struct volume_info_data * next;
|
|
|
|
|
|
|
|
|
|
/* time when info was obtained */
|
|
|
|
|
DWORD timestamp;
|
|
|
|
|
|
|
|
|
|
/* actual volume info */
|
|
|
|
|
char * root_dir;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
DWORD serialnum;
|
|
|
|
|
DWORD maxcomp;
|
|
|
|
|
DWORD flags;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
char * name;
|
|
|
|
|
char * type;
|
|
|
|
|
} volume_info_data;
|
|
|
|
|
|
|
|
|
|
/* Global referenced by various functions. */
|
|
|
|
|
static volume_info_data volume_info;
|
|
|
|
|
|
|
|
|
|
/* Vector to indicate which drives are local and fixed (for which cached
|
|
|
|
|
data never expires). */
|
|
|
|
|
static BOOL fixed_drives[26];
|
|
|
|
|
|
|
|
|
|
/* Consider cached volume information to be stale if older than 10s,
|
|
|
|
|
at least for non-local drives. Info for fixed drives is never stale. */
|
|
|
|
|
#define DRIVE_INDEX( c ) ( (c) <= 'Z' ? (c) - 'A' : (c) - 'a' )
|
|
|
|
|
#define VOLINFO_STILL_VALID( root_dir, info ) \
|
|
|
|
|
( ( isalpha (root_dir[0]) && \
|
|
|
|
|
fixed_drives[ DRIVE_INDEX (root_dir[0]) ] ) \
|
|
|
|
|
|| GetTickCount () - info->timestamp < 10000 )
|
|
|
|
|
|
|
|
|
|
/* Cache support functions. */
|
|
|
|
|
|
|
|
|
|
/* Simple linked list with linear search is sufficient. */
|
|
|
|
|
static volume_info_data *volume_cache = NULL;
|
|
|
|
|
|
|
|
|
|
static volume_info_data *
|
|
|
|
|
lookup_volume_info (char * root_dir)
|
|
|
|
|
{
|
|
|
|
|
volume_info_data * info;
|
|
|
|
|
|
|
|
|
|
for (info = volume_cache; info; info = info->next)
|
|
|
|
|
if (stricmp (info->root_dir, root_dir) == 0)
|
|
|
|
|
break;
|
|
|
|
|
return info;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
add_volume_info (char * root_dir, volume_info_data * info)
|
|
|
|
|
{
|
2000-08-22 22:49:27 +00:00
|
|
|
|
info->root_dir = xstrdup (root_dir);
|
1997-09-03 00:43:20 +00:00
|
|
|
|
info->next = volume_cache;
|
|
|
|
|
volume_cache = info;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Wrapper for GetVolumeInformation, which uses caching to avoid
|
|
|
|
|
performance penalty (~2ms on 486 for local drives, 7.5ms for local
|
|
|
|
|
cdrom drive, ~5-10ms or more for remote drives on LAN). */
|
|
|
|
|
volume_info_data *
|
|
|
|
|
GetCachedVolumeInformation (char * root_dir)
|
|
|
|
|
{
|
|
|
|
|
volume_info_data * info;
|
|
|
|
|
char default_root[ MAX_PATH ];
|
|
|
|
|
|
|
|
|
|
/* NULL for root_dir means use root from current directory. */
|
|
|
|
|
if (root_dir == NULL)
|
|
|
|
|
{
|
|
|
|
|
if (GetCurrentDirectory (MAX_PATH, default_root) == 0)
|
|
|
|
|
return NULL;
|
|
|
|
|
parse_root (default_root, &root_dir);
|
|
|
|
|
*root_dir = 0;
|
|
|
|
|
root_dir = default_root;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Local fixed drives can be cached permanently. Removable drives
|
|
|
|
|
cannot be cached permanently, since the volume name and serial
|
|
|
|
|
number (if nothing else) can change. Remote drives should be
|
|
|
|
|
treated as if they are removable, since there is no sure way to
|
|
|
|
|
tell whether they are or not. Also, the UNC association of drive
|
|
|
|
|
letters mapped to remote volumes can be changed at any time (even
|
|
|
|
|
by other processes) without notice.
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
As a compromise, so we can benefit from caching info for remote
|
|
|
|
|
volumes, we use a simple expiry mechanism to invalidate cache
|
|
|
|
|
entries that are more than ten seconds old. */
|
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
/* No point doing this, because WNetGetConnection is even slower than
|
|
|
|
|
GetVolumeInformation, consistently taking ~50ms on a 486 (FWIW,
|
|
|
|
|
GetDriveType is about the only call of this type which does not
|
|
|
|
|
involve network access, and so is extremely quick). */
|
|
|
|
|
|
|
|
|
|
/* Map drive letter to UNC if remote. */
|
|
|
|
|
if ( isalpha( root_dir[0] ) && !fixed[ DRIVE_INDEX( root_dir[0] ) ] )
|
|
|
|
|
{
|
|
|
|
|
char remote_name[ 256 ];
|
|
|
|
|
char drive[3] = { root_dir[0], ':' };
|
|
|
|
|
|
|
|
|
|
if (WNetGetConnection (drive, remote_name, sizeof (remote_name))
|
|
|
|
|
== NO_ERROR)
|
|
|
|
|
/* do something */ ;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
info = lookup_volume_info (root_dir);
|
|
|
|
|
|
|
|
|
|
if (info == NULL || ! VOLINFO_STILL_VALID (root_dir, info))
|
|
|
|
|
{
|
|
|
|
|
char name[ 256 ];
|
|
|
|
|
DWORD serialnum;
|
|
|
|
|
DWORD maxcomp;
|
|
|
|
|
DWORD flags;
|
|
|
|
|
char type[ 256 ];
|
|
|
|
|
|
|
|
|
|
/* Info is not cached, or is stale. */
|
|
|
|
|
if (!GetVolumeInformation (root_dir,
|
|
|
|
|
name, sizeof (name),
|
|
|
|
|
&serialnum,
|
|
|
|
|
&maxcomp,
|
|
|
|
|
&flags,
|
|
|
|
|
type, sizeof (type)))
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
/* Cache the volume information for future use, overwriting existing
|
|
|
|
|
entry if present. */
|
|
|
|
|
if (info == NULL)
|
|
|
|
|
{
|
|
|
|
|
info = (volume_info_data *) xmalloc (sizeof (volume_info_data));
|
|
|
|
|
add_volume_info (root_dir, info);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2000-08-22 22:49:27 +00:00
|
|
|
|
xfree (info->name);
|
|
|
|
|
xfree (info->type);
|
1997-09-03 00:43:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-08-22 22:49:27 +00:00
|
|
|
|
info->name = xstrdup (name);
|
1997-09-03 00:43:20 +00:00
|
|
|
|
info->serialnum = serialnum;
|
|
|
|
|
info->maxcomp = maxcomp;
|
|
|
|
|
info->flags = flags;
|
2000-08-22 22:49:27 +00:00
|
|
|
|
info->type = xstrdup (type);
|
1997-09-03 00:43:20 +00:00
|
|
|
|
info->timestamp = GetTickCount ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return info;
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
/* Get information on the volume where name is held; set path pointer to
|
|
|
|
|
start of pathname in name (past UNC header\volume header if present). */
|
|
|
|
|
int
|
|
|
|
|
get_volume_info (const char * name, const char ** pPath)
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
char temp[MAX_PATH];
|
|
|
|
|
char *rootname = NULL; /* default to current volume */
|
1997-09-03 00:43:20 +00:00
|
|
|
|
volume_info_data * info;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
if (name == NULL)
|
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
|
|
/* find the root name of the volume if given */
|
|
|
|
|
if (isalpha (name[0]) && name[1] == ':')
|
|
|
|
|
{
|
|
|
|
|
rootname = temp;
|
|
|
|
|
temp[0] = *name++;
|
|
|
|
|
temp[1] = *name++;
|
|
|
|
|
temp[2] = '\\';
|
|
|
|
|
temp[3] = 0;
|
|
|
|
|
}
|
|
|
|
|
else if (IS_DIRECTORY_SEP (name[0]) && IS_DIRECTORY_SEP (name[1]))
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
char *str = temp;
|
|
|
|
|
int slashes = 4;
|
|
|
|
|
rootname = temp;
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
if (IS_DIRECTORY_SEP (*name) && --slashes == 0)
|
|
|
|
|
break;
|
|
|
|
|
*str++ = *name++;
|
|
|
|
|
}
|
|
|
|
|
while ( *name );
|
|
|
|
|
|
|
|
|
|
*str++ = '\\';
|
|
|
|
|
*str = 0;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
if (pPath)
|
|
|
|
|
*pPath = name;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
info = GetCachedVolumeInformation (rootname);
|
|
|
|
|
if (info != NULL)
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* Set global referenced by other functions. */
|
|
|
|
|
volume_info = *info;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
return TRUE;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Determine if volume is FAT format (ie. only supports short 8.3
|
|
|
|
|
names); also set path pointer to start of pathname in name. */
|
|
|
|
|
int
|
|
|
|
|
is_fat_volume (const char * name, const char ** pPath)
|
|
|
|
|
{
|
|
|
|
|
if (get_volume_info (name, pPath))
|
|
|
|
|
return (volume_info.maxcomp == 12);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Map filename to a legal 8.3 name if necessary. */
|
|
|
|
|
const char *
|
1996-11-19 04:59:23 +00:00
|
|
|
|
map_w32_filename (const char * name, const char ** pPath)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
static char shortname[MAX_PATH];
|
|
|
|
|
char * str = shortname;
|
|
|
|
|
char c;
|
|
|
|
|
char * path;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
const char * save_name = name;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
2000-07-05 16:00:13 +00:00
|
|
|
|
if (strlen (name) >= MAX_PATH)
|
|
|
|
|
{
|
|
|
|
|
/* Return a filename which will cause callers to fail. */
|
|
|
|
|
strcpy (shortname, "?");
|
|
|
|
|
return shortname;
|
|
|
|
|
}
|
|
|
|
|
|
2000-08-22 22:49:27 +00:00
|
|
|
|
if (is_fat_volume (name, (const char **)&path)) /* truncate to 8.3 */
|
1994-11-01 10:27:02 +00:00
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
register int left = 8; /* maximum number of chars in part */
|
|
|
|
|
register int extn = 0; /* extension added? */
|
|
|
|
|
register int dots = 2; /* maximum number of dots allowed */
|
|
|
|
|
|
|
|
|
|
while (name < path)
|
|
|
|
|
*str++ = *name++; /* skip past UNC header */
|
|
|
|
|
|
|
|
|
|
while ((c = *name++))
|
|
|
|
|
{
|
|
|
|
|
switch ( c )
|
|
|
|
|
{
|
|
|
|
|
case '\\':
|
|
|
|
|
case '/':
|
|
|
|
|
*str++ = '\\';
|
|
|
|
|
extn = 0; /* reset extension flags */
|
|
|
|
|
dots = 2; /* max 2 dots */
|
|
|
|
|
left = 8; /* max length 8 for main part */
|
|
|
|
|
break;
|
|
|
|
|
case ':':
|
|
|
|
|
*str++ = ':';
|
|
|
|
|
extn = 0; /* reset extension flags */
|
|
|
|
|
dots = 2; /* max 2 dots */
|
|
|
|
|
left = 8; /* max length 8 for main part */
|
|
|
|
|
break;
|
|
|
|
|
case '.':
|
|
|
|
|
if ( dots )
|
|
|
|
|
{
|
|
|
|
|
/* Convert path components of the form .xxx to _xxx,
|
|
|
|
|
but leave . and .. as they are. This allows .emacs
|
|
|
|
|
to be read as _emacs, for example. */
|
|
|
|
|
|
|
|
|
|
if (! *name ||
|
|
|
|
|
*name == '.' ||
|
|
|
|
|
IS_DIRECTORY_SEP (*name))
|
|
|
|
|
{
|
|
|
|
|
*str++ = '.';
|
|
|
|
|
dots--;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*str++ = '_';
|
|
|
|
|
left--;
|
|
|
|
|
dots = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if ( !extn )
|
|
|
|
|
{
|
|
|
|
|
*str++ = '.';
|
|
|
|
|
extn = 1; /* we've got an extension */
|
|
|
|
|
left = 3; /* 3 chars in extension */
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* any embedded dots after the first are converted to _ */
|
|
|
|
|
*str++ = '_';
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case '~':
|
|
|
|
|
case '#': /* don't lose these, they're important */
|
|
|
|
|
if ( ! left )
|
|
|
|
|
str[-1] = c; /* replace last character of part */
|
|
|
|
|
/* FALLTHRU */
|
|
|
|
|
default:
|
|
|
|
|
if ( left )
|
|
|
|
|
{
|
|
|
|
|
*str++ = tolower (c); /* map to lower case (looks nicer) */
|
|
|
|
|
left--;
|
|
|
|
|
dots = 0; /* started a path component */
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
*str = '\0';
|
1996-05-11 17:26:00 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
strcpy (shortname, name);
|
|
|
|
|
unixtodos_filename (shortname);
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
if (pPath)
|
1997-09-03 00:43:20 +00:00
|
|
|
|
*pPath = shortname + (path - save_name);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
1996-05-11 17:26:00 +00:00
|
|
|
|
return shortname;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-06-05 16:08:32 +00:00
|
|
|
|
static int
|
|
|
|
|
is_exec (const char * name)
|
|
|
|
|
{
|
|
|
|
|
char * p = strrchr (name, '.');
|
|
|
|
|
return
|
|
|
|
|
(p != NULL
|
|
|
|
|
&& (stricmp (p, ".exe") == 0 ||
|
|
|
|
|
stricmp (p, ".com") == 0 ||
|
|
|
|
|
stricmp (p, ".bat") == 0 ||
|
|
|
|
|
stricmp (p, ".cmd") == 0));
|
|
|
|
|
}
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
/* Emulate the Unix directory procedures opendir, closedir,
|
1997-09-03 00:43:20 +00:00
|
|
|
|
and readdir. We can't use the procedures supplied in sysdep.c,
|
|
|
|
|
so we provide them here. */
|
|
|
|
|
|
|
|
|
|
struct direct dir_static; /* simulated directory contents */
|
|
|
|
|
static HANDLE dir_find_handle = INVALID_HANDLE_VALUE;
|
|
|
|
|
static int dir_is_fat;
|
|
|
|
|
static char dir_pathname[MAXPATHLEN+1];
|
|
|
|
|
static WIN32_FIND_DATA dir_find_data;
|
|
|
|
|
|
1998-06-17 04:22:58 +00:00
|
|
|
|
/* Support shares on a network resource as subdirectories of a read-only
|
|
|
|
|
root directory. */
|
|
|
|
|
static HANDLE wnet_enum_handle = INVALID_HANDLE_VALUE;
|
2004-05-17 21:33:16 +00:00
|
|
|
|
HANDLE open_unc_volume (const char *);
|
1998-06-17 04:22:58 +00:00
|
|
|
|
char *read_unc_volume (HANDLE, char *, int);
|
|
|
|
|
void close_unc_volume (HANDLE);
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
DIR *
|
|
|
|
|
opendir (char *filename)
|
|
|
|
|
{
|
|
|
|
|
DIR *dirp;
|
|
|
|
|
|
|
|
|
|
/* Opening is done by FindFirstFile. However, a read is inherent to
|
|
|
|
|
this operation, so we defer the open until read time. */
|
|
|
|
|
|
|
|
|
|
if (dir_find_handle != INVALID_HANDLE_VALUE)
|
|
|
|
|
return NULL;
|
1998-06-17 04:22:58 +00:00
|
|
|
|
if (wnet_enum_handle != INVALID_HANDLE_VALUE)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
if (is_unc_volume (filename))
|
|
|
|
|
{
|
|
|
|
|
wnet_enum_handle = open_unc_volume (filename);
|
|
|
|
|
if (wnet_enum_handle == INVALID_HANDLE_VALUE)
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(dirp = (DIR *) malloc (sizeof (DIR))))
|
|
|
|
|
return NULL;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
|
|
|
|
|
dirp->dd_fd = 0;
|
|
|
|
|
dirp->dd_loc = 0;
|
|
|
|
|
dirp->dd_size = 0;
|
|
|
|
|
|
|
|
|
|
strncpy (dir_pathname, map_w32_filename (filename, NULL), MAXPATHLEN);
|
|
|
|
|
dir_pathname[MAXPATHLEN] = '\0';
|
|
|
|
|
dir_is_fat = is_fat_volume (filename, NULL);
|
|
|
|
|
|
|
|
|
|
return dirp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
closedir (DIR *dirp)
|
|
|
|
|
{
|
|
|
|
|
/* If we have a find-handle open, close it. */
|
|
|
|
|
if (dir_find_handle != INVALID_HANDLE_VALUE)
|
|
|
|
|
{
|
|
|
|
|
FindClose (dir_find_handle);
|
|
|
|
|
dir_find_handle = INVALID_HANDLE_VALUE;
|
|
|
|
|
}
|
1998-06-17 04:22:58 +00:00
|
|
|
|
else if (wnet_enum_handle != INVALID_HANDLE_VALUE)
|
|
|
|
|
{
|
|
|
|
|
close_unc_volume (wnet_enum_handle);
|
|
|
|
|
wnet_enum_handle = INVALID_HANDLE_VALUE;
|
|
|
|
|
}
|
1997-09-03 00:43:20 +00:00
|
|
|
|
xfree ((char *) dirp);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct direct *
|
|
|
|
|
readdir (DIR *dirp)
|
|
|
|
|
{
|
1998-06-17 04:22:58 +00:00
|
|
|
|
if (wnet_enum_handle != INVALID_HANDLE_VALUE)
|
|
|
|
|
{
|
2003-02-04 14:56:31 +00:00
|
|
|
|
if (!read_unc_volume (wnet_enum_handle,
|
|
|
|
|
dir_find_data.cFileName,
|
1998-06-17 04:22:58 +00:00
|
|
|
|
MAX_PATH))
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* If we aren't dir_finding, do a find-first, otherwise do a find-next. */
|
1998-06-17 04:22:58 +00:00
|
|
|
|
else if (dir_find_handle == INVALID_HANDLE_VALUE)
|
1997-09-03 00:43:20 +00:00
|
|
|
|
{
|
|
|
|
|
char filename[MAXNAMLEN + 3];
|
|
|
|
|
int ln;
|
|
|
|
|
|
|
|
|
|
strcpy (filename, dir_pathname);
|
|
|
|
|
ln = strlen (filename) - 1;
|
|
|
|
|
if (!IS_DIRECTORY_SEP (filename[ln]))
|
|
|
|
|
strcat (filename, "\\");
|
|
|
|
|
strcat (filename, "*");
|
|
|
|
|
|
|
|
|
|
dir_find_handle = FindFirstFile (filename, &dir_find_data);
|
|
|
|
|
|
|
|
|
|
if (dir_find_handle == INVALID_HANDLE_VALUE)
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (!FindNextFile (dir_find_handle, &dir_find_data))
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* Emacs never uses this value, so don't bother making it match
|
|
|
|
|
value returned by stat(). */
|
|
|
|
|
dir_static.d_ino = 1;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
dir_static.d_reclen = sizeof (struct direct) - MAXNAMLEN + 3 +
|
|
|
|
|
dir_static.d_namlen - dir_static.d_namlen % 4;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
dir_static.d_namlen = strlen (dir_find_data.cFileName);
|
|
|
|
|
strcpy (dir_static.d_name, dir_find_data.cFileName);
|
|
|
|
|
if (dir_is_fat)
|
|
|
|
|
_strlwr (dir_static.d_name);
|
|
|
|
|
else if (!NILP (Vw32_downcase_file_names))
|
|
|
|
|
{
|
|
|
|
|
register char *p;
|
|
|
|
|
for (p = dir_static.d_name; *p; p++)
|
|
|
|
|
if (*p >= 'a' && *p <= 'z')
|
|
|
|
|
break;
|
|
|
|
|
if (!*p)
|
|
|
|
|
_strlwr (dir_static.d_name);
|
|
|
|
|
}
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
return &dir_static;
|
|
|
|
|
}
|
|
|
|
|
|
1998-06-17 04:22:58 +00:00
|
|
|
|
HANDLE
|
2004-05-17 21:33:16 +00:00
|
|
|
|
open_unc_volume (const char *path)
|
1998-06-17 04:22:58 +00:00
|
|
|
|
{
|
2003-02-04 14:56:31 +00:00
|
|
|
|
NETRESOURCE nr;
|
1998-06-17 04:22:58 +00:00
|
|
|
|
HANDLE henum;
|
|
|
|
|
int result;
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
nr.dwScope = RESOURCE_GLOBALNET;
|
|
|
|
|
nr.dwType = RESOURCETYPE_DISK;
|
|
|
|
|
nr.dwDisplayType = RESOURCEDISPLAYTYPE_SERVER;
|
|
|
|
|
nr.dwUsage = RESOURCEUSAGE_CONTAINER;
|
|
|
|
|
nr.lpLocalName = NULL;
|
2005-06-04 20:39:15 +00:00
|
|
|
|
nr.lpRemoteName = (LPSTR)map_w32_filename (path, NULL);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
nr.lpComment = NULL;
|
|
|
|
|
nr.lpProvider = NULL;
|
1998-06-17 04:22:58 +00:00
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
result = WNetOpenEnum(RESOURCE_GLOBALNET, RESOURCETYPE_DISK,
|
1998-06-17 04:22:58 +00:00
|
|
|
|
RESOURCEUSAGE_CONNECTABLE, &nr, &henum);
|
|
|
|
|
|
|
|
|
|
if (result == NO_ERROR)
|
|
|
|
|
return henum;
|
|
|
|
|
else
|
|
|
|
|
return INVALID_HANDLE_VALUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
read_unc_volume (HANDLE henum, char *readbuf, int size)
|
|
|
|
|
{
|
2000-08-22 22:49:27 +00:00
|
|
|
|
DWORD count;
|
1998-06-17 04:22:58 +00:00
|
|
|
|
int result;
|
2000-08-22 22:49:27 +00:00
|
|
|
|
DWORD bufsize = 512;
|
1998-06-17 04:22:58 +00:00
|
|
|
|
char *buffer;
|
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
|
|
count = 1;
|
|
|
|
|
buffer = alloca (bufsize);
|
|
|
|
|
result = WNetEnumResource (wnet_enum_handle, &count, buffer, &bufsize);
|
|
|
|
|
if (result != NO_ERROR)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
/* WNetEnumResource returns \\resource\share...skip forward to "share". */
|
|
|
|
|
ptr = ((LPNETRESOURCE) buffer)->lpRemoteName;
|
|
|
|
|
ptr += 2;
|
|
|
|
|
while (*ptr && !IS_DIRECTORY_SEP (*ptr)) ptr++;
|
|
|
|
|
ptr++;
|
|
|
|
|
|
|
|
|
|
strncpy (readbuf, ptr, size);
|
|
|
|
|
return readbuf;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
close_unc_volume (HANDLE henum)
|
|
|
|
|
{
|
|
|
|
|
if (henum != INVALID_HANDLE_VALUE)
|
|
|
|
|
WNetCloseEnum (henum);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DWORD
|
2004-05-17 21:33:16 +00:00
|
|
|
|
unc_volume_file_attributes (const char *path)
|
1998-06-17 04:22:58 +00:00
|
|
|
|
{
|
|
|
|
|
HANDLE henum;
|
|
|
|
|
DWORD attrs;
|
|
|
|
|
|
|
|
|
|
henum = open_unc_volume (path);
|
|
|
|
|
if (henum == INVALID_HANDLE_VALUE)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
attrs = FILE_ATTRIBUTE_READONLY | FILE_ATTRIBUTE_DIRECTORY;
|
|
|
|
|
|
|
|
|
|
close_unc_volume (henum);
|
|
|
|
|
|
|
|
|
|
return attrs;
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
/* Shadow some MSVC runtime functions to map requests for long filenames
|
|
|
|
|
to reasonable short names if necessary. This was originally added to
|
2003-02-04 14:56:31 +00:00
|
|
|
|
permit running Emacs on NT 3.1 on a FAT partition, which doesn't support
|
1996-05-03 18:32:52 +00:00
|
|
|
|
long file names. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_access (const char * path, int mode)
|
|
|
|
|
{
|
1998-06-05 16:08:32 +00:00
|
|
|
|
DWORD attributes;
|
|
|
|
|
|
|
|
|
|
/* MSVC implementation doesn't recognize D_OK. */
|
|
|
|
|
path = map_w32_filename (path, NULL);
|
1998-06-17 04:22:58 +00:00
|
|
|
|
if (is_unc_volume (path))
|
|
|
|
|
{
|
|
|
|
|
attributes = unc_volume_file_attributes (path);
|
|
|
|
|
if (attributes == -1) {
|
|
|
|
|
errno = EACCES;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if ((attributes = GetFileAttributes (path)) == -1)
|
1998-06-05 16:08:32 +00:00
|
|
|
|
{
|
|
|
|
|
/* Should try mapping GetLastError to errno; for now just indicate
|
|
|
|
|
that path doesn't exist. */
|
|
|
|
|
errno = EACCES;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if ((mode & X_OK) != 0 && !is_exec (path))
|
|
|
|
|
{
|
|
|
|
|
errno = EACCES;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if ((mode & W_OK) != 0 && (attributes & FILE_ATTRIBUTE_READONLY) != 0)
|
|
|
|
|
{
|
|
|
|
|
errno = EACCES;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if ((mode & D_OK) != 0 && (attributes & FILE_ATTRIBUTE_DIRECTORY) == 0)
|
|
|
|
|
{
|
|
|
|
|
errno = EACCES;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_chdir (const char * path)
|
|
|
|
|
{
|
1996-11-19 04:59:23 +00:00
|
|
|
|
return _chdir (map_w32_filename (path, NULL));
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_chmod (const char * path, int mode)
|
|
|
|
|
{
|
1996-11-19 04:59:23 +00:00
|
|
|
|
return _chmod (map_w32_filename (path, NULL), mode);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_creat (const char * path, int mode)
|
|
|
|
|
{
|
1996-11-19 04:59:23 +00:00
|
|
|
|
return _creat (map_w32_filename (path, NULL), mode);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FILE *
|
|
|
|
|
sys_fopen(const char * path, const char * mode)
|
|
|
|
|
{
|
|
|
|
|
int fd;
|
|
|
|
|
int oflag;
|
|
|
|
|
const char * mode_save = mode;
|
|
|
|
|
|
|
|
|
|
/* Force all file handles to be non-inheritable. This is necessary to
|
|
|
|
|
ensure child processes don't unwittingly inherit handles that might
|
|
|
|
|
prevent future file access. */
|
|
|
|
|
|
|
|
|
|
if (mode[0] == 'r')
|
|
|
|
|
oflag = O_RDONLY;
|
|
|
|
|
else if (mode[0] == 'w' || mode[0] == 'a')
|
|
|
|
|
oflag = O_WRONLY | O_CREAT | O_TRUNC;
|
1994-11-01 10:27:02 +00:00
|
|
|
|
else
|
1996-05-03 18:32:52 +00:00
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
/* Only do simplistic option parsing. */
|
|
|
|
|
while (*++mode)
|
|
|
|
|
if (mode[0] == '+')
|
|
|
|
|
{
|
|
|
|
|
oflag &= ~(O_RDONLY | O_WRONLY);
|
|
|
|
|
oflag |= O_RDWR;
|
|
|
|
|
}
|
|
|
|
|
else if (mode[0] == 'b')
|
|
|
|
|
{
|
|
|
|
|
oflag &= ~O_TEXT;
|
|
|
|
|
oflag |= O_BINARY;
|
|
|
|
|
}
|
|
|
|
|
else if (mode[0] == 't')
|
|
|
|
|
{
|
|
|
|
|
oflag &= ~O_BINARY;
|
|
|
|
|
oflag |= O_TEXT;
|
|
|
|
|
}
|
|
|
|
|
else break;
|
|
|
|
|
|
1996-11-19 04:59:23 +00:00
|
|
|
|
fd = _open (map_w32_filename (path, NULL), oflag | _O_NOINHERIT, 0644);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (fd < 0)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
return _fdopen (fd, mode_save);
|
1994-11-01 10:27:02 +00:00
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* This only works on NTFS volumes, but is useful to have. */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
int
|
1997-09-03 00:43:20 +00:00
|
|
|
|
sys_link (const char * old, const char * new)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
1997-09-03 00:43:20 +00:00
|
|
|
|
HANDLE fileh;
|
|
|
|
|
int result = -1;
|
|
|
|
|
char oldname[MAX_PATH], newname[MAX_PATH];
|
|
|
|
|
|
|
|
|
|
if (old == NULL || new == NULL)
|
|
|
|
|
{
|
|
|
|
|
errno = ENOENT;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
strcpy (oldname, map_w32_filename (old, NULL));
|
|
|
|
|
strcpy (newname, map_w32_filename (new, NULL));
|
|
|
|
|
|
|
|
|
|
fileh = CreateFile (oldname, 0, 0, NULL, OPEN_EXISTING,
|
|
|
|
|
FILE_FLAG_BACKUP_SEMANTICS, NULL);
|
|
|
|
|
if (fileh != INVALID_HANDLE_VALUE)
|
|
|
|
|
{
|
|
|
|
|
int wlen;
|
|
|
|
|
|
|
|
|
|
/* Confusingly, the "alternate" stream name field does not apply
|
|
|
|
|
when restoring a hard link, and instead contains the actual
|
|
|
|
|
stream data for the link (ie. the name of the link to create).
|
|
|
|
|
The WIN32_STREAM_ID structure before the cStreamName field is
|
|
|
|
|
the stream header, which is then immediately followed by the
|
|
|
|
|
stream data. */
|
|
|
|
|
|
|
|
|
|
struct {
|
|
|
|
|
WIN32_STREAM_ID wid;
|
|
|
|
|
WCHAR wbuffer[MAX_PATH]; /* extra space for link name */
|
|
|
|
|
} data;
|
|
|
|
|
|
|
|
|
|
wlen = MultiByteToWideChar (CP_ACP, MB_PRECOMPOSED, newname, -1,
|
|
|
|
|
data.wid.cStreamName, MAX_PATH);
|
|
|
|
|
if (wlen > 0)
|
|
|
|
|
{
|
|
|
|
|
LPVOID context = NULL;
|
|
|
|
|
DWORD wbytes = 0;
|
|
|
|
|
|
|
|
|
|
data.wid.dwStreamId = BACKUP_LINK;
|
|
|
|
|
data.wid.dwStreamAttributes = 0;
|
|
|
|
|
data.wid.Size.LowPart = wlen * sizeof(WCHAR);
|
|
|
|
|
data.wid.Size.HighPart = 0;
|
|
|
|
|
data.wid.dwStreamNameSize = 0;
|
|
|
|
|
|
|
|
|
|
if (BackupWrite (fileh, (LPBYTE)&data,
|
|
|
|
|
offsetof (WIN32_STREAM_ID, cStreamName)
|
|
|
|
|
+ data.wid.Size.LowPart,
|
|
|
|
|
&wbytes, FALSE, FALSE, &context)
|
|
|
|
|
&& BackupWrite (fileh, NULL, 0, &wbytes, TRUE, FALSE, &context))
|
|
|
|
|
{
|
|
|
|
|
/* succeeded */
|
|
|
|
|
result = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Should try mapping GetLastError to errno; for now just
|
|
|
|
|
indicate a general error (eg. links not supported). */
|
|
|
|
|
errno = EINVAL; // perhaps EMLINK?
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CloseHandle (fileh);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
errno = ENOENT;
|
|
|
|
|
|
|
|
|
|
return result;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_mkdir (const char * path)
|
|
|
|
|
{
|
1996-11-19 04:59:23 +00:00
|
|
|
|
return _mkdir (map_w32_filename (path, NULL));
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
1996-06-07 23:07:00 +00:00
|
|
|
|
/* Because of long name mapping issues, we need to implement this
|
|
|
|
|
ourselves. Also, MSVC's _mktemp returns NULL when it can't generate
|
|
|
|
|
a unique name, instead of setting the input template to an empty
|
|
|
|
|
string.
|
|
|
|
|
|
|
|
|
|
Standard algorithm seems to be use pid or tid with a letter on the
|
|
|
|
|
front (in place of the 6 X's) and cycle through the letters to find a
|
|
|
|
|
unique name. We extend that to allow any reasonable character as the
|
|
|
|
|
first of the 6 X's. */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
char *
|
|
|
|
|
sys_mktemp (char * template)
|
|
|
|
|
{
|
1996-06-07 23:07:00 +00:00
|
|
|
|
char * p;
|
|
|
|
|
int i;
|
|
|
|
|
unsigned uid = GetCurrentThreadId ();
|
|
|
|
|
static char first_char[] = "abcdefghijklmnopqrstuvwyz0123456789!%-_@#";
|
|
|
|
|
|
|
|
|
|
if (template == NULL)
|
|
|
|
|
return NULL;
|
|
|
|
|
p = template + strlen (template);
|
|
|
|
|
i = 5;
|
|
|
|
|
/* replace up to the last 5 X's with uid in decimal */
|
|
|
|
|
while (--p >= template && p[0] == 'X' && --i >= 0)
|
|
|
|
|
{
|
|
|
|
|
p[0] = '0' + uid % 10;
|
|
|
|
|
uid /= 10;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (i < 0 && p[0] == 'X')
|
|
|
|
|
{
|
|
|
|
|
i = 0;
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
int save_errno = errno;
|
|
|
|
|
p[0] = first_char[i];
|
|
|
|
|
if (sys_access (template, 0) < 0)
|
|
|
|
|
{
|
|
|
|
|
errno = save_errno;
|
|
|
|
|
return template;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
while (++i < sizeof (first_char));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Template is badly formed or else we can't generate a unique name,
|
|
|
|
|
so return empty string */
|
|
|
|
|
template[0] = 0;
|
|
|
|
|
return template;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_open (const char * path, int oflag, int mode)
|
|
|
|
|
{
|
2001-04-10 12:15:25 +00:00
|
|
|
|
const char* mpath = map_w32_filename (path, NULL);
|
|
|
|
|
/* Try to open file without _O_CREAT, to be able to write to hidden
|
|
|
|
|
and system files. Force all file handles to be
|
|
|
|
|
non-inheritable. */
|
|
|
|
|
int res = _open (mpath, (oflag & ~_O_CREAT) | _O_NOINHERIT, mode);
|
|
|
|
|
if (res >= 0)
|
|
|
|
|
return res;
|
|
|
|
|
return _open (mpath, oflag | _O_NOINHERIT, mode);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_rename (const char * oldname, const char * newname)
|
|
|
|
|
{
|
2000-12-17 23:14:55 +00:00
|
|
|
|
BOOL result;
|
1998-06-05 16:08:32 +00:00
|
|
|
|
char temp[MAX_PATH];
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
1997-01-20 00:38:22 +00:00
|
|
|
|
/* MoveFile on Windows 95 doesn't correctly change the short file name
|
1996-07-07 01:59:24 +00:00
|
|
|
|
alias in a number of circumstances (it is not easy to predict when
|
|
|
|
|
just by looking at oldname and newname, unfortunately). In these
|
|
|
|
|
cases, renaming through a temporary name avoids the problem.
|
|
|
|
|
|
1997-01-20 00:38:22 +00:00
|
|
|
|
A second problem on Windows 95 is that renaming through a temp name when
|
1996-07-07 01:59:24 +00:00
|
|
|
|
newname is uppercase fails (the final long name ends up in
|
|
|
|
|
lowercase, although the short alias might be uppercase) UNLESS the
|
|
|
|
|
long temp name is not 8.3.
|
|
|
|
|
|
1997-01-20 00:38:22 +00:00
|
|
|
|
So, on Windows 95 we always rename through a temp name, and we make sure
|
1996-07-07 01:59:24 +00:00
|
|
|
|
the temp name has a long extension to ensure correct renaming. */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
1996-11-19 04:59:23 +00:00
|
|
|
|
strcpy (temp, map_w32_filename (oldname, NULL));
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
if (os_subtype == OS_WIN95)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
1998-06-05 16:08:32 +00:00
|
|
|
|
char * o;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
char * p;
|
1998-06-05 16:08:32 +00:00
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
|
|
oldname = map_w32_filename (oldname, NULL);
|
|
|
|
|
if (o = strrchr (oldname, '\\'))
|
|
|
|
|
o++;
|
|
|
|
|
else
|
|
|
|
|
o = (char *) oldname;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
if (p = strrchr (temp, '\\'))
|
|
|
|
|
p++;
|
|
|
|
|
else
|
|
|
|
|
p = temp;
|
1998-06-05 16:08:32 +00:00
|
|
|
|
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
/* Force temp name to require a manufactured 8.3 alias - this
|
|
|
|
|
seems to make the second rename work properly. */
|
1998-06-16 04:33:08 +00:00
|
|
|
|
sprintf (p, "_.%s.%u", o, i);
|
1998-06-05 16:08:32 +00:00
|
|
|
|
i++;
|
1998-06-10 21:00:39 +00:00
|
|
|
|
result = rename (oldname, temp);
|
1998-06-05 16:08:32 +00:00
|
|
|
|
}
|
|
|
|
|
/* This loop must surely terminate! */
|
2000-12-17 23:14:55 +00:00
|
|
|
|
while (result < 0 && errno == EEXIST);
|
1998-06-10 21:00:39 +00:00
|
|
|
|
if (result < 0)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Emulate Unix behaviour - newname is deleted if it already exists
|
1996-07-07 01:59:24 +00:00
|
|
|
|
(at least if it is a file; don't do this for directories).
|
1997-09-03 00:43:20 +00:00
|
|
|
|
|
1998-06-05 16:08:32 +00:00
|
|
|
|
Since we mustn't do this if we are just changing the case of the
|
|
|
|
|
file name (we would end up deleting the file we are trying to
|
|
|
|
|
rename!), we let rename detect if the destination file already
|
|
|
|
|
exists - that way we avoid the possible pitfalls of trying to
|
|
|
|
|
determine ourselves whether two names really refer to the same
|
|
|
|
|
file, which is not always possible in the general case. (Consider
|
|
|
|
|
all the permutations of shared or subst'd drives, etc.) */
|
|
|
|
|
|
|
|
|
|
newname = map_w32_filename (newname, NULL);
|
1998-05-30 23:41:57 +00:00
|
|
|
|
result = rename (temp, newname);
|
1998-06-05 16:08:32 +00:00
|
|
|
|
|
|
|
|
|
if (result < 0
|
2000-12-17 23:14:55 +00:00
|
|
|
|
&& errno == EEXIST
|
1998-06-05 16:08:32 +00:00
|
|
|
|
&& _chmod (newname, 0666) == 0
|
|
|
|
|
&& _unlink (newname) == 0)
|
|
|
|
|
result = rename (temp, newname);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
1998-05-30 23:41:57 +00:00
|
|
|
|
return result;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_rmdir (const char * path)
|
|
|
|
|
{
|
1996-11-19 04:59:23 +00:00
|
|
|
|
return _rmdir (map_w32_filename (path, NULL));
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_unlink (const char * path)
|
|
|
|
|
{
|
1998-04-23 23:41:12 +00:00
|
|
|
|
path = map_w32_filename (path, NULL);
|
|
|
|
|
|
|
|
|
|
/* On Unix, unlink works without write permission. */
|
|
|
|
|
_chmod (path, 0666);
|
|
|
|
|
return _unlink (path);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static FILETIME utc_base_ft;
|
|
|
|
|
static long double utc_base;
|
|
|
|
|
static int init = 0;
|
|
|
|
|
|
|
|
|
|
static time_t
|
|
|
|
|
convert_time (FILETIME ft)
|
|
|
|
|
{
|
|
|
|
|
long double ret;
|
|
|
|
|
|
|
|
|
|
if (!init)
|
|
|
|
|
{
|
|
|
|
|
/* Determine the delta between 1-Jan-1601 and 1-Jan-1970. */
|
|
|
|
|
SYSTEMTIME st;
|
|
|
|
|
|
|
|
|
|
st.wYear = 1970;
|
|
|
|
|
st.wMonth = 1;
|
|
|
|
|
st.wDay = 1;
|
|
|
|
|
st.wHour = 0;
|
|
|
|
|
st.wMinute = 0;
|
|
|
|
|
st.wSecond = 0;
|
|
|
|
|
st.wMilliseconds = 0;
|
|
|
|
|
|
|
|
|
|
SystemTimeToFileTime (&st, &utc_base_ft);
|
|
|
|
|
utc_base = (long double) utc_base_ft.dwHighDateTime
|
|
|
|
|
* 4096 * 1024 * 1024 + utc_base_ft.dwLowDateTime;
|
|
|
|
|
init = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (CompareFileTime (&ft, &utc_base_ft) < 0)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
ret = (long double) ft.dwHighDateTime * 4096 * 1024 * 1024 + ft.dwLowDateTime;
|
|
|
|
|
ret -= utc_base;
|
|
|
|
|
return (time_t) (ret * 1e-7);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
convert_from_time_t (time_t time, FILETIME * pft)
|
|
|
|
|
{
|
|
|
|
|
long double tmp;
|
|
|
|
|
|
|
|
|
|
if (!init)
|
|
|
|
|
{
|
|
|
|
|
/* Determine the delta between 1-Jan-1601 and 1-Jan-1970. */
|
|
|
|
|
SYSTEMTIME st;
|
|
|
|
|
|
|
|
|
|
st.wYear = 1970;
|
|
|
|
|
st.wMonth = 1;
|
|
|
|
|
st.wDay = 1;
|
|
|
|
|
st.wHour = 0;
|
|
|
|
|
st.wMinute = 0;
|
|
|
|
|
st.wSecond = 0;
|
|
|
|
|
st.wMilliseconds = 0;
|
|
|
|
|
|
|
|
|
|
SystemTimeToFileTime (&st, &utc_base_ft);
|
|
|
|
|
utc_base = (long double) utc_base_ft.dwHighDateTime
|
|
|
|
|
* 4096 * 1024 * 1024 + utc_base_ft.dwLowDateTime;
|
|
|
|
|
init = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* time in 100ns units since 1-Jan-1601 */
|
|
|
|
|
tmp = (long double) time * 1e7 + utc_base;
|
|
|
|
|
pft->dwHighDateTime = (DWORD) (tmp / (4096.0 * 1024 * 1024));
|
1998-04-23 23:41:12 +00:00
|
|
|
|
pft->dwLowDateTime = (DWORD) (tmp - (4096.0 * 1024 * 1024) * pft->dwHighDateTime);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
#if 0
|
|
|
|
|
/* No reason to keep this; faking inode values either by hashing or even
|
|
|
|
|
using the file index from GetInformationByHandle, is not perfect and
|
|
|
|
|
so by default Emacs doesn't use the inode values on Windows.
|
|
|
|
|
Instead, we now determine file-truename correctly (except for
|
|
|
|
|
possible drive aliasing etc). */
|
|
|
|
|
|
|
|
|
|
/* Modified version of "PJW" algorithm (see the "Dragon" compiler book). */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
static unsigned
|
1997-09-03 00:43:20 +00:00
|
|
|
|
hashval (const unsigned char * str)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
unsigned h = 0;
|
|
|
|
|
while (*str)
|
|
|
|
|
{
|
|
|
|
|
h = (h << 4) + *str++;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
h ^= (h >> 28);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
return h;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Return the hash value of the canonical pathname, excluding the
|
|
|
|
|
drive/UNC header, to get a hopefully unique inode number. */
|
1997-09-03 00:43:20 +00:00
|
|
|
|
static DWORD
|
1996-05-03 18:32:52 +00:00
|
|
|
|
generate_inode_val (const char * name)
|
|
|
|
|
{
|
|
|
|
|
char fullname[ MAX_PATH ];
|
|
|
|
|
char * p;
|
|
|
|
|
unsigned hash;
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* Get the truly canonical filename, if it exists. (Note: this
|
|
|
|
|
doesn't resolve aliasing due to subst commands, or recognise hard
|
|
|
|
|
links. */
|
|
|
|
|
if (!w32_get_long_filename ((char *)name, fullname, MAX_PATH))
|
|
|
|
|
abort ();
|
|
|
|
|
|
|
|
|
|
parse_root (fullname, &p);
|
1996-11-19 04:59:23 +00:00
|
|
|
|
/* Normal W32 filesystems are still case insensitive. */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
_strlwr (p);
|
1997-09-03 00:43:20 +00:00
|
|
|
|
return hashval (p);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* MSVC stat function can't cope with UNC names and has other bugs, so
|
|
|
|
|
replace it with our own. This also allows us to calculate consistent
|
|
|
|
|
inode values without hacks in the main Emacs code. */
|
|
|
|
|
int
|
|
|
|
|
stat (const char * path, struct stat * buf)
|
|
|
|
|
{
|
1998-05-30 23:41:57 +00:00
|
|
|
|
char *name, *r;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
WIN32_FIND_DATA wfd;
|
|
|
|
|
HANDLE fh;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
DWORD fake_inode;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
int permission;
|
|
|
|
|
int len;
|
|
|
|
|
int rootdir = FALSE;
|
|
|
|
|
|
|
|
|
|
if (path == NULL || buf == NULL)
|
|
|
|
|
{
|
|
|
|
|
errno = EFAULT;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
1996-11-19 04:59:23 +00:00
|
|
|
|
name = (char *) map_w32_filename (path, &path);
|
2000-05-16 14:29:08 +00:00
|
|
|
|
/* must be valid filename, no wild cards or other invalid characters */
|
1998-05-12 21:33:44 +00:00
|
|
|
|
if (strpbrk (name, "*?|<>\""))
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
errno = ENOENT;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
1998-05-30 23:41:57 +00:00
|
|
|
|
/* If name is "c:/.." or "/.." then stat "c:/" or "/". */
|
|
|
|
|
r = IS_DEVICE_SEP (name[1]) ? &name[2] : name;
|
|
|
|
|
if (IS_DIRECTORY_SEP (r[0]) && r[1] == '.' && r[2] == '.' && r[3] == '\0')
|
|
|
|
|
{
|
|
|
|
|
r[1] = r[2] = '\0';
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Remove trailing directory separator, unless name is the root
|
|
|
|
|
directory of a drive or UNC volume in which case ensure there
|
|
|
|
|
is a trailing separator. */
|
|
|
|
|
len = strlen (name);
|
|
|
|
|
rootdir = (path >= name + len - 1
|
|
|
|
|
&& (IS_DIRECTORY_SEP (*path) || *path == 0));
|
|
|
|
|
name = strcpy (alloca (len + 2), name);
|
|
|
|
|
|
1998-06-17 04:22:58 +00:00
|
|
|
|
if (is_unc_volume (name))
|
|
|
|
|
{
|
|
|
|
|
DWORD attrs = unc_volume_file_attributes (name);
|
|
|
|
|
|
|
|
|
|
if (attrs == -1)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
memset (&wfd, 0, sizeof (wfd));
|
|
|
|
|
wfd.dwFileAttributes = attrs;
|
|
|
|
|
wfd.ftCreationTime = utc_base_ft;
|
|
|
|
|
wfd.ftLastAccessTime = utc_base_ft;
|
|
|
|
|
wfd.ftLastWriteTime = utc_base_ft;
|
|
|
|
|
strcpy (wfd.cFileName, name);
|
|
|
|
|
}
|
|
|
|
|
else if (rootdir)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
if (!IS_DIRECTORY_SEP (name[len-1]))
|
|
|
|
|
strcat (name, "\\");
|
|
|
|
|
if (GetDriveType (name) < 2)
|
|
|
|
|
{
|
|
|
|
|
errno = ENOENT;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
memset (&wfd, 0, sizeof (wfd));
|
|
|
|
|
wfd.dwFileAttributes = FILE_ATTRIBUTE_DIRECTORY;
|
|
|
|
|
wfd.ftCreationTime = utc_base_ft;
|
|
|
|
|
wfd.ftLastAccessTime = utc_base_ft;
|
|
|
|
|
wfd.ftLastWriteTime = utc_base_ft;
|
|
|
|
|
strcpy (wfd.cFileName, name);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (IS_DIRECTORY_SEP (name[len-1]))
|
|
|
|
|
name[len - 1] = 0;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
|
|
|
|
|
/* (This is hacky, but helps when doing file completions on
|
|
|
|
|
network drives.) Optimize by using information available from
|
|
|
|
|
active readdir if possible. */
|
1999-08-06 20:40:36 +00:00
|
|
|
|
len = strlen (dir_pathname);
|
|
|
|
|
if (IS_DIRECTORY_SEP (dir_pathname[len-1]))
|
|
|
|
|
len--;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
if (dir_find_handle != INVALID_HANDLE_VALUE
|
1999-08-06 20:40:36 +00:00
|
|
|
|
&& strnicmp (name, dir_pathname, len) == 0
|
1997-09-03 00:43:20 +00:00
|
|
|
|
&& IS_DIRECTORY_SEP (name[len])
|
|
|
|
|
&& stricmp (name + len + 1, dir_static.d_name) == 0)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* This was the last entry returned by readdir. */
|
|
|
|
|
wfd = dir_find_data;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
fh = FindFirstFile (name, &wfd);
|
|
|
|
|
if (fh == INVALID_HANDLE_VALUE)
|
|
|
|
|
{
|
|
|
|
|
errno = ENOENT;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
FindClose (fh);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-28 18:52:26 +00:00
|
|
|
|
if (!NILP (Vw32_get_true_file_attributes)
|
|
|
|
|
/* No access rights required to get info. */
|
|
|
|
|
&& (fh = CreateFile (name, 0, 0, NULL, OPEN_EXISTING,
|
|
|
|
|
FILE_FLAG_BACKUP_SEMANTICS, NULL))
|
|
|
|
|
!= INVALID_HANDLE_VALUE)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
/* This is more accurate in terms of gettting the correct number
|
|
|
|
|
of links, but is quite slow (it is noticable when Emacs is
|
|
|
|
|
making a list of file name completions). */
|
|
|
|
|
BY_HANDLE_FILE_INFORMATION info;
|
|
|
|
|
|
|
|
|
|
if (GetFileInformationByHandle (fh, &info))
|
|
|
|
|
{
|
|
|
|
|
buf->st_nlink = info.nNumberOfLinks;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* Might as well use file index to fake inode values, but this
|
|
|
|
|
is not guaranteed to be unique unless we keep a handle open
|
|
|
|
|
all the time (even then there are situations where it is
|
|
|
|
|
not unique). Reputedly, there are at most 48 bits of info
|
|
|
|
|
(on NTFS, presumably less on FAT). */
|
|
|
|
|
fake_inode = info.nFileIndexLow ^ info.nFileIndexHigh;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
1998-11-03 22:39:04 +00:00
|
|
|
|
buf->st_nlink = 1;
|
|
|
|
|
fake_inode = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-28 18:52:26 +00:00
|
|
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
1998-11-03 22:39:04 +00:00
|
|
|
|
{
|
2002-04-28 18:52:26 +00:00
|
|
|
|
buf->st_mode = _S_IFDIR;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
switch (GetFileType (fh))
|
|
|
|
|
{
|
|
|
|
|
case FILE_TYPE_DISK:
|
|
|
|
|
buf->st_mode = _S_IFREG;
|
|
|
|
|
break;
|
|
|
|
|
case FILE_TYPE_PIPE:
|
|
|
|
|
buf->st_mode = _S_IFIFO;
|
|
|
|
|
break;
|
|
|
|
|
case FILE_TYPE_CHAR:
|
|
|
|
|
case FILE_TYPE_UNKNOWN:
|
|
|
|
|
default:
|
|
|
|
|
buf->st_mode = _S_IFCHR;
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
1998-11-03 22:39:04 +00:00
|
|
|
|
CloseHandle (fh);
|
1997-09-03 00:43:20 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Don't bother to make this information more accurate. */
|
2002-04-28 18:52:26 +00:00
|
|
|
|
buf->st_mode = (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) ?
|
2002-06-13 20:41:33 +00:00
|
|
|
|
_S_IFDIR : _S_IFREG;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
buf->st_nlink = 1;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
fake_inode = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
/* Not sure if there is any point in this. */
|
|
|
|
|
if (!NILP (Vw32_generate_fake_inodes))
|
|
|
|
|
fake_inode = generate_inode_val (name);
|
|
|
|
|
else if (fake_inode == 0)
|
|
|
|
|
{
|
|
|
|
|
/* For want of something better, try to make everything unique. */
|
|
|
|
|
static DWORD gen_num = 0;
|
|
|
|
|
fake_inode = ++gen_num;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
1997-09-03 00:43:20 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/* MSVC defines _ino_t to be short; other libc's might not. */
|
|
|
|
|
if (sizeof (buf->st_ino) == 2)
|
|
|
|
|
buf->st_ino = fake_inode ^ (fake_inode >> 16);
|
|
|
|
|
else
|
|
|
|
|
buf->st_ino = fake_inode;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
/* consider files to belong to current user */
|
|
|
|
|
buf->st_uid = the_passwd.pw_uid;
|
|
|
|
|
buf->st_gid = the_passwd.pw_gid;
|
|
|
|
|
|
1996-11-19 04:59:23 +00:00
|
|
|
|
/* volume_info is set indirectly by map_w32_filename */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
buf->st_dev = volume_info.serialnum;
|
|
|
|
|
buf->st_rdev = volume_info.serialnum;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
buf->st_size = wfd.nFileSizeLow;
|
|
|
|
|
|
|
|
|
|
/* Convert timestamps to Unix format. */
|
|
|
|
|
buf->st_mtime = convert_time (wfd.ftLastWriteTime);
|
|
|
|
|
buf->st_atime = convert_time (wfd.ftLastAccessTime);
|
|
|
|
|
if (buf->st_atime == 0) buf->st_atime = buf->st_mtime;
|
|
|
|
|
buf->st_ctime = convert_time (wfd.ftCreationTime);
|
|
|
|
|
if (buf->st_ctime == 0) buf->st_ctime = buf->st_mtime;
|
|
|
|
|
|
|
|
|
|
/* determine rwx permissions */
|
|
|
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_READONLY)
|
|
|
|
|
permission = _S_IREAD;
|
|
|
|
|
else
|
|
|
|
|
permission = _S_IREAD | _S_IWRITE;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
|
|
|
permission |= _S_IEXEC;
|
1998-06-05 16:08:32 +00:00
|
|
|
|
else if (is_exec (name))
|
|
|
|
|
permission |= _S_IEXEC;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
buf->st_mode |= permission | (permission >> 3) | (permission >> 6);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:41:12 +00:00
|
|
|
|
/* Provide fstat and utime as well as stat for consistent handling of
|
|
|
|
|
file timestamps. */
|
|
|
|
|
int
|
|
|
|
|
fstat (int desc, struct stat * buf)
|
|
|
|
|
{
|
|
|
|
|
HANDLE fh = (HANDLE) _get_osfhandle (desc);
|
|
|
|
|
BY_HANDLE_FILE_INFORMATION info;
|
|
|
|
|
DWORD fake_inode;
|
|
|
|
|
int permission;
|
|
|
|
|
|
|
|
|
|
switch (GetFileType (fh) & ~FILE_TYPE_REMOTE)
|
|
|
|
|
{
|
|
|
|
|
case FILE_TYPE_DISK:
|
|
|
|
|
buf->st_mode = _S_IFREG;
|
|
|
|
|
if (!GetFileInformationByHandle (fh, &info))
|
|
|
|
|
{
|
|
|
|
|
errno = EACCES;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case FILE_TYPE_PIPE:
|
|
|
|
|
buf->st_mode = _S_IFIFO;
|
|
|
|
|
goto non_disk;
|
|
|
|
|
case FILE_TYPE_CHAR:
|
|
|
|
|
case FILE_TYPE_UNKNOWN:
|
|
|
|
|
default:
|
|
|
|
|
buf->st_mode = _S_IFCHR;
|
|
|
|
|
non_disk:
|
|
|
|
|
memset (&info, 0, sizeof (info));
|
|
|
|
|
info.dwFileAttributes = 0;
|
|
|
|
|
info.ftCreationTime = utc_base_ft;
|
|
|
|
|
info.ftLastAccessTime = utc_base_ft;
|
|
|
|
|
info.ftLastWriteTime = utc_base_ft;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (info.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
|
|
|
buf->st_mode = _S_IFDIR;
|
2002-04-28 18:52:26 +00:00
|
|
|
|
|
|
|
|
|
buf->st_nlink = info.nNumberOfLinks;
|
|
|
|
|
/* Might as well use file index to fake inode values, but this
|
|
|
|
|
is not guaranteed to be unique unless we keep a handle open
|
|
|
|
|
all the time (even then there are situations where it is
|
|
|
|
|
not unique). Reputedly, there are at most 48 bits of info
|
|
|
|
|
(on NTFS, presumably less on FAT). */
|
|
|
|
|
fake_inode = info.nFileIndexLow ^ info.nFileIndexHigh;
|
1998-04-23 23:41:12 +00:00
|
|
|
|
|
|
|
|
|
/* MSVC defines _ino_t to be short; other libc's might not. */
|
|
|
|
|
if (sizeof (buf->st_ino) == 2)
|
|
|
|
|
buf->st_ino = fake_inode ^ (fake_inode >> 16);
|
|
|
|
|
else
|
|
|
|
|
buf->st_ino = fake_inode;
|
|
|
|
|
|
|
|
|
|
/* consider files to belong to current user */
|
|
|
|
|
buf->st_uid = 0;
|
|
|
|
|
buf->st_gid = 0;
|
|
|
|
|
|
|
|
|
|
buf->st_dev = info.dwVolumeSerialNumber;
|
|
|
|
|
buf->st_rdev = info.dwVolumeSerialNumber;
|
|
|
|
|
|
|
|
|
|
buf->st_size = info.nFileSizeLow;
|
|
|
|
|
|
|
|
|
|
/* Convert timestamps to Unix format. */
|
|
|
|
|
buf->st_mtime = convert_time (info.ftLastWriteTime);
|
|
|
|
|
buf->st_atime = convert_time (info.ftLastAccessTime);
|
|
|
|
|
if (buf->st_atime == 0) buf->st_atime = buf->st_mtime;
|
|
|
|
|
buf->st_ctime = convert_time (info.ftCreationTime);
|
|
|
|
|
if (buf->st_ctime == 0) buf->st_ctime = buf->st_mtime;
|
|
|
|
|
|
|
|
|
|
/* determine rwx permissions */
|
|
|
|
|
if (info.dwFileAttributes & FILE_ATTRIBUTE_READONLY)
|
|
|
|
|
permission = _S_IREAD;
|
|
|
|
|
else
|
|
|
|
|
permission = _S_IREAD | _S_IWRITE;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1998-04-23 23:41:12 +00:00
|
|
|
|
if (info.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
|
|
|
permission |= _S_IEXEC;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
#if 0 /* no way of knowing the filename */
|
|
|
|
|
char * p = strrchr (name, '.');
|
|
|
|
|
if (p != NULL &&
|
|
|
|
|
(stricmp (p, ".exe") == 0 ||
|
|
|
|
|
stricmp (p, ".com") == 0 ||
|
|
|
|
|
stricmp (p, ".bat") == 0 ||
|
|
|
|
|
stricmp (p, ".cmd") == 0))
|
|
|
|
|
permission |= _S_IEXEC;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
buf->st_mode |= permission | (permission >> 3) | (permission >> 6);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
utime (const char *name, struct utimbuf *times)
|
|
|
|
|
{
|
|
|
|
|
struct utimbuf deftime;
|
|
|
|
|
HANDLE fh;
|
|
|
|
|
FILETIME mtime;
|
|
|
|
|
FILETIME atime;
|
|
|
|
|
|
|
|
|
|
if (times == NULL)
|
|
|
|
|
{
|
|
|
|
|
deftime.modtime = deftime.actime = time (NULL);
|
|
|
|
|
times = &deftime;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Need write access to set times. */
|
|
|
|
|
fh = CreateFile (name, GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
|
|
|
0, OPEN_EXISTING, 0, NULL);
|
|
|
|
|
if (fh)
|
|
|
|
|
{
|
|
|
|
|
convert_from_time_t (times->actime, &atime);
|
|
|
|
|
convert_from_time_t (times->modtime, &mtime);
|
|
|
|
|
if (!SetFileTime (fh, NULL, &atime, &mtime))
|
|
|
|
|
{
|
|
|
|
|
CloseHandle (fh);
|
|
|
|
|
errno = EACCES;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
CloseHandle (fh);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
errno = EINVAL;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#ifdef HAVE_SOCKETS
|
|
|
|
|
|
|
|
|
|
/* Wrappers for winsock functions to map between our file descriptors
|
|
|
|
|
and winsock's handles; also set h_errno for convenience.
|
|
|
|
|
|
|
|
|
|
To allow Emacs to run on systems which don't have winsock support
|
|
|
|
|
installed, we dynamically link to winsock on startup if present, and
|
|
|
|
|
otherwise provide the minimum necessary functionality
|
|
|
|
|
(eg. gethostname). */
|
|
|
|
|
|
|
|
|
|
/* function pointers for relevant socket functions */
|
|
|
|
|
int (PASCAL *pfn_WSAStartup) (WORD wVersionRequired, LPWSADATA lpWSAData);
|
|
|
|
|
void (PASCAL *pfn_WSASetLastError) (int iError);
|
|
|
|
|
int (PASCAL *pfn_WSAGetLastError) (void);
|
|
|
|
|
int (PASCAL *pfn_socket) (int af, int type, int protocol);
|
|
|
|
|
int (PASCAL *pfn_bind) (SOCKET s, const struct sockaddr *addr, int namelen);
|
|
|
|
|
int (PASCAL *pfn_connect) (SOCKET s, const struct sockaddr *addr, int namelen);
|
|
|
|
|
int (PASCAL *pfn_ioctlsocket) (SOCKET s, long cmd, u_long *argp);
|
|
|
|
|
int (PASCAL *pfn_recv) (SOCKET s, char * buf, int len, int flags);
|
|
|
|
|
int (PASCAL *pfn_send) (SOCKET s, const char * buf, int len, int flags);
|
|
|
|
|
int (PASCAL *pfn_closesocket) (SOCKET s);
|
|
|
|
|
int (PASCAL *pfn_shutdown) (SOCKET s, int how);
|
|
|
|
|
int (PASCAL *pfn_WSACleanup) (void);
|
|
|
|
|
|
|
|
|
|
u_short (PASCAL *pfn_htons) (u_short hostshort);
|
|
|
|
|
u_short (PASCAL *pfn_ntohs) (u_short netshort);
|
|
|
|
|
unsigned long (PASCAL *pfn_inet_addr) (const char * cp);
|
|
|
|
|
int (PASCAL *pfn_gethostname) (char * name, int namelen);
|
|
|
|
|
struct hostent * (PASCAL *pfn_gethostbyname) (const char * name);
|
|
|
|
|
struct servent * (PASCAL *pfn_getservbyname) (const char * name, const char * proto);
|
2002-05-03 20:40:03 +00:00
|
|
|
|
int (PASCAL *pfn_getpeername) (SOCKET s, struct sockaddr *addr, int * namelen);
|
2002-03-20 20:38:44 +00:00
|
|
|
|
int (PASCAL *pfn_setsockopt) (SOCKET s, int level, int optname,
|
|
|
|
|
const char * optval, int optlen);
|
|
|
|
|
int (PASCAL *pfn_listen) (SOCKET s, int backlog);
|
|
|
|
|
int (PASCAL *pfn_getsockname) (SOCKET s, struct sockaddr * name,
|
|
|
|
|
int * namelen);
|
|
|
|
|
SOCKET (PASCAL *pfn_accept) (SOCKET s, struct sockaddr * addr, int * addrlen);
|
|
|
|
|
int (PASCAL *pfn_recvfrom) (SOCKET s, char * buf, int len, int flags,
|
|
|
|
|
struct sockaddr * from, int * fromlen);
|
|
|
|
|
int (PASCAL *pfn_sendto) (SOCKET s, const char * buf, int len, int flags,
|
|
|
|
|
const struct sockaddr * to, int tolen);
|
|
|
|
|
|
1996-06-06 17:07:00 +00:00
|
|
|
|
/* SetHandleInformation is only needed to make sockets non-inheritable. */
|
|
|
|
|
BOOL (WINAPI *pfn_SetHandleInformation) (HANDLE object, DWORD mask, DWORD flags);
|
|
|
|
|
#ifndef HANDLE_FLAG_INHERIT
|
|
|
|
|
#define HANDLE_FLAG_INHERIT 1
|
|
|
|
|
#endif
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
HANDLE winsock_lib;
|
|
|
|
|
static int winsock_inuse;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
BOOL
|
1996-05-03 18:32:52 +00:00
|
|
|
|
term_winsock (void)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib != NULL && winsock_inuse == 0)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
/* Not sure what would cause WSAENETDOWN, or even if it can happen
|
|
|
|
|
after WSAStartup returns successfully, but it seems reasonable
|
|
|
|
|
to allow unloading winsock anyway in that case. */
|
|
|
|
|
if (pfn_WSACleanup () == 0 ||
|
|
|
|
|
pfn_WSAGetLastError () == WSAENETDOWN)
|
|
|
|
|
{
|
|
|
|
|
if (FreeLibrary (winsock_lib))
|
|
|
|
|
winsock_lib = NULL;
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
return FALSE;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
BOOL
|
|
|
|
|
init_winsock (int load_now)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
WSADATA winsockData;
|
|
|
|
|
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib != NULL)
|
|
|
|
|
return TRUE;
|
1996-06-06 17:07:00 +00:00
|
|
|
|
|
|
|
|
|
pfn_SetHandleInformation = NULL;
|
|
|
|
|
pfn_SetHandleInformation
|
|
|
|
|
= (void *) GetProcAddress (GetModuleHandle ("kernel32.dll"),
|
|
|
|
|
"SetHandleInformation");
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
winsock_lib = LoadLibrary ("wsock32.dll");
|
|
|
|
|
|
|
|
|
|
if (winsock_lib != NULL)
|
|
|
|
|
{
|
|
|
|
|
/* dynamically link to socket functions */
|
|
|
|
|
|
|
|
|
|
#define LOAD_PROC(fn) \
|
|
|
|
|
if ((pfn_##fn = (void *) GetProcAddress (winsock_lib, #fn)) == NULL) \
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
LOAD_PROC( WSAStartup );
|
|
|
|
|
LOAD_PROC( WSASetLastError );
|
|
|
|
|
LOAD_PROC( WSAGetLastError );
|
|
|
|
|
LOAD_PROC( socket );
|
|
|
|
|
LOAD_PROC( bind );
|
|
|
|
|
LOAD_PROC( connect );
|
|
|
|
|
LOAD_PROC( ioctlsocket );
|
|
|
|
|
LOAD_PROC( recv );
|
|
|
|
|
LOAD_PROC( send );
|
|
|
|
|
LOAD_PROC( closesocket );
|
|
|
|
|
LOAD_PROC( shutdown );
|
|
|
|
|
LOAD_PROC( htons );
|
|
|
|
|
LOAD_PROC( ntohs );
|
|
|
|
|
LOAD_PROC( inet_addr );
|
|
|
|
|
LOAD_PROC( gethostname );
|
|
|
|
|
LOAD_PROC( gethostbyname );
|
|
|
|
|
LOAD_PROC( getservbyname );
|
2002-05-03 20:40:03 +00:00
|
|
|
|
LOAD_PROC( getpeername );
|
1996-05-03 18:32:52 +00:00
|
|
|
|
LOAD_PROC( WSACleanup );
|
2002-03-20 20:38:44 +00:00
|
|
|
|
LOAD_PROC( setsockopt );
|
|
|
|
|
LOAD_PROC( listen );
|
|
|
|
|
LOAD_PROC( getsockname );
|
|
|
|
|
LOAD_PROC( accept );
|
|
|
|
|
LOAD_PROC( recvfrom );
|
|
|
|
|
LOAD_PROC( sendto );
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
#undef LOAD_PROC
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* specify version 1.1 of winsock */
|
|
|
|
|
if (pfn_WSAStartup (0x101, &winsockData) == 0)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsockData.wVersion != 0x101)
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!load_now)
|
|
|
|
|
{
|
|
|
|
|
/* Report that winsock exists and is usable, but leave
|
|
|
|
|
socket functions disabled. I am assuming that calling
|
|
|
|
|
WSAStartup does not require any network interaction,
|
|
|
|
|
and in particular does not cause or require a dial-up
|
|
|
|
|
connection to be established. */
|
|
|
|
|
|
|
|
|
|
pfn_WSACleanup ();
|
|
|
|
|
FreeLibrary (winsock_lib);
|
|
|
|
|
winsock_lib = NULL;
|
|
|
|
|
}
|
|
|
|
|
winsock_inuse = 0;
|
|
|
|
|
return TRUE;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
FreeLibrary (winsock_lib);
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
winsock_lib = NULL;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
|
|
|
|
|
return FALSE;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int h_errno = 0;
|
|
|
|
|
|
|
|
|
|
/* function to set h_errno for compatability; map winsock error codes to
|
|
|
|
|
normal system codes where they overlap (non-overlapping definitions
|
|
|
|
|
are already in <sys/socket.h> */
|
|
|
|
|
static void set_errno ()
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib == NULL)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
h_errno = EINVAL;
|
|
|
|
|
else
|
|
|
|
|
h_errno = pfn_WSAGetLastError ();
|
|
|
|
|
|
|
|
|
|
switch (h_errno)
|
|
|
|
|
{
|
|
|
|
|
case WSAEACCES: h_errno = EACCES; break;
|
|
|
|
|
case WSAEBADF: h_errno = EBADF; break;
|
|
|
|
|
case WSAEFAULT: h_errno = EFAULT; break;
|
|
|
|
|
case WSAEINTR: h_errno = EINTR; break;
|
|
|
|
|
case WSAEINVAL: h_errno = EINVAL; break;
|
|
|
|
|
case WSAEMFILE: h_errno = EMFILE; break;
|
|
|
|
|
case WSAENAMETOOLONG: h_errno = ENAMETOOLONG; break;
|
|
|
|
|
case WSAENOTEMPTY: h_errno = ENOTEMPTY; break;
|
|
|
|
|
}
|
|
|
|
|
errno = h_errno;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void check_errno ()
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (h_errno == 0 && winsock_lib != NULL)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
pfn_WSASetLastError (0);
|
|
|
|
|
}
|
|
|
|
|
|
1999-06-16 20:00:19 +00:00
|
|
|
|
/* Extend strerror to handle the winsock-specific error codes. */
|
|
|
|
|
struct {
|
|
|
|
|
int errnum;
|
|
|
|
|
char * msg;
|
|
|
|
|
} _wsa_errlist[] = {
|
|
|
|
|
WSAEINTR , "Interrupted function call",
|
|
|
|
|
WSAEBADF , "Bad file descriptor",
|
|
|
|
|
WSAEACCES , "Permission denied",
|
|
|
|
|
WSAEFAULT , "Bad address",
|
|
|
|
|
WSAEINVAL , "Invalid argument",
|
|
|
|
|
WSAEMFILE , "Too many open files",
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1999-06-16 20:00:19 +00:00
|
|
|
|
WSAEWOULDBLOCK , "Resource temporarily unavailable",
|
|
|
|
|
WSAEINPROGRESS , "Operation now in progress",
|
|
|
|
|
WSAEALREADY , "Operation already in progress",
|
|
|
|
|
WSAENOTSOCK , "Socket operation on non-socket",
|
|
|
|
|
WSAEDESTADDRREQ , "Destination address required",
|
|
|
|
|
WSAEMSGSIZE , "Message too long",
|
|
|
|
|
WSAEPROTOTYPE , "Protocol wrong type for socket",
|
|
|
|
|
WSAENOPROTOOPT , "Bad protocol option",
|
|
|
|
|
WSAEPROTONOSUPPORT , "Protocol not supported",
|
|
|
|
|
WSAESOCKTNOSUPPORT , "Socket type not supported",
|
|
|
|
|
WSAEOPNOTSUPP , "Operation not supported",
|
|
|
|
|
WSAEPFNOSUPPORT , "Protocol family not supported",
|
|
|
|
|
WSAEAFNOSUPPORT , "Address family not supported by protocol family",
|
|
|
|
|
WSAEADDRINUSE , "Address already in use",
|
|
|
|
|
WSAEADDRNOTAVAIL , "Cannot assign requested address",
|
|
|
|
|
WSAENETDOWN , "Network is down",
|
|
|
|
|
WSAENETUNREACH , "Network is unreachable",
|
|
|
|
|
WSAENETRESET , "Network dropped connection on reset",
|
|
|
|
|
WSAECONNABORTED , "Software caused connection abort",
|
|
|
|
|
WSAECONNRESET , "Connection reset by peer",
|
|
|
|
|
WSAENOBUFS , "No buffer space available",
|
|
|
|
|
WSAEISCONN , "Socket is already connected",
|
|
|
|
|
WSAENOTCONN , "Socket is not connected",
|
|
|
|
|
WSAESHUTDOWN , "Cannot send after socket shutdown",
|
|
|
|
|
WSAETOOMANYREFS , "Too many references", /* not sure */
|
|
|
|
|
WSAETIMEDOUT , "Connection timed out",
|
|
|
|
|
WSAECONNREFUSED , "Connection refused",
|
|
|
|
|
WSAELOOP , "Network loop", /* not sure */
|
|
|
|
|
WSAENAMETOOLONG , "Name is too long",
|
|
|
|
|
WSAEHOSTDOWN , "Host is down",
|
|
|
|
|
WSAEHOSTUNREACH , "No route to host",
|
|
|
|
|
WSAENOTEMPTY , "Buffer not empty", /* not sure */
|
|
|
|
|
WSAEPROCLIM , "Too many processes",
|
|
|
|
|
WSAEUSERS , "Too many users", /* not sure */
|
|
|
|
|
WSAEDQUOT , "Double quote in host name", /* really not sure */
|
|
|
|
|
WSAESTALE , "Data is stale", /* not sure */
|
|
|
|
|
WSAEREMOTE , "Remote error", /* not sure */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1999-06-16 20:00:19 +00:00
|
|
|
|
WSASYSNOTREADY , "Network subsystem is unavailable",
|
|
|
|
|
WSAVERNOTSUPPORTED , "WINSOCK.DLL version out of range",
|
|
|
|
|
WSANOTINITIALISED , "Winsock not initialized successfully",
|
|
|
|
|
WSAEDISCON , "Graceful shutdown in progress",
|
|
|
|
|
#ifdef WSAENOMORE
|
|
|
|
|
WSAENOMORE , "No more operations allowed", /* not sure */
|
|
|
|
|
WSAECANCELLED , "Operation cancelled", /* not sure */
|
|
|
|
|
WSAEINVALIDPROCTABLE , "Invalid procedure table from service provider",
|
|
|
|
|
WSAEINVALIDPROVIDER , "Invalid service provider version number",
|
|
|
|
|
WSAEPROVIDERFAILEDINIT , "Unable to initialize a service provider",
|
|
|
|
|
WSASYSCALLFAILURE , "System call failured",
|
|
|
|
|
WSASERVICE_NOT_FOUND , "Service not found", /* not sure */
|
|
|
|
|
WSATYPE_NOT_FOUND , "Class type not found",
|
|
|
|
|
WSA_E_NO_MORE , "No more resources available", /* really not sure */
|
|
|
|
|
WSA_E_CANCELLED , "Operation already cancelled", /* really not sure */
|
|
|
|
|
WSAEREFUSED , "Operation refused", /* not sure */
|
|
|
|
|
#endif
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1999-06-16 20:00:19 +00:00
|
|
|
|
WSAHOST_NOT_FOUND , "Host not found",
|
|
|
|
|
WSATRY_AGAIN , "Authoritative host not found during name lookup",
|
|
|
|
|
WSANO_RECOVERY , "Non-recoverable error during name lookup",
|
|
|
|
|
WSANO_DATA , "Valid name, no data record of requested type",
|
|
|
|
|
|
|
|
|
|
-1, NULL
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
sys_strerror(int error_no)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
static char unknown_msg[40];
|
|
|
|
|
|
2000-08-22 22:49:27 +00:00
|
|
|
|
if (error_no >= 0 && error_no < sys_nerr)
|
|
|
|
|
return sys_errlist[error_no];
|
1999-06-16 20:00:19 +00:00
|
|
|
|
|
|
|
|
|
for (i = 0; _wsa_errlist[i].errnum >= 0; i++)
|
|
|
|
|
if (_wsa_errlist[i].errnum == error_no)
|
|
|
|
|
return _wsa_errlist[i].msg;
|
|
|
|
|
|
|
|
|
|
sprintf(unknown_msg, "Unidentified error: %d", error_no);
|
|
|
|
|
return unknown_msg;
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* [andrewi 3-May-96] I've had conflicting results using both methods,
|
|
|
|
|
but I believe the method of keeping the socket handle separate (and
|
|
|
|
|
insuring it is not inheritable) is the correct one. */
|
|
|
|
|
|
|
|
|
|
//#define SOCK_REPLACE_HANDLE
|
|
|
|
|
|
|
|
|
|
#ifdef SOCK_REPLACE_HANDLE
|
|
|
|
|
#define SOCK_HANDLE(fd) ((SOCKET) _get_osfhandle (fd))
|
|
|
|
|
#else
|
|
|
|
|
#define SOCK_HANDLE(fd) ((SOCKET) fd_info[fd].hnd)
|
|
|
|
|
#endif
|
|
|
|
|
|
2002-03-20 20:38:44 +00:00
|
|
|
|
int socket_to_fd (SOCKET s);
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
int
|
|
|
|
|
sys_socket(int af, int type, int protocol)
|
|
|
|
|
{
|
2002-03-20 20:38:44 +00:00
|
|
|
|
SOCKET s;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib == NULL)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return INVALID_SOCKET;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
|
|
|
|
|
/* call the real socket function */
|
2002-03-20 20:38:44 +00:00
|
|
|
|
s = pfn_socket (af, type, protocol);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (s != INVALID_SOCKET)
|
2002-03-20 20:38:44 +00:00
|
|
|
|
return socket_to_fd (s);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
2002-03-20 20:38:44 +00:00
|
|
|
|
set_errno ();
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Convert a SOCKET to a file descriptor. */
|
|
|
|
|
int
|
|
|
|
|
socket_to_fd (SOCKET s)
|
|
|
|
|
{
|
|
|
|
|
int fd;
|
|
|
|
|
child_process * cp;
|
|
|
|
|
|
|
|
|
|
/* Although under NT 3.5 _open_osfhandle will accept a socket
|
|
|
|
|
handle, if opened with SO_OPENTYPE == SO_SYNCHRONOUS_NONALERT,
|
|
|
|
|
that does not work under NT 3.1. However, we can get the same
|
|
|
|
|
effect by using a backdoor function to replace an existing
|
|
|
|
|
descriptor handle with the one we want. */
|
|
|
|
|
|
|
|
|
|
/* allocate a file descriptor (with appropriate flags) */
|
|
|
|
|
fd = _open ("NUL:", _O_RDWR);
|
|
|
|
|
if (fd >= 0)
|
|
|
|
|
{
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#ifdef SOCK_REPLACE_HANDLE
|
2002-03-20 20:38:44 +00:00
|
|
|
|
/* now replace handle to NUL with our socket handle */
|
|
|
|
|
CloseHandle ((HANDLE) _get_osfhandle (fd));
|
|
|
|
|
_free_osfhnd (fd);
|
|
|
|
|
_set_osfhnd (fd, s);
|
|
|
|
|
/* setmode (fd, _O_BINARY); */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#else
|
2002-03-20 20:38:44 +00:00
|
|
|
|
/* Make a non-inheritable copy of the socket handle. Note
|
|
|
|
|
that it is possible that sockets aren't actually kernel
|
|
|
|
|
handles, which appears to be the case on Windows 9x when
|
|
|
|
|
the MS Proxy winsock client is installed. */
|
|
|
|
|
{
|
|
|
|
|
/* Apparently there is a bug in NT 3.51 with some service
|
|
|
|
|
packs, which prevents using DuplicateHandle to make a
|
|
|
|
|
socket handle non-inheritable (causes WSACleanup to
|
|
|
|
|
hang). The work-around is to use SetHandleInformation
|
|
|
|
|
instead if it is available and implemented. */
|
|
|
|
|
if (pfn_SetHandleInformation)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
2002-03-20 20:38:44 +00:00
|
|
|
|
pfn_SetHandleInformation ((HANDLE) s, HANDLE_FLAG_INHERIT, 0);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
HANDLE parent = GetCurrentProcess ();
|
|
|
|
|
HANDLE new_s = INVALID_HANDLE_VALUE;
|
|
|
|
|
|
|
|
|
|
if (DuplicateHandle (parent,
|
|
|
|
|
(HANDLE) s,
|
|
|
|
|
parent,
|
|
|
|
|
&new_s,
|
|
|
|
|
0,
|
|
|
|
|
FALSE,
|
|
|
|
|
DUPLICATE_SAME_ACCESS))
|
1996-06-06 17:07:00 +00:00
|
|
|
|
{
|
2002-03-20 20:38:44 +00:00
|
|
|
|
/* It is possible that DuplicateHandle succeeds even
|
|
|
|
|
though the socket wasn't really a kernel handle,
|
|
|
|
|
because a real handle has the same value. So
|
|
|
|
|
test whether the new handle really is a socket. */
|
|
|
|
|
long nonblocking = 0;
|
|
|
|
|
if (pfn_ioctlsocket ((SOCKET) new_s, FIONBIO, &nonblocking) == 0)
|
2000-07-05 16:00:13 +00:00
|
|
|
|
{
|
2002-03-20 20:38:44 +00:00
|
|
|
|
pfn_closesocket (s);
|
|
|
|
|
s = (SOCKET) new_s;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
CloseHandle (new_s);
|
|
|
|
|
}
|
2003-02-04 14:56:31 +00:00
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
2002-03-20 20:38:44 +00:00
|
|
|
|
}
|
|
|
|
|
fd_info[fd].hnd = (HANDLE) s;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
2002-03-20 20:38:44 +00:00
|
|
|
|
/* set our own internal flags */
|
|
|
|
|
fd_info[fd].flags = FILE_SOCKET | FILE_BINARY | FILE_READ | FILE_WRITE;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
2002-03-20 20:38:44 +00:00
|
|
|
|
cp = new_child ();
|
|
|
|
|
if (cp)
|
|
|
|
|
{
|
|
|
|
|
cp->fd = fd;
|
|
|
|
|
cp->status = STATUS_READ_ACKNOWLEDGED;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
2002-03-20 20:38:44 +00:00
|
|
|
|
/* attach child_process to fd_info */
|
|
|
|
|
if (fd_info[ fd ].cp != NULL)
|
|
|
|
|
{
|
|
|
|
|
DebPrint (("sys_socket: fd_info[%d] apparently in use!\n", fd));
|
|
|
|
|
abort ();
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2002-03-20 20:38:44 +00:00
|
|
|
|
fd_info[ fd ].cp = cp;
|
|
|
|
|
|
|
|
|
|
/* success! */
|
|
|
|
|
winsock_inuse++; /* count open sockets */
|
|
|
|
|
return fd;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2002-03-20 20:38:44 +00:00
|
|
|
|
/* clean up */
|
|
|
|
|
_close (fd);
|
|
|
|
|
}
|
|
|
|
|
pfn_closesocket (s);
|
|
|
|
|
h_errno = EMFILE;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_bind (int s, const struct sockaddr * addr, int namelen)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib == NULL)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
int rc = pfn_bind (SOCK_HANDLE (s), addr, namelen);
|
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_connect (int s, const struct sockaddr * name, int namelen)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib == NULL)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
int rc = pfn_connect (SOCK_HANDLE (s), name, namelen);
|
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u_short
|
|
|
|
|
sys_htons (u_short hostshort)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
return (winsock_lib != NULL) ?
|
1996-05-03 18:32:52 +00:00
|
|
|
|
pfn_htons (hostshort) : hostshort;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u_short
|
|
|
|
|
sys_ntohs (u_short netshort)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
return (winsock_lib != NULL) ?
|
1996-05-03 18:32:52 +00:00
|
|
|
|
pfn_ntohs (netshort) : netshort;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned long
|
|
|
|
|
sys_inet_addr (const char * cp)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
return (winsock_lib != NULL) ?
|
1996-05-03 18:32:52 +00:00
|
|
|
|
pfn_inet_addr (cp) : INADDR_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_gethostname (char * name, int namelen)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib != NULL)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
return pfn_gethostname (name, namelen);
|
|
|
|
|
|
|
|
|
|
if (namelen > MAX_COMPUTERNAME_LENGTH)
|
2000-08-22 22:49:27 +00:00
|
|
|
|
return !GetComputerName (name, (DWORD *)&namelen);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
h_errno = EFAULT;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct hostent *
|
|
|
|
|
sys_gethostbyname(const char * name)
|
|
|
|
|
{
|
|
|
|
|
struct hostent * host;
|
|
|
|
|
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib == NULL)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
host = pfn_gethostbyname (name);
|
|
|
|
|
if (!host)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return host;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct servent *
|
|
|
|
|
sys_getservbyname(const char * name, const char * proto)
|
|
|
|
|
{
|
|
|
|
|
struct servent * serv;
|
|
|
|
|
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib == NULL)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
serv = pfn_getservbyname (name, proto);
|
|
|
|
|
if (!serv)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return serv;
|
|
|
|
|
}
|
|
|
|
|
|
2002-05-03 20:40:03 +00:00
|
|
|
|
int
|
|
|
|
|
sys_getpeername (int s, struct sockaddr *addr, int * namelen)
|
|
|
|
|
{
|
|
|
|
|
if (winsock_lib == NULL)
|
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
int rc = pfn_getpeername (SOCK_HANDLE (s), addr, namelen);
|
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1997-07-10 20:10:09 +00:00
|
|
|
|
int
|
|
|
|
|
sys_shutdown (int s, int how)
|
|
|
|
|
{
|
|
|
|
|
if (winsock_lib == NULL)
|
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
int rc = pfn_shutdown (SOCK_HANDLE (s), how);
|
2002-03-20 20:38:44 +00:00
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_setsockopt (int s, int level, int optname, const char * optval, int optlen)
|
|
|
|
|
{
|
|
|
|
|
if (winsock_lib == NULL)
|
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
int rc = pfn_setsockopt (SOCK_HANDLE (s), level, optname,
|
|
|
|
|
optval, optlen);
|
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
return SOCKET_ERROR;
|
2002-03-20 20:38:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_listen (int s, int backlog)
|
|
|
|
|
{
|
|
|
|
|
if (winsock_lib == NULL)
|
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
int rc = pfn_listen (SOCK_HANDLE (s), backlog);
|
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
return SOCKET_ERROR;
|
2002-03-20 20:38:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_getsockname (int s, struct sockaddr * name, int * namelen)
|
|
|
|
|
{
|
|
|
|
|
if (winsock_lib == NULL)
|
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
int rc = pfn_getsockname (SOCK_HANDLE (s), name, namelen);
|
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
return SOCKET_ERROR;
|
2002-03-20 20:38:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_accept (int s, struct sockaddr * addr, int * addrlen)
|
|
|
|
|
{
|
|
|
|
|
if (winsock_lib == NULL)
|
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
2002-04-08 16:45:43 +00:00
|
|
|
|
SOCKET t = pfn_accept (SOCK_HANDLE (s), addr, addrlen);
|
|
|
|
|
if (t != INVALID_SOCKET)
|
|
|
|
|
return socket_to_fd (t);
|
2002-03-20 20:38:44 +00:00
|
|
|
|
|
|
|
|
|
set_errno ();
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_recvfrom (int s, char * buf, int len, int flags,
|
|
|
|
|
struct sockaddr * from, int * fromlen)
|
|
|
|
|
{
|
|
|
|
|
if (winsock_lib == NULL)
|
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
int rc = pfn_recvfrom (SOCK_HANDLE (s), buf, len, flags, from, fromlen);
|
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_sendto (int s, const char * buf, int len, int flags,
|
|
|
|
|
const struct sockaddr * to, int tolen)
|
|
|
|
|
{
|
|
|
|
|
if (winsock_lib == NULL)
|
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
int rc = pfn_sendto (SOCK_HANDLE (s), buf, len, flags, to, tolen);
|
1997-07-10 20:10:09 +00:00
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno ();
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
2002-05-03 20:40:03 +00:00
|
|
|
|
/* Windows does not have an fcntl function. Provide an implementation
|
|
|
|
|
solely for making sockets non-blocking. */
|
|
|
|
|
int
|
|
|
|
|
fcntl (int s, int cmd, int options)
|
|
|
|
|
{
|
|
|
|
|
if (winsock_lib == NULL)
|
|
|
|
|
{
|
|
|
|
|
h_errno = ENETDOWN;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check_errno ();
|
|
|
|
|
if (fd_info[s].flags & FILE_SOCKET)
|
|
|
|
|
{
|
|
|
|
|
if (cmd == F_SETFL && options == O_NDELAY)
|
|
|
|
|
{
|
|
|
|
|
unsigned long nblock = 1;
|
|
|
|
|
int rc = pfn_ioctlsocket (SOCK_HANDLE (s), FIONBIO, &nblock);
|
|
|
|
|
if (rc == SOCKET_ERROR)
|
|
|
|
|
set_errno();
|
|
|
|
|
/* Keep track of the fact that we set this to non-blocking. */
|
|
|
|
|
fd_info[s].flags |= FILE_NDELAY;
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
h_errno = EINVAL;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
h_errno = ENOTSOCK;
|
|
|
|
|
return SOCKET_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#endif /* HAVE_SOCKETS */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Shadow main io functions: we need to handle pipes and sockets more
|
|
|
|
|
intelligently, and implement non-blocking mode as well. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_close (int fd)
|
|
|
|
|
{
|
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
|
|
if (fd < 0 || fd >= MAXDESC)
|
|
|
|
|
{
|
|
|
|
|
errno = EBADF;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (fd_info[fd].cp)
|
|
|
|
|
{
|
|
|
|
|
child_process * cp = fd_info[fd].cp;
|
|
|
|
|
|
|
|
|
|
fd_info[fd].cp = NULL;
|
|
|
|
|
|
|
|
|
|
if (CHILD_ACTIVE (cp))
|
|
|
|
|
{
|
|
|
|
|
/* if last descriptor to active child_process then cleanup */
|
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i < MAXDESC; i++)
|
|
|
|
|
{
|
|
|
|
|
if (i == fd)
|
|
|
|
|
continue;
|
|
|
|
|
if (fd_info[i].cp == cp)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (i == MAXDESC)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
#ifdef HAVE_SOCKETS
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (fd_info[fd].flags & FILE_SOCKET)
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
#ifndef SOCK_REPLACE_HANDLE
|
|
|
|
|
if (winsock_lib == NULL) abort ();
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
pfn_shutdown (SOCK_HANDLE (fd), 2);
|
|
|
|
|
rc = pfn_closesocket (SOCK_HANDLE (fd));
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
#endif
|
|
|
|
|
winsock_inuse--; /* count open sockets */
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
delete_child (cp);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Note that sockets do not need special treatment here (at least on
|
1997-01-20 00:38:22 +00:00
|
|
|
|
NT and Windows 95 using the standard tcp/ip stacks) - it appears that
|
1996-05-03 18:32:52 +00:00
|
|
|
|
closesocket is equivalent to CloseHandle, which is to be expected
|
|
|
|
|
because socket handles are fully fledged kernel handles. */
|
|
|
|
|
rc = _close (fd);
|
|
|
|
|
|
|
|
|
|
if (rc == 0)
|
|
|
|
|
fd_info[fd].flags = 0;
|
|
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_dup (int fd)
|
|
|
|
|
{
|
|
|
|
|
int new_fd;
|
|
|
|
|
|
|
|
|
|
new_fd = _dup (fd);
|
|
|
|
|
if (new_fd >= 0)
|
|
|
|
|
{
|
|
|
|
|
/* duplicate our internal info as well */
|
|
|
|
|
fd_info[new_fd] = fd_info[fd];
|
|
|
|
|
}
|
|
|
|
|
return new_fd;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_dup2 (int src, int dst)
|
|
|
|
|
{
|
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
|
|
if (dst < 0 || dst >= MAXDESC)
|
|
|
|
|
{
|
|
|
|
|
errno = EBADF;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* make sure we close the destination first if it's a pipe or socket */
|
|
|
|
|
if (src != dst && fd_info[dst].flags != 0)
|
|
|
|
|
sys_close (dst);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
rc = _dup2 (src, dst);
|
|
|
|
|
if (rc == 0)
|
|
|
|
|
{
|
|
|
|
|
/* duplicate our internal info as well */
|
|
|
|
|
fd_info[dst] = fd_info[src];
|
|
|
|
|
}
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Unix pipe() has only one arg */
|
|
|
|
|
int
|
|
|
|
|
sys_pipe (int * phandles)
|
|
|
|
|
{
|
|
|
|
|
int rc;
|
|
|
|
|
unsigned flags;
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* make pipe handles non-inheritable; when we spawn a child, we
|
|
|
|
|
replace the relevant handle with an inheritable one. Also put
|
|
|
|
|
pipes into binary mode; we will do text mode translation ourselves
|
|
|
|
|
if required. */
|
|
|
|
|
rc = _pipe (phandles, 0, _O_NOINHERIT | _O_BINARY);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
if (rc == 0)
|
|
|
|
|
{
|
2003-08-27 22:57:54 +00:00
|
|
|
|
/* Protect against overflow, since Windows can open more handles than
|
|
|
|
|
our fd_info array has room for. */
|
|
|
|
|
if (phandles[0] >= MAXDESC || phandles[1] >= MAXDESC)
|
|
|
|
|
{
|
|
|
|
|
_close (phandles[0]);
|
|
|
|
|
_close (phandles[1]);
|
|
|
|
|
rc = -1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
flags = FILE_PIPE | FILE_READ | FILE_BINARY;
|
|
|
|
|
fd_info[phandles[0]].flags = flags;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
2003-08-27 22:57:54 +00:00
|
|
|
|
flags = FILE_PIPE | FILE_WRITE | FILE_BINARY;
|
|
|
|
|
fd_info[phandles[1]].flags = flags;
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-17 21:45:30 +00:00
|
|
|
|
/* From ntproc.c */
|
2004-05-17 21:05:58 +00:00
|
|
|
|
extern int w32_pipe_read_delay;
|
1996-05-17 21:45:30 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* Function to do blocking read of one byte, needed to implement
|
|
|
|
|
select. It is only allowed on sockets and pipes. */
|
|
|
|
|
int
|
|
|
|
|
_sys_read_ahead (int fd)
|
|
|
|
|
{
|
|
|
|
|
child_process * cp;
|
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
|
|
if (fd < 0 || fd >= MAXDESC)
|
|
|
|
|
return STATUS_READ_ERROR;
|
|
|
|
|
|
|
|
|
|
cp = fd_info[fd].cp;
|
|
|
|
|
|
|
|
|
|
if (cp == NULL || cp->fd != fd || cp->status != STATUS_READ_READY)
|
|
|
|
|
return STATUS_READ_ERROR;
|
|
|
|
|
|
|
|
|
|
if ((fd_info[fd].flags & (FILE_PIPE | FILE_SOCKET)) == 0
|
|
|
|
|
|| (fd_info[fd].flags & FILE_READ) == 0)
|
|
|
|
|
{
|
|
|
|
|
DebPrint (("_sys_read_ahead: internal error: fd %d is not a pipe or socket!\n", fd));
|
|
|
|
|
abort ();
|
|
|
|
|
}
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
cp->status = STATUS_READ_IN_PROGRESS;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (fd_info[fd].flags & FILE_PIPE)
|
1996-05-17 21:45:30 +00:00
|
|
|
|
{
|
|
|
|
|
rc = _read (fd, &cp->chr, sizeof (char));
|
|
|
|
|
|
|
|
|
|
/* Give subprocess time to buffer some more output for us before
|
1997-01-20 00:38:22 +00:00
|
|
|
|
reporting that input is available; we need this because Windows 95
|
1996-05-17 21:45:30 +00:00
|
|
|
|
connects DOS programs to pipes by making the pipe appear to be
|
|
|
|
|
the normal console stdout - as a result most DOS programs will
|
|
|
|
|
write to stdout without buffering, ie. one character at a
|
1996-11-19 04:59:23 +00:00
|
|
|
|
time. Even some W32 programs do this - "dir" in a command
|
1996-05-17 21:45:30 +00:00
|
|
|
|
shell on NT is very slow if we don't do this. */
|
|
|
|
|
if (rc > 0)
|
|
|
|
|
{
|
2004-05-17 21:05:58 +00:00
|
|
|
|
int wait = w32_pipe_read_delay;
|
1996-05-17 21:45:30 +00:00
|
|
|
|
|
|
|
|
|
if (wait > 0)
|
|
|
|
|
Sleep (wait);
|
|
|
|
|
else if (wait < 0)
|
|
|
|
|
while (++wait <= 0)
|
|
|
|
|
/* Yield remainder of our time slice, effectively giving a
|
|
|
|
|
temporary priority boost to the child process. */
|
|
|
|
|
Sleep (0);
|
|
|
|
|
}
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#ifdef HAVE_SOCKETS
|
|
|
|
|
else if (fd_info[fd].flags & FILE_SOCKET)
|
2002-05-03 20:40:03 +00:00
|
|
|
|
{
|
|
|
|
|
unsigned long nblock = 0;
|
|
|
|
|
/* We always want this to block, so temporarily disable NDELAY. */
|
|
|
|
|
if (fd_info[fd].flags & FILE_NDELAY)
|
|
|
|
|
pfn_ioctlsocket (SOCK_HANDLE (fd), FIONBIO, &nblock);
|
|
|
|
|
|
|
|
|
|
rc = pfn_recv (SOCK_HANDLE (fd), &cp->chr, sizeof (char), 0);
|
|
|
|
|
|
|
|
|
|
if (fd_info[fd].flags & FILE_NDELAY)
|
|
|
|
|
{
|
|
|
|
|
nblock = 1;
|
|
|
|
|
pfn_ioctlsocket (SOCK_HANDLE (fd), FIONBIO, &nblock);
|
|
|
|
|
}
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#endif
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (rc == sizeof (char))
|
|
|
|
|
cp->status = STATUS_READ_SUCCEEDED;
|
|
|
|
|
else
|
|
|
|
|
cp->status = STATUS_READ_FAILED;
|
|
|
|
|
|
|
|
|
|
return cp->status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
sys_read (int fd, char * buffer, unsigned int count)
|
|
|
|
|
{
|
|
|
|
|
int nchars;
|
|
|
|
|
int to_read;
|
|
|
|
|
DWORD waiting;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
char * orig_buffer = buffer;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
if (fd < 0 || fd >= MAXDESC)
|
|
|
|
|
{
|
|
|
|
|
errno = EBADF;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (fd_info[fd].flags & (FILE_PIPE | FILE_SOCKET))
|
|
|
|
|
{
|
|
|
|
|
child_process *cp = fd_info[fd].cp;
|
|
|
|
|
|
|
|
|
|
if ((fd_info[fd].flags & FILE_READ) == 0)
|
|
|
|
|
{
|
|
|
|
|
errno = EBADF;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
nchars = 0;
|
|
|
|
|
|
|
|
|
|
/* re-read CR carried over from last read */
|
|
|
|
|
if (fd_info[fd].flags & FILE_LAST_CR)
|
|
|
|
|
{
|
|
|
|
|
if (fd_info[fd].flags & FILE_BINARY) abort ();
|
|
|
|
|
*buffer++ = 0x0d;
|
|
|
|
|
count--;
|
|
|
|
|
nchars++;
|
1998-04-17 05:07:15 +00:00
|
|
|
|
fd_info[fd].flags &= ~FILE_LAST_CR;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* presence of a child_process structure means we are operating in
|
|
|
|
|
non-blocking mode - otherwise we just call _read directly.
|
|
|
|
|
Note that the child_process structure might be missing because
|
|
|
|
|
reap_subprocess has been called; in this case the pipe is
|
|
|
|
|
already broken, so calling _read on it is okay. */
|
|
|
|
|
if (cp)
|
|
|
|
|
{
|
|
|
|
|
int current_status = cp->status;
|
|
|
|
|
|
|
|
|
|
switch (current_status)
|
|
|
|
|
{
|
|
|
|
|
case STATUS_READ_FAILED:
|
|
|
|
|
case STATUS_READ_ERROR:
|
1998-04-17 05:07:15 +00:00
|
|
|
|
/* report normal EOF if nothing in buffer */
|
|
|
|
|
if (nchars <= 0)
|
|
|
|
|
fd_info[fd].flags |= FILE_AT_EOF;
|
|
|
|
|
return nchars;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
case STATUS_READ_READY:
|
|
|
|
|
case STATUS_READ_IN_PROGRESS:
|
|
|
|
|
DebPrint (("sys_read called when read is in progress\n"));
|
|
|
|
|
errno = EWOULDBLOCK;
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
case STATUS_READ_SUCCEEDED:
|
|
|
|
|
/* consume read-ahead char */
|
|
|
|
|
*buffer++ = cp->chr;
|
|
|
|
|
count--;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
nchars++;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
cp->status = STATUS_READ_ACKNOWLEDGED;
|
|
|
|
|
ResetEvent (cp->char_avail);
|
|
|
|
|
|
|
|
|
|
case STATUS_READ_ACKNOWLEDGED:
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
DebPrint (("sys_read: bad status %d\n", current_status));
|
|
|
|
|
errno = EBADF;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (fd_info[fd].flags & FILE_PIPE)
|
|
|
|
|
{
|
|
|
|
|
PeekNamedPipe ((HANDLE) _get_osfhandle (fd), NULL, 0, NULL, &waiting, NULL);
|
|
|
|
|
to_read = min (waiting, (DWORD) count);
|
1998-04-17 05:07:15 +00:00
|
|
|
|
|
|
|
|
|
if (to_read > 0)
|
|
|
|
|
nchars += _read (fd, buffer, to_read);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
#ifdef HAVE_SOCKETS
|
|
|
|
|
else /* FILE_SOCKET */
|
|
|
|
|
{
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib == NULL) abort ();
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
/* do the equivalent of a non-blocking read */
|
|
|
|
|
pfn_ioctlsocket (SOCK_HANDLE (fd), FIONREAD, &waiting);
|
1997-09-03 00:43:20 +00:00
|
|
|
|
if (waiting == 0 && nchars == 0)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
h_errno = errno = EWOULDBLOCK;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (waiting)
|
|
|
|
|
{
|
|
|
|
|
/* always use binary mode for sockets */
|
1997-09-03 00:43:20 +00:00
|
|
|
|
int res = pfn_recv (SOCK_HANDLE (fd), buffer, count, 0);
|
|
|
|
|
if (res == SOCKET_ERROR)
|
1996-05-03 18:32:52 +00:00
|
|
|
|
{
|
|
|
|
|
DebPrint(("sys_read.recv failed with error %d on socket %ld\n",
|
|
|
|
|
pfn_WSAGetLastError (), SOCK_HANDLE (fd)));
|
1997-09-03 00:43:20 +00:00
|
|
|
|
set_errno ();
|
|
|
|
|
return -1;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
1997-09-03 00:43:20 +00:00
|
|
|
|
nchars += res;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
else
|
1998-04-17 05:07:15 +00:00
|
|
|
|
{
|
|
|
|
|
int nread = _read (fd, buffer, count);
|
|
|
|
|
if (nread >= 0)
|
|
|
|
|
nchars += nread;
|
|
|
|
|
else if (nchars == 0)
|
|
|
|
|
nchars = nread;
|
|
|
|
|
}
|
1997-09-03 00:43:20 +00:00
|
|
|
|
|
1998-04-17 05:07:15 +00:00
|
|
|
|
if (nchars <= 0)
|
|
|
|
|
fd_info[fd].flags |= FILE_AT_EOF;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
/* Perform text mode translation if required. */
|
1998-04-17 05:07:15 +00:00
|
|
|
|
else if ((fd_info[fd].flags & FILE_BINARY) == 0)
|
1997-09-03 00:43:20 +00:00
|
|
|
|
{
|
|
|
|
|
nchars = crlf_to_lf (nchars, orig_buffer);
|
|
|
|
|
/* If buffer contains only CR, return that. To be absolutely
|
|
|
|
|
sure we should attempt to read the next char, but in
|
|
|
|
|
practice a CR to be followed by LF would not appear by
|
|
|
|
|
itself in the buffer. */
|
|
|
|
|
if (nchars > 1 && orig_buffer[nchars - 1] == 0x0d)
|
|
|
|
|
{
|
|
|
|
|
fd_info[fd].flags |= FILE_LAST_CR;
|
|
|
|
|
nchars--;
|
|
|
|
|
}
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
nchars = _read (fd, buffer, count);
|
|
|
|
|
|
1997-09-03 00:43:20 +00:00
|
|
|
|
return nchars;
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* For now, don't bother with a non-blocking mode */
|
|
|
|
|
int
|
|
|
|
|
sys_write (int fd, const void * buffer, unsigned int count)
|
|
|
|
|
{
|
|
|
|
|
int nchars;
|
|
|
|
|
|
|
|
|
|
if (fd < 0 || fd >= MAXDESC)
|
|
|
|
|
{
|
|
|
|
|
errno = EBADF;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (fd_info[fd].flags & (FILE_PIPE | FILE_SOCKET))
|
1997-09-03 00:43:20 +00:00
|
|
|
|
{
|
|
|
|
|
if ((fd_info[fd].flags & FILE_WRITE) == 0)
|
|
|
|
|
{
|
|
|
|
|
errno = EBADF;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Perform text mode translation if required. */
|
|
|
|
|
if ((fd_info[fd].flags & FILE_BINARY) == 0)
|
|
|
|
|
{
|
|
|
|
|
char * tmpbuf = alloca (count * 2);
|
|
|
|
|
unsigned char * src = (void *)buffer;
|
|
|
|
|
unsigned char * dst = tmpbuf;
|
|
|
|
|
int nbytes = count;
|
|
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
|
{
|
|
|
|
|
unsigned char *next;
|
|
|
|
|
/* copy next line or remaining bytes */
|
|
|
|
|
next = _memccpy (dst, src, '\n', nbytes);
|
|
|
|
|
if (next)
|
|
|
|
|
{
|
|
|
|
|
/* copied one line ending with '\n' */
|
|
|
|
|
int copied = next - dst;
|
|
|
|
|
nbytes -= copied;
|
|
|
|
|
src += copied;
|
|
|
|
|
/* insert '\r' before '\n' */
|
|
|
|
|
next[-1] = '\r';
|
|
|
|
|
next[0] = '\n';
|
|
|
|
|
dst = next + 1;
|
|
|
|
|
count++;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
}
|
1997-09-03 00:43:20 +00:00
|
|
|
|
else
|
|
|
|
|
/* copied remaining partial line -> now finished */
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
buffer = tmpbuf;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#ifdef HAVE_SOCKETS
|
|
|
|
|
if (fd_info[fd].flags & FILE_SOCKET)
|
|
|
|
|
{
|
2002-11-25 23:30:19 +00:00
|
|
|
|
unsigned long nblock = 0;
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
if (winsock_lib == NULL) abort ();
|
2002-11-25 23:30:19 +00:00
|
|
|
|
|
|
|
|
|
/* TODO: implement select() properly so non-blocking I/O works. */
|
|
|
|
|
/* For now, make sure the write blocks. */
|
|
|
|
|
if (fd_info[fd].flags & FILE_NDELAY)
|
|
|
|
|
pfn_ioctlsocket (SOCK_HANDLE (fd), FIONBIO, &nblock);
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
nchars = pfn_send (SOCK_HANDLE (fd), buffer, count, 0);
|
2002-11-25 23:30:19 +00:00
|
|
|
|
|
|
|
|
|
/* Set the socket back to non-blocking if it was before,
|
|
|
|
|
for other operations that support it. */
|
|
|
|
|
if (fd_info[fd].flags & FILE_NDELAY)
|
|
|
|
|
{
|
|
|
|
|
nblock = 1;
|
|
|
|
|
pfn_ioctlsocket (SOCK_HANDLE (fd), FIONBIO, &nblock);
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
if (nchars == SOCKET_ERROR)
|
|
|
|
|
{
|
2002-11-27 23:48:33 +00:00
|
|
|
|
DebPrint(("sys_write.send failed with error %d on socket %ld\n",
|
1996-05-03 18:32:52 +00:00
|
|
|
|
pfn_WSAGetLastError (), SOCK_HANDLE (fd)));
|
|
|
|
|
set_errno ();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
#endif
|
|
|
|
|
nchars = _write (fd, buffer, count);
|
|
|
|
|
|
|
|
|
|
return nchars;
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-17 05:07:15 +00:00
|
|
|
|
static void
|
|
|
|
|
check_windows_init_file ()
|
|
|
|
|
{
|
|
|
|
|
extern int noninteractive, inhibit_window_system;
|
|
|
|
|
|
|
|
|
|
/* A common indication that Emacs is not installed properly is when
|
|
|
|
|
it cannot find the Windows installation file. If this file does
|
|
|
|
|
not exist in the expected place, tell the user. */
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
if (!noninteractive && !inhibit_window_system)
|
1998-10-07 19:47:48 +00:00
|
|
|
|
{
|
|
|
|
|
extern Lisp_Object Vwindow_system, Vload_path, Qfile_exists_p;
|
1998-12-03 03:14:46 +00:00
|
|
|
|
Lisp_Object objs[2];
|
1998-10-27 23:42:26 +00:00
|
|
|
|
Lisp_Object full_load_path;
|
1998-10-07 19:47:48 +00:00
|
|
|
|
Lisp_Object init_file;
|
|
|
|
|
int fd;
|
1998-04-17 05:07:15 +00:00
|
|
|
|
|
1998-12-03 03:14:46 +00:00
|
|
|
|
objs[0] = Vload_path;
|
|
|
|
|
objs[1] = decode_env_path (0, (getenv ("EMACSLOADPATH")));
|
|
|
|
|
full_load_path = Fappend (2, objs);
|
1998-10-07 19:47:48 +00:00
|
|
|
|
init_file = build_string ("term/w32-win");
|
2002-04-30 01:02:01 +00:00
|
|
|
|
fd = openp (full_load_path, init_file, Vload_suffixes, NULL, Qnil);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
if (fd < 0)
|
1998-10-07 19:47:48 +00:00
|
|
|
|
{
|
1998-10-27 23:42:26 +00:00
|
|
|
|
Lisp_Object load_path_print = Fprin1_to_string (full_load_path, Qnil);
|
2002-07-15 00:01:34 +00:00
|
|
|
|
char *init_file_name = SDATA (init_file);
|
|
|
|
|
char *load_path = SDATA (load_path_print);
|
1998-10-07 19:47:48 +00:00
|
|
|
|
char *buffer = alloca (1024);
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
sprintf (buffer,
|
1998-10-07 19:47:48 +00:00
|
|
|
|
"The Emacs Windows initialization file \"%s.el\" "
|
|
|
|
|
"could not be found in your Emacs installation. "
|
|
|
|
|
"Emacs checked the following directories for this file:\n"
|
|
|
|
|
"\n%s\n\n"
|
|
|
|
|
"When Emacs cannot find this file, it usually means that it "
|
|
|
|
|
"was not installed properly, or its distribution file was "
|
|
|
|
|
"not unpacked properly.\nSee the README.W32 file in the "
|
|
|
|
|
"top-level Emacs directory for more information.",
|
|
|
|
|
init_file_name, load_path);
|
|
|
|
|
MessageBox (NULL,
|
|
|
|
|
buffer,
|
|
|
|
|
"Emacs Abort Dialog",
|
|
|
|
|
MB_OK | MB_ICONEXCLAMATION | MB_TASKMODAL);
|
1998-04-17 05:07:15 +00:00
|
|
|
|
/* Use the low-level Emacs abort. */
|
|
|
|
|
#undef abort
|
1998-10-07 19:47:48 +00:00
|
|
|
|
abort ();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2000-08-22 22:49:27 +00:00
|
|
|
|
_close (fd);
|
1998-10-07 19:47:48 +00:00
|
|
|
|
}
|
1998-04-17 05:07:15 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
term_ntproc ()
|
|
|
|
|
{
|
|
|
|
|
#ifdef HAVE_SOCKETS
|
|
|
|
|
/* shutdown the socket interface if necessary */
|
|
|
|
|
term_winsock ();
|
|
|
|
|
#endif
|
2004-11-08 Benjamin Riefenstahl <Benjamin.Riefenstahl@epost.de>
* w32select.c: Summary: Thorough rework to implement Unicode
clipboard operations and delayed rendering.
Drop last_clipboard_text and related code, keep track of
ownership via clipboard_owner instead. Drop old #if0
sections.
(DEFAULT_LCID, ANSICP, OEMCP, QUNICODE, QANSICP, QOEMCP)
(clipboard_owner, modifying_clipboard, cfg_coding_system)
(cfg_codepage, cfg_lcid, cfg_clipboard_type, current_text)
(current_coding_system, current_requires_encoding)
(current_num_nls, current_clipboard_type, current_lcid): New
static variables.
(convert_to_handle_as_ascii, convert_to_handle_as_coded)
(render, render_all, run_protected, lisp_error_handler)
(owner_callback, create_owner, setup_config)
(enum_locale_callback, cp_from_locale, coding_from_cp): New
local functions.
(term_w32select, globals_of_w32select): New global functions.
(Fw32_set_clipboard_data): Ignore parameter FRAME, use
clipboard_owner instead. Use delayed rendering and provide
all text formats. Provide CF_LOCALE if necessary.
(Fw32_get_clipboard_data): Handle CF_UNICODETEXT and
CF_LOCALE. Fall back to CF_TEXT, if CF_UNICODETEXT is not
available. Force DOS line-ends for decoding.
(Fx_selection_exists_p): Handle CF_UNICODETEXT.
(syms_of_w32select): Init and register new variables.
* w32.h: Add prototypes for globals_of_w32select and
term_w32select. Make the neighboring K&R declarations into
prototypes, too.
* emacs.c: Include w32.h to get function prototypes.
(main): Call globals_of_w32select.
* w32.c (term_ntproc): Call term_w32select.
* mule-cmds.el (set-locale-environment): Remove call to
set-selection-coding-system on Windows.
* s/ms-w32.h: Guard MSC-specific #pragmas with an #ifdef.
2005-02-15 23:19:26 +00:00
|
|
|
|
|
|
|
|
|
term_w32select ();
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
init_ntproc ()
|
|
|
|
|
{
|
|
|
|
|
#ifdef HAVE_SOCKETS
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
/* Initialise the socket interface now if available and requested by
|
|
|
|
|
the user by defining PRELOAD_WINSOCK; otherwise loading will be
|
1996-11-19 04:59:23 +00:00
|
|
|
|
delayed until open-network-stream is called (w32-has-winsock can
|
(init_environment): Read PRELOAD_WINSOCK from registry if
not set in environment.
(winsock_inuse) [HAVE_SOCKETS]: New variable.
(have_winsock) [HAVE_SOCKETS]: Obsolete variable removed.
(term_winsock) [HAVE_SOCKETS]: Only unload winsock library if there
are no active sockets still open, and if the cleanup function
succeeds. Return TRUE if winsock is unloaded.
(init_winsock) [HAVE_SOCKETS]: Load winsock if not already loaded,
and return TRUE if winsock support is available. Unload winsock
immediately if new parameter load_now is false. Check that
WSAStartup supports the winsock version we requested.
(set_errno, check_errno, sys_socket, sys_bind, sys_connect, sys_htons,
sys_ntohs, sys_inet_addr, sys_gethostname, sys_gethostbyname,
sys_getservbyname, sys_close, sys_read, sys_write) [HAVE_SOCKETS]:
Check winsock_lib instead of have_winsock to determine if winsock
support is available.
(sys_socket, sys_close) [HAVE_SOCKETS]: Count sockets in use.
(init_ntproc) [HAVE_SOCKETS]: Only load winsock library on startup
if PRELOAD_WINSOCK is set in environment (or registry).
1996-06-08 00:22:50 +00:00
|
|
|
|
also be used to dynamically load or reload winsock).
|
|
|
|
|
|
|
|
|
|
Conveniently, init_environment is called before us, so
|
|
|
|
|
PRELOAD_WINSOCK can be set in the registry. */
|
|
|
|
|
|
|
|
|
|
/* Always initialize this correctly. */
|
|
|
|
|
winsock_lib = NULL;
|
|
|
|
|
|
|
|
|
|
if (getenv ("PRELOAD_WINSOCK") != NULL)
|
|
|
|
|
init_winsock (TRUE);
|
1996-05-03 18:32:52 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/* Initial preparation for subprocess support: replace our standard
|
|
|
|
|
handles with non-inheritable versions. */
|
|
|
|
|
{
|
|
|
|
|
HANDLE parent;
|
|
|
|
|
HANDLE stdin_save = INVALID_HANDLE_VALUE;
|
|
|
|
|
HANDLE stdout_save = INVALID_HANDLE_VALUE;
|
|
|
|
|
HANDLE stderr_save = INVALID_HANDLE_VALUE;
|
|
|
|
|
|
|
|
|
|
parent = GetCurrentProcess ();
|
|
|
|
|
|
|
|
|
|
/* ignore errors when duplicating and closing; typically the
|
|
|
|
|
handles will be invalid when running as a gui program. */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
DuplicateHandle (parent,
|
|
|
|
|
GetStdHandle (STD_INPUT_HANDLE),
|
1996-05-03 18:32:52 +00:00
|
|
|
|
parent,
|
2003-02-04 14:56:31 +00:00
|
|
|
|
&stdin_save,
|
|
|
|
|
0,
|
|
|
|
|
FALSE,
|
1996-05-03 18:32:52 +00:00
|
|
|
|
DUPLICATE_SAME_ACCESS);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
DuplicateHandle (parent,
|
|
|
|
|
GetStdHandle (STD_OUTPUT_HANDLE),
|
|
|
|
|
parent,
|
|
|
|
|
&stdout_save,
|
|
|
|
|
0,
|
|
|
|
|
FALSE,
|
|
|
|
|
DUPLICATE_SAME_ACCESS);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
DuplicateHandle (parent,
|
|
|
|
|
GetStdHandle (STD_ERROR_HANDLE),
|
|
|
|
|
parent,
|
|
|
|
|
&stderr_save,
|
|
|
|
|
0,
|
|
|
|
|
FALSE,
|
|
|
|
|
DUPLICATE_SAME_ACCESS);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
fclose (stdin);
|
|
|
|
|
fclose (stdout);
|
|
|
|
|
fclose (stderr);
|
|
|
|
|
|
|
|
|
|
if (stdin_save != INVALID_HANDLE_VALUE)
|
|
|
|
|
_open_osfhandle ((long) stdin_save, O_TEXT);
|
|
|
|
|
else
|
1997-09-03 00:43:20 +00:00
|
|
|
|
_open ("nul", O_TEXT | O_NOINHERIT | O_RDONLY);
|
|
|
|
|
_fdopen (0, "r");
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
if (stdout_save != INVALID_HANDLE_VALUE)
|
|
|
|
|
_open_osfhandle ((long) stdout_save, O_TEXT);
|
|
|
|
|
else
|
1997-09-03 00:43:20 +00:00
|
|
|
|
_open ("nul", O_TEXT | O_NOINHERIT | O_WRONLY);
|
|
|
|
|
_fdopen (1, "w");
|
1996-05-03 18:32:52 +00:00
|
|
|
|
|
|
|
|
|
if (stderr_save != INVALID_HANDLE_VALUE)
|
|
|
|
|
_open_osfhandle ((long) stderr_save, O_TEXT);
|
|
|
|
|
else
|
1997-09-03 00:43:20 +00:00
|
|
|
|
_open ("nul", O_TEXT | O_NOINHERIT | O_WRONLY);
|
|
|
|
|
_fdopen (2, "w");
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* unfortunately, atexit depends on implementation of malloc */
|
|
|
|
|
/* atexit (term_ntproc); */
|
|
|
|
|
signal (SIGABRT, term_ntproc);
|
1997-09-03 00:43:20 +00:00
|
|
|
|
|
|
|
|
|
/* determine which drives are fixed, for GetCachedVolumeInformation */
|
|
|
|
|
{
|
|
|
|
|
/* GetDriveType must have trailing backslash. */
|
|
|
|
|
char drive[] = "A:\\";
|
|
|
|
|
|
|
|
|
|
/* Loop over all possible drive letters */
|
|
|
|
|
while (*drive <= 'Z')
|
|
|
|
|
{
|
|
|
|
|
/* Record if this drive letter refers to a fixed drive. */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
fixed_drives[DRIVE_INDEX (*drive)] =
|
1997-09-03 00:43:20 +00:00
|
|
|
|
(GetDriveType (drive) == DRIVE_FIXED);
|
|
|
|
|
|
|
|
|
|
(*drive)++;
|
|
|
|
|
}
|
2000-08-22 22:49:27 +00:00
|
|
|
|
|
|
|
|
|
/* Reset the volume info cache. */
|
|
|
|
|
volume_cache = NULL;
|
1997-09-03 00:43:20 +00:00
|
|
|
|
}
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1998-10-07 19:47:48 +00:00
|
|
|
|
/* Check to see if Emacs has been installed correctly. */
|
|
|
|
|
check_windows_init_file ();
|
1996-05-03 18:32:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2002-12-18 06:16:28 +00:00
|
|
|
|
/*
|
|
|
|
|
globals_of_w32 is used to initialize those global variables that
|
|
|
|
|
must always be initialized on startup even when the global variable
|
|
|
|
|
initialized is non zero (see the function main in emacs.c).
|
|
|
|
|
*/
|
|
|
|
|
void globals_of_w32 ()
|
|
|
|
|
{
|
|
|
|
|
g_b_init_is_windows_9x = 0;
|
|
|
|
|
g_b_init_open_process_token = 0;
|
|
|
|
|
g_b_init_get_token_information = 0;
|
|
|
|
|
g_b_init_lookup_account_sid = 0;
|
|
|
|
|
g_b_init_get_sid_identifier_authority = 0;
|
|
|
|
|
}
|
|
|
|
|
|
1996-05-03 18:32:52 +00:00
|
|
|
|
/* end of nt.c */
|
2003-09-01 15:45:59 +00:00
|
|
|
|
|
|
|
|
|
/* arch-tag: 90442dd3-37be-482b-b272-ac752e3049f1
|
|
|
|
|
(do not change this comment) */
|