• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

openmc-dev / openmc / 22500302709

27 Feb 2026 07:16PM UTC coverage: 81.512% (-0.3%) from 81.826%
22500302709

Pull #3830

github

web-flow
Merge 25fbb4266 into b3788f11e
Pull Request #3830: Parallelize sampling external sources and threadsafe rejection counters

17488 of 25193 branches covered (69.42%)

Branch coverage included in aggregate %.

59 of 66 new or added lines in 6 files covered. (89.39%)

841 existing lines in 44 files now uncovered.

57726 of 67081 relevant lines covered (86.05%)

44920080.48 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

78.88
/src/weight_windows.cpp
1
#include "openmc/weight_windows.h"
2

3
#include <algorithm>
4
#include <cassert>
5
#include <cmath>
6
#include <set>
7
#include <string>
8

9
#include "openmc/tensor.h"
10

11
#include "openmc/error.h"
12
#include "openmc/file_utils.h"
13
#include "openmc/hdf5_interface.h"
14
#include "openmc/mesh.h"
15
#include "openmc/message_passing.h"
16
#include "openmc/nuclide.h"
17
#include "openmc/output.h"
18
#include "openmc/particle.h"
19
#include "openmc/particle_data.h"
20
#include "openmc/physics_common.h"
21
#include "openmc/random_ray/flat_source_domain.h"
22
#include "openmc/search.h"
23
#include "openmc/settings.h"
24
#include "openmc/simulation.h"
25
#include "openmc/tallies/filter_energy.h"
26
#include "openmc/tallies/filter_mesh.h"
27
#include "openmc/tallies/filter_particle.h"
28
#include "openmc/tallies/tally.h"
29
#include "openmc/xml_interface.h"
30

31
#include <fmt/core.h>
32

33
namespace openmc {
34

35
//==============================================================================
36
// Global variables
37
//==============================================================================
38

39
namespace variance_reduction {
40

41
std::unordered_map<int32_t, int32_t> ww_map;
42
openmc::vector<unique_ptr<WeightWindows>> weight_windows;
43
openmc::vector<unique_ptr<WeightWindowsGenerator>> weight_windows_generators;
44

45
} // namespace variance_reduction
46

47
//==============================================================================
48
// WeightWindowSettings implementation
49
//==============================================================================
50

51
WeightWindows::WeightWindows(int32_t id)
244✔
52
{
53
  index_ = variance_reduction::weight_windows.size();
244✔
54
  set_id(id);
244✔
55
  set_defaults();
244✔
56
}
244✔
57

58
WeightWindows::WeightWindows(pugi::xml_node node)
105✔
59
{
60
  // Make sure required elements are present
61
  const vector<std::string> required_elems {
105✔
62
    "id", "particle_type", "lower_ww_bounds", "upper_ww_bounds"};
525!
63
  for (const auto& elem : required_elems) {
525✔
64
    if (!check_for_node(node, elem.c_str())) {
420!
UNCOV
65
      fatal_error(fmt::format("Must specify <{}> for weight windows.", elem));
×
66
    }
67
  }
68

69
  // Get weight windows ID
70
  int32_t id = std::stoi(get_node_value(node, "id"));
210✔
71
  this->set_id(id);
105✔
72

73
  // get the particle type
74
  auto particle_type_str = std::string(get_node_value(node, "particle_type"));
105✔
75
  particle_type_ = ParticleType {particle_type_str};
105✔
76

77
  // Determine associated mesh
78
  int32_t mesh_id = std::stoi(get_node_value(node, "mesh"));
210✔
79
  set_mesh(model::mesh_map.at(mesh_id));
105✔
80

81
  // energy bounds
82
  if (check_for_node(node, "energy_bounds"))
105✔
83
    energy_bounds_ = get_node_array<double>(node, "energy_bounds");
75✔
84

85
  // get the survival value - optional
86
  if (check_for_node(node, "survival_ratio")) {
105!
87
    survival_ratio_ = std::stod(get_node_value(node, "survival_ratio"));
210✔
88
    if (survival_ratio_ <= 1)
105!
UNCOV
89
      fatal_error("Survival to lower weight window ratio must bigger than 1 "
×
90
                  "and less than the upper to lower weight window ratio.");
91
  }
92

93
  // get the max lower bound ratio - optional
94
  if (check_for_node(node, "max_lower_bound_ratio")) {
105✔
95
    max_lb_ratio_ = std::stod(get_node_value(node, "max_lower_bound_ratio"));
64✔
96
    if (max_lb_ratio_ < 1.0) {
32!
97
      fatal_error("Maximum lower bound ratio must be larger than 1");
×
98
    }
99
  }
100

101
  // get the max split - optional
102
  if (check_for_node(node, "max_split")) {
105!
103
    max_split_ = std::stod(get_node_value(node, "max_split"));
210✔
104
    if (max_split_ <= 1)
105!
UNCOV
105
      fatal_error("max split must be larger than 1");
×
106
  }
107

108
  // weight cutoff - optional
109
  if (check_for_node(node, "weight_cutoff")) {
105!
110
    weight_cutoff_ = std::stod(get_node_value(node, "weight_cutoff"));
210✔
111
    if (weight_cutoff_ <= 0)
105!
UNCOV
112
      fatal_error("weight_cutoff must be larger than 0");
×
113
    if (weight_cutoff_ > 1)
105!
UNCOV
114
      fatal_error("weight_cutoff must be less than 1");
×
115
  }
116

117
  // read the lower/upper weight bounds
118
  this->set_bounds(get_node_array<double>(node, "lower_ww_bounds"),
105✔
119
    get_node_array<double>(node, "upper_ww_bounds"));
105✔
120

121
  set_defaults();
105✔
122
}
105✔
123

124
WeightWindows::~WeightWindows()
349✔
125
{
126
  variance_reduction::ww_map.erase(id());
349✔
127
}
1,047✔
128

129
WeightWindows* WeightWindows::create(int32_t id)
90✔
130
{
131
  variance_reduction::weight_windows.push_back(make_unique<WeightWindows>());
90✔
132
  auto wws = variance_reduction::weight_windows.back().get();
90✔
133
  variance_reduction::ww_map[wws->id()] =
90✔
134
    variance_reduction::weight_windows.size() - 1;
90✔
135
  return wws;
90✔
136
}
137

138
WeightWindows* WeightWindows::from_hdf5(
11✔
139
  hid_t wws_group, const std::string& group_name)
140
{
141
  // collect ID from the name of this group
142
  hid_t ww_group = open_group(wws_group, group_name);
11✔
143

144
  auto wws = WeightWindows::create();
11✔
145

146
  std::string particle_type;
11✔
147
  read_dataset(ww_group, "particle_type", particle_type);
11✔
148
  wws->particle_type_ = ParticleType {particle_type};
11✔
149

150
  read_dataset<double>(ww_group, "energy_bounds", wws->energy_bounds_);
11✔
151

152
  int32_t mesh_id;
11✔
153
  read_dataset(ww_group, "mesh", mesh_id);
11✔
154

155
  if (model::mesh_map.count(mesh_id) == 0) {
11!
UNCOV
156
    fatal_error(
×
UNCOV
157
      fmt::format("Mesh {} used in weight windows does not exist.", mesh_id));
×
158
  }
159
  wws->set_mesh(model::mesh_map[mesh_id]);
11✔
160

161
  wws->lower_ww_ =
11✔
162
    tensor::Tensor<double>({static_cast<size_t>(wws->bounds_size()[0]),
11✔
163
      static_cast<size_t>(wws->bounds_size()[1])});
11✔
164
  wws->upper_ww_ =
11✔
165
    tensor::Tensor<double>({static_cast<size_t>(wws->bounds_size()[0]),
11✔
166
      static_cast<size_t>(wws->bounds_size()[1])});
11✔
167

168
  read_dataset<double>(ww_group, "lower_ww_bounds", wws->lower_ww_);
11✔
169
  read_dataset<double>(ww_group, "upper_ww_bounds", wws->upper_ww_);
11✔
170
  read_dataset(ww_group, "survival_ratio", wws->survival_ratio_);
11✔
171
  read_dataset(ww_group, "max_lower_bound_ratio", wws->max_lb_ratio_);
11✔
172
  read_dataset(ww_group, "max_split", wws->max_split_);
11✔
173
  read_dataset(ww_group, "weight_cutoff", wws->weight_cutoff_);
11✔
174

175
  close_group(ww_group);
11✔
176

177
  return wws;
11✔
178
}
11✔
179

180
void WeightWindows::set_defaults()
428✔
181
{
182
  // set energy bounds to the min/max energy supported by the data
183
  if (energy_bounds_.size() == 0) {
428✔
184
    int p_type = particle_type_.transport_index();
274✔
185
    if (p_type == C_NONE) {
274!
UNCOV
186
      fatal_error("Weight windows particle is not supported for transport.");
×
187
    }
188
    energy_bounds_.push_back(data::energy_min[p_type]);
274✔
189
    energy_bounds_.push_back(data::energy_max[p_type]);
274✔
190
  }
191
}
428✔
192

193
void WeightWindows::allocate_ww_bounds()
560✔
194
{
195
  auto shape = bounds_size();
560✔
196
  if (shape[0] * shape[1] == 0) {
560!
UNCOV
197
    auto msg = fmt::format(
×
UNCOV
198
      "Size of weight window bounds is zero for WeightWindows {}", id());
×
199
    warning(msg);
×
UNCOV
200
  }
×
201
  lower_ww_ = tensor::Tensor<double>(
560✔
202
    {static_cast<size_t>(shape[0]), static_cast<size_t>(shape[1])});
560✔
203
  lower_ww_.fill(-1);
560✔
204
  upper_ww_ = tensor::Tensor<double>(
560✔
205
    {static_cast<size_t>(shape[0]), static_cast<size_t>(shape[1])});
560✔
206
  upper_ww_.fill(-1);
560✔
207
}
560✔
208

209
void WeightWindows::set_id(int32_t id)
503✔
210
{
211
  assert(id >= 0 || id == C_NONE);
503!
212

213
  // Clear entry in mesh map in case one was already assigned
214
  if (id_ != C_NONE) {
503!
215
    variance_reduction::ww_map.erase(id_);
503✔
216
    id_ = C_NONE;
503✔
217
  }
218

219
  // Ensure no other mesh has the same ID
220
  if (variance_reduction::ww_map.find(id) != variance_reduction::ww_map.end()) {
503!
UNCOV
221
    throw std::runtime_error {
×
UNCOV
222
      fmt::format("Two weight windows have the same ID: {}", id)};
×
223
  }
224

225
  // If no ID is specified, auto-assign the next ID in the sequence
226
  if (id == C_NONE) {
503✔
227
    id = 0;
244✔
228
    for (const auto& m : variance_reduction::weight_windows) {
266✔
229
      id = std::max(id, m->id_);
44!
230
    }
231
    ++id;
244✔
232
  }
233

234
  // Update ID and entry in the mesh map
235
  id_ = id;
503✔
236
  variance_reduction::ww_map[id] = index_;
503✔
237
}
503✔
238

239
void WeightWindows::set_energy_bounds(span<const double> bounds)
211✔
240
{
241
  energy_bounds_.clear();
211!
242
  energy_bounds_.insert(energy_bounds_.begin(), bounds.begin(), bounds.end());
211✔
243
  // if the mesh is set, allocate space for weight window bounds
244
  if (mesh_idx_ != C_NONE)
211!
245
    allocate_ww_bounds();
211✔
246
}
211✔
247

248
void WeightWindows::set_particle_type(ParticleType p_type)
255✔
249
{
250
  if (!p_type.is_neutron() && !p_type.is_photon())
255!
UNCOV
251
    fatal_error(fmt::format(
×
UNCOV
252
      "Particle type '{}' cannot be applied to weight windows.", p_type.str()));
×
253
  particle_type_ = p_type;
255✔
254
}
255✔
255

256
void WeightWindows::set_mesh(int32_t mesh_idx)
349✔
257
{
258
  if (mesh_idx < 0 || mesh_idx >= model::meshes.size())
349!
259
    fatal_error(fmt::format("Could not find a mesh for index {}", mesh_idx));
×
260

261
  mesh_idx_ = mesh_idx;
349✔
262
  model::meshes[mesh_idx_]->prepare_for_point_location();
349✔
263
  allocate_ww_bounds();
349✔
264
}
349✔
265

UNCOV
266
void WeightWindows::set_mesh(const std::unique_ptr<Mesh>& mesh)
×
267
{
UNCOV
268
  set_mesh(mesh.get());
×
UNCOV
269
}
×
270

UNCOV
271
void WeightWindows::set_mesh(const Mesh* mesh)
×
272
{
UNCOV
273
  set_mesh(model::mesh_map[mesh->id_]);
×
UNCOV
274
}
×
275

276
std::pair<bool, WeightWindow> WeightWindows::get_weight_window(
95,059,237✔
277
  const Particle& p) const
278
{
279
  // check for particle type
280
  if (particle_type_ != p.type()) {
95,059,237✔
281
    return {false, {}};
19,992,074✔
282
  }
283

284
  // particle energy
285
  double E = p.E();
75,067,163✔
286

287
  // check to make sure energy is in range, expects sorted energy values
288
  if (E < energy_bounds_.front() || E > energy_bounds_.back())
75,067,163!
289
    return {false, {}};
92,818✔
290

291
  // Get mesh index for particle's position
292
  const auto& mesh = this->mesh();
74,974,345✔
293
  int mesh_bin = mesh->get_bin(p.r());
74,974,345✔
294

295
  // particle is outside the weight window mesh
296
  if (mesh_bin < 0)
74,974,345✔
297
    return {false, {}};
55,556✔
298

299
  // get the mesh bin in energy group
300
  int energy_bin =
74,918,789✔
301
    lower_bound_index(energy_bounds_.begin(), energy_bounds_.end(), E);
74,918,789✔
302

303
  // mesh_bin += energy_bin * mesh->n_bins();
304
  // Create individual weight window
305
  WeightWindow ww;
74,918,789✔
306
  ww.lower_weight = lower_ww_(energy_bin, mesh_bin);
74,918,789✔
307
  ww.upper_weight = upper_ww_(energy_bin, mesh_bin);
74,918,789✔
308
  ww.survival_weight = ww.lower_weight * survival_ratio_;
74,918,789✔
309
  ww.max_lb_ratio = max_lb_ratio_;
74,918,789✔
310
  ww.max_split = max_split_;
74,918,789✔
311
  ww.weight_cutoff = weight_cutoff_;
74,918,789✔
312
  return {true, ww};
74,918,789✔
313
}
314

315
std::array<int, 2> WeightWindows::bounds_size() const
836✔
316
{
317
  int num_spatial_bins = this->mesh()->n_bins();
836✔
318
  int num_energy_bins =
836✔
319
    energy_bounds_.size() > 0 ? energy_bounds_.size() - 1 : 1;
836✔
320
  return {num_energy_bins, num_spatial_bins};
836✔
321
}
322

323
template<class T>
324
void WeightWindows::check_bounds(const T& lower, const T& upper) const
116!
325
{
326
  // make sure that the upper and lower bounds have the same size
327
  if (lower.size() != upper.size()) {
116!
UNCOV
328
    auto msg = fmt::format("The upper and lower weight window lengths do not "
×
329
                           "match.\n Lower size: {}\n Upper size: {}",
UNCOV
330
      lower.size(), upper.size());
×
UNCOV
331
    fatal_error(msg);
×
UNCOV
332
  }
×
333
  this->check_bounds(lower);
116✔
334
}
116✔
335

336
template<class T>
337
void WeightWindows::check_bounds(const T& bounds) const
116✔
338
{
339
  // check that the number of weight window entries is correct
340
  auto dims = this->bounds_size();
116✔
341
  if (bounds.size() != dims[0] * dims[1]) {
116!
UNCOV
342
    auto err_msg =
×
343
      fmt::format("In weight window domain {} the number of spatial "
344
                  "energy/spatial bins ({}) does not match the number "
345
                  "of weight bins ({})",
UNCOV
346
        id_, dims, bounds.size());
×
UNCOV
347
    fatal_error(err_msg);
×
UNCOV
348
  }
×
349
}
116✔
350

UNCOV
351
void WeightWindows::set_bounds(const tensor::Tensor<double>& lower_bounds,
×
352
  const tensor::Tensor<double>& upper_bounds)
353
{
354

UNCOV
355
  this->check_bounds(lower_bounds, upper_bounds);
×
356

357
  // set new weight window values
UNCOV
358
  lower_ww_ = lower_bounds;
×
UNCOV
359
  upper_ww_ = upper_bounds;
×
UNCOV
360
}
×
361

UNCOV
362
void WeightWindows::set_bounds(
×
363
  const tensor::Tensor<double>& lower_bounds, double ratio)
364
{
UNCOV
365
  this->check_bounds(lower_bounds);
×
366

367
  // set new weight window values
368
  lower_ww_ = lower_bounds;
×
UNCOV
369
  upper_ww_ = lower_bounds;
×
370
  upper_ww_ *= ratio;
×
371
}
×
372

373
void WeightWindows::set_bounds(
116✔
374
  span<const double> lower_bounds, span<const double> upper_bounds)
375
{
376
  check_bounds(lower_bounds, upper_bounds);
116✔
377
  auto shape = this->bounds_size();
116✔
378
  lower_ww_ = tensor::Tensor<double>(
116✔
379
    {static_cast<size_t>(shape[0]), static_cast<size_t>(shape[1])});
116✔
380
  upper_ww_ = tensor::Tensor<double>(
116✔
381
    {static_cast<size_t>(shape[0]), static_cast<size_t>(shape[1])});
116✔
382

383
  // Copy weight window values from input spans into the tensors
384
  std::copy(lower_bounds.data(), lower_bounds.data() + lower_ww_.size(),
116✔
385
    lower_ww_.data());
386
  std::copy(upper_bounds.data(), upper_bounds.data() + upper_ww_.size(),
116✔
387
    upper_ww_.data());
388
}
116✔
389

UNCOV
390
void WeightWindows::set_bounds(span<const double> lower_bounds, double ratio)
×
391
{
UNCOV
392
  this->check_bounds(lower_bounds);
×
393

UNCOV
394
  auto shape = this->bounds_size();
×
UNCOV
395
  lower_ww_ = tensor::Tensor<double>(
×
UNCOV
396
    {static_cast<size_t>(shape[0]), static_cast<size_t>(shape[1])});
×
UNCOV
397
  upper_ww_ = tensor::Tensor<double>(
×
UNCOV
398
    {static_cast<size_t>(shape[0]), static_cast<size_t>(shape[1])});
×
399

400
  // Copy lower bounds into both arrays, then scale upper by ratio
UNCOV
401
  std::copy(lower_bounds.data(), lower_bounds.data() + lower_ww_.size(),
×
402
    lower_ww_.data());
UNCOV
403
  std::copy(lower_bounds.data(), lower_bounds.data() + upper_ww_.size(),
×
404
    upper_ww_.data());
UNCOV
405
  upper_ww_ *= ratio;
×
UNCOV
406
}
×
407

408
void WeightWindows::update_weights(const Tally* tally, const std::string& value,
246✔
409
  double threshold, double ratio, WeightWindowUpdateMethod method)
410
{
411
  ///////////////////////////
412
  // Setup and checks
413
  ///////////////////////////
414
  this->check_tally_update_compatibility(tally);
246✔
415

416
  // Dimensions of weight window arrays
417
  int e_bins = lower_ww_.shape(0);
246!
418
  int64_t mesh_bins = lower_ww_.shape(1);
246!
419

420
  // Initialize weight window arrays to -1.0 by default
421
#pragma omp parallel for collapse(2) schedule(static)
140✔
422
  for (int e = 0; e < e_bins; e++) {
922✔
423
    for (int64_t m = 0; m < mesh_bins; m++) {
1,176,133✔
424
      lower_ww_(e, m) = -1.0;
1,175,317✔
425
      upper_ww_(e, m) = -1.0;
1,175,317✔
426
    }
427
  }
428

429
  // determine which value to use
430
  const std::set<std::string> allowed_values = {"mean", "rel_err"};
738!
431
  if (allowed_values.count(value) == 0) {
246!
432
    fatal_error(fmt::format("Invalid value '{}' specified for weight window "
×
433
                            "generation. Must be one of: 'mean' or 'rel_err'",
434
      value));
435
  }
436

437
  // determine the index of the specified score
438
  int score_index = tally->score_index("flux");
246✔
439
  if (score_index == C_NONE) {
246!
UNCOV
440
    fatal_error(
×
UNCOV
441
      fmt::format("A 'flux' score required for weight window generation "
×
442
                  "is not present on tally {}.",
443
        tally->id()));
×
444
  }
445

446
  ///////////////////////////
447
  // Extract tally data
448
  //
449
  // At the end of this section, mean and rel_err are
450
  // 2D tensors of tally data (n_e_groups, n_mesh_bins)
451
  //
452
  ///////////////////////////
453

454
  // build a shape for the tally results, this will always be
455
  // dimension 5 (3 filter dimensions, 1 score dimension, 1 results dimension)
456
  // Look for the size of the last dimension of the results tensor
457
  const auto& results = tally->results();
246!
458
  const int results_dim = static_cast<int>(results.shape(2));
246!
459
  std::array<int, 5> shape = {1, 1, 1, tally->n_scores(), results_dim};
246✔
460

461
  // set the shape for the filters applied on the tally
462
  for (int i = 0; i < tally->filters().size(); i++) {
940✔
463
    const auto& filter = model::tally_filters[tally->filters(i)];
694✔
464
    shape[i] = filter->n_bins();
694✔
465
  }
466

467
  // build the transpose information to re-order data according to filter type
468
  std::array<int, 5> transpose = {0, 1, 2, 3, 4};
246✔
469

470
  // track our filter types and where we've added new ones
471
  std::vector<FilterType> filter_types = tally->filter_types();
246✔
472

473
  // assign other filter types to dummy positions if needed
474
  if (!tally->has_filter(FilterType::PARTICLE))
246✔
475
    filter_types.push_back(FilterType::PARTICLE);
22✔
476

477
  if (!tally->has_filter(FilterType::ENERGY))
246✔
478
    filter_types.push_back(FilterType::ENERGY);
22✔
479

480
  // particle axis mapping
481
  transpose[0] =
246✔
482
    std::find(filter_types.begin(), filter_types.end(), FilterType::PARTICLE) -
246✔
483
    filter_types.begin();
246✔
484

485
  // energy axis mapping
486
  transpose[1] =
246✔
487
    std::find(filter_types.begin(), filter_types.end(), FilterType::ENERGY) -
246✔
488
    filter_types.begin();
246✔
489

490
  // mesh axis mapping
491
  transpose[2] =
246✔
492
    std::find(filter_types.begin(), filter_types.end(), FilterType::MESH) -
246✔
493
    filter_types.begin();
246✔
494

495
  // determine the index of the particle within its filter
496
  int particle_idx = 0;
246✔
497
  if (tally->has_filter(FilterType::PARTICLE)) {
246✔
498
    auto pf = tally->get_filter<ParticleFilter>();
224✔
499
    const auto& particles = pf->particles();
224!
500

501
    auto p_it =
224✔
502
      std::find(particles.begin(), particles.end(), this->particle_type_);
224!
503
    if (p_it == particles.end()) {
224!
504
      auto msg = fmt::format("Particle type '{}' not present on Filter {} for "
×
505
                             "Tally {} used to update WeightWindows {}",
506
        this->particle_type_.str(), pf->id(), tally->id(), this->id());
×
507
      fatal_error(msg);
×
UNCOV
508
    }
×
509

510
    particle_idx = p_it - particles.begin();
224✔
511
  }
512

513
  // The tally results array is 3D: (n_filter_combos, n_scores, n_result_types).
514
  // The first dimension is a row-major flattening of up to 3 filter dimensions
515
  // (particle, energy, mesh) whose storage order depends on which filters the
516
  // tally has. We need to map our desired indices (particle, energy, mesh)
517
  // into the correct flat filter combination index.
518
  //
519
  // transpose[i] tells us which storage position holds dimension i:
520
  //   i=0 -> particle, i=1 -> energy, i=2 -> mesh
521
  // shape[j] gives the number of bins for filter storage position j.
522

523
  // Row-major strides for the 3 filter dimensions
524
  const int stride0 = shape[1] * shape[2];
246✔
525
  const int stride1 = shape[2];
246✔
526

527
  tensor::Tensor<double> sum(
246✔
528
    {static_cast<size_t>(e_bins), static_cast<size_t>(mesh_bins)});
246✔
529
  tensor::Tensor<double> sum_sq(
246✔
530
    {static_cast<size_t>(e_bins), static_cast<size_t>(mesh_bins)});
246✔
531

532
  const int i_sum = static_cast<int>(TallyResult::SUM);
246✔
533
  const int i_sum_sq = static_cast<int>(TallyResult::SUM_SQ);
246✔
534

535
  for (int e = 0; e < e_bins; e++) {
2,054✔
536
    for (int64_t m = 0; m < mesh_bins; m++) {
2,612,711✔
537
      // Place particle, energy, and mesh indices into their storage positions
538
      std::array<int, 3> idx = {0, 0, 0};
2,610,903✔
539
      idx[transpose[0]] = particle_idx;
2,610,903✔
540
      idx[transpose[1]] = e;
2,610,903✔
541
      idx[transpose[2]] = static_cast<int>(m);
2,610,903✔
542

543
      // Compute flat filter combination index (row-major over filter dims)
544
      int flat = idx[0] * stride0 + idx[1] * stride1 + idx[2];
2,610,903✔
545

546
      sum(e, m) = results(flat, score_index, i_sum);
2,610,903✔
547
      sum_sq(e, m) = results(flat, score_index, i_sum_sq);
2,610,903✔
548
    }
549
  }
550
  int n = tally->n_realizations_;
246✔
551

552
  //////////////////////////////////////////////
553
  //
554
  // Assign new weight windows
555
  //
556
  // Use references to the existing weight window data
557
  // to store and update the values
558
  //
559
  //////////////////////////////////////////////
560

561
  // up to this point the data arrays are views into the tally results (no
562
  // computation has been performed) now we'll switch references to the tally's
563
  // bounds to avoid allocating additional memory
564
  auto& new_bounds = this->lower_ww_;
246✔
565
  auto& rel_err = this->upper_ww_;
246✔
566

567
  // get mesh volumes
568
  auto mesh_vols = this->mesh()->volumes();
246✔
569

570
  // Calculate mean (new_bounds) and relative error
571
#pragma omp parallel for collapse(2) schedule(static)
140✔
572
  for (int e = 0; e < e_bins; e++) {
922✔
573
    for (int64_t m = 0; m < mesh_bins; m++) {
1,176,133✔
574
      // Calculate mean
575
      new_bounds(e, m) = sum(e, m) / n;
1,175,317✔
576
      // Calculate relative error
577
      if (sum(e, m) > 0.0) {
1,175,317✔
578
        double mean_val = new_bounds(e, m);
101,480✔
579
        double variance = (sum_sq(e, m) / n - mean_val * mean_val) / (n - 1);
101,480✔
580
        rel_err(e, m) = std::sqrt(variance) / mean_val;
101,480✔
581
      } else {
582
        rel_err(e, m) = INFTY;
1,073,837✔
583
      }
584
      if (value == "rel_err") {
1,175,317✔
585
        new_bounds(e, m) = 1.0 / rel_err(e, m);
345,000✔
586
      }
587
    }
588
  }
589

590
  // Divide by volume of mesh elements
591
#pragma omp parallel for collapse(2) schedule(static)
140✔
592
  for (int e = 0; e < e_bins; e++) {
922✔
593
    for (int64_t m = 0; m < mesh_bins; m++) {
1,176,133✔
594
      new_bounds(e, m) /= mesh_vols[m];
1,175,317✔
595
    }
596
  }
597

598
  if (method == WeightWindowUpdateMethod::MAGIC) {
246✔
599
    // For MAGIC, weight windows are proportional to the forward fluxes.
600
    // We normalize weight windows independently for each energy group.
601

602
    // Find group maximum and normalize (per energy group)
603
    for (int e = 0; e < e_bins; e++) {
1,870✔
604
      double group_max = 0.0;
936✔
605

606
      // Find maximum value across all elements in this energy group
607
#pragma omp parallel for schedule(static) reduction(max : group_max)
936✔
608
      for (int64_t m = 0; m < mesh_bins; m++) {
1,092,505✔
609
        if (new_bounds(e, m) > group_max) {
1,091,725✔
610
          group_max = new_bounds(e, m);
2,520✔
611
        }
612
      }
613

614
      // Normalize values in this energy group by the maximum value
615
      if (group_max > 0.0) {
1,716✔
616
        double norm_factor = 1.0 / (2.0 * group_max);
1,683✔
617
#pragma omp parallel for schedule(static)
918✔
618
        for (int64_t m = 0; m < mesh_bins; m++) {
1,091,590✔
619
          new_bounds(e, m) *= norm_factor;
1,090,825✔
620
        }
621
      }
622
    }
623
  } else {
624
    // For FW-CADIS, weight windows are inversely proportional to the adjoint
625
    // fluxes. We normalize the weight windows across all energy groups.
626
#pragma omp parallel for collapse(2) schedule(static)
56✔
627
    for (int e = 0; e < e_bins; e++) {
72✔
628
      for (int64_t m = 0; m < mesh_bins; m++) {
83,628✔
629
        // Take the inverse, but are careful not to divide by zero
630
        if (new_bounds(e, m) != 0.0) {
83,592✔
631
          new_bounds(e, m) = 1.0 / new_bounds(e, m);
69,660✔
632
        } else {
633
          new_bounds(e, m) = 0.0;
13,932✔
634
        }
635
      }
636
    }
637

638
    // Find the maximum value across all elements
639
    double max_val = 0.0;
56✔
640
#pragma omp parallel for collapse(2) schedule(static) reduction(max : max_val)
56✔
641
    for (int e = 0; e < e_bins; e++) {
72✔
642
      for (int64_t m = 0; m < mesh_bins; m++) {
83,628✔
643
        if (new_bounds(e, m) > max_val) {
83,592✔
644
          max_val = new_bounds(e, m);
405✔
645
        }
646
      }
647
    }
648

649
    // Parallel normalization
650
    if (max_val > 0.0) {
92✔
651
      double norm_factor = 1.0 / (2.0 * max_val);
68✔
652
#pragma omp parallel for collapse(2) schedule(static)
38✔
653
      for (int e = 0; e < e_bins; e++) {
60✔
654
        for (int64_t m = 0; m < mesh_bins; m++) {
69,690✔
655
          new_bounds(e, m) *= norm_factor;
69,660✔
656
        }
657
      }
658
    }
659
  }
660

661
  // Final processing
662
#pragma omp parallel for collapse(2) schedule(static)
140✔
663
  for (int e = 0; e < e_bins; e++) {
922✔
664
    for (int64_t m = 0; m < mesh_bins; m++) {
1,176,133✔
665
      // Values where the mean is zero should be ignored
666
      if (sum(e, m) <= 0.0) {
1,175,317✔
667
        new_bounds(e, m) = -1.0;
1,073,837✔
668
      }
669
      // Values where the relative error is higher than the threshold should be
670
      // ignored
671
      else if (rel_err(e, m) > threshold) {
101,480✔
672
        new_bounds(e, m) = -1.0;
1,420✔
673
      }
674
      // Set the upper bounds
675
      upper_ww_(e, m) = ratio * lower_ww_(e, m);
1,175,317✔
676
    }
677
  }
678
}
738✔
679

680
void WeightWindows::check_tally_update_compatibility(const Tally* tally)
246✔
681
{
682
  // define the set of allowed filters for the tally
683
  const std::set<FilterType> allowed_filters = {
246✔
684
    FilterType::MESH, FilterType::ENERGY, FilterType::PARTICLE};
246✔
685

686
  // retrieve a mapping of filter type to filter index for the tally
687
  auto filter_indices = tally->filter_indices();
246✔
688

689
  // a mesh filter is required for a tally used to update weight windows
690
  if (!filter_indices.count(FilterType::MESH)) {
246!
UNCOV
691
    fatal_error(
×
692
      "A mesh filter is required for a tally to update weight window bounds");
693
  }
694

695
  // ensure the mesh filter is using the same mesh as this weight window object
696
  auto mesh_filter = tally->get_filter<MeshFilter>();
246✔
697

698
  // make sure that all of the filters present on the tally are allowed
699
  for (auto filter_pair : filter_indices) {
940✔
700
    if (allowed_filters.find(filter_pair.first) == allowed_filters.end()) {
694!
UNCOV
701
      fatal_error(fmt::format("Invalid filter type '{}' found on tally "
×
702
                              "used for weight window generation.",
UNCOV
703
        model::tally_filters[tally->filters(filter_pair.second)]->type_str()));
×
704
    }
705
  }
706

707
  if (mesh_filter->mesh() != mesh_idx_) {
246!
UNCOV
708
    int32_t mesh_filter_id = model::meshes[mesh_filter->mesh()]->id();
×
UNCOV
709
    int32_t ww_mesh_id = model::meshes[this->mesh_idx_]->id();
×
UNCOV
710
    fatal_error(fmt::format("Mesh filter {} uses a different mesh ({}) than "
×
711
                            "weight window {} mesh ({})",
UNCOV
712
      mesh_filter->id(), mesh_filter_id, id_, ww_mesh_id));
×
713
  }
714

715
  // if an energy filter exists, make sure the energy grid matches that of this
716
  // weight window object
717
  if (auto energy_filter = tally->get_filter<EnergyFilter>()) {
246✔
718
    std::vector<double> filter_bins = energy_filter->bins();
224✔
719
    std::set<double> filter_e_bounds(
224✔
720
      energy_filter->bins().begin(), energy_filter->bins().end());
224✔
721
    if (filter_e_bounds.size() != energy_bounds().size()) {
224!
UNCOV
722
      fatal_error(
×
UNCOV
723
        fmt::format("Energy filter {} does not have the same number of energy "
×
724
                    "bounds ({}) as weight window object {} ({})",
UNCOV
725
          energy_filter->id(), filter_e_bounds.size(), id_,
×
UNCOV
726
          energy_bounds().size()));
×
727
    }
728

729
    for (auto e : energy_bounds()) {
2,234✔
730
      if (filter_e_bounds.count(e) == 0) {
2,010!
UNCOV
731
        fatal_error(fmt::format(
×
732
          "Energy bounds of filter {} and weight windows {} do not match",
UNCOV
733
          energy_filter->id(), id_));
×
734
      }
735
    }
736
  }
224✔
737
}
246✔
738

739
void WeightWindows::to_hdf5(hid_t group) const
134✔
740
{
741
  hid_t ww_group = create_group(group, fmt::format("weight_windows_{}", id()));
134✔
742

743
  write_dataset(ww_group, "mesh", this->mesh()->id());
134✔
744
  write_dataset(ww_group, "particle_type", particle_type_.str());
134✔
745
  write_dataset(ww_group, "energy_bounds", energy_bounds_);
134✔
746
  write_dataset(ww_group, "lower_ww_bounds", lower_ww_);
134✔
747
  write_dataset(ww_group, "upper_ww_bounds", upper_ww_);
134✔
748
  write_dataset(ww_group, "survival_ratio", survival_ratio_);
134✔
749
  write_dataset(ww_group, "max_lower_bound_ratio", max_lb_ratio_);
134✔
750
  write_dataset(ww_group, "max_split", max_split_);
134✔
751
  write_dataset(ww_group, "weight_cutoff", weight_cutoff_);
134✔
752

753
  close_group(ww_group);
134✔
754
}
134✔
755

756
WeightWindowsGenerator::WeightWindowsGenerator(pugi::xml_node node)
79✔
757
{
758
  // read information from the XML node
759
  int32_t mesh_id = std::stoi(get_node_value(node, "mesh"));
158✔
760
  int32_t mesh_idx = model::mesh_map[mesh_id];
79✔
761
  max_realizations_ = std::stoi(get_node_value(node, "max_realizations"));
158✔
762

763
  int32_t active_batches = settings::n_batches - settings::n_inactive;
79✔
764
  if (max_realizations_ > active_batches) {
79✔
765
    auto msg =
16✔
766
      fmt::format("The maximum number of specified tally realizations ({}) is "
767
                  "greater than the number of active batches ({}).",
768
        max_realizations_, active_batches);
16✔
769
    warning(msg);
16✔
770
  }
16✔
771
  auto tmp_str = get_node_value(node, "particle_type", false, true);
79✔
772
  auto particle_type = ParticleType {tmp_str};
79✔
773

774
  update_interval_ = std::stoi(get_node_value(node, "update_interval"));
158✔
775
  on_the_fly_ = get_node_value_bool(node, "on_the_fly");
79✔
776

777
  std::vector<double> e_bounds;
79✔
778
  if (check_for_node(node, "energy_bounds")) {
79✔
779
    e_bounds = get_node_array<double>(node, "energy_bounds");
46✔
780
  } else {
781
    int p_type = particle_type.transport_index();
56✔
782
    if (p_type == C_NONE) {
56!
UNCOV
783
      fatal_error("Weight windows particle is not supported for transport.");
×
784
    }
785
    e_bounds.push_back(data::energy_min[p_type]);
56✔
786
    e_bounds.push_back(data::energy_max[p_type]);
56✔
787
  }
788

789
  // set method
790
  std::string method_string = get_node_value(node, "method");
79✔
791
  if (method_string == "magic") {
79✔
792
    method_ = WeightWindowUpdateMethod::MAGIC;
33✔
793
    if (settings::solver_type == SolverType::RANDOM_RAY &&
33!
794
        FlatSourceDomain::adjoint_) {
UNCOV
795
      fatal_error("Random ray weight window generation with MAGIC cannot be "
×
796
                  "done in adjoint mode.");
797
    }
798
  } else if (method_string == "fw_cadis") {
46!
799
    method_ = WeightWindowUpdateMethod::FW_CADIS;
46✔
800
    if (settings::solver_type != SolverType::RANDOM_RAY) {
46!
UNCOV
801
      fatal_error("FW-CADIS can only be run in random ray solver mode.");
×
802
    }
803
    FlatSourceDomain::adjoint_ = true;
46✔
804
  } else {
UNCOV
805
    fatal_error(fmt::format(
×
806
      "Unknown weight window update method '{}' specified", method_string));
807
  }
808

809
  // parse non-default update parameters if specified
810
  if (check_for_node(node, "update_parameters")) {
79✔
811
    pugi::xml_node params_node = node.child("update_parameters");
22✔
812
    if (check_for_node(params_node, "value"))
22!
813
      tally_value_ = get_node_value(params_node, "value");
22✔
814
    if (check_for_node(params_node, "threshold"))
22!
815
      threshold_ = std::stod(get_node_value(params_node, "threshold"));
44✔
816
    if (check_for_node(params_node, "ratio")) {
22!
817
      ratio_ = std::stod(get_node_value(params_node, "ratio"));
44✔
818
    }
819
  }
820

821
  // check update parameter values
822
  if (tally_value_ != "mean" && tally_value_ != "rel_err") {
79!
823
    fatal_error(fmt::format("Unsupported tally value '{}' specified for "
×
824
                            "weight window generation.",
UNCOV
825
      tally_value_));
×
826
  }
827
  if (threshold_ <= 0.0)
79!
UNCOV
828
    fatal_error(fmt::format("Invalid relative error threshold '{}' (<= 0.0) "
×
829
                            "specified for weight window generation",
UNCOV
830
      ratio_));
×
831
  if (ratio_ <= 1.0)
79!
832
    fatal_error(fmt::format("Invalid weight window ratio '{}' (<= 1.0) "
×
833
                            "specified for weight window generation"));
834

835
  // create a matching weight windows object
836
  auto wws = WeightWindows::create();
79✔
837
  ww_idx_ = wws->index();
79✔
838
  wws->set_mesh(mesh_idx);
79✔
839
  if (e_bounds.size() > 0)
79!
840
    wws->set_energy_bounds(e_bounds);
79✔
841
  wws->set_particle_type(particle_type);
79✔
842
  wws->set_defaults();
79✔
843
}
79✔
844

845
void WeightWindowsGenerator::create_tally()
79✔
846
{
847
  const auto& wws = variance_reduction::weight_windows[ww_idx_];
79✔
848

849
  // create a tally based on the WWG information
850
  Tally* ww_tally = Tally::create();
79✔
851
  tally_idx_ = model::tally_map[ww_tally->id()];
79✔
852
  ww_tally->set_scores({"flux"});
158!
853

854
  int32_t mesh_id = wws->mesh()->id();
79✔
855
  int32_t mesh_idx = model::mesh_map.at(mesh_id);
79✔
856
  // see if there's already a mesh filter using this mesh
857
  bool found_mesh_filter = false;
79✔
858
  for (const auto& f : model::tally_filters) {
247✔
859
    if (f->type() == FilterType::MESH) {
179✔
860
      const auto* mesh_filter = dynamic_cast<MeshFilter*>(f.get());
11!
861
      if (mesh_filter->mesh() == mesh_idx && !mesh_filter->translated() &&
22!
862
          !mesh_filter->rotated()) {
11✔
863
        ww_tally->add_filter(f.get());
11✔
864
        found_mesh_filter = true;
865
        break;
866
      }
867
    }
868
  }
869

870
  if (!found_mesh_filter) {
68✔
871
    auto mesh_filter = Filter::create("mesh");
68✔
872
    openmc_mesh_filter_set_mesh(mesh_filter->index(), model::mesh_map[mesh_id]);
68✔
873
    ww_tally->add_filter(mesh_filter);
68✔
874
  }
875

876
  const auto& e_bounds = wws->energy_bounds();
79!
877
  if (e_bounds.size() > 0) {
79!
878
    auto energy_filter = Filter::create("energy");
79✔
879
    openmc_energy_filter_set_bins(
79✔
880
      energy_filter->index(), e_bounds.size(), e_bounds.data());
79✔
881
    ww_tally->add_filter(energy_filter);
79✔
882
  }
883

884
  // add a particle filter
885
  auto particle_type = wws->particle_type();
79✔
886
  auto particle_filter = Filter::create("particle");
79✔
887
  auto pf = dynamic_cast<ParticleFilter*>(particle_filter);
79!
888
  pf->set_particles({&particle_type, 1});
79✔
889
  ww_tally->add_filter(particle_filter);
79✔
890
}
79✔
891

892
void WeightWindowsGenerator::update() const
2,277✔
893
{
894
  const auto& wws = variance_reduction::weight_windows[ww_idx_];
2,277✔
895

896
  Tally* tally = model::tallies[tally_idx_].get();
2,277✔
897

898
  // If in random ray mode, only update on the last batch
899
  if (settings::solver_type == SolverType::RANDOM_RAY) {
2,277✔
900
    if (simulation::current_batch != settings::n_batches) {
2,112✔
901
      return;
902
    }
903
    // If in Monte Carlo mode and beyond the number of max realizations or
904
    // not at the correct update interval, skip the update
905
  } else if (max_realizations_ < tally->n_realizations_ ||
165✔
906
             tally->n_realizations_ % update_interval_ != 0) {
33!
907
    return;
908
  }
909

910
  wws->update_weights(tally, tally_value_, threshold_, ratio_, method_);
125✔
911

912
  // if we're not doing on the fly generation, reset the tally results once
913
  // we're done with the update
914
  if (!on_the_fly_)
125!
UNCOV
915
    tally->reset();
×
916

917
  // TODO: deactivate or remove tally once weight window generation is
918
  // complete
919
}
920

921
//==============================================================================
922
// Non-member functions
923
//==============================================================================
924

925
std::pair<bool, WeightWindow> search_weight_window(const Particle& p)
87,842,819✔
926
{
927
  // TODO: this is a linear search - should do something more clever
928
  for (const auto& ww : variance_reduction::weight_windows) {
107,983,267✔
929
    auto [ww_found, weight_window] = ww->get_weight_window(p);
95,059,237✔
930
    if (ww_found)
95,059,237✔
931
      return {true, weight_window};
74,918,789✔
932
  }
933
  return {false, {}};
12,924,030✔
934
}
935

936
void apply_weight_windows(Particle& p)
168,763,596✔
937
{
938
  if (!settings::weight_windows_on)
168,763,596✔
939
    return;
167,945,727✔
940

941
  // WW on photon and neutron only
942
  if (!p.type().is_neutron() && !p.type().is_photon())
821,942!
943
    return;
944

945
  // skip dead or no energy
946
  if (p.E() <= 0 || !p.alive())
821,942!
947
    return;
948

949
  auto [ww_found, ww] = search_weight_window(p);
817,869✔
950
  if (ww_found && ww.is_valid()) {
817,869✔
951
    apply_weight_window(p, ww);
710,739✔
952
  } else {
953
    if (p.wgt_ww_born() == -1.0)
107,130✔
954
      p.wgt_ww_born() = 1.0;
50,402✔
955
  }
956
}
957

958
void apply_weight_window(Particle& p, WeightWindow weight_window)
87,668,694✔
959
{
960
  if (!weight_window.is_valid())
87,668,694✔
961
    return;
962

963
  // skip dead or no energy
964
  if (p.E() <= 0 || !p.alive())
62,173,141✔
965
    return;
966

967
  // If particle has not yet had its birth weight window value set, set it to
968
  // the current weight window.
969
  if (p.wgt_ww_born() == -1.0)
58,567,916✔
970
    p.wgt_ww_born() =
696,628✔
971
      (weight_window.lower_weight + weight_window.upper_weight) / 2;
696,628✔
972

973
  // Normalize weight windows based on particle's starting weight
974
  // and the value of the weight window the particle was born in.
975
  weight_window.scale(p.wgt_born() / p.wgt_ww_born());
58,567,916✔
976

977
  // get the paramters
978
  double weight = p.wgt();
58,567,916✔
979

980
  // first check to see if particle should be killed for weight cutoff
981
  if (p.wgt() < weight_window.weight_cutoff) {
58,567,916✔
982
    p.wgt() = 0.0;
297✔
983
    return;
297✔
984
  }
985

986
  // check if particle is far above current weight window
987
  // only do this if the factor is not already set on the particle and a
988
  // maximum lower bound ratio is specified
989
  if (p.ww_factor() == 0.0 && weight_window.max_lb_ratio > 1.0 &&
58,567,619✔
990
      p.wgt() > weight_window.lower_weight * weight_window.max_lb_ratio) {
11,000!
991
    p.ww_factor() =
11,000✔
992
      p.wgt() / (weight_window.lower_weight * weight_window.max_lb_ratio);
11,000✔
993
  }
994

995
  // move weight window closer to the particle weight if needed
996
  if (p.ww_factor() > 1.0)
58,567,619✔
997
    weight_window.scale(p.ww_factor());
3,193,641✔
998

999
  // if particle's weight is above the weight window split until they are within
1000
  // the window
1001
  if (weight > weight_window.upper_weight) {
58,567,619✔
1002
    // do not further split the particle if above the limit
1003
    if (p.n_split() >= settings::max_history_splits)
12,290,230✔
1004
      return;
10,874,932✔
1005

1006
    double n_split = std::ceil(weight / weight_window.upper_weight);
1,415,298✔
1007
    double max_split = weight_window.max_split;
1,415,298✔
1008
    n_split = std::min(n_split, max_split);
1,415,298✔
1009

1010
    p.n_split() += n_split;
1,415,298✔
1011

1012
    // Create secondaries and divide weight among all particles
1013
    int i_split = std::round(n_split);
1,415,298✔
1014
    for (int l = 0; l < i_split - 1; l++) {
5,665,685✔
1015
      p.split(weight / n_split);
4,250,387✔
1016
    }
1017
    // remaining weight is applied to current particle
1018
    p.wgt() = weight / n_split;
1,415,298✔
1019

1020
  } else if (weight <= weight_window.lower_weight) {
46,277,389✔
1021
    // if the particle weight is below the window, play Russian roulette
1022
    double weight_survive =
1,462,891✔
1023
      std::min(weight * weight_window.max_split, weight_window.survival_weight);
1,462,891✔
1024
    russian_roulette(p, weight_survive);
1,462,891✔
1025
  } // else particle is in the window, continue as normal
1026
}
1027

1028
void free_memory_weight_windows()
8,206✔
1029
{
1030
  variance_reduction::ww_map.clear();
8,206✔
1031
  variance_reduction::weight_windows.clear();
8,206✔
1032
}
8,206✔
1033

1034
void finalize_variance_reduction()
8,049✔
1035
{
1036
  for (const auto& wwg : variance_reduction::weight_windows_generators) {
8,128✔
1037
    wwg->create_tally();
79✔
1038
  }
1039
}
8,049✔
1040

1041
//==============================================================================
1042
// C API
1043
//==============================================================================
1044

1045
int verify_ww_index(int32_t index)
1,991✔
1046
{
1047
  if (index < 0 || index >= variance_reduction::weight_windows.size()) {
1,991!
UNCOV
1048
    set_errmsg(fmt::format("Index '{}' for weight windows is invalid", index));
×
UNCOV
1049
    return OPENMC_E_OUT_OF_BOUNDS;
×
1050
  }
1051
  return 0;
1052
}
1053

1054
extern "C" int openmc_get_weight_windows_index(int32_t id, int32_t* idx)
165✔
1055
{
1056
  auto it = variance_reduction::ww_map.find(id);
165!
1057
  if (it == variance_reduction::ww_map.end()) {
165!
UNCOV
1058
    set_errmsg(fmt::format("No weight windows exist with ID={}", id));
×
UNCOV
1059
    return OPENMC_E_INVALID_ID;
×
1060
  }
1061

1062
  *idx = it->second;
165✔
1063
  return 0;
165✔
1064
}
1065

1066
extern "C" int openmc_weight_windows_get_id(int32_t index, int32_t* id)
517✔
1067
{
1068
  if (int err = verify_ww_index(index))
517!
1069
    return err;
1070

1071
  const auto& wws = variance_reduction::weight_windows.at(index);
517✔
1072
  *id = wws->id();
517✔
1073
  return 0;
517✔
1074
}
1075

1076
extern "C" int openmc_weight_windows_set_id(int32_t index, int32_t id)
154✔
1077
{
1078
  if (int err = verify_ww_index(index))
154!
1079
    return err;
1080

1081
  const auto& wws = variance_reduction::weight_windows.at(index);
154✔
1082
  wws->set_id(id);
154✔
1083
  return 0;
154✔
1084
}
1085

1086
extern "C" int openmc_weight_windows_update_magic(int32_t ww_idx,
121✔
1087
  int32_t tally_idx, const char* value, double threshold, double ratio)
1088
{
1089
  if (int err = verify_ww_index(ww_idx))
121!
1090
    return err;
1091

1092
  if (tally_idx < 0 || tally_idx >= model::tallies.size()) {
121!
UNCOV
1093
    set_errmsg(fmt::format("Index '{}' for tally is invalid", tally_idx));
×
UNCOV
1094
    return OPENMC_E_OUT_OF_BOUNDS;
×
1095
  }
1096

1097
  // get the requested tally
1098
  const Tally* tally = model::tallies.at(tally_idx).get();
121✔
1099

1100
  // get the WeightWindows object
1101
  const auto& wws = variance_reduction::weight_windows.at(ww_idx);
121✔
1102

1103
  wws->update_weights(tally, value, threshold, ratio);
121✔
1104

1105
  return 0;
121✔
1106
}
1107

1108
extern "C" int openmc_weight_windows_set_mesh(int32_t ww_idx, int32_t mesh_idx)
154✔
1109
{
1110
  if (int err = verify_ww_index(ww_idx))
154!
1111
    return err;
1112
  const auto& wws = variance_reduction::weight_windows.at(ww_idx);
154✔
1113
  wws->set_mesh(mesh_idx);
154✔
1114
  return 0;
154✔
1115
}
1116

1117
extern "C" int openmc_weight_windows_get_mesh(int32_t ww_idx, int32_t* mesh_idx)
11✔
1118
{
1119
  if (int err = verify_ww_index(ww_idx))
11!
1120
    return err;
1121
  const auto& wws = variance_reduction::weight_windows.at(ww_idx);
11✔
1122
  *mesh_idx = model::mesh_map.at(wws->mesh()->id());
11✔
1123
  return 0;
11✔
1124
}
1125

1126
extern "C" int openmc_weight_windows_set_energy_bounds(
132✔
1127
  int32_t ww_idx, double* e_bounds, size_t e_bounds_size)
1128
{
1129
  if (int err = verify_ww_index(ww_idx))
132!
1130
    return err;
1131
  const auto& wws = variance_reduction::weight_windows.at(ww_idx);
132✔
1132
  wws->set_energy_bounds({e_bounds, e_bounds_size});
132✔
1133
  return 0;
132✔
1134
}
1135

1136
extern "C" int openmc_weight_windows_get_energy_bounds(
11✔
1137
  int32_t ww_idx, const double** e_bounds, size_t* e_bounds_size)
1138
{
1139
  if (int err = verify_ww_index(ww_idx))
11!
1140
    return err;
1141
  const auto& wws = variance_reduction::weight_windows[ww_idx].get();
11✔
1142
  *e_bounds = wws->energy_bounds().data();
11✔
1143
  *e_bounds_size = wws->energy_bounds().size();
11✔
1144
  return 0;
11✔
1145
}
1146

1147
extern "C" int openmc_weight_windows_set_particle(
176✔
1148
  int32_t index, int32_t particle)
1149
{
1150
  if (int err = verify_ww_index(index))
176!
1151
    return err;
1152

1153
  const auto& wws = variance_reduction::weight_windows.at(index);
176✔
1154
  wws->set_particle_type(ParticleType {particle});
176✔
1155
  return 0;
176✔
1156
}
1157

1158
extern "C" int openmc_weight_windows_get_particle(
44✔
1159
  int32_t index, int32_t* particle)
1160
{
1161
  if (int err = verify_ww_index(index))
44!
1162
    return err;
1163

1164
  const auto& wws = variance_reduction::weight_windows.at(index);
44✔
1165
  *particle = wws->particle_type().pdg_number();
44✔
1166
  return 0;
44✔
1167
}
1168

1169
extern "C" int openmc_weight_windows_get_bounds(int32_t index,
484✔
1170
  const double** lower_bounds, const double** upper_bounds, size_t* size)
1171
{
1172
  if (int err = verify_ww_index(index))
484!
1173
    return err;
1174

1175
  const auto& wws = variance_reduction::weight_windows[index];
484✔
1176
  *size = wws->lower_ww_bounds().size();
484✔
1177
  *lower_bounds = wws->lower_ww_bounds().data();
484✔
1178
  *upper_bounds = wws->upper_ww_bounds().data();
484✔
1179
  return 0;
484✔
1180
}
1181

1182
extern "C" int openmc_weight_windows_set_bounds(int32_t index,
11✔
1183
  const double* lower_bounds, const double* upper_bounds, size_t size)
1184
{
1185
  if (int err = verify_ww_index(index))
11!
1186
    return err;
1187

1188
  const auto& wws = variance_reduction::weight_windows[index];
11✔
1189
  wws->set_bounds(span<const double>(lower_bounds, size),
11✔
1190
    span<const double>(upper_bounds, size));
1191
  return 0;
11✔
1192
}
1193

1194
extern "C" int openmc_weight_windows_get_survival_ratio(
33✔
1195
  int32_t index, double* ratio)
1196
{
1197
  if (int err = verify_ww_index(index))
33!
1198
    return err;
1199
  const auto& wws = variance_reduction::weight_windows[index];
33✔
1200
  *ratio = wws->survival_ratio();
33✔
1201
  return 0;
33✔
1202
}
1203

1204
extern "C" int openmc_weight_windows_set_survival_ratio(
11✔
1205
  int32_t index, double ratio)
1206
{
1207
  if (int err = verify_ww_index(index))
11!
1208
    return err;
1209
  const auto& wws = variance_reduction::weight_windows[index];
11✔
1210
  wws->survival_ratio() = ratio;
11✔
1211
  std::cout << "Survival ratio: " << wws->survival_ratio() << std::endl;
11✔
1212
  return 0;
11✔
1213
}
1214

1215
extern "C" int openmc_weight_windows_get_max_lower_bound_ratio(
33✔
1216
  int32_t index, double* lb_ratio)
1217
{
1218
  if (int err = verify_ww_index(index))
33!
1219
    return err;
1220
  const auto& wws = variance_reduction::weight_windows[index];
33✔
1221
  *lb_ratio = wws->max_lower_bound_ratio();
33✔
1222
  return 0;
33✔
1223
}
1224

1225
extern "C" int openmc_weight_windows_set_max_lower_bound_ratio(
11✔
1226
  int32_t index, double lb_ratio)
1227
{
1228
  if (int err = verify_ww_index(index))
11!
1229
    return err;
1230
  const auto& wws = variance_reduction::weight_windows[index];
11✔
1231
  wws->max_lower_bound_ratio() = lb_ratio;
11✔
1232
  return 0;
11✔
1233
}
1234

1235
extern "C" int openmc_weight_windows_get_weight_cutoff(
33✔
1236
  int32_t index, double* cutoff)
1237
{
1238
  if (int err = verify_ww_index(index))
33!
1239
    return err;
1240
  const auto& wws = variance_reduction::weight_windows[index];
33✔
1241
  *cutoff = wws->weight_cutoff();
33✔
1242
  return 0;
33✔
1243
}
1244

1245
extern "C" int openmc_weight_windows_set_weight_cutoff(
11✔
1246
  int32_t index, double cutoff)
1247
{
1248
  if (int err = verify_ww_index(index))
11!
1249
    return err;
1250
  const auto& wws = variance_reduction::weight_windows[index];
11✔
1251
  wws->weight_cutoff() = cutoff;
11✔
1252
  return 0;
11✔
1253
}
1254

1255
extern "C" int openmc_weight_windows_get_max_split(
33✔
1256
  int32_t index, int* max_split)
1257
{
1258
  if (int err = verify_ww_index(index))
33!
1259
    return err;
1260
  const auto& wws = variance_reduction::weight_windows[index];
33✔
1261
  *max_split = wws->max_split();
33✔
1262
  return 0;
33✔
1263
}
1264

1265
extern "C" int openmc_weight_windows_set_max_split(int32_t index, int max_split)
11✔
1266
{
1267
  if (int err = verify_ww_index(index))
11!
1268
    return err;
1269
  const auto& wws = variance_reduction::weight_windows[index];
11✔
1270
  wws->max_split() = max_split;
11✔
1271
  return 0;
11✔
1272
}
1273

1274
extern "C" int openmc_extend_weight_windows(
154✔
1275
  int32_t n, int32_t* index_start, int32_t* index_end)
1276
{
1277
  if (index_start)
154!
1278
    *index_start = variance_reduction::weight_windows.size();
154✔
1279
  if (index_end)
154!
UNCOV
1280
    *index_end = variance_reduction::weight_windows.size() + n - 1;
×
1281
  for (int i = 0; i < n; ++i)
308✔
1282
    variance_reduction::weight_windows.push_back(make_unique<WeightWindows>());
154✔
1283
  return 0;
154✔
1284
}
1285

1286
extern "C" size_t openmc_weight_windows_size()
154✔
1287
{
1288
  return variance_reduction::weight_windows.size();
154✔
1289
}
1290

1291
extern "C" int openmc_weight_windows_export(const char* filename)
158✔
1292
{
1293

1294
  if (!mpi::master)
158✔
1295
    return 0;
1296

1297
  std::string name = filename ? filename : "weight_windows.h5";
235✔
1298

1299
  write_message(fmt::format("Exporting weight windows to {}...", name), 5);
158✔
1300

1301
  hid_t ww_file = file_open(name, 'w');
134✔
1302

1303
  // Write file type
1304
  write_attribute(ww_file, "filetype", "weight_windows");
134✔
1305

1306
  // Write revisiion number for state point file
1307
  write_attribute(ww_file, "version", VERSION_WEIGHT_WINDOWS);
134✔
1308

1309
  hid_t weight_windows_group = create_group(ww_file, "weight_windows");
134✔
1310

1311
  hid_t mesh_group = create_group(ww_file, "meshes");
134✔
1312

1313
  std::vector<int32_t> mesh_ids;
134✔
1314
  std::vector<int32_t> ww_ids;
134✔
1315
  for (const auto& ww : variance_reduction::weight_windows) {
268✔
1316

1317
    ww->to_hdf5(weight_windows_group);
134✔
1318
    ww_ids.push_back(ww->id());
134✔
1319

1320
    // if the mesh has already been written, move on
1321
    int32_t mesh_id = ww->mesh()->id();
134!
1322
    if (std::find(mesh_ids.begin(), mesh_ids.end(), mesh_id) != mesh_ids.end())
134!
UNCOV
1323
      continue;
×
1324

1325
    mesh_ids.push_back(mesh_id);
134✔
1326
    ww->mesh()->to_hdf5(mesh_group);
134✔
1327
  }
1328

1329
  write_attribute(mesh_group, "n_meshes", mesh_ids.size());
134✔
1330
  write_attribute(mesh_group, "ids", mesh_ids);
134✔
1331
  close_group(mesh_group);
134✔
1332

1333
  write_attribute(weight_windows_group, "n_weight_windows", ww_ids.size());
134✔
1334
  write_attribute(weight_windows_group, "ids", ww_ids);
134✔
1335
  close_group(weight_windows_group);
134✔
1336

1337
  file_close(ww_file);
134✔
1338

1339
  return 0;
134✔
1340
}
292✔
1341

1342
extern "C" int openmc_weight_windows_import(const char* filename)
11✔
1343
{
1344
  std::string name = filename ? filename : "weight_windows.h5";
11!
1345

1346
  if (mpi::master)
11!
1347
    write_message(fmt::format("Importing weight windows from {}...", name), 5);
24✔
1348

1349
  if (!file_exists(name)) {
11!
UNCOV
1350
    set_errmsg(fmt::format("File '{}' does not exist", name));
×
1351
  }
1352

1353
  hid_t ww_file = file_open(name, 'r');
11✔
1354

1355
  // Check that filetype is correct
1356
  std::string filetype;
11✔
1357
  read_attribute(ww_file, "filetype", filetype);
11✔
1358
  if (filetype != "weight_windows") {
11!
UNCOV
1359
    file_close(ww_file);
×
UNCOV
1360
    set_errmsg(fmt::format("File '{}' is not a weight windows file.", name));
×
UNCOV
1361
    return OPENMC_E_INVALID_ARGUMENT;
×
1362
  }
1363

1364
  // Check that the file version is compatible
1365
  std::array<int, 2> file_version;
11✔
1366
  read_attribute(ww_file, "version", file_version);
11✔
1367
  if (file_version[0] != VERSION_WEIGHT_WINDOWS[0]) {
11!
UNCOV
1368
    std::string err_msg =
×
1369
      fmt::format("File '{}' has version {} which is incompatible with the "
1370
                  "expected version ({}).",
UNCOV
1371
        name, file_version, VERSION_WEIGHT_WINDOWS);
×
UNCOV
1372
    set_errmsg(err_msg);
×
UNCOV
1373
    return OPENMC_E_INVALID_ARGUMENT;
×
UNCOV
1374
  }
×
1375

1376
  hid_t weight_windows_group = open_group(ww_file, "weight_windows");
11✔
1377

1378
  hid_t mesh_group = open_group(ww_file, "meshes");
11✔
1379

1380
  read_meshes(mesh_group);
11✔
1381

1382
  std::vector<std::string> names = group_names(weight_windows_group);
11✔
1383

1384
  for (const auto& name : names) {
22✔
1385
    WeightWindows::from_hdf5(weight_windows_group, name);
11✔
1386
  }
1387

1388
  close_group(weight_windows_group);
11✔
1389

1390
  file_close(ww_file);
11✔
1391

1392
  return 0;
11✔
1393
}
22✔
1394

1395
} // namespace openmc
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc