diff --git a/Makefile b/Makefile index d6ca58b..908989d 100644 --- a/Makefile +++ b/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) diff --git a/bb.c b/bb.c index 1c156f3..95cad43 100644 --- a/bb.c +++ b/bb.c @@ -7,6 +7,7 @@ // #include +#include #include #include #include @@ -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::