LSST Applications  21.0.0-172-gfb10e10a+18fedfabac,22.0.0+297cba6710,22.0.0+80564b0ff1,22.0.0+8d77f4f51a,22.0.0+a28f4c53b1,22.0.0+dcf3732eb2,22.0.1-1-g7d6de66+2a20fdde0d,22.0.1-1-g8e32f31+297cba6710,22.0.1-1-geca5380+7fa3b7d9b6,22.0.1-12-g44dc1dc+2a20fdde0d,22.0.1-15-g6a90155+515f58c32b,22.0.1-16-g9282f48+790f5f2caa,22.0.1-2-g92698f7+dcf3732eb2,22.0.1-2-ga9b0f51+7fa3b7d9b6,22.0.1-2-gd1925c9+bf4f0e694f,22.0.1-24-g1ad7a390+a9625a72a8,22.0.1-25-g5bf6245+3ad8ecd50b,22.0.1-25-gb120d7b+8b5510f75f,22.0.1-27-g97737f7+2a20fdde0d,22.0.1-32-gf62ce7b1+aa4237961e,22.0.1-4-g0b3f228+2a20fdde0d,22.0.1-4-g243d05b+871c1b8305,22.0.1-4-g3a563be+32dcf1063f,22.0.1-4-g44f2e3d+9e4ab0f4fa,22.0.1-42-gca6935d93+ba5e5ca3eb,22.0.1-5-g15c806e+85460ae5f3,22.0.1-5-g58711c4+611d128589,22.0.1-5-g75bb458+99c117b92f,22.0.1-6-g1c63a23+7fa3b7d9b6,22.0.1-6-g50866e6+84ff5a128b,22.0.1-6-g8d3140d+720564cf76,22.0.1-6-gd805d02+cc5644f571,22.0.1-8-ge5750ce+85460ae5f3,master-g6e05de7fdc+babf819c66,master-g99da0e417a+8d77f4f51a,w.2021.48
LSST Data Management Base Package
MaskDict.cc
Go to the documentation of this file.
1 /*
2  * Developed for the LSST Data Management System.
3  * This product includes software developed by the LSST Project
4  * (https://www.lsst.org).
5  * See the COPYRIGHT file at the top-level directory of this distribution
6  * for details of code ownership.
7  *
8  * This program is free software: you can redistribute it and/or modify
9  * it under the terms of the GNU General Public License as published by
10  * the Free Software Foundation, either version 3 of the License, or
11  * (at your option) any later version.
12  *
13  * This program is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16  * GNU General Public License for more details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with this program. If not, see <https://www.gnu.org/licenses/>.
20  */
21 
22 #include <mutex>
23 #include <set>
24 
25 #include "boost/functional/hash.hpp"
27 
28 namespace lsst { namespace afw { namespace image { namespace detail {
29 
30 // A thread-safe singleton that manages MaskDict's global state.
32 public:
33 
34  static GlobalState & get() {
35  static GlobalState instance;
36  return instance;
37  }
38 
41  if (!mpd.empty()) {
42  std::shared_ptr<MaskDict> dict(new MaskDict(mpd));
43  _allMaskDicts.insert(dict);
44  return dict;
45  }
46  return _defaultMaskDict;
47  }
48 
52  _allMaskDicts.insert(result);
53  return result;
54  }
55 
57  return _defaultMaskDict;
58  }
59 
61  _defaultMaskDict = std::move(dict);
62  }
63 
66  _defaultMaskDict = copy(*_defaultMaskDict);
67  return _defaultMaskDict;
68  }
69 
70  template <typename Functor>
71  void forEachMaskDict(Functor functor) {
73  _prune(); // guarantees dereference below is safe
74  for (auto const & ptr : _allMaskDicts) {
75  functor(*ptr.lock());
76  }
77  }
78 
79 private:
80 
81  GlobalState() : _defaultMaskDict(new MaskDict()) {
82  _allMaskDicts.insert(_defaultMaskDict);
83  _defaultMaskDict->_addInitialMaskPlanes();
84  }
85 
86  GlobalState(GlobalState const &) = delete;
87  GlobalState(GlobalState &&) = delete;
88 
89  GlobalState & operator=(GlobalState const &) = delete;
90  GlobalState & operator=(GlobalState &&) = delete;
91 
92  ~GlobalState() = default;
93 
94  // Prune expired weak_ptrs from _allMaskDicts. Not thread safe; should
95  // only be called by routines that have already locked the mutex.
96  void _prune() {
97  for (auto iter = _allMaskDicts.begin(); iter != _allMaskDicts.end(); ) {
98  if (iter->expired()) {
99  iter = _allMaskDicts.erase(iter);
100  } else {
101  ++iter;
102  }
103  }
104  }
105 
106  std::recursive_mutex _mutex; // guards _allMaskDicts and synchronizes updates to it and _defaultMaskDict
108  std::shared_ptr<MaskDict> _defaultMaskDict;
109 };
110 
112  return GlobalState::get().copyOrGetDefault(mpd);
113 }
114 
116  return GlobalState::get().getDefault();
117 }
118 
121 }
122 
124  return GlobalState::get().detachDefault();
125 }
126 
127 void MaskDict::addAllMasksPlane(std::string const & name, int bitId) {
129  [&name, bitId](MaskDict & dict) {
130  auto const found = std::find_if(
131  dict.begin(), dict.end(),
132  [bitId](auto const & item) { return item.second == bitId; }
133  );
134  if (found == dict.end()) {
135  // is name already in use?
136  if (dict.find(name) == dict.end()) {
137  dict.add(name, bitId);
138  }
139  }
140  }
141  );
142 }
143 
144 MaskDict::~MaskDict() noexcept = default;
145 
146 std::shared_ptr<MaskDict> MaskDict::clone() const {
147  return GlobalState::get().copy(*this);
148 }
149 
151  if (empty()) {
152  return 0;
153  }
154 
155  auto const maxIter = std::max_element(
156  begin(), end(),
157  [](auto const & a, auto const & b) { return a.second < b.second; }
158  );
159  assert(maxIter != end());
160  int id = maxIter->second + 1; // The maskPlane to use if there are no gaps
161 
162  for (int i = 0; i < id; ++i) {
163  // is i already used in this Mask?
164  auto const sameIter = std::find_if(
165  begin(), end(),
166  [i](auto const & item) { return item.second == i; }
167  );
168  if (sameIter == end()) { // Not used; so we'll use it
169  return i;
170  }
171  }
172 
173  return id;
174 }
175 
177  auto iter = find(name);
178  return (iter == end()) ? -1 : iter->second;
179 }
180 
181 void MaskDict::print() const {
182  for (auto const & item : *this) {
183  std::cout << "Plane " << item.second << " -> " << item.first << std::endl;
184  }
185 }
186 
187 bool MaskDict::operator==(MaskDict const& rhs) const {
188  return this == &rhs || (_hash == rhs._hash && _dict == rhs._dict);
189 }
190 
191 void MaskDict::add(std::string const & name, int bitId) {
192  _dict[name] = bitId;
193  _hash = boost::hash<MaskPlaneDict>()(_dict);
194 }
195 
197  _dict.erase(name);
198  _hash = boost::hash<MaskPlaneDict>()(_dict);
199 }
200 
202  _dict.clear();
203  _hash = 0x0;
204 }
205 
206 void MaskDict::_addInitialMaskPlanes() {
207  int i = -1;
208  _dict["BAD"] = ++i;
209  _dict["SAT"] = ++i;
210  _dict["INTRP"] = ++i;
211  _dict["CR"] = ++i;
212  _dict["EDGE"] = ++i;
213  _dict["DETECTED"] = ++i;
214  _dict["DETECTED_NEGATIVE"] = ++i;
215  _dict["SUSPECT"] = ++i;
216  _dict["NO_DATA"] = ++i;
217  _hash = boost::hash<MaskPlaneDict>()(_dict);
218 }
219 
220 MaskDict::MaskDict() : _dict(), _hash(0x0) {}
221 
222 MaskDict::MaskDict(MaskPlaneDict const & dict) : _dict(dict), _hash(boost::hash<MaskPlaneDict>()(_dict)) {}
223 
224 }}}} // lsst::afw::image::detail
py::object result
Definition: _schema.cc:429
table::Key< std::string > name
Definition: Amplifier.cc:116
table::Key< int > id
Definition: Detector.cc:162
uint64_t * ptr
Definition: RangeSet.cc:88
table::Key< int > b
table::Key< int > a
std::shared_ptr< MaskDict > detachDefault()
Definition: MaskDict.cc:64
std::shared_ptr< MaskDict > getDefault() const noexcept
Definition: MaskDict.cc:56
std::shared_ptr< MaskDict > copy(MaskDict const &dict)
Definition: MaskDict.cc:49
std::shared_ptr< MaskDict > copyOrGetDefault(MaskPlaneDict const &mpd)
Definition: MaskDict.cc:39
void setDefault(std::shared_ptr< MaskDict > dict)
Definition: MaskDict.cc:60
int getMaskPlane(std::string const &name) const
Definition: MaskDict.cc:176
const_iterator begin() const noexcept
Definition: MaskDict.h:131
static void addAllMasksPlane(std::string const &name, int bitId)
Definition: MaskDict.cc:127
static std::shared_ptr< MaskDict > detachDefault()
Definition: MaskDict.cc:123
const_iterator end() const noexcept
Definition: MaskDict.h:132
static std::shared_ptr< MaskDict > getDefault()
Definition: MaskDict.cc:115
static void setDefault(std::shared_ptr< MaskDict > dict)
Definition: MaskDict.cc:119
bool operator==(MaskDict const &rhs) const
Definition: MaskDict.cc:187
const_iterator find(std::string const &name) const
Definition: MaskDict.h:135
void add(std::string const &name, int bitId)
Definition: MaskDict.cc:191
static std::shared_ptr< MaskDict > copyOrGetDefault(MaskPlaneDict const &dict)
Definition: MaskDict.cc:111
bool empty() const noexcept
Definition: MaskDict.h:141
void erase(std::string const &name)
Definition: MaskDict.cc:196
T clear(T... args)
T empty(T... args)
T endl(T... args)
T erase(T... args)
T find_if(T... args)
T max_element(T... args)
T move(T... args)
Definition: Polygon.cc:24
std::map< std::string, int > MaskPlaneDict
Definition: Mask.h:58
Backwards-compatibility support for depersisting the old Calib (FluxMag0/FluxMag0Err) objects.
class[[deprecated("Removed with no replacement (but see lsst::afw::image::TransmissionCurve). Will be " "removed after v22.")]] FilterProperty final
Describe the properties of a Filter (e.g.
Definition: Filter.h:53
A base class for image defects.
STL namespace.