File: SafeArray.h

package info (click to toggle)
pcsx2 1.6.0%2Bdfsg-2
  • links: PTS, VCS
  • area: main
  • in suites: bookworm
  • size: 28,464 kB
  • sloc: cpp: 299,796; ansic: 23,973; lisp: 2,689; asm: 908; perl: 852; sh: 789; xml: 116; makefile: 60
file content (215 lines) | stat: -rw-r--r-- 7,338 bytes parent folder | download | duplicates (4)
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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
/*  PCSX2 - PS2 Emulator for PCs
 *  Copyright (C) 2002-2010  PCSX2 Dev Team
 *
 *  PCSX2 is free software: you can redistribute it and/or modify it under the terms
 *  of the GNU Lesser General Public License as published by the Free Software Found-
 *  ation, either version 3 of the License, or (at your option) any later version.
 *
 *  PCSX2 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 PCSX2.
 *  If not, see <http://www.gnu.org/licenses/>.
 */

#pragma once

// pxUSE_SECURE_MALLOC - enables bounds checking on scoped malloc allocations.

#ifndef pxUSE_SECURE_MALLOC
#define pxUSE_SECURE_MALLOC 0
#endif


// Microsoft Windows only macro, useful for freeing out COM objects:
#define safe_release(ptr) \
    ((void)((((ptr) != NULL) && ((ptr)->Release(), !!0)), (ptr) = NULL))

// --------------------------------------------------------------------------------------
//  SafeArray
// --------------------------------------------------------------------------------------
// Handy little class for allocating a resizable memory block, complete with exception
// error handling and automatic cleanup.  A lightweight alternative to std::vector.
//
template <typename T>
class SafeArray
{
    DeclareNoncopyableObject(SafeArray);

public:
    static const int DefaultChunkSize = 0x1000 * sizeof(T);

public:
    wxString Name; // user-assigned block name
    int ChunkSize;

protected:
    T *m_ptr;
    int m_size; // size of the allocation of memory

protected:
    SafeArray(const wxChar *name, T *allocated_mem, int initSize);
    virtual T *_virtual_realloc(int newsize);

    // A safe array index fetcher.  Asserts if the index is out of bounds (dev and debug
    // builds only -- no bounds checking is done in release builds).
    T *_getPtr(uint i) const;

public:
    virtual ~SafeArray();

    explicit SafeArray(const wxChar *name = L"Unnamed");
    explicit SafeArray(int initialSize, const wxChar *name = L"Unnamed");

    void Dispose();
    void ExactAlloc(int newsize);
    void MakeRoomFor(int newsize)
    {
        if (newsize > m_size)
            ExactAlloc(newsize);
    }

    bool IsDisposed() const { return (m_ptr == NULL); }

    // Returns the size of the memory allocation, as according to the array type.
    int GetLength() const { return m_size; }
    // Returns the size of the memory allocation in bytes.
    int GetSizeInBytes() const { return m_size * sizeof(T); }

    // Extends the containment area of the array.  Extensions are performed
    // in chunks.
    void GrowBy(int items)
    {
        MakeRoomFor(m_size + ChunkSize + items + 1);
    }

    // Gets a pointer to the requested allocation index.
    // DevBuilds : Generates assertion if the index is invalid.
    T *GetPtr(uint idx = 0) { return _getPtr(idx); }
    const T *GetPtr(uint idx = 0) const { return _getPtr(idx); }

    // Gets a pointer to the element directly after the last element in the array.
    // This is equivalent to doing GetPtr(GetLength()), except that this call *avoids*
    // the out-of-bounds assertion check that typically occurs when you do that. :)
    T *GetPtrEnd() { return &m_ptr[m_size]; }
    const T *GetPtrEnd() const { return &m_ptr[m_size]; }

    // Gets an element of this memory allocation much as if it were an array.
    // DevBuilds : Generates assertion if the index is invalid.
    T &operator[](int idx) { return *_getPtr((uint)idx); }
    const T &operator[](int idx) const { return *_getPtr((uint)idx); }

    virtual SafeArray<T> *Clone() const;
};

//////////////////////////////////////////////////////////////////////////////////////////
// SafeList - Simple growable container without all the mess or hassle of std containers.
//
// This container is intended for reasonably simple class types only.  Things which this
// container does not handle with desired robustness:
//
//  * Classes with non-trivial constructors (such that construction creates much overhead)
//  * Classes with copy constructors (copying is done using performance memcpy)
//  * Classes with destructors (they're not called, sorry!)
//
template <typename T>
class SafeList
{
    DeclareNoncopyableObject(SafeList);

public:
    static const int DefaultChunkSize = 0x80 * sizeof(T);

public:
    wxString Name; // user-assigned block name
    int ChunkSize; // assigned DefaultChunkSize on init, reconfigurable at any time.

protected:
    T *m_ptr;
    int m_allocsize; // size of the allocation of memory
    uint m_length;   // length of the array (active items, not buffer allocation)

protected:
    virtual T *_virtual_realloc(int newsize);
    void _MakeRoomFor_threshold(int newsize);

    T *_getPtr(uint i) const;

public:
    virtual ~SafeList();
    explicit SafeList(const wxChar *name = L"Unnamed");
    explicit SafeList(int initialSize, const wxChar *name = L"Unnamed");
    virtual SafeList<T> *Clone() const;

    void Remove(int index);
    void MakeRoomFor(int blockSize);

    T &New();
    int Add(const T &src);
    T &AddNew(const T &src);

    // Returns the size of the list, as according to the array type.  This includes
    // mapped items only.  The actual size of the allocation may differ.
    int GetLength() const { return m_length; }

    // Returns the size of the list, in bytes.  This includes mapped items only.
    // The actual size of the allocation may differ.
    int GetSizeInBytes() const { return m_length * sizeof(T); }

    void MatchLengthToAllocatedSize()
    {
        m_length = m_allocsize;
    }

    void GrowBy(int items)
    {
        MakeRoomFor(m_length + ChunkSize + items + 1);
    }

    // Sets the item length to zero.  Does not free memory allocations.
    void Clear()
    {
        m_length = 0;
    }

    // Gets an element of this memory allocation much as if it were an array.
    // DevBuilds : Generates assertion if the index is invalid.
    T &operator[](int idx) { return *_getPtr((uint)idx); }
    const T &operator[](int idx) const { return *_getPtr((uint)idx); }

    T *GetPtr() { return m_ptr; }
    const T *GetPtr() const { return m_ptr; }

    T &GetLast() { return m_ptr[m_length - 1]; }
    const T &GetLast() const { return m_ptr[m_length - 1]; }
};

// --------------------------------------------------------------------------------------
//  SafeAlignedArray<T>
// --------------------------------------------------------------------------------------
// Handy little class for allocating a resizable memory block, complete with
// exception-based error handling and automatic cleanup.
// This one supports aligned data allocations too!

template <typename T, uint Alignment>
class SafeAlignedArray : public SafeArray<T>
{
    typedef SafeArray<T> _parent;

protected:
    T *_virtual_realloc(int newsize);

public:
    using _parent::operator[];

    virtual ~SafeAlignedArray();

    explicit SafeAlignedArray(const wxChar *name = L"Unnamed")
        : SafeArray<T>::SafeArray(name)
    {
    }

    explicit SafeAlignedArray(int initialSize, const wxChar *name = L"Unnamed");
    virtual SafeAlignedArray<T, Alignment> *Clone() const;
};