PSCF v1.2
MemoryIArchive.h
1#ifndef UTIL_MEMORY_I_ARCHIVE_H
2#define UTIL_MEMORY_I_ARCHIVE_H
3
4/*
5* Util Package - C++ Utilities for Scientific Computation
6*
7* Copyright 2010 - 2017, The Regents of the University of Minnesota
8* Distributed under the terms of the GNU General Public License.
9*/
10
11#include "Byte.h"
12#include "serialize.h"
13
14#include <util/space/Vector.h>
15#include <util/space/IntVector.h>
16
17#include <complex>
18#include <string>
19#include <vector>
20
21namespace Util
22{
23
24 class MemoryOArchive;
25
32 {
33
34 public:
35
37 static bool is_saving();
38
40 static bool is_loading();
41
46
51
57 void allocate(size_t capacity);
58
63
67 void reset();
68
74 void clear();
75
79 void release();
80
86 template <typename T>
88
94 template <typename T>
96
106 template <typename T, size_t N>
107 MemoryIArchive& operator >> (T (& data)[N]);
108
116 template <typename T, size_t N>
117 MemoryIArchive& operator & (T (& data)[N]);
118
119 // Pack functions and function templates
120
126 template <typename T>
127 void unpack(T& data);
128
135 template <typename T>
136 void unpack(T* array, int n);
137
149 template <typename T>
150 void unpack(T* array, int m, int n, int np);
151
152 #ifdef UTIL_MPI
159 void recv(MPI::Intracomm& comm, int source);
160 #endif
161
165 Byte* begin() const;
166
170 Byte* cursor() const;
171
175 Byte* end() const;
176
180 size_t capacity() const;
181
185 bool isAllocated() const;
186
187 private:
188
190 Byte* buffer_;
191
193 Byte* begin_;
194
196 Byte* cursor_;
197
199 Byte* end_;
200
202 Byte* endAllocated_;
203
205 MemoryOArchive* oArchivePtr_;
206
208 size_t capacity_;
209
211 unsigned int version_;
212
214 bool ownsData_;
215
216 };
217
218 // Inline static methods
219
221 { return false; }
222
224 { return true; }
225
226 // Inline non-static methods
227
228 /*
229 * Return pointer to beginning of block.
230 */
232 { return begin_; }
233
234 /*
235 * Return pointer to cursor position.
236 */
238 { return cursor_; }
239
240 /*
241 * Return end of packed block (one Byte past the last).
242 */
243 inline Byte* MemoryIArchive::end() const
244 { return end_; }
245
246 /*
247 * Return capacity in Bytes.
248 */
249 inline size_t MemoryIArchive::capacity() const
250 { return capacity_; }
251
252 /*
253 * Has a memory block been allocated?
254 */
255 inline bool MemoryIArchive::isAllocated() const
256 { return (bool) begin_; }
257
258 // Template methods
259
260 /*
261 * Load one T object from this MemoryIArchive.
262 */
263 template <typename T>
265 {
266 serialize(*this, data, version_);
267 return *this;
268 }
269
270 /*
271 * Load one T object from a MemoryIArchive.
272 */
273 template <typename T>
275 {
276 serialize(*this, data, version_);
277 return *this;
278 }
279
280 /*
281 * Load a fixed size array of objects via operator >>.
282 */
283 template <typename T, size_t N>
285 {
286 for (size_t i=0; i < N; ++i) {
287 serialize(*this, data[i], version_);
288 }
289 return *this;
290 }
291
292 /*
293 * Load a fixed size array of objects via operator &.
294 */
295 template <typename T, size_t N>
297 {
298 for (size_t i=0; i < N; ++i) {
299 serialize(*this, data[i], version_);
300 }
301 return *this;
302 }
303
304 /*
305 * Load one T object from this MemoryIArchive.
306 */
307 template <typename T>
309 {
310 if (cursor_ + sizeof(data) > end_) {
311 UTIL_THROW("Attempted read past end of packed block");
312 }
313 T* ptr = (T *)cursor_;
314 data = *ptr;
315 ++ptr;
316 cursor_ = (Byte *)ptr;
317 }
318
319 /*
320 * Load a C-array of objects of type T.
321 */
322 template <typename T>
323 void MemoryIArchive::unpack(T* array, int n)
324 {
325 if (cursor_ + n*sizeof(T) > end_) {
326 UTIL_THROW("Attempted read past end of data");
327 }
328 T* ptr = (T *)cursor_;
329 for (int i=0; i < n; ++i) {
330 array[i] = *ptr;
331 ++ptr;
332 }
333 cursor_ = (Byte *)ptr;
334 }
335
336 /*
337 * Bitwise pack a 2D C-array of objects of type T.
338 */
339 template <typename T>
340 void MemoryIArchive::unpack(T* array, int m, int n, int np)
341 {
342 if (cursor_ + m*n*sizeof(T) > end_) {
343 UTIL_THROW("Attempted read past end of data");
344 }
345 int i, j;
346 T* ptr = (T *)cursor_;
347 for (i = 0; i < m; ++i) {
348 for (j = 0; j < n; ++j) {
349 array[i*np + j] = *ptr;
350 ++ptr;
351 }
352 }
353 cursor_ = (Byte *)ptr;
354 }
355
356 // Explicit specializations of serialize function
357
358 /*
359 * Load a char from a MemoryIArchive.
360 */
361 template <>
362 inline void serialize(MemoryIArchive& ar, char& data,
363 const unsigned int version)
364 { ar.unpack(data); }
365
366 /*
367 * Load a bool from a MemoryIArchive.
368 */
369 template <>
370 inline void serialize(MemoryIArchive& ar, bool& data,
371 const unsigned int version)
372 { ar.unpack(data); }
373
374 /*
375 * Load an unsigned int from a MemoryIArchive.
376 */
377 template <>
378 inline void serialize(MemoryIArchive& ar, unsigned int& data,
379 const unsigned int version)
380 { ar.unpack(data); }
381
382 /*
383 * Load an int from a MemoryIArchive.
384 */
385 template <>
386 inline void serialize(MemoryIArchive& ar, int& data,
387 const unsigned int version)
388 { ar.unpack(data); }
389
390 /*
391 * Load an unsigned long int from a MemoryIArchive.
392 */
393 template <>
394 inline void serialize(MemoryIArchive& ar, unsigned long& data,
395 const unsigned int version)
396 { ar.unpack(data); }
397
398 /*
399 * Load a long int from a MemoryIArchive.
400 */
401 template <>
402 inline void serialize(MemoryIArchive& ar, long& data,
403 const unsigned int version)
404 { ar.unpack(data); }
405
406 /*
407 * Load a float from a MemoryIArchive.
408 */
409 template <>
410 inline void serialize(MemoryIArchive& ar, float& data,
411 const unsigned int version)
412 { ar.unpack(data); }
413
414 /*
415 * Load a double from a MemoryIArchive.
416 */
417 template <>
418 inline void serialize(MemoryIArchive& ar, double& data,
419 const unsigned int version)
420 { ar.unpack(data); }
421
422 /*
423 * Load a std::vector from a MemoryIArchive.
424 */
425 template <typename T>
426 void serialize(MemoryIArchive& ar, std::vector<T>& data,
427 const unsigned int version)
428 {
429 T element;
430 std::size_t size;
431 ar.unpack(size);
432 data.reserve(size);
433 data.clear();
434 for (size_t i = 0; i < size; ++i) {
435 ar & element;
436 data.push_back(element);
437 }
438 }
439
440 // Explicit serialize methods for std library types.
441
442 /*
443 * Load a std::complex<float> from a MemoryIArchive.
444 */
445 template <>
446 inline void serialize(MemoryIArchive& ar, std::complex<float>& data,
447 const unsigned int version)
448 { ar.unpack(data); }
449
450 /*
451 * Load a std::complex<double> from a MemoryIArchive.
452 */
453 template <>
454 inline void serialize(MemoryIArchive& ar, std::complex<double>& data,
455 const unsigned int version)
456 { ar.unpack(data); }
457
458 /*
459 * Load a std::string from MemoryIArchive.
460 */
461 template <>
462 void serialize(MemoryIArchive& ar, std::string& data,
463 const unsigned int version);
464
465 // Explicit serialize methods for Util namespace types.
466
467 /*
468 * Load a Util::Vector from a MemoryIArchive.
469 */
470 template <>
471 inline void serialize(MemoryIArchive& ar, Vector& data,
472 const unsigned int version)
473 { ar.unpack(data); }
474
475 /*
476 * Load a Util::IntVector from a MemoryIArchive.
477 */
478 template <>
479 inline void serialize(MemoryIArchive& ar, IntVector& data,
480 const unsigned int version)
481 { ar.unpack(data); }
482
483}
484#endif
Input archive for packed heterogeneous binary data.
void recv(MPI::Intracomm &comm, int source)
Receive packed data via MPI.
Byte * begin() const
Return pointer to beginning of block.
static bool is_saving()
Returns true;.
MemoryIArchive & operator&(T &data)
Load (read) one object of type T via the & operator.
static bool is_loading()
Returns false;.
MemoryIArchive & operator=(MemoryOArchive &other)
Assignment from MemoryOArchive.
void allocate(size_t capacity)
Allocate memory block.
void reset()
Reset the cursor to the beginning (for rereading).
size_t capacity() const
Return capacity in Bytes.
void clear()
Reset to empty state.
~MemoryIArchive()
Destructor.
void unpack(T &data)
Unpack one object of type T.
Byte * cursor() const
Return pointer to current position (cursor).
MemoryIArchive & operator>>(T &data)
Load (read) one object of type T via the >> operator.
void release()
Release memory obtained by assignment.
MemoryIArchive()
Constructor.
Byte * end() const
Return pointer to end of packed block (one Byte past the last).
bool isAllocated() const
Has memory been allocated?
Save archive for packed heterogeneous binary data.
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
Definition global.h:51
Utility classes for scientific computation.
unsigned char Byte
Define a "Byte" type.
Definition Byte.h:19