aboutsummaryrefslogtreecommitdiff
path: root/src/rt/memory_region.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/rt/memory_region.cpp')
-rw-r--r--src/rt/memory_region.cpp100
1 files changed, 100 insertions, 0 deletions
diff --git a/src/rt/memory_region.cpp b/src/rt/memory_region.cpp
new file mode 100644
index 00000000..797a7c1d
--- /dev/null
+++ b/src/rt/memory_region.cpp
@@ -0,0 +1,100 @@
+/*
+ *
+ */
+
+#include "rust_internal.h"
+#include "memory_region.h"
+
+#define TRACK_ALLOCATIONS
+
+memory_region::memory_region(rust_srv *srv, bool synchronized) :
+ _srv(srv), _parent(NULL), _live_allocations(0),
+ _synchronized(synchronized) {
+ // Nop.
+}
+
+memory_region::memory_region(memory_region *parent) :
+ _srv(parent->_srv), _parent(parent), _live_allocations(0),
+ _synchronized(parent->_synchronized) {
+ // Nop.
+}
+
+void memory_region::free(void *mem) {
+ if (_synchronized) { _lock.lock(); }
+#ifdef TRACK_ALLOCATIONS
+ if (_allocation_list.replace(mem, NULL) == false) {
+ printf("free: ptr 0x%" PRIxPTR " is not in allocation_list\n",
+ (uintptr_t) mem);
+ _srv->fatal("not in allocation_list", __FILE__, __LINE__, "");
+ }
+#endif
+ if (_live_allocations < 1) {
+ _srv->fatal("live_allocs < 1", __FILE__, __LINE__, "");
+ }
+ _live_allocations--;
+ _srv->free(mem);
+ if (_synchronized) { _lock.unlock(); }
+
+}
+
+void *
+memory_region::realloc(void *mem, size_t size) {
+ if (_synchronized) { _lock.lock(); }
+ if (!mem) {
+ _live_allocations++;
+ }
+ void *newMem = _srv->realloc(mem, size);
+#ifdef TRACK_ALLOCATIONS
+ if (_allocation_list.replace(mem, newMem) == false) {
+ printf("realloc: ptr 0x%" PRIxPTR " is not in allocation_list\n",
+ (uintptr_t) mem);
+ _srv->fatal("not in allocation_list", __FILE__, __LINE__, "");
+ }
+#endif
+ if (_synchronized) { _lock.unlock(); }
+ return newMem;
+}
+
+void *
+memory_region::malloc(size_t size) {
+ if (_synchronized) { _lock.lock(); }
+ _live_allocations++;
+ void *mem = _srv->malloc(size);
+#ifdef TRACK_ALLOCATIONS
+ _allocation_list.append(mem);
+#endif
+ if (_synchronized) { _lock.unlock(); }
+ return mem;
+}
+
+void *
+memory_region::calloc(size_t size) {
+ if (_synchronized) { _lock.lock(); }
+ _live_allocations++;
+ void *mem = _srv->malloc(size);
+ memset(mem, 0, size);
+#ifdef TRACK_ALLOCATIONS
+ _allocation_list.append(mem);
+#endif
+ if (_synchronized) { _lock.unlock(); }
+ return mem;
+}
+
+memory_region::~memory_region() {
+ if (_live_allocations == 0) {
+ return;
+ }
+ char msg[128];
+ snprintf(msg, sizeof(msg),
+ "leaked memory in rust main loop (%" PRIuPTR " objects)",
+ _live_allocations);
+#ifdef TRACK_ALLOCATIONS
+ for (size_t i = 0; i < _allocation_list.size(); i++) {
+ if (_allocation_list[i] != NULL) {
+ printf("allocation 0x%" PRIxPTR " was not freed\n",
+ (uintptr_t) _allocation_list[i]);
+ }
+ }
+#endif
+ _srv->fatal(msg, __FILE__, __LINE__, "%d objects", _live_allocations);
+}