#include #include #include #include #include #include #include #include "buffer.h" #include "util.h" /* Panic with a message */ static void vfail(const char *fmt, va_list args) __attribute__((__noreturn__)); static void vfail(const char *fmt, va_list args) { vfprintf(stderr, fmt, args); fflush(stdin); fflush(stderr); fflush(stdout); #ifdef __MTA__ mta_sync(); #else sync(); #endif fsync(STDIN_FILENO); fsync(STDERR_FILENO); fsync(STDOUT_FILENO); raise(SIGABRT); exit(EXIT_FAILURE); } /* Panic with a nice message */ void __fail(const char *file, unsigned int line, const char *func, const char *fmt, ...) { va_list args; va_start(args, fmt); fprintf(stderr, "\n%s:%u ", file, line); vfail(fmt, args); va_end(args); } #ifndef HAVE_VARIADIC_MACROS /* Panic with a not-quite-as-nice message */ void fail(const char *fmt, ...) { va_list args; va_start(args, fmt); vfail(fmt, args); } #endif /* Concatenate 2 strings, allocating space in r for the result */ char *rstrcat(region r, const char *s1, const char *s2) { char *result = rarrayalloc(r, strlen(s1)+strlen(s2)+1, char); result[0] = '\0'; strcat(result, s1); strcat(result, s2); return result; } /* Concatenate n strings, allocating space in r for the result. The last argument should be a null pointer. */ char *rstrscat(region r, ...) { char *result; int len = 0; const char *s; va_list args; va_start(args, r); while ((s = va_arg(args, const char *))) len += strlen(s); result = rarrayalloc(r, len+1, char); result[0] = '\0'; va_start(args, r); while ((s = va_arg(args, const char *))) strcat(result, s); return result; } /* Convert an integer to a string, storing the result in r */ const char *inttostr(region r, int i) { char *result; int width; if (i == 0) width = 1; else width = (int) (floor(log10(abs(i))) + 1); width++; /* allow an extra character to deal with buggy math roundoff implementation (HPUX) */ if (i<0) width++; result = rarrayalloc(r, width + 1, char); if (snprintf(result, width + 1, "%d", i) == -1) { printf("i=%d, width=%d\n", i, width); fail ("inttostr width wrong\n"); } return result; } /* sprintf a string, allocating space in r for the result */ char *rsprintf(region r, const char *fmt, ...) { va_list args; va_start(args, fmt); return rvsprintf(r, fmt, args); } char *rvsprintf(region r, const char *fmt, va_list args) { growbuf buf = growbuf_new(r, 100); gvprintf(buf, fmt, args); return growbuf_contents(buf); } /* Space for the ASCII representation of a pointer -- 2 hex chars per byte, plus 3 chars for 0x prefix and trailing \0 */ #define PTR_ASCII_SIZE ((int) (3 + sizeof(void *)*2)) /* Convert a pointer to an ascii string with leading 0x. Re-uses internal buffer. */ char *ptr_to_ascii(void *ptr) { static char addr[PTR_ASCII_SIZE]; int nchars; nchars = snprintf(addr, PTR_ASCII_SIZE, "%p", ptr); if (nchars == -1 || nchars >= PTR_ASCII_SIZE) fail("Unable to convert ptr to ascii (need %d bytes, have %d)\n", nchars, PTR_ASCII_SIZE); return addr; } #if 0 /* DOB: this looks dangerous for LP64 platforms, and is currently unused anyhow */ /* Convert a pointer to an integer */ int ptr_hash(void *ptr) { return (int) ptr; } #endif /* Return TRUE iff ptr1 == ptr2 */ bool ptr_eq(void *ptr1, void *ptr2) { return ptr1 == ptr2; } /* Return TRUE iff s1 == s2 */ bool str_eq(const char *s1, const char *s2) { return (strcmp(s1, s2) == 0); } /* A total ordering on pointers. Returns 0 if ptr1 = ptr2, a value <0 if ptr1 < ptr2, or a value >0 if ptr1 > ptr2. */ int ptr_cmp(void *ptr1, void *ptr2) { return (char *) ptr1 - (char *) ptr2; } /* Hash a string */ int string_hash(char *str) { char *c; int h; c = str; h = 0; while (*c) h = 33*h + 720 + *c++; /* SML/NJ's string hash function */ return h; } /* Return TRUE iff s1 == s2 */ bool string_eq(char *s1, char *s2) { return !strcmp(s1, s2); } #ifndef HAVE_INLINE int min(int a, int b) { if (a < b) return a; else return b; } int max(int a, int b) { if (a < b) return b; else return a; } #endif