2 * msvcrt.dll errno functions
4 * Copyright 2000 Jon Griffiths
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
25 #include "wine/debug.h"
27 WINE_DEFAULT_DEBUG_CHANNEL(msvcrt);
29 /* error strings generated with glibc strerror */
30 static char str_success[] = "Success";
31 static char str_EPERM[] = "Operation not permitted";
32 static char str_ENOENT[] = "No such file or directory";
33 static char str_ESRCH[] = "No such process";
34 static char str_EINTR[] = "Interrupted system call";
35 static char str_EIO[] = "Input/output error";
36 static char str_ENXIO[] = "No such device or address";
37 static char str_E2BIG[] = "Argument list too long";
38 static char str_ENOEXEC[] = "Exec format error";
39 static char str_EBADF[] = "Bad file descriptor";
40 static char str_ECHILD[] = "No child processes";
41 static char str_EAGAIN[] = "Resource temporarily unavailable";
42 static char str_ENOMEM[] = "Cannot allocate memory";
43 static char str_EACCES[] = "Permission denied";
44 static char str_EFAULT[] = "Bad address";
45 static char str_EBUSY[] = "Device or resource busy";
46 static char str_EEXIST[] = "File exists";
47 static char str_EXDEV[] = "Invalid cross-device link";
48 static char str_ENODEV[] = "No such device";
49 static char str_ENOTDIR[] = "Not a directory";
50 static char str_EISDIR[] = "Is a directory";
51 static char str_EINVAL[] = "Invalid argument";
52 static char str_ENFILE[] = "Too many open files in system";
53 static char str_EMFILE[] = "Too many open files";
54 static char str_ENOTTY[] = "Inappropriate ioctl for device";
55 static char str_EFBIG[] = "File too large";
56 static char str_ENOSPC[] = "No space left on device";
57 static char str_ESPIPE[] = "Illegal seek";
58 static char str_EROFS[] = "Read-only file system";
59 static char str_EMLINK[] = "Too many links";
60 static char str_EPIPE[] = "Broken pipe";
61 static char str_EDOM[] = "Numerical argument out of domain";
62 static char str_ERANGE[] = "Numerical result out of range";
63 static char str_EDEADLK[] = "Resource deadlock avoided";
64 static char str_ENAMETOOLONG[] = "File name too long";
65 static char str_ENOLCK[] = "No locks available";
66 static char str_ENOSYS[] = "Function not implemented";
67 static char str_ENOTEMPTY[] = "Directory not empty";
68 static char str_EILSEQ[] = "Invalid or incomplete multibyte or wide character";
69 static char str_generic_error[] = "Unknown error";
71 char *MSVCRT__sys_errlist[] =
119 unsigned int MSVCRT__sys_nerr = sizeof(MSVCRT__sys_errlist)/sizeof(MSVCRT__sys_errlist[0]) - 1;
120 MSVCRT_invalid_parameter_handler MSVCRT_invalid_parameter = NULL;
122 /* INTERNAL: Set the crt and dos errno's from the OS error given. */
123 void msvcrt_set_errno(int err)
125 int *errno = MSVCRT__errno();
126 MSVCRT_ulong *doserrno = MSVCRT___doserrno();
132 #define ERR_CASE(oserr) case oserr:
133 #define ERR_MAPS(oserr, crterr) case oserr: *errno = crterr; break
134 ERR_CASE(ERROR_ACCESS_DENIED)
135 ERR_CASE(ERROR_NETWORK_ACCESS_DENIED)
136 ERR_CASE(ERROR_CANNOT_MAKE)
137 ERR_CASE(ERROR_SEEK_ON_DEVICE)
138 ERR_CASE(ERROR_LOCK_FAILED)
139 ERR_CASE(ERROR_FAIL_I24)
140 ERR_CASE(ERROR_CURRENT_DIRECTORY)
141 ERR_CASE(ERROR_DRIVE_LOCKED)
142 ERR_CASE(ERROR_NOT_LOCKED)
143 ERR_CASE(ERROR_INVALID_ACCESS)
144 ERR_CASE(ERROR_SHARING_VIOLATION)
145 ERR_MAPS(ERROR_LOCK_VIOLATION, MSVCRT_EACCES);
146 ERR_CASE(ERROR_FILE_NOT_FOUND)
147 ERR_CASE(ERROR_NO_MORE_FILES)
148 ERR_CASE(ERROR_BAD_PATHNAME)
149 ERR_CASE(ERROR_BAD_NETPATH)
150 ERR_CASE(ERROR_INVALID_DRIVE)
151 ERR_CASE(ERROR_BAD_NET_NAME)
152 ERR_CASE(ERROR_FILENAME_EXCED_RANGE)
153 ERR_MAPS(ERROR_PATH_NOT_FOUND, MSVCRT_ENOENT);
154 ERR_MAPS(ERROR_IO_DEVICE, MSVCRT_EIO);
155 ERR_MAPS(ERROR_BAD_FORMAT, MSVCRT_ENOEXEC);
156 ERR_MAPS(ERROR_INVALID_HANDLE, MSVCRT_EBADF);
157 ERR_CASE(ERROR_OUTOFMEMORY)
158 ERR_CASE(ERROR_INVALID_BLOCK)
159 ERR_CASE(ERROR_NOT_ENOUGH_QUOTA);
160 ERR_MAPS(ERROR_ARENA_TRASHED, MSVCRT_ENOMEM);
161 ERR_MAPS(ERROR_BUSY, MSVCRT_EBUSY);
162 ERR_CASE(ERROR_ALREADY_EXISTS)
163 ERR_MAPS(ERROR_FILE_EXISTS, MSVCRT_EEXIST);
164 ERR_MAPS(ERROR_BAD_DEVICE, MSVCRT_ENODEV);
165 ERR_MAPS(ERROR_TOO_MANY_OPEN_FILES, MSVCRT_EMFILE);
166 ERR_MAPS(ERROR_DISK_FULL, MSVCRT_ENOSPC);
167 ERR_MAPS(ERROR_BROKEN_PIPE, MSVCRT_EPIPE);
168 ERR_MAPS(ERROR_POSSIBLE_DEADLOCK, MSVCRT_EDEADLK);
169 ERR_MAPS(ERROR_DIR_NOT_EMPTY, MSVCRT_ENOTEMPTY);
170 ERR_MAPS(ERROR_BAD_ENVIRONMENT, MSVCRT_E2BIG);
171 ERR_CASE(ERROR_WAIT_NO_CHILDREN)
172 ERR_MAPS(ERROR_CHILD_NOT_COMPLETE, MSVCRT_ECHILD);
173 ERR_CASE(ERROR_NO_PROC_SLOTS)
174 ERR_CASE(ERROR_MAX_THRDS_REACHED)
175 ERR_MAPS(ERROR_NESTING_NOT_ALLOWED, MSVCRT_EAGAIN);
177 /* Remaining cases map to EINVAL */
178 /* FIXME: may be missing some errors above */
179 *errno = MSVCRT_EINVAL;
183 /*********************************************************************
186 int* CDECL MSVCRT__errno(void)
188 return &msvcrt_get_thread_data()->thread_errno;
191 /*********************************************************************
192 * __doserrno (MSVCRT.@)
194 MSVCRT_ulong* CDECL MSVCRT___doserrno(void)
196 return &msvcrt_get_thread_data()->thread_doserrno;
199 /*********************************************************************
200 * strerror (MSVCRT.@)
202 char* CDECL MSVCRT_strerror(int err)
204 thread_data_t *data = msvcrt_get_thread_data();
206 if (!data->strerror_buffer)
207 if (!(data->strerror_buffer = MSVCRT_malloc(256))) return NULL;
209 if (err < 0 || err > MSVCRT__sys_nerr) err = MSVCRT__sys_nerr;
210 strcpy( data->strerror_buffer, MSVCRT__sys_errlist[err] );
211 return data->strerror_buffer;
214 /**********************************************************************
215 * _strerror (MSVCRT.@)
217 char* CDECL _strerror(const char* str)
219 thread_data_t *data = msvcrt_get_thread_data();
222 if (!data->strerror_buffer)
223 if (!(data->strerror_buffer = MSVCRT_malloc(256))) return NULL;
225 err = data->thread_errno;
226 if (err < 0 || err > MSVCRT__sys_nerr) err = MSVCRT__sys_nerr;
229 sprintf( data->strerror_buffer, "%s: %s\n", str, MSVCRT__sys_errlist[err] );
231 sprintf( data->strerror_buffer, "%s\n", MSVCRT__sys_errlist[err] );
233 return data->strerror_buffer;
236 /*********************************************************************
239 void CDECL MSVCRT_perror(const char* str)
241 int err = *MSVCRT__errno();
242 if (err < 0 || err > MSVCRT__sys_nerr) err = MSVCRT__sys_nerr;
246 MSVCRT__write( 2, str, strlen(str) );
247 MSVCRT__write( 2, ": ", 2 );
249 MSVCRT__write( 2, MSVCRT__sys_errlist[err], strlen(MSVCRT__sys_errlist[err]) );
250 MSVCRT__write( 2, "\n", 1 );
253 /******************************************************************************
254 * _set_error_mode (MSVCRT.@)
256 * Set the error mode, which describes where the C run-time writes error
260 * mode - the new error mode
263 * The old error mode.
266 * This function does not have a proper implementation; the error mode is
269 int CDECL _set_error_mode(int mode)
271 static int current_mode = MSVCRT__OUT_TO_DEFAULT;
273 const int old = current_mode;
274 if ( MSVCRT__REPORT_ERRMODE != mode ) {
276 FIXME("dummy implementation (old mode: %d, new mode: %d)\n",
282 /******************************************************************************
283 * _seterrormode (MSVCRT.@)
285 void CDECL _seterrormode(int mode)
287 SetErrorMode( mode );
290 /* _get_invalid_parameter_handler - not exported in native msvcrt, added in msvcr80 */
291 MSVCRT_invalid_parameter_handler CDECL _get_invalid_parameter_handler(void)
294 return MSVCRT_invalid_parameter;
297 /* _set_invalid_parameter_handler - not exproted in native msvcrt, added in msvcr80 */
298 MSVCRT_invalid_parameter_handler CDECL _set_invalid_parameter_handler(
299 MSVCRT_invalid_parameter_handler handler)
301 MSVCRT_invalid_parameter_handler old = MSVCRT_invalid_parameter;
303 TRACE("(%p)\n", handler);
305 MSVCRT_invalid_parameter = handler;