factor/vm/local_roots.hpp

53 lines
1.2 KiB
C++
Raw Normal View History

2009-05-04 02:46:13 -04:00
namespace factor
{
2009-05-02 05:04:19 -04:00
/* If a runtime function needs to call another function which potentially
allocates memory, it must wrap any local variable references to Factor
objects in gc_root instances */
extern std::vector<cell> gc_locals;
2009-05-02 05:04:19 -04:00
template <typename T>
2009-05-02 10:19:09 -04:00
struct gc_root : public tagged<T>
2009-05-02 05:04:19 -04:00
{
void push() { check_tagged_pointer(tagged<T>::value()); gc_locals.push_back((cell)this); }
2009-05-02 10:19:09 -04:00
2009-05-04 05:50:24 -04:00
explicit gc_root(cell value_) : tagged<T>(value_) { push(); }
2009-05-02 05:04:19 -04:00
explicit gc_root(T *value_) : tagged<T>(value_) { push(); }
2009-05-02 10:19:09 -04:00
const gc_root<T>& operator=(const T *x) { tagged<T>::operator=(x); return *this; }
2009-05-04 05:50:24 -04:00
const gc_root<T>& operator=(const cell &x) { tagged<T>::operator=(x); return *this; }
2009-05-02 10:19:09 -04:00
2009-05-04 06:07:14 -04:00
~gc_root() {
#ifdef FACTOR_DEBUG
assert(gc_locals.back() == (cell)this);
2009-05-04 06:07:14 -04:00
#else
gc_locals.pop_back();
2009-05-04 06:07:14 -04:00
#endif
}
2009-05-02 05:04:19 -04:00
};
/* A similar hack for the bignum implementation */
extern std::vector<cell> gc_bignums;
2009-05-02 05:04:19 -04:00
struct gc_bignum
{
2009-05-04 05:50:24 -04:00
bignum **addr;
2009-05-04 05:50:24 -04:00
gc_bignum(bignum **addr_) : addr(addr_) {
if(*addr_)
check_data_pointer(*addr_);
gc_bignums.push_back((cell)addr);
}
~gc_bignum() {
#ifdef FACTOR_DEBUG
assert(gc_bignums.back() == (cell)addr);
#endif
gc_bignums.pop_back();
}
};
#define GC_BIGNUM(x) gc_bignum x##__gc_root(&x)
2009-05-04 02:46:13 -04:00
}