1 /* Utility and Unix shadow routines for GNU Emacs support programs on NT.
2 Copyright (C) 1994, 2001, 2002, 2003, 2004, 2005, 2006, 2007,
3 2008, 2009 Free Software Foundation, Inc.
5 This file is part of GNU Emacs.
7 GNU Emacs is free software: you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation, either version 3 of the License, or
10 (at your option) any later version.
12 GNU Emacs is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
21 Geoff Voelker (voelker@cs.washington.edu) 10-8-94
29 #include <sys/types.h>
35 #define MAXPATHLEN _MAX_PATH
37 /* Emulate sleep...we could have done this with a define, but that
38 would necessitate including windows.h in the files that used it.
39 This is much easier. */
41 sleep(unsigned long seconds
)
43 Sleep (seconds
* 1000);
46 /* Get the current working directory. */
50 if (GetCurrentDirectory (MAXPATHLEN
, dir
) > 0)
55 static HANDLE getppid_parent
;
56 static int getppid_ppid
;
64 ppid
= getenv ("EM_PARENT_PROCESS_ID");
72 getppid_ppid
= atoi (ppid
);
77 getppid_parent
= OpenProcess (SYNCHRONIZE
, FALSE
, atoi(ppid
));
80 printf ("Failed to open handle to parent process: %d\n",
86 result
= WaitForSingleObject (getppid_parent
, 0);
90 /* The parent is still alive. */
93 /* The parent is gone. Return the pid of Unix init (1). */
97 printf ("Checking parent status failed: %d\n", GetLastError());
105 static char user_name
[256];
106 DWORD length
= sizeof (user_name
);
108 if (GetUserName (user_name
, &length
))
116 char * name
= getlogin ();
118 return strcpy (s
, name
? name
: "");
129 setuid (unsigned uid
)
135 getpwuid (unsigned uid
)
141 getpass (const char * prompt
)
143 static char input
[256];
148 in
= GetStdHandle (STD_INPUT_HANDLE
);
149 err
= GetStdHandle (STD_ERROR_HANDLE
);
151 if (in
== INVALID_HANDLE_VALUE
|| err
== INVALID_HANDLE_VALUE
)
154 if (WriteFile (err
, prompt
, strlen (prompt
), &count
, NULL
))
156 int istty
= (GetFileType (in
) == FILE_TYPE_CHAR
);
162 if (GetConsoleMode (in
, &old_flags
))
163 SetConsoleMode (in
, ENABLE_LINE_INPUT
| ENABLE_PROCESSED_INPUT
);
167 rc
= ReadFile (in
, input
, sizeof (input
), &count
, NULL
);
168 if (count
>= 2 && input
[count
- 2] == '\r')
169 input
[count
- 2] = '\0';
173 while (ReadFile (in
, buf
, sizeof (buf
), &count
, NULL
) > 0)
174 if (count
>= 2 && buf
[count
- 2] == '\r')
177 WriteFile (err
, "\r\n", 2, &count
, NULL
);
179 SetConsoleMode (in
, old_flags
);
188 fchown (int fd
, unsigned uid
, unsigned gid
)
193 /* Place a wrapper around the MSVC version of ctime. It returns NULL
194 on network directories, so we handle that case here.
195 (Ulrich Leodolter, 1/11/95). */
197 sys_ctime (const time_t *t
)
199 char *str
= (char *) ctime (t
);
200 return (str
? str
: "Sun Jan 01 00:00:00 1970");
204 sys_fopen(const char * path
, const char * mode
)
206 return fopen (path
, mode
);
210 sys_chdir (const char * path
)
212 return _chdir (path
);
215 static FILETIME utc_base_ft
;
216 static long double utc_base
;
220 convert_time (FILETIME ft
)
224 if (CompareFileTime (&ft
, &utc_base_ft
) < 0)
227 ret
= (long double) ft
.dwHighDateTime
228 * 4096.0L * 1024.0L * 1024.0L + ft
.dwLowDateTime
;
230 return (time_t) (ret
* 1e-7L);
234 is_exec (const char * name
)
236 char * p
= strrchr (name
, '.');
239 && (stricmp (p
, ".exe") == 0 ||
240 stricmp (p
, ".com") == 0 ||
241 stricmp (p
, ".bat") == 0 ||
242 stricmp (p
, ".cmd") == 0));
245 #define IS_DIRECTORY_SEP(x) ((x) == '/' || (x) == '\\')
247 /* We need this because nt/inc/sys/stat.h defines struct stat that is
248 incompatible with the MS run-time libraries. */
250 stat (const char * path
, struct stat
* buf
)
257 char *name
= alloca (FILENAME_MAX
);
261 /* Determine the delta between 1-Jan-1601 and 1-Jan-1970. */
270 st
.wMilliseconds
= 0;
272 SystemTimeToFileTime (&st
, &utc_base_ft
);
273 utc_base
= (long double) utc_base_ft
.dwHighDateTime
274 * 4096.0L * 1024.0L * 1024.0L + utc_base_ft
.dwLowDateTime
;
278 if (path
== NULL
|| buf
== NULL
|| *path
== '\0')
283 if (_mbspbrk (path
, "*?|<>\""))
290 /* Remove trailing directory separator, unless name is the root
291 directory of a drive in which case ensure there is a trailing
294 rootdir
= IS_DIRECTORY_SEP (name
[0])
295 || (len
== 3 && name
[1] == ':' && IS_DIRECTORY_SEP (name
[2]));
298 if (GetDriveType (name
) < 2)
303 memset (&wfd
, 0, sizeof (wfd
));
304 wfd
.dwFileAttributes
= FILE_ATTRIBUTE_DIRECTORY
;
305 wfd
.ftCreationTime
= utc_base_ft
;
306 wfd
.ftLastAccessTime
= utc_base_ft
;
307 wfd
.ftLastWriteTime
= utc_base_ft
;
308 strcpy (wfd
.cFileName
, name
);
312 if (IS_DIRECTORY_SEP (name
[len
-1]))
315 fh
= FindFirstFile (name
, &wfd
);
316 if (fh
== INVALID_HANDLE_VALUE
)
323 buf
->st_mode
= (wfd
.dwFileAttributes
& FILE_ATTRIBUTE_DIRECTORY
) ?
328 if (name
[0] && name
[1] == ':')
329 buf
->st_dev
= tolower (name
[0]) - 'a' + 1;
331 buf
->st_dev
= _getdrive ();
332 buf
->st_rdev
= buf
->st_dev
;
334 buf
->st_size
= wfd
.nFileSizeLow
;
336 /* Convert timestamps to Unix format. */
337 buf
->st_mtime
= convert_time (wfd
.ftLastWriteTime
);
338 buf
->st_atime
= convert_time (wfd
.ftLastAccessTime
);
339 if (buf
->st_atime
== 0) buf
->st_atime
= buf
->st_mtime
;
340 buf
->st_ctime
= convert_time (wfd
.ftCreationTime
);
341 if (buf
->st_ctime
== 0) buf
->st_ctime
= buf
->st_mtime
;
343 /* determine rwx permissions */
344 if (wfd
.dwFileAttributes
& FILE_ATTRIBUTE_READONLY
)
345 permission
= S_IREAD
;
347 permission
= S_IREAD
| S_IWRITE
;
349 if (wfd
.dwFileAttributes
& FILE_ATTRIBUTE_DIRECTORY
)
350 permission
|= S_IEXEC
;
351 else if (is_exec (name
))
352 permission
|= S_IEXEC
;
354 buf
->st_mode
|= permission
| (permission
>> 3) | (permission
>> 6);
359 /* arch-tag: 7b63fb83-70ee-4124-8822-54e53e5d0773
360 (do not change this comment) */