linux memory tracking

master
Alex Baines 2020-02-16 22:47:42 +00:00
parent a3b096df1c
commit 687cfd7850
2 changed files with 76 additions and 9 deletions

View File

@ -140,6 +140,13 @@ struct Linux_Input_Chunk {
Linux_Input_Chunk_Persistent pers; Linux_Input_Chunk_Persistent pers;
}; };
struct Linux_Memory_Tracker_Node {
Linux_Memory_Tracker_Node* prev;
Linux_Memory_Tracker_Node* next;
String_Const_u8 location;
u64 size;
};
struct Linux_Vars { struct Linux_Vars {
Thread_Context tctx; Thread_Context tctx;
Arena *frame_arena; Arena *frame_arena;
@ -173,6 +180,10 @@ struct Linux_Vars {
Node timer_objects; Node timer_objects;
System_Mutex global_frame_mutex; System_Mutex global_frame_mutex;
pthread_mutex_t memory_tracker_mutex;
Linux_Memory_Tracker_Node* memory_tracker_head;
Linux_Memory_Tracker_Node* memory_tracker_tail;
int memory_tracker_count;
Arena* clipboard_arena; Arena* clipboard_arena;
String_Const_u8 clipboard_contents; String_Const_u8 clipboard_contents;
@ -1552,6 +1563,11 @@ linux_epoll_process(struct epoll_event* events, int num_events) {
int int
main(int argc, char **argv){ main(int argc, char **argv){
pthread_mutexattr_t attr;
pthread_mutexattr_init(&attr);
pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
pthread_mutex_init(&linuxvars.memory_tracker_mutex, &attr);
// NOTE(allen): context setup // NOTE(allen): context setup
{ {
Base_Allocator* alloc = get_base_allocator_system(); Base_Allocator* alloc = get_base_allocator_system();

View File

@ -616,20 +616,42 @@ system_condition_variable_signal(System_Condition_Variable cv){
internal void internal void
system_condition_variable_free(System_Condition_Variable cv){ system_condition_variable_free(System_Condition_Variable cv){
Linux_Object* object = *(Linux_Object**)&cv; Linux_Object* object = *(Linux_Object**)&cv;
//LINUX_FN_DEBUG("%p", object); LINUX_FN_DEBUG("%p", &object->condition_variable);
Assert(object->kind == LinuxObjectKind_ConditionVariable); Assert(object->kind == LinuxObjectKind_ConditionVariable);
pthread_cond_destroy(&object->condition_variable); pthread_cond_destroy(&object->condition_variable);
linux_free_object(object); linux_free_object(object);
} }
#define MEMORY_PREFIX_SIZE 64
internal void* internal void*
system_memory_allocate(u64 size, String_Const_u8 location){ system_memory_allocate(u64 size, String_Const_u8 location){
void* result = mmap( static_assert(MEMORY_PREFIX_SIZE >= sizeof(Memory_Annotation_Node));
NULL, size, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); u64 adjusted_size = size + MEMORY_PREFIX_SIZE;
// TODO(andrew): Allocation tracking?
//LINUX_FN_DEBUG("%" PRIu64 ", %.*s %p", size, (int)location.size, location.str, result); Assert(adjusted_size > size);
return result;
const int prot = PROT_READ | PROT_WRITE;
const int flags = MAP_PRIVATE | MAP_ANONYMOUS;
void* result = mmap(NULL, adjusted_size, prot, flags, -1, 0);
if(result == MAP_FAILED) {
perror("mmap");
return NULL;
}
Linux_Memory_Tracker_Node* node = (Linux_Memory_Tracker_Node*)result;
node->location = location;
node->size = size;
pthread_mutex_lock(&linuxvars.memory_tracker_mutex);
zdll_push_back(linuxvars.memory_tracker_head, linuxvars.memory_tracker_tail, node);
linuxvars.memory_tracker_count++;
pthread_mutex_unlock(&linuxvars.memory_tracker_mutex);
return (u8*)result + MEMORY_PREFIX_SIZE;
} }
internal b32 internal b32
@ -645,14 +667,43 @@ system_memory_set_protection(void* ptr, u64 size, u32 flags){
internal void internal void
system_memory_free(void* ptr, u64 size){ system_memory_free(void* ptr, u64 size){
//LINUX_FN_DEBUG("%p / %ld", ptr, size); u64 adjusted_size = size + MEMORY_PREFIX_SIZE;
munmap(ptr, size); Linux_Memory_Tracker_Node* node = (Linux_Memory_Tracker_Node*)((u8*)ptr - MEMORY_PREFIX_SIZE);
pthread_mutex_lock(&linuxvars.memory_tracker_mutex);
zdll_remove(linuxvars.memory_tracker_head, linuxvars.memory_tracker_tail, node);
linuxvars.memory_tracker_count--;
pthread_mutex_unlock(&linuxvars.memory_tracker_mutex);
if(munmap(node, adjusted_size) == -1) {
perror("munmap");
}
} }
internal Memory_Annotation internal Memory_Annotation
system_memory_annotation(Arena* arena){ system_memory_annotation(Arena* arena){
LINUX_FN_DEBUG(); LINUX_FN_DEBUG();
// TODO;
Memory_Annotation result;
Memory_Annotation_Node** ptr = &result.first;
pthread_mutex_lock(&linuxvars.memory_tracker_mutex);
for(Linux_Memory_Tracker_Node* node = linuxvars.memory_tracker_head; node; node = node->next) {
*ptr = push_array(arena, Memory_Annotation_Node, 1);
(*ptr)->location = node->location;
(*ptr)->size = node->size;
(*ptr)->address = (u8*)node + MEMORY_PREFIX_SIZE;
ptr = &(*ptr)->next;
result.count++;
}
pthread_mutex_unlock(&linuxvars.memory_tracker_mutex);
*ptr = NULL;
result.last = CastFromMember(Memory_Annotation_Node, next, ptr);
return result;
} }
internal void internal void