1/*
2 * regerror - error-code expansion
3 *
4 * Copyright (c) 1998, 1999 Henry Spencer. All rights reserved.
5 *
6 * Development of this software was funded, in part, by Cray Research Inc.,
7 * UUNET Communications Services Inc., Sun Microsystems Inc., and Scriptics
8 * Corporation, none of whom are responsible for the results. The author
9 * thanks all of them.
10 *
11 * Redistribution and use in source and binary forms -- with or without
12 * modification -- are permitted for any purpose, provided that
13 * redistributions in source form retain this entire copyright notice and
14 * indicate the origin and nature of any modifications.
15 *
16 * I'd appreciate being given credit for this package in the documentation
17 * of software which uses it, but that is not a requirement.
18 *
19 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
20 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
21 * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
22 * HENRY SPENCER BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
23 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
24 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
25 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
26 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
27 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
28 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 *
30 * src/backend/regex/regerror.c
31 *
32 */
33
34#include "regex/regguts.h"
35
36/* unknown-error explanation */
37static const char unk[] = "*** unknown regex error code 0x%x ***";
38
39/* struct to map among codes, code names, and explanations */
40static const struct rerr
41{
42 int code;
43 const char *name;
44 const char *explain;
45} rerrs[] =
46
47{
48 /* the actual table is built from regex.h */
49#include "regex/regerrs.h" /* pgrminclude ignore */
50 {
51 -1, "", "oops"
52 }, /* explanation special-cased in code */
53};
54
55/*
56 * pg_regerror - the interface to error numbers
57 */
58/* ARGSUSED */
59size_t /* actual space needed (including NUL) */
60pg_regerror(int errcode, /* error code, or REG_ATOI or REG_ITOA */
61 const regex_t *preg, /* associated regex_t (unused at present) */
62 char *errbuf, /* result buffer (unless errbuf_size==0) */
63 size_t errbuf_size) /* available space in errbuf, can be 0 */
64{
65 const struct rerr *r;
66 const char *msg;
67 char convbuf[sizeof(unk) + 50]; /* 50 = plenty for int */
68 size_t len;
69 int icode;
70
71 switch (errcode)
72 {
73 case REG_ATOI: /* convert name to number */
74 for (r = rerrs; r->code >= 0; r++)
75 if (strcmp(r->name, errbuf) == 0)
76 break;
77 sprintf(convbuf, "%d", r->code); /* -1 for unknown */
78 msg = convbuf;
79 break;
80 case REG_ITOA: /* convert number to name */
81 icode = atoi(errbuf); /* not our problem if this fails */
82 for (r = rerrs; r->code >= 0; r++)
83 if (r->code == icode)
84 break;
85 if (r->code >= 0)
86 msg = r->name;
87 else
88 { /* unknown; tell him the number */
89 sprintf(convbuf, "REG_%u", (unsigned) icode);
90 msg = convbuf;
91 }
92 break;
93 default: /* a real, normal error code */
94 for (r = rerrs; r->code >= 0; r++)
95 if (r->code == errcode)
96 break;
97 if (r->code >= 0)
98 msg = r->explain;
99 else
100 { /* unknown; say so */
101 sprintf(convbuf, unk, errcode);
102 msg = convbuf;
103 }
104 break;
105 }
106
107 len = strlen(msg) + 1; /* space needed, including NUL */
108 if (errbuf_size > 0)
109 {
110 if (errbuf_size > len)
111 strcpy(errbuf, msg);
112 else
113 { /* truncate to fit */
114 memcpy(errbuf, msg, errbuf_size - 1);
115 errbuf[errbuf_size - 1] = '\0';
116 }
117 }
118
119 return len;
120}
121