Line data Source code
1 : /****************************************************************************/
2 : // Eclipse SUMO, Simulation of Urban MObility; see https://eclipse.dev/sumo
3 : // Copyright (C) 2001-2026 German Aerospace Center (DLR) and others.
4 : // This program and the accompanying materials are made available under the
5 : // terms of the Eclipse Public License 2.0 which is available at
6 : // https://www.eclipse.org/legal/epl-2.0/
7 : // This Source Code may also be made available under the following Secondary
8 : // Licenses when the conditions for such availability set forth in the Eclipse
9 : // Public License 2.0 are satisfied: GNU General Public License, version 2
10 : // or later which is available at
11 : // https://www.gnu.org/licenses/old-licenses/gpl-2.0-standalone.html
12 : // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-or-later
13 : /****************************************************************************/
14 : /// @file MSDetectorControl.cpp
15 : /// @author Daniel Krajzewicz
16 : /// @author Jakob Erdmann
17 : /// @author Clemens Honomichl
18 : /// @author Sascha Krieg
19 : /// @author Michael Behrisch
20 : /// @author Laura Bieker
21 : /// @date 2005-09-15
22 : ///
23 : // Detectors container; responsible for string and output generation
24 : /****************************************************************************/
25 : #include <config.h>
26 :
27 : #include <iostream>
28 : #include <utils/options/OptionsCont.h>
29 : #include <utils/options/Option.h>
30 : #include <utils/common/MsgHandler.h>
31 : #include "MSMeanData_Emissions.h"
32 : #include "MSMeanData_Net.h"
33 : #include "MSDetectorControl.h"
34 :
35 :
36 : // ===========================================================================
37 : // member method definitions
38 : // ===========================================================================
39 42501 : MSDetectorControl::MSDetectorControl() {
40 42501 : }
41 :
42 :
43 41881 : MSDetectorControl::~MSDetectorControl() {
44 53971 : for (auto i = myDetectors.begin(); i != myDetectors.end(); ++i) {
45 12090 : (*i).second.clear();
46 : }
47 64677 : for (auto item : myMeanData) {
48 46031 : for (MSMeanData* md : item.second) {
49 23235 : delete md;
50 : }
51 : }
52 : myMeanData.clear();
53 41881 : }
54 :
55 :
56 : void
57 40367 : MSDetectorControl::close(SUMOTime step) {
58 : // flush the last values
59 40367 : writeOutput(step, true);
60 : // [...] files are closed on another place [...]
61 : myIntervals.clear();
62 40367 : }
63 :
64 :
65 : void
66 35174 : MSDetectorControl::add(SumoXMLTag type, MSDetectorFileOutput* d, const std::string& device, SUMOTime interval, SUMOTime begin) {
67 35174 : if (!myDetectors[type].add(d->getID(), d)) {
68 : const std::string id = d->getID();
69 4 : delete d;
70 12 : throw ProcessError(toString(type) + " detector '" + id + "' could not be built (declared twice?).");
71 : }
72 35170 : addDetectorAndInterval(d, &OutputDevice::getDevice(device), interval, begin);
73 35156 : }
74 :
75 :
76 : void
77 3175 : MSDetectorControl::add(SumoXMLTag type, MSDetectorFileOutput* d) {
78 3175 : if (!myDetectors[type].add(d->getID(), d)) {
79 : const std::string id = d->getID();
80 0 : delete d;
81 0 : throw ProcessError(toString(type) + " detector '" + id + "' could not be built (declared twice?).");
82 : }
83 3175 : }
84 :
85 :
86 : void
87 23290 : MSDetectorControl::add(MSMeanData* md, const std::string& device,
88 : SUMOTime frequency, SUMOTime begin) {
89 23290 : myMeanData[md->getID()].push_back(md);
90 23290 : addDetectorAndInterval(md, &OutputDevice::getDevice(device), frequency, begin);
91 46580 : if (begin <= string2time(OptionsCont::getOptions().getString("begin"))) {
92 22647 : md->init();
93 : }
94 23290 : MSGlobals::gHaveEmissions |= typeid(*md) == typeid(MSMeanData_Emissions);
95 23290 : }
96 :
97 :
98 : const std::vector<SumoXMLTag>
99 8507 : MSDetectorControl::getAvailableTypes() const {
100 : std::vector<SumoXMLTag> result;
101 9189 : for (std::map<SumoXMLTag, NamedObjectCont<MSDetectorFileOutput*> >::const_iterator i = myDetectors.begin(); i != myDetectors.end(); ++i) {
102 682 : result.push_back(i->first);
103 : }
104 8507 : return result;
105 0 : }
106 :
107 :
108 : const NamedObjectCont<MSDetectorFileOutput*>&
109 11455218 : MSDetectorControl::getTypedDetectors(SumoXMLTag type) const {
110 11455218 : if (myDetectors.find(type) == myDetectors.end()) {
111 116 : return myEmptyContainer;
112 : }
113 11455102 : return myDetectors.find(type)->second;
114 : }
115 :
116 :
117 : void
118 64914559 : MSDetectorControl::updateDetectors(const SUMOTime step) {
119 76165266 : for (const auto& i : myDetectors) {
120 57597882 : for (const auto& j : getTypedDetectors(i.first)) {
121 46347175 : j.second->detectorUpdate(step);
122 : }
123 : }
124 90357554 : for (auto item : myMeanData) {
125 51792327 : for (MSMeanData* md : item.second) {
126 26349323 : md->detectorUpdate(step);
127 : }
128 : }
129 64914550 : }
130 :
131 :
132 : void
133 64954911 : MSDetectorControl::writeOutput(SUMOTime step, bool closing) {
134 88824307 : for (Intervals::iterator i = myIntervals.begin(); i != myIntervals.end(); ++i) {
135 23869397 : IntervalsKey interval = (*i).first;
136 23869397 : if (myLastCalls[interval] + interval.first <= step || (closing && myLastCalls[interval] < step)) {
137 392383 : DetectorFileVec dfVec = (*i).second;
138 392383 : SUMOTime startTime = myLastCalls[interval];
139 : // check whether at the end the output was already generated
140 1496335 : for (DetectorFileVec::iterator it = dfVec.begin(); it != dfVec.end(); ++it) {
141 1103953 : MSDetectorFileOutput* det = it->first;
142 1103953 : det->writeXMLOutput(*(it->second), startTime, step);
143 : }
144 392382 : myLastCalls[interval] = step;
145 392383 : }
146 : }
147 64954910 : }
148 :
149 :
150 : void
151 58446 : MSDetectorControl::addDetectorAndInterval(MSDetectorFileOutput* det,
152 : OutputDevice* device,
153 : SUMOTime interval,
154 : SUMOTime begin) {
155 58446 : const SUMOTime simBegin = string2time(OptionsCont::getOptions().getString("begin"));
156 58446 : if (begin == -1) {
157 : begin = simBegin;
158 : }
159 58446 : IntervalsKey key = std::make_pair(interval, begin);
160 : Intervals::iterator it = myIntervals.find(key);
161 : // Add command for given key only once to MSEventControl...
162 58446 : if (it == myIntervals.end()) {
163 : DetectorFileVec detAndFileVec;
164 23751 : detAndFileVec.push_back(std::make_pair(det, device));
165 47502 : myIntervals.insert(std::make_pair(key, detAndFileVec));
166 : SUMOTime lastCall = begin;
167 23751 : if (begin < simBegin) {
168 18 : SUMOTime divRest = (simBegin - begin) % interval;
169 18 : lastCall = simBegin - divRest;
170 : }
171 23751 : myLastCalls[key] = lastCall;
172 23751 : } else {
173 34695 : DetectorFileVec& detAndFileVec = it->second;
174 34695 : if (find_if(detAndFileVec.begin(), detAndFileVec.end(), [&](const DetectorFilePair & pair) {
175 101006 : return pair.first == det;
176 : }) == detAndFileVec.end()) {
177 34695 : detAndFileVec.push_back(std::make_pair(det, device));
178 : } else {
179 : // detector already in container. Don't add several times
180 0 : WRITE_WARNING("MSDetectorControl::addDetectorAndInterval: detector already in container. Ignoring.");
181 0 : return;
182 : }
183 : }
184 58446 : det->writeXMLDetectorProlog(*device);
185 : }
186 :
187 : const std::map<std::string, MSMoveReminder*>&
188 179 : MSDetectorControl::getAllReminders() {
189 179 : if (myReminders.empty()) {
190 30 : for (auto item : myMeanData) {
191 2 : for (MSMeanData* md : item.second) {
192 45 : for (MSMoveReminder* rem : md->getReminders()) {
193 44 : myReminders[rem->getDescription()] = rem;
194 1 : }
195 : }
196 : }
197 29 : if (myReminders.empty()) {
198 : // ensure non-empty map
199 28 : myReminders[""] = nullptr;
200 : }
201 : }
202 179 : return myReminders;
203 : }
204 :
205 : void
206 168 : MSDetectorControl::clearState(SUMOTime step) {
207 219 : for (const auto& i : myDetectors) {
208 102 : for (const auto& j : getTypedDetectors(i.first)) {
209 51 : j.second->clearState(step);
210 : }
211 : }
212 : myReminders.clear();
213 168 : }
214 :
215 : /****************************************************************************/
|