This isn't a library function, but a program that I needed to polish up a bit.
Fiddling and tweaking can turn it into a library function without too much 
trouble.
The last file ("err-names.h") is a derived file that depends on the "mk" script
that is unabashedly bash.  It is also c99.  That is pretty recent, too.
/**
 *  @file errno.c
 *
 *  Time-stamp:        "2011-08-02 14:04:06 bkorb"
 *
 *  Print out error numbers and meanings
 *
 *  Copyright (c) 2002-2011 Bruce Korb - all rights reserved
 *
 * errno is free software: you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the
 * Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * errno is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 * See the GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program.  If not, see <http://www.gnu.org/licenses/>.
 */
#include <errno.h>
#include <regex.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#include "err-names.h"

static char const usage_text[] =
    "USAGE: errno <name-number-or-description> [...]\n"
    "For each argument, the program will find the error(s) that match\n"
    "the name, error number or description pattern and print a line:\n"
    "  nnn (ENAME. . . . . ) == Description\n"
    "where 'nnn' is the numerical number, 'ENAME' is the #defined name,\n"
    "and 'Description' may be plain text or a regular expression.\n";

#ifndef NUL
#define NUL '\0'
#endif

/**
 * Print usage and exit.  This is always a success procedure because
 * all command line arguments are presumed to be an error number
 * search expression.  This is only called when no arguments are provided.
 */
static void
usage(void)
{
    fwrite(usage_text, sizeof(usage_text) - 1, 1, stdout);
    exit(EXIT_SUCCESS);
}

/**
 * parse a number.  If there is any problem with the number, return -1.
 * The entire input string must be consumed.  If it looks like a number,
 * but is not a valid errno number, "print_errno()" will catch it and
 * return "false" to its caller.
 *
 * @param arg input string
 * @returns -1 or the number indicated by the string
 */
static int
parse_errno(char const * arg)
{
    errno = 0;
    char * end;
    unsigned long val = strtoul(arg, &end, 0);
    if ((errno != 0) || (*end != NUL)) return -1;
    return (int)val;
}

/**
 * print the error number, name and description.
 * The error number must be in the error name table.
 *
 * @param err  the error number
 * @param desc the error description
 * @returns false if the error number is invalid, otherwise true.
 */
static bool
print_errno(int err, char const * desc)
{
    if ((unsigned)err >= ERRNO_LIMIT)
        return false;

    char const * name = errno_names[err];
    if (name == NULL)
        return false;

    char buf[1024];
    int len = snprintf(buf, sizeof(buf), show_fmt, err, name, desc);
    char * ps = strchr(buf, '(');
    char * pe = strchr(ps,  ')');
    if ((ps != NULL) && (pe != NULL)) {
        for (;;) {
            ps += 2;
            if (ps >= pe)   break;
            if (*ps == ' ') *ps = '.';
        }
    }
    fwrite(buf, len, 1, stdout);
    return true;
}

static int
re_find_err(regex_t * re)
{
    int find_ct = 0;
    for (int err = 1; err < 255; err++) {
        char const * errstr = strerror(err);

        if (regexec(re, errstr, 0, NULL, 0) == 0) {
            find_ct++;
            print_errno(err, errstr);
        }
    }
    return find_ct;
}

static int
print_by_name(char const * ename)
{
    for (int idx = 1; idx < ERRNO_LIMIT; idx++) {
        char const * name = errno_names[idx];

        if ((name != NULL) && (strcmp(ename, name) == 0)) {
            char const * txt = strerror(idx);
            print_errno(idx, txt);
            return EXIT_SUCCESS;
        }
    }
    fprintf(stderr, "no error is named '%s'\n", ename);
    return EXIT_FAILURE;
}

static int
find_errno(char const * arg)
{
    static char const not_name_chars[] =
        " \t.,;/abcdefghijklmnopqrstuvwxyz";

    if ((*arg == 'E') && (strpbrk(arg, not_name_chars) == NULL))
        return print_by_name(arg);

    regex_t re;
    int     err = regcomp(&re, arg, 0);
    int     match_ct = 0;

    printf("looking for matches to:  '%s'\n", arg);

    if (err ==  0) {
        match_ct = re_find_err(&re);
        regfree(&re);

    } else {
        for (err = 1; err < 255; err++) {
            char const * errstr = strerror(err);

            if (strstr(errstr, arg) != NULL) {
                match_ct++;
                print_errno(err, errstr);
            }
        }
    }

    if (match_ct == 0) {
        fprintf(stderr, "no error descriptions match '%s'\n\n", arg);
        return EXIT_FAILURE;
    }

    putc('\n', stdout);
    return EXIT_SUCCESS;
}

int
main(int argc, char ** argv)
{
    int res = EXIT_SUCCESS;
    if (argc <= 1)
        usage();

    while (--argc > 0) {
        char * arg = *++argv;
        int    err = parse_errno(arg);
        if (err < 0)
            res |= find_errno(arg);

        else if (! print_errno(err, strerror(err))) {
            res |= EXIT_FAILURE;
            fprintf(stderr, "error number %d is invalid\n", err);
        }
    }

    return res;
}
/*
 * Local Variables:
 * mode: C
 * c-file-style: "stroustrup"
 * indent-tabs-mode: nil
 * End:
 * end of agen5/autogen.c */
# errno Makefile
#
# $Id: Makefile,v 1.2 2003/04/12 18:09:00 bkorb Exp $

EXE    := errno
CC     := $(shell which cc) -std=c99
SRC    := $(EXE).c
HDR    := err-names.h
OBJ    := $(SRC:.c=.o)
CFLAGS =  -O4 -g

$(EXE) : $(OBJ)
        $(CC) -g -o $@ $(OBJ)

$(OBJ) : $(SRC) $(HDR)
        $(CC) $(CFLAGS) -c -o $@ $(SRC)

err-names.h : mk-err-names.sh
        $(SHELL) mk-err-names.sh

clean :
        rm -f $(OBJ) *~ './.#'*

clobber : clean
        rm -f $(EXE) $(HDR)

Attachment: mk-err-names.sh
Description: application/shellscript

/* -*- buffer-read-only: t -*- vi: set ro:
 *
 * DO NOT EDIT THIS FILE
 *
 * This file has been derived by scanning system headers:
 *
 * /usr/include/asm-generic/errno-base.h
 * /usr/include/asm-generic/errno.h
 * /usr/include/asm/errno.h
 * /usr/include/bits/errno.h
 * /usr/include/errno.h
 * /usr/include/linux/errno.h
 *
 *  @file errno.c
 *
 *  Time-stamp:        "2011-08-02 14:04:06 bkorb"
 *
 *  Print out error numbers and meanings
 *
 *  Copyright (c) 2002-2011 Bruce Korb - all rights reserved
 *
 * errno is free software: you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the
 * Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * errno is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 * See the GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program.  If not, see <http://www.gnu.org/licenses/>.
 */
#ifndef ERRNO_LIMIT
#define ERRNO_LIMIT 133

static char const * const errno_names[ERRNO_LIMIT] = {
    [  1] = "EPERM",
    [  2] = "ENOENT",
    [  3] = "ESRCH",
    [  4] = "EINTR",
    [  5] = "EIO",
    [  6] = "ENXIO",
    [  7] = "E2BIG",
    [  8] = "ENOEXEC",
    [  9] = "EBADF",
    [ 10] = "ECHILD",
    [ 11] = "EAGAIN",
    [ 12] = "ENOMEM",
    [ 13] = "EACCES",
    [ 14] = "EFAULT",
    [ 15] = "ENOTBLK",
    [ 16] = "EBUSY",
    [ 17] = "EEXIST",
    [ 18] = "EXDEV",
    [ 19] = "ENODEV",
    [ 20] = "ENOTDIR",
    [ 21] = "EISDIR",
    [ 22] = "EINVAL",
    [ 23] = "ENFILE",
    [ 24] = "EMFILE",
    [ 25] = "ENOTTY",
    [ 26] = "ETXTBSY",
    [ 27] = "EFBIG",
    [ 28] = "ENOSPC",
    [ 29] = "ESPIPE",
    [ 30] = "EROFS",
    [ 31] = "EMLINK",
    [ 32] = "EPIPE",
    [ 33] = "EDOM",
    [ 34] = "ERANGE",
    [ 35] = "EDEADLK",
    [ 36] = "ENAMETOOLONG",
    [ 37] = "ENOLCK",
    [ 38] = "ENOSYS",
    [ 39] = "ENOTEMPTY",
    [ 40] = "ELOOP",
    [ 42] = "ENOMSG",
    [ 43] = "EIDRM",
    [ 44] = "ECHRNG",
    [ 45] = "EL2NSYNC",
    [ 46] = "EL3HLT",
    [ 47] = "EL3RST",
    [ 48] = "ELNRNG",
    [ 49] = "EUNATCH",
    [ 50] = "ENOCSI",
    [ 51] = "EL2HLT",
    [ 52] = "EBADE",
    [ 53] = "EBADR",
    [ 54] = "EXFULL",
    [ 55] = "ENOANO",
    [ 56] = "EBADRQC",
    [ 57] = "EBADSLT",
    [ 59] = "EBFONT",
    [ 60] = "ENOSTR",
    [ 61] = "ENODATA",
    [ 62] = "ETIME",
    [ 63] = "ENOSR",
    [ 64] = "ENONET",
    [ 65] = "ENOPKG",
    [ 66] = "EREMOTE",
    [ 67] = "ENOLINK",
    [ 68] = "EADV",
    [ 69] = "ESRMNT",
    [ 70] = "ECOMM",
    [ 71] = "EPROTO",
    [ 72] = "EMULTIHOP",
    [ 73] = "EDOTDOT",
    [ 74] = "EBADMSG",
    [ 75] = "EOVERFLOW",
    [ 76] = "ENOTUNIQ",
    [ 77] = "EBADFD",
    [ 78] = "EREMCHG",
    [ 79] = "ELIBACC",
    [ 80] = "ELIBBAD",
    [ 81] = "ELIBSCN",
    [ 82] = "ELIBMAX",
    [ 83] = "ELIBEXEC",
    [ 84] = "EILSEQ",
    [ 85] = "ERESTART",
    [ 86] = "ESTRPIPE",
    [ 87] = "EUSERS",
    [ 88] = "ENOTSOCK",
    [ 89] = "EDESTADDRREQ",
    [ 90] = "EMSGSIZE",
    [ 91] = "EPROTOTYPE",
    [ 92] = "ENOPROTOOPT",
    [ 93] = "EPROTONOSUPPORT",
    [ 94] = "ESOCKTNOSUPPORT",
    [ 95] = "EOPNOTSUPP",
    [ 96] = "EPFNOSUPPORT",
    [ 97] = "EAFNOSUPPORT",
    [ 98] = "EADDRINUSE",
    [ 99] = "EADDRNOTAVAIL",
    [100] = "ENETDOWN",
    [101] = "ENETUNREACH",
    [102] = "ENETRESET",
    [103] = "ECONNABORTED",
    [104] = "ECONNRESET",
    [105] = "ENOBUFS",
    [106] = "EISCONN",
    [107] = "ENOTCONN",
    [108] = "ESHUTDOWN",
    [109] = "ETOOMANYREFS",
    [110] = "ETIMEDOUT",
    [111] = "ECONNREFUSED",
    [112] = "EHOSTDOWN",
    [113] = "EHOSTUNREACH",
    [114] = "EALREADY",
    [115] = "EINPROGRESS",
    [116] = "ESTALE",
    [117] = "EUCLEAN",
    [118] = "ENOTNAM",
    [119] = "ENAVAIL",
    [120] = "EISNAM",
    [121] = "EREMOTEIO",
    [122] = "EDQUOT",
    [123] = "ENOMEDIUM",
    [124] = "EMEDIUMTYPE",
    [125] = "ECANCELED",
    [126] = "ENOKEY",
    [127] = "EKEYEXPIRED",
    [128] = "EKEYREVOKED",
    [129] = "EKEYREJECTED",
    [130] = "EOWNERDEAD",
    [131] = "ENOTRECOVERABLE",
    [132] = "ERFKILL",
};

static char const show_fmt[] = "%5d (%-15s) == %s\n";

#endif /* ERRNO_LIMIT */

Reply via email to