aboutsummaryrefslogtreecommitdiff
path: root/src/stable/archivebinary.h
blob: 4c12e0212722f75678e01311fe37c088259838e6 (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
111
112
113
114
115
116
117
118
119
120
/*
 * Copyright (C) 2007-2019 Xagasoft, All rights reserved.
 *
 * This file is part of the libbu++ library and is released under the
 * terms of the license contained in the file LICENSE.
 */

#ifndef BU_ARCHIVE_BINARY_H
#define BU_ARCHIVE_BINARY_H

#include <stdint.h>
#include "bu/archive.h"
#include "bu/hash.h"
#include "bu/util.h"
#include "bu/variant.h"
#include "bu/blob.h"

namespace Bu
{
    class Stream;

    /**
     * Provides a framework for serialization of objects and primitives.  The
     * archive will handle any basic primitive, a few special types, like char *
     * strings, as well as STL classes and anything that inherits from the
     * Archival class.  Each ArchiveBinary operates on a Stream, so you can send the
     * data using an ArchiveBinary almost anywhere.
     *
     * In order to use an ArchiveBinary to store something to a file, try something
     * like:
     *@code
     * File sOut("output", "wb"); // This is a stream subclass
     * ArchiveBinary ar( sOut, ArchiveBinary::save );
     * ar << myClass;
     @endcode
     * In this example myClass is any class that inherits from Archival.  When
     * the storage operator is called, the Archival::archive() function in the
     * myClass object is called with a reference to the ArchiveBinary.  This can be
     * handled in one of two ways:
     *@code
     * void MyClass::archive( ArchiveBinary &ar )
     * {
     *  ar && sName && nAge && sJob;
     * }
     @endcode
     * Here we don't worry about weather we're loading or saving by using the
     * smart && operator.  This allows us to write very consistent, very simple
     * archive functions that really do a lot of work.  If we wanted to do
     * something different in the case of loading or saving we would do:
     *@code
     * void MyClass::archive( ArchiveBinary &ar )
     * {
     *  if( ar.isLoading() )
     *  {
     *      ar >> sName >> nAge >> sJob;
     *  } else
     *  {
     *      ar << sName << nAge << sJob;
     *  }
     * }
     @endcode
     * ArchiveBinary currently does not provide facility to make fully portable
     * archives.  For example, it will not convert between endianness for you,
     * nor will it take into account differences between primitive sizes on
     * different platforms.  This, at the moment, is up to the user to ensure.
     * One way of dealing with the latter problem is to make sure and use
     * explicit primitive types from the stdint.h header, i.e. int32_t.
     */
    class ArchiveBinary : public Archive
    {
    private:
        bool bLoading;
    public:
        bool isLoading();

        ArchiveBinary( Stream &rStream, bool bLoading );
        virtual ~ArchiveBinary();
        virtual void close();

        virtual void write( const void *pData, size_t iSize );
        virtual void read( void *pData, size_t iSize );
        
        /**
         * For storage, get an ID for the pointer to the object you're going to
         * write.
         */
        uint32_t getID( const void *ptr );

        /**
         * For loading.  Assosiates an empty pointer with an id.  When you wind
         * up loading an id reference to a pointer for an object that may or
         * may not have loaded yet, call this with the id, if it has been loaded
         * already, you'll immediately get a pointer, if not, it will write one
         * for you when the time comes.
         */
        void assocPtrID( void **ptr, uint32_t id );

        /**
         * For loading.  Call this when you load an object that other things may
         * have pointers to.  It will assosiate every pointer that's been
         * registered with assocPtrID to the pointer passed in, and id passed
         * in.  It will also set things up so future calls to assocPtrID will
         * automatically succeed immediately.
         */
        void readID( const void *ptr, uint32_t id );

        virtual void setProperty( const Bu::Blob &rKey,
                const Bu::Variant &rValue );
        virtual Bu::Variant getProperty( const Bu::Blob &rKey ) const;

    private:
        Stream &rStream;
        uint32_t nNextID;
        Hash<uint32_t,ptrdiff_t> hPtrID;
        Hash<uint32_t,List<void **> > hPtrDest;
        Hash<Bu::Blob, Variant> hProps;
    };
}

#endif