#include "tap.hpp" #include "memory/buffer/paged.hpp" #include "debug.hpp" #include "except.hpp" #include <signal.h> #include <setjmp.h> /////////////////////////////////////////////////////////////////////////////// sigjmp_buf fault_jmp; __attribute__((no_sanitize_address)) bool has_fault (const volatile char* addr) { if (sigsetjmp (fault_jmp, 1) == 0) { *addr; return false; } else { return true; } } //----------------------------------------------------------------------------- static bool fault_seen; static void *fault_address; __attribute__((no_sanitize_address)) void segv_handler (int num, siginfo_t *info, void *cookie) { CHECK_EQ (num, SIGSEGV); (void)num; (void)cookie; fault_seen = true; fault_address = info->si_addr; siglongjmp (fault_jmp, SIGSEGV); } /////////////////////////////////////////////////////////////////////////////// int main (void) { cruft::TAP::logger tap; // setup a trap to record SEGV events struct sigaction newhandler {}; newhandler.sa_sigaction = segv_handler; newhandler.sa_flags = SA_SIGINFO; auto err = sigaction (SIGSEGV, &newhandler, nullptr); if (err) cruft::errno_error::throw_code (); // initialise a partially unmapped buffer. the tests assume that the // window is substantially less than half the capacity (so that probing // the centre doesn't trigger a mapping overlapping the end). constexpr size_t CAPACITY = 16 * 1024 * 1024; constexpr size_t WINDOW = 1024 * 1024; cruft::memory::buffer::paged buffer (CAPACITY, WINDOW); typedef decltype(buffer)::value_type value_type; const value_type *first = buffer.begin (); const value_type *last = buffer.end () - 1; const value_type *centre = buffer.begin () + buffer.capacity () / 2; const value_type *window = centre + buffer.window () - 1; // ensure correct initial mappings tap.expect (!has_fault (first), "first is initially valid"); tap.expect ( has_fault (last), "last is intially invalid"); // allocate half the buffer and check mappings buffer.access (const_cast<value_type*> (centre)); tap.expect (!has_fault (first), "first remains valid after commit"); tap.expect (!has_fault (centre), "centre is valid after partial commit"); tap.expect (!has_fault (window), "centre window is valid after partial commit"); tap.expect ( has_fault (last), "last is invalid after partial commit"); // allocate the entire buffer and check the last address buffer.access (const_cast<value_type*> (last)); tap.expect (!has_fault (last), "last is valid after total commit"); // unmap the buffer and check centre and last are invalid buffer.access (const_cast<value_type*> (first)); tap.expect (!has_fault (first), "first value remains valid after release"); tap.expect ( has_fault (centre), "centre is invalid after release"); tap.expect ( has_fault (last), "last is invalid after release"); return tap.status (); }