diff options
| author | Alex Crichton <alex@alexcrichton.com> | 2013-10-26 01:10:39 -0700 |
|---|---|---|
| committer | Alex Crichton <alex@alexcrichton.com> | 2013-10-26 01:10:39 -0700 |
| commit | 357ef1f69cd1ffbd3c36a9e7e171b10263d94139 (patch) | |
| tree | fdbf34015e80a82799979b60454dcf9c01cafb56 /src/rt/memory_region.cpp | |
| parent | c5074ae6463876bbb9511e980fe890bebc881e41 (diff) | |
| download | rust-357ef1f69cd1ffbd3c36a9e7e171b10263d94139.tar.gz rust-357ef1f69cd1ffbd3c36a9e7e171b10263d94139.zip | |
Rewrite boxed_region/memory_region in Rust
This drops more of the old C++ runtime to rather be written in rust. A few features were lost along the way, but hopefully not too many. The main loss is that there are no longer backtraces associated with allocations (rust doesn't have a way of acquiring those just yet). Other than that though, I believe that the rest of the debugging utilities made their way over into rust. Closes #8704
Diffstat (limited to 'src/rt/memory_region.cpp')
| -rw-r--r-- | src/rt/memory_region.cpp | 258 |
1 files changed, 0 insertions, 258 deletions
diff --git a/src/rt/memory_region.cpp b/src/rt/memory_region.cpp deleted file mode 100644 index 4a34312c6d4..00000000000 --- a/src/rt/memory_region.cpp +++ /dev/null @@ -1,258 +0,0 @@ -// Copyright 2012 The Rust Project Developers. See the COPYRIGHT -// file at the top-level directory of this distribution and at -// http://rust-lang.org/COPYRIGHT. -// -// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or -// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license -// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your -// option. This file may not be copied, modified, or distributed -// except according to those terms. - - -#include "memory_region.h" - -#if RUSTRT_TRACK_ALLOCATIONS >= 3 -#include <execinfo.h> -#endif - -#if RUSTRT_TRACK_ALLOCATIONS >= 1 -// For some platforms, 16 byte alignment is required. -# define PTR_SIZE 16 -# define ALIGN_PTR(x) (((x)+PTR_SIZE-1)/PTR_SIZE*PTR_SIZE) -# define HEADER_SIZE ALIGN_PTR(sizeof(alloc_header)) -# define MAGIC 0xbadc0ffe -#else -# define HEADER_SIZE 0 -#endif - -memory_region::alloc_header *memory_region::get_header(void *mem) { - return (alloc_header *)((char *)mem - HEADER_SIZE); -} - -void *memory_region::get_data(alloc_header *ptr) { - return (void*)((char *)ptr + HEADER_SIZE); -} - -inline void memory_region::maybe_print_backtrace(const alloc_header *header) const { -# if RUSTRT_TRACK_ALLOCATIONS >= 3 - if (_detailed_leaks) { - backtrace_symbols_fd(header->bt + 1, header->btframes - 1, 2); - } -# endif -} - -memory_region::memory_region(bool detailed_leaks, - bool poison_on_free) : - _parent(NULL), _live_allocations(0), - _detailed_leaks(detailed_leaks), - _poison_on_free(poison_on_free) { -} - -memory_region::memory_region(memory_region *parent) : - _parent(parent), _live_allocations(0), - _detailed_leaks(parent->_detailed_leaks), - _poison_on_free(parent->_poison_on_free) { -} - -void memory_region::add_alloc() { - _live_allocations++; -} - -void memory_region::dec_alloc() { - _live_allocations--; -} - -void memory_region::free(void *mem) { - // printf("free: ptr 0x%" PRIxPTR" region=%p\n", (uintptr_t) mem, this); - if (!mem) { return; } - alloc_header *alloc = get_header(mem); - -# if RUSTRT_TRACK_ALLOCATIONS >= 1 - assert(alloc->magic == MAGIC); -# endif - - if (_live_allocations < 1) { - assert(false && "live_allocs < 1"); - } - release_alloc(mem); - maybe_poison(mem); - ::free(alloc); -} - -void * -memory_region::realloc(void *mem, size_t orig_size) { - if (!mem) { - add_alloc(); - } - - alloc_header *alloc = get_header(mem); -# if RUSTRT_TRACK_ALLOCATIONS >= 1 - assert(alloc->magic == MAGIC); -# endif - - size_t size = orig_size + HEADER_SIZE; - alloc_header *newMem = (alloc_header *)::realloc(alloc, size); - if (newMem == NULL) { - fprintf(stderr, - "memory_region::realloc> " - "Out of memory allocating %ld bytes", - (long int) size); - abort(); - } - -# if RUSTRT_TRACK_ALLOCATIONS >= 1 - assert(newMem->magic == MAGIC); - newMem->size = orig_size; -# endif - -# if RUSTRT_TRACK_ALLOCATIONS >= 2 - if (_allocation_list[newMem->index] != alloc) { - printf("at index %d, found %p, expected %p\n", - alloc->index, _allocation_list[alloc->index], alloc); - printf("realloc: ptr 0x%" PRIxPTR " (%s) is not in allocation_list\n", - (uintptr_t) get_data(alloc), alloc->tag); - assert(false && "not in allocation_list"); - } - else { - _allocation_list[newMem->index] = newMem; - // printf("realloc: stored %p at index %d, replacing %p\n", - // newMem, index, mem); - } -# endif - - return get_data(newMem); -} - -void * -memory_region::malloc(size_t size, const char *tag) { -# if RUSTRT_TRACK_ALLOCATIONS >= 1 - size_t old_size = size; -# endif - size += HEADER_SIZE; - alloc_header *mem = (alloc_header *)::malloc(size); - if (mem == NULL) { - fprintf(stderr, - "memory_region::malloc> " - "Out of memory allocating %ld bytes", - (long int) size); - abort(); - } - -# if RUSTRT_TRACK_ALLOCATIONS >= 1 - mem->magic = MAGIC; - mem->tag = tag; - mem->index = -1; - mem->size = old_size; -# endif - - void *data = get_data(mem); - claim_alloc(data); - - return data; -} - -memory_region::~memory_region() { - if (_live_allocations == 0 && !_detailed_leaks) { - return; - } - char msg[128]; - if(_live_allocations > 0) { - snprintf(msg, sizeof(msg), - "leaked memory in rust main loop (%d objects)", - _live_allocations); - } - -# if RUSTRT_TRACK_ALLOCATIONS >= 2 - if (_detailed_leaks) { - int leak_count = 0; - for (size_t i = 0; i < _allocation_list.size(); i++) { - if (_allocation_list[i] != NULL) { - alloc_header *header = (alloc_header*)_allocation_list[i]; - printf("allocation (%s) 0x%" PRIxPTR " was not freed\n", - header->tag, - (uintptr_t) get_data(header)); - ++leak_count; - maybe_print_backtrace(header); - } - } - assert(leak_count == _live_allocations); - } -# endif - - if (_live_allocations > 0) { - fprintf(stderr, "%s\n", msg); - assert(false); - } -} - -void -memory_region::release_alloc(void *mem) { -# if RUSTRT_TRACK_ALLOCATIONS >= 1 - alloc_header *alloc = get_header(mem); - assert(alloc->magic == MAGIC); -# endif - -# if RUSTRT_TRACK_ALLOCATIONS >= 2 - if (((size_t) alloc->index) >= _allocation_list.size()) { - printf("free: ptr 0x%" PRIxPTR " (%s) index %d is beyond allocation_list of size %zu\n", - (uintptr_t) get_data(alloc), alloc->tag, alloc->index, _allocation_list.size()); - maybe_print_backtrace(alloc); - assert(false && "index beyond allocation_list"); - } - if (_allocation_list[alloc->index] != alloc) { - printf("free: ptr 0x%" PRIxPTR " (%s) is not in allocation_list\n", - (uintptr_t) get_data(alloc), alloc->tag); - maybe_print_backtrace(alloc); - assert(false && "not in allocation_list"); - } - else { - // printf("freed index %d\n", index); - _allocation_list[alloc->index] = NULL; - alloc->index = -1; - } -# endif - - dec_alloc(); -} - -void -memory_region::claim_alloc(void *mem) { -# if RUSTRT_TRACK_ALLOCATIONS >= 1 - alloc_header *alloc = get_header(mem); - assert(alloc->magic == MAGIC); -# endif - -# if RUSTRT_TRACK_ALLOCATIONS >= 2 - alloc->index = _allocation_list.append(alloc); -# endif - -# if RUSTRT_TRACK_ALLOCATIONS >= 3 - if (_detailed_leaks) { - alloc->btframes = ::backtrace(alloc->bt, 32); - } -# endif - - add_alloc(); -} - -void -memory_region::maybe_poison(void *mem) { - - if (!_poison_on_free) - return; - -# if RUSTRT_TRACK_ALLOCATIONS >= 1 - alloc_header *alloc = get_header(mem); - memset(mem, '\xcd', alloc->size); -# endif -} - -// -// Local Variables: -// mode: C++ -// fill-column: 78; -// indent-tabs-mode: nil -// c-basic-offset: 4 -// buffer-file-coding-system: utf-8-unix -// End: -// |
