Commit 2da0271f authored by David Gibson's avatar David Gibson Committed by Dan Good

altstack: Consolidate thread-local variables

altstack uses a number of __thread variables to track internal state.  This
allows altstack to be thread-safe, although it's still not re-entrant.
This patch gathers all these variables into a single per-thread state
structure.  This makes it easy to see at a glance what the whole of the
required state is, and thereby easier to reason about correctness of
changes to the implementation.
Signed-off-by: default avatarDavid Gibson <david@gibson.dropbear.id.au>
parent e0b86f0c
...@@ -11,43 +11,48 @@ ...@@ -11,43 +11,48 @@
#include <unistd.h> #include <unistd.h>
#include <sys/mman.h> #include <sys/mman.h>
static __thread char ebuf[ALTSTACK_ERR_MAXLEN]; static __thread struct altstack_state {
static __thread unsigned elen; char ebuf[ALTSTACK_ERR_MAXLEN];
unsigned elen;
jmp_buf jmp;
void *rsp_save[2];
rlim_t max;
void *(*fn)(void *);
void *arg, *out;
} state;
#define bang(x) \ #define bang(x) \
(elen += snprintf(ebuf + elen, sizeof(ebuf) - elen, \ (state.elen += snprintf(state.ebuf + state.elen, \
sizeof(state.ebuf) - state.elen, \
"%s(altstack@%d) %s%s%s", \ "%s(altstack@%d) %s%s%s", \
elen ? "; " : "", __LINE__, (x), \ state.elen ? "; " : "", __LINE__, (x), \
errno ? ": " : "", errno ? strerror(errno) : "")) errno ? ": " : "", \
errno ? strerror(errno) : ""))
void altstack_perror(void) void altstack_perror(void)
{ {
fprintf(stderr, "%s\n", ebuf); fprintf(stderr, "%s\n", state.ebuf);
} }
char *altstack_geterr(void) char *altstack_geterr(void)
{ {
return ebuf; return state.ebuf;
} }
static __thread jmp_buf jmp;
static void segvjmp(int signum) static void segvjmp(int signum)
{ {
longjmp(jmp, 1); longjmp(state.jmp, 1);
} }
static __thread void *rsp_save_[2];
static __thread rlim_t max_;
rlim_t altstack_max(void) { rlim_t altstack_max(void) {
return max_; return state.max;
} }
static ptrdiff_t rsp_save(unsigned i) { static ptrdiff_t rsp_save(unsigned i) {
assert(i < 2); assert(i < 2);
asm volatile ("movq %%rsp, %0" : "=g" (rsp_save_[i])); asm volatile ("movq %%rsp, %0" : "=g" (state.rsp_save[i]));
return (char *) rsp_save_[0] - (char *) rsp_save_[i]; return (char *) state.rsp_save[0] - (char *) state.rsp_save[i];
} }
void altstack_rsp_save(void) { void altstack_rsp_save(void) {
...@@ -58,9 +63,6 @@ ptrdiff_t altstack_used(void) { ...@@ -58,9 +63,6 @@ ptrdiff_t altstack_used(void) {
return rsp_save(1); return rsp_save(1);
} }
static __thread void *(*fn_)(void *);
static __thread void *arg_, *out_;
int altstack(rlim_t max, void *(*fn)(void *), void *arg, void **out) int altstack(rlim_t max, void *(*fn)(void *), void *arg, void **out)
{ {
long pgsz = sysconf(_SC_PAGESIZE); long pgsz = sysconf(_SC_PAGESIZE);
...@@ -73,11 +75,11 @@ int altstack(rlim_t max, void *(*fn)(void *), void *arg, void **out) ...@@ -73,11 +75,11 @@ int altstack(rlim_t max, void *(*fn)(void *), void *arg, void **out)
assert(max > 0 && fn); assert(max > 0 && fn);
#define ok(x, y) ({ long __r = (long) (x); if (__r == -1) { bang(#x); if (y) goto out; } __r; }) #define ok(x, y) ({ long __r = (long) (x); if (__r == -1) { bang(#x); if (y) goto out; } __r; })
fn_ = fn; state.fn = fn;
arg_ = arg; state.arg = arg;
out_ = 0; state.out = 0;
max_ = max; state.max = max;
ebuf[elen = 0] = '\0'; state.ebuf[state.elen = 0] = '\0';
if (out) *out = 0; if (out) *out = 0;
// if the first page below the mapping is in use, we get max-pgsz usable bytes // if the first page below the mapping is in use, we get max-pgsz usable bytes
...@@ -85,13 +87,13 @@ int altstack(rlim_t max, void *(*fn)(void *), void *arg, void **out) ...@@ -85,13 +87,13 @@ int altstack(rlim_t max, void *(*fn)(void *), void *arg, void **out)
max += pgsz; max += pgsz;
ok(getrlimit(RLIMIT_STACK, &rl_save), 1); ok(getrlimit(RLIMIT_STACK, &rl_save), 1);
ok(setrlimit(RLIMIT_STACK, &(struct rlimit) { max_, rl_save.rlim_max }), 1); ok(setrlimit(RLIMIT_STACK, &(struct rlimit) { state.max, rl_save.rlim_max }), 1);
undo++; undo++;
ok(m = mmap(0, max, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS|MAP_GROWSDOWN|MAP_NORESERVE, -1, 0), 1); ok(m = mmap(0, max, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS|MAP_GROWSDOWN|MAP_NORESERVE, -1, 0), 1);
undo++; undo++;
if (setjmp(jmp) == 0) { if (setjmp(state.jmp) == 0) {
unsigned char sigstk[SIGSTKSZ]; unsigned char sigstk[SIGSTKSZ];
stack_t ss = { .ss_sp = sigstk, .ss_size = sizeof(sigstk) }; stack_t ss = { .ss_sp = sigstk, .ss_size = sizeof(sigstk) };
struct sigaction sa = { .sa_handler = segvjmp, .sa_flags = SA_NODEFER|SA_RESETHAND|SA_ONSTACK }; struct sigaction sa = { .sa_handler = segvjmp, .sa_flags = SA_NODEFER|SA_RESETHAND|SA_ONSTACK };
...@@ -108,12 +110,13 @@ int altstack(rlim_t max, void *(*fn)(void *), void *arg, void **out) ...@@ -108,12 +110,13 @@ int altstack(rlim_t max, void *(*fn)(void *), void *arg, void **out)
"mov %1, %%rsp\n\t" "mov %1, %%rsp\n\t"
"sub $8, %%rsp\n\t" "sub $8, %%rsp\n\t"
"push %%r10" "push %%r10"
: "=r" (rsp_save_[0]) : "0" (m + max) : "r10", "memory"); : "=r" (state.rsp_save[0])
out_ = fn_(arg_); : "0" (m + max) : "r10", "memory");
state.out = state.fn(state.arg);
asm volatile ("pop %%rsp" asm volatile ("pop %%rsp"
: : : "memory"); : : : "memory");
ret = 0; ret = 0;
if (out) *out = out_; if (out) *out = state.out;
} }
else { else {
errno = 0; errno = 0;
...@@ -137,5 +140,5 @@ out: ...@@ -137,5 +140,5 @@ out:
if (errno_save) if (errno_save)
errno = errno_save; errno = errno_save;
return !ret && elen ? 1 : ret; return !ret && state.elen ? 1 : ret;
} }
...@@ -24,7 +24,7 @@ char *m_; ...@@ -24,7 +24,7 @@ char *m_;
rlim_t msz_; rlim_t msz_;
#define e(x) (900+(x)) #define e(x) (900+(x))
#define seterr(x) (errno = e(x)) #define seterr(x) (errno = e(x))
#define setcall(x) ((call1 |= !errno ? (x) : 0), (call2 |= errno || out_ ? (x) : 0)) #define setcall(x) ((call1 |= !errno ? (x) : 0), (call2 |= errno || state.out ? (x) : 0))
#define getrlimit(...) (fail&getrlimit_ ? (seterr(getrlimit_), -1) : (setcall(getrlimit_), getrlimit(__VA_ARGS__))) #define getrlimit(...) (fail&getrlimit_ ? (seterr(getrlimit_), -1) : (setcall(getrlimit_), getrlimit(__VA_ARGS__)))
#define mmap(...) (fail&mmap_ ? (seterr(mmap_), (void *)-1) : (setcall(mmap_), mmap(__VA_ARGS__))) #define mmap(...) (fail&mmap_ ? (seterr(mmap_), (void *)-1) : (setcall(mmap_), mmap(__VA_ARGS__)))
#define munmap(a, b) (fail&munmap_ ? (seterr(munmap_), -1) : (setcall(munmap_), munmap(m_=(a), msz_=(b)))) #define munmap(a, b) (fail&munmap_ ? (seterr(munmap_), -1) : (setcall(munmap_), munmap(m_=(a), msz_=(b))))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment