libctf, ld, binutils: add textual error/warning reporting for libctf
[deliverable/binutils-gdb.git] / libctf / ctf-subr.c
CommitLineData
60da9d95 1/* Simple subrs.
b3adc24a 2 Copyright (C) 2019-2020 Free Software Foundation, Inc.
60da9d95
NA
3
4 This file is part of libctf.
5
6 libctf is free software; you can redistribute it and/or modify it under
7 the terms of the GNU General Public License as published by the Free
8 Software Foundation; either version 3, or (at your option) any later
9 version.
10
11 This program is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
14 See the GNU General Public License for more details.
15
16 You should have received a copy of the GNU General Public License
17 along with this program; see the file COPYING. If not see
18 <http://www.gnu.org/licenses/>. */
19
20#include <ctf-impl.h>
21#ifdef HAVE_MMAP
22#include <sys/mman.h>
23#endif
24#include <sys/types.h>
25#include <stdarg.h>
26#include <string.h>
27#include <unistd.h>
28
6c33b742 29int _libctf_version = CTF_VERSION; /* Library client version. */
60da9d95
NA
30int _libctf_debug = 0; /* Debugging messages enabled. */
31
60da9d95
NA
32/* Private, read-only mmap from a file, with fallback to copying.
33
34 No handling of page-offset issues at all: the caller must allow for that. */
35
36_libctf_malloc_ void *
37ctf_mmap (size_t length, size_t offset, int fd)
38{
39 void *data;
40
41#ifdef HAVE_MMAP
42 data = mmap (NULL, length, PROT_READ, MAP_PRIVATE, fd, offset);
43 if (data == MAP_FAILED)
44 data = NULL;
45#else
46 if ((data = malloc (length)) != NULL)
47 {
48 if (ctf_pread (fd, data, length, offset) <= 0)
49 {
50 free (data);
51 data = NULL;
52 }
53 }
54#endif
55 return data;
56}
57
58void
59ctf_munmap (void *buf, size_t length _libctf_unused_)
60{
61#ifdef HAVE_MMAP
62 (void) munmap (buf, length);
63#else
64 free (buf);
65#endif
66}
67
60da9d95
NA
68ssize_t
69ctf_pread (int fd, void *buf, ssize_t count, off_t offset)
70{
71 ssize_t len;
72 size_t acc = 0;
73 char *data = (char *) buf;
74
75#ifdef HAVE_PREAD
76 while (count > 0)
77 {
78 errno = 0;
79 if (((len = pread (fd, data, count, offset)) < 0) &&
80 errno != EINTR)
81 return len;
82 if (errno == EINTR)
83 continue;
84
85 acc += len;
86 if (len == 0) /* EOF. */
87 return acc;
88
89 count -= len;
90 offset += len;
91 data += len;
92 }
93 return acc;
94#else
95 off_t orig_off;
96
97 if ((orig_off = lseek (fd, 0, SEEK_CUR)) < 0)
98 return -1;
99 if ((lseek (fd, offset, SEEK_SET)) < 0)
100 return -1;
101
102 while (count > 0)
103 {
104 errno = 0;
105 if (((len = read (fd, data, count)) < 0) &&
106 errno != EINTR)
107 return len;
108 if (errno == EINTR)
109 continue;
110
111 acc += len;
112 if (len == 0) /* EOF. */
113 break;
114
115 count -= len;
116 data += len;
117 }
118 if ((lseek (fd, orig_off, SEEK_SET)) < 0)
119 return -1; /* offset is smashed. */
120#endif
121
122 return acc;
123}
124
125const char *
126ctf_strerror (int err)
127{
128 return (const char *) (strerror (err));
129}
130
6c33b742
NA
131/* Set the CTF library client version to the specified version. If version is
132 zero, we just return the default library version number. */
133int
134ctf_version (int version)
135{
136 if (version < 0)
137 {
138 errno = EINVAL;
139 return -1;
140 }
141
142 if (version > 0)
143 {
144 /* Dynamic version switching is not presently supported. */
145 if (version != CTF_VERSION)
146 {
147 errno = ENOTSUP;
148 return -1;
149 }
150 ctf_dprintf ("ctf_version: client using version %d\n", version);
151 _libctf_version = version;
152 }
153
154 return _libctf_version;
155}
156
60da9d95
NA
157void
158libctf_init_debug (void)
159{
160 static int inited;
161 if (!inited)
162 {
163 _libctf_debug = getenv ("LIBCTF_DEBUG") != NULL;
164 inited = 1;
165 }
166}
167
168void ctf_setdebug (int debug)
169{
170 /* Ensure that libctf_init_debug() has been called, so that we don't get our
171 debugging-on-or-off smashed by the next call. */
172
173 libctf_init_debug();
174 _libctf_debug = debug;
175 ctf_dprintf ("CTF debugging set to %i\n", debug);
176}
177
178int ctf_getdebug (void)
179{
180 return _libctf_debug;
181}
182
183_libctf_printflike_ (1, 2)
184void ctf_dprintf (const char *format, ...)
185{
5ec7465f 186 if (_libctf_unlikely_ (_libctf_debug))
60da9d95
NA
187 {
188 va_list alist;
189
190 va_start (alist, format);
191 fflush (stdout);
192 (void) fputs ("libctf DEBUG: ", stderr);
193 (void) vfprintf (stderr, format, alist);
194 va_end (alist);
195 }
196}
8b37e7b6
NA
197
198/* Errors and warnings. */
199_libctf_printflike_ (3, 4)
200extern void
201ctf_err_warn (ctf_file_t *fp, int is_warning, const char *format, ...)
202{
203 va_list alist;
204 ctf_err_warning_t *cew;
205
206 /* Don't bother reporting errors here: we can't do much about them if they
207 happen. If we're so short of memory that a tiny malloc doesn't work, a
208 vfprintf isn't going to work either and the caller will have to rely on the
209 ENOMEM return they'll be getting in short order anyway. */
210
211 if ((cew = malloc (sizeof (ctf_err_warning_t))) == NULL)
212 return;
213
214 cew->cew_is_warning = is_warning;
215 va_start (alist, format);
216 if (vasprintf (&cew->cew_text, format, alist) < 0)
217 {
218 free (cew);
219 va_end (alist);
220 return;
221 }
222 va_end (alist);
223
224 ctf_dprintf ("%s: %s\n", is_warning ? "error" : "warning", cew->cew_text);
225
226 ctf_list_append (&fp->ctf_errs_warnings, cew);
227}
228
229/* Error-warning reporting: an 'iterator' that returns errors and warnings from
230 the error/warning list, in order of emission. Errors and warnings are popped
231 after return: the caller must free the returned error-text pointer. */
232char *
233ctf_errwarning_next (ctf_file_t *fp, ctf_next_t **it, int *is_warning)
234{
235 ctf_next_t *i = *it;
236 char *ret;
237 ctf_err_warning_t *cew;
238
239 if (!i)
240 {
241 if ((i = ctf_next_create ()) == NULL)
242 {
243 ctf_set_errno (fp, ENOMEM);
244 return NULL;
245 }
246
247 i->cu.ctn_fp = fp;
248 i->ctn_iter_fun = (void (*) (void)) ctf_errwarning_next;
249 *it = i;
250 }
251
252 if ((void (*) (void)) ctf_errwarning_next != i->ctn_iter_fun)
253 {
254 ctf_set_errno (fp, ECTF_NEXT_WRONGFUN);
255 return NULL;
256 }
257
258 if (fp != i->cu.ctn_fp)
259 {
260 ctf_set_errno (fp, ECTF_NEXT_WRONGFP);
261 return NULL;
262 }
263
264 cew = ctf_list_next (&fp->ctf_errs_warnings);
265
266 if (!cew)
267 {
268 ctf_next_destroy (i);
269 *it = NULL;
270 ctf_set_errno (fp, ECTF_NEXT_END);
271 return NULL;
272 }
273
274 if (is_warning)
275 *is_warning = cew->cew_is_warning;
276 ret = cew->cew_text;
277 ctf_list_delete (&fp->ctf_errs_warnings, cew);
278 free (cew);
279 return ret;
280}
281
282void
283ctf_assert_fail_internal (ctf_file_t *fp, const char *file, size_t line,
284 const char *exprstr)
285{
286 ctf_err_warn (fp, 0, "%s: %lu: libctf assertion failed: %s", file,
287 (long unsigned int) line, exprstr);
288 ctf_set_errno (fp, ECTF_INTERNAL);
289}
This page took 0.092425 seconds and 4 git commands to generate.