MagAO-X
Operations Applications Utilities Source
logHeader.hpp
Go to the documentation of this file.
1 /** \file logHeader.hpp
2  * \brief The flatlogs buffer header format.
3  * \author Jared R. Males (jaredmales@gmail.com)
4  *
5  * \ingroup flatlogs_files
6  *
7  * History:
8  * - 2017-08-29 created by JRM
9  * - 2018-08-17 moved to flatlogs
10  *
11  */
12 #ifndef flatlogs_logHeader_hpp
13 #define flatlogs_logHeader_hpp
14 
15 #include "logDefs.hpp"
16 #include "timespecX.hpp"
17 #include "logPriority.hpp"
18 
19 namespace flatlogs
20 {
21 
22 /** The log entry is a binary buffer with the following format:
23  * \addtogroup flatlogs
24  * \verbatim
25  1 1 1 1 1 1 1 1 1 1 2 2 2 2 2 2 2 2 2
26  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8
27  |p|evt|time_s |time_ns|l| [message (0-254)]
28  |p|evt|time_s |time_ns|E|len| [message (255-65534)]
29  |p|evt|time_s |time_ns|F|len | [message (>65535)]
30  \endverbatim
31  * where the header consists of everything up to the [message]. Table notations are:
32  * - p (uint8_t) denotes the log priority.
33  * - evt (uint16_t) denotes the event code.
34  * - time_s (uint32_t) is the time in seconds since the epoch
35  * - time_ns (uint32_t) denotes the nanoseconds past that second.
36  * - l/len is the message length field, which is variable in length
37  * depending on the size of the message:
38  * - Message length 0-253: uint8_t in a single byte.
39  * - Message length 255-65534: uint8_t = 0xFF-1 in the first byte, uint16_t in the next two bytes.
40  * - Message length 65535 and up: uint8_t=0xFF in the first byte, uint64_t in the next 8 bytes.
41  *
42  * Rationale for variable length: this keeps the space used for 0 length and short messages
43  * to a minimum, while allowing for progressively larger messages to be saved.
44  * The savings will be large for small messages, while the the cost is proportionally small
45  * for larger messages. For larger messages, this requires a progressive read to
46  * sniff out the entire length. Likewise, the cost is small relative to the cost of reading
47  * the larger message.
48  *
49  * If a non-zero message exists, it is usually a flatlogs serialized buffer.
50  *
51  *
52  */
53 
54 ///The log entry buffer smart pointer.
55 /** \ingroup logbuff
56  */
57 typedef std::shared_ptr<char> bufferPtrT;
58 
59 
60 /// The log entry header
61 /**
62  * This class is designed to work with the log header only as a shared pointer to it,
63  * not directly on the members. The actual header struct is private so we ensure that it is
64  * accessed properly.As such all of the member methods are static and take a shared_ptr as
65  * first argument.
66  *
67  * \ingroup logbuff
68  */
69 class logHeader
70 {
71 
72 public:
73 
74  ///The max value in the msgLen0 field.
75  constexpr static size_t MAX_LEN0 = std::numeric_limits<msgLen0T>::max() ;
76 
77  ///The max value in the msgLen1 field.
78  constexpr static size_t MAX_LEN1 = std::numeric_limits<msgLen1T>::max();
79 
80  ///The minimum header size
81  /** A buffer must be allocated to at least this size.
82  */
83  constexpr static int minHeadSize = sizeof(logPrioT) + sizeof(eventCodeT) +
84  sizeof(secT) + sizeof(nanosecT) + sizeof(msgLen0T);
85 
86  ///The maximum header size
87  /** The header component could be as big as this.
88  */
89  constexpr static int maxHeadSize = minHeadSize + sizeof(msgLen2T);
90 
91 private:
92 
94  {
98  msgLen0T msgLen0; ///< The short message length. Always present.
99  union
100  {
101  msgLen1T msgLen1; ///< The intermediate message length. Only present if msgLen0 == max-1 of msgLen0T.
102  msgLen2T msgLen2; ///< The long message length. Only present if msgLen1 == max-value of msgLen0T.
103  };
104  } __attribute__((packed));
105 
106 public:
107 
108  ///Set the level of a log entry in a logBuffer header
109  /**
110  * \returns 0 on success
111  * \returns -1 on error
112  *
113  * \ingroup logbuff
114  */
115  static int logLevel( bufferPtrT & logBuffer, ///< [in/out] a shared_ptr<char> containing a raw lag entry buffer.
116  const logPrioT & lvl ///< [in] the new log level.
117  );
118 
119  ///Extract the level of a log entry
120  /**
121  * \returns the level
122  *
123  * \ingroup logbuff
124  */
125  static logPrioT logLevel( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
126 
127  ///Set the event code of a log entry
128  /**
129  * \returns 0 on success
130  * \returns -1 on error
131  *
132  * \ingroup logbuff
133  */
134  static int eventCode( bufferPtrT & logBuffer, ///< [in,out] a shared_ptr<char> containing a raw lag entry buffer.
135  const eventCodeT & ec ///< [in] the new event code.
136  );
137 
138  ///Extract the event code of a log entry
139  /**
140  * \returns the event code
141  *
142  * \ingroup logbuff
143  */
144  static eventCodeT eventCode( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
145 
146  ///Extract the timespec of a log entry
147  /**
148  * \returns the timespec
149  *
150  * \ingroup logbuff
151  */
152  static timespecX timespec( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
153 
154  ///Set the timespec of a log entry
155  /**
156  * \returns 0 on success
157  * \returns -1 on error
158  *
159  * \ingroup logbuff
160  */
161  static int timespec( bufferPtrT & logBuffer, ///< [in, out] a shared_ptr<char> containing a raw lag entry buffer.*/
162  const timespecX & ts ///< [in] the new timespec
163  );
164 
165  ///Get the size in bytes of the length field for an existing logBuffer.
166  /**
167  * \returns the number of bytes in the length field.
168  *
169  * \ingroup logbuff
170  */
171  static size_t lenSize( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
172 
173  ///Get the size in bytes of the length field for a logBuffer given the intended message length
174  /**
175  * \returns the number of bytes to be put in the length field.
176  *
177  * \ingroup logbuff
178  */
179  static size_t lenSize( msgLenT & msgSz /**< [in] the size of the intended message.*/);
180 
181  ///Set the message length of a log entry message
182  /**
183  * \note The logBuffer must already be allocated with a header large enough for this message size.
184  *
185  * \returns 0 on success
186  * \returns -1 on error
187  *
188  * \ingroup logbuff
189  */
190  static int msgLen( bufferPtrT & logBuffer, ///< [out] a shared_ptr<char> containing a raw lag entry buffer allocated with large enough header for this message length.
191  const msgLenT & msgLen ///< [in] the message length to set.
192  );
193 
194  ///Extract the short message length of a log entry message
195  /** This is always safe on a minimally allocated logBuffer, can be used to test for progressive reading.
196  *
197  * \returns the short message length field
198  *
199  * \ingroup logbuff
200  */
201  static msgLen0T msgLen0( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
202 
203  ///Extract the medium message length of a log entry message
204  /** This is NOT always safe, and should only be caled if msgLen0 is 0xFE. Can be used to test for progressive reading.
205  *
206  * \returns the medium message length field
207  *
208  * \ingroup logbuff
209  */
210  static msgLen1T msgLen1( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
211 
212  ///Extract the message length of a log entry message
213  /**
214  * \returns the message length
215  *
216  * \ingroup logbuff
217  */
218  static msgLenT msgLen( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
219 
220  ///Get the size of the header, including the variable size length field, for an existing logBuffer.
221  /**
222  * \returns the size of the header in this log entry.
223  *
224  * \ingroup logbuff
225  */
226  static size_t headerSize( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
227 
228  ///Get the size of the header, including the variable size length field, given a message size.
229  /**
230  * \returns the size to be used for the header.
231  *
232  * \ingroup logbuff
233  */
234  static size_t headerSize( msgLenT & msgSz /**< [in] the size of the intended message.*/);
235 
236  ///Get the total size of the log entry, including the message buffer.
237  /**
238  * \returns the total size of this log entry.
239  *
240  * \ingroup logbuff
241  */
242  static size_t totalSize( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
243 
244  ///Get the total size of a log entry, given the message buffer size.
245  /**
246  * \returns the total size to be used for a log entry.
247  *
248  * \ingroup logbuff
249  */
250  static size_t totalSize( msgLenT & msgSz /**< [in] the intended size of the message buffer.*/);
251 
252  ///Get the message buffer address.
253  /**
254  * \returns the address of the message buffer.
255  *
256  * \ingroup logbuff
257  */
258  static void * messageBuffer( bufferPtrT & logBuffer /**< [in] a shared_ptr<char> containing a raw lag entry buffer.*/);
259 
260  /// Create a formatted log entry, filling in a buffer.
261  /** This version has the timestamp provided.
262  *
263  * \tparam logT is a log entry type
264  *
265  * \returns 0 on success, -1 on error.
266  */
267  template<typename logT>
268  static int createLog( bufferPtrT & logBuffer, ///< [out] a shared_ptr<logBuffer>, which will be allocated and populated with the log entry
269  const timespecX & ts, ///< [in] the timestamp of this log entry.
270  const typename logT::messageT & msg, ///< [in] the message to log (could be of type emptyMessage)
271  const logPrioT & level ///< [in] the level (verbosity) of this log
272  );
273 
274  ///Extract the basic details of a log entry
275  /** Convenience wrapper for the other extraction functions.
276  *
277  * \returns 0 on success, -1 on error.
278  *
279  * \ingroup logbuff
280  */
281  static int extractBasicLog( logPrioT & lvl, ///< [out] The log level
282  eventCodeT & ec, ///< [out] the event code
283  timespecX & ts, ///< [out] the timestamp of the log entry
284  msgLenT & len, ///< [out] the message length
285  bufferPtrT & logBuffer ///< [in] a shared_ptr<char> containing a raw log entry buffer.
286  );
287 
288 };
289 
290 inline
291 int logHeader::logLevel( bufferPtrT & logBuffer,
292  const logPrioT & lvl
293  )
294 {
295  reinterpret_cast<internal_logHeader *>(logBuffer.get())->m_logLevel = lvl;
296 
297  return 0;
298 }
299 
300 inline
301 logPrioT logHeader::logLevel( bufferPtrT & logBuffer)
302 {
303  return reinterpret_cast<internal_logHeader *>(logBuffer.get())->m_logLevel;
304 }
305 
306 inline
307 int logHeader::eventCode( bufferPtrT & logBuffer,
308  const eventCodeT & ec
309  )
310 {
311  reinterpret_cast<internal_logHeader *>(logBuffer.get())->m_eventCode = ec;
312 
313  return 0;
314 }
315 
316 inline
317 eventCodeT logHeader::eventCode( bufferPtrT & logBuffer)
318 {
319  return reinterpret_cast<internal_logHeader *>(logBuffer.get())->m_eventCode;
320 }
321 
322 inline
323 timespecX logHeader::timespec( bufferPtrT & logBuffer)
324 {
325  return reinterpret_cast<internal_logHeader *>(logBuffer.get())->m_timespecX;
326 }
327 
328 inline
329 int logHeader::timespec( bufferPtrT & logBuffer,
330  const timespecX & ts
331  )
332 {
333  reinterpret_cast<internal_logHeader *>(logBuffer.get())->m_timespecX = ts;
334 
335  return 0;
336 }
337 
338 inline
339 size_t logHeader::lenSize( bufferPtrT & logBuffer)
340 {
341  msgLen0T len0 = reinterpret_cast<internal_logHeader *>(logBuffer.get())->msgLen0;
342 
343  if(len0 < MAX_LEN0-1) return sizeof(msgLen0T);
344 
345  if(len0 == MAX_LEN0-1)
346  {
347  return sizeof(msgLen0T) + sizeof(msgLen1T);
348  }
349 
350  return sizeof(msgLen0T) + sizeof(msgLen2T);
351 }
352 
353 inline
354 size_t logHeader::lenSize( msgLenT & msgSz /**< [in] the size of the intended message.*/)
355 {
356  if(msgSz < MAX_LEN0-1) return sizeof(msgLen0T);
357 
358  if(msgSz < MAX_LEN1) return sizeof(msgLen0T) + sizeof(msgLen1T);
359 
360  return sizeof(msgLen0T) + sizeof(msgLen2T);
361 }
362 
363 inline
364 size_t logHeader::headerSize( bufferPtrT & logBuffer )
365 {
366  return sizeof(logPrioT) + sizeof(eventCodeT) + sizeof(secT) + sizeof(nanosecT) + lenSize(logBuffer);
367 }
368 
369 inline
371 {
372  return sizeof(logPrioT) + sizeof(eventCodeT) + sizeof(timespecX) + lenSize(msgSz);
373 }
374 
375 inline
376 int logHeader::msgLen( bufferPtrT & logBuffer,
377  const msgLenT & msgLen
378  )
379 {
380  internal_logHeader * lh = reinterpret_cast<internal_logHeader *>(logBuffer.get());
381 
382  if( msgLen < MAX_LEN0-1 ) //254 for uint8_t
383  {
384  lh->msgLen0 = msgLen;
385  return 0;
386  }
387 
388  if(msgLen < MAX_LEN1) //65535 for uint16_t
389  {
390  lh->msgLen0 = MAX_LEN0-1; //254 for uint8_t
391  lh->msgLen1 = msgLen;
392  return 0;
393  }
394 
395  lh->msgLen0 = MAX_LEN0; //255 for uint8_t
396  lh->msgLen2 = msgLen;
397  return 0;
398 }
399 
400 inline
401 msgLen0T logHeader::msgLen0( bufferPtrT & logBuffer )
402 {
403  return reinterpret_cast<internal_logHeader *>(logBuffer.get())->msgLen0;
404 }
405 
406 inline
407 msgLen1T logHeader::msgLen1( bufferPtrT & logBuffer )
408 {
409  return reinterpret_cast<internal_logHeader *>(logBuffer.get())->msgLen1;
410 
411 }
412 
413 inline
414 msgLenT logHeader::msgLen( bufferPtrT & logBuffer )
415 {
416  msgLen0T len0 = reinterpret_cast<internal_logHeader *>(logBuffer.get())->msgLen0;
417 
418  if(len0 < MAX_LEN0-1) return len0;
419 
420  if(len0 == MAX_LEN0-1)
421  {
422  return reinterpret_cast<internal_logHeader *>(logBuffer.get())->msgLen1;
423  }
424 
425  return reinterpret_cast<internal_logHeader *>(logBuffer.get())->msgLen2;
426 }
427 
428 inline
429 size_t logHeader::totalSize( bufferPtrT & logBuffer )
430 {
431  return headerSize(logBuffer) + msgLen(logBuffer);
432 }
433 
434 
435 inline
436 size_t logHeader::totalSize( msgLenT & msgSz )
437 {
438  return headerSize(msgSz) + msgSz;
439 }
440 
441 inline
442 void * logHeader::messageBuffer( bufferPtrT & logBuffer )
443 {
444  return logBuffer.get() + headerSize(logBuffer);
445 }
446 
447 template<typename logT>
448 int logHeader::createLog( bufferPtrT & logBuffer,
449  const timespecX & ts,
450  const typename logT::messageT & msg,
451  const logPrioT & level
452  )
453 {
454  logPrioT lvl;
455  if(level == logPrio::LOG_DEFAULT)
456  {
457  lvl = logT::defaultLevel;
458  }
459  else lvl = level;
460 
461  //We first allocate the buffer.
462  msgLenT len = logT::length(msg);
463  logBuffer = bufferPtrT( (char *) ::operator new(totalSize(len)*sizeof(char)) );
464 
465  //Now load the basics.
466  logLevel(logBuffer, lvl);
467  eventCode(logBuffer, logT::eventCode);
468  timespec(logBuffer, ts);
469 
470  msgLen( logBuffer, len);
471 
472 
473  //Each log-type is responsible for loading its message
474  logT::format( messageBuffer(logBuffer), msg);
475 
476  return 0;
477 
478 }
479 
480 inline
482  eventCodeT & ec,
483  timespecX & ts,
484  msgLenT & len,
485  bufferPtrT & logBuffer
486  )
487 {
488 
489  lvl = logLevel(logBuffer);
490 
491  ec = eventCode(logBuffer);
492 
493  ts = timespec(logBuffer);
494 
495  len = logHeader::msgLen(logBuffer);
496 
497  return 0;
498 }
499 
500 
501 
502 
503 
504 } //namespace flatlogs
505 
506 #endif //flatlogs_logHeader_hpp
507 
static msgLen0T msgLen0(bufferPtrT &logBuffer)
Extract the short message length of a log entry message.
Definition: logHeader.hpp:401
static size_t lenSize(bufferPtrT &logBuffer)
Get the size in bytes of the length field for an existing logBuffer.
Definition: logHeader.hpp:339
uint8_t msgLen0T
The type used for the short message length.
Definition: logDefs.hpp:45
msgLen0T msgLen0
The short message length. Always present.
Definition: logHeader.hpp:98
A fixed-width timespec structure and utilities.
static int eventCode(bufferPtrT &logBuffer, const eventCodeT &ec)
Set the event code of a log entry.
Definition: logHeader.hpp:307
uint16_t msgLen1T
The type used for intermediate message length.
Definition: logDefs.hpp:52
uint32_t secT
The type used for seconds.
Definition: logDefs.hpp:27
static int createLog(bufferPtrT &logBuffer, const timespecX &ts, const typename logT::messageT &msg, const logPrioT &level)
Create a formatted log entry, filling in a buffer.
Definition: logHeader.hpp:448
static constexpr int maxHeadSize
The maximum header size.
Definition: logHeader.hpp:89
static int logLevel(bufferPtrT &logBuffer, const logPrioT &lvl)
Set the level of a log entry in a logBuffer header.
Definition: logHeader.hpp:291
static size_t totalSize(bufferPtrT &logBuffer)
Get the total size of the log entry, including the message buffer.
Definition: logHeader.hpp:429
int8_t logPrioT
The type of the log priority code.
Definition: logDefs.hpp:19
The log entry header.
Definition: logHeader.hpp:69
static int extractBasicLog(logPrioT &lvl, eventCodeT &ec, timespecX &ts, msgLenT &len, bufferPtrT &logBuffer)
Extract the basic details of a log entry.
Definition: logHeader.hpp:481
The MagAO-X logger log priority levels.
static constexpr size_t MAX_LEN0
The max value in the msgLen0 field.
Definition: logHeader.hpp:75
static timespecX timespec(bufferPtrT &logBuffer)
Extract the timespec of a log entry.
Definition: logHeader.hpp:323
std::shared_ptr< char > bufferPtrT
The log entry buffer smart pointer.
Definition: logHeader.hpp:57
static msgLen1T msgLen1(bufferPtrT &logBuffer)
Extract the medium message length of a log entry message.
Definition: logHeader.hpp:407
Type definitions for the flatlogs format.
uint32_t nanosecT
The type used for nanoseconds.
Definition: logDefs.hpp:32
static int msgLen(bufferPtrT &logBuffer, const msgLenT &msgLen)
Set the message length of a log entry message.
Definition: logHeader.hpp:376
msgLen2T msgLenT
The type used to refer to the message length, regardless of length.
Definition: logDefs.hpp:67
static constexpr int minHeadSize
The minimum header size.
Definition: logHeader.hpp:83
static constexpr size_t MAX_LEN1
The max value in the msgLen1 field.
Definition: logHeader.hpp:78
static constexpr logPrioT LOG_DEFAULT
Used to denote "use the default level for this log type".
Definition: logPriority.hpp:58
uint16_t eventCodeT
The type of an event code (16-bit unsigned int).
Definition: logDefs.hpp:38
A fixed-width timespec structure.
Definition: timespecX.hpp:31
static size_t headerSize(bufferPtrT &logBuffer)
Get the size of the header, including the variable size length field, for an existing logBuffer...
Definition: logHeader.hpp:364
uint64_t msgLen2T
The type used for long message length.
Definition: logDefs.hpp:60
static void * messageBuffer(bufferPtrT &logBuffer)
Get the message buffer address.
Definition: logHeader.hpp:442