Bagaimana cara mengetahui apa arti 'errno'?


221

Saat menelepon execl(...), saya mendapat errno=2. Apa artinya? Bagaimana saya bisa tahu artinya ini errno?

Jawaban:


339

Anda dapat menggunakan strerror()untuk mendapatkan string yang dapat dibaca manusia untuk nomor kesalahan. Ini adalah string yang sama yang dicetak oleh perror()tetapi berguna jika Anda memformat pesan kesalahan untuk sesuatu selain output kesalahan standar.

Sebagai contoh:

#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 juga mendukung varian eksplisit-threadsafe strerror_r().


5
1. TIDAK dijamin menjadi reentrant atau thread aman 3. Sebagian besar implementasi DO menulis ke buffer statis internal, termasuk implementasi LibU GNU
StaceyGirl

5
@Van benar dan @ris salah. strerror()tidak aman saat strerror_r()thread aman. Fungsi MT-Safe atau Thread-Safe aman untuk dipanggil di hadapan utas lainnya. MT, dalam MT-Safe, singkatan dari Multi Thread. -p26, Pustaka GNU C char * strerror(int errnum ) [Fungsi] Pendahuluan: | Balapan MT-Tidak Aman: strerror | Tumpukan AS-Tidak Aman i18n | AC-Tidak Aman mem | Lihat Bagian 1.2.2.1 [Konsep Keselamatan POSIX], halaman 2. -p58, Perpustakaan GNU C

@StaceyGirl glibc sebenarnya menggunakan fungsi untuk mendapatkan alamat errno sekarang, jadi errnoaman-utas. Lihat:#define errno *__errno_location()
SS Anne

@ JL2210 Ini bukan tentang errno, libc GNU strerrormenulis ke buffer global internal . Tidak selalu melakukan ini, tetapi dalam beberapa kasus memang demikian. Anda akan menimpa buffer itu saat utas lainnya membacanya.
StaceyGirl

Maaf. Saya pikir Anda bicarakan errno, tidak strerror().
SS Anne

55

Alih-alih berjalan perrorpada kode kesalahan apa pun yang Anda dapatkan, Anda dapat mengambil daftar lengkap errnonilai pada sistem Anda dengan one-liner berikut:

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


3
Untuk referensi, berikut adalah daftar dari Ubuntu 14.04 amd64 pastebin.com/407gDBPq
kevinf

45

Di Linux ada juga alat yang sangat rapi yang dapat langsung mengetahui apa arti setiap kode kesalahan. Pada Ubuntu: apt-get install errno.

Kemudian jika misalnya Anda ingin mendapatkan deskripsi kesalahan tipe 2, ketikkan saja errno 2di terminal.

Dengan errno -lAnda mendapatkan daftar dengan semua kesalahan dan deskripsinya. Jauh lebih mudah daripada metode lain yang disebutkan oleh poster sebelumnya.


+ janneb ya, namun nama paket errno masih sepenuhnya valid.
carefulnow1

Tidak kecuali Anda melakukan sesuatu yang aneh. Biasanya, menanamkan perror/ strerrorlebih baik karena PENGGUNA tidak perlu mencarinya. Meskipun, errno -llebih baik untuk menemukannya.
jgh fun-run

29

Berikut ini adalah output dari errno -ldiformat ulang untuk dibaca:

  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

Saya menggunakan tabularise di Vim untuk meluruskan kolom:

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

20

Kode kesalahan 2 berarti "File / Direktori tidak ditemukan". Secara umum, Anda bisa menggunakan fungsi perror untuk mencetak string yang dapat dibaca manusia.


8

Ada beberapa fungsi yang berguna untuk menangani errnos. (Hanya untuk memperjelas, ini adalah bawaan untuk libc- Saya hanya memberikan contoh implementasi karena beberapa orang menemukan membaca kode lebih jelas daripada membaca bahasa Inggris.)

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

strerrormengembalikan string yang menggambarkan nomor kesalahan yang Anda berikan. Perhatian, ini bukan thread-atau interrupt-safe; bebas menulis ulang string dan mengembalikan pointer yang sama pada doa berikutnya. Gunakan strerror_rjika Anda perlu khawatir tentang itu.

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

perrormencetak pesan yang Anda berikan, ditambah string yang menggambarkan saat ini errno, ke kesalahan standar.


2
some people find reading code clearer than reading Englishkebenaran.
Qix - MONICA DISALAHKAN

7

Ini lebih cepat daripada mencari kode errno.h, lebih pendek dari sebagian besar solusi yang diposting di sini dan tidak memerlukan instalasi alat pihak ketiga:

perl -E 'say $!=shift' 2

hasil panen

No such file or directory


Whoohoo. Cepat, tidak bergantung pada kompilasi ulang, dan bekerja (hampir) di mana-mana, termasuk pada mesin lama di mana errno.h tidak tersedia. Terima kasih banyak.
Adrien Clerc

Apa fungsinya? (Saya tidak tahu Perl)
Jonathan Lam

1
Argumen -Ememperlakukan kode yang dikutip berikut ini sebagai skrip Perl. saysedang menulis argumennya ke output standar. $!adalah variabel khusus yang memegang nilai errno. Jika digunakan dalam konteks string menghasilkan string kesalahan yang sesuai. Script memberikan nilai 2ke variabel ini dengan menggunakan shiftperintah, yang memotong kepala array argumen @ARGVdan menempatkan kepala ini di tempatnya. Baris perintah juga bisa ditulis sebagai 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); 
}    

Bekerja pada Solaris.
cc perror.c -o perror<< gunakan baris ini untuk mengkompilasinya


Bukankah seharusnya Anda menggunakan perror(NULL);? Dengan perror("");format itu : ERROR NAME. Dengan perror(NULL);outputnya sajaERROR NAME
jgh fun-run

5

Ini dokumentasinya . Itu akan memberi tahu Anda apa artinya dan apa yang harus dilakukan dengan mereka. Anda harus menghindari menggunakan nilai numerik dan menggunakan konstanta yang tercantum di sana juga, karena jumlahnya dapat berubah di antara sistem yang berbeda.


2
Saya benci bagaimana dokumentasi unix tidak mengaitkan konstanta dengan nilai integer. Apa nilai "EIO"? Documents tidak berharga seperti ini.
Seseorang di suatu tempat

4
@ Seseorang di suatu tempat Itu adalah fitur, bukan bug. Anda harus selalu menggunakan konstanta kode kesalahan simbolis dalam kode Anda, bukan angka literal. Ini membuat kode Anda jauh lebih mudah dibaca, karena sesuatu seperti EQFULL jauh lebih bermakna daripada 106. Sayangnya, bahasa ini tidak menegakkan ini, sehingga Anda membuat orang yang malas atau kacau di kepala yang menulis 106 bukannya EQFULL. Jangan ragu untuk mengirimi orang-orang itu pukulan yang bagus.
allyourcode

3
Masalahnya adalah bahwa perror tidak memberi tahu Anda makro yang sesuai dengan kesalahan terkait, ia mencetak beberapa pesan kesalahan yang sama sekali tidak terkait yang separuh waktunya bahkan tidak muncul di halaman manual. Saya ingin perror () yang mencetak nama MACRO sehingga saya bisa mencari kesalahan berdarah di halaman manual!
DarwinSurvivor

@DarwinSurvivor Anda mungkin tertarik pada errnonameperpustakaan saya saat itu. Ini mengurus bagian yang membosankan mengumpulkan semua nama errno yang mungkin menjadi fungsi C yang dapat digunakan hanya sepele untuk mendapatkan nama makro errno dari nomor tersebut. Menggunakannya sebagai basis, membuat perrorvarian Anda harus cepat dan mudah.
mtraceur

3

Saya menggunakan skrip berikut:

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

Ketik sudo apt-get install moreutilsdan kemudian setelah yang diinstal, ketik errno 2. Anda juga dapat menggunakan errno -luntuk semua nomor kesalahan, atau hanya melihat orang-orang file dengan pipa ke grep, seperti ini: errno | grep file.


2

Panggilan

perror("execl");

dalam hal terjadi kesalahan.

Sampel:

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

Halaman dari errno(3)dan perror(3)menarik, juga ...


0

Ketika Anda menggunakan strace (di Linux) untuk menjalankan biner Anda, itu akan menampilkan pengembalian dari panggilan sistem dan apa artinya nomor kesalahan. Ini terkadang bermanfaat bagi Anda.


0

Saya memiliki fungsi berikut dalam file .bashrc saya - itu mendongak nilai errno dari file header (dapat berupa /usr/include/errno.h, /usr/include/linux/errno.h, dll, dll)

Ini berfungsi jika file header diinstal pada mesin ;-)

Biasanya file header memiliki kesalahan + selanjutnya muncul penjelasan dalam komentar; sesuatu dari yang berikut:

./asm-generic/errno-base.h:#define EAGAIN 11 / * Coba lagi * /

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
}

Dengan menggunakan situs kami, Anda mengakui telah membaca dan memahami Kebijakan Cookie dan Kebijakan Privasi kami.
Licensed under cc by-sa 3.0 with attribution required.