add test.

This commit is contained in:
amass 2024-03-25 00:09:33 +08:00
parent 4178f405f2
commit d745f7d8c6
6 changed files with 100 additions and 39 deletions

View File

@ -26,7 +26,9 @@ set(ZLMediaKit_ROOT /opt/aarch64-v01c01-linux-gnu-gcc/lib/ZLMediaKit)
set(ZLMediaKit_INCLUDE_DIR ${ZLMediaKit_ROOT}/include) set(ZLMediaKit_INCLUDE_DIR ${ZLMediaKit_ROOT}/include)
set(ZLMediaKit_LIBRARY_DIRS ${ZLMediaKit_ROOT}/lib) set(ZLMediaKit_LIBRARY_DIRS ${ZLMediaKit_ROOT}/lib)
if(${CROSS_BUILD})
add_subdirectory(Main) add_subdirectory(Main)
endif()
add_subdirectory(Tools) add_subdirectory(Tools)
include(FetchContent) include(FetchContent)

View File

@ -3,6 +3,7 @@
#include <fstream> #include <fstream>
#include <functional> #include <functional>
#include <memory>
#include <string> #include <string>
#include <thread> #include <thread>

View File

@ -3,6 +3,10 @@ add_executable(LeakTracer main.cpp
MemoryAllocationStackTracer.h MemoryAllocationStackTracer.cpp MemoryAllocationStackTracer.h MemoryAllocationStackTracer.cpp
) )
target_compile_definitions(LeakTracer
PRIVATE BOOST_STACKTRACE_USE_ADDR2LINE
)
target_include_directories(LeakTracer target_include_directories(LeakTracer
PRIVATE ${Libraries_ROOT}/LeakTracer/include PRIVATE ${Libraries_ROOT}/LeakTracer/include
) )

View File

@ -1,63 +1,98 @@
#include "MemoryAllocationStackTracer.h" #include "MemoryAllocationStackTracer.h"
#include <fstream>
#include <iostream> #include <iostream>
#include <sstream>
#include <stdio.h> #include <stdio.h>
extern "C" { extern "C" {
void *__wrap_malloc(size_t size) { void *__wrap_malloc(size_t size) {
void *address = __real_malloc(size); void *address = __real_malloc(size);
// if (address != nullptr) { auto self = MemoryAllocationStackTracer::instance();
// MemoryAllocationStackTracer::instance()->push(reinterpret_cast<uint64_t>(address), if (address != nullptr && self->enabled()) {
// boost::stacktrace::stacktrace()); self->push(reinterpret_cast<uint64_t>(address), MemoryAllocationStackTracer::Frame());
// } }
// std::cout << "malloc " << size << " bytes." << std::endl; // printf("%p = malloc(%zu)\n", address, size);
return address; return address;
} }
void __wrap_free(void *__ptr) { void __wrap_free(void *__ptr) {
// if (__ptr != nullptr) { if (__ptr != nullptr) {
// MemoryAllocationStackTracer::instance()->pop(reinterpret_cast<uint64_t>(__ptr)); MemoryAllocationStackTracer::instance()->pop(reinterpret_cast<uint64_t>(__ptr));
// } }
// printf("free(%p)\n", __ptr);
return __real_free(__ptr); return __real_free(__ptr);
} }
} }
void *operator new(std::size_t size) { void *operator new(std::size_t size) {
printf("new %d %d\n ", size, sizeof(boost::stacktrace::stacktrace)); // printf("void *operator new(std::size_t size),%zu\n", size);
void *address = __real_malloc(size);
// if (size != sizeof(boost::stacktrace::stacktrace) || size != 2048) { return malloc(size);
// MemoryAllocationStackTracer::instance()->push(reinterpret_cast<uint64_t>(address),
// boost::stacktrace::stacktrace());
// }
return address;
} }
void operator delete(void *p) { void operator delete(void *p) noexcept {
free(p); // printf("%s", "void operator delete(void *p)\n");
return free(p);
}
void *operator new[](std::size_t count) {
// printf("%s", "void *operator new[](std::size_t count)\n");
return malloc(count);
}
void operator delete[](void *ptr) noexcept {
// printf("%s", "void operator delete[](void *ptr) noexcept\n");
return free(ptr);
} }
MemoryAllocationStackTracer *MemoryAllocationStackTracer::m_instance = nullptr; MemoryAllocationStackTracer *MemoryAllocationStackTracer::m_instance = nullptr;
void MemoryAllocationStackTracer::initialize() { void MemoryAllocationStackTracer::start() {
if (!m_enabled) {
m_enabled = true;
}
}
void MemoryAllocationStackTracer::push(uint64_t address, Frame &&stacktrace) {
Infomation info;
info.time = std::chrono::system_clock::now();
info.frame = std::move(stacktrace);
m_stacktraces.insert({address, info});
// m_stacktraces.insert({address, stacktrace});
}
void MemoryAllocationStackTracer::pop(uint64_t address) {
if (m_stacktraces.count(address) > 0) {
m_stacktraces.erase(address);
}
}
MemoryAllocationStackTracer *MemoryAllocationStackTracer::instance() {
if (m_instance == nullptr) { if (m_instance == nullptr) {
auto buffer = __real_malloc(sizeof(MemoryAllocationStackTracer)); auto buffer = __real_malloc(sizeof(MemoryAllocationStackTracer));
if (buffer != nullptr) { if (buffer != nullptr) {
m_instance = new (buffer) MemoryAllocationStackTracer(); m_instance = new (buffer) MemoryAllocationStackTracer();
} }
} }
}
void MemoryAllocationStackTracer::push(uint64_t address, const boost::stacktrace::stacktrace &stacktrace) {
m_stacktraces.insert({address, stacktrace});
}
void MemoryAllocationStackTracer::pop(uint64_t address) {
m_stacktraces.erase(address);
}
MemoryAllocationStackTracer *MemoryAllocationStackTracer::instance() {
return m_instance; return m_instance;
} }
void MemoryAllocationStackTracer::dump() {
using namespace std::chrono;
m_enabled = false;
std::cout << "size: " << m_stacktraces.size() << std::endl;
int index = 0;
auto now = std::chrono::system_clock::now();
std::ofstream ofs("test.txt");
for (auto &info : m_stacktraces) {
std::ostringstream oss;
auto duration = duration_cast<seconds>(now - info.second.time);
oss << "-----index[" << index << "] duration: " << duration.count() << "s:-----" << std::endl;
oss << info.second.frame << std::endl;
auto content = oss.str();
ofs << content;
std::cout << content;
index++;
}
m_enabled = true;
}

View File

@ -2,6 +2,7 @@
#define __MEMORYALLOCATIONSTACKTRACER_H__ #define __MEMORYALLOCATIONSTACKTRACER_H__
#include <boost/stacktrace/stacktrace.hpp> #include <boost/stacktrace/stacktrace.hpp>
#include <chrono>
#include <unordered_map> #include <unordered_map>
extern "C" { extern "C" {
@ -33,7 +34,7 @@ public:
} }
T *allocate(std::size_t n) { T *allocate(std::size_t n) {
return reinterpret_cast<T *>(__real_malloc(n)); return reinterpret_cast<T *>(__real_malloc(n * sizeof(T)));
} }
void deallocate(T *p, std::size_t n) { void deallocate(T *p, std::size_t n) {
__real_free(p); __real_free(p);
@ -60,17 +61,28 @@ public:
class MemoryAllocationStackTracer { class MemoryAllocationStackTracer {
public: public:
static void initialize(); using Frame = boost::stacktrace::basic_stacktrace<DebugAllocator<boost::stacktrace::frame>>;
class Infomation {
public:
std::chrono::system_clock::time_point time;
Frame frame;
};
static MemoryAllocationStackTracer *instance(); static MemoryAllocationStackTracer *instance();
void push(uint64_t address, const boost::stacktrace::stacktrace &stacktrace); inline bool enabled() const {
return m_enabled;
}
void start();
void dump();
void push(uint64_t address, Frame &&stacktrace);
void pop(uint64_t address); void pop(uint64_t address);
private: private:
std::unordered_map<uint64_t, boost::stacktrace::stacktrace, std::hash<uint64_t>, std::equal_to<uint64_t>, std::unordered_map<uint64_t, Infomation, std::hash<uint64_t>, std::equal_to<uint64_t>,
DebugAllocator<MemoryAllocationStackTracer>> DebugAllocator<std::pair<const uint64_t, Infomation>>>
m_stacktraces; m_stacktraces;
static MemoryAllocationStackTracer *m_instance; static MemoryAllocationStackTracer *m_instance;
bool m_enabled = false;
}; };
#endif // __MEMORYALLOCATIONSTACKTRACER_H__ #endif // __MEMORYALLOCATIONSTACKTRACER_H__

View File

@ -1,3 +1,4 @@
#include "BoostLog.h"
#include "LeakTracer.h" #include "LeakTracer.h"
#include "MemoryAllocationStackTracer.h" #include "MemoryAllocationStackTracer.h"
#include <boost/stacktrace.hpp> #include <boost/stacktrace.hpp>
@ -31,7 +32,13 @@ class Test {
}; };
int main(int argc, char const *argv[]) { int main(int argc, char const *argv[]) {
MemoryAllocationStackTracer::initialize(); LOG(info) << "hello world.";
MemoryAllocationStackTracer::instance()->start();
auto a = new Test(); auto a = new Test();
make_memory_leak();
MemoryAllocationStackTracer::instance()->dump();
return 0; return 0;
} }