¿Cómo saber lo que significa 'errno'?


221

Cuando llamo execl(...), recibo un errno=2. Qué significa eso? ¿Cómo puedo saber el significado de esto errno?

Respuestas:


339

Puede usar strerror()para obtener una cadena legible por humanos para el número de error. Esta es la misma cadena impresa perror()pero es útil si está formateando el mensaje de error para algo que no sea la salida de error estándar.

Por ejemplo:

#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 también admite la variante explícitamente segura para subprocesos strerror_r().


55
1. NO se garantiza que sea reentrante o seguro para subprocesos 3. La mayoría de las implementaciones escriben en el búfer estático interno, incluida la implementación de GNU LibC
StaceyGirl

55
@Ivan es correcto y @Chris en incorrecto. strerror()no es seguro para hilos mientras strerror_r()es seguro para hilos. Las funciones MT-Safe o Thread-Safe son seguras para llamar en presencia de otros hilos. MT, en MT-Safe, significa Multi Thread. -p26, La Biblioteca GNU C char * strerror(int errnum ) [Función] Preliminar: | Carrera MT-insegura: strerror | AS-Inseguro montón i18n | AC-Inseguro mem | Consulte la Sección 1.2.2.1 [Conceptos de seguridad POSIX], página 2. -p58, The GNU C Library

@StaceyGirl glibc en realidad usa una función para obtener la dirección de errno ahora, por lo que errnoes seguro para subprocesos. Mira:#define errno *__errno_location()
SS Anne

@ JL2210 Esto no se trata errno, GNU libc strerrorescribe en el búfer global interno . No siempre hace esto, pero en algunos casos lo hace. Sobreescribirá ese búfer mientras otro hilo lo está leyendo.
StaceyGirl

Lo siento. Pensé que estabas hablando errno, no strerror().
SS Anne

55

En lugar de ejecutar perrorcualquier código de error que obtenga, puede recuperar una lista completa de errnovalores en su sistema con la siguiente línea:

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


3
Como referencia, aquí hay una lista de Ubuntu 14.04 amd64 pastebin.com/407gDBPq
kevinf

45

En Linux también hay una herramienta muy ordenada que puede decir de inmediato qué significa cada código de error. En Ubuntu: apt-get install errno.

Entonces, si por ejemplo desea obtener la descripción del tipo de error 2, simplemente escriba errno 2el terminal.

Con errno -lusted obtendrá una lista con todos los errores y sus descripciones. Mucho más fácil que otros métodos mencionados en carteles anteriores.


+ janneb sí, sin embargo, el nombre del paquete errno sigue siendo completamente válido.
cuidadoso1 de

No, a menos que estés haciendo algo extraño. Normalmente, incrustar perror/ strerrores mejor porque entonces el USUARIO no tiene que buscarlo. Aunque, errno -les mejor encontrarlos.
Jgh Fun-Run

29

Aquí está el resultado de errno -lreformateado para facilitar la lectura:

  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

Usé tabularise en Vim para alinear las columnas:

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

20

El código de error 2 significa "Archivo / Directorio no encontrado". En general, puede usar la función perror para imprimir una cadena legible por humanos.


8

Hay algunas funciones útiles para tratar con errnos. (Solo para que quede claro, estos están integrados en libc- Solo proporciono implementaciones de muestra porque algunas personas encuentran que leer códigos es más claro que leer inglés).

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

strerrordevuelve una cadena que describe el número de error que le ha pasado. Precaución, esto no es a prueba de hilos o interrupciones; es libre de reescribir la cadena y devolver el mismo puntero en la próxima invocación. Úselo strerror_rsi necesita preocuparse por eso.

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

perrorimprime el mensaje que le da, más una cadena que describe el actual errno, al error estándar.


2
some people find reading code clearer than reading Englishverdad.
Qix - MONICA FUE MALTRATADA el

7

Esto es más rápido que buscar el código errno.h, más corto que la mayoría de las soluciones publicadas aquí y no requiere la instalación de herramientas de terceros:

perl -E 'say $!=shift' 2

rendimientos

No such file or directory


Whoohoo Rápido, no se basa en una recompilación y funciona (casi) en todas partes, incluso en máquinas antiguas donde errno.h no está disponible. Muchas gracias.
Adrien Clerc

¿Qué hace esto? (No sé Perl)
Jonathan Lam

1
El argumento -Etrata el siguiente código citado como un script Perl. sayestá escribiendo sus argumentos a la salida estándar. $!es una variable especial que contiene el valor de errno. Si se usa en contexto de cadena, produce la cadena de error correspondiente. El script asigna el valor 2a esta variable mediante el shiftcomando, que corta el encabezado de la matriz de argumentos @ARGVy coloca este encabezado en su lugar. La línea de comando también podría haberse escrito como 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); 
}    

Funciona en Solaris.
cc perror.c -o perror<< usa esta línea para compilarlo


¿No deberías estar usando perror(NULL);? Con perror("");su formato es : ERROR NAME. Con perror(NULL);su salida es soloERROR NAME
jgh fun-run

5

Aquí está la documentación . Eso debería decirle qué significa y qué hacer con ellos. Debe evitar usar el valor numérico y usar las constantes que figuran allí también, ya que el número puede cambiar entre diferentes sistemas.


2
Odio cómo la documentación de Unix no asocia constante a valor entero. ¿Qué valor es "EIO"? Los documentos no valen nada así.
Alguien en algún lugar

44
@SomeoneSomewhere Esa es una característica, no un error. Siempre debe usar constantes de código de error simbólico en su código, no literales de números. Esto hace que su código sea mucho más legible, porque algo como EQFULL es mucho más significativo que 106. Lamentablemente, el lenguaje no impone esto, por lo que obtiene personas que son perezosas o desordenadas en la cabeza que escriben 106 en lugar de EQFULL. Siéntase libre de enviarles a esas personas una buena paliza.
allyourcode

3
El problema es que el error no le dice a qué macro equivalente se relaciona el error, imprime algún otro mensaje de error completamente no relacionado que la mitad del tiempo ni siquiera aparece en la página del manual. ¡Quiero un perror () que imprima el nombre MACRO para poder buscar el error sangriento en la página del manual!
DarwinSurvivor

@DarwinSurvivor Puede que te interese mi errnonamebiblioteca entonces. Se encarga de la parte tediosa de reunir todos los nombres posibles de errno en una función C que se puede usar trivialmente para obtener el nombre de macro errno del número. Usando eso como base, hacer que su perrorvariante sea rápida y fácil.
mtraceur

3

Yo uso el siguiente script:

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

Escriba sudo apt-get install moreutilsy luego una vez que se haya instalado, escriba errno 2. También se puede utilizar errno -lpara todos los números de error, o ver sólo los archivos mediante la canalización que grep, como esto: errno | grep file.


2

Llamada

perror("execl");

en caso de error

Muestra:

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

Las páginas de manual de errno(3)y también perror(3)son interesantes ...


0

Cuando usa strace (en Linux) para ejecutar su binario, generará los retornos de las llamadas del sistema y lo que significa el número de error. Esto a veces puede ser útil para usted.


0

Tengo la siguiente función en mi archivo .bashrc - se consulta el valor de errno a partir de los archivos de cabecera (puede ser cualquiera /usr/include/errno.h, /usr/include/linux/errno.hetc., etc.)

Funciona si los archivos de encabezado están instalados en la máquina ;-)

Por lo general, el archivo de encabezado tiene un error + luego viene la explicación en el comentario; algo de lo siguiente:

./asm-generic/errno-base.h:#define EAGAIN 11 / * Inténtalo de nuevo * /

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
}

Al usar nuestro sitio, usted reconoce que ha leído y comprende nuestra Política de Cookies y Política de Privacidad.
Licensed under cc by-sa 3.0 with attribution required.