blob: 707389b5844f066e1eeb76045edfdf32fea17142 [file] [log] [blame]
Neale Ranns812ed392017-10-16 04:20:13 -07001/*
2 * Copyright (c) 2017 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16#ifndef __VOM_INST_DB_H__
17#define __VOM_INST_DB_H__
18
19#include <memory>
20#include <ostream>
21
22namespace VOM {
23/**
24 * A Database to store the unique 'singular' instances of a single object
25 * type.
26 * The instances are stored as weak pointers. So the DB does not own these
27 * objects, they are owned by object in the client_db.
28 */
29template <typename KEY, typename OBJ>
30class singular_db
31{
32public:
33 /**
34 * Constructor
35 */
36 singular_db() {}
37
38 /**
39 * Iterator
40 */
41 typedef
42 typename std::map<KEY, std::weak_ptr<OBJ>>::const_iterator const_iterator;
43
44 /**
45 * Get iterator to the beginning of the DB
46 */
47 const_iterator cbegin() { return m_map.cbegin(); }
48
49 /**
50 * Get iterator to the beginning of the DB
51 */
52 const_iterator cend() { return m_map.cend(); }
53
54 /**
55 * Find or add the object to the store.
56 * The object passed is deisred state. A new instance will be copy
57 * constructed from it. This function is templatised on the object type
58 * passed, which may be drrived from, the object type stored. this
59 * prevents slicing during the make_shared construction.
60 */
61 template <typename DERIVED>
62 std::shared_ptr<OBJ> find_or_add(const KEY& key, const DERIVED& obj)
63 {
64 auto search = m_map.find(key);
65
66 if (search == m_map.end()) {
67 std::shared_ptr<OBJ> sp = std::make_shared<DERIVED>(obj);
68
69 m_map[key] = sp;
70
71 VOM_LOG(log_level_t::DEBUG) << *sp;
72 return (sp);
73 }
74
75 return (search->second.lock());
76 }
77
78 /**
79 * Find the object to the store.
80 */
81 std::shared_ptr<OBJ> find(const KEY& key)
82 {
83 auto search = m_map.find(key);
84
85 if (search == m_map.end()) {
86 std::shared_ptr<OBJ> sp(NULL);
87
88 return (sp);
89 }
90
91 return (search->second.lock());
92 }
93
94 /**
95 * Release the object from the DB store, if it's the one we have stored
96 */
97 void release(const KEY& key, const OBJ* obj)
98 {
99 auto search = m_map.find(key);
100
101 if (search != m_map.end()) {
102 if (search->second.expired()) {
103 m_map.erase(key);
104 } else {
105 std::shared_ptr<OBJ> sp = m_map[key].lock();
106
107 if (sp.get() == obj) {
108 m_map.erase(key);
109 }
110 }
111 }
112 }
113
114 /**
115 * Find the object to the store.
116 */
117 void add(const KEY& key, std::shared_ptr<OBJ> sp) { m_map[key] = sp; }
118
119 /**
120 * Print each of the object in the DB into the stream provided
121 */
122 void dump(std::ostream& os)
123 {
124 for (auto entry : m_map) {
125 os << "key: " << entry.first << std::endl;
126 os << " " << entry.second.lock()->to_string() << std::endl;
127 }
128 }
129
130 /**
131 * Populate VPP from current state, on VPP restart
132 */
133 void replay()
134 {
135 for (auto entry : m_map) {
136 entry.second.lock()->replay();
137 }
138 }
139
140private:
141 /**
142 * the map of objects against their key
143 */
144 std::map<KEY, std::weak_ptr<OBJ>> m_map;
145};
146};
147
148/*
149 * fd.io coding-style-patch-verification: ON
150 *
151 * Local Variables:
152 * eval: (c-set-style "mozilla")
153 * End:
154 */
155
156#endif