summaryrefslogtreecommitdiff
path: root/common/utility/Lazy.H
blob: 84bf426fcce872fdf049ac9d98333242a1c2224a (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
102
103
104
105
106
107
108
109
110
// Copyright 2008, 2009 Brady J. Garvin

// This file is part of Covering Arrays by Simulated Annealing (CASA).

// CASA 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.

// CASA 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 CASA.  If not, see <http://www.gnu.org/licenses/>.


#ifndef LAZY_H
#define LAZY_H

#include <cassert>
#include <iostream>

template<class T>class Lazy {
protected:
  T*					implementation;
  unsigned*				referenceCount;

  void destroy() {
    if (referenceCount) {
      assert(implementation);
      if (!(--*referenceCount)) {
	delete implementation;
	delete referenceCount;
      }
      implementation = NULL;
    } else {
      assert(!implementation);
    }
  }

public:
  Lazy() :
    implementation(NULL),
    referenceCount(NULL) {}
  Lazy(T*implementation) :
    implementation(implementation),
    referenceCount(implementation ? new unsigned(1) : NULL) {}
  Lazy(const Lazy&copy) :
    implementation(copy.implementation),
    referenceCount(copy.referenceCount) {
    if (implementation) {
      assert(referenceCount);
      ++*referenceCount;
    } else {
      assert(!referenceCount);
    }
  }

  Lazy&operator =(T*implementation) {
    destroy();
    this->implementation = implementation;
    this->referenceCount = implementation ? new unsigned(1) : NULL;
    return *this;
  }
  Lazy&operator =(const Lazy&copy) {
    destroy();
    implementation = copy.implementation;
    if (implementation) {
      referenceCount = copy.referenceCount;
      assert(referenceCount);
      ++*referenceCount;
    } else {
      assert(!copy.referenceCount);
      referenceCount = NULL;
    }
    return *this;
  }

  virtual ~Lazy(){
    destroy();
  }

  const T*operator ->() const {
    return implementation;
  }
  T*operator ->() {
    if (referenceCount) {
      assert(implementation);
      if (*referenceCount > 1) {
	T*copy = new T(*implementation);
	destroy();
	implementation = copy;
	referenceCount = new unsigned(1);
      }
    } else {
      assert(!implementation);
    }
    return implementation;
  }
  operator const T*() const {
    return operator ->();
  }
  operator T*() {
    return operator ->();
  }
};

#endif