summaryrefslogtreecommitdiffstats
path: root/src/Common/SharedPtr.h
blob: 3696344b143b5d48954c955bcc4fcd5cbc418c7d (plain)
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
/*
 * SharedPtr.h
 *
 * Copyright (C) 2008 Johannes Thorn <dante@g4t3.de>
 *
 * This program is free software: you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the
 * Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 * See the GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program. If not, see <http://www.gnu.org/licenses/>.
 */

#ifndef MAD_COMMON_SHAREDPTR_H_
#define MAD_COMMON_SHAREDPTR_H_

namespace Mad {
namespace Common {

template <typename T>
class SharedPtr {
  protected:
    class Counter {
      public:
        Counter(T *o) : object(o), refCount(1), cCount(new unsigned int(1)) {}
        Counter(T *o, unsigned int *count) : object(o), refCount(1), cCount(count) {++*cCount;}

        virtual ~Counter() {
          if(--cCount)
            return;

          if(object)
            delete object;

          delete cCount;
        }

        T *object;
        unsigned int refCount;
        unsigned int *cCount;
    };

    Counter *counter;

    template <typename T2>
    class CastPtr : public SharedPtr<T2> {
      public:
        CastPtr(T *o, unsigned int *cCount) : SharedPtr<T2>(o, cCount) {}
    };

    SharedPtr(T *o, unsigned int *cCount) : counter(new Counter(o, cCount)) {}

  public:
    SharedPtr(T *o = 0) : counter(new Counter(o)) {}

    SharedPtr(const SharedPtr<T> &c) : counter(c.counter) {
      ++counter->refCount;
    }

    virtual ~SharedPtr() {
      if(--counter->refCount == 0)
        delete counter;
    }

    SharedPtr<T>& operator=(const SharedPtr<T>& c) {
      ++c.counter->refCount;
      if(--counter->refCount == 0)
        delete counter;

      counter = c.counter;
      return *this;
    }

    bool empty() const {
      return counter->object;
    }

    T* operator->() {return counter->object;}
    T& operator*() {return *counter->object;}

    const T* operator->() const {return counter->object;}
    const T& operator*() const {return *counter->object;}

    bool operator<(const SharedPtr<T> &o) const {return counter->object < o.counter->object;}
    bool operator==(const SharedPtr<T> &o) const {return counter->object == o.counter->object;}

    template <typename T2> SharedPtr<T2> cast() const {
      return CastPtr<T2>(counter->object, counter->cCount);
    }
};

}
}

#endif /*MAD_COMMON_SHAREDPTR_H_*/