Làm sao để biết 'errno' nghĩa là gì?


221

Khi gọi execl(...), tôi nhận được một errno=2. Nó có nghĩa là gì? Làm thế nào tôi có thể biết ý nghĩa của điều này errno?

Câu trả lời:


339

Bạn có thể sử dụng strerror()để có được một chuỗi có thể đọc được cho con số cho số lỗi. Đây là cùng một chuỗi được in bởi perror()nhưng nó hữu ích nếu bạn định dạng thông báo lỗi cho một cái gì đó không phải là đầu ra lỗi tiêu chuẩn.

Ví dụ:

#include <errno.h>
#include <string.h>

/* ... */

if(read(fd, buf, 1)==-1) {
    printf("Oh dear, something went wrong with read()! %s\n", strerror(errno));
}

Linux cũng hỗ trợ biến thể chủ đề rõ ràng strerror_r().


5
1. KHÔNG được bảo đảm là reentrant hoặc thread safe 3. Hầu hết các triển khai DO ghi vào bộ đệm tĩnh bên trong, bao gồm cả triển khai GNU LibC
StaceyGirl

5
@Ivan là chính xác và @Chris không chính xác. strerror()không phải là chủ đề an toàn trong khi strerror_r()chủ đề là an toàn. Các chức năng MT-Safe hoặc Thread-Safe an toàn để gọi khi có sự hiện diện của các luồng khác. MT, trong MT-Safe, là viết tắt của Multi Thread. -p26, Thư viện GNU C char * strerror(int errnum ) [Chức năng] Sơ bộ: | Cuộc đua MT-không an toàn: strerror | AS-không an toàn đống i18n | AC-Không an toàn mem | Xem Phần 1.2.2.1 [Khái niệm an toàn POSIX], trang 2. -p58, Thư viện GNU C

@StaceyGirl glibc thực sự sử dụng một chức năng để lấy địa chỉ của errno bây giờ, do đó, errnoan toàn cho chuỗi. Nhìn:#define errno *__errno_location()
SS Anne

@ JL2210 Đây không phải là về errno, GNU libc strerrorghi vào bộ đệm toàn cục bên trong . Nó không luôn luôn làm điều này, nhưng trong một số trường hợp nó làm. Bạn sẽ ghi đè lên bộ đệm đó trong khi một luồng khác đang đọc nó.
StaceyGirl

Lấy làm tiếc. Tôi nghĩ bạn đang nói về errno, không phải strerror().
SS Anne

55

Thay vì chạy perrortrên bất kỳ mã lỗi nào bạn nhận được, bạn có thể truy xuất danh sách đầy đủ các errnogiá trị trên hệ thống của mình bằng một lớp lót sau:

cpp -dM /usr/include/errno.h | grep 'define E' | sort -n -k 3


3
Để tham khảo, đây là danh sách từ Ubuntu 14.04 amd64 pastebin.com/407gDBPq
kevinf

45

Trên Linux cũng có một công cụ rất gọn gàng có thể cho biết ngay mỗi mã lỗi có nghĩa là gì. Trên Ubuntu : apt-get install errno.

Sau đó, nếu bạn muốn lấy mô tả về lỗi loại 2, chỉ cần gõ errno 2vào thiết bị đầu cuối.

Với errno -lbạn có được một danh sách với tất cả các lỗi và mô tả của họ. Dễ dàng hơn nhiều mà các phương pháp khác được đề cập bởi các áp phích trước.


+ janneb có, tuy nhiên tên gói errno vẫn hoàn toàn hợp lệ.
cẩn thận1

Không trừ khi bạn đang làm một cái gì đó kỳ lạ. Thông thường, nhúng perror/ strerrortốt hơn bởi vì sau đó NGƯỜI DÙNG không phải tìm kiếm nó. Mặc dù, errno -llà tốt hơn để tìm thấy chúng.
jgh vui vẻ chạy

29

Đây là đầu ra từ errno -lđịnh dạng lại cho dễ đọc:

  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

Tôi đã sử dụng tabularise trong Vim để căn chỉnh các cột:

:%Tab /^[^ ]*\zs /r1l1l1
:%Tab /^ *[^ ]* *[^ ]*\zs /l1

20

Mã lỗi 2 có nghĩa là "Không tìm thấy tệp / thư mục". Nói chung, bạn có thể sử dụng hàm perror để in một chuỗi có thể đọc được của con người.


8

Có một vài chức năng hữu ích để xử lý errnos. (Chỉ cần làm rõ, những điều này được tích hợp sẵn libc- Tôi chỉ cung cấp các triển khai mẫu vì một số người thấy đọc mã rõ ràng hơn đọc tiếng Anh.)

#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;
}

strerrortrả về một chuỗi mô tả số lỗi bạn đã truyền cho nó. Lưu ý, đây không phải là chủ đề hoặc ngắt an toàn; nó được tự do viết lại chuỗi và trả về cùng một con trỏ trong lần gọi tiếp theo. Sử dụng strerror_rnếu bạn cần phải lo lắng về điều đó.

#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));
}

perrorin ra thông điệp bạn đưa ra, cộng với một chuỗi mô tả hiện tại errno, lỗi tiêu chuẩn.


2
some people find reading code clearer than reading Englishsự thật
Qix - MONICA ĐƯỢC PHÂN BIỆT

7

Điều này nhanh hơn tìm kiếm mã trong errno.h, ngắn hơn hầu hết các giải pháp được đăng ở đây và nó không yêu cầu cài đặt các công cụ của bên thứ ba:

perl -E 'say $!=shift' 2

sản lượng

No such file or directory


Ôi trời. Nhanh, không phụ thuộc vào việc biên dịch lại và hoạt động (gần như) ở mọi nơi, kể cả trên máy cũ, nơi errno.h không khả dụng. Cảm ơn rất nhiều.
Adrien Clerc

Cái này làm gì (Tôi không biết Perl)
Jonathan Lam

1
Đối số -Ecoi mã được trích dẫn sau đây là tập lệnh Perl. sayđang viết các đối số của nó để đầu ra tiêu chuẩn. $!là một biến đặc biệt giữ giá trị của errno. Nếu được sử dụng trong ngữ cảnh chuỗi, nó mang lại chuỗi lỗi tương ứng. Kịch bản gán giá trị 2cho biến này bằng cách sử dụng shiftlệnh, loại bỏ phần đầu của mảng đối số @ARGVvà đặt phần đầu này vào vị trí của nó. Dòng lệnh cũng có thể được viết là perl -E 'say $!=2'.
LCC

5
#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); 
}    

Hoạt động trên Solaris.
cc perror.c -o perror<< sử dụng dòng này để biên dịch nó


Bạn không nên sử dụng perror(NULL);? Với perror("");định dạng của nó là : ERROR NAME. Với perror(NULL);đầu ra của nó chỉ làERROR NAME
jgh vui vẻ chạy

5

Đây là tài liệu . Điều đó sẽ cho bạn biết ý nghĩa của nó và phải làm gì với chúng. Bạn nên tránh sử dụng giá trị số và sử dụng các hằng số được liệt kê ở đó, vì số có thể thay đổi giữa các hệ thống khác nhau.


2
Tôi ghét cách tài liệu unix không liên kết với giá trị nguyên. Giá trị nào là "EIO"? Tài liệu là vô giá trị như thế này.
Ai đó ở đâu đó

4
@SomeoneSomewhere Đó là một tính năng, không phải là một lỗi. Bạn nên luôn luôn sử dụng các hằng số mã lỗi tượng trưng trong mã của bạn, không phải là số bằng chữ. Điều này làm cho mã của bạn dễ đọc hơn nhiều, bởi vì một cái gì đó như EQFULL có ý nghĩa hơn nhiều so với 106. Đáng buồn thay, ngôn ngữ không thực thi điều này, vì vậy bạn nhận được những người lười biếng hoặc rối tung trong đầu, người viết 106 thay vì EQFULL. Hãy gửi cho những người đó một nhịp đập tốt đẹp.
mã allyour

3
Vấn đề là perror không cho bạn biết lỗi tương đương với macro nào, nó in một số thông báo lỗi hoàn toàn không liên quan khác mà một nửa thời gian thậm chí không xuất hiện trong trang man. Tôi muốn một perror () in tên MACRO để tôi có thể tra cứu lỗi đẫm máu trong trang man!
DarwinSurvivor

@DarwinSurvivor Bạn có thể quan tâm đến tôi errnoname thư viện sau đó. Nó đảm nhiệm phần tẻ nhạt của việc tập hợp tất cả các tên errno có thể vào một hàm C có thể được sử dụng một cách tầm thường để lấy tên macro errno từ số đó. Sử dụng nó làm cơ sở, làm cho perrorbiến thể của bạn phải nhanh chóng và dễ dàng.
mtraceur

3

Tôi sử dụng đoạn script sau:

#!/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

3

Kiểu sudo apt-get install moreutils và sau đó một khi đã cài đặt, gõ errno 2. Bạn cũng có thể sử dụng errno -lcho tất cả các số lỗi hoặc chỉ xem các tập tin bằng cách chuyển nó tới grep, như thế này : errno | grep file.


2

Gọi

perror("execl");

trong trường hợp có lỗi

Mẫu vật:

if(read(fd, buf, 1)==-1) {
    perror("read");
}

Các trang web của errno(3)perror(3)cũng thú vị ...


0

Khi bạn sử dụng strace (trên Linux) để chạy nhị phân của mình, nó sẽ xuất trả về từ các cuộc gọi hệ thống và số lỗi có nghĩa là gì. Điều này đôi khi có thể hữu ích cho bạn.


0

Tôi có chức năng sau đây trong tập tin .bashrc của tôi - nó nhìn lên giá trị errno từ các tập tin tiêu đề (có thể là /usr/include/errno.h, /usr/include/linux/errno.h, vv, vv)

Nó hoạt động nếu các tệp tiêu đề được cài đặt trên máy ;-)

Thông thường tệp tiêu đề có lỗi + tiếp theo là phần giải thích trong bình luận; một số thứ sau đây:

./asm-generic/errno-base.h:#define EAGAIN 11 / * Thử lại * /

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
}

Khi sử dụng trang web của chúng tôi, bạn xác nhận rằng bạn đã đọc và hiểu Chính sách cookieChính sách bảo mật của chúng tôi.
Licensed under cc by-sa 3.0 with attribution required.