diff options
author | Nick Alcock <nick.alcock@oracle.com> | 2019-04-23 18:55:27 +0100 |
---|---|---|
committer | Nick Alcock <nick.alcock@oracle.com> | 2019-05-28 17:07:15 +0100 |
commit | 60da9d955964759b1f52690bff587ad32a198507 (patch) | |
tree | e3d9dd8f6667916ef29c557e62c2d0f3fba6f6c0 /libctf/ctf-subr.c | |
parent | 2e94b05630514109994abb77ee2dae730f17c263 (diff) | |
download | gdb-60da9d955964759b1f52690bff587ad32a198507.zip gdb-60da9d955964759b1f52690bff587ad32a198507.tar.gz gdb-60da9d955964759b1f52690bff587ad32a198507.tar.bz2 |
libctf: lowest-level memory allocation and debug-dumping wrappers
The memory-allocation wrappers are simple things to allow malloc
interposition: they are only used inconsistently at present, usually
where malloc debugging was required in the past.
These provide a default implementation that is environment-variable
triggered (initialized on the first call to the libctf creation and
file-opening functions, the first functions people will use), and
a ctf_setdebug()/ctf_getdebug() pair that allows the caller to
explicitly turn debugging off and on. If ctf_setdebug() is called,
the automatic setting from an environment variable is skipped.
libctf/
* ctf-impl.h: New file.
* ctf-subr.c: New file.
include/
* ctf-api.h (ctf_setdebug): New.
(ctf_getdebug): Likewise.
Diffstat (limited to 'libctf/ctf-subr.c')
-rw-r--r-- | libctf/ctf-subr.c | 232 |
1 files changed, 232 insertions, 0 deletions
diff --git a/libctf/ctf-subr.c b/libctf/ctf-subr.c new file mode 100644 index 0000000..3103e28 --- /dev/null +++ b/libctf/ctf-subr.c @@ -0,0 +1,232 @@ +/* Simple subrs. + Copyright (C) 2019 Free Software Foundation, Inc. + + This file is part of libctf. + + libctf 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, or (at your option) any later + version. + + This program 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; see the file COPYING. If not see + <http://www.gnu.org/licenses/>. */ + +#include <ctf-impl.h> +#ifdef HAVE_MMAP +#include <sys/mman.h> +#endif +#include <sys/types.h> +#include <stdarg.h> +#include <string.h> +#include <unistd.h> + +static size_t _PAGESIZE _libctf_unused_; +int _libctf_debug = 0; /* Debugging messages enabled. */ + +_libctf_malloc_ void * +ctf_data_alloc (size_t size) +{ + void *ret; + +#ifdef HAVE_MMAP + if (_PAGESIZE == 0) + _PAGESIZE = sysconf(_SC_PAGESIZE); + + if (size > _PAGESIZE) + { + ret = mmap (NULL, size, PROT_READ | PROT_WRITE, + MAP_PRIVATE | MAP_ANON, -1, 0); + if (ret == MAP_FAILED) + ret = NULL; + } + else + ret = calloc (1, size); +#else + ret = calloc (1, size); +#endif + return ret; +} + +void +ctf_data_free (void *buf, size_t size _libctf_unused_) +{ +#ifdef HAVE_MMAP + /* Must be the same as the check in ctf_data_alloc(). */ + + if (size > _PAGESIZE) + (void) munmap (buf, size); + else + free (buf); +#else + free (buf); +#endif +} + +/* Private, read-only mmap from a file, with fallback to copying. + + No handling of page-offset issues at all: the caller must allow for that. */ + +_libctf_malloc_ void * +ctf_mmap (size_t length, size_t offset, int fd) +{ + void *data; + +#ifdef HAVE_MMAP + data = mmap (NULL, length, PROT_READ, MAP_PRIVATE, fd, offset); + if (data == MAP_FAILED) + data = NULL; +#else + if ((data = malloc (length)) != NULL) + { + if (ctf_pread (fd, data, length, offset) <= 0) + { + free (data); + data = NULL; + } + } +#endif + return data; +} + +void +ctf_munmap (void *buf, size_t length _libctf_unused_) +{ +#ifdef HAVE_MMAP + (void) munmap (buf, length); +#else + free (buf); +#endif +} + +void +ctf_data_protect (void *buf, size_t size) +{ +#ifdef HAVE_MMAP + /* Must be the same as the check in ctf_data_alloc(). */ + + if (size > _PAGESIZE) + (void) mprotect (buf, size, PROT_READ); +#endif +} + +_libctf_malloc_ void * +ctf_alloc (size_t size) +{ + return (malloc (size)); +} + +void +ctf_free (void *buf) +{ + free (buf); +} + +ssize_t +ctf_pread (int fd, void *buf, ssize_t count, off_t offset) +{ + ssize_t len; + size_t acc = 0; + char *data = (char *) buf; + +#ifdef HAVE_PREAD + while (count > 0) + { + errno = 0; + if (((len = pread (fd, data, count, offset)) < 0) && + errno != EINTR) + return len; + if (errno == EINTR) + continue; + + acc += len; + if (len == 0) /* EOF. */ + return acc; + + count -= len; + offset += len; + data += len; + } + return acc; +#else + off_t orig_off; + + if ((orig_off = lseek (fd, 0, SEEK_CUR)) < 0) + return -1; + if ((lseek (fd, offset, SEEK_SET)) < 0) + return -1; + + while (count > 0) + { + errno = 0; + if (((len = read (fd, data, count)) < 0) && + errno != EINTR) + return len; + if (errno == EINTR) + continue; + + acc += len; + if (len == 0) /* EOF. */ + break; + + count -= len; + data += len; + } + if ((lseek (fd, orig_off, SEEK_SET)) < 0) + return -1; /* offset is smashed. */ +#endif + + return acc; +} + +const char * +ctf_strerror (int err) +{ + return (const char *) (strerror (err)); +} + +void +libctf_init_debug (void) +{ + static int inited; + if (!inited) + { + _libctf_debug = getenv ("LIBCTF_DEBUG") != NULL; + inited = 1; + } +} + +void ctf_setdebug (int debug) +{ + /* Ensure that libctf_init_debug() has been called, so that we don't get our + debugging-on-or-off smashed by the next call. */ + + libctf_init_debug(); + _libctf_debug = debug; + ctf_dprintf ("CTF debugging set to %i\n", debug); +} + +int ctf_getdebug (void) +{ + return _libctf_debug; +} + +_libctf_printflike_ (1, 2) +void ctf_dprintf (const char *format, ...) +{ + if (_libctf_debug) + { + va_list alist; + + va_start (alist, format); + fflush (stdout); + (void) fputs ("libctf DEBUG: ", stderr); + (void) vfprintf (stderr, format, alist); + va_end (alist); + } +} |