X-Git-Url: https://ruderich.org/simon/gitweb/?p=coloredstderr%2Fcoloredstderr.git;a=blobdiff_plain;f=src%2Fhookmacros.h;h=539d0d9d93fc241ca360df59cc9e4a9db6ee8033;hp=09e718a594f05216fb0034286ebb493ccdbd42cb;hb=a58d1a9017a1a15a237f5b516fe2c44a0f01482e;hpb=73870d5ed5fc275c462eefff724e3b26502ce26a diff --git a/src/hookmacros.h b/src/hookmacros.h index 09e718a..539d0d9 100644 --- a/src/hookmacros.h +++ b/src/hookmacros.h @@ -22,122 +22,216 @@ /* Hook the function by creating a function with the same name. With * LD_PRELOAD our function will be preferred. The original function is stored - * in a static variable (real_*). */ + * in a static variable (real_*). Any function called in these macros must + * make sure to restore the errno if it changes it. + * + * "Pseudo code" for the following macros. is the name of the hooked + * function, is either a file descriptor or a FILE pointer. + * + * if (!real_) { + * real_ = dlsym_function(); + * if (!initialized) { + * init_from_environment(); + * } + * } + * if (tracked_fds_find()) { + * if (force_write_to_non_tty) { + * handle = 1; + * } else { + * handle = isatty_noinline(); + * } + * } else { + * handle = 0; + * } + * + * if (handle) { + * handle__pre(); + * } + * result = real_(); + * if (handle) { + * handle__post(); + * } + * return result; + */ -#define _HOOK_PRE(type, name) \ +#define _HOOK_PRE(type, name, fd) \ int handle; \ - DLSYM_FUNCTION(real_ ## name, #name); + if (unlikely(!(real_ ## name ))) { \ + *(void **) (&(real_ ## name)) = dlsym_function(#name); \ + /* Initialize our data while we're at it. */ \ + if (unlikely(!initialized)) { \ + init_from_environment(); \ + } \ + } \ + /* Check if this fd should be handled. */ \ + if (unlikely(tracked_fds_find(fd))) { \ + if (unlikely(force_write_to_non_tty)) { \ + handle = 1; \ + } else { \ + handle = isatty_noinline(fd); \ + } \ + } else { \ + handle = 0; \ + } #define _HOOK_PRE_FD(type, name, fd) \ type result; \ _HOOK_PRE_FD_(type, name, fd) #define _HOOK_PRE_FD_(type, name, fd) \ - _HOOK_PRE(type, name) \ - handle = check_handle_fd(fd); \ - if (handle) { \ - handle_fd_pre(fd, handle); \ + _HOOK_PRE(type, name, fd) \ + if (unlikely(handle)) { \ + handle_fd_pre(fd); \ } #define _HOOK_PRE_FILE(type, name, file) \ type result; \ - _HOOK_PRE(type, name) \ - handle = check_handle_fd(fileno(file)); \ - if (handle) { \ - handle_file_pre(file, handle); \ + _HOOK_PRE(type, name, fileno(file)) \ + if (unlikely(handle)) { \ + handle_file_pre(file); \ } -/* Save and restore the errno to make sure we return the errno of the original - * function call. */ #define _HOOK_POST_FD_(fd) \ - if (handle) { \ - int saved_errno = errno; \ - handle_fd_post(fd, handle); \ - errno = saved_errno; \ + if (unlikely(handle)) { \ + handle_fd_post(fd); \ } #define _HOOK_POST_FD(fd) \ _HOOK_POST_FD_(fd) \ return result; #define _HOOK_POST_FILE(file) \ - if (handle) { \ - int saved_errno = errno; \ - handle_file_post(file, handle); \ - errno = saved_errno; \ + if (unlikely(handle)) { \ + handle_file_post(file); \ } \ return result; -#define HOOK_VOID1(type, name, fd, type1, arg1) \ +#define HOOK_FUNC_DEF1(type, name, type1, arg1) \ static type (*real_ ## name)(type1); \ - type name(type1 arg1) { \ + type name(type1) visibility_protected; \ + type name(type1 arg1) +#define HOOK_FUNC_DEF2(type, name, type1, arg1, type2, arg2) \ + static type (*real_ ## name)(type1, type2); \ + type name(type1, type2) visibility_protected; \ + type name(type1 arg1, type2 arg2) +#define HOOK_FUNC_DEF3(type, name, type1, arg1, type2, arg2, type3, arg3) \ + static type (*real_ ## name)(type1, type2, type3); \ + type name(type1, type2, type3) visibility_protected; \ + type name(type1 arg1, type2 arg2, type3 arg3) +#define HOOK_FUNC_DEF4(type, name, type1, arg1, type2, arg2, type3, arg3, type4, arg4) \ + static type (*real_ ## name)(type1, type2, type3, type4); \ + type name(type1, type2, type3, type4) visibility_protected; \ + type name(type1 arg1, type2 arg2, type3 arg3, type4 arg4) + +#define HOOK_FUNC_VAR_DEF2(type, name, type1, arg1, type2, arg2) \ + static type (*real_ ## name)(type1, type2, ...); \ + type name(type1, type2, ...) visibility_protected; \ + type name(type1 arg1, type2 arg2, ...) + +#define HOOK_FUNC_SIMPLE3(type, name, type1, arg1, type2, arg2, type3, arg3) \ + type name(type1, type2, type3) visibility_protected; \ + type name(type1 arg1, type2 arg2, type3 arg3) + +#define HOOK_FUNC_VAR_SIMPLE1(type, name, type1, arg1) \ + type name(type1, ...) visibility_protected; \ + type name(type1 arg1, ...) +#define HOOK_FUNC_VAR_SIMPLE2(type, name, type1, arg1, type2, arg2) \ + type name(type1, type2, ...) visibility_protected; \ + type name(type1 arg1, type2 arg2, ...) +#define HOOK_FUNC_VAR_SIMPLE3(type, name, type1, arg1, type2, arg2, type3, arg3) \ + type name(type1, type2, type3, ...) visibility_protected; \ + type name(type1 arg1, type2 arg2, type3 arg3, ...) + +#define HOOK_VOID1(type, name, fd, type1, arg1) \ + HOOK_FUNC_DEF1(type, name, type1, arg1) { \ _HOOK_PRE_FD_(type, name, fd) \ real_ ## name(arg1); \ _HOOK_POST_FD_(fd) \ } +#define HOOK_VOID2(type, name, fd, type1, arg1, type2, arg2) \ + HOOK_FUNC_DEF2(type, name, type1, arg1, type2, arg2) { \ + _HOOK_PRE_FD_(type, name, fd) \ + real_ ## name(arg1, arg2); \ + _HOOK_POST_FD_(fd) \ + } +#define HOOK_VOID3(type, name, fd, type1, arg1, type2, arg2, type3, arg3) \ + HOOK_FUNC_DEF3(type, name, type1, arg1, type2, arg2, type3, arg3) { \ + _HOOK_PRE_FD_(type, name, fd) \ + real_ ## name(arg1, arg2, arg3); \ + _HOOK_POST_FD_(fd) \ + } +#define HOOK_VAR_VOID1(type, name, fd, func, type1, arg1) \ + HOOK_FUNC_VAR_SIMPLE1(type, name, type1, arg1) { \ + va_list ap; \ + va_start(ap, arg1); \ + func(arg1, ap); \ + va_end(ap); \ + } +#define HOOK_VAR_VOID2(type, name, fd, func, type1, arg1, type2, arg2) \ + HOOK_FUNC_VAR_SIMPLE2(type, name, type1, arg1, type2, arg2) { \ + va_list ap; \ + va_start(ap, arg2); \ + func(arg1, arg2, ap); \ + va_end(ap); \ + } + +#define HOOK_FD2(type, name, fd, type1, arg1, type2, arg2) \ + HOOK_FUNC_DEF2(type, name, type1, arg1, type2, arg2) { \ + _HOOK_PRE_FD(type, name, fd) \ + result = real_ ## name(arg1, arg2); \ + _HOOK_POST_FD(fd) \ + } #define HOOK_FD3(type, name, fd, type1, arg1, type2, arg2, type3, arg3) \ - static type (*real_ ## name)(type1, type2, type3); \ - type name(type1 arg1, type2 arg2, type3 arg3) { \ + HOOK_FUNC_DEF3(type, name, type1, arg1, type2, arg2, type3, arg3) { \ _HOOK_PRE_FD(type, name, fd) \ result = real_ ## name(arg1, arg2, arg3); \ _HOOK_POST_FD(fd) \ } #define HOOK_FILE1(type, name, file, type1, arg1) \ - static type (*real_ ## name)(type1); \ - type name(type1 arg1) { \ + HOOK_FUNC_DEF1(type, name, type1, arg1) { \ _HOOK_PRE_FILE(type, name, file) \ result = real_ ## name(arg1); \ _HOOK_POST_FILE(file) \ } #define HOOK_FILE2(type, name, file, type1, arg1, type2, arg2) \ - static type (*real_ ## name)(type1, type2); \ - type name(type1 arg1, type2 arg2) { \ + HOOK_FUNC_DEF2(type, name, type1, arg1, type2, arg2) { \ _HOOK_PRE_FILE(type, name, file) \ result = real_ ## name(arg1, arg2); \ _HOOK_POST_FILE(file) \ } #define HOOK_FILE3(type, name, file, type1, arg1, type2, arg2, type3, arg3) \ - static type (*real_ ## name)(type1, type2, type3); \ - type name(type1 arg1, type2 arg2, type3 arg3) { \ + HOOK_FUNC_DEF3(type, name, type1, arg1, type2, arg2, type3, arg3) { \ _HOOK_PRE_FILE(type, name, file) \ result = real_ ## name(arg1, arg2, arg3); \ _HOOK_POST_FILE(file) \ } #define HOOK_FILE4(type, name, file, type1, arg1, type2, arg2, type3, arg3, type4, arg4) \ - static type (*real_ ## name)(type1, type2, type3, type4); \ - type name(type1 arg1, type2 arg2, type3 arg3, type4 arg4) { \ + HOOK_FUNC_DEF4(type, name, type1, arg1, type2, arg2, type3, arg3, type4, arg4) { \ _HOOK_PRE_FILE(type, name, file) \ result = real_ ## name(arg1, arg2, arg3, arg4); \ _HOOK_POST_FILE(file) \ } #define HOOK_VAR_FILE1(type, name, file, func, type1, arg1) \ - static type (*real_ ## func)(type1, va_list); \ - type name(type1 arg1, ...) { \ + HOOK_FUNC_VAR_SIMPLE1(type, name, type1, arg1) { \ va_list ap; \ - _HOOK_PRE_FILE(type, func, file) \ va_start(ap, arg1); \ - result = real_ ## func(arg1, ap); \ + type result = func(arg1, ap); \ va_end(ap); \ - _HOOK_POST_FILE(file) \ + return result; \ } #define HOOK_VAR_FILE2(type, name, file, func, type1, arg1, type2, arg2) \ - static type (*real_ ## func)(type1, type2, va_list); \ - type name(type1 arg1, type2 arg2, ...) { \ + HOOK_FUNC_VAR_SIMPLE2(type, name, type1, arg1, type2, arg2) { \ va_list ap; \ - _HOOK_PRE_FILE(type, func, file) \ va_start(ap, arg2); \ - result = real_ ## func(arg1, arg2, ap); \ + type result = func(arg1, arg2, ap); \ va_end(ap); \ - _HOOK_POST_FILE(file) \ + return result; \ } #define HOOK_VAR_FILE3(type, name, file, func, type1, arg1, type2, arg2, type3, arg3) \ - static type (*real_ ## func)(type1, type2, type3, va_list); \ - type name(type1 arg1, type2 arg2, type3 arg3, ...) \ - { \ + HOOK_FUNC_VAR_SIMPLE3(type, name, type1, arg1, type2, arg2, type3, arg3) { \ va_list ap; \ - _HOOK_PRE_FILE(type, func, file) \ va_start(ap, arg3); \ - result = real_ ## func(arg1, arg2, arg3, ap); \ + type result = func(arg1, arg2, arg3, ap); \ va_end(ap); \ - _HOOK_POST_FILE(file) \ + return result; \ } #endif