Make safe_ptr able to track callstacks
The code stays disabled, as the performance penalty is quite high, but can be enabled to help debugging safe pointers.
This commit is contained in:
parent
1b54b65bb5
commit
d356ae2419
|
@ -9,7 +9,7 @@ bindir := $(DESTDIR)$(PREFIX)/bin
|
||||||
sharedir := $(DESTDIR)$(PREFIX)/share/kak
|
sharedir := $(DESTDIR)$(PREFIX)/share/kak
|
||||||
docdir := $(DESTDIR)$(PREFIX)/share/doc/kak
|
docdir := $(DESTDIR)$(PREFIX)/share/doc/kak
|
||||||
|
|
||||||
CXXFLAGS += -std=gnu++11 -g -Wall -Wno-reorder -Wno-sign-compare -pedantic
|
CXXFLAGS += -std=gnu++11 -g -Wall -Wno-reorder -Wno-sign-compare -pedantic -rdynamic
|
||||||
|
|
||||||
os := $(shell uname)
|
os := $(shell uname)
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
#include "hook_manager.hh"
|
#include "hook_manager.hh"
|
||||||
#include "option_manager.hh"
|
#include "option_manager.hh"
|
||||||
#include "keymap_manager.hh"
|
#include "keymap_manager.hh"
|
||||||
|
#include "safe_ptr.hh"
|
||||||
#include "string.hh"
|
#include "string.hh"
|
||||||
#include "value.hh"
|
#include "value.hh"
|
||||||
|
|
||||||
|
|
|
@ -3,6 +3,7 @@
|
||||||
|
|
||||||
#include "completion.hh"
|
#include "completion.hh"
|
||||||
#include "utils.hh"
|
#include "utils.hh"
|
||||||
|
#include "safe_ptr.hh"
|
||||||
|
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
|
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
#ifndef client_hh_INCLUDED
|
#ifndef client_hh_INCLUDED
|
||||||
#define client_hh_INCLUDED
|
#define client_hh_INCLUDED
|
||||||
|
|
||||||
|
#include "display_buffer.hh"
|
||||||
|
#include "env_vars.hh"
|
||||||
|
#include "input_handler.hh"
|
||||||
|
#include "safe_ptr.hh"
|
||||||
#include "string.hh"
|
#include "string.hh"
|
||||||
#include "utils.hh"
|
#include "utils.hh"
|
||||||
#include "display_buffer.hh"
|
|
||||||
#include "input_handler.hh"
|
|
||||||
#include "env_vars.hh"
|
|
||||||
|
|
||||||
namespace Kakoune
|
namespace Kakoune
|
||||||
{
|
{
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include "keys.hh"
|
#include "keys.hh"
|
||||||
#include "string.hh"
|
#include "string.hh"
|
||||||
#include "utils.hh"
|
#include "utils.hh"
|
||||||
|
#include "safe_ptr.hh"
|
||||||
|
|
||||||
namespace Kakoune
|
namespace Kakoune
|
||||||
{
|
{
|
||||||
|
|
60
src/safe_ptr.cc
Normal file
60
src/safe_ptr.cc
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
#include "safe_ptr.hh"
|
||||||
|
|
||||||
|
#ifdef SAFE_PTR_TRACK_CALLSTACKS
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#if defined(__linux__)
|
||||||
|
# include <execinfo.h>
|
||||||
|
#elif defined(__CYGWIN__)
|
||||||
|
# include <windows.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
namespace Kakoune
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
SafeCountable::Backtrace::Backtrace()
|
||||||
|
{
|
||||||
|
#if defined(__linux__)
|
||||||
|
num_frames = backtrace(stackframes, max_frames);
|
||||||
|
#elif defined(__CYGWIN__)
|
||||||
|
num_frames = CaptureStackBackTrace(0, max_frames, stackframes, nullptr);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* SafeCountable::Backtrace::desc() const
|
||||||
|
{
|
||||||
|
#if defined(__linux__)
|
||||||
|
char** symbols = backtrace_symbols(stackframes, num_frames);
|
||||||
|
int size = 0;
|
||||||
|
for (int i = 0; i < num_frames; ++i)
|
||||||
|
size += strlen(symbols[i]) + 1;
|
||||||
|
|
||||||
|
char* res = (char*)malloc(size+1);
|
||||||
|
res[0] = 0;
|
||||||
|
for (int i = 0; i < num_frames; ++i)
|
||||||
|
{
|
||||||
|
strcat(res, symbols[i]);
|
||||||
|
strcat(res, "\n");
|
||||||
|
}
|
||||||
|
free(symbols);
|
||||||
|
return res;
|
||||||
|
#elif defined(__CYGWIN__)
|
||||||
|
char* res = (char*)malloc(num_frames * 20);
|
||||||
|
res[0] = 0;
|
||||||
|
for (int i = 0; i < num_frames; ++i)
|
||||||
|
{
|
||||||
|
char addr[20];
|
||||||
|
snprintf(addr, 20, "0x%p", stackframes[i]);
|
||||||
|
strcat(res, addr);
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
#else
|
||||||
|
return "<not implemented>";
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
#endif
|
167
src/safe_ptr.hh
Normal file
167
src/safe_ptr.hh
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
#ifndef safe_ptr_hh_INCLUDED
|
||||||
|
#define safe_ptr_hh_INCLUDED
|
||||||
|
|
||||||
|
// #define SAFE_PTR_TRACK_CALLSTACKS
|
||||||
|
|
||||||
|
#include "assert.hh"
|
||||||
|
|
||||||
|
#ifdef SAFE_PTR_TRACK_CALLSTACKS
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
#include <algorithm>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
namespace Kakoune
|
||||||
|
{
|
||||||
|
|
||||||
|
// *** safe_ptr: objects that assert nobody references them when they die ***
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
class safe_ptr
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
safe_ptr() : m_ptr(nullptr) {}
|
||||||
|
explicit safe_ptr(T* ptr) : m_ptr(ptr)
|
||||||
|
{
|
||||||
|
#ifdef KAK_DEBUG
|
||||||
|
if (m_ptr)
|
||||||
|
m_ptr->inc_safe_count(this);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
safe_ptr(const safe_ptr& other) : safe_ptr(other.m_ptr) {}
|
||||||
|
safe_ptr(safe_ptr&& other) : m_ptr(other.m_ptr)
|
||||||
|
{
|
||||||
|
other.m_ptr = nullptr;
|
||||||
|
#ifdef KAK_DEBUG
|
||||||
|
if (m_ptr)
|
||||||
|
m_ptr->safe_ptr_moved(&other, this);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
~safe_ptr()
|
||||||
|
{
|
||||||
|
#ifdef KAK_DEBUG
|
||||||
|
if (m_ptr)
|
||||||
|
m_ptr->dec_safe_count(this);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
safe_ptr& operator=(const safe_ptr& other)
|
||||||
|
{
|
||||||
|
#ifdef KAK_DEBUG
|
||||||
|
if (m_ptr != other.m_ptr)
|
||||||
|
{
|
||||||
|
if (m_ptr)
|
||||||
|
m_ptr->dec_safe_count(this);
|
||||||
|
if (other.m_ptr)
|
||||||
|
other.m_ptr->inc_safe_count(this);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
m_ptr = other.m_ptr;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
safe_ptr& operator=(safe_ptr&& other)
|
||||||
|
{
|
||||||
|
#ifdef KAK_DEBUG
|
||||||
|
if (m_ptr)
|
||||||
|
m_ptr->dec_safe_count(this);
|
||||||
|
if (other.m_ptr)
|
||||||
|
other.m_ptr->safe_ptr_moved(&other, this);
|
||||||
|
#endif
|
||||||
|
m_ptr = other.m_ptr;
|
||||||
|
other.m_ptr = nullptr;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
void reset(T* ptr = nullptr)
|
||||||
|
{
|
||||||
|
*this = safe_ptr(ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool operator== (const safe_ptr& other) const { return m_ptr == other.m_ptr; }
|
||||||
|
bool operator!= (const safe_ptr& other) const { return m_ptr != other.m_ptr; }
|
||||||
|
bool operator== (T* ptr) const { return m_ptr == ptr; }
|
||||||
|
bool operator!= (T* ptr) const { return m_ptr != ptr; }
|
||||||
|
|
||||||
|
T& operator* () const { return *m_ptr; }
|
||||||
|
T* operator-> () const { return m_ptr; }
|
||||||
|
|
||||||
|
T* get() const { return m_ptr; }
|
||||||
|
|
||||||
|
explicit operator bool() const { return m_ptr; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
T* m_ptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
class SafeCountable
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
#ifdef KAK_DEBUG
|
||||||
|
SafeCountable() : m_count(0) {}
|
||||||
|
~SafeCountable()
|
||||||
|
{
|
||||||
|
kak_assert(m_count == 0);
|
||||||
|
#ifdef SAFE_PTR_TRACK_CALLSTACKS
|
||||||
|
kak_assert(m_callstacks.empty());
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void inc_safe_count(void* ptr) const
|
||||||
|
{
|
||||||
|
++m_count;
|
||||||
|
#ifdef SAFE_PTR_TRACK_CALLSTACKS
|
||||||
|
m_callstacks.emplace_back(ptr);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
void dec_safe_count(void* ptr) const
|
||||||
|
{
|
||||||
|
--m_count;
|
||||||
|
kak_assert(m_count >= 0);
|
||||||
|
#ifdef SAFE_PTR_TRACK_CALLSTACKS
|
||||||
|
auto it = std::find_if(m_callstacks.begin(), m_callstacks.end(),
|
||||||
|
[=](const Callstack& cs) { return cs.ptr == ptr; });
|
||||||
|
kak_assert(it != m_callstacks.end());
|
||||||
|
m_callstacks.erase(it);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void safe_ptr_moved(void* from, void* to) const
|
||||||
|
{
|
||||||
|
#ifdef SAFE_PTR_TRACK_CALLSTACKS
|
||||||
|
auto it = std::find_if(m_callstacks.begin(), m_callstacks.end(),
|
||||||
|
[=](const Callstack& cs) { return cs.ptr == from; });
|
||||||
|
kak_assert(it != m_callstacks.end());
|
||||||
|
it->ptr = to;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
#ifdef SAFE_PTR_TRACK_CALLSTACKS
|
||||||
|
struct Backtrace
|
||||||
|
{
|
||||||
|
static constexpr int max_frames = 16;
|
||||||
|
void* stackframes[max_frames];
|
||||||
|
int num_frames = 0;
|
||||||
|
|
||||||
|
Backtrace();
|
||||||
|
const char* desc() const;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Callstack
|
||||||
|
{
|
||||||
|
Callstack(void* p) : ptr(p) {}
|
||||||
|
void* ptr;
|
||||||
|
Backtrace bt;
|
||||||
|
};
|
||||||
|
|
||||||
|
mutable std::vector<Callstack> m_callstacks;
|
||||||
|
#endif
|
||||||
|
mutable int m_count;
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // safe_ptr_hh_INCLUDED
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
#include "color.hh"
|
#include "color.hh"
|
||||||
#include "keys.hh"
|
#include "keys.hh"
|
||||||
#include "memoryview.hh"
|
#include "memoryview.hh"
|
||||||
#include "utils.hh"
|
#include "safe_ptr.hh"
|
||||||
|
|
||||||
namespace Kakoune
|
namespace Kakoune
|
||||||
{
|
{
|
||||||
|
|
86
src/utils.hh
86
src/utils.hh
|
@ -53,92 +53,6 @@ private:
|
||||||
template<typename T>
|
template<typename T>
|
||||||
T* Singleton<T>::ms_instance = nullptr;
|
T* Singleton<T>::ms_instance = nullptr;
|
||||||
|
|
||||||
// *** safe_ptr: objects that assert nobody references them when they die ***
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
class safe_ptr
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
safe_ptr() : m_ptr(nullptr) {}
|
|
||||||
explicit safe_ptr(T* ptr) : m_ptr(ptr)
|
|
||||||
{
|
|
||||||
#ifdef KAK_DEBUG
|
|
||||||
if (m_ptr)
|
|
||||||
m_ptr->inc_safe_count();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
safe_ptr(const safe_ptr& other) : safe_ptr(other.m_ptr) {}
|
|
||||||
safe_ptr(safe_ptr&& other) : m_ptr(other.m_ptr) { other.m_ptr = nullptr; }
|
|
||||||
~safe_ptr()
|
|
||||||
{
|
|
||||||
#ifdef KAK_DEBUG
|
|
||||||
if (m_ptr)
|
|
||||||
m_ptr->dec_safe_count();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
safe_ptr& operator=(const safe_ptr& other)
|
|
||||||
{
|
|
||||||
#ifdef KAK_DEBUG
|
|
||||||
if (m_ptr != other.m_ptr)
|
|
||||||
{
|
|
||||||
if (m_ptr)
|
|
||||||
m_ptr->dec_safe_count();
|
|
||||||
if (other.m_ptr)
|
|
||||||
other.m_ptr->inc_safe_count();
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
m_ptr = other.m_ptr;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
safe_ptr& operator=(safe_ptr&& other)
|
|
||||||
{
|
|
||||||
#ifdef KAK_DEBUG
|
|
||||||
if (m_ptr)
|
|
||||||
m_ptr->dec_safe_count();
|
|
||||||
#endif
|
|
||||||
m_ptr = other.m_ptr;
|
|
||||||
other.m_ptr = nullptr;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
void reset(T* ptr = nullptr)
|
|
||||||
{
|
|
||||||
*this = safe_ptr(ptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator== (const safe_ptr& other) const { return m_ptr == other.m_ptr; }
|
|
||||||
bool operator!= (const safe_ptr& other) const { return m_ptr != other.m_ptr; }
|
|
||||||
bool operator== (T* ptr) const { return m_ptr == ptr; }
|
|
||||||
bool operator!= (T* ptr) const { return m_ptr != ptr; }
|
|
||||||
|
|
||||||
T& operator* () const { return *m_ptr; }
|
|
||||||
T* operator-> () const { return m_ptr; }
|
|
||||||
|
|
||||||
T* get() const { return m_ptr; }
|
|
||||||
|
|
||||||
explicit operator bool() const { return m_ptr; }
|
|
||||||
|
|
||||||
private:
|
|
||||||
T* m_ptr;
|
|
||||||
};
|
|
||||||
|
|
||||||
class SafeCountable
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
#ifdef KAK_DEBUG
|
|
||||||
SafeCountable() : m_count(0) {}
|
|
||||||
~SafeCountable() { kak_assert(m_count == 0); }
|
|
||||||
|
|
||||||
void inc_safe_count() const { ++m_count; }
|
|
||||||
void dec_safe_count() const { --m_count; kak_assert(m_count >= 0); }
|
|
||||||
|
|
||||||
private:
|
|
||||||
mutable int m_count;
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
|
|
||||||
// *** Containers helpers ***
|
// *** Containers helpers ***
|
||||||
|
|
||||||
template<typename Container>
|
template<typename Container>
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include "hook_manager.hh"
|
#include "hook_manager.hh"
|
||||||
#include "option_manager.hh"
|
#include "option_manager.hh"
|
||||||
#include "keymap_manager.hh"
|
#include "keymap_manager.hh"
|
||||||
|
#include "safe_ptr.hh"
|
||||||
|
|
||||||
namespace Kakoune
|
namespace Kakoune
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue
Block a user