WritableBuffer.hxx 6.72 KB
Newer Older
1
/*
2
 * Copyright 2013-2020 Max Kellermann <max.kellermann@gmail.com>
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
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * - Redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer.
 *
 * - Redistributions in binary form must reproduce the above copyright
 * notice, this list of conditions and the following disclaimer in the
 * documentation and/or other materials provided with the
 * distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
 * FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
 * OF THE POSSIBILITY OF SUCH DAMAGE.
 */

30 31
#ifndef WRITABLE_BUFFER_HXX
#define WRITABLE_BUFFER_HXX
32

33
#include "ConstBuffer.hxx"
34 35
#include "Compiler.h"

36 37 38
#include <cstddef>

#ifndef NDEBUG
39
#include <cassert>
40
#endif
41

42 43 44 45 46
template<typename T>
struct WritableBuffer;

template<>
struct WritableBuffer<void> {
47
	typedef std::size_t size_type;
48
	typedef void value_type;
49 50 51 52
	typedef void *pointer;
	typedef const void *const_pointer;
	typedef pointer iterator;
	typedef const_pointer const_iterator;
53

54
	pointer data;
55 56 57 58
	size_type size;

	WritableBuffer() = default;

59 60
	constexpr WritableBuffer(std::nullptr_t) noexcept
		:data(nullptr), size(0) {}
61

62
	constexpr WritableBuffer(pointer _data, size_type _size) noexcept
63 64
		:data(_data), size(_size) {}

65 66 67 68
	constexpr operator ConstBuffer<void>() const noexcept {
		return {data, size};
	}

69
	constexpr bool IsNull() const noexcept {
70 71 72
		return data == nullptr;
	}

73
	constexpr bool operator==(std::nullptr_t) const noexcept {
74 75 76
		return data == nullptr;
	}

77
	constexpr bool operator!=(std::nullptr_t) const noexcept {
78 79 80
		return data != nullptr;
	}

81
	constexpr bool empty() const noexcept {
82 83 84 85
		return size == 0;
	}
};

86 87 88 89 90 91 92
/**
 * A reference to a memory area that is writable.
 *
 * @see ConstBuffer
 */
template<typename T>
struct WritableBuffer {
93
	typedef std::size_t size_type;
94
	typedef T value_type;
95 96
	typedef T &reference;
	typedef const T &const_reference;
97 98 99 100
	typedef T *pointer;
	typedef const T *const_pointer;
	typedef pointer iterator;
	typedef const_pointer const_iterator;
101

102
	pointer data;
103
	size_type size;
104

105
	WritableBuffer() = default;
106

107 108
	constexpr WritableBuffer(std::nullptr_t) noexcept
		:data(nullptr), size(0) {}
109

110
	constexpr WritableBuffer(pointer _data, size_type _size) noexcept
111
		:data(_data), size(_size) {}
112

113
	constexpr WritableBuffer(pointer _data, pointer _end) noexcept
114 115
		:data(_data), size(_end - _data) {}

116 117 118 119
	/**
	 * Convert array to WritableBuffer instance.
	 */
	template<size_type _size>
120
	constexpr WritableBuffer(T (&_data)[_size]) noexcept
121 122
		:data(_data), size(_size) {}

123 124 125 126
	constexpr operator ConstBuffer<T>() const noexcept {
		return {data, size};
	}

127 128 129 130
	/**
	 * Cast a WritableBuffer<void> to a WritableBuffer<T>,
	 * rounding down to the next multiple of T's size.
	 */
131
	static constexpr WritableBuffer<T> FromVoidFloor(WritableBuffer<void> other) noexcept {
132
		static_assert(sizeof(T) > 0, "Empty base type");
133
		return WritableBuffer<T>(pointer(other.data),
134 135 136
					 other.size / sizeof(T));
	}

137 138 139 140 141 142 143 144 145
	/**
	 * Cast a WritableBuffer<void> to a WritableBuffer<T>.  A "void"
	 * buffer records its size in bytes, and when casting to "T",
	 * the assertion below ensures that the size is a multiple of
	 * sizeof(T).
	 */
#ifdef NDEBUG
	constexpr
#endif
146
	static WritableBuffer<T> FromVoid(WritableBuffer<void> other) noexcept {
147
		static_assert(sizeof(T) > 0, "Empty base type");
148 149 150
#ifndef NDEBUG
		assert(other.size % sizeof(T) == 0);
#endif
151
		return FromVoidFloor(other);
152 153
	}

154
	constexpr WritableBuffer<void> ToVoid() const noexcept {
155 156 157 158
		static_assert(sizeof(T) > 0, "Empty base type");
		return WritableBuffer<void>(data, size * sizeof(T));
	}

159
	constexpr bool IsNull() const noexcept {
160 161
		return data == nullptr;
	}
162

163
	constexpr bool operator==(std::nullptr_t) const noexcept {
164 165 166
		return data == nullptr;
	}

167
	constexpr bool operator!=(std::nullptr_t) const noexcept {
168 169 170
		return data != nullptr;
	}

171
	constexpr bool empty() const noexcept {
172 173
		return size == 0;
	}
174

175
	constexpr iterator begin() const noexcept {
176 177
		return data;
	}
178

179
	constexpr iterator end() const noexcept {
180 181
		return data + size;
	}
182

183
	constexpr const_iterator cbegin() const noexcept {
184 185
		return data;
	}
186

187
	constexpr const_iterator cend() const noexcept {
188 189
		return data + size;
	}
190 191 192 193

#ifdef NDEBUG
	constexpr
#endif
194
	reference operator[](size_type i) const noexcept {
195 196 197 198 199 200
#ifndef NDEBUG
		assert(i < size);
#endif

		return data[i];
	}
201 202 203 204 205 206 207 208

	/**
	 * Returns a reference to the first element.  Buffer must not
	 * be empty.
	 */
#ifdef NDEBUG
	constexpr
#endif
209
	reference front() const noexcept {
210
#ifndef NDEBUG
211
		assert(!empty());
212 213 214 215 216 217 218 219 220 221 222
#endif
		return data[0];
	}

	/**
	 * Returns a reference to the last element.  Buffer must not
	 * be empty.
	 */
#ifdef NDEBUG
	constexpr
#endif
223
	reference back() const noexcept {
224
#ifndef NDEBUG
225
		assert(!empty());
226 227 228 229 230 231 232 233
#endif
		return data[size - 1];
	}

	/**
	 * Remove the first element (by moving the head pointer, does
	 * not actually modify the buffer).  Buffer must not be empty.
	 */
234
	void pop_front() noexcept {
235
		assert(!empty());
236 237 238 239 240 241 242 243 244

		++data;
		--size;
	}

	/**
	 * Remove the last element (by moving the tail pointer, does
	 * not actually modify the buffer).  Buffer must not be empty.
	 */
245
	void pop_back() noexcept {
246
		assert(!empty());
247 248 249 250 251 252 253 254

		--size;
	}

	/**
	 * Remove the first element and return a reference to it.
	 * Buffer must not be empty.
	 */
255 256
	reference shift() noexcept {
		reference result = front();
257 258 259
		pop_front();
		return result;
	}
260

261
	void skip_front(size_type n) noexcept {
262 263 264 265 266 267 268
#ifndef NDEBUG
		assert(size >= n);
#endif

		data += n;
		size -= n;
	}
269 270 271 272 273

	/**
	 * Move the front pointer to the given address, and adjust the
	 * size attribute to retain the old end address.
	 */
274
	void MoveFront(pointer new_data) noexcept {
275 276 277 278 279 280 281 282
#ifndef NDEBUG
		assert(IsNull() == (new_data == nullptr));
		assert(new_data <= end());
#endif

		size = end() - new_data;
		data = new_data;
	}
283 284 285 286 287

	/**
	 * Move the end pointer to the given address (by adjusting the
	 * size).
	 */
288
	void SetEnd(pointer new_end) noexcept {
289 290 291 292 293 294 295
#ifndef NDEBUG
		assert(IsNull() == (new_end == nullptr));
		assert(new_end >= begin());
#endif

		size = new_end - data;
	}
296 297 298
};

#endif