Improved memory allocation/error checking helper functions. Also
reworked stderr so that it prints to the console on exit.
This commit is contained in:
parent
fe75508665
commit
783f127199
2
Makefile
2
Makefile
@ -22,7 +22,7 @@ CWARN=-Wall -Wextra
|
||||
CFLAGS += '-DBB_NAME="$(NAME)"'
|
||||
OSFLAGS != case $$(uname -s) in *BSD|Darwin) echo '-D_BSD_SOURCE';; Linux) echo '-D_GNU_SOURCE';; *) echo '-D_DEFAULT_SOURCE';; esac
|
||||
|
||||
CFILES=draw.c terminal.c
|
||||
CFILES=draw.c terminal.c utils.c
|
||||
OBJFILES=$(CFILES:.c=.o)
|
||||
|
||||
all: $(NAME)
|
||||
|
162
bb.c
162
bb.c
@ -7,6 +7,7 @@
|
||||
//
|
||||
|
||||
#include <ctype.h>
|
||||
#include <err.h>
|
||||
#include <fcntl.h>
|
||||
#include <glob.h>
|
||||
#include <limits.h>
|
||||
@ -36,11 +37,6 @@
|
||||
#define SCROLLOFF MIN(5, (winsize.ws_row-4)/2)
|
||||
#define ONSCREEN (winsize.ws_row - 3)
|
||||
|
||||
#define new(t) memcheck(calloc(1, sizeof(t)))
|
||||
#define xcalloc(a,b) memcheck(calloc(a,b))
|
||||
#define xrealloc(a,b) memcheck(realloc(a,b))
|
||||
#define clean_err(...) do { cleanup(); err(1, __VA_ARGS__); } while (0)
|
||||
|
||||
// Functions
|
||||
void bb_browse(bb_t *bb, int argc, char *argv[]);
|
||||
static void check_cmdfile(bb_t *bb);
|
||||
@ -54,7 +50,6 @@ static void init_term(void);
|
||||
static int is_simple_bbcmd(const char *s);
|
||||
static entry_t* load_entry(bb_t *bb, const char *path);
|
||||
static int matches_cmd(const char *str, const char *cmd);
|
||||
static void* memcheck(void *p);
|
||||
static char* normalize_path(const char *path, char *pbuf);
|
||||
static int populate_files(bb_t *bb, const char *path);
|
||||
static void print_bindings(int fd);
|
||||
@ -91,6 +86,10 @@ static struct winsize winsize = {0};
|
||||
static char cmdfilename[PATH_MAX] = {0};
|
||||
static bb_t *current_bb = NULL;
|
||||
|
||||
static char err_tmpfilename[PATH_MAX] = {0};
|
||||
int stderr_dup_fd = -1, err_tmp_fd = -1;
|
||||
FILE *err_file = NULL;
|
||||
|
||||
//
|
||||
// Use bb to browse the filesystem.
|
||||
//
|
||||
@ -112,8 +111,7 @@ void bb_browse(bb_t *bb, int argc, char *argv[])
|
||||
|
||||
struct stat path_stat;
|
||||
const char *goto_file = NULL;
|
||||
if (stat(full_initial_path, &path_stat) != 0)
|
||||
clean_err("Could not find initial path: \"%s\"", initial_path);
|
||||
nonnegative(stat(full_initial_path, &path_stat), "Could not find initial path: \"%s\"", initial_path);
|
||||
if (!S_ISDIR(path_stat.st_mode)) {
|
||||
char *slash = strrchr(full_initial_path, '/');
|
||||
*slash = '\0';
|
||||
@ -121,12 +119,12 @@ void bb_browse(bb_t *bb, int argc, char *argv[])
|
||||
}
|
||||
|
||||
if (populate_files(bb, full_initial_path))
|
||||
clean_err("Could not find initial path: \"%s\"", full_initial_path);
|
||||
errx(EXIT_FAILURE, "Could not find initial path: \"%s\"", full_initial_path);
|
||||
|
||||
// Emergency fallback:
|
||||
bindings[0].key = KEY_CTRL_C;
|
||||
bindings[0].script = strdup("kill -INT $PPID");
|
||||
bindings[0].description = strdup("Kill the bb process");
|
||||
bindings[0].script = check_strdup("kill -INT $PPID");
|
||||
bindings[0].description = check_strdup("Kill the bb process");
|
||||
run_script(bb, "bbstartup");
|
||||
|
||||
FILE *cmdfile = fopen(cmdfilename, "a");
|
||||
@ -170,7 +168,7 @@ static void check_cmdfile(bb_t *bb)
|
||||
run_bbcmd(bb, cmd);
|
||||
if (bb->should_quit) break;
|
||||
}
|
||||
free(cmd);
|
||||
delete(&cmd);
|
||||
fclose(cmdfile);
|
||||
unlink(cmdfilename);
|
||||
}
|
||||
@ -209,6 +207,17 @@ static void cleanup(void)
|
||||
fflush(tty_out);
|
||||
tcsetattr(fileno(tty_out), TCSANOW, &orig_termios);
|
||||
}
|
||||
if (err_tmp_fd != -1) {
|
||||
fflush(stderr);
|
||||
dup2(stderr_dup_fd, STDERR_FILENO); // Put stderr back to normal
|
||||
char buf[256];
|
||||
lseek(err_tmp_fd, 0, SEEK_SET);
|
||||
for (ssize_t len; (len = read(err_tmp_fd, buf, sizeof(buf))) > 0; )
|
||||
write(STDERR_FILENO, buf, len);
|
||||
close(err_tmp_fd);
|
||||
err_tmp_fd = stderr_dup_fd = -1;
|
||||
unlink(err_tmpfilename);
|
||||
}
|
||||
}
|
||||
|
||||
//
|
||||
@ -359,8 +368,7 @@ static void handle_next_key_binding(bb_t *bb)
|
||||
//
|
||||
static void init_term(void)
|
||||
{
|
||||
if (tcsetattr(fileno(tty_out), TCSANOW, &bb_termios) == -1)
|
||||
clean_err("Couldn't tcsetattr");
|
||||
nonnegative(tcsetattr(fileno(tty_out), TCSANOW, &bb_termios));
|
||||
update_term_size(0);
|
||||
// Initiate mouse tracking and disable text wrapping:
|
||||
fputs(T_ENTER_BBMODE, tty_out);
|
||||
@ -415,15 +423,14 @@ static entry_t* load_entry(bb_t *bb, const char *path)
|
||||
ssize_t linkpathlen = -1;
|
||||
char linkbuf[PATH_MAX];
|
||||
if (S_ISLNK(filestat.st_mode)) {
|
||||
linkpathlen = readlink(pbuf, linkbuf, sizeof(linkbuf));
|
||||
if (linkpathlen < 0) clean_err("Couldn't read link: '%s'", pbuf);
|
||||
linkpathlen = nonnegative(readlink(pbuf, linkbuf, sizeof(linkbuf)), "Couldn't read link: '%s'", pbuf);
|
||||
linkbuf[linkpathlen] = '\0';
|
||||
while (linkpathlen > 0 && linkbuf[linkpathlen-1] == '/') linkbuf[--linkpathlen] = '\0';
|
||||
if (stat(pbuf, &linkedstat) == -1) memset(&linkedstat, 0, sizeof(linkedstat));
|
||||
}
|
||||
size_t pathlen = strlen(pbuf);
|
||||
size_t entry_size = sizeof(entry_t) + (pathlen + 1) + (size_t)(linkpathlen + 1);
|
||||
entry_t *entry = xcalloc(entry_size, 1);
|
||||
entry_t *entry = new_bytes(entry_size);
|
||||
char *end = stpcpy(entry->fullname, pbuf);
|
||||
if (linkpathlen >= 0)
|
||||
entry->linkname = strcpy(end + 1, linkbuf);
|
||||
@ -453,16 +460,6 @@ static int matches_cmd(const char *str, const char *cmd)
|
||||
return *str == '\0' || *str == ':';
|
||||
}
|
||||
|
||||
//
|
||||
// Memory allocation failures are unrecoverable in bb, so this wrapper just
|
||||
// prints an error message and exits if that happens.
|
||||
//
|
||||
static void* memcheck(void *p)
|
||||
{
|
||||
if (!p) clean_err("Allocation failure");
|
||||
return p;
|
||||
}
|
||||
|
||||
//
|
||||
// Prepend `./` to relative paths, replace "~" with $HOME.
|
||||
// The normalized path is stored in `normalized`.
|
||||
@ -532,7 +529,7 @@ static int populate_files(bb_t *bb, const char *path)
|
||||
if (clear_future_history && !samedir) {
|
||||
for (bb_history_t *next, *h = bb->history ? bb->history->next : NULL; h; h = next) {
|
||||
next = h->next;
|
||||
free(h);
|
||||
delete(&h);
|
||||
}
|
||||
|
||||
bb_history_t *h = new(bb_history_t);
|
||||
@ -553,8 +550,7 @@ static int populate_files(bb_t *bb, const char *path)
|
||||
try_free_entry(bb->files[i]);
|
||||
bb->files[i] = NULL;
|
||||
}
|
||||
free(bb->files);
|
||||
bb->files = NULL;
|
||||
delete(&bb->files);
|
||||
}
|
||||
bb->nfiles = 0;
|
||||
bb->cursor = 0;
|
||||
@ -565,10 +561,10 @@ static int populate_files(bb_t *bb, const char *path)
|
||||
|
||||
size_t space = 0;
|
||||
glob_t globbuf = {0};
|
||||
char *pat, *tmpglob = memcheck(strdup(bb->globpats));
|
||||
char *pat, *tmpglob = check_strdup(bb->globpats);
|
||||
while ((pat = strsep(&tmpglob, " ")) != NULL)
|
||||
glob(pat, GLOB_NOSORT|GLOB_APPEND, NULL, &globbuf);
|
||||
free(tmpglob);
|
||||
delete(&tmpglob);
|
||||
for (size_t i = 0; i < globbuf.gl_pathc; i++) {
|
||||
// Don't normalize path so we can get "." and ".."
|
||||
entry_t *entry = load_entry(bb, globbuf.gl_pathv[i]);
|
||||
@ -578,7 +574,7 @@ static int populate_files(bb_t *bb, const char *path)
|
||||
}
|
||||
entry->index = bb->nfiles;
|
||||
if ((size_t)bb->nfiles + 1 > space)
|
||||
bb->files = xrealloc(bb->files, (space += 100)*sizeof(void*));
|
||||
bb->files = grow(bb->files, space += 100);
|
||||
bb->files[bb->nfiles++] = entry;
|
||||
}
|
||||
globfree(&globbuf);
|
||||
@ -657,12 +653,12 @@ static void run_bbcmd(bb_t *bb, const char *cmd)
|
||||
const char *value = strchr(cmd, ':');
|
||||
if (value) ++value;
|
||||
if (matches_cmd(cmd, "bind:")) { // +bind:<keys>:<script>
|
||||
char *value_copy = memcheck(strdup(value));
|
||||
char *value_copy = check_strdup(value);
|
||||
char *keys = trim(value_copy);
|
||||
if (!keys[0]) { free(value_copy); return; }
|
||||
if (!keys[0]) { delete(&value_copy); return; }
|
||||
char *script = strchr(keys+1, ':');
|
||||
if (!script) {
|
||||
free(value_copy);
|
||||
delete(&value_copy);
|
||||
flash_warn(bb, "No script provided.");
|
||||
return;
|
||||
}
|
||||
@ -683,16 +679,16 @@ static void run_bbcmd(bb_t *bb, const char *cmd)
|
||||
continue;
|
||||
char *script2;
|
||||
if (is_simple_bbcmd(script)) {
|
||||
script2 = memcheck(strdup(script));
|
||||
script2 = check_strdup(script);
|
||||
} else {
|
||||
const char *prefix = "set -e\n";
|
||||
script2 = xcalloc(strlen(prefix) + strlen(script) + 1, 1);
|
||||
script2 = new_bytes(strlen(prefix) + strlen(script) + 1);
|
||||
sprintf(script2, "%s%s", prefix, script);
|
||||
}
|
||||
binding_t binding = {keyval, script2, memcheck(strdup(description))};
|
||||
binding_t binding = {keyval, script2, check_strdup(description)};
|
||||
if (bindings[i].key == keyval) {
|
||||
free((char*)bindings[i].description);
|
||||
free((char*)bindings[i].script);
|
||||
delete((char**)&bindings[i].description);
|
||||
delete((char**)&bindings[i].script);
|
||||
for (; i + 1 < sizeof(bindings)/sizeof(bindings[0]) && bindings[i+1].key; i++)
|
||||
bindings[i] = bindings[i+1];
|
||||
}
|
||||
@ -700,7 +696,7 @@ static void run_bbcmd(bb_t *bb, const char *cmd)
|
||||
break;
|
||||
}
|
||||
}
|
||||
free(value_copy);
|
||||
delete(&value_copy);
|
||||
} else if (matches_cmd(cmd, "cd:")) { // +cd:
|
||||
if (populate_files(bb, value))
|
||||
flash_warn(bb, "Could not open directory: \"%s\"", value);
|
||||
@ -737,8 +733,7 @@ static void run_bbcmd(bb_t *bb, const char *cmd)
|
||||
fputs("\033[K", tty_out);
|
||||
restore_term(&orig_termios);
|
||||
signal(SIGTTOU, SIG_IGN);
|
||||
if (tcsetpgrp(fileno(tty_out), child->pid))
|
||||
clean_err("Couldn't set pgrp");
|
||||
nonnegative(tcsetpgrp(fileno(tty_out), child->pid));
|
||||
kill(-(child->pid), SIGCONT);
|
||||
wait_for_process(&child);
|
||||
signal(SIGTTOU, SIG_DFL);
|
||||
@ -758,7 +753,7 @@ static void run_bbcmd(bb_t *bb, const char *cmd)
|
||||
char pbuf[PATH_MAX];
|
||||
strcpy(pbuf, e->fullname);
|
||||
char *lastslash = strrchr(pbuf, '/');
|
||||
if (!lastslash) clean_err("No slash found in filename: %s", pbuf);
|
||||
if (!lastslash) errx(EXIT_FAILURE, "No slash found in filename: %s", pbuf);
|
||||
*lastslash = '\0'; // Split in two
|
||||
try_free_entry(e);
|
||||
// Move to dir and reselect
|
||||
@ -773,14 +768,13 @@ static void run_bbcmd(bb_t *bb, const char *cmd)
|
||||
} else if (matches_cmd(cmd, "help")) { // +help
|
||||
char filename[PATH_MAX];
|
||||
sprintf(filename, "%s/bbhelp.XXXXXX", getenv("TMPDIR"));
|
||||
int fd = mkstemp(filename);
|
||||
if (fd == -1) clean_err("Couldn't create temporary help file at %s", filename);
|
||||
int fd = nonnegative(mkstemp(filename), "Couldn't create temporary help file at %s", filename);
|
||||
print_bindings(fd);
|
||||
close(fd);
|
||||
char script[512] = "less -rKX < ";
|
||||
strcat(script, filename);
|
||||
run_script(bb, script);
|
||||
if (unlink(filename) == -1) clean_err("Couldn't delete temporary help file: '%s'", filename);
|
||||
nonnegative(unlink(filename), "Couldn't delete temporary help file: '%s'", filename);
|
||||
} else if (matches_cmd(cmd, "interleave:") || matches_cmd(cmd, "interleave")) { // +interleave
|
||||
bb->interleave_dirs = value ? (value[0] == '1') : !bb->interleave_dirs;
|
||||
set_interleave(bb, bb->interleave_dirs);
|
||||
@ -858,12 +852,11 @@ static int run_script(bb_t *bb, const char *cmd)
|
||||
{
|
||||
proc_t *proc = new(proc_t);
|
||||
signal(SIGTTOU, SIG_IGN);
|
||||
if ((proc->pid = fork()) == 0) {
|
||||
if ((proc->pid = nonnegative(fork())) == 0) {
|
||||
pid_t pgrp = getpid();
|
||||
(void)setpgid(0, pgrp);
|
||||
if (tcsetpgrp(STDIN_FILENO, pgrp))
|
||||
clean_err("Couldn't set pgrp");
|
||||
const char **args = xcalloc(4 + (size_t)bb->nselected + 1, sizeof(char*));
|
||||
nonnegative(tcsetpgrp(STDIN_FILENO, pgrp));
|
||||
const char **args = new(char*[4 + (size_t)bb->nselected + 1]);
|
||||
int i = 0;
|
||||
args[i++] = "sh";
|
||||
args[i++] = "-c";
|
||||
@ -880,13 +873,10 @@ static int run_script(bb_t *bb, const char *cmd)
|
||||
dup2(fileno(tty_out), STDOUT_FILENO);
|
||||
dup2(fileno(tty_in), STDIN_FILENO);
|
||||
execvp(args[0], (char**)args);
|
||||
clean_err("Failed to execute command: '%s'", cmd);
|
||||
err(EXIT_FAILURE, "Failed to execute command: '%s'", cmd);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (proc->pid == -1)
|
||||
clean_err("Failed to fork");
|
||||
|
||||
(void)setpgid(getpid(), getpid());
|
||||
LL_PREPEND(bb->running_procs, proc, running);
|
||||
int status = wait_for_process(&proc);
|
||||
@ -937,8 +927,8 @@ static void set_cursor(bb_t *bb, int newcur)
|
||||
//
|
||||
static void set_globs(bb_t *bb, const char *globs)
|
||||
{
|
||||
free(bb->globpats);
|
||||
bb->globpats = memcheck(strdup(globs));
|
||||
delete(&bb->globpats);
|
||||
bb->globpats = check_strdup(globs);
|
||||
setenv("BBGLOB", bb->globpats, 1);
|
||||
}
|
||||
|
||||
@ -1036,7 +1026,7 @@ static int try_free_entry(entry_t *e)
|
||||
{
|
||||
if (IS_SELECTED(e) || IS_VIEWED(e) || !IS_LOADED(e)) return 0;
|
||||
LL_REMOVE(e, hash);
|
||||
free(e);
|
||||
delete(&e);
|
||||
return 1;
|
||||
}
|
||||
|
||||
@ -1086,13 +1076,11 @@ static int wait_for_process(proc_t **proc)
|
||||
continue;
|
||||
if (WIFEXITED(status) || WIFSIGNALED(status)) {
|
||||
LL_REMOVE((*proc), running);
|
||||
free(*proc);
|
||||
*proc = NULL;
|
||||
delete(proc);
|
||||
} else if (WIFSTOPPED(status))
|
||||
break;
|
||||
}
|
||||
if (tcsetpgrp(fileno(tty_out), getpid()))
|
||||
clean_err("Failed to set pgrp");
|
||||
nonnegative(tcsetpgrp(fileno(tty_out), getpid()));
|
||||
signal(SIGTTOU, SIG_DFL);
|
||||
return status;
|
||||
}
|
||||
@ -1146,9 +1134,9 @@ int main(int argc, char *argv[])
|
||||
// Default values
|
||||
setenv("TMPDIR", "/tmp", 0);
|
||||
sprintf(cmdfilename, "%s/"BB_NAME".XXXXXX", getenv("TMPDIR"));
|
||||
int cmdfd;
|
||||
if ((cmdfd = mkostemp(cmdfilename, O_APPEND)) == -1)
|
||||
clean_err("Couldn't create "BB_NAME" command file: '%s'", cmdfilename);
|
||||
int cmdfd = nonnegative(
|
||||
mkostemp(cmdfilename, O_APPEND),
|
||||
"Couldn't create "BB_NAME" command file: '%s'", cmdfilename);
|
||||
close(cmdfd);
|
||||
setenv("BBCMD", cmdfilename, 1);
|
||||
char xdg_config_home[PATH_MAX], xdg_data_home[PATH_MAX];
|
||||
@ -1162,21 +1150,22 @@ int main(int argc, char *argv[])
|
||||
static char bbpath[PATH_MAX];
|
||||
// Hacky fix to allow `bb` to be run out of its build directory:
|
||||
if (strncmp(argv[0], "./", 2) == 0) {
|
||||
if (realpath(argv[0], bbpath) == NULL)
|
||||
clean_err("Could not resolve path: %s", bbpath);
|
||||
nonnull(realpath(argv[0], bbpath), "Could not resolve path: %s", bbpath);
|
||||
char *slash = strrchr(bbpath, '/');
|
||||
if (!slash) clean_err("No slash found in real path: %s", bbpath);
|
||||
if (!slash) errx(EXIT_FAILURE, "No slash found in real path: %s", bbpath);
|
||||
*slash = '\0';
|
||||
setenv("BBPATH", bbpath, 1);
|
||||
}
|
||||
if (getenv("BBPATH")) {
|
||||
if (asprintf(&newpath, "%s/"BB_NAME":%s/scripts:%s",
|
||||
getenv("XDG_CONFIG_HOME"), getenv("BBPATH"), getenv("PATH")) < 0)
|
||||
clean_err("Could not allocate memory for PATH");
|
||||
nonnegative(
|
||||
asprintf(&newpath, "%s/"BB_NAME":%s/scripts:%s",
|
||||
getenv("XDG_CONFIG_HOME"), getenv("BBPATH"), getenv("PATH")),
|
||||
"Could not allocate memory for PATH");
|
||||
} else {
|
||||
if (asprintf(&newpath, "%s/"BB_NAME":%s/"BB_NAME":%s",
|
||||
getenv("XDG_CONFIG_HOME"), getenv("sysconfdir"), getenv("PATH")) < 0)
|
||||
clean_err("Could not allocate memory for PATH");
|
||||
nonnegative(
|
||||
asprintf(&newpath, "%s/"BB_NAME":%s/"BB_NAME":%s",
|
||||
getenv("XDG_CONFIG_HOME"), getenv("sysconfdir"), getenv("PATH")),
|
||||
"Could not allocate memory for PATH");
|
||||
}
|
||||
setenv("PATH", newpath, 1);
|
||||
|
||||
@ -1188,17 +1177,21 @@ int main(int argc, char *argv[])
|
||||
sprintf(depthstr, "%d", depth + 1);
|
||||
setenv("BBDEPTH", depthstr, 1);
|
||||
|
||||
tty_in = fopen("/dev/tty", "r");
|
||||
if (!tty_in) clean_err("Couldn't open /dev/tty file for reading");
|
||||
tty_out = fopen("/dev/tty", "w");
|
||||
if (!tty_out) clean_err("Couldn't open /dev/tty file for writing");
|
||||
if (tcgetattr(fileno(tty_out), &orig_termios))
|
||||
clean_err("Couldn't tcgetattr");
|
||||
atexit(cleanup);
|
||||
|
||||
sprintf(err_tmpfilename, "%s/"BB_NAME".err.XXXXXX", getenv("TMPDIR"));
|
||||
err_tmp_fd = nonnegative(mkostemp(err_tmpfilename, O_RDWR), "Couldn't create error file");
|
||||
stderr_dup_fd = nonnegative(dup(STDERR_FILENO));
|
||||
nonnegative(dup2(err_tmp_fd, STDERR_FILENO), "Couldn't redirect error output");
|
||||
|
||||
tty_in = nonnull(fopen("/dev/tty", "r"));
|
||||
tty_out = nonnull(fopen("/dev/tty", "w"));
|
||||
nonnegative(tcgetattr(fileno(tty_out), &orig_termios));
|
||||
memcpy(&bb_termios, &orig_termios, sizeof(bb_termios));
|
||||
cfmakeraw(&bb_termios);
|
||||
bb_termios.c_cc[VMIN] = 0;
|
||||
bb_termios.c_cc[VTIME] = 1;
|
||||
atexit(cleanup);
|
||||
|
||||
int signals[] = {SIGTERM, SIGINT, SIGXCPU, SIGXFSZ, SIGVTALRM, SIGPROF, SIGSEGV, SIGTSTP};
|
||||
struct sigaction sa = {.sa_handler = &cleanup_and_raise, .sa_flags = (int)(SA_NODEFER | SA_RESETHAND)};
|
||||
for (size_t i = 0; i < sizeof(signals)/sizeof(signals[0]); i++)
|
||||
@ -1213,7 +1206,6 @@ int main(int argc, char *argv[])
|
||||
set_globs(&bb, "*");
|
||||
init_term();
|
||||
bb_browse(&bb, argc, argv);
|
||||
cleanup();
|
||||
|
||||
if (bb.selected && print_selection) {
|
||||
for (entry_t *e = bb.selected; e; e = e->selected.next) {
|
||||
@ -1229,10 +1221,10 @@ int main(int argc, char *argv[])
|
||||
populate_files(&bb, NULL);
|
||||
while (bb.selected)
|
||||
set_selected(&bb, bb.selected, 0);
|
||||
free(bb.globpats);
|
||||
delete(&bb.globpats);
|
||||
for (bb_history_t *next; bb.history; bb.history = next) {
|
||||
next = bb.history->next;
|
||||
free(bb.history);
|
||||
delete(&bb.history);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
57
utils.c
Normal file
57
utils.c
Normal file
@ -0,0 +1,57 @@
|
||||
//
|
||||
// utils.c
|
||||
// Copyright 2021 Bruce Hill
|
||||
// Released under the MIT license with the Commons Clause
|
||||
//
|
||||
// This file contains implementations of some convenience functions for more
|
||||
// easily error checking.
|
||||
//
|
||||
|
||||
#include <err.h>
|
||||
#include <stdarg.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
//
|
||||
// If the given argument is nonnegative, print the error message and exit with
|
||||
// failure. Otherwise, return the given argument.
|
||||
//
|
||||
int check_nonnegative(int negative_err, const char *err_msg, ...)
|
||||
{
|
||||
if (negative_err < 0) {
|
||||
va_list args;
|
||||
va_start(args, err_msg);
|
||||
verr(EXIT_FAILURE, err_msg, args);
|
||||
va_end(args);
|
||||
}
|
||||
return negative_err;
|
||||
}
|
||||
|
||||
//
|
||||
// If the given argument is NULL, print the error message and exit with
|
||||
// failure. Otherwise return the given argument.
|
||||
//
|
||||
void *check_nonnull(void *p, const char *err_msg, ...)
|
||||
{
|
||||
if (p == NULL) {
|
||||
va_list args;
|
||||
va_start(args, err_msg);
|
||||
verr(EXIT_FAILURE, err_msg, args);
|
||||
va_end(args);
|
||||
}
|
||||
return p;
|
||||
}
|
||||
|
||||
//
|
||||
// For a given pointer to a memory-allocated pointer, free its memory and set
|
||||
// the pointer to NULL. (This is a safer alternative to free() that
|
||||
// automatically NULLs out the pointer so it can't be used after freeing)
|
||||
//
|
||||
void delete(void *p)
|
||||
{
|
||||
if (*(void**)p != NULL) {
|
||||
free(*(void**)p);
|
||||
*(void**)p = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1
|
24
utils.h
24
utils.h
@ -1,16 +1,14 @@
|
||||
//
|
||||
// utils.h
|
||||
// Copyright 2020 Bruce Hill
|
||||
// Copyright 2021 Bruce Hill
|
||||
// Released under the MIT license with the Commons Clause
|
||||
//
|
||||
// This file contains some definitions of some utility macros.
|
||||
// This file contains some definitions of some utility macros and functions.
|
||||
//
|
||||
|
||||
#ifndef FILE_UTILS__H
|
||||
#define FILE_UTILS__H
|
||||
|
||||
#include <err.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
|
||||
#ifndef streq
|
||||
@ -55,6 +53,24 @@
|
||||
((node)->name).next = NULL; \
|
||||
} while (0)
|
||||
|
||||
#define S1(x) #x
|
||||
#define S2(x) S1(x)
|
||||
#define __LOCATION__ __FILE__ ":" S2(__LINE__)
|
||||
|
||||
// Error checking helper macros:
|
||||
#define nonnegative(exp, ...) check_nonnegative(exp, __LOCATION__ ": `" #exp "` " __VA_ARGS__)
|
||||
#define nonnull(exp, ...) check_nonnull(exp, __LOCATION__ ": `" #exp "` " __VA_ARGS__)
|
||||
// Error-checking memory allocation helper macros:
|
||||
#define new(t) check_nonnull(calloc(1, sizeof(t)), __LOCATION__ ": new(" #t ") failed")
|
||||
#define new_bytes(n) check_nonnull(calloc(1, n), __LOCATION__ ": new_bytes(" #n ") failed")
|
||||
#define grow(obj, new_count) check_nonnull(reallocarray(obj, (new_count), sizeof(obj[0])), __LOCATION__ ": grow(" #obj ", " #new_count ") failed")
|
||||
#define check_strdup(s) check_nonnull(strdup(s), __LOCATION__ ": check_strdup(" #s ") failed")
|
||||
|
||||
int check_nonnegative(int negative_err, const char *err_msg, ...);
|
||||
__attribute__((returns_nonnull))
|
||||
void *check_nonnull(void *p, const char *err_msg, ...);
|
||||
__attribute__((nonnull))
|
||||
void delete(void *p);
|
||||
|
||||
#endif
|
||||
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1
|
||||
|
Loading…
Reference in New Issue
Block a user