forked from lix-project/lix
2cd468874f
Because config.h can #define things like _FILE_OFFSET_BITS=64 and not every compilation unit includes config.h, we currently compile half of Nix with _FILE_OFFSET_BITS=64 and other half with _FILE_OFFSET_BITS unset. This causes major havoc with the Settings class on e.g. 32-bit ARM, where different compilation units disagree with the struct layout. E.g.: diff --git a/src/libstore/globals.cc b/src/libstore/globals.cc @@ -166,6 +166,8 @@ void Settings::update() _get(useSubstitutes, "build-use-substitutes"); + fprintf(stderr, "at Settings::update(): &useSubstitutes = %p\n", &nix::settings.useSubstitutes); _get(buildUsersGroup, "build-users-group"); diff --git a/src/libstore/remote-store.cc b/src/libstore/remote-store.cc +++ b/src/libstore/remote-store.cc @@ -138,6 +138,8 @@ void RemoteStore::initConnection(Connection & conn) void RemoteStore::setOptions(Connection & conn) { + fprintf(stderr, "at RemoteStore::setOptions(): &useSubstitutes = %p\n", &nix::settings.useSubstitutes); conn.to << wopSetOptions Gave me: at Settings::update(): &useSubstitutes = 0xb6e5c5cb at RemoteStore::setOptions(): &useSubstitutes = 0xb6e5c5c7 That was not a fun one to debug!
70 lines
2 KiB
C++
70 lines
2 KiB
C++
#include "types.hh"
|
|
|
|
#include <cstring>
|
|
#include <cstddef>
|
|
#include <cstdlib>
|
|
|
|
#include <unistd.h>
|
|
#include <signal.h>
|
|
|
|
namespace nix {
|
|
|
|
|
|
static void sigsegvHandler(int signo, siginfo_t * info, void * ctx)
|
|
{
|
|
/* Detect stack overflows by comparing the faulting address with
|
|
the stack pointer. Unfortunately, getting the stack pointer is
|
|
not portable. */
|
|
bool haveSP = true;
|
|
char * sp = 0;
|
|
#if defined(__x86_64__) && defined(REG_RSP)
|
|
sp = (char *) ((ucontext *) ctx)->uc_mcontext.gregs[REG_RSP];
|
|
#elif defined(REG_ESP)
|
|
sp = (char *) ((ucontext *) ctx)->uc_mcontext.gregs[REG_ESP];
|
|
#else
|
|
haveSP = false;
|
|
#endif
|
|
|
|
if (haveSP) {
|
|
ptrdiff_t diff = (char *) info->si_addr - sp;
|
|
if (diff < 0) diff = -diff;
|
|
if (diff < 4096) {
|
|
char msg[] = "error: stack overflow (possible infinite recursion)\n";
|
|
[[gnu::unused]] int res = write(2, msg, strlen(msg));
|
|
_exit(1); // maybe abort instead?
|
|
}
|
|
}
|
|
|
|
/* Restore default behaviour (i.e. segfault and dump core). */
|
|
struct sigaction act;
|
|
sigfillset(&act.sa_mask);
|
|
act.sa_handler = SIG_DFL;
|
|
act.sa_flags = 0;
|
|
if (sigaction(SIGSEGV, &act, 0)) abort();
|
|
}
|
|
|
|
|
|
void detectStackOverflow()
|
|
{
|
|
#if defined(SA_SIGINFO) && defined (SA_ONSTACK)
|
|
/* Install a SIGSEGV handler to detect stack overflows. This
|
|
requires an alternative stack, otherwise the signal cannot be
|
|
delivered when we're out of stack space. */
|
|
stack_t stack;
|
|
stack.ss_size = 4096 * 4 + MINSIGSTKSZ;
|
|
stack.ss_sp = new char[stack.ss_size];
|
|
if (!stack.ss_sp) throw Error("cannot allocate alternative stack");
|
|
stack.ss_flags = 0;
|
|
if (sigaltstack(&stack, 0) == -1) throw SysError("cannot set alternative stack");
|
|
|
|
struct sigaction act;
|
|
sigfillset(&act.sa_mask);
|
|
act.sa_sigaction = sigsegvHandler;
|
|
act.sa_flags = SA_SIGINFO | SA_ONSTACK;
|
|
if (sigaction(SIGSEGV, &act, 0))
|
|
throw SysError("resetting SIGCHLD");
|
|
#endif
|
|
}
|
|
|
|
|
|
}
|