GM6000 Digital Heater Controller Branch: main
SDX-1330
ModelPointCommon_.h
Go to the documentation of this file.
1#ifndef Cpl_Dm_ModelPointBase_h_
2#define Cpl_Dm_ModelPointBase_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
16#include "Cpl/Dm/ModelPoint.h"
18#include "Cpl/Dm/Subscriber.h"
19#include "Cpl/Container/DList.h"
20#include <stdint.h>
21#include <stdlib.h>
22
23///
24namespace Cpl {
25///
26namespace Dm {
27
28
29/** This concrete class provide common infrastructure for a Model Point.
30 */
32{
33protected:
34 /// Constructor
36 const char* symbolicName,
37 void* myDataPtr,
38 size_t dataSizeInBytes,
39 bool isValid = false );
40
41public:
42 /// See Cpl::Dm::ModelPoint
43 const char* getName() const noexcept;
44
45 /// See Cpl::Dm::ModelPoint. This method IS thread safe.
46 size_t getSize() const noexcept;
47
48 /// See Cpl::Dm::ModelPoint
49 uint16_t getSequenceNumber() const noexcept;
50
51 /// See Cpl::Dm::ModelPoint
52 uint16_t touch() noexcept;
53
54 /// See Cpl::Dm::ModelPoint
55 uint16_t setInvalid( LockRequest_T lockRequest = eNO_REQUEST ) noexcept;
56
57 /// See Cpl::Dm::ModelPoint
58 bool isNotValid( uint16_t* seqNumPtr=0 ) const noexcept;
59
60 /// See Cpl::Dm::ModelPoint
61 bool isLocked() const noexcept;
62
63 /// See Cpl::Dm::ModelPoint
64 uint16_t setLockState( LockRequest_T lockRequest ) noexcept;
65
66 /// See Cpl::Dm::ModelPoint
67 size_t getExternalSize( bool includeLockedState = false ) const noexcept;
68
69 /// See Cpl::Dm::ModelPoint. Note: The implementation does NOT account for Endianess, i.e. assumes the 'platform' is the same for export/import
70 size_t exportData( void* dstDataStream, size_t maxDstLength, uint16_t* retSequenceNumber = 0, bool includeLockedState = false ) const noexcept;
71
72 /// See Cpl::Dm::ModelPoint. Note: The implementation does NOT account for Endianess, i.e. assumes the 'platform' is the same for export/import
73 size_t importData( const void* srcDataStream, size_t srcLength, uint16_t* retSequenceNumber = 0, bool includeLockedState = false ) noexcept;
74
75 /// See Cpl::Dm::ModelPoint
76 void genericAttach( SubscriberApi& observer, uint16_t initialSeqNumber=SEQUENCE_NUMBER_UNKNOWN ) noexcept;
77
78 /// See Cpl::Dm::ModelPoint
79 void genericDetach( SubscriberApi& observer ) noexcept;
80
81 /// See Cpl::Dm::ModelPoint
82 bool toJSON( char* dst, size_t dstSize, bool& truncated, bool verbose=true, bool pretty=false ) noexcept;
83
84protected:
85 /** This method is used to read the MP contents and synchronize
86 the observer with the current MP contents. This method should ONLY be
87 used in the notification callback method and the 'observerToSync'
88 argument MUST be the argument provided by the callback method
89
90 Note: The observer will be subscribed for change notifications after
91 this call.
92 */
93 inline bool readAndSync( void* dstData, size_t dstSize, SubscriberApi& observerToSync )
94 {
95 uint16_t seqNum;
96 bool result = readData( dstData, dstSize , &seqNum );
97 attachSubscriber( observerToSync, seqNum );
98 return result;
99 }
100
101public:
102 /** This method is used to test the validate state of the MP and synchronize
103 the observer with the current MP contents. This method should ONLY be
104 used in the notification callback method and the 'observerToSync'
105 argument MUST be the argument provided by the callback method
106
107 Note: The observer will be subscribed for change notifications after
108 this call
109 */
110 inline bool isNotValidAndSync( SubscriberApi& observerToSync )
111 {
112 uint16_t seqNum;
113 bool result = isNotValid( &seqNum );
114 attachSubscriber( observerToSync, seqNum );
115 return result;
116 }
117
118
119protected:
120 /// See Cpl::Dm::ModelPoint
121 bool readData( void* dstData, size_t dstSize, uint16_t* seqNumPtr=0 ) const noexcept;
122
123 /// See Cpl::Dm::ModelPoint
124 uint16_t writeData( const void* srcData, size_t srcSize, LockRequest_T lockRequest = eNO_REQUEST ) noexcept;
125
126 /// Updates the MP with the valid-state/data from 'src'. Note: the src.lock state is NOT copied
127 virtual uint16_t copyDataAndStateFrom( const ModelPointCommon_& src, LockRequest_T lockRequest ) noexcept;
128
129 /// See Cpl::Dm::ModelPoint
130 void attachSubscriber( SubscriberApi& observer, uint16_t initialSeqNumber=SEQUENCE_NUMBER_UNKNOWN ) noexcept;
131
132 /// See Cpl::Dm::ModelPoint
133 void detachSubscriber( SubscriberApi& observer ) noexcept;
134
135 /// See Cpl::Dm::ModelPoint
136 void copyDataTo_( void* dstData, size_t dstSize ) const noexcept;
137
138 /// See Cpl::Dm::ModelPoint
139 void copyDataFrom_( const void* srcData, size_t srcSize ) noexcept;
140
141
142 /// See Cpl::Dm::ModelPoint. Note: This implementation does NOT work if the any of the data content are floats/double data types
143 bool isDataEqual_( const void* otherData ) const noexcept;
144
145 /// See Cpl::Dm::ModelPoint.
146 const void* getImportExportDataPointer_() const noexcept;
147
148 /// See Cpl::Dm::ModelPoint.
149 size_t getInternalDataSize_() const noexcept;
150
151public:
152 /// See Cpl::Dm::ModelPoint
153 void processSubscriptionEvent_( SubscriberApi& subscriber, Event_T event ) noexcept;
154
155
156protected:
157 /** Internal helper method that completes the data update process as well
158 as ensuring any change notifications get generated. Note: This method
159 ALWAYS sets the MP's state to 'valid'
160
161 This method is NOT thread safe.
162 */
163 virtual void processDataUpdated() noexcept;
164
165 /** Internal helper method that handles generating change notifications
166 when the Model Point's data/state has changed.
167
168 This method is NOT thread safe.
169 */
170 virtual void processChangeNotifications() noexcept;
171
172 /** Internal helper method that advances/updates the Model Point's
173 sequence number.
174
175 This method is NOT thread safe.
176 */
177 virtual void advanceSequenceNumber() noexcept;
178
179 /** Internal helper method that manages testing and updating the locked
180 state.
181
182 Rules:
183 1) If 'lockRequest' is eNO_REQUEST, the method only returns true if
184 the MP is in the unlocked state
185 2) If 'lockRequest' is eLOCK, the method only returns if the MP is in
186 the unlocked state. In addition, when true is returned the MP is
187 put into the locked state.
188 3) If 'lockRequest' is eUNLOCK, the method always returns true and
189 the MP is left in the unlocked state.
190
191 This method is NOT thread safe.
192 */
193 virtual bool testAndUpdateLock( LockRequest_T lockRequest ) noexcept;
194
195
196protected:
197 /// Helper FSM method
198 virtual void transitionToNotifyPending( SubscriberApi& subscriber ) noexcept;
199
200 /// Helper FSM method
201 virtual void transitionToSubscribed( SubscriberApi& subscriber ) noexcept;
202
203 /// Helper method when converting MP to a JSON string
204 virtual JsonDocument& beginJSON( bool isValid, bool locked, uint16_t seqnum, bool verbose=true ) noexcept;
205
206 /// Helper method when converting MP to a JSON string
207 virtual void endJSON( char* dst, size_t dstSize, bool& truncated, bool verbose=true, bool pretty=false ) noexcept;
208
209 /** Helper method that a child a class can override to change behavior when
210 an MP is set to the invalid state. The default behavior is to zero out
211 the data (i.e. perform a memset(m_dataPtr,0, m_dataSize) call on the data)
212 */
213 virtual void hookSetInvalid() noexcept;
214
215
216protected:
217 /// List of Active Subscribers
218 Cpl::Container::DList<SubscriberApi> m_subscribers;
219
220 /// The model point's symbolic name
221 const char* m_name;
222
223 /// Reference to the containing Model Base
225
226 /// Reference to my Data
228
229 /// Size of my data
231
232 /// Sequence number used for tracking changes in the Point data
233 uint16_t m_seqNum;
234
235 /// Locked state
237
238 /// valid/invalid state
240};
241
242}; // end namespaces
243};
244#endif // end header latch
This concrete class implements a simple Model Database.
Definition ModelDatabase.h:56
This concrete class provide common infrastructure for a Model Point.
Definition ModelPointCommon_.h:32
ModelPointCommon_(ModelDatabase &myModelBase, const char *symbolicName, void *myDataPtr, size_t dataSizeInBytes, bool isValid=false)
Constructor.
void attachSubscriber(SubscriberApi &observer, uint16_t initialSeqNumber=SEQUENCE_NUMBER_UNKNOWN) noexcept
See Cpl::Dm::ModelPoint.
void copyDataTo_(void *dstData, size_t dstSize) const noexcept
See Cpl::Dm::ModelPoint.
bool m_valid
valid/invalid state
Definition ModelPointCommon_.h:239
size_t getExternalSize(bool includeLockedState=false) const noexcept
See Cpl::Dm::ModelPoint.
uint16_t setInvalid(LockRequest_T lockRequest=eNO_REQUEST) noexcept
See Cpl::Dm::ModelPoint.
Cpl::Container::DList< SubscriberApi > m_subscribers
List of Active Subscribers.
Definition ModelPointCommon_.h:218
virtual JsonDocument & beginJSON(bool isValid, bool locked, uint16_t seqnum, bool verbose=true) noexcept
Helper method when converting MP to a JSON string.
virtual void endJSON(char *dst, size_t dstSize, bool &truncated, bool verbose=true, bool pretty=false) noexcept
Helper method when converting MP to a JSON string.
virtual void transitionToNotifyPending(SubscriberApi &subscriber) noexcept
Helper FSM method.
void genericAttach(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
const char * m_name
The model point's symbolic name.
Definition ModelPointCommon_.h:221
virtual bool testAndUpdateLock(LockRequest_T lockRequest) noexcept
Internal helper method that manages testing and updating the locked state.
bool isLocked() const noexcept
See Cpl::Dm::ModelPoint.
ModelDatabase & m_modelDatabase
Reference to the containing Model Base.
Definition ModelPointCommon_.h:224
bool isNotValid(uint16_t *seqNumPtr=0) const noexcept
See Cpl::Dm::ModelPoint.
void copyDataFrom_(const void *srcData, size_t srcSize) noexcept
See Cpl::Dm::ModelPoint.
virtual void advanceSequenceNumber() noexcept
Internal helper method that advances/updates the Model Point's sequence number.
bool isNotValidAndSync(SubscriberApi &observerToSync)
This method is used to test the validate state of the MP and synchronize the observer with the curren...
Definition ModelPointCommon_.h:110
uint16_t getSequenceNumber() const noexcept
See Cpl::Dm::ModelPoint.
void genericDetach(SubscriberApi &observer) noexcept
See Cpl::Dm::ModelPoint.
virtual void processChangeNotifications() noexcept
Internal helper method that handles generating change notifications when the Model Point's data/state...
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.
uint16_t m_seqNum
Sequence number used for tracking changes in the Point data.
Definition ModelPointCommon_.h:233
uint16_t touch() noexcept
See Cpl::Dm::ModelPoint.
const char * getName() const noexcept
See Cpl::Dm::ModelPoint.
virtual void transitionToSubscribed(SubscriberApi &subscriber) noexcept
Helper FSM method.
uint16_t setLockState(LockRequest_T lockRequest) noexcept
See Cpl::Dm::ModelPoint.
size_t m_dataSize
Size of my data.
Definition ModelPointCommon_.h:230
const void * getImportExportDataPointer_() const noexcept
See Cpl::Dm::ModelPoint.
size_t getInternalDataSize_() const noexcept
See Cpl::Dm::ModelPoint.
size_t getSize() const noexcept
See Cpl::Dm::ModelPoint. This method IS thread safe.
virtual void hookSetInvalid() noexcept
Helper method that a child a class can override to change behavior when an MP is set to the invalid s...
bool toJSON(char *dst, size_t dstSize, bool &truncated, bool verbose=true, bool pretty=false) noexcept
See Cpl::Dm::ModelPoint.
void processSubscriptionEvent_(SubscriberApi &subscriber, Event_T event) noexcept
See Cpl::Dm::ModelPoint.
bool isDataEqual_(const void *otherData) const noexcept
See Cpl::Dm::ModelPoint. Note: This implementation does NOT work if the any of the data content are f...
bool readData(void *dstData, size_t dstSize, uint16_t *seqNumPtr=0) const noexcept
See Cpl::Dm::ModelPoint.
size_t exportData(void *dstDataStream, size_t maxDstLength, uint16_t *retSequenceNumber=0, bool includeLockedState=false) const noexcept
See Cpl::Dm::ModelPoint. Note: The implementation does NOT account for Endianess, i....
bool m_locked
Locked state.
Definition ModelPointCommon_.h:236
uint16_t writeData(const void *srcData, size_t srcSize, LockRequest_T lockRequest=eNO_REQUEST) noexcept
See Cpl::Dm::ModelPoint.
void * m_dataPtr
Reference to my Data.
Definition ModelPointCommon_.h:227
size_t importData(const void *srcDataStream, size_t srcLength, uint16_t *retSequenceNumber=0, bool includeLockedState=false) noexcept
See Cpl::Dm::ModelPoint. Note: The implementation does NOT account for Endianess, i....
virtual void processDataUpdated() noexcept
Internal helper method that completes the data update process as well as ensuring any change notifica...
This mostly abstract class defines the interface for a Model Point.
Definition ModelPoint.h:46
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
Event_T
Subscriber events.
Definition ModelPoint.h:365
This abstract class defines the Subscriber interface - for change notifications - to a Model Points d...
Definition SubscriberApi.h:34
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20