เมื่อโทรผมได้รับexecl(...)
errno=2
มันหมายความว่าอะไร? ฉันจะรู้ความหมายของสิ่งนี้ได้errno
อย่างไร
เมื่อโทรผมได้รับexecl(...)
errno=2
มันหมายความว่าอะไร? ฉันจะรู้ความหมายของสิ่งนี้ได้errno
อย่างไร
คำตอบ:
คุณสามารถใช้strerror()
เพื่อรับสตริงที่มนุษย์อ่านได้สำหรับหมายเลขข้อผิดพลาด นี่คือสตริงเดียวกันที่พิมพ์โดยperror()
แต่จะมีประโยชน์หากคุณกำลังจัดรูปแบบข้อความแสดงข้อผิดพลาดสำหรับสิ่งอื่นที่ไม่ใช่ข้อผิดพลาดมาตรฐาน
ตัวอย่างเช่น:
#include <errno.h>
#include <string.h>
/* ... */
if(read(fd, buf, 1)==-1) {
printf("Oh dear, something went wrong with read()! %s\n", strerror(errno));
}
strerror_r()
ลินุกซ์ยังสนับสนุนแตกต่างอย่างชัดเจนด้าย
strerror()
ไม่ปลอดภัยเธรดในขณะที่strerror_r()
ปลอดภัยเธรด ฟังก์ชั่น MT-Safe หรือ Thread-Safe นั้นปลอดภัยในการโทรเมื่อมีเธรดอื่นอยู่ MT ใน MT-Safe ย่อมาจาก Multi Thread -p26, GNU C Library char * strerror(int errnum )
[ฟังก์ชั่น] เบื้องต้น: | MT-Unsafe race: strerror | AS-Unsafe heap i18n | AC-Unsafe mem | ดูหัวข้อ 1.2.2.1 [POSIX Safety Concepts], หน้า 2 -p58, GNU C Library
errno
ปลอดภัยต่อเธรด ดู:#define errno *__errno_location()
errno
, GNU libc strerror
เขียนไปยังบัฟเฟอร์ทั่วโลกภายใน มันไม่ได้ทำเช่นนี้เสมอไป แต่ในบางกรณี คุณจะเขียนทับบัฟเฟอร์นั้นในขณะที่เธรดอื่นกำลังอ่านอยู่
errno
strerror()
แทนที่จะเรียกใช้perror
รหัสข้อผิดพลาดที่คุณได้รับคุณสามารถดึงรายการของerrno
ค่าทั้งหมดในระบบของคุณด้วยหนึ่งซับต่อไปนี้:
cpp -dM /usr/include/errno.h | grep 'define E' | sort -n -k 3
บนลีนุกซ์นั้นยังมีเครื่องมือที่เรียบร้อยมากซึ่งสามารถบอกได้ทันทีว่ารหัสข้อผิดพลาดแต่ละตัวหมายถึงอะไร บน apt-get install errno
Ubuntu:
จากนั้นถ้าเช่นคุณต้องการได้รับรายละเอียดของประเภทข้อผิดพลาด 2 เพียงแค่พิมพ์errno 2
ใน terminal
เมื่อerrno -l
คุณได้รับรายการพร้อมข้อผิดพลาดและคำอธิบายทั้งหมด ง่ายกว่าวิธีอื่น ๆ ที่กล่าวถึงโดยผู้โพสต์ก่อนหน้านี้
perror
/ strerror
ดีกว่าเพราะผู้ใช้ไม่ต้องค้นหา แม้ว่าerrno -l
จะดีกว่าสำหรับการค้นหาพวกเขา
นี่คือผลลัพธ์จากการerrno -l
จัดรูปแบบใหม่เพื่อให้สามารถอ่านได้:
1 EPERM Operation not permitted
2 ENOENT No such file or directory
3 ESRCH No such process
4 EINTR Interrupted system call
5 EIO Input/output error
6 ENXIO No such device or address
7 E2BIG Argument list too long
8 ENOEXEC Exec format error
9 EBADF Bad file descriptor
10 ECHILD No child processes
11 EAGAIN Resource temporarily unavailable
11 EWOULDBLOCK Resource temporarily unavailable
12 ENOMEM Cannot allocate memory
13 EACCES Permission denied
14 EFAULT Bad address
15 ENOTBLK Block device required
16 EBUSY Device or resource busy
17 EEXIST File exists
18 EXDEV Invalid cross-device link
19 ENODEV No such device
20 ENOTDIR Not a directory
21 EISDIR Is a directory
22 EINVAL Invalid argument
23 ENFILE Too many open files in system
24 EMFILE Too many open files
25 ENOTTY Inappropriate ioctl for device
26 ETXTBSY Text file busy
27 EFBIG File too large
28 ENOSPC No space left on device
29 ESPIPE Illegal seek
30 EROFS Read-only file system
31 EMLINK Too many links
32 EPIPE Broken pipe
33 EDOM Numerical argument out of domain
34 ERANGE Numerical result out of range
35 EDEADLK Resource deadlock avoided
35 EDEADLOCK Resource deadlock avoided
36 ENAMETOOLONG File name too long
37 ENOLCK No locks available
38 ENOSYS Function not implemented
39 ENOTEMPTY Directory not empty
40 ELOOP Too many levels of symbolic links
42 ENOMSG No message of desired type
43 EIDRM Identifier removed
44 ECHRNG Channel number out of range
45 EL2NSYNC Level 2 not synchronized
46 EL3HLT Level 3 halted
47 EL3RST Level 3 reset
48 ELNRNG Link number out of range
49 EUNATCH Protocol driver not attached
50 ENOCSI No CSI structure available
51 EL2HLT Level 2 halted
52 EBADE Invalid exchange
53 EBADR Invalid request descriptor
54 EXFULL Exchange full
55 ENOANO No anode
56 EBADRQC Invalid request code
57 EBADSLT Invalid slot
59 EBFONT Bad font file format
60 ENOSTR Device not a stream
61 ENODATA No data available
62 ETIME Timer expired
63 ENOSR Out of streams resources
64 ENONET Machine is not on the network
65 ENOPKG Package not installed
66 EREMOTE Object is remote
67 ENOLINK Link has been severed
68 EADV Advertise error
69 ESRMNT Srmount error
70 ECOMM Communication error on send
71 EPROTO Protocol error
72 EMULTIHOP Multihop attempted
73 EDOTDOT RFS specific error
74 EBADMSG Bad message
75 EOVERFLOW Value too large for defined data type
76 ENOTUNIQ Name not unique on network
77 EBADFD File descriptor in bad state
78 EREMCHG Remote address changed
79 ELIBACC Can not access a needed shared library
80 ELIBBAD Accessing a corrupted shared library
81 ELIBSCN .lib section in a.out corrupted
82 ELIBMAX Attempting to link in too many shared libraries
83 ELIBEXEC Cannot exec a shared library directly
84 EILSEQ Invalid or incomplete multibyte or wide character
85 ERESTART Interrupted system call should be restarted
86 ESTRPIPE Streams pipe error
87 EUSERS Too many users
88 ENOTSOCK Socket operation on non-socket
89 EDESTADDRREQ Destination address required
90 EMSGSIZE Message too long
91 EPROTOTYPE Protocol wrong type for socket
92 ENOPROTOOPT Protocol not available
93 EPROTONOSUPPORT Protocol not supported
94 ESOCKTNOSUPPORT Socket type not supported
95 ENOTSUP Operation not supported
95 EOPNOTSUPP Operation not supported
96 EPFNOSUPPORT Protocol family not supported
97 EAFNOSUPPORT Address family not supported by protocol
98 EADDRINUSE Address already in use
99 EADDRNOTAVAIL Cannot assign requested address
100 ENETDOWN Network is down
101 ENETUNREACH Network is unreachable
102 ENETRESET Network dropped connection on reset
103 ECONNABORTED Software caused connection abort
104 ECONNRESET Connection reset by peer
105 ENOBUFS No buffer space available
106 EISCONN Transport endpoint is already connected
107 ENOTCONN Transport endpoint is not connected
108 ESHUTDOWN Cannot send after transport endpoint shutdown
109 ETOOMANYREFS Too many references: cannot splice
110 ETIMEDOUT Connection timed out
111 ECONNREFUSED Connection refused
112 EHOSTDOWN Host is down
113 EHOSTUNREACH No route to host
114 EALREADY Operation already in progress
115 EINPROGRESS Operation now in progress
116 ESTALE Stale file handle
117 EUCLEAN Structure needs cleaning
118 ENOTNAM Not a XENIX named type file
119 ENAVAIL No XENIX semaphores available
120 EISNAM Is a named type file
121 EREMOTEIO Remote I/O error
122 EDQUOT Disk quota exceeded
123 ENOMEDIUM No medium found
124 EMEDIUMTYPE Wrong medium type
125 ECANCELED Operation canceled
126 ENOKEY Required key not available
127 EKEYEXPIRED Key has expired
128 EKEYREVOKED Key has been revoked
129 EKEYREJECTED Key was rejected by service
130 EOWNERDEAD Owner died
131 ENOTRECOVERABLE State not recoverable
132 ERFKILL Operation not possible due to RF-kill
133 EHWPOISON Memory page has hardware error
ฉันใช้ tabularise ใน Vim เพื่อจัดเรียงคอลัมน์:
:%Tab /^[^ ]*\zs /r1l1l1
:%Tab /^ *[^ ]* *[^ ]*\zs /l1
รหัสข้อผิดพลาด 2 หมายถึง "ไม่พบไฟล์ / ไดเรกทอรี" โดยทั่วไปแล้วคุณสามารถใช้ฟังก์ชันperrorเพื่อพิมพ์สตริงที่มนุษย์สามารถอ่านได้
มีฟังก์ชั่นที่มีประโยชน์บางอย่างสำหรับจัดการกับerrno
s (เพียงเพื่อให้ชัดเจนสิ่งเหล่านี้มีอยู่ในตัวlibc
- ฉันแค่ให้การใช้งานตัวอย่างเพราะบางคนพบว่าการอ่านรหัสชัดเจนกว่าการอ่านภาษาอังกฤษ)
#include <string.h>
char *strerror(int errnum);
/* you can think of it as being implemented like this: */
static char strerror_buf[1024];
const char *sys_errlist[] = {
[EPERM] = "Operation not permitted",
[ENOENT] = "No such file or directory",
[ESRCH] = "No such process",
[EINTR] = "Interrupted system call",
[EIO] = "I/O error",
[ENXIO] = "No such device or address",
[E2BIG] = "Argument list too long",
/* etc. */
};
int sys_nerr = sizeof(sys_errlist) / sizeof(char *);
char *strerror(int errnum) {
if (0 <= errnum && errnum < sys_nerr && sys_errlist[errnum])
strcpy(strerror_buf, sys_errlist[errnum]);
else
sprintf(strerror_buf, "Unknown error %d", errnum);
return strerror_buf;
}
strerror
ส่งคืนสตริงที่อธิบายถึงหมายเลขข้อผิดพลาดที่คุณส่งให้ ข้อควรระวังนี่ไม่ใช่แบบเธรดหรืออินเตอร์รัปต์ที่ปลอดภัย มีอิสระในการเขียนสตริงและส่งกลับตัวชี้เดียวกันในการเรียกใช้ครั้งถัดไป ใช้strerror_r
หากคุณจำเป็นต้องกังวลเกี่ยวกับเรื่องนั้น
#include <stdio.h>
void perror(const char *s);
/* you can think of it as being implemented like this: */
void perror(const char *s) {
fprintf(stderr, "%s: %s\n", s, strerror(errno));
}
perror
พิมพ์ข้อความที่คุณให้รวมถึงสตริงที่อธิบายถึงerrno
ข้อผิดพลาดมาตรฐานในปัจจุบัน
some people find reading code clearer than reading English
ความจริง
นี่คือเร็วกว่าการค้นหารหัสในerrno.h
สั้นกว่าโซลูชั่นส่วนใหญ่โพสต์ที่นี่และไม่จำเป็นต้องติดตั้งเครื่องมือของบุคคลที่สาม:
perl -E 'say $!=shift' 2
อัตราผลตอบแทน
No such file or directory
-E
ใช้รหัสที่ยกมาต่อไปนี้เป็นสคริปต์ Perl say
กำลังเขียนอาร์กิวเมนต์ลงในเอาต์พุตมาตรฐาน เป็นตัวแปรพิเศษถือค่าของ$!
errno
หากใช้ในบริบทสตริงจะทำให้เกิดข้อผิดพลาดสตริงที่สอดคล้องกัน สคริปต์กำหนดค่า2
ให้กับตัวแปรนี้โดยใช้shift
คำสั่งซึ่งตัดส่วนหัวของอาเรย์อาร์กิวเมนต์@ARGV
และวางส่วนหัวนี้แทน perl -E 'say $!=2'
บรรทัดคำสั่งนอกจากนี้ยังจะได้รับการเขียนเป็น
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
int main(int i, char *c[]) {
if (i != 2)
fprintf(stderr, "Usage: perror errno\n");
else {
errno = atoi(c[1]);
perror("");
}
exit(0);
}
ทำงานบน Solaris
cc perror.c -o perror
<< ใช้บรรทัดนี้เพื่อคอมไพล์
perror(NULL);
หรือ ด้วยรูปแบบของมันคือperror("");
: ERROR NAME
ด้วยperror(NULL);
เอาต์พุตของมันเป็นเพียงERROR NAME
นี่คือเอกสาร ควรบอกคุณว่ามันหมายถึงอะไรและจะทำอย่างไรกับพวกเขา คุณควรหลีกเลี่ยงการใช้ค่าตัวเลขและใช้ค่าคงที่ที่ระบุไว้เช่นกันเนื่องจากตัวเลขอาจเปลี่ยนแปลงระหว่างระบบต่างๆ
errnoname
ห้องสมุดของฉันแล้ว ดูแลส่วนที่น่าเบื่อของการรวบรวมชื่อ errno ที่เป็นไปได้ทั้งหมดลงในฟังก์ชัน C ซึ่งสามารถนำมาใช้เพียงเล็กน้อยเพื่อรับชื่อแมโคร errno จากหมายเลข การใช้สิ่งนั้นเป็นฐานทำให้perror
ตัวแปรของคุณควรรวดเร็วและง่ายดาย
ฉันใช้สคริปต์ต่อไปนี้:
#!/usr/bin/python
import errno
import os
import sys
toname = dict((str(getattr(errno, x)), x)
for x in dir(errno)
if x.startswith("E"))
tocode = dict((x, getattr(errno, x))
for x in dir(errno)
if x.startswith("E"))
for arg in sys.argv[1:]:
if arg in tocode:
print arg, tocode[arg], os.strerror(tocode[arg])
elif arg in toname:
print toname[arg], arg, os.strerror(int(arg))
else:
print "Unknown:", arg
พิมพ์แล้วครั้งว่ามีการติดตั้งชนิดsudo apt-get install moreutils
errno 2
นอกจากนี้คุณยังสามารถใช้errno -l
สำหรับหมายเลขข้อผิดพลาดทั้งหมดหรือดูคนเดียวแฟ้มโดยท่อมันเช่นนี้grep
errno | grep file
โทร
perror("execl");
ในกรณีที่มีข้อผิดพลาด
ตัวอย่าง:
if(read(fd, buf, 1)==-1) {
perror("read");
}
manpages errno(3)
และperror(3)
น่าสนใจเช่นกัน ...
เมื่อคุณใช้strace (บน Linux) เพื่อเรียกใช้ไบนารี่ของคุณมันจะแสดงผลลัพธ์จากการเรียกของระบบและหมายเลขข้อผิดพลาดหมายถึงอะไร บางครั้งสิ่งนี้อาจเป็นประโยชน์กับคุณ
ฉันมีฟังก์ชั่นดังต่อไปนี้ในแฟ้ม .bashrc ของฉัน - มันเงยหน้าขึ้นค่า errno จากส่วนหัวของไฟล์ (สามารถเป็นได้ทั้ง/usr/include/errno.h
, /usr/include/linux/errno.h
ฯลฯ ฯลฯ )
ใช้งานได้หากติดตั้งไฟล์ส่วนหัวไว้ในเครื่อง ;-)
โดยปกติแล้วไฟล์ส่วนหัวมีข้อผิดพลาด + ถัดไปจะมีคำอธิบายในความคิดเห็น สิ่งต่อไปนี้:
./asm-generic/errno-base.h:#define EAGAIN 11 / * ลองอีกครั้ง * /
function errno()
{
local arg=$1
if [[ "x$arg" == "x-h" ]]; then
cat <<EOF
Usage: errno <num>
Prints text that describes errno error number
EOF
else
pushd /usr/include
find . -name "errno*.h" | xargs grep "[[:space:]]${arg}[[:space:]]"
popd
fi
}
มันหมายถึง:
ไม่พบไฟล์หรือไดเรกทอรี