Как узнать, что означает "errno"?

При звонке execl(...) я получаю errno=2. Что это значит? Как я могу узнать значение этого errno?


person Barth    schedule 02.02.2009    source источник


Ответы (15)


Вы можете использовать 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));
}

Linux также поддерживает вариант с явной поточной безопасностью strerror_r().

person Commodore Jaeger    schedule 02.02.2009
comment
1. НЕ гарантируется реентерабельность или потокобезопасность. 3. Большинство реализаций ДОЛЖНЫ записывать во внутренний статический буфер, включая реализацию GNU LibC. - person StaceyGirl; 09.06.2017
comment
@Ivan правильный, а @Chris неверный. strerror() не является потокобезопасным, а strerror_r() - потокобезопасным. Функции MT-Safe или Thread-Safe можно безопасно вызывать в присутствии других потоков. MT в переводе с MT-Safe означает многопоточность. -p26, Библиотека GNU C char * strerror(int errnum ) [Функция] Предварительные сведения: | MT-Небезопасная гонка: strerror | AS-Небезопасная куча i18n | AC-Unsafe mem | См. Раздел 1.2.2.1 [Концепции безопасности POSIX], стр. 2. -p58, Библиотека GNU C - person ; 15.09.2017
comment
@StaceyGirl glibc теперь использует функцию для получения адреса errno, поэтому errno является потокобезопасным. Посмотрите: #define errno *__errno_location() - person S.S. Anne; 22.03.2019
comment
@ JL2210 Речь идет не о errno, GNU libc strerror записывает во внутренний глобальный буфер. Это происходит не всегда, но в некоторых случаях это так. Вы будете перезаписывать этот буфер, пока другой поток его читает. - person StaceyGirl; 22.03.2019
comment
Извините. Я думал, вы говорите о errno, а не о strerror(). - person S.S. Anne; 22.03.2019

Вместо того, чтобы запускать perror для любого полученного кода ошибки, вы можете получить полный список errno значений в вашей системе с помощью следующего однострочника:

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

person Josh Kelley    schedule 02.02.2009
comment
Для справки, вот список из Ubuntu 14.04 amd64 pastebin.com/407gDBPq - person Kevin; 08.08.2014

В Linux также есть очень удобный инструмент, который может сразу сказать, что означает каждый код ошибки. В Ubuntu: apt-get install errno.

Затем, если, например, вы хотите получить описание ошибки типа 2, просто введите errno 2 в терминале.

С errno -l вы получаете список со всеми ошибками и их описаниями. Намного проще, чем другие методы, упомянутые в предыдущих плакатах.

person Pithikos    schedule 05.10.2011
comment
+ janneb да, однако имя пакета errno по-прежнему полностью актуально. - person carefulnow1; 06.11.2016
comment
Нет, если вы не делаете что-то странное. Обычно встраивание perror / strerror лучше, потому что тогда ПОЛЬЗОВАТЕЛЮ не нужно искать его. Хотя, errno -l их лучше найти. - person jgh fun-run; 19.03.2020

Вот результат 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
person joeytwiddle    schedule 10.10.2016

Код ошибки 2 означает «Файл / каталог не найден». В общем, вы можете использовать функцию perror для печати удобочитаемой строки.

person schnaader    schedule 02.02.2009

Есть несколько полезных функций для работы с errnos. (Чтобы прояснить, они встроены в 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, до стандартной ошибки.

person ephemient    schedule 02.02.2009
comment
some people find reading code clearer than reading English правда. - person Qix - MONICA WAS MISTREATED; 19.04.2016

Это быстрее, чем искать код в errno.h, короче, чем большинство решений, размещенных здесь, и не требует установки сторонних инструментов:

perl -E 'say $!=shift' 2

дает

No such file or directory

person LCC    schedule 28.10.2015
comment
Ухуху. Быстро, не требует перекомпиляции и работает (почти) везде, в том числе на старых машинах, где errno.h недоступен. Большое спасибо. - person Adrien Clerc; 20.07.2016
comment
Что это значит? (Я не знаю Perl) - person Jonathan Lam; 17.08.2018
comment
Аргумент -E рассматривает приведенный ниже код в кавычках как сценарий Perl. say записывает свои аргументы в стандартный вывод. $! - это специальная переменная, содержащая значение errno. При использовании в строковом контексте выдает соответствующую строку ошибки. Сценарий присваивает этой переменной значение 2 с помощью команды shift, которая отрезает головку массива аргументов @ARGV и помещает эту головку на свое место. Командная строка также могла быть записана как perl -E 'say $!=2'. - person LCC; 23.10.2018

#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 ‹< используйте эту строку для компиляции

person Johan Snowgoose    schedule 01.11.2011
comment
Разве вы не должны использовать perror(NULL);? С perror(""); это формат : ERROR NAME. С perror(NULL); вывод будет просто ERROR NAME - person jgh fun-run; 19.03.2020

Вот документация. Это должно сказать вам, что это значит и что с ними делать. Вам следует избегать использования числовых значений, а также использовать перечисленные там константы, так как число может меняться в разных системах.

person dsm    schedule 02.02.2009
comment
Я ненавижу то, что документация unix не связывает константу с целочисленным значением. Какое значение имеет EIO? Документы так бесполезны. - person Someone Somewhere; 02.09.2011
comment
@SomeoneSomewhere Это особенность, а не ошибка. Вы всегда должны использовать в коде символьные константы кода ошибки, а не числовые литералы. Это делает ваш код более читабельным, потому что что-то вроде EQFULL гораздо более значимо, чем 106. К сожалению, язык не обеспечивает этого, поэтому вы получаете людей, которые ленивы или запутались в голове, которые пишут 106 вместо EQFULL. Не стесняйтесь посылать этим людям приятную шутку. - person allyourcode; 25.11.2012
comment
Проблема в том, что perror не сообщает вам, к какому макросу относится ошибка, он выводит какое-то другое совершенно несвязанное сообщение об ошибке, которое в половине случаев даже не появляется на странице руководства. Мне нужен perror (), который печатает имя МАКРОСА, чтобы я мог найти чертову ошибку на странице руководства! - person DarwinSurvivor; 24.02.2013
comment
@DarwinSurvivor Тогда вас может заинтересовать моя errnoname библиотека. Он берет на себя утомительную часть сбора всех возможных имен errno в функцию C, которую можно тривиально использовать для получения имени макроса errno из числа. Используя это как основу, создание вашего perror варианта должно быть быстрым и легким. - person mtraceur; 11.08.2019

Я использую следующий скрипт:

#!/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
person Community    schedule 03.02.2009

Введите sudo apt-get install moreutils в оболочку, а затем, когда она будет установлена, введите errno 2. Вы также можете использовать errno -l для всех номеров ошибок или просматривать только файловые, подключив их к grep, например: errno -l | grep file.

person carefulnow1    schedule 06.11.2016

Вызов

perror("execl");

в случае ошибки.

Образец:

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

Справочники errno(3) и perror(3) тоже интересны ...

person Johannes Weiss    schedule 02.02.2009

Когда вы используете strace (в Linux) для запуска своего двоичного файла, он будет выводить результаты из системы звонков и что означает номер ошибки. Иногда это может быть вам полезно.

person Sarel Botha    schedule 02.02.2009

У меня есть следующая функция в моем файле .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
}
person MichaelMoser    schedule 22.05.2014

Это означает:

Файл или каталог не найден.

person Otávio Décio    schedule 02.02.2009