PSCF v1.2
MemoryOArchive.h
1#ifndef UTIL_MEMORY_O_ARCHIVE_H
2#define UTIL_MEMORY_O_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 MemoryIArchive;
25
32 {
33
34 public:
35
37 static bool is_saving();
38
40 static bool is_loading();
41
46
50 virtual ~MemoryOArchive();
51
57 virtual void allocate(size_t capacity);
58
62 void clear();
63
69 template <typename T>
71
79 template <typename T>
81
91 template <typename T, size_t N>
92 MemoryOArchive& operator << (T (& data)[N]);
93
101 template <typename T, size_t N>
102 MemoryOArchive& operator & (T (& data)[N]);
103
104 // Pack functions and function templates
105
109 template <typename T>
110 void pack(const T& data);
111
118 template <typename T>
119 void pack(const T* array, int n);
120
132 template <typename T>
133 void pack(const T* array, int m, int n, int np);
134
135 #ifdef UTIL_MPI
142 void send(MPI::Intracomm& comm, int dest);
143
151 void iSend(MPI::Intracomm& comm, MPI::Request& req, int dest);
152 #endif
153
157 Byte* begin() const;
158
162 Byte* cursor() const;
163
167 size_t capacity() const;
168
172 bool isAllocated() const;
173
174 private:
175
177 Byte* buffer_;
178
180 Byte* begin_;
181
183 Byte* cursor_;
184
186 Byte* endAllocated_;
187
189 size_t capacity_;
190
192 unsigned int version_;
193
195 bool isLocked_;
196
198 bool ownsData_;
199
203 MemoryOArchive (const MemoryOArchive& other);
204
208 MemoryOArchive& operator = (const MemoryOArchive& other);
209
210 // friends:
211
212 friend class MemoryIArchive;
213
214 };
215
216 // Inline methods
217
219 { return true; }
220
222 { return false; }
223
224 // Inline non-static methods
225
226 /*
227 * Has a memory block been allocated?
228 */
229 inline bool MemoryOArchive::isAllocated() const
230 { return (bool) begin_; }
231
232 /*
233 * Return capacity in Bytes.
234 */
235 inline size_t MemoryOArchive::capacity() const
236 { return capacity_; }
237
238 /*
239 * Return pointer to beginning of block.
240 */
242 { return begin_; }
243
244 /*
245 * Return pointer to cursor position.
246 */
248 { return cursor_; }
249
250 /*
251 * Save one object of type T to this archive via the << operator.
252 */
253 template <typename T>
255 {
256 serialize(*this, data, version_);
257 return *this;
258 }
259
260 /*
261 * Save one object of type T to this archive via the & operator.
262 */
263 template <typename T>
265 {
266 serialize(*this, data, version_);
267 return *this;
268 }
269
270 /*
271 * Save a fixed size array of objects via operator <<.
272 */
273 template <typename T, size_t N>
275 {
276 for (size_t i = 0; i < N; ++i) {
277 serialize(*this, data[i], version_);
278 }
279 return *this;
280 }
281
282 /*
283 * Save a fixed size array of objects via operator &.
284 */
285 template <typename T, size_t N>
287 {
288 for (size_t i = 0; i < N; ++i) {
289 serialize(*this, data[i], version_);
290 }
291 return *this;
292 }
293
294 // Method templates
295
296 /*
297 * Save a single object of type T.
298 */
299 template <typename T>
300 inline void MemoryOArchive::pack(const T& data)
301 {
302 if (isLocked_) {
303 UTIL_THROW("Locked archive");
304 }
305 if (cursor_ + sizeof(data) > endAllocated_) {
306 UTIL_THROW("Attempted write past end of allocated block");
307 }
308 T* ptr = (T *)cursor_;
309 *ptr = data;
310 ++ptr;
311 cursor_ = (Byte *)ptr;
312 }
313
314 /*
315 * Save a C-array of objects of type T.
316 */
317 template <typename T>
318 inline void MemoryOArchive::pack(const T* array, int n)
319 {
320 if (isLocked_) {
321 UTIL_THROW("Locked archive");
322 }
323 if (cursor_ + n*sizeof(T) > endAllocated_) {
324 UTIL_THROW("Attempted write past end of allocated block");
325 }
326 T* ptr = (T *)cursor_;
327 for (int i=0; i < n; ++i) {
328 *ptr = array[i];
329 ++ptr;
330 }
331 cursor_ = (Byte *)ptr;
332 }
333
334 /*
335 * Bitwise pack a 2D C-array of objects of type T.
336 */
337 template <typename T>
338 inline void MemoryOArchive::pack(const T* array, int m, int n, int np)
339 {
340 if (isLocked_) {
341 UTIL_THROW("Locked archive");
342 }
343 if (cursor_ + m*n*sizeof(T) > endAllocated_) {
344 UTIL_THROW("Attempted write past end of allocated block");
345 }
346 int i, j;
347 T* ptr = (T *)cursor_;
348 for (i=0; i < m; ++i) {
349 for (j=0; j < n; ++j) {
350 *ptr = array[i*np + j];
351 ++ptr;
352 }
353 }
354 cursor_ = (Byte *)ptr;
355 }
356
357 // Explicit serialize functions for primitive types
358
359 /*
360 * Save a bool to a MemoryOArchive.
361 */
362 template <>
363 inline void serialize(MemoryOArchive& ar, bool& data,
364 const unsigned int version)
365 { ar.pack(data); }
366
367 /*
368 * Save a char to a MemoryOArchive.
369 */
370 template <>
371 inline void serialize(MemoryOArchive& ar, char& data,
372 const unsigned int version)
373 { ar.pack(data); }
374
375 /*
376 * Save an unsigned int to a MemoryOArchive.
377 */
378 template <>
379 inline void serialize(MemoryOArchive& ar, unsigned int& data,
380 const unsigned int version)
381 { ar.pack(data); }
382
383 /*
384 * Save an int to a MemoryOArchive.
385 */
386 template <>
387 inline void serialize(MemoryOArchive& ar, int& data,
388 const unsigned int version)
389 { ar.pack(data); }
390
391 /*
392 * Save a unsigned long int to a MemoryOArchive.
393 */
394 template <>
395 inline void serialize(MemoryOArchive& ar, unsigned long& data,
396 const unsigned int version)
397 { ar.pack(data); }
398
399 /*
400 * Save a long int to a MemoryOArchive.
401 */
402 template <>
403 inline void serialize(MemoryOArchive& ar, long& data,
404 const unsigned int version)
405 { ar.pack(data); }
406
407 /*
408 * Save a float to a MemoryOArchive.
409 */
410 template <>
411 inline void serialize(MemoryOArchive& ar, float& data,
412 const unsigned int version)
413 { ar.pack(data); }
414
415 /*
416 * Save an double to a MemoryOArchive.
417 */
418 template <>
419 inline void serialize(MemoryOArchive& ar, double& data,
420 const unsigned int version)
421 { ar.pack(data); }
422
423 /*
424 * Save a std::vector to a MemoryOArchive.
425 */
426 template <typename T>
427 void serialize(MemoryOArchive& ar, std::vector<T>& data,
428 const unsigned int version)
429 {
430 size_t size = data.size();
431 ar.pack(size);
432 for (size_t i = 0; i < size; ++i) {
433 ar & data[i];
434 }
435 }
436
437 // Explicit serialize functions for standard library types
438
439 /*
440 * Save a std::complex<float> to a MemoryOArchive.
441 */
442 template <>
443 inline void serialize(MemoryOArchive& ar, std::complex<float>& data,
444 const unsigned int version)
445 { ar.pack(data); }
446
447 /*
448 * Save a std::complex<double> to a MemoryOArchive.
449 */
450 template <>
451 inline void serialize(MemoryOArchive& ar, std::complex<double>& data,
452 const unsigned int version)
453 { ar.pack(data); }
454
455 /*
456 * Save a std::string to a MemoryOArchive.
457 */
458 template <>
459 inline void serialize(MemoryOArchive& ar, std::string& data,
460 const unsigned int version)
461 {
462 size_t size = data.size() + 1; // the +1 is for the NULL
463 ar.pack(size);
464 const char* temp = data.c_str();
465 ar.pack(temp, size);
466 }
467
468 // Explicit serialize functions for namespace Util
469
470 /*
471 * Save a Util::Vector to a MemoryOArchive.
472 */
473 template <>
474 inline void serialize(MemoryOArchive& ar, Vector& data,
475 const unsigned int version)
476 { ar.pack(data); }
477
478 /*
479 * Save a Util::IntVector to a MemoryOArchive.
480 */
481 template <>
482 inline void serialize(MemoryOArchive& ar, IntVector& data,
483 const unsigned int version)
484 { ar.pack(data); }
485
486}
487#endif
Input archive for packed heterogeneous binary data.
Save archive for packed heterogeneous binary data.
virtual ~MemoryOArchive()
Destructor.
void iSend(MPI::Intracomm &comm, MPI::Request &req, int dest)
Send packed data via MPI (non-blocking)
virtual void allocate(size_t capacity)
Allocate memory.
void pack(const T &data)
Pack a T object.
static bool is_loading()
Returns false;.
bool isAllocated() const
Has memory been allocated?
Byte * cursor() const
Return pointer to current position (cursor).
static bool is_saving()
Returns true;.
MemoryOArchive & operator<<(T &data)
Save (write) one object to this archive via the << operator.
MemoryOArchive & operator&(T &data)
Save (write) one object to this archive via the & operator.
Byte * begin() const
Return pointer to beginning of block.
size_t capacity() const
Return capacity in Bytes.
MemoryOArchive()
Constructor.
void send(MPI::Intracomm &comm, int dest)
Send packed data via MPI.
void clear()
Resets the cursor to the beginning.
#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