GM6000 Digital Heater Controller Branch: main
SDX-1330
Numeric.h
Go to the documentation of this file.
1#ifndef Cpl_Dm_Mp_Numeric_h_
2#define Cpl_Dm_Mp_Numeric_h_
3/*-----------------------------------------------------------------------------
4* This file is part of the Colony.Core Project. The Colony.Core Project is an
5* open source project with a BSD type of licensing agreement. See the license
6* agreement (license.txt) in the top/ directory or on the Internet at
7* http://integerfox.com/colony.core/license.txt
8*
9* Copyright (c) 2014-2022 John T. Taylor
10*
11* Redistributions of the source code must retain the above copyright notice.
12*----------------------------------------------------------------------------*/
13/** @file */
14
15
17#include "Cpl/System/Assert.h"
19#include "Cpl/Text/atob.h"
20#include "Cpl/Text/format.h"
21#include "Cpl/Text/FString.h"
22#include <string.h>
23#include <stdint.h>
24
25/// Hack to get around that NOT all compilers support the "%llx" notation for printf
26#if INTPTR_MAX == INT32_MAX
27/// print format max integer
28#define PRINTF_SIZET_FMT "%lx"
29/// type for max integer
30#define PRINTF_SIZET_TYPE unsigned long
31
32#elif INTPTR_MAX == INT64_MAX
33/// print format max integer
34#define PRINTF_SIZET_FMT "%llx"
35/// print format max integer
36#define PRINTF_SIZET_TYPE unsigned long long
37#else
38#error "Environment not 32 or 64-bit."
39#endif
40
41/// Endianess of a Bit array. For little endian set to true; else set to false
42#ifndef OPTION_CPL_DM_MP_BITARRAY_IS_LITTLE_ENDIAN
43#define OPTION_CPL_DM_MP_BITARRAY_IS_LITTLE_ENDIAN true
44#endif
45
46
47///
48namespace Cpl {
49///
50namespace Dm {
51///
52namespace Mp {
53
54
55/** This template class provides a mostly concrete implementation for a Model
56 Point who's data is a C numeric primitive type of type: 'ELEMTYPE'.
57
58 NOTES:
59 1) All methods in this class are NOT thread Safe unless explicitly
60 documented otherwise.
61 */
62template<class ELEMTYPE, class MPTYPE>
64{
65protected:
66 /// The element's value
67 ELEMTYPE m_data;
68
69protected:
70 /// Constructor: Invalid MP
71 Numeric( Cpl::Dm::ModelDatabase& myModelBase, const char* symbolicName )
72 :Cpl::Dm::ModelPointCommon_( myModelBase, symbolicName, &m_data, sizeof( m_data ), false )
73 {
74 }
75
76 /// Constructor: Valid MP (requires initial value)
77 Numeric( Cpl::Dm::ModelDatabase& myModelBase, const char* symbolicName, ELEMTYPE initialValue )
78 :Cpl::Dm::ModelPointCommon_( myModelBase, symbolicName, &m_data, sizeof( m_data ), true )
79 {
80 m_data = initialValue;
81 }
82
83public:
84 /// Type safe read. See Cpl::Dm::ModelPoint
85 inline bool read( ELEMTYPE& dstData, uint16_t* seqNumPtr = 0 ) const noexcept
86 {
87 return Cpl::Dm::ModelPointCommon_::readData( &dstData, sizeof( ELEMTYPE ), seqNumPtr );
88 }
89
90 /// Type safe write. See Cpl::Dm::ModelPoint
91 inline uint16_t write( ELEMTYPE newValue, Cpl::Dm::ModelPoint::LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
92 {
93 return Cpl::Dm::ModelPointCommon_::writeData( &newValue, sizeof( ELEMTYPE ), lockRequest );
94 }
95
96 /// Atomic increment
97 inline uint16_t increment( ELEMTYPE incSize = 1, Cpl::Dm::ModelPoint::LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
98 {
100 uint16_t result = write( m_data + incSize, lockRequest );
102 return result;
103 }
104
105 /// Atomic decrement
106 inline uint16_t decrement( ELEMTYPE decSize = 1, Cpl::Dm::ModelPoint::LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
107 {
109 uint16_t result = write( m_data - decSize, lockRequest );
111 return result;
112 }
113
114 /// Updates the MP with the valid-state/data from 'src'. Note: the src.lock state is NOT copied
115 inline uint16_t copyFrom( const MPTYPE& src, LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
116 {
117 return copyDataAndStateFrom( src, lockRequest );
118 }
119
120 /// Type safe register observer
121 inline void attach( Cpl::Dm::Subscriber<MPTYPE>& observer, uint16_t initialSeqNumber = SEQUENCE_NUMBER_UNKNOWN ) noexcept
122 {
123 attachSubscriber( observer, initialSeqNumber );
124 }
125
126 /// Type safe un-register observer
127 inline void detach( Cpl::Dm::Subscriber<MPTYPE>& observer ) noexcept
128 {
129 detachSubscriber( observer );
130 }
131
132 /// See Cpl::Dm::ModelPointCommon
133 inline bool readAndSync( ELEMTYPE& dstData, SubscriberApi& observerToSync )
134 {
135 return ModelPointCommon_::readAndSync( &dstData, sizeof( ELEMTYPE ), observerToSync );
136 }
137
138protected:
139 /// See Cpl::Dm::Point.
140 void setJSONVal( JsonDocument& doc ) noexcept
141 {
142 doc["val"] = m_data;
143 }
144
145public:
146 /// See Cpl::Dm::Point.
147 bool fromJSON_( JsonVariant& src, Cpl::Dm::ModelPoint::LockRequest_T lockRequest, uint16_t& retSequenceNumber, Cpl::Text::String* errorMsg ) noexcept
148 {
149 if ( src.is<ELEMTYPE>() )
150 {
151 retSequenceNumber = write( src.as<ELEMTYPE>(), lockRequest );
152 return true;
153 }
154 if ( errorMsg )
155 {
156 *errorMsg = "Invalid syntax for the 'val' key/value pair";
157 }
158 return false;
159 }
160};
161
162
163/** This template class extends the Numeric<> class to provide bit operation
164 on the numeric value. The datatype of the numeric MUST be an integer
165 type.
166
167 The underlying storage of the bit array is N bit integers. A side effect of
168 this storage mechanism the bit ordering in the JSON 'val' string is dependent on the
169 target platform's Endian architecture.
170
171 The toJSON()/fromJSON format is:
172 \code
173
174 { name:"<mpname>", type:"<mptypestring>", valid:true|false, seqnum:nnnn, locked:true|false, val:"<bits>" }
175
176 where <bits> is a string of N digits ('1' or '0') where the left most digit is
177 is the MSb of byte[0] and the right most digit is the LSb of byte[N].
178 Whether byte[0] is the MSB or LSB is dependent on the big/little Endian
179 architecture of the target platform.
180
181 For example a 16bit Array (as binary hex: dataword[0]=0x30, dataword[1]=0x09)
182
183 val:"0011000000001001"
184
185 \endcode
186
187 NOTE: All methods in this class ARE thread Safe unless explicitly
188 documented otherwise.
189
190 */
191template <class WORDSIZE, class MPTYPE>
192class BitArray_ : public Numeric<WORDSIZE, MPTYPE>
193{
194protected:
195 /// Constructor. Invalid MP.
196 BitArray_( Cpl::Dm::ModelDatabase& myModelBase, const char* symbolicName )
197 : Numeric<WORDSIZE, MPTYPE>( myModelBase, symbolicName )
198 {
199 }
200
201 /// Constructor. Valid MP. Requires an initial value
202 BitArray_( Cpl::Dm::ModelDatabase& myModelBase, const char* symbolicName, WORDSIZE initialValue )
203 : Numeric<WORDSIZE, MPTYPE>( myModelBase, symbolicName, initialValue )
204 {
205 }
206
207public:
208 /// Atomic operation to set the zero indexed bit to a 1.
209 inline uint16_t setBit( uint8_t bitPosition, Cpl::Dm::ModelPoint::LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
210 {
212 uint16_t result = Numeric<WORDSIZE, MPTYPE>::write( Numeric<WORDSIZE, MPTYPE>::m_data | ( 1 << bitPosition ), lockRequest );
214 return result;
215 }
216
217 /// Atomic operation to set the zero indexed bit to a 0.
218 inline uint16_t clearBit( uint8_t bitPosition, Cpl::Dm::ModelPoint::LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
219 {
221 uint16_t result = Numeric<WORDSIZE, MPTYPE>::write( Numeric<WORDSIZE, MPTYPE>::m_data & ( ~( 1 << bitPosition ) ), lockRequest );
223 return result;
224 }
225
226 /// Atomic operation to toggle the zero indexed bit.
227 inline uint16_t flipBit( uint8_t bitPosition, Cpl::Dm::ModelPoint::LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
228 {
230 uint16_t result = Numeric<WORDSIZE, MPTYPE>::write( Numeric<WORDSIZE, MPTYPE>::m_data ^ ( 1 << bitPosition ), lockRequest );
232 return result;
233 }
234
235
236public:
237 /// Atomic operation to clear ONLY the bits as specified by the bit mask.
238 inline uint16_t clearBitsByMask( uint16_t bitMask, Cpl::Dm::ModelPoint::LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
239 {
241 uint16_t result = Numeric<WORDSIZE, MPTYPE>::write( Numeric<WORDSIZE, MPTYPE>::m_data & ~( bitMask ), lockRequest );
243 return result;
244 }
245
246 /// Atomic operation to set the bits specified by the bit mask
247 inline uint16_t setBitsByMask( uint16_t bitMask, Cpl::Dm::ModelPoint::LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
248 {
250 uint16_t result = Numeric<WORDSIZE, MPTYPE>::write( Numeric<WORDSIZE, MPTYPE>::m_data | bitMask, lockRequest );
252 return result;
253 }
254
255 /// Atomic operation to flip/toggle ONLY the bits as specified the bit mask
256 inline uint16_t flipBitsByMask( uint16_t bitMask, Cpl::Dm::ModelPoint::LockRequest_T lockRequest = Cpl::Dm::ModelPoint::eNO_REQUEST ) noexcept
257 {
259 uint16_t result = Numeric<WORDSIZE, MPTYPE>::write( Numeric<WORDSIZE, MPTYPE>::m_data ^ bitMask, lockRequest );
261 return result;
262 }
263
264
265
266protected:
267 /// See Cpl::Dm::Point.
268 void setJSONVal( JsonDocument& doc ) noexcept
269 {
271 const void* dataPtr = &(Numeric<WORDSIZE, MPTYPE>::m_data);
273 doc["val"] = (char*) tmp.getString();
274 }
275
276
277public:
278 /// See Cpl::Dm::Point.
279 bool fromJSON_( JsonVariant& src, Cpl::Dm::ModelPoint::LockRequest_T lockRequest, uint16_t& retSequenceNumber, Cpl::Text::String* errorMsg ) noexcept
280 {
281 if ( src.is<const char*>() )
282 {
283 const char* val = src.as<const char*>();
284 uint16_t value = 0;
285 if ( Cpl::Text::asciiBinaryToBuffer( &value, val, sizeof( value ), OPTION_CPL_DM_MP_BITARRAY_IS_LITTLE_ENDIAN ) > 0 )
286 {
287 retSequenceNumber = Numeric<WORDSIZE, MPTYPE>::write( value, lockRequest );
288 return true;
289 }
290 }
291
292 if ( errorMsg )
293 {
294 *errorMsg = "Invalid syntax for the 'val' key/value pair";
295 }
296 return false;
297 }
298};
299
300
301/** This template class extends the implementation of Numeric<> class to support
302 the pointers instead of integers
303
304 NOTES:
305 1) All methods in this class are NOT thread Safe unless explicitly
306 documented otherwise.
307 */
308template <class MPTYPE>
309class Pointer_ : public Numeric<size_t, MPTYPE>
310{
311protected:
312 /// Constructor. Invalid MP.
313 Pointer_( Cpl::Dm::ModelDatabase& myModelBase, const char* symbolicName )
314 : Numeric<size_t, MPTYPE>( myModelBase, symbolicName )
315 {
316 }
317
318 /// Constructor. Valid MP. Requires an initial value
319 Pointer_( Cpl::Dm::ModelDatabase& myModelBase, const char* symbolicName, void* initialValue )
320 : Numeric<size_t, MPTYPE>( myModelBase, symbolicName, (size_t) initialValue )
321 {
322 }
323
324public:
325 /// See Cpl::Dm::Point.
326 void setJSONVal( JsonDocument& doc ) noexcept
327 {
330 doc["val"] = (char*) tmp.getString();
331 }
332
333 /// See Cpl::Dm::Point.
334 bool fromJSON_( JsonVariant& src, Cpl::Dm::ModelPoint::LockRequest_T lockRequest, uint16_t& retSequenceNumber, Cpl::Text::String* errorMsg ) noexcept
335 {
336 if ( src.is<const char*>() )
337 {
338 const char* val = src.as<const char*>();
339 unsigned long long value = 0;
340 if ( Cpl::Text::a2ull( value, val, 16 ) )
341 {
342 retSequenceNumber = Numeric<size_t, MPTYPE>::write( (size_t) value, lockRequest );
343 return true;
344 }
345 }
346
347 if ( errorMsg )
348 {
349 *errorMsg = "Invalid syntax for the 'val' key/value pair";
350 }
351 return false;
352 }
353};
354
355}; // end namespaces
356};
357};
358#endif // end header latch
#define PRINTF_SIZET_FMT
Hack to get around that NOT all compilers support the "%llx" notation for printf.
Definition Numeric.h:28
#define PRINTF_SIZET_TYPE
type for max integer
Definition Numeric.h:30
#define OPTION_CPL_DM_MP_BITARRAY_IS_LITTLE_ENDIAN
Endianess of a Bit array. For little endian set to true; else set to false.
Definition Numeric.h:43
This file contains a collection of methods that wrap the standard C library functions for converting ...
This concrete class implements a simple Model Database.
Definition ModelDatabase.h:56
void lock_() noexcept
This method has 'PACKAGE Scope' in that is should only be called by other classes in the Cpl::Dm name...
void unlock_() noexcept
This method has 'PACKAGE Scope' in that is should only be called by other classes in the Cpl::Dm name...
This concrete class provide common infrastructure for a Model Point.
Definition ModelPointCommon_.h:32
void attachSubscriber(SubscriberApi &observer, uint16_t initialSeqNumber=SEQUENCE_NUMBER_UNKNOWN) noexcept
See Cpl::Dm::ModelPoint.
void detachSubscriber(SubscriberApi &observer) noexcept
See Cpl::Dm::ModelPoint.
bool readAndSync(void *dstData, size_t dstSize, SubscriberApi &observerToSync)
This method is used to read the MP contents and synchronize the observer with the current MP contents...
Definition ModelPointCommon_.h:93
ModelDatabase & m_modelDatabase
Reference to the containing Model Base.
Definition ModelPointCommon_.h:224
virtual uint16_t copyDataAndStateFrom(const ModelPointCommon_ &src, LockRequest_T lockRequest) noexcept
Updates the MP with the valid-state/data from 'src'. Note: the src.lock state is NOT copied.
bool readData(void *dstData, size_t dstSize, uint16_t *seqNumPtr=0) const noexcept
See Cpl::Dm::ModelPoint.
uint16_t writeData(const void *srcData, size_t srcSize, LockRequest_T lockRequest=eNO_REQUEST) noexcept
See Cpl::Dm::ModelPoint.
LockRequest_T
Options related to the Model Point's locked state.
Definition ModelPoint.h:50
@ eNO_REQUEST
No change in the MP's lock state is requested.
Definition ModelPoint.h:51
static const uint16_t SEQUENCE_NUMBER_UNKNOWN
Magic value to use when registering for a change notification and application does not 'know' the cur...
Definition ModelPoint.h:62
This template class extends the Numeric<> class to provide bit operation on the numeric value.
Definition Numeric.h:193
bool fromJSON_(JsonVariant &src, Cpl::Dm::ModelPoint::LockRequest_T lockRequest, uint16_t &retSequenceNumber, Cpl::Text::String *errorMsg) noexcept
See Cpl::Dm::Point.
Definition Numeric.h:279
uint16_t setBit(uint8_t bitPosition, Cpl::Dm::ModelPoint::LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Atomic operation to set the zero indexed bit to a 1.
Definition Numeric.h:209
uint16_t flipBit(uint8_t bitPosition, Cpl::Dm::ModelPoint::LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Atomic operation to toggle the zero indexed bit.
Definition Numeric.h:227
uint16_t clearBitsByMask(uint16_t bitMask, Cpl::Dm::ModelPoint::LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Atomic operation to clear ONLY the bits as specified by the bit mask.
Definition Numeric.h:238
BitArray_(Cpl::Dm::ModelDatabase &myModelBase, const char *symbolicName, WORDSIZE initialValue)
Constructor. Valid MP. Requires an initial value.
Definition Numeric.h:202
uint16_t clearBit(uint8_t bitPosition, Cpl::Dm::ModelPoint::LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Atomic operation to set the zero indexed bit to a 0.
Definition Numeric.h:218
void setJSONVal(JsonDocument &doc) noexcept
See Cpl::Dm::Point.
Definition Numeric.h:268
BitArray_(Cpl::Dm::ModelDatabase &myModelBase, const char *symbolicName)
Constructor. Invalid MP.
Definition Numeric.h:196
uint16_t flipBitsByMask(uint16_t bitMask, Cpl::Dm::ModelPoint::LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Atomic operation to flip/toggle ONLY the bits as specified the bit mask.
Definition Numeric.h:256
uint16_t setBitsByMask(uint16_t bitMask, Cpl::Dm::ModelPoint::LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Atomic operation to set the bits specified by the bit mask.
Definition Numeric.h:247
This template class provides a mostly concrete implementation for a Model Point who's data is a C num...
Definition Numeric.h:64
uint16_t write(ELEMTYPE newValue, Cpl::Dm::ModelPoint::LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Type safe write. See Cpl::Dm::ModelPoint.
Definition Numeric.h:91
bool fromJSON_(JsonVariant &src, Cpl::Dm::ModelPoint::LockRequest_T lockRequest, uint16_t &retSequenceNumber, Cpl::Text::String *errorMsg) noexcept
See Cpl::Dm::Point.
Definition Numeric.h:147
bool read(ELEMTYPE &dstData, uint16_t *seqNumPtr=0) const noexcept
Type safe read. See Cpl::Dm::ModelPoint.
Definition Numeric.h:85
bool readAndSync(ELEMTYPE &dstData, SubscriberApi &observerToSync)
See Cpl::Dm::ModelPointCommon.
Definition Numeric.h:133
Numeric(Cpl::Dm::ModelDatabase &myModelBase, const char *symbolicName)
Constructor: Invalid MP.
Definition Numeric.h:71
uint16_t decrement(ELEMTYPE decSize=1, Cpl::Dm::ModelPoint::LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Atomic decrement.
Definition Numeric.h:106
Numeric(Cpl::Dm::ModelDatabase &myModelBase, const char *symbolicName, ELEMTYPE initialValue)
Constructor: Valid MP (requires initial value)
Definition Numeric.h:77
uint16_t copyFrom(const MPTYPE &src, LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Updates the MP with the valid-state/data from 'src'. Note: the src.lock state is NOT copied.
Definition Numeric.h:115
void setJSONVal(JsonDocument &doc) noexcept
See Cpl::Dm::Point.
Definition Numeric.h:140
ELEMTYPE m_data
The element's value.
Definition Numeric.h:67
uint16_t increment(ELEMTYPE incSize=1, Cpl::Dm::ModelPoint::LockRequest_T lockRequest=Cpl::Dm::ModelPoint::eNO_REQUEST) noexcept
Atomic increment.
Definition Numeric.h:97
void attach(Cpl::Dm::Subscriber< MPTYPE > &observer, uint16_t initialSeqNumber=SEQUENCE_NUMBER_UNKNOWN) noexcept
Type safe register observer.
Definition Numeric.h:121
void detach(Cpl::Dm::Subscriber< MPTYPE > &observer) noexcept
Type safe un-register observer.
Definition Numeric.h:127
This template class extends the implementation of Numeric<> class to support the pointers instead of ...
Definition Numeric.h:310
Pointer_(Cpl::Dm::ModelDatabase &myModelBase, const char *symbolicName, void *initialValue)
Constructor. Valid MP. Requires an initial value.
Definition Numeric.h:319
bool fromJSON_(JsonVariant &src, Cpl::Dm::ModelPoint::LockRequest_T lockRequest, uint16_t &retSequenceNumber, Cpl::Text::String *errorMsg) noexcept
See Cpl::Dm::Point.
Definition Numeric.h:334
void setJSONVal(JsonDocument &doc) noexcept
See Cpl::Dm::Point.
Definition Numeric.h:326
Pointer_(Cpl::Dm::ModelDatabase &myModelBase, const char *symbolicName)
Constructor. Invalid MP.
Definition Numeric.h:313
This abstract class defines the Subscriber interface - for change notifications - to a Model Points d...
Definition SubscriberApi.h:34
This template class defines a type safe Subscriber.
Definition Subscriber.h:82
This template class represents a NULL terminated string of a specific length.
Definition FString.h:38
const char * getString() const
See Cpl::Text::String.
void format(const char *format,...)
See Cpl::Text::String.
This abstract class defines the operations that can be before on a NULL terminated string.
Definition String.h:40
This file contains some general purpose string formatting functions.
long asciiBinaryToBuffer(void *dstBinary, const char *srcString, size_t dstMaxLen, bool reverse=false)
This method will convert an 'ASCII BINARY' string to an equivalent binary buffer, i....
bool a2ull(unsigned long long &convertedValue, const char *string, int base=10, const char *validStopChars=0, const char **endptr=0)
This method is the same as a2i() except that it converts unsigned long long integer.
bool bufferToAsciiBinary(const void *binaryData, int len, Cpl::Text::String &destString, bool appendToString=false, bool reverse=false)
This method converts the binary buffer to a single string that is ASCII BINARY.
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20