GM6000 Digital Heater Controller Branch: main
SDX-1330
ModelDatabase.h
Go to the documentation of this file.
1#ifndef Cpl_Dm_ModelDatabase_h_
2#define Cpl_Dm_ModelDatabase_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#include "colony_config.h"
17#include "Cpl/System/Mutex.h"
18#include "Cpl/Container/SList.h"
19#include "Cpl/Json/Arduino.h"
20
21
22/** This symbol defines the size, in bytes, of a single/global JSON document
23 buffer that is used for the toJSON() and fromJSON() operations. Only one
24 instance of this buffer is allocated.
25*/
26#ifndef OPTION_CPL_DM_MODEL_DATABASE_MAX_CAPACITY_JSON_DOC
27#define OPTION_CPL_DM_MODEL_DATABASE_MAX_CAPACITY_JSON_DOC (1024*2)
28#endif
29
30
31///
32namespace Cpl {
33///
34namespace Dm {
35
36
37/** This concrete class implements a simple Model Database.
38
39 Note: The model point instances are contained in SList. This is to reduce
40 the RAM overhead on each Model Point instance (e.g. a MapItem has 32
41 bytes of overhead vs. 8 bytes for Item). This means that looking up
42 a model point by name is not efficient. However, the typically use
43 case for the look-up function is when reading/writing model point
44 via the TShell command shell the performance impact should not be
45 noticeable.
46
47 The SList is sorted alphabetically on the first call to getFirstByName().
48
49 Note: All of the methods are thread safe. However, since the Model Database
50 instances are typically created statically and statically allocated
51 Model point self-register there is race condition on when the
52 Model Database instance's Mutex is created/initialize. As a result
53 mutex is dynamically created/initialized on its first usage.
54 */
56{
57public:
58 /// Constructor
59 ModelDatabase() noexcept;
60
61 /** This is a special constructor for when the Model Database is
62 statically declared (i.e. it is initialized as part of
63 C++ startup BEFORE main() is executed. C/C++ does NOT guarantee
64 the order of static initializers. Since the Model Database contains
65 a list of model points - and one or more of those model points can
66 be statically created/initialized - we need to do something to ensure
67 my internal list is properly initialized for this scenario - hence this
68 constructor.
69 */
70 ModelDatabase( const char* ignoreThisParameter_usedToCreateAUniqueConstructor ) noexcept;
71
72 /// Destructor
73 ~ModelDatabase() noexcept;
74
75public:
76 /// See Cpl::Dm::ModelDatabaseApi
77 ModelPoint* lookupModelPoint( const char* modelPointName ) noexcept;
78
79 /// See Cpl::Dm::ModelDatabaseApi
81
82 /// See Cpl::Dm::ModelDatabaseApi
83 ModelPoint* getNextByName( ModelPoint& currentModelPoint ) noexcept;
84
85 /// See Cpl::Dm::ModelDatabaseApi
86 bool fromJSON( const char* src, Cpl::Text::String* errorMsg=0, ModelPoint** retMp = 0, uint16_t* retSequenceNumber=0 ) noexcept;
87
88public:
89 /** This method has 'PACKAGE Scope' in that is should only be called by
90 other classes in the Cpl::Dm namespace. It is ONLY public to avoid
91 the tight coupling of C++ friend mechanism.
92
93 This method inserts a new Model Point into the Model Database.
94 */
95 void insert_( ModelPoint& mpToAdd ) noexcept;
96
97 /** This method has 'PACKAGE Scope' in that is should only be called by
98 other classes in the Cpl::Dm namespace. It is ONLY public to avoid
99 the tight coupling of C++ friend mechanism.
100
101 This method locks the Model Database. For every call to lock() there must
102 be corresponding call to unlock();
103 */
104 void lock_() noexcept;
105
106 /** This method has 'PACKAGE Scope' in that is should only be called by
107 other classes in the Cpl::Dm namespace. It is ONLY public to avoid
108 the tight coupling of C++ friend mechanism.
109
110 This method unlocks the Model Database.
111 */
112 void unlock_() noexcept;
113
114
115
116public:
117 /** This method has 'PACKAGE Scope' in that is should only be called by
118 other classes in the Cpl::Dm namespace. It is ONLY public to avoid
119 the tight coupling of C++ friend mechanism.
120
121 This method provides a single global lock for ALL Model Database
122 instances. The method is used to protect global Model Database (e.g.
123 the global parse buffer).
124
125 This method locks the global Model Database lock. For every call to
126 globalLock_() there must be corresponding call to globalUnlock_();
127 */
128 static void globalLock_() noexcept;
129
130 /** This method has 'PACKAGE Scope' in that is should only be called by
131 other classes in the Cpl::Dm namespace. It is ONLY public to avoid
132 the tight coupling of C++ friend mechanism.
133
134 This method unlocks the global Model Database lock
135 */
136 static void globalUnlock_() noexcept;
137
138 /** This variable has 'PACKAGE Scope' in that is should only be called by
139 other classes in the Cpl::Dm namespace. It is ONLY public to avoid
140 the tight coupling of C++ friend mechanism.
141
142 Global/single instance of a JSON document. Model Point's need to have
143 acquired the global lock before using this buffer
144 */
146
147
148protected:
149 /// Helper method to create the database lock
151
152 /// Helper method to sort the model point list alphabetical
153 virtual void sortList() noexcept;
154
155 /// Helper method to find a point by name
156 virtual ModelPoint* find( const char* name ) noexcept;
157
158protected:
159 /// Map to the store the Model Points
160 Cpl::Container::SList<ModelPoint> m_list;
161
162 /// Mutex for making the Database thread safe
163 Cpl::System::Mutex* m_lock;
164
165 /// Keep track if the point list has beed sorted
167
168private:
169 /// Prevent access to the copy constructor -->Model Databases can not be copied!
170 ModelDatabase( const ModelDatabase& m );
171
172 /// Prevent access to the assignment operator -->Model Databases can not be copied!
173 const ModelDatabase& operator=( const ModelDatabase& m );
174};
175
176
177}; // end namespaces
178};
179#endif // end header latch
#define OPTION_CPL_DM_MODEL_DATABASE_MAX_CAPACITY_JSON_DOC
This symbol defines the size, in bytes, of a single/global JSON document buffer that is used for the ...
Definition ModelDatabase.h:27
This class defines the basic operations that can be performed on a Model Base.
Definition ModelDatabaseApi.h:32
This concrete class implements a simple Model Database.
Definition ModelDatabase.h:56
ModelPoint * getFirstByName() noexcept
See Cpl::Dm::ModelDatabaseApi.
static StaticJsonDocument< OPTION_CPL_DM_MODEL_DATABASE_MAX_CAPACITY_JSON_DOC > g_doc_
This variable has 'PACKAGE Scope' in that is should only be called by other classes in the Cpl::Dm na...
Definition ModelDatabase.h:145
virtual ModelPoint * find(const char *name) noexcept
Helper method to find a point by name.
ModelDatabase() noexcept
Constructor.
bool m_listSorted
Keep track if the point list has beed sorted.
Definition ModelDatabase.h:166
ModelPoint * lookupModelPoint(const char *modelPointName) noexcept
See Cpl::Dm::ModelDatabaseApi.
ModelPoint * getNextByName(ModelPoint &currentModelPoint) noexcept
See Cpl::Dm::ModelDatabaseApi.
Cpl::System::Mutex * m_lock
Mutex for making the Database thread safe.
Definition ModelDatabase.h:163
bool createLock()
Helper method to create the database lock.
virtual void sortList() noexcept
Helper method to sort the model point list alphabetical.
static void globalLock_() noexcept
This method has 'PACKAGE Scope' in that is should only be called by other classes in the Cpl::Dm name...
Cpl::Container::SList< ModelPoint > m_list
Map to the store the Model Points.
Definition ModelDatabase.h:160
static void globalUnlock_() noexcept
This method has 'PACKAGE Scope' in that is should only be called by other classes in the Cpl::Dm name...
void lock_() noexcept
This method has 'PACKAGE Scope' in that is should only be called by other classes in the Cpl::Dm name...
void insert_(ModelPoint &mpToAdd) 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...
bool fromJSON(const char *src, Cpl::Text::String *errorMsg=0, ModelPoint **retMp=0, uint16_t *retSequenceNumber=0) noexcept
See Cpl::Dm::ModelDatabaseApi.
This mostly abstract class defines the interface for a Model Point.
Definition ModelPoint.h:46
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20