Woher wissen, was das "Errno" bedeutet?

221

Wenn execl(...)ich anrufe , bekomme ich eine errno=2. Was heißt das? Wie kann ich die Bedeutung davon erkennen errno?

Barth
quelle

Antworten:

339

Sie können verwenden strerror(), um eine für Menschen lesbare Zeichenfolge für die Fehlernummer abzurufen. Dies ist dieselbe Zeichenfolge, die von gedruckt wird, perror()aber es ist nützlich, wenn Sie die Fehlermeldung für etwas anderes als die Standardfehlerausgabe formatieren.

Beispielsweise:

#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 unterstützt auch die explizit threadsichere Variante strerror_r().

Commodore Jaeger
quelle
5
1. Es ist NICHT garantiert, dass es wiedereintritts- oder threadsicher ist. 3. Die meisten Implementierungen schreiben in den internen statischen Puffer, einschließlich der Implementierung von GNU LibC
StaceyGirl
5
@Ivan ist korrekt und @Chris ist falsch. strerror()ist nicht threadsicher, während strerror_r()threadsicher ist. MT-Safe- oder Thread-Safe-Funktionen können bei Vorhandensein anderer Threads sicher aufgerufen werden. MT steht in MT-Safe für Multi Thread. -p26, Die GNU C-Bibliothek char * strerror(int errnum ) [Funktion] Vorläufig: | MT-Unsicheres Rennen: Fehler | AS-unsicherer Heap i18n | AC-Unsichere mem | Siehe Abschnitt 1.2.2.1 [POSIX-Sicherheitskonzepte], Seite 2. -p58, The GNU C Library
@StaceyGirl glibc verwendet tatsächlich eine Funktion, um die Adresse von errno jetzt abzurufen, errnoist also threadsicher. Schauen Sie:#define errno *__errno_location()
SS Anne
@ JL2210 Hier geht es nicht darum errno, GNU libc strerrorschreibt in den internen globalen Puffer . Dies ist nicht immer der Fall, in einigen Fällen jedoch. Sie werden diesen Puffer überschreiben, während ein anderer Thread ihn liest.
StaceyGirl
Es tut uns leid. Ich dachte du redest errnonicht strerror().
SS Anne
55

Anstatt mit perroreinem Fehlercode zu arbeiten, den Sie erhalten, können Sie eine vollständige Liste der errnoWerte auf Ihrem System mit dem folgenden Einzeiler abrufen :

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

Josh Kelley
quelle
3
Als Referenz ist hier eine Liste von Ubuntu 14.04 amd64 pastebin.com/407gDBPq
kevinf
45

Unter Linux gibt es auch ein sehr übersichtliches Tool, das sofort erkennen kann, was jeder Fehlercode bedeutet. Unter Ubuntu : apt-get install errno.

Wenn Sie beispielsweise die Beschreibung des Fehlertyps 2 erhalten möchten, geben Sie einfach errno 2das Terminal ein.

Mit erhalten errno -lSie eine Liste mit allen Fehlern und deren Beschreibungen. Viel einfacher als andere Methoden, die auf früheren Postern erwähnt wurden.

Pithikos
quelle
+ janneb ja, aber der errno paketname ist noch vollständig gültig.
Vorsicht
Nicht, wenn Sie etwas Seltsames tun. Normalerweise ist das Einbetten von perror/ strerrorbesser, da der BENUTZER es dann nicht nachschlagen muss. Obwohl errno -les besser ist, sie zu finden.
JGH Fun-Run
29

Hier ist die Ausgabe von neu errno -lformatiert für die Lesbarkeit:

  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

Ich habe die Tabellierung in Vim verwendet, um die Spalten auszurichten:

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

Fehlercode 2 bedeutet "Datei / Verzeichnis nicht gefunden". Im Allgemeinen können Sie die Perror- Funktion verwenden, um eine lesbare Zeichenfolge zu drucken.

Schnaader
quelle
8

Es gibt einige nützliche Funktionen für den Umgang mit errnos. (Nur um es klar zu machen, diese sind integriert in libc- ich biete nur Beispielimplementierungen an, weil manche Leute das Lesen von Code klarer finden als das Lesen von Englisch.)

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

strerrorGibt eine Zeichenfolge zurück, die die Fehlernummer beschreibt, die Sie an sie übergeben haben. Achtung, dies ist nicht thread- oder Interrupt-sicher. Es ist frei, die Zeichenfolge neu zu schreiben und beim nächsten Aufruf denselben Zeiger zurückzugeben. Verwenden strerror_rSie diese Option, wenn Sie sich darüber Sorgen machen müssen.

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

perrordruckt die Nachricht aus, die Sie ihm geben, sowie eine Zeichenfolge, die den aktuellen errnoStandardfehler beschreibt.

kurzlebig
quelle
2
some people find reading code clearer than reading EnglishWahrheit.
Qix - MONICA wurde am
7

Dies ist schneller als das Nachschlagen des Codes errno.h, kürzer als die meisten hier veröffentlichten Lösungen und erfordert keine Installation von Tools von Drittanbietern:

perl -E 'say $!=shift' 2

ergibt

No such file or directory

LCC
quelle
Whoohoo. Schnell, ist nicht auf eine Neukompilierung angewiesen und funktioniert (fast) überall, auch auf alten Computern, auf denen errno.h nicht verfügbar ist. Vielen Dank.
Adrien Clerc
Was macht das? (Ich kenne Perl nicht)
Jonathan Lam
1
Das Argument -Ebehandelt den folgenden zitierten Code als Perl-Skript. sayschreibt seine Argumente in die Standardausgabe. $!ist eine spezielle Variable, die den Wert von enthält errno. Bei Verwendung im Zeichenfolgenkontext wird die entsprechende Fehlerzeichenfolge ausgegeben. Das Skript weist 2dieser Variablen den Wert zu, indem es den shiftBefehl verwendet, der den Kopf des Argumentarrays abschneidet @ARGVund diesen Kopf an seine Stelle setzt. Die Befehlszeile könnte auch als geschrieben worden sein 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); 
}    

Funktioniert unter Solaris.
cc perror.c -o perror<< Verwenden Sie diese Zeile, um sie zu kompilieren

Johan Snowgoose
quelle
Solltest du nicht benutzen perror(NULL);? Mit perror("");seinem Format ist : ERROR NAME. Mit perror(NULL);seiner Ausgabe ist nurERROR NAME
jgh Fun-Run
5

Hier ist die Dokumentation . Das sollte dir sagen, was es bedeutet und was du mit ihnen machen sollst. Sie sollten die Verwendung des numerischen Werts vermeiden und auch die dort aufgeführten Konstanten verwenden, da sich die Anzahl zwischen verschiedenen Systemen ändern kann.

dsm
quelle
2
Ich hasse es, wenn die Unix-Dokumentation keine Konstante mit einem ganzzahligen Wert verknüpft. Welcher Wert ist "EIO"? Docs sind so wertlos.
Jemand irgendwo
4
@SomeoneSomewhere Das ist eine Funktion, kein Fehler. Sie sollten immer symbolische Fehlercodekonstanten in Ihrem Code verwenden, keine Zahlenliterale. Dies macht Ihren Code viel lesbarer, da so etwas wie EQFULL viel aussagekräftiger ist als 106. Leider erzwingt die Sprache dies nicht, sodass Sie Leute bekommen, die faul oder durcheinander im Kopf sind und 106 anstelle von EQFULL schreiben. Fühlen Sie sich frei, diesen Leuten einen netten Peer-Prügel zu schicken.
Allyourcode
3
Das Problem ist, dass perror Ihnen nicht sagt, auf welches Makroäquivalent sich der Fehler bezieht, sondern eine andere, völlig unabhängige Fehlermeldung ausgibt, die die Hälfte der Zeit nicht einmal in der Manpage angezeigt wird. Ich möchte einen perror (), der den MACRO-Namen druckt, damit ich den blutigen Fehler in der Manpage nachschlagen kann!
DarwinSurvivor
@DarwinSurvivor Dann könnte dich meine errnonameBibliothek interessieren . Es kümmert sich um den mühsamen Teil des Sammelns aller möglichen Errno-Namen in einer C-Funktion, die nur trivial verwendet werden kann, um den Errno-Makronamen aus der Nummer zu erhalten. Auf dieser Grundlage sollte die Erstellung Ihrer perrorVariante schnell und einfach sein.
mtraceur
3

Ich benutze das folgende Skript:

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

quelle
3

Geben Sie ein sudo apt-get install moreutilsund geben Sie nach der Installation ein errno 2. Sie können auch errno -lfür alle Fehlernummern verwenden oder nur die Dateien anzeigen, indem Sie sie grepwie folgt weiterleiten : errno | grep file.

Vorsicht1
quelle
2

Anruf

perror("execl");

im Fehlerfall.

Stichprobe:

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

Die Manpages von errno(3)und perror(3)sind auch interessant ...

Johannes Weiss
quelle
0

Wenn Sie strace (unter Linux) verwenden, um Ihre Binärdatei auszuführen, werden die Rückgaben von Systemaufrufen und die Bedeutung der Fehlernummer ausgegeben. Dies kann manchmal nützlich für Sie sein.

Sarel Botha
quelle
0

Ich habe die folgende Funktion in meinem .bashrc - es sieht den errno - Wert aus den Header - Dateien auf (kann entweder /usr/include/errno.h, /usr/include/linux/errno.hetc., etc.)

Es funktioniert, wenn Header-Dateien auf dem Computer installiert sind ;-)

Normalerweise hat die Header-Datei einen Fehler + als nächstes kommt die Erklärung im Kommentar; etwas von folgendem:

./asm-generic/errno-base.h:#define EAGAIN 11 / * Versuchen Sie es erneut * /

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
}
MichaelMoser
quelle
0

Es bedeutet:

Datei oder Verzeichnis nicht gefunden.

Otávio Décio
quelle