diff options
Diffstat (limited to 'Modules/_tracemalloc.c')
| -rw-r--r-- | Modules/_tracemalloc.c | 1407 | 
1 files changed, 1407 insertions, 0 deletions
| diff --git a/Modules/_tracemalloc.c b/Modules/_tracemalloc.c new file mode 100644 index 0000000000..15ed7342b9 --- /dev/null +++ b/Modules/_tracemalloc.c @@ -0,0 +1,1407 @@ +#include "Python.h" +#include "hashtable.h" +#include "frameobject.h" +#include "pythread.h" +#include "osdefs.h" + +/* Trace memory blocks allocated by PyMem_RawMalloc() */ +#define TRACE_RAW_MALLOC + +/* Forward declaration */ +static void tracemalloc_stop(void); +static int tracemalloc_atexit_register(void); +static void* raw_malloc(size_t size); +static void raw_free(void *ptr); + +#ifdef Py_DEBUG +#  define TRACE_DEBUG +#endif + +#define _STR(VAL) #VAL +#define STR(VAL) _STR(VAL) + +/* Protected by the GIL */ +static struct { +    PyMemAllocator mem; +    PyMemAllocator raw; +    PyMemAllocator obj; +} allocators; + +/* Arbitrary limit of the number of frames in a traceback. The value was chosen +   to not allocate too much memory on the stack (see TRACEBACK_STACK_SIZE +   below). */ +#define MAX_NFRAME 100 + +static struct { +    /* Module initialized? +       Variable protected by the GIL */ +    enum { +        TRACEMALLOC_NOT_INITIALIZED, +        TRACEMALLOC_INITIALIZED, +        TRACEMALLOC_FINALIZED +    } initialized; + +    /* atexit handler registered? */ +    int atexit_registered; + +    /* Is tracemalloc tracing memory allocations? +       Variable protected by the GIL */ +    int tracing; + +    /* limit of the number of frames in a traceback, 1 by default. +       Variable protected by the GIL. */ +    int max_nframe; +} tracemalloc_config = {TRACEMALLOC_NOT_INITIALIZED, 0, 0, 1}; + +#if defined(TRACE_RAW_MALLOC) && defined(WITH_THREAD) +/* This lock is needed because tracemalloc_free() is called without +   the GIL held from PyMem_RawFree(). It cannot acquire the lock because it +   would introduce a deadlock in PyThreadState_DeleteCurrent(). */ +static PyThread_type_lock tables_lock; +#  define TABLES_LOCK() PyThread_acquire_lock(tables_lock, 1) +#  define TABLES_UNLOCK() PyThread_release_lock(tables_lock) +#else +   /* variables are protected by the GIL */ +#  define TABLES_LOCK() +#  define TABLES_UNLOCK() +#endif + +/* Pack the frame_t structure to reduce the memory footprint on 64-bit +   architectures: 12 bytes instead of 16. This optimization might produce +   SIGBUS on architectures not supporting unaligned memory accesses (64-bit +   IPS CPU?): on such architecture, the structure must not be packed. */ +#pragma pack(4) +typedef struct +#ifdef __GNUC__ +__attribute__((packed)) +#endif +{ +    PyObject *filename; +    int lineno; +} frame_t; + +typedef struct { +    Py_uhash_t hash; +    int nframe; +    frame_t frames[1]; +} traceback_t; + +#define TRACEBACK_SIZE(NFRAME) \ +        (sizeof(traceback_t) + sizeof(frame_t) * (NFRAME - 1)) +#define TRACEBACK_STACK_SIZE TRACEBACK_SIZE(MAX_NFRAME) + +static PyObject *unknown_filename = NULL; +static traceback_t tracemalloc_empty_traceback; + +typedef struct { +    size_t size; +    traceback_t *traceback; +} trace_t; + +/* Size in bytes of currently traced memory. +   Protected by TABLES_LOCK(). */ +static size_t tracemalloc_traced_memory = 0; + +/* Maximum size in bytes of traced memory. +   Protected by TABLES_LOCK(). */ +static size_t tracemalloc_max_traced_memory = 0; + +/* Hash table used as a set to to intern filenames: +   PyObject* => PyObject*. +   Protected by the GIL */ +static _Py_hashtable_t *tracemalloc_filenames = NULL; + +/* Hash table used as a set to intern tracebacks: +   traceback_t* => traceback_t* +   Protected by the GIL */ +static _Py_hashtable_t *tracemalloc_tracebacks = NULL; + +/* pointer (void*) => trace (trace_t). +   Protected by TABLES_LOCK(). */ +static _Py_hashtable_t *tracemalloc_traces = NULL; + +#ifdef TRACE_DEBUG +static void +tracemalloc_error(const char *format, ...) +{ +    va_list ap; +    fprintf(stderr, "tracemalloc: "); +    va_start(ap, format); +    vfprintf(stderr, format, ap); +    va_end(ap); +    fprintf(stderr, "\n"); +    fflush(stderr); +} +#endif + +#if defined(WITH_THREAD) && defined(TRACE_RAW_MALLOC) +#define REENTRANT_THREADLOCAL + +/* If your OS does not provide native thread local storage, you can implement +   it manually using a lock. Functions of thread.c cannot be used because +   they use PyMem_RawMalloc() which leads to a reentrant call. */ +#if !(defined(_POSIX_THREADS) || defined(NT_THREADS)) +#  error "need native thread local storage (TLS)" +#endif + +static int tracemalloc_reentrant_key; + +/* Any non-NULL pointer can be used */ +#define REENTRANT Py_True + +static int +get_reentrant(void) +{ +    void *ptr = PyThread_get_key_value(tracemalloc_reentrant_key); +    if (ptr != NULL) { +        assert(ptr == REENTRANT); +        return 1; +    } +    else +        return 0; +} + +static void +set_reentrant(int reentrant) +{ +    if (reentrant) { +        assert(PyThread_get_key_value(tracemalloc_reentrant_key) == NULL); +        PyThread_set_key_value(tracemalloc_reentrant_key, +                               REENTRANT); +    } +    else { +        /* FIXME: PyThread_set_key_value() cannot be used to set the flag +           to zero, because it does nothing if the variable has already +           a value set. */ +        PyThread_delete_key_value(tracemalloc_reentrant_key); +    } +} + +#else + +/* WITH_THREAD not defined: Python compiled without threads, +   or TRACE_RAW_MALLOC not defined: variable protected by the GIL */ +static int tracemalloc_reentrant = 0; + +static int +get_reentrant(void) +{ +    return tracemalloc_reentrant; +} + +static void +set_reentrant(int reentrant) +{ +    assert(!reentrant || !get_reentrant()); +    tracemalloc_reentrant = reentrant; +} +#endif + +static int +hashtable_compare_unicode(const void *key, const _Py_hashtable_entry_t *entry) +{ +    if (key != NULL && entry->key != NULL) +        return (PyUnicode_Compare((PyObject *)key, (PyObject *)entry->key) == 0); +    else +        return key == entry->key; +} + +static _Py_hashtable_allocator_t hashtable_alloc = {malloc, free}; + +static _Py_hashtable_t * +hashtable_new(size_t data_size, +              _Py_hashtable_hash_func hash_func, +              _Py_hashtable_compare_func compare_func) +{ +    return _Py_hashtable_new_full(data_size, 0, +                                  hash_func, compare_func, +                                  NULL, NULL, NULL, &hashtable_alloc); +} + +static void* +raw_malloc(size_t size) +{ +    return allocators.raw.malloc(allocators.raw.ctx, size); +} + +static void +raw_free(void *ptr) +{ +    allocators.raw.free(allocators.raw.ctx, ptr); +} + +static Py_uhash_t +hashtable_hash_traceback(const void *key) +{ +    const traceback_t *traceback = key; +    return traceback->hash; +} + +static int +hashtable_compare_traceback(const traceback_t *traceback1, +                            const _Py_hashtable_entry_t *he) +{ +    const traceback_t *traceback2 = he->key; +    const frame_t *frame1, *frame2; +    int i; + +    if (traceback1->nframe != traceback2->nframe) +        return 0; + +    for (i=0; i < traceback1->nframe; i++) { +        frame1 = &traceback1->frames[i]; +        frame2 = &traceback2->frames[i]; + +        if (frame1->lineno != frame2->lineno) +            return 0; + +        if (frame1->filename != frame2->filename) { +            assert(PyUnicode_Compare(frame1->filename, frame2->filename) != 0); +            return 0; +        } +    } +    return 1; +} + +static void +tracemalloc_get_frame(PyFrameObject *pyframe, frame_t *frame) +{ +    PyCodeObject *code; +    PyObject *filename; +    _Py_hashtable_entry_t *entry; + +    frame->filename = unknown_filename; +    frame->lineno = PyFrame_GetLineNumber(pyframe); +    assert(frame->lineno >= 0); +    if (frame->lineno < 0) +        frame->lineno = 0; + +    code = pyframe->f_code; +    if (code == NULL) { +#ifdef TRACE_DEBUG +        tracemalloc_error("failed to get the code object of the a frame"); +#endif +        return; +    } + +    if (code->co_filename == NULL) { +#ifdef TRACE_DEBUG +        tracemalloc_error("failed to get the filename of the code object"); +#endif +        return; +    } + +    filename = code->co_filename; +    assert(filename != NULL); +    if (filename == NULL) +        return; + +    if (!PyUnicode_Check(filename)) { +#ifdef TRACE_DEBUG +        tracemalloc_error("filename is not an unicode string"); +#endif +        return; +    } +    if (!PyUnicode_IS_READY(filename)) { +        /* Don't make a Unicode string ready to avoid reentrant calls +           to tracemalloc_malloc() or tracemalloc_realloc() */ +#ifdef TRACE_DEBUG +        tracemalloc_error("filename is not a ready unicode string"); +#endif +        return; +    } + +    /* intern the filename */ +    entry = _Py_hashtable_get_entry(tracemalloc_filenames, filename); +    if (entry != NULL) { +        filename = (PyObject *)entry->key; +    } +    else { +        /* tracemalloc_filenames is responsible to keep a reference +           to the filename */ +        Py_INCREF(filename); +        if (_Py_hashtable_set(tracemalloc_filenames, filename, NULL, 0) < 0) { +            Py_DECREF(filename); +#ifdef TRACE_DEBUG +            tracemalloc_error("failed to intern the filename"); +#endif +            return; +        } +    } + +    /* the tracemalloc_filenames table keeps a reference to the filename */ +    frame->filename = filename; +} + +static Py_uhash_t +traceback_hash(traceback_t *traceback) +{ +    /* code based on tuplehash() of Objects/tupleobject.c */ +    Py_uhash_t x;  /* Unsigned for defined overflow behavior. */ +    Py_hash_t y; +    int len = traceback->nframe; +    Py_uhash_t mult = _PyHASH_MULTIPLIER; +    frame_t *frame; + +    x = 0x345678UL; +    frame = traceback->frames; +    while (--len >= 0) { +        y = PyObject_Hash(frame->filename); +        y ^= frame->lineno; +        frame++; + +        x = (x ^ y) * mult; +        /* the cast might truncate len; that doesn't change hash stability */ +        mult += (Py_hash_t)(82520UL + len + len); +    } +    x += 97531UL; +    return x; +} + +static void +traceback_get_frames(traceback_t *traceback) +{ +    PyThreadState *tstate; +    PyFrameObject *pyframe; + +#ifdef WITH_THREAD +    tstate = PyGILState_GetThisThreadState(); +#else +    tstate = PyThreadState_Get(); +#endif +    if (tstate == NULL) { +#ifdef TRACE_DEBUG +        tracemalloc_error("failed to get the current thread state"); +#endif +        return; +    } + +    for (pyframe = tstate->frame; pyframe != NULL; pyframe = pyframe->f_back) { +        tracemalloc_get_frame(pyframe, &traceback->frames[traceback->nframe]); +        assert(traceback->frames[traceback->nframe].filename != NULL); +        assert(traceback->frames[traceback->nframe].lineno >= 0); +        traceback->nframe++; +        if (traceback->nframe == tracemalloc_config.max_nframe) +            break; +    } +} + +static traceback_t * +traceback_new(void) +{ +    char stack_buffer[TRACEBACK_STACK_SIZE]; +    traceback_t *traceback = (traceback_t *)stack_buffer; +    _Py_hashtable_entry_t *entry; + +#ifdef WITH_THREAD +    assert(PyGILState_Check()); +#endif + +    /* get frames */ +    traceback->nframe = 0; +    traceback_get_frames(traceback); +    if (traceback->nframe == 0) +        return &tracemalloc_empty_traceback; +    traceback->hash = traceback_hash(traceback); + +    /* intern the traceback */ +    entry = _Py_hashtable_get_entry(tracemalloc_tracebacks, traceback); +    if (entry != NULL) { +        traceback = (traceback_t *)entry->key; +    } +    else { +        traceback_t *copy; +        size_t traceback_size; + +        traceback_size = TRACEBACK_SIZE(traceback->nframe); + +        copy = raw_malloc(traceback_size); +        if (copy == NULL) { +#ifdef TRACE_DEBUG +            tracemalloc_error("failed to intern the traceback: malloc failed"); +#endif +            return NULL; +        } +        memcpy(copy, traceback, traceback_size); + +        if (_Py_hashtable_set(tracemalloc_tracebacks, copy, NULL, 0) < 0) { +            raw_free(copy); +#ifdef TRACE_DEBUG +            tracemalloc_error("failed to intern the traceback: putdata failed"); +#endif +            return NULL; +        } +        traceback = copy; +    } +    return traceback; +} + +static void +tracemalloc_log_alloc(void *ptr, size_t size) +{ +    traceback_t *traceback; +    trace_t trace; + +#ifdef WITH_THREAD +    assert(PyGILState_Check()); +#endif + +    traceback = traceback_new(); +    if (traceback == NULL) +        return; + +    trace.size = size; +    trace.traceback = traceback; + +    TABLES_LOCK(); +    assert(tracemalloc_traced_memory <= PY_SIZE_MAX - size); +    tracemalloc_traced_memory += size; +    if (tracemalloc_traced_memory > tracemalloc_max_traced_memory) +        tracemalloc_max_traced_memory = tracemalloc_traced_memory; + +    _Py_HASHTABLE_SET(tracemalloc_traces, ptr, trace); +    TABLES_UNLOCK(); +} + +static void +tracemalloc_log_free(void *ptr) +{ +    trace_t trace; + +    TABLES_LOCK(); +    if (_Py_hashtable_pop(tracemalloc_traces, ptr, &trace, sizeof(trace))) { +        assert(tracemalloc_traced_memory >= trace.size); +        tracemalloc_traced_memory -= trace.size; +    } +    TABLES_UNLOCK(); +} + +static void* +tracemalloc_malloc(void *ctx, size_t size, int gil_held) +{ +    PyMemAllocator *alloc = (PyMemAllocator *)ctx; +#if defined(TRACE_RAW_MALLOC) && defined(WITH_THREAD) +    PyGILState_STATE gil_state; +#endif +    void *ptr; + +    if (get_reentrant()) { +        return alloc->malloc(alloc->ctx, size); +    } + +    /* Ignore reentrant call. PyObjet_Malloc() calls PyMem_Malloc() +       for allocations larger than 512 bytes. PyGILState_Ensure() may call +       PyMem_RawMalloc() indirectly which would call PyGILState_Ensure() if +       reentrant are not disabled. */ +    set_reentrant(1); +#ifdef WITH_THREAD +#ifdef TRACE_RAW_MALLOC +    if (!gil_held) +        gil_state = PyGILState_Ensure(); +#else +    assert(gil_held); +#endif +#endif +    ptr = alloc->malloc(alloc->ctx, size); +    set_reentrant(0); + +    if (ptr != NULL) +        tracemalloc_log_alloc(ptr, size); + +#if defined(TRACE_RAW_MALLOC) && defined(WITH_THREAD) +    if (!gil_held) +        PyGILState_Release(gil_state); +#endif + +    return ptr; +} + +static void* +tracemalloc_realloc(void *ctx, void *ptr, size_t new_size, int gil_held) +{ +    PyMemAllocator *alloc = (PyMemAllocator *)ctx; +#if defined(TRACE_RAW_MALLOC) && defined(WITH_THREAD) +    PyGILState_STATE gil_state; +#endif +    void *ptr2; + +    if (get_reentrant()) { +        /* Reentrant call to PyMem_Realloc() and PyMem_RawRealloc(). +           Example: PyMem_RawRealloc() is called internally by pymalloc +           (_PyObject_Malloc() and  _PyObject_Realloc()) to allocate a new +           arena (new_arena()). */ +        ptr2 = alloc->realloc(alloc->ctx, ptr, new_size); + +        if (ptr2 != NULL && ptr != NULL) +            tracemalloc_log_free(ptr); + +        return ptr2; +    } + +    /* Ignore reentrant call. PyObjet_Realloc() calls PyMem_Realloc() for +       allocations larger than 512 bytes. PyGILState_Ensure() may call +       PyMem_RawMalloc() indirectly which would call PyGILState_Ensure() if +       reentrant are not disabled. */ +    set_reentrant(1); +#ifdef WITH_THREAD +#ifdef TRACE_RAW_MALLOC +    if (!gil_held) +        gil_state = PyGILState_Ensure(); +#else +    assert(gil_held); +#endif +#endif +    ptr2 = alloc->realloc(alloc->ctx, ptr, new_size); +    set_reentrant(0); + +    if (ptr2 != NULL) { +        if (ptr != NULL) +            tracemalloc_log_free(ptr); + +        tracemalloc_log_alloc(ptr2, new_size); +    } + +#if defined(TRACE_RAW_MALLOC) && defined(WITH_THREAD) +    if (!gil_held) +        PyGILState_Release(gil_state); +#endif + +    return ptr2; +} + +static void +tracemalloc_free(void *ctx, void *ptr) +{ +    PyMemAllocator *alloc = (PyMemAllocator *)ctx; + +    if (ptr == NULL) +        return; + +     /* GIL cannot be locked in PyMem_RawFree() because it would introduce +        a deadlock in PyThreadState_DeleteCurrent(). */ + +    alloc->free(alloc->ctx, ptr); +    tracemalloc_log_free(ptr); +} + +static void* +tracemalloc_malloc_gil(void *ctx, size_t size) +{ +    return tracemalloc_malloc(ctx, size, 1); +} + +static void* +tracemalloc_realloc_gil(void *ctx, void *ptr, size_t new_size) +{ +    return tracemalloc_realloc(ctx, ptr, new_size, 1); +} + +#ifdef TRACE_RAW_MALLOC +static void* +tracemalloc_raw_malloc(void *ctx, size_t size) +{ +    return tracemalloc_malloc(ctx, size, 0); +} + +static void* +tracemalloc_raw_realloc(void *ctx, void *ptr, size_t new_size) +{ +    return tracemalloc_realloc(ctx, ptr, new_size, 0); +} +#endif + +static int +tracemalloc_clear_filename(_Py_hashtable_entry_t *entry, void *user_data) +{ +    PyObject *filename = (PyObject *)entry->key; +    Py_DECREF(filename); +    return 0; +} + +static int +traceback_free_traceback(_Py_hashtable_entry_t *entry, void *user_data) +{ +    traceback_t *traceback = (traceback_t *)entry->key; +    raw_free(traceback); +    return 0; +} + +/* reentrant flag must be set to call this function and GIL must be held */ +static void +tracemalloc_clear_traces(void) +{ +#ifdef WITH_THREAD +    /* The GIL protects variables againt concurrent access */ +    assert(PyGILState_Check()); +#endif + +    /* Disable also reentrant calls to tracemalloc_malloc() to not add a new +       trace while we are clearing traces */ +    assert(get_reentrant()); + +    TABLES_LOCK(); +    _Py_hashtable_clear(tracemalloc_traces); +    tracemalloc_traced_memory = 0; +    tracemalloc_max_traced_memory = 0; +    TABLES_UNLOCK(); + +    _Py_hashtable_foreach(tracemalloc_tracebacks, traceback_free_traceback, NULL); +    _Py_hashtable_clear(tracemalloc_tracebacks); + +    _Py_hashtable_foreach(tracemalloc_filenames, tracemalloc_clear_filename, NULL); +    _Py_hashtable_clear(tracemalloc_filenames); +} + +static int +tracemalloc_init(void) +{ +    if (tracemalloc_config.initialized == TRACEMALLOC_FINALIZED) { +        PyErr_SetString(PyExc_RuntimeError, +                        "the tracemalloc module has been unloaded"); +        return -1; +    } + +    if (tracemalloc_config.initialized == TRACEMALLOC_INITIALIZED) +        return 0; + +    PyMem_GetAllocator(PYMEM_DOMAIN_RAW, &allocators.raw); + +#ifdef REENTRANT_THREADLOCAL +    tracemalloc_reentrant_key = PyThread_create_key(); +    if (tracemalloc_reentrant_key == -1) { +#ifdef MS_WINDOWS +        PyErr_SetFromWindowsErr(0); +#else +        PyErr_SetFromErrno(PyExc_OSError); +#endif +        return -1; +    } +#endif + +#if defined(WITH_THREAD) && defined(TRACE_RAW_MALLOC) +    if (tables_lock == NULL) { +        tables_lock = PyThread_allocate_lock(); +        if (tables_lock == NULL) { +            PyErr_SetString(PyExc_RuntimeError, "cannot allocate lock"); +            return -1; +        } +    } +#endif + +    tracemalloc_filenames = hashtable_new(0, +                                          (_Py_hashtable_hash_func)PyObject_Hash, +                                          hashtable_compare_unicode); + +    tracemalloc_tracebacks = hashtable_new(0, +                                           (_Py_hashtable_hash_func)hashtable_hash_traceback, +                                           (_Py_hashtable_compare_func)hashtable_compare_traceback); + +    tracemalloc_traces = hashtable_new(sizeof(trace_t), +                                       _Py_hashtable_hash_ptr, +                                       _Py_hashtable_compare_direct); + +    if (tracemalloc_filenames == NULL || tracemalloc_tracebacks == NULL +        || tracemalloc_traces == NULL) +    { +        PyErr_NoMemory(); +        return -1; +    } + +    unknown_filename = PyUnicode_FromString("<unknown>"); +    if (unknown_filename == NULL) +        return -1; +    PyUnicode_InternInPlace(&unknown_filename); + +    tracemalloc_empty_traceback.nframe = 1; +    /* borrowed reference */ +    tracemalloc_empty_traceback.frames[0].filename = unknown_filename; +    tracemalloc_empty_traceback.frames[0].lineno = 0; +    tracemalloc_empty_traceback.hash = traceback_hash(&tracemalloc_empty_traceback); + +    /* Disable tracing allocations until hooks are installed. Set +       also the reentrant flag to detect bugs: fail with an assertion error +       if set_reentrant(1) is called while tracing is disabled. */ +    set_reentrant(1); + +    tracemalloc_config.initialized = TRACEMALLOC_INITIALIZED; +    return 0; +} + +static void +tracemalloc_deinit(void) +{ +    if (tracemalloc_config.initialized != TRACEMALLOC_INITIALIZED) +        return; +    tracemalloc_config.initialized = TRACEMALLOC_FINALIZED; + +    tracemalloc_stop(); + +    /* destroy hash tables */ +    _Py_hashtable_destroy(tracemalloc_traces); +    _Py_hashtable_destroy(tracemalloc_tracebacks); +    _Py_hashtable_destroy(tracemalloc_filenames); + +#if defined(WITH_THREAD) && defined(TRACE_RAW_MALLOC) +    if (tables_lock != NULL) { +        PyThread_free_lock(tables_lock); +        tables_lock = NULL; +    } +#endif + +#ifdef REENTRANT_THREADLOCAL +    PyThread_delete_key(tracemalloc_reentrant_key); +#endif + +    Py_XDECREF(unknown_filename); +} + +static int +tracemalloc_start(void) +{ +    PyMemAllocator alloc; + +    if (tracemalloc_init() < 0) +        return -1; + +    if (tracemalloc_config.tracing) { +        /* hook already installed: do nothing */ +        return 0; +    } + +    if (tracemalloc_atexit_register() < 0) +        return -1; + +#ifdef TRACE_RAW_MALLOC +    alloc.malloc = tracemalloc_raw_malloc; +    alloc.realloc = tracemalloc_raw_realloc; +    alloc.free = tracemalloc_free; + +    alloc.ctx = &allocators.raw; +    PyMem_GetAllocator(PYMEM_DOMAIN_RAW, &allocators.raw); +    PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &alloc); +#endif + +    alloc.malloc = tracemalloc_malloc_gil; +    alloc.realloc = tracemalloc_realloc_gil; +    alloc.free = tracemalloc_free; + +    alloc.ctx = &allocators.mem; +    PyMem_GetAllocator(PYMEM_DOMAIN_MEM, &allocators.mem); +    PyMem_SetAllocator(PYMEM_DOMAIN_MEM, &alloc); + +    alloc.ctx = &allocators.obj; +    PyMem_GetAllocator(PYMEM_DOMAIN_OBJ, &allocators.obj); +    PyMem_SetAllocator(PYMEM_DOMAIN_OBJ, &alloc); + +    /* everything is ready: start tracing Python memory allocations */ +    tracemalloc_config.tracing = 1; +    set_reentrant(0); + +    return 0; +} + +static void +tracemalloc_stop(void) +{ +    if (!tracemalloc_config.tracing) +        return; + +    /* stop tracing Python memory allocations */ +    tracemalloc_config.tracing = 0; + +    /* set the reentrant flag to detect bugs: fail with an assertion error if +       set_reentrant(1) is called while tracing is disabled. */ +    set_reentrant(1); + +    /* unregister the hook on memory allocators */ +#ifdef TRACE_RAW_MALLOC +    PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &allocators.raw); +#endif +    PyMem_SetAllocator(PYMEM_DOMAIN_MEM, &allocators.mem); +    PyMem_SetAllocator(PYMEM_DOMAIN_OBJ, &allocators.obj); + +    /* release memory */ +    tracemalloc_clear_traces(); +} + + +static PyObject* +lineno_as_obj(int lineno) +{ +    if (lineno >= 0) +        return PyLong_FromLong(lineno); +    else +        Py_RETURN_NONE; +} + +PyDoc_STRVAR(tracemalloc_is_tracing_doc, +    "is_tracing()->bool\n" +    "\n" +    "True if the tracemalloc module is tracing Python memory allocations,\n" +    "False otherwise."); + +static PyObject* +py_tracemalloc_is_tracing(PyObject *self) +{ +    return PyBool_FromLong(tracemalloc_config.tracing); +} + +PyDoc_STRVAR(tracemalloc_clear_traces_doc, +    "clear_traces()\n" +    "\n" +    "Clear traces of memory blocks allocated by Python."); + +static PyObject* +py_tracemalloc_clear_traces(PyObject *self) +{ +    if (!tracemalloc_config.tracing) +        Py_RETURN_NONE; + +    set_reentrant(1); +    tracemalloc_clear_traces(); +    set_reentrant(0); + +    Py_RETURN_NONE; +} + +static PyObject* +frame_to_pyobject(frame_t *frame) +{ +    PyObject *frame_obj, *lineno_obj; + +    frame_obj = PyTuple_New(2); +    if (frame_obj == NULL) +        return NULL; + +    if (frame->filename == NULL) +        frame->filename = Py_None; +    Py_INCREF(frame->filename); +    PyTuple_SET_ITEM(frame_obj, 0, frame->filename); + +    assert(frame->lineno >= 0); +    lineno_obj = lineno_as_obj(frame->lineno); +    if (lineno_obj == NULL) { +        Py_DECREF(frame_obj); +        return NULL; +    } +    PyTuple_SET_ITEM(frame_obj, 1, lineno_obj); + +    return frame_obj; +} + +static PyObject* +traceback_to_pyobject(traceback_t *traceback, _Py_hashtable_t *intern_table) +{ +    int i; +    PyObject *frames, *frame; + +    if (intern_table != NULL) { +        if (_Py_HASHTABLE_GET(intern_table, traceback, frames)) { +            Py_INCREF(frames); +            return frames; +        } +    } + +    frames = PyTuple_New(traceback->nframe); +    if (frames == NULL) +        return NULL; + +    for (i=0; i < traceback->nframe; i++) { +        frame = frame_to_pyobject(&traceback->frames[i]); +        if (frame == NULL) { +            Py_DECREF(frames); +            return NULL; +        } +        PyTuple_SET_ITEM(frames, i, frame); +    } + +    if (intern_table != NULL) { +        if (_Py_HASHTABLE_SET(intern_table, traceback, frames) < 0) { +            Py_DECREF(frames); +            PyErr_NoMemory(); +            return NULL; +        } +        /* intern_table keeps a new reference to frames */ +        Py_INCREF(frames); +    } +    return frames; +} + +static PyObject* +trace_to_pyobject(trace_t *trace, _Py_hashtable_t *intern_tracebacks) +{ +    PyObject *trace_obj = NULL; +    PyObject *size, *traceback; + +    trace_obj = PyTuple_New(2); +    if (trace_obj == NULL) +        return NULL; + +    size = PyLong_FromSize_t(trace->size); +    if (size == NULL) { +        Py_DECREF(trace_obj); +        return NULL; +    } +    PyTuple_SET_ITEM(trace_obj, 0, size); + +    traceback = traceback_to_pyobject(trace->traceback, intern_tracebacks); +    if (traceback == NULL) { +        Py_DECREF(trace_obj); +        return NULL; +    } +    PyTuple_SET_ITEM(trace_obj, 1, traceback); + +    return trace_obj; +} + +typedef struct { +    _Py_hashtable_t *traces; +    _Py_hashtable_t *tracebacks; +    PyObject *list; +} get_traces_t; + +static int +tracemalloc_get_traces_fill(_Py_hashtable_entry_t *entry, void *user_data) +{ +    get_traces_t *get_traces = user_data; +    trace_t *trace; +    PyObject *tracemalloc_obj; +    int res; + +    trace = (trace_t *)_PY_HASHTABLE_ENTRY_DATA(entry); + +    tracemalloc_obj = trace_to_pyobject(trace, get_traces->tracebacks); +    if (tracemalloc_obj == NULL) +        return 1; + +    res = PyList_Append(get_traces->list, tracemalloc_obj); +    Py_DECREF(tracemalloc_obj); +    if (res < 0) +        return 1; + +    return 0; +} + +static int +tracemalloc_pyobject_decref_cb(_Py_hashtable_entry_t *entry, void *user_data) +{ +    PyObject *obj = (PyObject *)_Py_HASHTABLE_ENTRY_DATA_AS_VOID_P(entry); +    Py_DECREF(obj); +    return 0; +} + +PyDoc_STRVAR(tracemalloc_get_traces_doc, +    "get_traces() -> list\n" +    "\n" +    "Get traces of all memory blocks allocated by Python.\n" +    "Return a list of (size: int, traceback: tuple) tuples.\n" +    "traceback is a tuple of (filename: str, lineno: int) tuples.\n" +    "\n" +    "Return an empty list if the tracemalloc module is disabled."); + +static PyObject* +py_tracemalloc_get_traces(PyObject *self, PyObject *obj) +{ +    get_traces_t get_traces; +    int err; + +    get_traces.traces = NULL; +    get_traces.tracebacks = NULL; +    get_traces.list = PyList_New(0); +    if (get_traces.list == NULL) +        goto error; + +    if (!tracemalloc_config.tracing) +        return get_traces.list; + +    get_traces.tracebacks = hashtable_new(sizeof(PyObject *), +                                          _Py_hashtable_hash_ptr, +                                          _Py_hashtable_compare_direct); +    if (get_traces.tracebacks == NULL) { +        PyErr_NoMemory(); +        goto error; +    } + +    TABLES_LOCK(); +    get_traces.traces = _Py_hashtable_copy(tracemalloc_traces); +    TABLES_UNLOCK(); + +    if (get_traces.traces == NULL) { +        PyErr_NoMemory(); +        goto error; +    } + +    set_reentrant(1); +    err = _Py_hashtable_foreach(get_traces.traces, +                                tracemalloc_get_traces_fill, &get_traces); +    set_reentrant(0); +    if (err) +        goto error; + +    goto finally; + +error: +    Py_CLEAR(get_traces.list); + +finally: +    if (get_traces.tracebacks != NULL) { +        _Py_hashtable_foreach(get_traces.tracebacks, +                         tracemalloc_pyobject_decref_cb, NULL); +        _Py_hashtable_destroy(get_traces.tracebacks); +    } +    if (get_traces.traces != NULL) +        _Py_hashtable_destroy(get_traces.traces); + +    return get_traces.list; +} + +PyDoc_STRVAR(tracemalloc_get_object_traceback_doc, +    "get_object_traceback(obj)\n" +    "\n" +    "Get the traceback where the Python object obj was allocated.\n" +    "Return a tuple of (filename: str, lineno: int) tuples.\n" +    "\n" +    "Return None if the tracemalloc module is disabled or did not\n" +    "trace the allocation of the object."); + +static PyObject* +py_tracemalloc_get_object_traceback(PyObject *self, PyObject *obj) +{ +    PyTypeObject *type; +    void *ptr; +    trace_t trace; +    int found; + +    if (!tracemalloc_config.tracing) +        Py_RETURN_NONE; + +    type = Py_TYPE(obj); +    if (PyType_IS_GC(type)) +        ptr = (void *)((char *)obj - sizeof(PyGC_Head)); +    else +        ptr = (void *)obj; + +    TABLES_LOCK(); +    found = _Py_HASHTABLE_GET(tracemalloc_traces, ptr, trace); +    TABLES_UNLOCK(); + +    if (!found) +        Py_RETURN_NONE; + +    return traceback_to_pyobject(trace.traceback, NULL); +} + +static PyObject* +tracemalloc_atexit(PyObject *self) +{ +#ifdef WITH_THREAD +    assert(PyGILState_Check()); +#endif +    tracemalloc_deinit(); +    Py_RETURN_NONE; +} + +static PyMethodDef atexit_method = { +    "_atexit", (PyCFunction)tracemalloc_atexit, METH_NOARGS, NULL}; + +static int +tracemalloc_atexit_register(void) +{ +    PyObject *method = NULL, *atexit = NULL, *func = NULL; +    PyObject *result; +    int ret = -1; + +    if (tracemalloc_config.atexit_registered) +        return 0; +    tracemalloc_config.atexit_registered = 1; + +    /* private functions */ +    method = PyCFunction_New(&atexit_method, NULL); +    if (method == NULL) +        goto done; + +    atexit = PyImport_ImportModule("atexit"); +    if (atexit == NULL) { +        if (!PyErr_Warn(PyExc_ImportWarning, +                       "atexit module is missing: " +                       "cannot automatically disable tracemalloc at exit")) +        { +            PyErr_Clear(); +            return 0; +        } +        goto done; +    } + +    func = PyObject_GetAttrString(atexit, "register"); +    if (func == NULL) +        goto done; + +    result = PyObject_CallFunction(func, "O", method); +    if (result == NULL) +        goto done; +    Py_DECREF(result); + +    ret = 0; + +done: +    Py_XDECREF(method); +    Py_XDECREF(func); +    Py_XDECREF(atexit); +    return ret; +} + +PyDoc_STRVAR(tracemalloc_start_doc, +    "start()\n" +    "\n" +    "Start tracing Python memory allocations."); + +static PyObject* +py_tracemalloc_start(PyObject *self) +{ +    if (tracemalloc_start() < 0) +        return NULL; + +    Py_RETURN_NONE; +} + +PyDoc_STRVAR(tracemalloc_stop_doc, +    "stop()\n" +    "\n" +    "Stop tracing Python memory allocations and clear traces\n" +    "of memory blocks allocated by Python."); + +static PyObject* +py_tracemalloc_stop(PyObject *self) +{ +    tracemalloc_stop(); +    Py_RETURN_NONE; +} + +PyDoc_STRVAR(tracemalloc_get_traceback_limit_doc, +    "get_traceback_limit() -> int\n" +    "\n" +    "Get the maximum number of frames stored in the traceback\n" +    "of a trace.\n" +    "\n" +    "By default, a trace of an allocated memory block only stores\n" +    "the most recent frame: the limit is 1."); + +static PyObject* +py_tracemalloc_get_traceback_limit(PyObject *self) +{ +    return PyLong_FromLong(tracemalloc_config.max_nframe); +} + +PyDoc_STRVAR(tracemalloc_set_traceback_limit_doc, +    "set_traceback_limit(nframe: int)\n" +    "\n" +    "Set the maximum number of frames stored in the traceback of a trace."); + +static PyObject* +tracemalloc_set_traceback_limit(PyObject *self, PyObject *args) +{ +    Py_ssize_t nframe; + +    if (!PyArg_ParseTuple(args, "n:set_traceback_limit", +                          &nframe)) +        return NULL; + +    if (nframe < 1 || nframe > MAX_NFRAME) { +        PyErr_Format(PyExc_ValueError, +                     "the number of frames must be in range [1; %i]", +                     MAX_NFRAME); +        return NULL; +    } +    tracemalloc_config.max_nframe = Py_SAFE_DOWNCAST(nframe, Py_ssize_t, int); + +    Py_RETURN_NONE; +} + +PyDoc_STRVAR(tracemalloc_get_tracemalloc_memory_doc, +    "get_tracemalloc_memory() -> int\n" +    "\n" +    "Get the memory usage in bytes of the tracemalloc module\n" +    "used internally to trace memory allocations."); + +static PyObject* +tracemalloc_get_tracemalloc_memory(PyObject *self) +{ +    size_t size; +    PyObject *size_obj; + +    size = _Py_hashtable_size(tracemalloc_tracebacks); +    size += _Py_hashtable_size(tracemalloc_filenames); + +    TABLES_LOCK(); +    size += _Py_hashtable_size(tracemalloc_traces); +    TABLES_UNLOCK(); + +    size_obj = PyLong_FromSize_t(size); +    return Py_BuildValue("N", size_obj); +} + +PyDoc_STRVAR(tracemalloc_get_traced_memory_doc, +    "get_traced_memory() -> int\n" +    "\n" +    "Get the current size and maximum size of memory blocks traced\n" +    "by the tracemalloc module as a tuple: (size: int, max_size: int)."); + +static PyObject* +tracemalloc_get_traced_memory(PyObject *self) +{ +    Py_ssize_t size, max_size; +    PyObject *size_obj, *max_size_obj; + +    if (!tracemalloc_config.tracing) +        return Py_BuildValue("ii", 0, 0); + +    TABLES_LOCK(); +    size = tracemalloc_traced_memory; +    max_size = tracemalloc_max_traced_memory; +    TABLES_UNLOCK(); + +    size_obj = PyLong_FromSize_t(size); +    max_size_obj = PyLong_FromSize_t(max_size); +    return Py_BuildValue("NN", size_obj, max_size_obj); +} + +static PyMethodDef module_methods[] = { +    {"is_tracing", (PyCFunction)py_tracemalloc_is_tracing, +     METH_NOARGS, tracemalloc_is_tracing_doc}, +    {"clear_traces", (PyCFunction)py_tracemalloc_clear_traces, +     METH_NOARGS, tracemalloc_clear_traces_doc}, +    {"_get_traces", (PyCFunction)py_tracemalloc_get_traces, +     METH_NOARGS, tracemalloc_get_traces_doc}, +    {"_get_object_traceback", (PyCFunction)py_tracemalloc_get_object_traceback, +     METH_O, tracemalloc_get_object_traceback_doc}, +    {"start", (PyCFunction)py_tracemalloc_start, +      METH_NOARGS, tracemalloc_start_doc}, +    {"stop", (PyCFunction)py_tracemalloc_stop, +      METH_NOARGS, tracemalloc_stop_doc}, +    {"get_traceback_limit", (PyCFunction)py_tracemalloc_get_traceback_limit, +     METH_NOARGS, tracemalloc_get_traceback_limit_doc}, +    {"set_traceback_limit", (PyCFunction)tracemalloc_set_traceback_limit, +     METH_VARARGS, tracemalloc_set_traceback_limit_doc}, +    {"get_tracemalloc_memory", (PyCFunction)tracemalloc_get_tracemalloc_memory, +     METH_NOARGS, tracemalloc_get_tracemalloc_memory_doc}, +    {"get_traced_memory", (PyCFunction)tracemalloc_get_traced_memory, +     METH_NOARGS, tracemalloc_get_traced_memory_doc}, + +    /* sentinel */ +    {NULL, NULL} +}; + +PyDoc_STRVAR(module_doc, +"Debug module to trace memory blocks allocated by Python."); + +static struct PyModuleDef module_def = { +    PyModuleDef_HEAD_INIT, +    "_tracemalloc", +    module_doc, +    0, /* non-negative size to be able to unload the module */ +    module_methods, +    NULL, +}; + +PyMODINIT_FUNC +PyInit__tracemalloc(void) +{ +    PyObject *m; +    m = PyModule_Create(&module_def); +    if (m == NULL) +        return NULL; + +    if (tracemalloc_init() < 0) +        return NULL; + +    return m; +} + +static int +parse_sys_xoptions(PyObject *value) +{ +    PyObject *valuelong; +    long nframe; + +    if (value == Py_True) +        return 1; + +    assert(PyUnicode_Check(value)); +    if (PyUnicode_GetLength(value) == 0) +        return -1; + +    valuelong = PyLong_FromUnicodeObject(value, 10); +    if (valuelong == NULL) +        return -1; + +    nframe = PyLong_AsLong(valuelong); +    Py_DECREF(valuelong); +    if (nframe == -1 && PyErr_Occurred()) +        return -1; + +    if (nframe < 1 || nframe > MAX_NFRAME) +        return -1; + +    return Py_SAFE_DOWNCAST(nframe, long, int); +} + +int +_PyTraceMalloc_Init(void) +{ +    char *p; +    int nframe; + +#ifdef WITH_THREAD +    assert(PyGILState_Check()); +#endif + +    if ((p = Py_GETENV("PYTHONTRACEMALLOC")) && *p != '\0') { +        char *endptr = p; +        unsigned long value; + +        value = strtoul(p, &endptr, 10); +        if (*endptr != '\0' +            || value < 1 +            || value > MAX_NFRAME +            || (errno == ERANGE && value == ULONG_MAX)) +        { +            Py_FatalError("PYTHONTRACEMALLOC must be an integer " +                          "in range [1; " STR(MAX_NFRAME) "]"); +            return -1; +        } + +        nframe = (int)value; +    } +    else { +        PyObject *xoptions, *key, *value; + +        xoptions = PySys_GetXOptions(); +        if (xoptions == NULL) +            return -1; + +        key = PyUnicode_FromString("tracemalloc"); +        if (key == NULL) +            return -1; + +        value = PyDict_GetItemWithError(xoptions, key); +        Py_DECREF(key); +        if (value == NULL) { +            if (PyErr_Occurred()) +                return -1; + +            /* -X tracemalloc is not used */ +            return 0; +        } + +        nframe = parse_sys_xoptions(value); +        Py_DECREF(value); +        if (nframe < 0) { +            Py_FatalError("-X tracemalloc=NFRAME: number of frame must be " +                          "an integer in range [1; " STR(MAX_NFRAME) "]"); +        } +    } + +    tracemalloc_config.max_nframe = nframe; +    return tracemalloc_start(); +} + | 
