Cycles: svn merge -r41225:41232 ^/trunk/blender
[blender.git] / intern / memutil / intern / MEM_CacheLimiterC-Api.cpp
1 /** \file memutil/intern/MEM_CacheLimiterC-Api.cpp
2  *  \ingroup memutil
3  */
4 /**
5  * This program is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU General Public License
7  * as published by the Free Software Foundation; either version 2
8  * of the License, or (at your option) any later version. The Blender
9  * Foundation also sells licenses for use in proprietary software under
10  * the Blender License.  See http://www.blender.org/BL/ for information
11  * about this.
12  *
13  * This program is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16  * GNU General Public License for more details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with this program; if not, write to the Free Software Foundation,
20  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
21  *
22  * Contributor(s): Peter Schlaile <peter@schlaile.de> 2005
23  *
24  */
25
26 #include <cstddef>
27
28 #include "MEM_CacheLimiter.h"
29 #include "MEM_CacheLimiterC-Api.h"
30
31 static intptr_t & get_max() 
32 {
33         static intptr_t m = 32*1024*1024;
34         return m;
35 }
36
37 void MEM_CacheLimiter_set_maximum(intptr_t m)
38 {
39         get_max() = m;
40 }
41
42 intptr_t MEM_CacheLimiter_get_maximum()
43 {
44         return get_max();
45 }
46
47 class MEM_CacheLimiterHandleCClass;
48 class MEM_CacheLimiterCClass;
49
50 typedef MEM_CacheLimiterHandle<MEM_CacheLimiterHandleCClass> handle_t;
51 typedef MEM_CacheLimiter<MEM_CacheLimiterHandleCClass> cache_t;
52 typedef std::list<MEM_CacheLimiterHandleCClass*,
53                   MEM_Allocator<MEM_CacheLimiterHandleCClass* > > list_t;
54
55 class MEM_CacheLimiterCClass {
56 public:
57         MEM_CacheLimiterCClass(MEM_CacheLimiter_Destruct_Func data_destructor_)
58                 : data_destructor(data_destructor_) { 
59         }
60         ~MEM_CacheLimiterCClass();
61         
62         handle_t * insert(void * data);
63
64         void destruct(void * data,
65                       list_t::iterator it);
66
67         cache_t * get_cache() {
68                 return &cache;
69         }
70 private:
71         MEM_CacheLimiter_Destruct_Func data_destructor;
72
73         MEM_CacheLimiter<MEM_CacheLimiterHandleCClass> cache;
74         
75         list_t cclass_list;
76 };
77
78 class MEM_CacheLimiterHandleCClass {
79 public:
80         MEM_CacheLimiterHandleCClass(void * data_,
81                                    MEM_CacheLimiterCClass * parent_)
82                 : data(data_), parent(parent_) { }
83         ~MEM_CacheLimiterHandleCClass();
84         void set_iter(list_t::iterator it_) {
85                 it = it_;
86         }
87         void set_data(void * data_) {
88                 data = data_;
89         }
90         void * get_data() const {
91                 return data;
92         }
93 private:
94         void * data;
95         MEM_CacheLimiterCClass * parent;
96         list_t::iterator it;
97 };
98
99 handle_t * MEM_CacheLimiterCClass::insert(void * data) 
100 {
101         cclass_list.push_back(new MEM_CacheLimiterHandleCClass(data, this));
102         list_t::iterator it = cclass_list.end();
103         --it;
104         cclass_list.back()->set_iter(it);
105         
106         return cache.insert(cclass_list.back());
107 }
108
109 void MEM_CacheLimiterCClass::destruct(void * data, list_t::iterator it) 
110 {
111         data_destructor(data);
112         cclass_list.erase(it);
113 }
114
115 MEM_CacheLimiterHandleCClass::~MEM_CacheLimiterHandleCClass()
116 {
117         if (data) {
118                 parent->destruct(data, it);
119         }
120 }
121
122 MEM_CacheLimiterCClass::~MEM_CacheLimiterCClass()
123 {
124         // should not happen, but don't leak memory in this case...
125         for (list_t::iterator it = cclass_list.begin();
126              it != cclass_list.end(); it++) {
127                 (*it)->set_data(0);
128                 delete *it;
129         }
130 }
131
132 // ----------------------------------------------------------------------
133
134 static inline MEM_CacheLimiterCClass* cast(MEM_CacheLimiterC * l)
135 {
136         return (MEM_CacheLimiterCClass*) l;
137 }
138
139 static inline handle_t* cast(MEM_CacheLimiterHandleC * l)
140 {
141         return (handle_t*) l;
142 }
143
144 MEM_CacheLimiterC * new_MEM_CacheLimiter(
145         MEM_CacheLimiter_Destruct_Func data_destructor)
146 {
147         return (MEM_CacheLimiterC*) new MEM_CacheLimiterCClass(
148                 data_destructor);
149 }
150
151 void delete_MEM_CacheLimiter(MEM_CacheLimiterC * This)
152 {
153         delete cast(This);
154 }
155
156 MEM_CacheLimiterHandleC * MEM_CacheLimiter_insert(
157         MEM_CacheLimiterC * This, void * data)
158 {
159         return (MEM_CacheLimiterHandleC *) cast(This)->insert(data);
160 }
161
162 void MEM_CacheLimiter_enforce_limits(MEM_CacheLimiterC * This)
163 {
164         cast(This)->get_cache()->enforce_limits();
165 }
166         
167 void MEM_CacheLimiter_unmanage(MEM_CacheLimiterHandleC * handle)
168 {
169         cast(handle)->unmanage();
170 }
171         
172 void MEM_CacheLimiter_touch(MEM_CacheLimiterHandleC * handle)
173 {
174         cast(handle)->touch();
175 }
176         
177 void MEM_CacheLimiter_ref(MEM_CacheLimiterHandleC * handle)
178 {
179         cast(handle)->ref();
180 }
181         
182 void MEM_CacheLimiter_unref(MEM_CacheLimiterHandleC * handle)
183 {
184         cast(handle)->unref();
185 }
186
187 int MEM_CacheLimiter_get_refcount(MEM_CacheLimiterHandleC * handle)
188 {
189         return cast(handle)->get_refcount();
190 }
191
192         
193 void * MEM_CacheLimiter_get(MEM_CacheLimiterHandleC * handle)
194 {
195         return cast(handle)->get()->get_data();
196 }