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
27 #define WIN32_NO_STATUS
33 #include "wine/debug.h"
35 WINE_DEFAULT_DEBUG_CHANNEL(msvcrt
);
37 /* error strings generated with glibc strerror */
38 static char str_success
[] = "Success";
39 static char str_EPERM
[] = "Operation not permitted";
40 static char str_ENOENT
[] = "No such file or directory";
41 static char str_ESRCH
[] = "No such process";
42 static char str_EINTR
[] = "Interrupted system call";
43 static char str_EIO
[] = "Input/output error";
44 static char str_ENXIO
[] = "No such device or address";
45 static char str_E2BIG
[] = "Argument list too long";
46 static char str_ENOEXEC
[] = "Exec format error";
47 static char str_EBADF
[] = "Bad file descriptor";
48 static char str_ECHILD
[] = "No child processes";
49 static char str_EAGAIN
[] = "Resource temporarily unavailable";
50 static char str_ENOMEM
[] = "Cannot allocate memory";
51 static char str_EACCES
[] = "Permission denied";
52 static char str_EFAULT
[] = "Bad address";
53 static char str_EBUSY
[] = "Device or resource busy";
54 static char str_EEXIST
[] = "File exists";
55 static char str_EXDEV
[] = "Invalid cross-device link";
56 static char str_ENODEV
[] = "No such device";
57 static char str_ENOTDIR
[] = "Not a directory";
58 static char str_EISDIR
[] = "Is a directory";
59 static char str_EINVAL
[] = "Invalid argument";
60 static char str_ENFILE
[] = "Too many open files in system";
61 static char str_EMFILE
[] = "Too many open files";
62 static char str_ENOTTY
[] = "Inappropriate ioctl for device";
63 static char str_EFBIG
[] = "File too large";
64 static char str_ENOSPC
[] = "No space left on device";
65 static char str_ESPIPE
[] = "Illegal seek";
66 static char str_EROFS
[] = "Read-only file system";
67 static char str_EMLINK
[] = "Too many links";
68 static char str_EPIPE
[] = "Broken pipe";
69 static char str_EDOM
[] = "Numerical argument out of domain";
70 static char str_ERANGE
[] = "Numerical result out of range";
71 static char str_EDEADLK
[] = "Resource deadlock avoided";
72 static char str_ENAMETOOLONG
[] = "File name too long";
73 static char str_ENOLCK
[] = "No locks available";
74 static char str_ENOSYS
[] = "Function not implemented";
75 static char str_ENOTEMPTY
[] = "Directory not empty";
76 static char str_EILSEQ
[] = "Invalid or incomplete multibyte or wide character";
77 static char str_generic_error
[] = "Unknown error";
79 char *MSVCRT__sys_errlist
[] =
127 unsigned int MSVCRT__sys_nerr
= ARRAY_SIZE(MSVCRT__sys_errlist
) - 1;
129 static _invalid_parameter_handler invalid_parameter_handler
= NULL
;
131 /* INTERNAL: Set the crt and dos errno's from the OS error given. */
132 void msvcrt_set_errno(int err
)
134 int *errno_ptr
= _errno();
135 __msvcrt_ulong
*doserrno
= __doserrno();
141 #define ERR_CASE(oserr) case oserr:
142 #define ERR_MAPS(oserr, crterr) case oserr: *errno_ptr = crterr; break
143 ERR_CASE(ERROR_ACCESS_DENIED
)
144 ERR_CASE(ERROR_NETWORK_ACCESS_DENIED
)
145 ERR_CASE(ERROR_CANNOT_MAKE
)
146 ERR_CASE(ERROR_SEEK_ON_DEVICE
)
147 ERR_CASE(ERROR_LOCK_FAILED
)
148 ERR_CASE(ERROR_FAIL_I24
)
149 ERR_CASE(ERROR_CURRENT_DIRECTORY
)
150 ERR_CASE(ERROR_DRIVE_LOCKED
)
151 ERR_CASE(ERROR_NOT_LOCKED
)
152 ERR_CASE(ERROR_INVALID_ACCESS
)
153 ERR_CASE(ERROR_SHARING_VIOLATION
)
154 ERR_MAPS(ERROR_LOCK_VIOLATION
, EACCES
);
155 ERR_CASE(ERROR_FILE_NOT_FOUND
)
156 ERR_CASE(ERROR_NO_MORE_FILES
)
157 ERR_CASE(ERROR_BAD_PATHNAME
)
158 ERR_CASE(ERROR_BAD_NETPATH
)
159 ERR_CASE(ERROR_INVALID_DRIVE
)
160 ERR_CASE(ERROR_BAD_NET_NAME
)
161 ERR_CASE(ERROR_FILENAME_EXCED_RANGE
)
162 ERR_MAPS(ERROR_PATH_NOT_FOUND
, ENOENT
);
163 ERR_MAPS(ERROR_IO_DEVICE
, EIO
);
164 ERR_MAPS(ERROR_BAD_FORMAT
, ENOEXEC
);
165 ERR_MAPS(ERROR_INVALID_HANDLE
, EBADF
);
166 ERR_CASE(ERROR_OUTOFMEMORY
)
167 ERR_CASE(ERROR_INVALID_BLOCK
)
168 ERR_CASE(ERROR_NOT_ENOUGH_QUOTA
)
169 ERR_MAPS(ERROR_ARENA_TRASHED
, ENOMEM
);
170 ERR_MAPS(ERROR_BUSY
, EBUSY
);
171 ERR_CASE(ERROR_ALREADY_EXISTS
)
172 ERR_MAPS(ERROR_FILE_EXISTS
, EEXIST
);
173 ERR_MAPS(ERROR_BAD_DEVICE
, ENODEV
);
174 ERR_MAPS(ERROR_TOO_MANY_OPEN_FILES
, EMFILE
);
175 ERR_MAPS(ERROR_DISK_FULL
, ENOSPC
);
176 ERR_MAPS(ERROR_BROKEN_PIPE
, EPIPE
);
177 ERR_MAPS(ERROR_POSSIBLE_DEADLOCK
, EDEADLK
);
178 ERR_MAPS(ERROR_DIR_NOT_EMPTY
, ENOTEMPTY
);
179 ERR_MAPS(ERROR_BAD_ENVIRONMENT
, E2BIG
);
180 ERR_CASE(ERROR_WAIT_NO_CHILDREN
)
181 ERR_MAPS(ERROR_CHILD_NOT_COMPLETE
, ECHILD
);
182 ERR_CASE(ERROR_NO_PROC_SLOTS
)
183 ERR_CASE(ERROR_MAX_THRDS_REACHED
)
184 ERR_MAPS(ERROR_NESTING_NOT_ALLOWED
, EAGAIN
);
186 /* Remaining cases map to EINVAL */
187 /* FIXME: may be missing some errors above */
194 /*********************************************************************
195 * __sys_nerr (MSVCR80.@)
197 int* CDECL
__sys_nerr(void)
199 return (int*)&MSVCRT__sys_nerr
;
202 /*********************************************************************
203 * __sys_errlist (MSVCR80.@)
205 char** CDECL
__sys_errlist(void)
207 return MSVCRT__sys_errlist
;
210 #endif /* _MSVCR_VER >= 80 */
212 /*********************************************************************
215 int* CDECL
_errno(void)
217 return &msvcrt_get_thread_data()->thread_errno
;
220 /*********************************************************************
221 * __doserrno (MSVCRT.@)
223 __msvcrt_ulong
* CDECL
__doserrno(void)
225 return &msvcrt_get_thread_data()->thread_doserrno
;
228 /*********************************************************************
229 * _get_errno (MSVCRT.@)
231 int CDECL
_get_errno(int *pValue
)
240 /*********************************************************************
241 * _get_doserrno (MSVCRT.@)
243 int CDECL
_get_doserrno(int *pValue
)
248 *pValue
= *__doserrno();
252 /*********************************************************************
253 * _set_errno (MSVCRT.@)
255 int CDECL
_set_errno(int value
)
261 /*********************************************************************
262 * _set_doserrno (MSVCRT.@)
264 int CDECL
_set_doserrno(int value
)
266 *__doserrno() = value
;
270 /*********************************************************************
271 * strerror (MSVCRT.@)
273 char* CDECL
strerror(int err
)
275 thread_data_t
*data
= msvcrt_get_thread_data();
277 if (!data
->strerror_buffer
)
278 if (!(data
->strerror_buffer
= malloc(256))) return NULL
;
280 if (err
< 0 || err
> MSVCRT__sys_nerr
) err
= MSVCRT__sys_nerr
;
281 strcpy( data
->strerror_buffer
, MSVCRT__sys_errlist
[err
] );
282 return data
->strerror_buffer
;
285 /**********************************************************************
286 * strerror_s (MSVCRT.@)
288 int CDECL
strerror_s(char *buffer
, size_t numberOfElements
, int errnum
)
292 if (!buffer
|| !numberOfElements
)
298 if (errnum
< 0 || errnum
> MSVCRT__sys_nerr
)
299 errnum
= MSVCRT__sys_nerr
;
301 ptr
= MSVCRT__sys_errlist
[errnum
];
302 while (*ptr
&& numberOfElements
> 1)
312 /**********************************************************************
313 * _strerror (MSVCRT.@)
315 char* CDECL
_strerror(const char* str
)
317 thread_data_t
*data
= msvcrt_get_thread_data();
320 if (!data
->strerror_buffer
)
321 if (!(data
->strerror_buffer
= malloc(256))) return NULL
;
323 err
= data
->thread_errno
;
324 if (err
< 0 || err
> MSVCRT__sys_nerr
) err
= MSVCRT__sys_nerr
;
327 sprintf( data
->strerror_buffer
, "%s: %s\n", str
, MSVCRT__sys_errlist
[err
] );
329 sprintf( data
->strerror_buffer
, "%s\n", MSVCRT__sys_errlist
[err
] );
331 return data
->strerror_buffer
;
334 /*********************************************************************
337 void CDECL
perror(const char* str
)
340 if (err
< 0 || err
> MSVCRT__sys_nerr
) err
= MSVCRT__sys_nerr
;
344 _write( 2, str
, strlen(str
) );
345 _write( 2, ": ", 2 );
347 _write( 2, MSVCRT__sys_errlist
[err
], strlen(MSVCRT__sys_errlist
[err
]) );
348 _write( 2, "\n", 1 );
351 /*********************************************************************
352 * _wperror (MSVCRT.@)
354 void CDECL
_wperror(const wchar_t* str
)
361 size
= wcstombs(NULL
, str
, 0);
362 if (size
== -1) return;
364 buffer
= malloc(size
);
366 if (wcstombs(buffer
, str
, size
) == -1)
376 /*********************************************************************
377 * _wcserror_s (MSVCRT.@)
379 int CDECL
_wcserror_s(wchar_t* buffer
, size_t nc
, int err
)
381 if (!MSVCRT_CHECK_PMT(buffer
!= NULL
)) return EINVAL
;
382 if (!MSVCRT_CHECK_PMT(nc
> 0)) return EINVAL
;
384 if (err
< 0 || err
> MSVCRT__sys_nerr
) err
= MSVCRT__sys_nerr
;
385 MultiByteToWideChar(CP_ACP
, 0, MSVCRT__sys_errlist
[err
], -1, buffer
, nc
);
389 /*********************************************************************
390 * _wcserror (MSVCRT.@)
392 wchar_t* CDECL
_wcserror(int err
)
394 thread_data_t
*data
= msvcrt_get_thread_data();
396 if (!data
->wcserror_buffer
)
397 if (!(data
->wcserror_buffer
= malloc(256 * sizeof(wchar_t)))) return NULL
;
398 _wcserror_s(data
->wcserror_buffer
, 256, err
);
399 return data
->wcserror_buffer
;
402 /**********************************************************************
403 * __wcserror_s (MSVCRT.@)
405 int CDECL
__wcserror_s(wchar_t* buffer
, size_t nc
, const wchar_t* str
)
411 if (err
< 0 || err
> MSVCRT__sys_nerr
) err
= MSVCRT__sys_nerr
;
413 len
= MultiByteToWideChar(CP_ACP
, 0, MSVCRT__sys_errlist
[err
], -1, NULL
, 0) + 1 /* \n */;
414 if (str
&& *str
) len
+= wcslen(str
) + 2 /* ': ' */;
417 MSVCRT_INVALID_PMT("buffer[nc] is too small", ERANGE
);
422 lstrcpyW(buffer
, str
);
423 lstrcatW(buffer
, L
": ");
425 else buffer
[0] = '\0';
426 len
= wcslen(buffer
);
427 MultiByteToWideChar(CP_ACP
, 0, MSVCRT__sys_errlist
[err
], -1, buffer
+ len
, 256 - len
);
428 lstrcatW(buffer
, L
"\n");
433 /**********************************************************************
434 * __wcserror (MSVCRT.@)
436 wchar_t* CDECL
__wcserror(const wchar_t* str
)
438 thread_data_t
*data
= msvcrt_get_thread_data();
441 if (!data
->wcserror_buffer
)
442 if (!(data
->wcserror_buffer
= malloc(256 * sizeof(wchar_t)))) return NULL
;
444 err
= __wcserror_s(data
->wcserror_buffer
, 256, str
);
445 if (err
) FIXME("bad wcserror call (%d)\n", err
);
447 return data
->wcserror_buffer
;
450 /******************************************************************************
451 * _seterrormode (MSVCRT.@)
453 void CDECL
_seterrormode(int mode
)
455 SetErrorMode( mode
);
458 /******************************************************************************
459 * _invalid_parameter (MSVCRT.@)
461 void __cdecl
_invalid_parameter(const wchar_t *expr
, const wchar_t *func
,
462 const wchar_t *file
, unsigned int line
, uintptr_t arg
)
464 #if _MSVCR_VER >= 140
465 thread_data_t
*data
= msvcrt_get_thread_data();
467 if (data
->invalid_parameter_handler
)
469 data
->invalid_parameter_handler( expr
, func
, file
, line
, arg
);
474 if (invalid_parameter_handler
) invalid_parameter_handler( expr
, func
, file
, line
, arg
);
477 ERR( "%s:%u %s: %s %Ix\n", debugstr_w(file
), line
, debugstr_w(func
), debugstr_w(expr
), arg
);
479 RaiseException( STATUS_INVALID_CRUNTIME_PARAMETER
, EXCEPTION_NONCONTINUABLE
, 0, NULL
);
486 /*********************************************************************
487 * _invalid_parameter_noinfo (MSVCR80.@)
489 void CDECL
_invalid_parameter_noinfo(void)
491 _invalid_parameter( NULL
, NULL
, NULL
, 0, 0 );
494 /*********************************************************************
495 * _invalid_parameter_noinfo_noreturn (MSVCR80.@)
497 void CDECL
_invalid_parameter_noinfo_noreturn(void)
499 _invalid_parameter( NULL
, NULL
, NULL
, 0, 0 );
500 _exit( STATUS_INVALID_CRUNTIME_PARAMETER
);
503 /*********************************************************************
504 * _get_invalid_parameter_handler (MSVCR80.@)
506 _invalid_parameter_handler CDECL
_get_invalid_parameter_handler(void)
509 return invalid_parameter_handler
;
512 /*********************************************************************
513 * _set_invalid_parameter_handler (MSVCR80.@)
515 _invalid_parameter_handler CDECL
_set_invalid_parameter_handler(
516 _invalid_parameter_handler handler
)
518 _invalid_parameter_handler old
= invalid_parameter_handler
;
520 TRACE("(%p)\n", handler
);
522 invalid_parameter_handler
= handler
;
526 #endif /* _MSVCR_VER >= 80 */
528 #if _MSVCR_VER >= 140
530 /*********************************************************************
531 * _get_thread_local_invalid_parameter_handler (UCRTBASE.@)
533 _invalid_parameter_handler CDECL
_get_thread_local_invalid_parameter_handler(void)
536 return msvcrt_get_thread_data()->invalid_parameter_handler
;
539 /*********************************************************************
540 * _set_thread_local_invalid_parameter_handler (UCRTBASE.@)
542 _invalid_parameter_handler CDECL
_set_thread_local_invalid_parameter_handler(
543 _invalid_parameter_handler handler
)
545 thread_data_t
*data
= msvcrt_get_thread_data();
546 _invalid_parameter_handler old
= data
->invalid_parameter_handler
;
548 TRACE("(%p)\n", handler
);
550 data
->invalid_parameter_handler
= handler
;
554 #endif /* _MSVCR_VER >= 140 */