Line data Source code
1 : /****************************************************************************/
2 : // Eclipse SUMO, Simulation of Urban MObility; see https://eclipse.dev/sumo
3 : // Copyright (C) 2001-2024 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 marouter_main.cpp
15 : /// @author Daniel Krajzewicz
16 : /// @author Jakob Erdmann
17 : /// @author Laura Bieker
18 : /// @author Michael Behrisch
19 : /// @date Thu, 06 Jun 2002
20 : ///
21 : // Main for MAROUTER
22 : /****************************************************************************/
23 : #include <config.h>
24 :
25 : #ifdef HAVE_VERSION_H
26 : #include <version.h>
27 : #endif
28 :
29 : #include <iostream>
30 : #include <string>
31 : #include <limits.h>
32 : #include <ctime>
33 : #include <vector>
34 : #include <xercesc/sax/SAXException.hpp>
35 : #include <xercesc/sax/SAXParseException.hpp>
36 : #include <utils/common/FileHelpers.h>
37 : #include <utils/common/StringUtils.h>
38 : #include <utils/common/MsgHandler.h>
39 : #include <utils/common/UtilExceptions.h>
40 : #include <utils/common/SystemFrame.h>
41 : #include <utils/common/RandHelper.h>
42 : #include <utils/common/ToString.h>
43 : #include <utils/distribution/Distribution_Points.h>
44 : #include <utils/iodevices/OutputDevice.h>
45 : #include <utils/iodevices/OutputDevice_String.h>
46 : #include <utils/options/Option.h>
47 : #include <utils/options/OptionsCont.h>
48 : #include <utils/options/OptionsIO.h>
49 : #include <utils/router/RouteCostCalculator.h>
50 : #include <utils/router/DijkstraRouter.h>
51 : #include <utils/router/AStarRouter.h>
52 : #include <utils/router/CHRouter.h>
53 : #include <utils/router/CHRouterWrapper.h>
54 : #include <utils/xml/XMLSubSys.h>
55 : #include <od/ODCell.h>
56 : #include <od/ODDistrict.h>
57 : #include <od/ODDistrictCont.h>
58 : #include <od/ODDistrictHandler.h>
59 : #include <od/ODMatrix.h>
60 : #include <router/ROEdge.h>
61 : #include <router/ROLoader.h>
62 : #include <router/RONet.h>
63 : #include <router/RORoute.h>
64 : #include <router/RORoutable.h>
65 :
66 : #include "ROMAFrame.h"
67 : #include "ROMAAssignments.h"
68 : #include "ROMAEdgeBuilder.h"
69 : #include "ROMARouteHandler.h"
70 : #include "ROMAEdge.h"
71 :
72 :
73 : // ===========================================================================
74 : // Method implementation
75 : // ===========================================================================
76 : /* -------------------------------------------------------------------------
77 : * data processing methods
78 : * ----------------------------------------------------------------------- */
79 : /**
80 : * loads the net
81 : * The net is in this meaning made up by the net itself and the dynamic
82 : * weights which may be supplied in a separate file
83 : */
84 : void
85 96 : initNet(RONet& net, ROLoader& loader, OptionsCont& oc) {
86 : // load the net
87 96 : ROMAEdgeBuilder builder;
88 96 : ROEdge::setGlobalOptions(oc.getBool("weights.interpolate"));
89 96 : loader.loadNet(net, builder);
90 : // load the weights when wished/available
91 186 : if (oc.isSet("weight-files")) {
92 0 : loader.loadWeights(net, "weight-files", oc.getString("weight-attribute"), false, oc.getBool("weights.expand"));
93 : }
94 186 : if (oc.isSet("lane-weight-files")) {
95 0 : loader.loadWeights(net, "lane-weight-files", oc.getString("weight-attribute"), true, oc.getBool("weights.expand"));
96 : }
97 96 : }
98 :
99 :
100 : double
101 32 : getTravelTime(const ROEdge* const edge, const ROVehicle* const /* veh */, double /* time */) {
102 32 : return edge->getLength() / edge->getSpeedLimit();
103 : }
104 :
105 :
106 : /**
107 : * Computes all pair shortest paths, saving them
108 : */
109 : void
110 1 : computeAllPairs(RONet& net, OptionsCont& oc) {
111 2 : OutputDevice::createDeviceByOption("all-pairs-output");
112 1 : OutputDevice& outFile = OutputDevice::getDeviceByOption("all-pairs-output");
113 : // build the router
114 : typedef DijkstraRouter<ROEdge, ROVehicle> Dijkstra;
115 2 : Dijkstra router(ROEdge::getAllEdges(), oc.getBool("ignore-errors"), &getTravelTime);
116 : ConstROEdgeVector into;
117 1 : const int numInternalEdges = net.getInternalEdgeNumber();
118 1 : const int numTotalEdges = (int)net.getEdgeNumber();
119 8 : for (int i = numInternalEdges; i < numTotalEdges; i++) {
120 : const Dijkstra::EdgeInfo& ei = router.getEdgeInfo(i);
121 7 : if (!ei.edge->isInternal()) {
122 7 : router.compute(ei.edge, nullptr, nullptr, 0, into);
123 7 : double fromEffort = router.getEffort(ei.edge, nullptr, 0);
124 56 : for (int j = numInternalEdges; j < numTotalEdges; j++) {
125 49 : double heuTT = router.getEdgeInfo(j).effort - fromEffort;
126 49 : outFile << heuTT;
127 : /*
128 : if (heuTT >
129 : ei.edge->getDistanceTo(router.getEdgeInfo(j).edge)
130 : && router.getEdgeInfo(j).traveltime != std::numeric_limits<double>::max()
131 : ) {
132 : std::cout << " heuristic failure: from=" << ei.edge->getID() << " to=" << router.getEdgeInfo(j).edge->getID()
133 : << " fromEffort=" << fromEffort << " heuTT=" << heuTT << " airDist=" << ei.edge->getDistanceTo(router.getEdgeInfo(j).edge) << "\n";
134 : }
135 : */
136 : }
137 : }
138 : }
139 2 : }
140 :
141 :
142 : /**
143 : * Computes the routes saving them
144 : */
145 : void
146 63 : computeRoutes(RONet& net, OptionsCont& oc, ODMatrix& matrix) {
147 : // build the router
148 : SUMOAbstractRouter<ROEdge, ROVehicle>* router = nullptr;
149 63 : const std::string measure = oc.getString("weight-attribute");
150 66 : const std::string routingAlgorithm = oc.getString("routing-algorithm");
151 63 : const double priorityFactor = oc.getFloat("weights.priority-factor");
152 63 : SUMOTime begin = string2time(oc.getString("begin"));
153 63 : SUMOTime end = string2time(oc.getString("end"));
154 126 : if (oc.isDefault("begin") && matrix.getBegin() >= 0) {
155 : begin = matrix.getBegin();
156 : }
157 126 : if (oc.isDefault("end")) {
158 61 : end = matrix.getEnd() >= 0 ? matrix.getEnd() : SUMOTime_MAX;
159 : }
160 63 : DijkstraRouter<ROEdge, ROVehicle>::Operation ttOp = oc.getInt("paths") > 1 ? &ROMAAssignments::getPenalizedTT : &ROEdge::getTravelTimeStatic;
161 63 : if (measure == "traveltime" && priorityFactor == 0) {
162 62 : if (routingAlgorithm == "dijkstra") {
163 118 : router = new DijkstraRouter<ROEdge, ROVehicle>(ROEdge::getAllEdges(), oc.getBool("ignore-errors"), ttOp, nullptr, false, nullptr, net.hasPermissions());
164 3 : } else if (routingAlgorithm == "astar") {
165 0 : router = new AStarRouter<ROEdge, ROVehicle, ROMapMatcher>(ROEdge::getAllEdges(), oc.getBool("ignore-errors"), ttOp, nullptr, net.hasPermissions());
166 3 : } else if (routingAlgorithm == "CH" && !net.hasPermissions()) {
167 3 : const SUMOTime weightPeriod = (oc.isSet("weight-files") ?
168 3 : string2time(oc.getString("weight-period")) :
169 : SUMOTime_MAX);
170 6 : router = new CHRouter<ROEdge, ROVehicle>(ROEdge::getAllEdges(), oc.getBool("ignore-errors"), &ROEdge::getTravelTimeStatic, SVC_IGNORING, weightPeriod, net.hasPermissions(), false);
171 0 : } else if (routingAlgorithm == "CHWrapper" || routingAlgorithm == "CH") {
172 : // use CHWrapper instead of CH if the net has permissions
173 0 : const SUMOTime weightPeriod = (oc.isSet("weight-files") ?
174 0 : string2time(oc.getString("weight-period")) :
175 : SUMOTime_MAX);
176 : router = new CHRouterWrapper<ROEdge, ROVehicle>(
177 0 : ROEdge::getAllEdges(), oc.getBool("ignore-errors"), &ROEdge::getTravelTimeStatic,
178 0 : begin, end, weightPeriod, net.hasPermissions(), oc.getInt("routing-threads"));
179 : } else {
180 0 : throw ProcessError(TLF("Unknown routing Algorithm '%'!", routingAlgorithm));
181 : }
182 : } else {
183 : DijkstraRouter<ROEdge, ROVehicle>::Operation op;
184 1 : if (measure == "traveltime") {
185 0 : if (ROEdge::initPriorityFactor(priorityFactor)) {
186 : op = &ROEdge::getTravelTimeStaticPriorityFactor;
187 : } else {
188 : op = &ROEdge::getTravelTimeStatic;
189 : }
190 1 : } else if (measure == "CO") {
191 : op = &ROEdge::getEmissionEffort<PollutantsInterface::CO>;
192 1 : } else if (measure == "CO2") {
193 : op = &ROEdge::getEmissionEffort<PollutantsInterface::CO2>;
194 1 : } else if (measure == "PMx") {
195 : op = &ROEdge::getEmissionEffort<PollutantsInterface::PM_X>;
196 1 : } else if (measure == "HC") {
197 : op = &ROEdge::getEmissionEffort<PollutantsInterface::HC>;
198 1 : } else if (measure == "NOx") {
199 : op = &ROEdge::getEmissionEffort<PollutantsInterface::NO_X>;
200 1 : } else if (measure == "fuel") {
201 : op = &ROEdge::getEmissionEffort<PollutantsInterface::FUEL>;
202 1 : } else if (measure == "electricity") {
203 : op = &ROEdge::getEmissionEffort<PollutantsInterface::ELEC>;
204 1 : } else if (measure == "noise") {
205 : op = &ROEdge::getNoiseEffort;
206 : } else {
207 : op = &ROEdge::getStoredEffort;
208 : }
209 1 : if (measure != "traveltime" && !net.hasLoadedEffort()) {
210 6 : WRITE_WARNINGF(TL("No weight data was loaded for attribute '%'."), measure);
211 : }
212 2 : router = new DijkstraRouter<ROEdge, ROVehicle>(ROEdge::getAllEdges(), oc.getBool("ignore-errors"), op, ttOp, false, nullptr, net.hasPermissions());
213 : }
214 : try {
215 : const RORouterProvider provider(router, nullptr, nullptr, nullptr);
216 : // prepare the output
217 63 : net.openOutput(oc);
218 : // process route definitions
219 124 : if (oc.isSet("timeline")) {
220 16 : matrix.applyCurve(matrix.parseTimeLine(oc.getStringVector("timeline"), oc.getBool("timeline.day-in-hours")));
221 : }
222 62 : matrix.sortByBeginTime();
223 125 : bool haveOutput = OutputDevice::createDeviceByOption("netload-output", "meandata");
224 61 : ROVehicle defaultVehicle(SUMOVehicleParameter(), nullptr, net.getVehicleTypeSecure(DEFAULT_VTYPE_ID), &net);
225 122 : ROMAAssignments a(begin, end, oc.getBool("additive-traffic"), oc.getFloat("weight-adaption"),
226 61 : oc.getInt("max-alternatives"), oc.getBool("capacities.default"), net, matrix, *router,
227 244 : haveOutput ? &OutputDevice::getDeviceByOption("netload-output") : nullptr);
228 61 : a.resetFlows();
229 : #ifdef HAVE_FOX
230 : // this is just to init the CHRouter with the default vehicle
231 61 : router->reset(&defaultVehicle);
232 61 : const int maxNumThreads = oc.getInt("routing-threads");
233 64 : while ((int)net.getThreadPool().size() < maxNumThreads) {
234 3 : new RONet::WorkerThread(net.getThreadPool(), provider);
235 : }
236 : #endif
237 61 : std::string assignMethod = oc.getString("assignment-method");
238 61 : if (assignMethod == "UE") {
239 2 : WRITE_WARNING(TL("Deterministic user equilibrium ('UE') is not implemented yet, using stochastic method ('SUE')."));
240 : assignMethod = "SUE";
241 : }
242 61 : if (assignMethod == "incremental") {
243 114 : a.incremental(oc.getInt("max-iterations"), oc.getBool("verbose"));
244 4 : } else if (assignMethod == "SUE") {
245 8 : a.sue(oc.getInt("max-iterations"), oc.getInt("max-inner-iterations"),
246 9 : oc.getInt("paths"), oc.getFloat("paths.penalty"), oc.getFloat("tolerance"), oc.getString("route-choice-method"));
247 : }
248 : // update path costs and output
249 : OutputDevice* dev = net.getRouteOutput();
250 61 : if (dev != nullptr) {
251 : std::vector<std::string> tazParamKeys;
252 120 : if (oc.isSet("taz-param")) {
253 2 : tazParamKeys = oc.getStringVector("taz-param");
254 : }
255 : std::map<SUMOTime, std::string> sortedOut;
256 : SUMOTime lastEnd = -1;
257 : int num = 0;
258 2091 : for (const ODCell* const c : matrix.getCells()) {
259 2031 : if (c->begin >= end || c->end <= begin ||
260 4062 : c->pathsVector.empty() || c->pathsVector.front()->getEdgeVector().empty()) {
261 3 : continue;
262 : }
263 2028 : if (lastEnd >= 0 && lastEnd <= c->begin) {
264 282 : for (std::map<SUMOTime, std::string>::const_iterator desc = sortedOut.begin(); desc != sortedOut.end(); ++desc) {
265 141 : dev->writePreformattedTag(desc->second);
266 : }
267 : sortedOut.clear();
268 : }
269 2028 : if (c->departures.empty()) {
270 2007 : const SUMOTime b = MAX2(begin, c->begin);
271 2007 : const SUMOTime e = MIN2(end, c->end);
272 2007 : const int numVehs = int(c->vehicleNumber * (double)(e - b) / (double)(c->end - c->begin));
273 2007 : OutputDevice_String od(1);
274 4014 : od.openTag(SUMO_TAG_FLOW).writeAttr(SUMO_ATTR_ID, oc.getString("prefix") + toString(num++));
275 6021 : od.writeAttr(SUMO_ATTR_BEGIN, time2string(b)).writeAttr(SUMO_ATTR_END, time2string(e));
276 : od.writeAttr(SUMO_ATTR_NUMBER, numVehs);
277 2007 : matrix.writeDefaultAttrs(od, oc.getBool("ignore-vehicle-type"), c);
278 2007 : od.openTag(SUMO_TAG_ROUTE_DISTRIBUTION);
279 4593 : for (RORoute* const r : c->pathsVector) {
280 2586 : r->setCosts(router->recomputeCosts(r->getEdgeVector(), &defaultVehicle, begin));
281 5172 : r->writeXMLDefinition(od, nullptr, true, true, false, false);
282 : }
283 2007 : od.closeTag();
284 2007 : od.closeTag();
285 2007 : sortedOut[c->begin] += od.getString();
286 2007 : } else {
287 42 : const bool ignoreType = oc.getBool("ignore-vehicle-type");
288 928 : for (auto deps : c->departures) {
289 907 : if (deps.first >= end || deps.first < begin) {
290 : continue;
291 : }
292 3332 : const std::string routeDistId = c->origin + "_" + c->destination + "_" + time2string(c->begin) + "_" + time2string(c->end);
293 1699 : for (const SUMOVehicleParameter& veh : deps.second) {
294 866 : OutputDevice_String od(1);
295 :
296 866 : veh.write(od, OptionsCont::getOptions(), SUMO_TAG_VEHICLE, ignoreType || veh.vtypeid == DEFAULT_VTYPE_ID ? "" : veh.vtypeid);
297 866 : od.openTag(SUMO_TAG_ROUTE_DISTRIBUTION);
298 2495 : for (RORoute* const r : c->pathsVector) {
299 1629 : r->setCosts(router->recomputeCosts(r->getEdgeVector(), &defaultVehicle, begin));
300 3258 : r->writeXMLDefinition(od, nullptr, true, true, false, false);
301 : }
302 1732 : od.closeTag();
303 866 : if (!tazParamKeys.empty()) {
304 33 : od.openTag(SUMO_TAG_PARAM).writeAttr(SUMO_ATTR_KEY, tazParamKeys[0]).writeAttr(SUMO_ATTR_VALUE, c->origin).closeTag();
305 11 : if (tazParamKeys.size() > 1) {
306 44 : od.openTag(SUMO_TAG_PARAM).writeAttr(SUMO_ATTR_KEY, tazParamKeys[1]).writeAttr(SUMO_ATTR_VALUE, c->destination).closeTag();
307 : }
308 : }
309 866 : od.closeTag();
310 866 : sortedOut[deps.first] += od.getString();
311 866 : }
312 : }
313 : }
314 2028 : if (c->end > lastEnd) {
315 : lastEnd = c->end;
316 : }
317 : }
318 931 : for (std::map<SUMOTime, std::string>::const_iterator desc = sortedOut.begin(); desc != sortedOut.end(); ++desc) {
319 871 : dev->writePreformattedTag(desc->second);
320 : }
321 : haveOutput = true;
322 60 : }
323 1 : if (!haveOutput) {
324 2 : throw ProcessError(TL("No output file given."));
325 : }
326 : // end the processing
327 60 : net.cleanup();
328 68 : } catch (ProcessError&) {
329 3 : net.cleanup();
330 3 : throw;
331 3 : }
332 60 : }
333 :
334 :
335 : /* -------------------------------------------------------------------------
336 : * main
337 : * ----------------------------------------------------------------------- */
338 : int
339 102 : main(int argc, char** argv) {
340 102 : OptionsCont& oc = OptionsCont::getOptions();
341 102 : oc.setApplicationDescription(TL("Import O/D-matrices for macroscopic traffic assignment to generate SUMO routes."));
342 204 : oc.setApplicationName("marouter", "Eclipse SUMO marouter Version " VERSION_STRING);
343 : int ret = 0;
344 : RONet* net = nullptr;
345 : try {
346 102 : XMLSubSys::init();
347 102 : ROMAFrame::fillOptions();
348 102 : OptionsIO::setArgs(argc, argv);
349 102 : OptionsIO::getOptions();
350 102 : if (oc.processMetaOptions(argc < 2)) {
351 6 : SystemFrame::close();
352 7 : return 0;
353 : }
354 96 : SystemFrame::checkOptions(oc);
355 323 : XMLSubSys::setValidation(oc.getString("xml-validation"), oc.getString("xml-validation.net"), oc.getString("xml-validation.routes"));
356 96 : MsgHandler::initOutputOptions();
357 96 : if (!ROMAFrame::checkOptions()) {
358 0 : throw ProcessError();
359 : }
360 96 : RandHelper::initRandGlobal();
361 : // load data
362 96 : ROLoader loader(oc, false, false);
363 96 : net = new RONet();
364 96 : initNet(*net, loader, oc);
365 186 : if (oc.isSet("all-pairs-output")) {
366 1 : computeAllPairs(*net, oc);
367 1 : if (net->getDistricts().empty()) {
368 1 : delete net;
369 1 : SystemFrame::close();
370 : if (ret == 0) {
371 : std::cout << "Success." << std::endl;
372 : }
373 : return ret;
374 : }
375 : }
376 92 : if (net->getDistricts().empty()) {
377 4 : WRITE_WARNING(TL("No districts loaded, will use edge ids!"));
378 : }
379 : // load districts
380 92 : ODDistrictCont districts;
381 92 : districts.makeDistricts(net->getDistricts());
382 : // load the matrix
383 92 : ODMatrix matrix(districts, oc.getFloat("scale"));
384 92 : matrix.loadMatrix(oc);
385 74 : ROMARouteHandler handler(matrix);
386 74 : matrix.loadRoutes(oc, handler);
387 74 : if (matrix.getNumLoaded() == matrix.getNumDiscarded()) {
388 6 : throw ProcessError(TL("No valid vehicles loaded."));
389 : }
390 97 : if (MsgHandler::getErrorInstance()->wasInformed() && !oc.getBool("ignore-errors")) {
391 16 : throw ProcessError(TL("Loading failed."));
392 : }
393 63 : MsgHandler::getErrorInstance()->clear();
394 252 : WRITE_MESSAGE(toString(matrix.getNumLoaded() - matrix.getNumDiscarded()) + " valid vehicles loaded (total seen: " + toString(matrix.getNumLoaded()) + ").");
395 :
396 : // build routes and parse the incremental rates if the incremental method is choosen.
397 : try {
398 63 : computeRoutes(*net, oc, matrix);
399 3 : } catch (XERCES_CPP_NAMESPACE::SAXParseException& e) {
400 0 : WRITE_ERROR(toString(e.getLineNumber()));
401 : ret = 1;
402 0 : } catch (XERCES_CPP_NAMESPACE::SAXException& e) {
403 0 : WRITE_ERROR(StringUtils::transcode(e.getMessage()));
404 : ret = 1;
405 0 : }
406 60 : if (MsgHandler::getErrorInstance()->wasInformed() || ret != 0) {
407 0 : throw ProcessError();
408 : }
409 209 : } catch (const ProcessError& e) {
410 105 : if (std::string(e.what()) != std::string("Process Error") && std::string(e.what()) != std::string("")) {
411 70 : WRITE_ERROR(e.what());
412 : }
413 35 : MsgHandler::getErrorInstance()->inform("Quitting (on error).", false);
414 : ret = 1;
415 35 : }
416 :
417 95 : delete net;
418 95 : SystemFrame::close();
419 95 : if (ret == 0) {
420 : std::cout << "Success." << std::endl;
421 : }
422 : return ret;
423 : }
424 :
425 :
426 : /****************************************************************************/
|