aboutsummaryrefslogtreecommitdiff
blob: 5960741419f2e2a3dcd7a12c7dea05cbbdb201ac (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
from rpython.rtyper.lltypesystem import lltype, rffi
from rpython.rlib import rgc
from rpython.rlib.rvmprof import traceback

from pypy.interpreter.pycode import PyCode
from pypy.module.faulthandler.cintf import pypy_faulthandler_write
from pypy.module.faulthandler.cintf import pypy_faulthandler_write_uint


MAX_STRING_LENGTH = 500

global_buf = lltype.malloc(rffi.CCHARP.TO, MAX_STRING_LENGTH, flavor='raw',
                           immortal=True, zero=True)

def _dump(fd, s):
    assert isinstance(s, str)
    l = len(s)
    if l >= MAX_STRING_LENGTH:
        l = MAX_STRING_LENGTH - 1
    i = 0
    while i < l:
        global_buf[i] = s[i]
        i += 1
    global_buf[l] = '\x00'
    pypy_faulthandler_write(fd, global_buf)

def _dump_nonneg_int(fd, i):
    pypy_faulthandler_write_uint(fd, rffi.cast(lltype.Unsigned, i),
                                 rffi.cast(rffi.INT, 1))


def dump_code(pycode, loc, fd):
    if pycode is None:
        _dump(fd, "  File ???")
    else:
        _dump(fd, '  File "')
        _dump(fd, pycode.co_filename)
        _dump(fd, '", line ')
        _dump_nonneg_int(fd, pycode.co_firstlineno)
        _dump(fd, " in ")
        _dump(fd, pycode.co_name)
    if loc == traceback.LOC_JITTED:
        _dump(fd, " [jitted]")
    elif loc == traceback.LOC_JITTED_INLINED:
        _dump(fd, " [jit inlined]")
    _dump(fd, "\n")


@rgc.no_collect
def _dump_callback(fd, array_p, array_length):
    """We are as careful as we can reasonably be here (i.e. not 100%,
    but hopefully close enough).  In particular, this is written as
    RPython but shouldn't allocate anything.
    """
    traceback.walk_traceback(PyCode, dump_code, fd, array_p, array_length)