File: Auto.h

package info (click to toggle)
storm-lang 0.7.4-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 52,004 kB
  • sloc: ansic: 261,462; cpp: 140,405; sh: 14,891; perl: 9,846; python: 2,525; lisp: 2,504; asm: 860; makefile: 678; pascal: 70; java: 52; xml: 37; awk: 12
file content (109 lines) | stat: -rw-r--r-- 1,595 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
#pragma once
#include "Utils/Templates.h"
#define IF_CONVERTIBLE(From, To) typename EnableIf<IsConvertible<From, To>::value, bool>::t = false

/**
 * Auto-ptr.
 */
template <class T>
class Auto {
public:
	Auto() : ptr(null) {}

	Auto(T *o) : ptr(o) {}

	Auto(const Auto<T> &o) : ptr(o.ptr) {
		if (ptr)
			ptr->addRef();
	}

	template <class U>
	Auto(const Auto<U> &o, IF_CONVERTIBLE(U, T)) : ptr(&*o) {
		if (ptr)
			ptr->addRef();
	}

	Auto<T> &operator =(const Auto<T> &o) {
		if (o.ptr)
			o.ptr->addRef();
		if (ptr)
			ptr->release();
		ptr = o.ptr;
		return *this;
	}

	~Auto() {
		if (ptr)
			ptr->release();
	}

	template <class U>
	Auto<U> as() const {
		U *o = dynamic_cast<U *>(ptr);
		if (o)
			o->addRef();
		return Auto<U>(o);
	}

	T *borrow() const {
		return ptr;
	}

	operator void *() const {
		return ptr;
	}

	bool operator ==(const Auto<T> &o) const {
		return ptr == o.ptr;
	}

	T *operator ->() const {
		return ptr;
	}

	T &operator *() const {
		return *ptr;
	}

	// Make sure < works as if calling < on the value (good in maps).
	bool operator <(const Auto<T> &o) const {
		return *ptr < *o.ptr;
	}

private:
	T *ptr;
};

template <class T>
wostream &operator <<(wostream &to, const Auto<T> &p) {
	return to << *p;
}

template <class T>
Auto<T> capture(T *ptr) {
	if (ptr)
		ptr->addRef();
	return Auto<T>(ptr);
}


/**
 * Refcounted object.
 */
class Refcount : NoCopy {
public:
	Refcount() : refs(1) {}
	Refcount(const Refcount &) : NoCopy(), refs(1) {}

	void addRef() {
		refs++;
	}

	void release() {
		if (--refs == 0)
			delete this;
	}

private:
	nat refs;
};