Don't leak memory from Ref<>()
parent
131cf2581e
commit
2e4de7b67d
|
@ -341,7 +341,7 @@ def generate_class_implementation(icalls, used_classes, c):
|
||||||
if is_enum(method["return_type"]):
|
if is_enum(method["return_type"]):
|
||||||
return_statement += "return (" + remove_enum_prefix(method["return_type"]) + ") "
|
return_statement += "return (" + remove_enum_prefix(method["return_type"]) + ") "
|
||||||
elif is_reference_type(method["return_type"]):
|
elif is_reference_type(method["return_type"]):
|
||||||
return_statement += "return Ref<" + strip_name(method["return_type"]) + ">::__internal_constructor(";
|
return_statement += "return Ref<" + strip_name(method["return_type"]) + ">(";
|
||||||
else:
|
else:
|
||||||
return_statement += "return " + ("(" + strip_name(method["return_type"]) + " *) " if is_class_type(method["return_type"]) else "")
|
return_statement += "return " + ("(" + strip_name(method["return_type"]) + " *) " if is_class_type(method["return_type"]) else "")
|
||||||
else:
|
else:
|
||||||
|
|
|
@ -10,24 +10,6 @@ template<class T>
|
||||||
class Ref {
|
class Ref {
|
||||||
T *reference;
|
T *reference;
|
||||||
|
|
||||||
void ref(const Ref &from)
|
|
||||||
{
|
|
||||||
if (from.reference == reference) return;
|
|
||||||
|
|
||||||
unref();
|
|
||||||
|
|
||||||
reference = from.reference;
|
|
||||||
if (reference) reference->reference();
|
|
||||||
}
|
|
||||||
|
|
||||||
void ref_pointer(T *r)
|
|
||||||
{
|
|
||||||
if (!r) return;
|
|
||||||
|
|
||||||
if (r->init_ref()) reference = r;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
inline bool operator==(const Ref<T> &r) const
|
inline bool operator==(const Ref<T> &r) const
|
||||||
{
|
{
|
||||||
|
@ -73,28 +55,37 @@ public:
|
||||||
|
|
||||||
void operator=(const Ref &from)
|
void operator=(const Ref &from)
|
||||||
{
|
{
|
||||||
ref(from);
|
if (reference)
|
||||||
|
unref();
|
||||||
|
|
||||||
|
if (from.reference) {
|
||||||
|
reference = from.reference;
|
||||||
|
reference->reference();
|
||||||
|
} else {
|
||||||
|
reference = nullptr;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template<class T_Other>
|
template<class T_Other>
|
||||||
void operator=(const Ref<T_Other> &from)
|
void operator=(const Ref<T_Other> &from)
|
||||||
{
|
{
|
||||||
Ref<T> n((T *) from.ptr());
|
if (reference)
|
||||||
ref(n);
|
unref();
|
||||||
|
|
||||||
|
if (from.reference) {
|
||||||
|
reference = (T *) from.reference;
|
||||||
|
reference->reference();
|
||||||
|
} else {
|
||||||
|
reference = nullptr;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void operator=(const Variant &variant)
|
void operator=(const Variant &variant)
|
||||||
{
|
{
|
||||||
T *r = (T *) (Object *) variant;
|
if (reference)
|
||||||
if (!r) {
|
|
||||||
unref();
|
unref();
|
||||||
return;
|
|
||||||
}
|
reference = (T *) (Object *) variant;
|
||||||
|
|
||||||
Ref re;
|
|
||||||
re.reference = r;
|
|
||||||
ref(re);
|
|
||||||
re.reference = nullptr;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
operator Variant() const
|
operator Variant() const
|
||||||
|
@ -105,22 +96,27 @@ public:
|
||||||
template<class T_Other>
|
template<class T_Other>
|
||||||
Ref(const Ref<T_Other> &from)
|
Ref(const Ref<T_Other> &from)
|
||||||
{
|
{
|
||||||
if (from.ptr())
|
if (from.reference) {
|
||||||
ref_pointer((T *) from.ptr());
|
reference = (T *) from.reference;
|
||||||
else
|
reference->reference();
|
||||||
|
} else {
|
||||||
reference = nullptr;
|
reference = nullptr;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ref(const Ref &from)
|
Ref(const Ref &from)
|
||||||
{
|
{
|
||||||
reference = nullptr;
|
if (from.reference) {
|
||||||
ref(from);
|
reference = from.reference;
|
||||||
|
reference->reference();
|
||||||
|
} else {
|
||||||
|
reference = nullptr;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Ref(T *r)
|
Ref(T *r)
|
||||||
{
|
{
|
||||||
r->reference();
|
|
||||||
reference = r;
|
reference = r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -129,28 +125,9 @@ public:
|
||||||
|
|
||||||
Ref(const Variant &variant)
|
Ref(const Variant &variant)
|
||||||
{
|
{
|
||||||
reference = nullptr;
|
reference = (T *) (Object *) variant;
|
||||||
T *r = (T *) (Object *) variant;
|
|
||||||
if (!r) {
|
|
||||||
unref();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ref re;
|
|
||||||
re.reference = r;
|
|
||||||
ref(re);
|
|
||||||
re.reference = nullptr;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template<class T_Other>
|
|
||||||
static Ref<T> __internal_constructor(T_Other *r)
|
|
||||||
{
|
|
||||||
Ref<T> ref;
|
|
||||||
ref.reference = (T *) r;
|
|
||||||
return ref;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
inline bool is_valid() const { return reference != nullptr; }
|
inline bool is_valid() const { return reference != nullptr; }
|
||||||
inline bool is_null() const { return reference == nullptr; }
|
inline bool is_null() const { return reference == nullptr; }
|
||||||
|
|
||||||
|
@ -164,7 +141,9 @@ public:
|
||||||
|
|
||||||
void instance()
|
void instance()
|
||||||
{
|
{
|
||||||
ref(new T);
|
unref();
|
||||||
|
|
||||||
|
reference = new T;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ref()
|
Ref()
|
||||||
|
|
Loading…
Reference in New Issue