mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
binVector.hpp
Go to the documentation of this file.
1/** \file binVector.hpp
2 * \author Jared R. Males
3 * \brief A utility to read/write vectors of data from/to a binary file.
4 * \ingroup utils_files
5 */
6
7//***********************************************************************//
8// Copyright 2015, 2016, 2017 Jared R. Males (jaredmales@gmail.com)
9//
10// This file is part of mxlib.
11//
12// mxlib is free software: you can redistribute it and/or modify
13// it under the terms of the GNU General Public License as published by
14// the Free Software Foundation, either version 3 of the License, or
15// (at your option) any later version.
16//
17// mxlib is distributed in the hope that it will be useful,
18// but WITHOUT ANY WARRANTY; without even the implied warranty of
19// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20// GNU General Public License for more details.
21//
22// You should have received a copy of the GNU General Public License
23// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
24//***********************************************************************//
25
26#ifndef binVector_hpp
27#define binVector_hpp
28
29#include <vector>
30#include <complex>
31#include <cstdint>
32
33#include "../mxlib.hpp"
34#include "../mxError.hpp"
35
36namespace mx
37{
38namespace ioutils
39{
40
41/** \defgroup binvector binVector Binary File Format
42 * \ingroup ioutils
43 * \brief A simple binary file format for storing vectors of data on disk.
44 *
45 * The binVector file format is very simple: the first 8 bytes contain a unit64_t integer which specifies
46 * the type of the data. The second 8 bytes contain a uint64_t integer which specifies the length L of the data vector.
47 * The remaining L*sizeof(dataT) bytes contain the data.
48 *
49 * The suggested extension for BinVector files is ".binv".
50 *
51 *
52 */
53
54/// The type used for binVector sizes.
55/** \ingroup binVector
56 */
57typedef uint64_t binVSizeT;
58
59/// The type of binVector type codes.
60/** \ingroup binvector
61 */
62typedef uint64_t binVTypeT;
63
64/// Namespace fo the bin-vector type-codes.
65/** \ingroup binvector
66 */
67namespace binVTypes
68{
69
70/// The pre-defined type codes for binVector.
71/** \ingroup binvector
72 */
73enum : binVTypeT
74{
75 Bool = 0,
76 SChar = 1,
77 UChar = 2,
78 Char = 3,
79 WChar = 4,
80 Char16 = 5,
81 Char32 = 6,
82 Int = 7,
83 UInt = 8,
84 SInt = 9,
85 SUInt = 10,
86 LInt = 11,
87 LUInt = 12,
88 LLInt = 13,
89 LLUInt = 14,
90 Float = 15,
91 Double = 16,
92 LDouble = 17,
93 Quad = 18,
94 CFloat = 19,
95 CDouble = 20,
96};
97
98} // namespace binVTypes
99
100/// Get the integer type code corresponding to the type.
101/**
102 * \returns an integer which uniquely identifies the type.
103 *
104 * \tparam dataT is the type
105 *
106 * \ingroup binvector
107 */
108template <typename dataT>
110
111template <>
112binVTypeT binVectorTypeCode<bool>()
113{
114 return binVTypes::Bool;
115}
116
117template <>
118binVTypeT binVectorTypeCode<signed char>()
119{
120 return binVTypes::SChar;
121}
122
123template <>
124binVTypeT binVectorTypeCode<unsigned char>()
125{
126 return binVTypes::UChar;
127}
128
129template <>
130binVTypeT binVectorTypeCode<char>()
131{
132 return binVTypes::Char;
133}
134
135template <>
136binVTypeT binVectorTypeCode<wchar_t>()
137{
138 return binVTypes::WChar;
139}
140
141template <>
142binVTypeT binVectorTypeCode<char16_t>()
143{
144 return binVTypes::Char16;
145}
146
147template <>
148binVTypeT binVectorTypeCode<char32_t>()
149{
150 return binVTypes::Char32;
151}
152
153template <>
154binVTypeT binVectorTypeCode<int>()
155{
156 return binVTypes::Int;
157}
158
159template <>
160binVTypeT binVectorTypeCode<unsigned int>()
161{
162 return binVTypes::UInt;
163}
164
165template <>
166binVTypeT binVectorTypeCode<short int>()
167{
168 return binVTypes::SInt;
169}
170
171template <>
172binVTypeT binVectorTypeCode<short unsigned int>()
173{
174 return binVTypes::SUInt;
175}
176
177template <>
178binVTypeT binVectorTypeCode<long int>()
179{
180 return binVTypes::LInt;
181}
182
183template <>
184binVTypeT binVectorTypeCode<long unsigned int>()
185{
186 return binVTypes::LUInt;
187}
188
189template <>
190binVTypeT binVectorTypeCode<long long int>()
191{
192 return binVTypes::LLInt;
193}
194
195template <>
196binVTypeT binVectorTypeCode<long long unsigned int>()
197{
198 return binVTypes::LLUInt;
199}
200
201template <>
202binVTypeT binVectorTypeCode<float>()
203{
204 return binVTypes::Float;
205}
206
207template <>
208binVTypeT binVectorTypeCode<double>()
209{
210 return binVTypes::Double;
211}
212
213template <>
214binVTypeT binVectorTypeCode<long double>()
215{
216 return binVTypes::LDouble;
217}
218
219template <>
220binVTypeT binVectorTypeCode<std::complex<float>>()
221{
222 return binVTypes::CFloat;
223}
224
225template <>
226binVTypeT binVectorTypeCode<std::complex<double>>()
227{
228 return binVTypes::CDouble;
229}
230
231/// Read a BinVector file from disk.
232/**
233 * \note dataT must match what was stored in the file.
234 *
235 * \returns 0 on success.
236 * \returns -1 if an error occurs.
237 *
238 * \ingroup binvector
239 */
240template <typename dataT>
241int readBinVector( std::vector<dataT> &vec, ///< [out] vec is a vector which will be resized and populated.
242 const std::string &fname ///< [in] fname is the name (full-path) of the file.
243)
244{
245 FILE *fin;
246 binVTypeT typecode;
247 binVSizeT sz;
248 size_t nrd;
249
250 fin = fopen( fname.c_str(), "r" );
251 if( fin == 0 )
252 {
253 mxPError( "readBinVector", errno, "Error from fopen [" + fname + "]" );
254 return -1;
255 }
256
257 errno = 0;
258 nrd = fread( &typecode, sizeof( binVTypeT ), 1, fin );
259
260 if( nrd != 1 )
261 {
262 // Have to handle case where EOF reached but no error.
263 if( errno != 0 )
264 {
265 mxPError( "readBinVector", errno, "Error reading data size [" + fname + "]" );
266 }
267 else
268 {
269 mxError(
270 "readBinVector", MXE_FILERERR, "Error reading data size, did not read enough bytes. [" + fname + "]" );
271 }
272 fclose( fin );
273 return -1;
274 }
275
276 if( typecode != binVectorTypeCode<dataT>() )
277 {
278 mxError( "readBinVector", MXE_SIZEERR, "Mismatch between type dataT and type in file [" + fname + "]" );
279 fclose( fin );
280 return -1;
281 }
282
283 errno = 0;
284 nrd = fread( &sz, sizeof( binVSizeT ), 1, fin );
285
286 if( nrd != 1 )
287 {
288 // Have to handle case where EOF reached but no error.
289 if( errno != 0 )
290 {
291 mxPError( "readBinVector", errno, "Error reading vector size [" + fname + "]" );
292 }
293 else
294 {
295 mxError(
296 "readBinVector", MXE_FILERERR, "Error reading vector size, did not read enough bytes [" + fname + "]" );
297 }
298 fclose( fin );
299 return -1;
300 }
301
302 vec.resize( sz );
303
304 errno = 0;
305 nrd = fread( vec.data(), sizeof( dataT ), sz, fin );
306
307 if( nrd != sz )
308 {
309 // Have to handle case where EOF reached but no error.
310 if( errno != 0 )
311 {
312 mxPError( "readBinVector", errno, "Error reading data [" + fname + "]" );
313 }
314 else
315 {
316 mxError( "readBinVector", MXE_FILERERR, "Did not read enough data [" + fname + "]" );
317 }
318 fclose( fin );
319 return -1;
320 }
321
322 fclose( fin );
323
324 return 0;
325}
326
327/// Write a BinVector file to disk.
328/**
329 *
330 * \returns 0 on success.
331 * \returns -1 if an error occurs.
332 *
333 * \ingroup binvector
334 */
335template <typename dataT>
336int writeBinVector( const std::string &fname, ///< [in] fname is the name (full-path) of the file.
337 std::vector<dataT> &vec ///< [in] vec is the vector which will be written to disk.
338)
339{
340
341 FILE *fout;
342 size_t nwr;
343 binVTypeT typecode = binVectorTypeCode<dataT>();
344 binVSizeT sz = vec.size();
345
346 fout = fopen( fname.c_str(), "wb" );
347 if( fout == 0 )
348 {
349 mxPError( "writeBinVector", errno, "Error from fopen [" + fname + "]" );
350 return -1;
351 }
352
353 nwr = fwrite( &typecode, sizeof( binVTypeT ), 1, fout );
354 if( nwr != 1 )
355 {
356 mxPError( "writeBinVector", errno, "Error writing typecode [" + fname + "]" );
357 fclose( fout );
358 return -1;
359 }
360
361 nwr = fwrite( &sz, sizeof( binVSizeT ), 1, fout );
362 if( nwr != 1 )
363 {
364 mxPError( "writeBinVector", errno, "Error writing vector size [" + fname + "]" );
365 fclose( fout );
366 return -1;
367 }
368
369 nwr = fwrite( vec.data(), sizeof( dataT ), vec.size(), fout );
370 if( nwr != sz )
371 {
372 mxPError( "writeBinVector", errno, "Error writing data [" + fname + "]" );
373 fclose( fout );
374 return -1;
375 }
376
377 fclose( fout );
378
379 return 0;
380}
381
382} // namespace ioutils
383} // namespace mx
384
385#endif // binVector_hpp
uint64_t binVSizeT
The type used for binVector sizes.
Definition binVector.hpp:57
int writeBinVector(const std::string &fname, std::vector< dataT > &vec)
Write a BinVector file to disk.
binVTypeT binVectorTypeCode()
Get the integer type code corresponding to the type.
uint64_t binVTypeT
The type of binVector type codes.
Definition binVector.hpp:62
int readBinVector(std::vector< dataT > &vec, const std::string &fname)
Read a BinVector file from disk.
The mxlib c++ namespace.
Definition mxError.hpp:106