Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
eclipse
GitHub Repository: eclipse/sumo
Path: blob/main/src/utils/common/RandHelper.cpp
193674 views
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 RandHelper.cpp
15
/// @author Daniel Krajzewicz
16
/// @author Michael Behrisch
17
/// @date Tue, 29.05.2005
18
///
19
//
20
/****************************************************************************/
21
#include <config.h>
22
23
#include <ctime>
24
#include <utils/options/OptionsCont.h>
25
#include <utils/common/SysUtils.h>
26
#include "StdDefs.h"
27
#include "RandHelper.h"
28
29
30
// debug vehicle movement randomness (dawdling in finalizeSpeed)
31
//#define DEBUG_FLAG1
32
//#define DEBUG_RANDCALLS
33
//#define DEBUG_RANDCALLS_PARALLEL
34
35
36
// ===========================================================================
37
// static member variables
38
// ===========================================================================
39
SumoRNG RandHelper::myRandomNumberGenerator("default");
40
41
#ifdef DEBUG_RANDCALLS
42
unsigned long long int myDebugIndex(7);
43
std::string myDebugId("");
44
#endif
45
46
#ifdef DEBUG_RANDCALLS_PARALLEL
47
#include <thread>
48
#include <utils/iodevices/OutputDevice.h>
49
std::map<std::thread::id, int> threadIndices;
50
std::map<std::string, int> lastThreadIndex; // by rng
51
#endif
52
53
54
// ===========================================================================
55
// member method definitions
56
// ===========================================================================
57
58
void
59
RandHelper::insertRandOptions(OptionsCont& oc) {
60
// registers random number options
61
oc.addOptionSubTopic("Random Number");
62
63
oc.doRegister("random", new Option_Bool(false));
64
oc.addSynonyme("random", "abs-rand", true);
65
oc.addDescription("random", "Random Number", TL("Initialises the random number generator with the current system time"));
66
67
oc.doRegister("seed", new Option_Integer(23423));
68
oc.addSynonyme("seed", "srand", true);
69
oc.addDescription("seed", "Random Number", TL("Initialises the random number generator with the given value"));
70
}
71
72
73
void
74
RandHelper::initRand(SumoRNG* which, const bool random, const int seed) {
75
if (which == nullptr) {
76
which = &myRandomNumberGenerator;
77
}
78
if (random) {
79
which->setSeed((unsigned long)time(nullptr));
80
} else {
81
which->setSeed(seed);
82
}
83
}
84
85
86
void
87
RandHelper::initRandGlobal(SumoRNG* which) {
88
OptionsCont& oc = OptionsCont::getOptions();
89
initRand(which, oc.getBool("random"), oc.getInt("seed"));
90
}
91
92
int
93
RandHelper::getSeed(SumoRNG* which) {
94
if (which == nullptr) {
95
which = &myRandomNumberGenerator;
96
}
97
return which->origSeed;
98
}
99
100
double
101
RandHelper::rand(SumoRNG* rng) {
102
if (rng == nullptr) {
103
rng = &myRandomNumberGenerator;
104
}
105
const double res = double((*rng)() / 4294967296.0);
106
rng->count++;
107
#ifdef DEBUG_RANDCALLS
108
if (rng->count == myDebugIndex
109
&& (myDebugId == "" || rng->id == myDebugId)) {
110
std::cout << "DEBUG\n"; // for setting breakpoint
111
}
112
std::stringstream stream; // to reduce output interleaving from different threads
113
#ifdef DEBUG_RANDCALLS_PARALLEL
114
auto threadID = std::this_thread::get_id();
115
if (threadIndices.count(threadID) == 0) {
116
const int tmp = threadIndices.size();
117
threadIndices[threadID] = tmp;
118
}
119
int threadIndex = threadIndices[threadID];
120
auto it = lastThreadIndex.find(rng->id);
121
if ((it == lastThreadIndex.end() || it->second != threadIndex)
122
&& (myDebugId == "" || rng->id == myDebugId)) {
123
std::cout << "DEBUG rng " << rng->id << " change thread old=" << (it == lastThreadIndex.end() ? -1 : it->second) << " new=" << threadIndex << " (" << std::this_thread::get_id() << ")\n"; // for setting breakpoint
124
}
125
lastThreadIndex[rng->id] = threadIndex;
126
stream << "rng " << rng->id << " call=" << rng->count << " thread=" << threadIndex << " val=" << res << "\n";
127
OutputDevice::getDevice(rng->id) << stream.str();
128
#else
129
stream << "rng " << rng->id << " call=" << rng->count << " val=" << res << "\n";
130
std::cout << stream.str();
131
#endif
132
#endif
133
#ifdef DEBUG_FLAG1
134
if (gDebugFlag1) {
135
std::stringstream stream; // to reduce output interleaving from different threads
136
stream << "rng " << rng->id << " call=" << rng->count << " val=" << res << "\n";
137
std::cout << stream.str();
138
}
139
#endif
140
return res;
141
}
142
143
double
144
RandHelper::randNorm(double mean, double variance, SumoRNG* rng) {
145
// Polar method to avoid cosine
146
double u, q;
147
do {
148
u = rand(2.0, rng) - 1;
149
const double v = rand(2.0, rng) - 1;
150
q = u * u + v * v;
151
} while (q == 0.0 || q >= 1.0);
152
const double logRounded = ceil(log(q) * 1e14) / 1e14;
153
return mean + variance * u * sqrt(-2 * logRounded / q);
154
}
155
156
double
157
RandHelper::randExp(double rate, SumoRNG* rng) {
158
double r = rand(rng);
159
while (r == 0) {
160
r = rand(rng);
161
}
162
return -log(r) / rate;
163
}
164
165
// template<class T>
166
// void RandHelper::shuffle(const std::vector<T>& v) {
167
// std::shuffle(v.begin(), v.end(), rng);
168
// }
169
170
/****************************************************************************/
171
172