2021-09-28 05:58:51 +00:00
|
|
|
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
|
|
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
|
|
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
|
|
|
│ Copyright 2021 Justine Alexandra Roberts Tunney │
|
|
|
|
|
│ │
|
|
|
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
|
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
|
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
|
|
|
|
│ │
|
|
|
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
|
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
|
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
|
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
|
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
|
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
|
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
|
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
|
|
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
2022-03-21 10:46:16 +00:00
|
|
|
|
#include "libc/assert.h"
|
2021-09-28 05:58:51 +00:00
|
|
|
|
#include "libc/calls/calls.h"
|
2021-10-14 00:27:13 +00:00
|
|
|
|
#include "libc/dce.h"
|
2021-09-28 05:58:51 +00:00
|
|
|
|
#include "libc/errno.h"
|
2021-10-14 00:27:13 +00:00
|
|
|
|
#include "libc/fmt/conv.h"
|
|
|
|
|
#include "libc/intrin/asan.internal.h"
|
2022-03-21 10:46:16 +00:00
|
|
|
|
#include "libc/intrin/kprintf.h"
|
2021-10-14 00:27:13 +00:00
|
|
|
|
#include "libc/log/libfatal.internal.h"
|
2021-09-28 05:58:51 +00:00
|
|
|
|
#include "libc/log/log.h"
|
|
|
|
|
#include "libc/mem/mem.h"
|
|
|
|
|
#include "libc/runtime/gc.internal.h"
|
2022-03-18 09:33:37 +00:00
|
|
|
|
#include "libc/runtime/internal.h"
|
2021-09-28 05:58:51 +00:00
|
|
|
|
#include "libc/runtime/runtime.h"
|
2021-10-14 00:27:13 +00:00
|
|
|
|
#include "libc/runtime/symbols.internal.h"
|
2021-09-28 05:58:51 +00:00
|
|
|
|
#include "libc/stdio/append.internal.h"
|
|
|
|
|
#include "libc/stdio/stdio.h"
|
|
|
|
|
#include "libc/str/str.h"
|
|
|
|
|
#include "libc/sysv/consts/o.h"
|
|
|
|
|
#include "libc/sysv/consts/sig.h"
|
|
|
|
|
#include "libc/testlib/testlib.h"
|
|
|
|
|
#include "libc/x/x.h"
|
|
|
|
|
#include "net/http/escape.h"
|
|
|
|
|
|
2021-10-14 00:27:13 +00:00
|
|
|
|
static bool OutputHasSymbol(const char *output, const char *s) {
|
|
|
|
|
return strstr(output, s) || (!FindDebugBinary() && strstr(output, "NULL"));
|
|
|
|
|
}
|
2021-09-28 05:58:51 +00:00
|
|
|
|
|
2021-10-14 00:27:13 +00:00
|
|
|
|
void FpuCrash(void) {
|
|
|
|
|
typedef char xmm_t __attribute__((__vector_size__(16)));
|
2021-09-28 05:58:51 +00:00
|
|
|
|
xmm_t v = {0x0, 0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7,
|
|
|
|
|
0x8, 0x9, 0xa, 0xb, 0xc, 0xd, 0xe, 0xf};
|
|
|
|
|
volatile int x = 0;
|
|
|
|
|
asm volatile("fldpi");
|
|
|
|
|
asm volatile("mov\t%0,%%r15" : /* no outputs */ : "g"(0x3133731337));
|
|
|
|
|
asm volatile("movaps\t%0,%%xmm15" : /* no outputs */ : "x"(v));
|
|
|
|
|
fputc(7 / x, stdout);
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-14 00:27:13 +00:00
|
|
|
|
char bss[10];
|
|
|
|
|
void BssOverrunCrash(int n) {
|
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i < n; ++i) {
|
|
|
|
|
bss[i] = i;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char data[10] = "abcdeabcde";
|
|
|
|
|
void DataOverrunCrash(int n) {
|
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i < n; ++i) {
|
|
|
|
|
data[i] = i;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char rodata[10] = "abcdeabcde";
|
|
|
|
|
int RodataOverrunCrash(int i) {
|
|
|
|
|
return rodata[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *StackOverrunCrash(int n) {
|
|
|
|
|
int i;
|
|
|
|
|
char stack[10];
|
|
|
|
|
bzero(stack, sizeof(stack));
|
|
|
|
|
for (i = 0; i < n; ++i) {
|
|
|
|
|
stack[i] = i;
|
|
|
|
|
}
|
|
|
|
|
return strdup(stack);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *MemoryLeakCrash(void) {
|
|
|
|
|
char *p = strdup("doge");
|
2022-03-18 09:33:37 +00:00
|
|
|
|
CheckForMemoryLeaks();
|
2021-10-14 00:27:13 +00:00
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int NpeCrash(char *p) {
|
2022-03-18 10:48:37 +00:00
|
|
|
|
asm("nop"); // xxx: due to backtrace addr-1 thing
|
2021-10-14 00:27:13 +00:00
|
|
|
|
return *p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void (*pFpuCrash)(void) = FpuCrash;
|
|
|
|
|
void (*pBssOverrunCrash)(int) = BssOverrunCrash;
|
|
|
|
|
void (*pDataOverrunCrash)(int) = DataOverrunCrash;
|
|
|
|
|
int (*pRodataOverrunCrash)(int) = RodataOverrunCrash;
|
|
|
|
|
char *(*pStackOverrunCrash)(int) = StackOverrunCrash;
|
|
|
|
|
char *(*pMemoryLeakCrash)(void) = MemoryLeakCrash;
|
|
|
|
|
int (*pNpeCrash)(char *) = NpeCrash;
|
|
|
|
|
|
2021-09-28 05:58:51 +00:00
|
|
|
|
void SetUp(void) {
|
|
|
|
|
ShowCrashReports();
|
2021-10-14 00:27:13 +00:00
|
|
|
|
if (__argc == 2) {
|
|
|
|
|
switch (atoi(__argv[1])) {
|
|
|
|
|
case 0:
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
pFpuCrash();
|
|
|
|
|
exit(0);
|
|
|
|
|
case 2:
|
|
|
|
|
pBssOverrunCrash(10 + 1);
|
|
|
|
|
exit(0);
|
|
|
|
|
case 3:
|
|
|
|
|
exit(pRodataOverrunCrash(10 + 1));
|
|
|
|
|
case 4:
|
|
|
|
|
pDataOverrunCrash(10 + 1);
|
|
|
|
|
exit(0);
|
|
|
|
|
case 5:
|
|
|
|
|
exit((intptr_t)pStackOverrunCrash(10 + 1));
|
|
|
|
|
case 6:
|
|
|
|
|
exit((intptr_t)pMemoryLeakCrash());
|
|
|
|
|
case 7:
|
|
|
|
|
exit(pNpeCrash(0));
|
2022-03-18 09:33:37 +00:00
|
|
|
|
case 8:
|
|
|
|
|
__cxa_finalize(0);
|
|
|
|
|
exit(pNpeCrash(0));
|
2021-10-14 00:27:13 +00:00
|
|
|
|
default:
|
|
|
|
|
printf("preventing fork recursion: %s\n", __argv[1]);
|
|
|
|
|
exit(1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// UNFREED MEMORY
|
|
|
|
|
// o/dbg/test/libc/log/backtrace_test.com
|
|
|
|
|
// max allocated space 655,360
|
|
|
|
|
// total allocated space 80
|
|
|
|
|
// total free space 327,600
|
|
|
|
|
// releasable space 0
|
|
|
|
|
// mmaped space 65,360
|
|
|
|
|
// non-mmapped space 327,680
|
|
|
|
|
//
|
|
|
|
|
// 100080040020 64 bytes 5 used
|
|
|
|
|
// 421871 strdup
|
|
|
|
|
// 416529 MemoryLeakCrash
|
|
|
|
|
// 41666d SetUp
|
|
|
|
|
// 45428c testlib_runtestcases
|
|
|
|
|
//
|
|
|
|
|
// 00007fff0000-000080010000 rw-pa-F 2x shadow of 000000000000
|
|
|
|
|
// 000080070000-0000800a0000 rw-pa-F 3x shadow of 0000003c0000
|
|
|
|
|
// 02008fff0000-020090020000 rw-pa-F 3x shadow of 10007ffc0000
|
|
|
|
|
// 020090060000-020090080000 rw-pa-F 2x shadow of 100080340000
|
|
|
|
|
// 0e007fff0000-0e0080010000 rw-pa-F 2x shadow of 6ffffffc0000
|
|
|
|
|
// 100006560000-100006580000 rw-pa-F 2x shadow of 7ffc32b40000
|
|
|
|
|
// 100080000000-100080050000 rw-pa-- 5x automap w/ 50 frame hole
|
|
|
|
|
// 100080370000-100080390000 rw-pa-- 2x automap w/ 1 frame hole
|
|
|
|
|
// 1000803a0000-1000803b0000 rw-pa-- 1x automap
|
|
|
|
|
// 6ffffffe0000-700000000000 rw-paSF 2x stack
|
|
|
|
|
// # 24 frames mapped w/ 51 frames gapped
|
|
|
|
|
TEST(ShowCrashReports, testMemoryLeakCrash) {
|
|
|
|
|
size_t got;
|
|
|
|
|
ssize_t rc;
|
|
|
|
|
int ws, pid, fds[2];
|
|
|
|
|
char *output, buf[512];
|
|
|
|
|
if (!IsAsan()) {
|
|
|
|
|
/* TODO(jart): How can we make this work without ASAN? */
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
ASSERT_NE(-1, pipe2(fds, O_CLOEXEC));
|
|
|
|
|
ASSERT_NE(-1, (pid = vfork()));
|
|
|
|
|
if (!pid) {
|
|
|
|
|
dup2(fds[1], 1);
|
|
|
|
|
dup2(fds[1], 2);
|
|
|
|
|
execv(program_executable_name,
|
|
|
|
|
(char *const[]){program_executable_name, "6", 0});
|
|
|
|
|
_exit(127);
|
|
|
|
|
}
|
|
|
|
|
close(fds[1]);
|
|
|
|
|
output = 0;
|
|
|
|
|
appends(&output, "");
|
|
|
|
|
for (;;) {
|
|
|
|
|
rc = read(fds[0], buf, sizeof(buf));
|
|
|
|
|
if (rc == -1) {
|
|
|
|
|
ASSERT_EQ(EINTR, errno);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if ((got = rc)) {
|
|
|
|
|
appendd(&output, buf, got);
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-04-20 16:56:53 +00:00
|
|
|
|
close(fds[0]);
|
2021-10-14 00:27:13 +00:00
|
|
|
|
ASSERT_NE(-1, wait(&ws));
|
|
|
|
|
EXPECT_TRUE(WIFEXITED(ws));
|
|
|
|
|
EXPECT_EQ(78, WEXITSTATUS(ws));
|
|
|
|
|
if (!strstr(output, "UNFREED MEMORY")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't report leak\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
|
|
|
|
if (IsAsan()) {
|
|
|
|
|
if (!OutputHasSymbol(output, "strdup") ||
|
|
|
|
|
!OutputHasSymbol(output, "MemoryLeakCrash")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't backtrace allocation\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2021-09-28 05:58:51 +00:00
|
|
|
|
}
|
2021-10-14 00:27:13 +00:00
|
|
|
|
free(output);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// clang-format off
|
|
|
|
|
// asan error: stack overrun 1-byte store at 0x6fffffffff0a shadow 0x0e007fff7fe1
|
|
|
|
|
// x
|
|
|
|
|
// uuuuuuuuuuuuuuuuuuuuuuuuuuuuuu..........oooooooooooooooooooooo..................
|
|
|
|
|
// |-15 |-15 |-15 |0 |2 |-13 |-13 |0 |0
|
|
|
|
|
// ╡A ΦCE └eA ☺☻♥♦♣♠•◘○○0000003fffffffff◙CapAmb:○0000↑ÿ_╟ⁿ⌂ ÿ
|
|
|
|
|
// 000000400000-000000462000 .text 000000462000-00000046a000 .data
|
|
|
|
|
// 00007fff0000-00008000ffff
|
|
|
|
|
// 000080070000-00008009ffff
|
|
|
|
|
// 02008fff0000-02009001ffff
|
|
|
|
|
// 020090060000-02009007ffff
|
|
|
|
|
// 0e007ffb0000-0e008000ffff ←shadow
|
|
|
|
|
// 100018eb0000-100018ecffff
|
|
|
|
|
// 100080000000-10008009ffff
|
|
|
|
|
// 100080360000-10008037ffff
|
|
|
|
|
// 100080390000-10008039ffff
|
|
|
|
|
// 6fffffe00000-6fffffffffff ←address
|
|
|
|
|
// 0x0000000000407c06: __die at libc/log/die.c:37
|
|
|
|
|
// 0x000000000040b1c1: __asan_report_store at libc/intrin/asan.c:1104
|
|
|
|
|
// 0x0000000000443302: __asan_report_store1 at libc/intrin/somanyasan.S:118
|
|
|
|
|
// 0x000000000041669a: StackOverrunCrash at test/libc/log/backtrace_test.c:76
|
|
|
|
|
// 0x00000000004167e7: SetUp at test/libc/log/backtrace_test.c:105
|
|
|
|
|
// 0x0000000000452d4b: testlib_runtestcases at libc/testlib/testrunner.c:98
|
|
|
|
|
// 0x000000000044c740: testlib_runalltests at libc/testlib/runner.c:37
|
|
|
|
|
// 0x00000000004026db: main at libc/testlib/testmain.c:155
|
|
|
|
|
// 0x000000000040324f: cosmo at libc/runtime/cosmo.S:64
|
|
|
|
|
// 0x000000000040219b: _start at libc/crt/crt.S:67
|
|
|
|
|
// clang-format on
|
|
|
|
|
TEST(ShowCrashReports, testStackOverrunCrash) {
|
|
|
|
|
if (!IsAsan()) return;
|
|
|
|
|
size_t got;
|
|
|
|
|
ssize_t rc;
|
|
|
|
|
int ws, pid, fds[2];
|
|
|
|
|
char *output, buf[512];
|
|
|
|
|
ASSERT_NE(-1, pipe2(fds, O_CLOEXEC));
|
|
|
|
|
ASSERT_NE(-1, (pid = vfork()));
|
|
|
|
|
if (!pid) {
|
|
|
|
|
dup2(fds[1], 1);
|
|
|
|
|
dup2(fds[1], 2);
|
|
|
|
|
execv(program_executable_name,
|
|
|
|
|
(char *const[]){program_executable_name, "5", 0});
|
|
|
|
|
_exit(127);
|
|
|
|
|
}
|
|
|
|
|
close(fds[1]);
|
|
|
|
|
output = 0;
|
|
|
|
|
appends(&output, "");
|
|
|
|
|
for (;;) {
|
|
|
|
|
rc = read(fds[0], buf, sizeof(buf));
|
|
|
|
|
if (rc == -1) {
|
|
|
|
|
ASSERT_EQ(EINTR, errno);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if ((got = rc)) {
|
|
|
|
|
appendd(&output, buf, got);
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-04-20 16:56:53 +00:00
|
|
|
|
close(fds[0]);
|
2021-10-14 00:27:13 +00:00
|
|
|
|
ASSERT_NE(-1, wait(&ws));
|
|
|
|
|
EXPECT_TRUE(WIFEXITED(ws));
|
|
|
|
|
EXPECT_EQ(77, WEXITSTATUS(ws));
|
|
|
|
|
/* NULL is stopgap until we can copy symbol tablces into binary */
|
|
|
|
|
if (!OutputHasSymbol(output, "StackOverrunCrash")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have backtrace\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2022-03-21 10:46:16 +00:00
|
|
|
|
if (strstr(output, "'int' index 10 into 'char [10]' out of bounds")) {
|
|
|
|
|
// ubsan nailed it
|
|
|
|
|
} else {
|
|
|
|
|
// asan nailed it
|
|
|
|
|
if (!strstr(output, "☺☻♥♦♣♠•◘○")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have memory diagram\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
|
|
|
|
if (!strstr(output, "stack overrun")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report misclassified stack overrun\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2021-10-14 00:27:13 +00:00
|
|
|
|
}
|
|
|
|
|
free(output);
|
2021-09-28 05:58:51 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// error: Uncaught SIGFPE (FPE_INTDIV) on nightmare pid 11724
|
|
|
|
|
// /home/jart/cosmo/o/dbg/test/libc/log/backtrace_test.com.tmp.11721
|
|
|
|
|
// ENOTTY[25]
|
|
|
|
|
// Linux nightmare SMP Thu, 12 Aug 2021 06:16:45 UTC
|
|
|
|
|
//
|
2021-10-14 00:27:13 +00:00
|
|
|
|
// 0x0000000000414659: FpuCrash at test/libc/log/backtrace_test.c:35
|
2021-09-28 05:58:51 +00:00
|
|
|
|
// 0x000000000045003b: testlib_runtestcases at libc/testlib/testrunner.c:98
|
|
|
|
|
// 0x000000000044b770: testlib_runalltests at libc/testlib/runner.c:37
|
|
|
|
|
// 0x000000000040278e: main at libc/testlib/testmain.c:86
|
|
|
|
|
// 0x0000000000403210: cosmo at libc/runtime/cosmo.S:65
|
|
|
|
|
// 0x0000000000402247: _start at libc/crt/crt.S:67
|
|
|
|
|
//
|
|
|
|
|
// RAX 0000000000000007 RBX 00006fffffffff10 RDI 00007ffe0745fde1 ST(0) 0.0
|
|
|
|
|
// RCX 0000000000000000 RDX 0000000000000000 RSI 0000000000489900 ST(1) 0.0
|
|
|
|
|
// RBP 00006fffffffff70 RSP 00006fffffffff10 RIP 000000000041465a ST(2) 0.0
|
|
|
|
|
// R8 0000000000000001 R9 00006ffffffffcc0 R10 00006ffffffffe60 ST(3) 0.0
|
|
|
|
|
// R11 000000000000000d R12 00000dffffffffe2 R13 00006fffffffff10 ST(4) 0.0
|
|
|
|
|
// R14 0000000000000003 R15 000000000049b700 VF PF ZF IF
|
|
|
|
|
//
|
|
|
|
|
// XMM0 00000000000000000000000000000000 XMM8 00000000000000000000000000000000
|
|
|
|
|
// XMM1 000000008000000400000000004160ea XMM9 00000000000000000000000000000000
|
|
|
|
|
// XMM2 00000000000000000000000000000000 XMM10 00000000000000000000000000000000
|
|
|
|
|
// XMM3 00000000000000000000000000000000 XMM11 00000000000000000000000000000000
|
|
|
|
|
// XMM4 00000000000000000000000000000000 XMM12 00000000000000000000000000000000
|
|
|
|
|
// XMM5 00000000000000000000000000000000 XMM13 00000000000000000000000000000000
|
|
|
|
|
// XMM6 00000000000000000000000000000000 XMM14 00000000000000000000000000000000
|
|
|
|
|
// XMM7 00000000000000000000000000000000 XMM15 00000000000000000000000000000000
|
|
|
|
|
//
|
|
|
|
|
// mm->i == 4;
|
|
|
|
|
// mm->p[ 0]=={0x00008007,0x00008008,-1,3,50}; /* 2 */
|
|
|
|
|
// /* 234,881,012 */
|
|
|
|
|
// mm->p[ 1]=={0x0e007ffd,0x0e007fff,-1,3,50}; /* 3 */
|
|
|
|
|
// /* 33,538,280 */
|
|
|
|
|
// mm->p[ 2]=={0x100040e8,0x100040e8,-1,3,50}; /* 1 */
|
|
|
|
|
// /* 1,610,596,103 */
|
|
|
|
|
// mm->p[ 3]=={0x6ffffff0,0x6fffffff,12884901888,306,0}; /* 16 */
|
|
|
|
|
// /* 22 frames mapped w/ 1,879,015,395 frames gapped */
|
|
|
|
|
//
|
|
|
|
|
// 00400000-0045b000 r-xp 00000000 08:03 4587526
|
|
|
|
|
// /home/jart/cosmo/o/dbg/test/libc/log/backtrace_test.com.tmp.11721
|
|
|
|
|
// 0045b000-00461000 rw-p 0005b000 08:03 4587526
|
|
|
|
|
// /home/jart/cosmo/o/dbg/test/libc/log/backtrace_test.com.tmp.11721
|
|
|
|
|
// 00461000-004a0000 rw-p 00000000 00:00 0
|
|
|
|
|
// 80070000-80090000 rw-p 00000000 00:00 0
|
|
|
|
|
// e007ffd0000-e0080000000 rw-p 00000000 00:00 0
|
|
|
|
|
// 100040e80000-100040e90000 rw-p 00000000 00:00 0
|
|
|
|
|
// 6ffffff00000-700000000000 rw-p 00000000 00:00 0
|
|
|
|
|
// 7ffe0743f000-7ffe07460000 rw-p 00000000 00:00 0 [stack]
|
|
|
|
|
// 7ffe075a8000-7ffe075ab000 r--p 00000000 00:00 0 [vvar]
|
|
|
|
|
// 7ffe075ab000-7ffe075ac000 r-xp 00000000 00:00 0 [vdso]
|
|
|
|
|
//
|
|
|
|
|
// /home/jart/cosmo/o/dbg/test/libc/log/backtrace_test.com.tmp.11721 1
|
|
|
|
|
TEST(ShowCrashReports, testDivideByZero) {
|
|
|
|
|
size_t got;
|
|
|
|
|
ssize_t rc;
|
|
|
|
|
int ws, pid, fds[2];
|
|
|
|
|
char *output, buf[512];
|
|
|
|
|
ASSERT_NE(-1, pipe2(fds, O_CLOEXEC));
|
|
|
|
|
ASSERT_NE(-1, (pid = vfork()));
|
|
|
|
|
if (!pid) {
|
|
|
|
|
dup2(fds[1], 1);
|
|
|
|
|
dup2(fds[1], 2);
|
|
|
|
|
execv(program_executable_name,
|
|
|
|
|
(char *const[]){program_executable_name, "1", 0});
|
|
|
|
|
_exit(127);
|
|
|
|
|
}
|
|
|
|
|
close(fds[1]);
|
|
|
|
|
output = 0;
|
|
|
|
|
appends(&output, "");
|
|
|
|
|
for (;;) {
|
|
|
|
|
rc = read(fds[0], buf, sizeof(buf));
|
|
|
|
|
if (rc == -1) {
|
|
|
|
|
ASSERT_EQ(EINTR, errno);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if ((got = rc)) {
|
|
|
|
|
appendd(&output, buf, got);
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-04-20 16:56:53 +00:00
|
|
|
|
close(fds[0]);
|
2021-09-28 05:58:51 +00:00
|
|
|
|
ASSERT_NE(-1, wait(&ws));
|
|
|
|
|
EXPECT_TRUE(WIFEXITED(ws));
|
2022-03-21 10:46:16 +00:00
|
|
|
|
assert(128 + SIGFPE == WEXITSTATUS(ws) || 77 == WEXITSTATUS(ws));
|
2021-09-28 05:58:51 +00:00
|
|
|
|
/* NULL is stopgap until we can copy symbol tablces into binary */
|
2021-10-15 02:36:49 +00:00
|
|
|
|
#ifdef __FNO_OMIT_FRAME_POINTER__
|
2021-10-14 00:27:13 +00:00
|
|
|
|
if (!OutputHasSymbol(output, "FpuCrash")) {
|
2021-09-28 05:58:51 +00:00
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have backtrace\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
2021-10-14 00:27:13 +00:00
|
|
|
|
__die();
|
2021-09-28 05:58:51 +00:00
|
|
|
|
}
|
2021-10-15 02:36:49 +00:00
|
|
|
|
#endif
|
2022-03-21 10:46:16 +00:00
|
|
|
|
if (strstr(output, "divrem overflow")) {
|
|
|
|
|
// UBSAN handled it
|
|
|
|
|
} else {
|
|
|
|
|
// ShowCrashReports() handled it
|
|
|
|
|
if (!strstr(output, gc(xasprintf("%d", pid)))) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have pid\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
|
|
|
|
if (!strstr(output, "SIGFPE")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have signal name\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
|
|
|
|
if (!strstr(output, "3.141")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have fpu register\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
|
|
|
|
if (!strstr(output, "0f0e0d0c0b0a09080706050403020100")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have sse register\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
|
|
|
|
if (!strstr(output, "3133731337")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have general register\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2021-09-28 05:58:51 +00:00
|
|
|
|
}
|
2021-10-14 00:27:13 +00:00
|
|
|
|
free(output);
|
|
|
|
|
}
|
2021-09-28 05:58:51 +00:00
|
|
|
|
|
2021-10-14 00:27:13 +00:00
|
|
|
|
// clang-format off
|
2022-03-21 10:46:16 +00:00
|
|
|
|
//
|
|
|
|
|
// test/libc/log/backtrace_test.c:59: ubsan error: 'int' index 10 into 'char [10]' out of bounds
|
|
|
|
|
// 0x000000000040a352: __die at libc/log/die.c:40
|
|
|
|
|
// 0x0000000000489bc8: __ubsan_abort at libc/intrin/ubsan.c:196
|
|
|
|
|
// 0x0000000000489e1c: __ubsan_handle_out_of_bounds at libc/intrin/ubsan.c:242
|
|
|
|
|
// 0x0000000000423666: BssOverrunCrash at test/libc/log/backtrace_test.c:59
|
|
|
|
|
// 0x0000000000423e0a: SetUp at test/libc/log/backtrace_test.c:115
|
|
|
|
|
// 0x000000000049350b: testlib_runtestcases at libc/testlib/testrunner.c:98
|
|
|
|
|
// 0x000000000048ab50: testlib_runalltests at libc/testlib/runner.c:37
|
|
|
|
|
// 0x00000000004028d0: main at libc/testlib/testmain.c:94
|
|
|
|
|
// 0x0000000000403977: cosmo at libc/runtime/cosmo.S:69
|
|
|
|
|
// 0x00000000004021ae: _start at libc/crt/crt.S:78
|
|
|
|
|
//
|
2021-10-14 00:27:13 +00:00
|
|
|
|
// asan error: global redzone 1-byte store at 0x00000048cf2a shadow 0x0000800899e5
|
|
|
|
|
// x
|
|
|
|
|
// ........................................OOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOOO
|
|
|
|
|
// |0 |0 |0 |0 |2 |-6 |-6 |-6 |-6
|
|
|
|
|
// ☺☻♥♦♣♠•◘○
|
|
|
|
|
// 000000400000-000000460000 .text
|
|
|
|
|
// 000000460000-000000468000 .data
|
|
|
|
|
// 00007fff0000-00008000ffff
|
|
|
|
|
// 000080070000-00008009ffff ←shadow
|
|
|
|
|
// 02008fff0000-02009001ffff
|
|
|
|
|
// 020090060000-02009007ffff
|
|
|
|
|
// 0e007ffb0000-0e008000ffff
|
|
|
|
|
// 1000286b0000-1000286cffff
|
|
|
|
|
// 100080000000-10008009ffff
|
|
|
|
|
// 100080350000-10008036ffff
|
|
|
|
|
// 100080380000-10008038ffff
|
|
|
|
|
// 6fffffe00000-6fffffffffff
|
|
|
|
|
// 0x0000000000407af6: __die at libc/log/die.c:36
|
|
|
|
|
// 0x0000000000444f13: __asan_die at libc/intrin/asan.c:318
|
|
|
|
|
// 0x0000000000445bc8: __asan_report at libc/intrin/asan.c:667
|
|
|
|
|
// 0x0000000000445e41: __asan_report_memory_fault at libc/intrin/asan.c:672
|
|
|
|
|
// 0x0000000000446312: __asan_report_store at libc/intrin/asan.c:1008
|
|
|
|
|
// 0x0000000000444442: __asan_report_store1 at libc/intrin/somanyasan.S:118
|
|
|
|
|
// 0x0000000000416216: BssOverrunCrash at test/libc/log/backtrace_test.c:52
|
|
|
|
|
// 0x000000000041642a: SetUp at test/libc/log/backtrace_test.c:73
|
|
|
|
|
// 0x00000000004513eb: testlib_runtestcases at libc/testlib/testrunner.c:98
|
|
|
|
|
// 0x000000000044bbe0: testlib_runalltests at libc/testlib/runner.c:37
|
|
|
|
|
// 0x00000000004026db: main at libc/testlib/testmain.c:155
|
|
|
|
|
// 0x000000000040323f: cosmo at libc/runtime/cosmo.S:64
|
|
|
|
|
// 0x000000000040219b: _start at libc/crt/crt.S:67
|
2022-03-21 10:46:16 +00:00
|
|
|
|
//
|
2021-10-14 00:27:13 +00:00
|
|
|
|
// clang-format on
|
2022-03-21 10:46:16 +00:00
|
|
|
|
|
2021-10-14 00:27:13 +00:00
|
|
|
|
TEST(ShowCrashReports, testBssOverrunCrash) {
|
|
|
|
|
if (!IsAsan()) return;
|
|
|
|
|
size_t got;
|
|
|
|
|
ssize_t rc;
|
|
|
|
|
int ws, pid, fds[2];
|
|
|
|
|
char *output, buf[512];
|
|
|
|
|
ASSERT_NE(-1, pipe2(fds, O_CLOEXEC));
|
|
|
|
|
ASSERT_NE(-1, (pid = vfork()));
|
|
|
|
|
if (!pid) {
|
|
|
|
|
dup2(fds[1], 1);
|
|
|
|
|
dup2(fds[1], 2);
|
|
|
|
|
execv(program_executable_name,
|
|
|
|
|
(char *const[]){program_executable_name, "2", 0});
|
|
|
|
|
_exit(127);
|
|
|
|
|
}
|
|
|
|
|
close(fds[1]);
|
|
|
|
|
output = 0;
|
|
|
|
|
appends(&output, "");
|
|
|
|
|
for (;;) {
|
|
|
|
|
rc = read(fds[0], buf, sizeof(buf));
|
|
|
|
|
if (rc == -1) {
|
|
|
|
|
ASSERT_EQ(EINTR, errno);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if ((got = rc)) {
|
|
|
|
|
appendd(&output, buf, got);
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-04-20 16:56:53 +00:00
|
|
|
|
close(fds[0]);
|
2021-10-14 00:27:13 +00:00
|
|
|
|
ASSERT_NE(-1, wait(&ws));
|
|
|
|
|
EXPECT_TRUE(WIFEXITED(ws));
|
|
|
|
|
EXPECT_EQ(77, WEXITSTATUS(ws));
|
|
|
|
|
/* NULL is stopgap until we can copy symbol tablces into binary */
|
2021-10-15 02:36:49 +00:00
|
|
|
|
#ifdef __FNO_OMIT_FRAME_POINTER__
|
2021-10-14 00:27:13 +00:00
|
|
|
|
if (!OutputHasSymbol(output, "BssOverrunCrash")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have backtrace\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2021-10-15 02:36:49 +00:00
|
|
|
|
#endif
|
2022-03-21 10:46:16 +00:00
|
|
|
|
if (!strstr(output, "'int' index 10 into 'char [10]' out of bounds") &&
|
|
|
|
|
(!strstr(output, "☺☻♥♦♣♠•◘○") || !strstr(output, "global redzone"))) {
|
2021-10-14 00:27:13 +00:00
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have memory diagram\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
|
|
|
|
free(output);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// clang-format off
|
|
|
|
|
// asan error: null pointer dereference 1-byte load at 0x000000000000 shadow 0x00007fff8000
|
|
|
|
|
// x
|
|
|
|
|
// MMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMM∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅∅
|
|
|
|
|
// |-17 |-17 |-17 |-17 |-17 |-1 |-1 |-1 |-1 |-1
|
|
|
|
|
// ⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅⋅
|
|
|
|
|
// 000000400000-000000464000 .text
|
|
|
|
|
// 000000464000-00000046d000 .data
|
|
|
|
|
// 00007fff0000-00008000ffff ←shadow
|
|
|
|
|
// 000080070000-00008009ffff
|
|
|
|
|
// 02008fff0000-02009001ffff
|
|
|
|
|
// 020090060000-02009007ffff
|
|
|
|
|
// 0e007fff0000-0e008000ffff
|
|
|
|
|
// 10000d3e0000-10000d3fffff
|
|
|
|
|
// 100080000000-10008009ffff
|
|
|
|
|
// 100080370000-10008038ffff
|
|
|
|
|
// 1000803a0000-1000803affff
|
|
|
|
|
// 6ffffffe0000-6fffffffffff
|
|
|
|
|
// 0x0000000000407c84: __die at libc/log/die.c:37
|
|
|
|
|
// 0x000000000040b1ee: __asan_report_load at libc/intrin/asan.c:1083
|
|
|
|
|
// 0x000000000041639e: NpeCrash at test/libc/log/backtrace_test.c:87
|
|
|
|
|
// 0x0000000000416733: SetUp at test/libc/log/backtrace_test.c:120
|
|
|
|
|
// 0x00000000004541fb: testlib_runtestcases at libc/testlib/testrunner.c:98
|
|
|
|
|
// 0x000000000044d000: testlib_runalltests at libc/testlib/runner.c:37
|
|
|
|
|
// 0x00000000004026db: main at libc/testlib/testmain.c:94
|
|
|
|
|
// 0x000000000040327f: cosmo at libc/runtime/cosmo.S:64
|
|
|
|
|
// 0x000000000040219b: _start at libc/crt/crt.S:67
|
|
|
|
|
// clang-format on
|
|
|
|
|
TEST(ShowCrashReports, testNpeCrash) {
|
|
|
|
|
if (!IsAsan()) return;
|
|
|
|
|
size_t got;
|
|
|
|
|
ssize_t rc;
|
|
|
|
|
int ws, pid, fds[2];
|
|
|
|
|
char *output, buf[512];
|
|
|
|
|
ASSERT_NE(-1, pipe2(fds, O_CLOEXEC));
|
|
|
|
|
ASSERT_NE(-1, (pid = vfork()));
|
|
|
|
|
if (!pid) {
|
|
|
|
|
dup2(fds[1], 1);
|
|
|
|
|
dup2(fds[1], 2);
|
|
|
|
|
execv(program_executable_name,
|
|
|
|
|
(char *const[]){program_executable_name, "7", 0});
|
|
|
|
|
_exit(127);
|
|
|
|
|
}
|
|
|
|
|
close(fds[1]);
|
|
|
|
|
output = 0;
|
|
|
|
|
appends(&output, "");
|
|
|
|
|
for (;;) {
|
|
|
|
|
rc = read(fds[0], buf, sizeof(buf));
|
|
|
|
|
if (rc == -1) {
|
|
|
|
|
ASSERT_EQ(EINTR, errno);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if ((got = rc)) {
|
|
|
|
|
appendd(&output, buf, got);
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-04-20 16:56:53 +00:00
|
|
|
|
close(fds[0]);
|
2021-10-14 00:27:13 +00:00
|
|
|
|
ASSERT_NE(-1, wait(&ws));
|
|
|
|
|
EXPECT_TRUE(WIFEXITED(ws));
|
|
|
|
|
EXPECT_EQ(77, WEXITSTATUS(ws));
|
2022-03-18 09:33:37 +00:00
|
|
|
|
/* NULL is stopgap until we can copy symbol tables into binary */
|
2022-03-21 10:46:16 +00:00
|
|
|
|
if (!strstr(output, "null pointer")) {
|
2021-10-14 00:27:13 +00:00
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't diagnose the problem\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2021-10-15 02:36:49 +00:00
|
|
|
|
#ifdef __FNO_OMIT_FRAME_POINTER__
|
2021-10-14 00:27:13 +00:00
|
|
|
|
if (!OutputHasSymbol(output, "NpeCrash")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have backtrace\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2021-10-15 02:36:49 +00:00
|
|
|
|
#endif
|
2022-03-21 10:46:16 +00:00
|
|
|
|
if (strstr(output, "null pointer access")) {
|
|
|
|
|
// ubsan nailed it
|
|
|
|
|
} else {
|
|
|
|
|
// asan nailed it
|
|
|
|
|
if (!strstr(output, "∅∅∅∅")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have shadow diagram\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2021-10-14 00:27:13 +00:00
|
|
|
|
}
|
|
|
|
|
free(output);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(ShowCrashReports, testDataOverrunCrash) {
|
|
|
|
|
if (!IsAsan()) return;
|
|
|
|
|
size_t got;
|
|
|
|
|
ssize_t rc;
|
|
|
|
|
int ws, pid, fds[2];
|
|
|
|
|
char *output, buf[512];
|
|
|
|
|
ASSERT_NE(-1, pipe2(fds, O_CLOEXEC));
|
|
|
|
|
ASSERT_NE(-1, (pid = vfork()));
|
|
|
|
|
if (!pid) {
|
|
|
|
|
dup2(fds[1], 1);
|
|
|
|
|
dup2(fds[1], 2);
|
|
|
|
|
execv(program_executable_name,
|
|
|
|
|
(char *const[]){program_executable_name, "4", 0});
|
|
|
|
|
_exit(127);
|
|
|
|
|
}
|
|
|
|
|
close(fds[1]);
|
|
|
|
|
output = 0;
|
|
|
|
|
appends(&output, "");
|
|
|
|
|
for (;;) {
|
|
|
|
|
rc = read(fds[0], buf, sizeof(buf));
|
|
|
|
|
if (rc == -1) {
|
|
|
|
|
ASSERT_EQ(EINTR, errno);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if ((got = rc)) {
|
|
|
|
|
appendd(&output, buf, got);
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-04-20 16:56:53 +00:00
|
|
|
|
close(fds[0]);
|
2021-10-14 00:27:13 +00:00
|
|
|
|
ASSERT_NE(-1, wait(&ws));
|
|
|
|
|
EXPECT_TRUE(WIFEXITED(ws));
|
|
|
|
|
EXPECT_EQ(77, WEXITSTATUS(ws));
|
|
|
|
|
/* NULL is stopgap until we can copy symbol tablces into binary */
|
2021-10-15 02:36:49 +00:00
|
|
|
|
#ifdef __FNO_OMIT_FRAME_POINTER__
|
2021-10-14 00:27:13 +00:00
|
|
|
|
if (!OutputHasSymbol(output, "DataOverrunCrash")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have backtrace\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2021-10-15 02:36:49 +00:00
|
|
|
|
#endif
|
2022-03-21 10:46:16 +00:00
|
|
|
|
if (!strstr(output, "'int' index 10 into 'char [10]' out of bounds") &&
|
|
|
|
|
(!strstr(output, "☺☻♥♦♣♠•◘○") || !strstr(output, "global redzone"))) {
|
2021-10-14 00:27:13 +00:00
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have memory diagram\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
2021-09-28 05:58:51 +00:00
|
|
|
|
free(output);
|
|
|
|
|
}
|
2022-03-18 09:33:37 +00:00
|
|
|
|
|
|
|
|
|
TEST(ShowCrashReports, testNpeCrashAfterFinalize) {
|
|
|
|
|
/*
|
|
|
|
|
* this test makes sure we're not doing things like depending on
|
|
|
|
|
* environment variables after __cxa_finalize is called in cases
|
|
|
|
|
* where putenv() is used
|
|
|
|
|
*/
|
|
|
|
|
size_t got;
|
|
|
|
|
ssize_t rc;
|
|
|
|
|
int ws, pid, fds[2];
|
|
|
|
|
char *output, buf[512];
|
|
|
|
|
ASSERT_NE(-1, pipe2(fds, O_CLOEXEC));
|
|
|
|
|
ASSERT_NE(-1, (pid = vfork()));
|
|
|
|
|
if (!pid) {
|
|
|
|
|
dup2(fds[1], 1);
|
|
|
|
|
dup2(fds[1], 2);
|
|
|
|
|
execv(program_executable_name,
|
|
|
|
|
(char *const[]){program_executable_name, "8", 0});
|
|
|
|
|
_exit(127);
|
|
|
|
|
}
|
|
|
|
|
close(fds[1]);
|
|
|
|
|
output = 0;
|
|
|
|
|
appends(&output, "");
|
|
|
|
|
for (;;) {
|
|
|
|
|
rc = read(fds[0], buf, sizeof(buf));
|
|
|
|
|
if (rc == -1) {
|
|
|
|
|
ASSERT_EQ(EINTR, errno);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if ((got = rc)) {
|
|
|
|
|
appendd(&output, buf, got);
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-04-20 16:56:53 +00:00
|
|
|
|
close(fds[0]);
|
2022-03-18 09:33:37 +00:00
|
|
|
|
ASSERT_NE(-1, wait(&ws));
|
|
|
|
|
EXPECT_TRUE(WIFEXITED(ws));
|
|
|
|
|
EXPECT_EQ(IsAsan() ? 77 : 128 + SIGSEGV, WEXITSTATUS(ws));
|
|
|
|
|
/* NULL is stopgap until we can copy symbol tables into binary */
|
2022-03-21 10:46:16 +00:00
|
|
|
|
if (!strstr(output, IsAsan() ? "null pointer" : "Uncaught SIGSEGV (SEGV_")) {
|
2022-03-18 09:33:37 +00:00
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't diagnose the problem\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
|
|
|
|
#ifdef __FNO_OMIT_FRAME_POINTER__
|
|
|
|
|
if (!OutputHasSymbol(output, "NpeCrash")) {
|
|
|
|
|
fprintf(stderr, "ERROR: crash report didn't have backtrace\n%s\n",
|
|
|
|
|
gc(IndentLines(output, -1, 0, 4)));
|
|
|
|
|
__die();
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
free(output);
|
|
|
|
|
}
|