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

openmc-dev / openmc / 22231282333

20 Feb 2026 04:04PM UTC coverage: 81.861% (-0.2%) from 82.058%
22231282333

Pull #2693

github

web-flow
Merge e5188d202 into 53ce1910f
Pull Request #2693: Add reactivity control to coupled transport-depletion analyses

17278 of 24307 branches covered (71.08%)

Branch coverage included in aggregate %.

75 of 80 new or added lines in 4 files covered. (93.75%)

3462 existing lines in 80 files now uncovered.

57630 of 67199 relevant lines covered (85.76%)

49119197.82 hits per line

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

72.64
/src/hdf5_interface.cpp
1
#include "openmc/hdf5_interface.h"
2

3
#include <cstring>
4
#include <stdexcept>
5
#include <string>
6

7
#include "openmc/tensor.h"
8
#include <fmt/core.h>
9

10
#include "hdf5.h"
11
#include "hdf5_hl.h"
12
#ifdef OPENMC_MPI
13
#include "mpi.h"
14
#include "openmc/message_passing.h"
15
#endif
16

17
#include "openmc/array.h"
18

19
namespace openmc {
20

21
bool attribute_exists(hid_t obj_id, const char* name)
1,356,599✔
22
{
23
  htri_t out = H5Aexists_by_name(obj_id, ".", name, H5P_DEFAULT);
1,356,599✔
24
  return out > 0;
1,356,599✔
25
}
26

27
size_t attribute_typesize(hid_t obj_id, const char* name)
15,624,262✔
28
{
29
  hid_t attr = H5Aopen(obj_id, name, H5P_DEFAULT);
15,624,262✔
30
  hid_t filetype = H5Aget_type(attr);
15,624,262✔
31
  size_t n = H5Tget_size(filetype);
15,624,262✔
32
  H5Tclose(filetype);
15,624,262✔
33
  H5Aclose(attr);
15,624,262✔
34
  return n;
15,624,262✔
35
}
36

UNCOV
37
void get_shape(hid_t obj_id, hsize_t* dims)
×
38
{
UNCOV
39
  auto type = H5Iget_type(obj_id);
×
40
  hid_t dspace;
UNCOV
41
  if (type == H5I_DATASET) {
×
42
    dspace = H5Dget_space(obj_id);
×
43
  } else if (type == H5I_ATTR) {
×
44
    dspace = H5Aget_space(obj_id);
×
45
  } else {
UNCOV
46
    throw std::runtime_error {
×
47
      "Expected dataset or attribute in call to get_shape."};
×
48
  }
UNCOV
49
  H5Sget_simple_extent_dims(dspace, dims, nullptr);
×
50
  H5Sclose(dspace);
×
51
}
×
52

53
vector<hsize_t> attribute_shape(hid_t obj_id, const char* name)
12,420,831✔
54
{
55
  hid_t attr = H5Aopen(obj_id, name, H5P_DEFAULT);
12,420,831✔
56
  vector<hsize_t> shape = object_shape(attr);
12,420,831✔
57
  H5Aclose(attr);
12,420,831✔
58
  return shape;
12,420,831✔
UNCOV
59
}
×
60

61
vector<hsize_t> object_shape(hid_t obj_id)
23,515,994✔
62
{
63
  // Get number of dimensions
64
  auto type = H5Iget_type(obj_id);
23,515,994✔
65
  hid_t dspace;
66
  if (type == H5I_DATASET) {
23,515,994✔
67
    dspace = H5Dget_space(obj_id);
11,095,163✔
68
  } else if (type == H5I_ATTR) {
12,420,831!
69
    dspace = H5Aget_space(obj_id);
12,420,831✔
70
  } else {
UNCOV
71
    throw std::runtime_error {
×
72
      "Expected dataset or attribute in call to object_shape."};
×
73
  }
74
  int n = H5Sget_simple_extent_ndims(dspace);
23,515,994✔
75

76
  // Get shape of array
77
  vector<hsize_t> shape(n);
23,515,994✔
78
  H5Sget_simple_extent_dims(dspace, shape.data(), nullptr);
23,515,994✔
79

80
  // Free resources and return
81
  H5Sclose(dspace);
23,515,994✔
82
  return shape;
23,515,994✔
UNCOV
83
}
×
84

UNCOV
85
void get_shape_attr(hid_t obj_id, const char* name, hsize_t* dims)
×
86
{
UNCOV
87
  hid_t attr = H5Aopen(obj_id, name, H5P_DEFAULT);
×
88
  hid_t dspace = H5Aget_space(attr);
×
89
  H5Sget_simple_extent_dims(dspace, dims, nullptr);
×
90
  H5Sclose(dspace);
×
91
  H5Aclose(attr);
×
92
}
×
93

94
hid_t create_group(hid_t parent_id, const char* name)
199,767✔
95
{
96
  hid_t out = H5Gcreate(parent_id, name, H5P_DEFAULT, H5P_DEFAULT, H5P_DEFAULT);
199,767✔
97
  if (out < 0) {
199,767!
UNCOV
98
    fatal_error(fmt::format("Failed to create HDF5 group \"{}\"", name));
×
99
  }
100
  return out;
199,767✔
101
}
102

103
hid_t create_group(hid_t parent_id, const std::string& name)
120,793✔
104
{
105
  return create_group(parent_id, name.c_str());
120,793✔
106
}
107

108
void close_dataset(hid_t dataset_id)
7,929,048✔
109
{
110
  if (H5Dclose(dataset_id) < 0)
7,929,048!
UNCOV
111
    fatal_error("Failed to close dataset");
×
112
}
7,929,048✔
113

114
void close_group(hid_t group_id)
14,983,780✔
115
{
116
  if (H5Gclose(group_id) < 0)
14,983,780!
UNCOV
117
    fatal_error("Failed to close group");
×
118
}
14,983,780✔
119

120
void close_object(hid_t obj_id)
3,190,981✔
121
{
122
  if (H5Oclose(obj_id) < 0)
3,190,981!
UNCOV
123
    fatal_error("Failed to close object");
×
124
}
3,190,981✔
125

126
int dataset_ndims(hid_t dset)
70✔
127
{
128
  hid_t dspace = H5Dget_space(dset);
70✔
129
  int ndims = H5Sget_simple_extent_ndims(dspace);
70✔
130
  H5Sclose(dspace);
70✔
131
  return ndims;
70✔
132
}
133

134
size_t dataset_typesize(hid_t obj_id, const char* name)
156✔
135
{
136
  hid_t dset = open_dataset(obj_id, name);
156✔
137
  hid_t filetype = H5Dget_type(dset);
156✔
138
  size_t n = H5Tget_size(filetype);
156✔
139
  H5Tclose(filetype);
156✔
140
  close_dataset(dset);
156✔
141
  return n;
156✔
142
}
143

144
void ensure_exists(hid_t obj_id, const char* name, bool attribute)
26,019,390✔
145
{
146
  if (attribute) {
26,019,390✔
147
    if (!attribute_exists(obj_id, name)) {
2,398!
UNCOV
148
      fatal_error(fmt::format("Attribute \"{}\" does not exist in object {}",
×
149
        name, object_name(obj_id)));
×
150
    }
151
  } else {
152
    if (!object_exists(obj_id, name)) {
26,016,992!
UNCOV
153
      fatal_error(fmt::format("Object \"{}\" does not exist in object {}", name,
×
154
        object_name(obj_id)));
×
155
    }
156
  }
157
}
26,019,390✔
158

159
hid_t file_open(const char* filename, char mode, bool parallel)
52,288✔
160
{
161
  bool create;
162
  unsigned int flags;
163
  switch (mode) {
52,288!
164
  case 'r':
37,090✔
165
  case 'a':
166
    create = false;
37,090✔
167
    flags = (mode == 'r' ? H5F_ACC_RDONLY : H5F_ACC_RDWR);
37,090✔
168
    break;
37,090✔
169
  case 'w':
15,198✔
170
  case 'x':
171
    create = true;
15,198✔
172
    flags = (mode == 'x' ? H5F_ACC_EXCL : H5F_ACC_TRUNC);
15,198!
173
    break;
15,198✔
UNCOV
174
  default:
×
175
    fatal_error(fmt::format("Invalid file mode: ", mode));
×
176
  }
177

178
  hid_t plist = H5P_DEFAULT;
52,288✔
179
#ifdef PHDF5
180
  if (parallel) {
23,796✔
181
    // Setup file access property list with parallel I/O access
182
    plist = H5Pcreate(H5P_FILE_ACCESS);
3,291✔
183
    H5Pset_fapl_mpio(plist, openmc::mpi::intracomm, MPI_INFO_NULL);
3,291✔
184
  }
185
#endif
186

187
  // Open the file collectively
188
  hid_t file_id;
189
  if (create) {
52,288✔
190
    file_id = H5Fcreate(filename, flags, H5P_DEFAULT, plist);
15,198✔
191
  } else {
192
    file_id = H5Fopen(filename, flags, plist);
37,090✔
193
  }
194
  if (file_id < 0) {
52,288!
UNCOV
195
    fatal_error(fmt::format(
×
196
      "Failed to open HDF5 file with mode '{}': {}", mode, filename));
197
  }
198

199
#ifdef PHDF5
200
  // Close the property list
201
  if (parallel)
23,796✔
202
    H5Pclose(plist);
3,291✔
203
#endif
204

205
  return file_id;
52,288✔
206
}
207

208
hid_t file_open(const std::string& filename, char mode, bool parallel)
50,684✔
209
{
210
  return file_open(filename.c_str(), mode, parallel);
50,684✔
211
}
212

213
hid_t open_group(hid_t group_id, const std::string& name)
260✔
214
{
215
  return open_group(group_id, name.c_str());
260✔
216
}
217

218
void file_close(hid_t file_id)
52,270✔
219
{
220
  H5Fclose(file_id);
52,270✔
221
}
52,270✔
222

223
void get_name(hid_t obj_id, std::string& name)
2,999✔
224
{
225
  size_t size = 1 + H5Iget_name(obj_id, nullptr, 0);
2,999✔
226
  name.resize(size);
2,999✔
227
  H5Iget_name(obj_id, name.data(), size);
2,999✔
228
}
2,999✔
229

230
int get_num_datasets(hid_t group_id)
2,999✔
231
{
232
  // Determine number of links in the group
233
  H5G_info_t info;
234
  H5Gget_info(group_id, &info);
2,999✔
235

236
  // Iterate over links to get number of groups
237
  H5O_info_t oinfo;
238
  int ndatasets = 0;
2,999✔
239
  for (hsize_t i = 0; i < info.nlinks; ++i) {
7,006✔
240
    // Determine type of object (and skip non-group)
241
    H5Oget_info_by_idx(
4,007✔
242
      group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i, &oinfo, H5P_DEFAULT);
243
    if (oinfo.type == H5O_TYPE_DATASET)
4,007!
244
      ndatasets += 1;
4,007✔
245
  }
246

247
  return ndatasets;
2,999✔
248
}
249

UNCOV
250
int get_num_groups(hid_t group_id)
×
251
{
252
  // Determine number of links in the group
253
  H5G_info_t info;
UNCOV
254
  H5Gget_info(group_id, &info);
×
255

256
  // Iterate over links to get number of groups
257
  H5O_info_t oinfo;
UNCOV
258
  int ngroups = 0;
×
259
  for (hsize_t i = 0; i < info.nlinks; ++i) {
×
260
    // Determine type of object (and skip non-group)
UNCOV
261
    H5Oget_info_by_idx(
×
262
      group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i, &oinfo, H5P_DEFAULT);
UNCOV
263
    if (oinfo.type == H5O_TYPE_GROUP)
×
264
      ngroups += 1;
×
265
  }
266

UNCOV
267
  return ngroups;
×
268
}
269

270
void get_datasets(hid_t group_id, char* name[])
2,999✔
271
{
272
  // Determine number of links in the group
273
  H5G_info_t info;
274
  H5Gget_info(group_id, &info);
2,999✔
275

276
  // Iterate over links to get names
277
  H5O_info_t oinfo;
278
  hsize_t count = 0;
2,999✔
279
  size_t size;
280
  for (hsize_t i = 0; i < info.nlinks; ++i) {
7,006✔
281
    // Determine type of object (and skip non-group)
282
    H5Oget_info_by_idx(
4,007✔
283
      group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i, &oinfo, H5P_DEFAULT);
284
    if (oinfo.type != H5O_TYPE_DATASET)
4,007!
UNCOV
285
      continue;
×
286

287
    // Get size of name
288
    size = 1 + H5Lget_name_by_idx(group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i,
4,007✔
289
                 nullptr, 0, H5P_DEFAULT);
290

291
    // Read name
292
    H5Lget_name_by_idx(group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i,
4,007✔
293
      name[count], size, H5P_DEFAULT);
4,007✔
294
    count += 1;
4,007✔
295
  }
296
}
2,999✔
297

UNCOV
298
void get_groups(hid_t group_id, char* name[])
×
299
{
300
  // Determine number of links in the group
301
  H5G_info_t info;
UNCOV
302
  H5Gget_info(group_id, &info);
×
303

304
  // Iterate over links to get names
305
  H5O_info_t oinfo;
UNCOV
306
  hsize_t count = 0;
×
307
  size_t size;
UNCOV
308
  for (hsize_t i = 0; i < info.nlinks; ++i) {
×
309
    // Determine type of object (and skip non-group)
UNCOV
310
    H5Oget_info_by_idx(
×
311
      group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i, &oinfo, H5P_DEFAULT);
UNCOV
312
    if (oinfo.type != H5O_TYPE_GROUP)
×
313
      continue;
×
314

315
    // Get size of name
UNCOV
316
    size = 1 + H5Lget_name_by_idx(group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i,
×
317
                 nullptr, 0, H5P_DEFAULT);
318

319
    // Read name
UNCOV
320
    H5Lget_name_by_idx(group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i,
×
321
      name[count], size, H5P_DEFAULT);
×
322
    count += 1;
×
323
  }
UNCOV
324
}
×
325

326
vector<std::string> member_names(hid_t group_id, H5O_type_t type)
1,277,448✔
327
{
328
  // Determine number of links in the group
329
  H5G_info_t info;
330
  H5Gget_info(group_id, &info);
1,277,448✔
331

332
  // Iterate over links to get names
333
  H5O_info_t oinfo;
334
  size_t size;
335
  vector<std::string> names;
1,277,448✔
336
  for (hsize_t i = 0; i < info.nlinks; ++i) {
6,949,969✔
337
    // Determine type of object (and skip non-group)
338
    H5Oget_info_by_idx(
5,672,521✔
339
      group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i, &oinfo, H5P_DEFAULT);
340
    if (oinfo.type != type)
5,672,521!
UNCOV
341
      continue;
×
342

343
    // Get size of name
344
    size = 1 + H5Lget_name_by_idx(group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i,
5,672,521✔
345
                 nullptr, 0, H5P_DEFAULT);
346

347
    // Read name
348
    char* buffer = new char[size];
5,672,521✔
349
    H5Lget_name_by_idx(
5,672,521✔
350
      group_id, ".", H5_INDEX_NAME, H5_ITER_INC, i, buffer, size, H5P_DEFAULT);
351
    names.emplace_back(&buffer[0]);
5,672,521✔
352
    delete[] buffer;
5,672,521!
353
  }
354
  return names;
2,554,896✔
UNCOV
355
}
×
356

357
vector<std::string> group_names(hid_t group_id)
1,249,838✔
358
{
359
  return member_names(group_id, H5O_TYPE_GROUP);
1,249,838✔
360
}
361

362
vector<std::string> dataset_names(hid_t group_id)
27,610✔
363
{
364
  return member_names(group_id, H5O_TYPE_DATASET);
27,610✔
365
}
366

367
bool object_exists(hid_t object_id, const char* name)
32,258,771✔
368
{
369
  htri_t out = H5LTpath_valid(object_id, name, true);
32,258,771✔
370
  if (out < 0) {
32,258,771!
UNCOV
371
    fatal_error(fmt::format("Failed to check if object \"{}\" exists.", name));
×
372
  }
373
  return (out > 0);
32,258,771✔
374
}
375

376
std::string object_name(hid_t obj_id)
30,537✔
377
{
378
  // Determine size and create buffer
379
  size_t size = 1 + H5Iget_name(obj_id, nullptr, 0);
30,537✔
380
  char* buffer = new char[size];
30,537✔
381

382
  // Read and return name
383
  H5Iget_name(obj_id, buffer, size);
30,537✔
384
  std::string str = buffer;
30,537✔
385
  delete[] buffer;
30,537!
386
  return str;
30,537✔
387
}
388

389
hid_t open_dataset(hid_t group_id, const char* name)
7,998,881✔
390
{
391
  ensure_exists(group_id, name);
7,998,881✔
392
  return H5Dopen(group_id, name, H5P_DEFAULT);
7,998,881✔
393
}
394

395
hid_t open_group(hid_t group_id, const char* name)
14,827,130✔
396
{
397
  ensure_exists(group_id, name);
14,827,130✔
398
  return H5Gopen(group_id, name, H5P_DEFAULT);
14,827,130✔
399
}
400

401
hid_t open_object(hid_t group_id, const std::string& name)
3,190,981✔
402
{
403
  ensure_exists(group_id, name.c_str());
3,190,981✔
404
  return H5Oopen(group_id, name.c_str(), H5P_DEFAULT);
3,190,981✔
405
}
406

407
void read_attr(hid_t obj_id, const char* name, hid_t mem_type_id, void* buffer)
45,235,455✔
408
{
409
  hid_t attr = H5Aopen(obj_id, name, H5P_DEFAULT);
45,235,455✔
410
  H5Aread(attr, mem_type_id, buffer);
45,235,455✔
411
  H5Aclose(attr);
45,235,455✔
412
}
45,235,455✔
413

414
void read_attr_double(hid_t obj_id, const char* name, double* buffer)
112✔
415
{
416
  read_attr(obj_id, name, H5T_NATIVE_DOUBLE, buffer);
112✔
417
}
112✔
418

419
void read_attr_int(hid_t obj_id, const char* name, int* buffer)
6,054✔
420
{
421
  read_attr(obj_id, name, H5T_NATIVE_INT, buffer);
6,054✔
422
}
6,054✔
423

424
void read_attr_string(hid_t obj_id, const char* name, size_t slen, char* buffer)
15,624,262✔
425
{
426
  // Create datatype for a string
427
  hid_t datatype = H5Tcopy(H5T_C_S1);
15,624,262✔
428
  H5Tset_size(datatype, slen);
15,624,262✔
429
  // numpy uses null-padding when writing fixed-length strings
430
  H5Tset_strpad(datatype, H5T_STR_NULLPAD);
15,624,262✔
431

432
  // Read data into buffer
433
  read_attr(obj_id, name, datatype, buffer);
15,624,262✔
434

435
  // Free resources
436
  H5Tclose(datatype);
15,624,262✔
437
}
15,624,262✔
438

439
void read_dataset_lowlevel(hid_t obj_id, const char* name, hid_t mem_type_id,
11,181,590✔
440
  hid_t mem_space_id, bool indep, void* buffer)
441
{
442
  hid_t dset = obj_id;
11,181,590✔
443
  if (name)
11,181,590✔
444
    dset = open_dataset(obj_id, name);
69,833✔
445

446
  if (using_mpio_device(dset)) {
11,181,590✔
447
#ifdef PHDF5
448
    // Set up collective vs independent I/O
449
    auto data_xfer_mode = indep ? H5FD_MPIO_INDEPENDENT : H5FD_MPIO_COLLECTIVE;
552!
450

451
    // Create dataset transfer property list
452
    hid_t plist = H5Pcreate(H5P_DATASET_XFER);
552✔
453
    H5Pset_dxpl_mpio(plist, data_xfer_mode);
552✔
454

455
    // Read data
456
    H5Dread(dset, mem_type_id, mem_space_id, H5S_ALL, plist, buffer);
552✔
457
    H5Pclose(plist);
552✔
458
#endif
459
  } else {
460
    H5Dread(dset, mem_type_id, mem_space_id, H5S_ALL, H5P_DEFAULT, buffer);
11,181,038✔
461
  }
462

463
  if (name)
11,181,590✔
464
    H5Dclose(dset);
69,833✔
465
}
11,181,590✔
466

467
template<>
468
void read_dataset(
1,104✔
469
  hid_t dset, tensor::Tensor<std::complex<double>>& tensor, bool indep)
470
{
471
  // Get shape of dataset
472
  vector<hsize_t> shape = object_shape(dset);
1,104✔
473

474
  // Resize tensor and read data directly
475
  vector<size_t> tshape(shape.begin(), shape.end());
1,104✔
476
  tensor.resize(tshape);
1,104✔
477

478
  // Read data from dataset
479
  read_complex(dset, nullptr,
1,104✔
480
    reinterpret_cast<std::complex<double>*>(tensor.data()), indep);
481
}
1,104✔
482

483
void read_double(hid_t obj_id, const char* name, double* buffer, bool indep)
7,146✔
484
{
485
  read_dataset_lowlevel(
7,146✔
486
    obj_id, name, H5T_NATIVE_DOUBLE, H5S_ALL, indep, buffer);
7,146✔
487
}
7,146✔
488

UNCOV
489
void read_int(hid_t obj_id, const char* name, int* buffer, bool indep)
×
490
{
UNCOV
491
  read_dataset_lowlevel(obj_id, name, H5T_NATIVE_INT, H5S_ALL, indep, buffer);
×
UNCOV
492
}
×
493

UNCOV
494
void read_llong(hid_t obj_id, const char* name, long long* buffer, bool indep)
×
495
{
496
  read_dataset_lowlevel(obj_id, name, H5T_NATIVE_LLONG, H5S_ALL, indep, buffer);
×
UNCOV
497
}
×
498

499
void read_string(
156✔
500
  hid_t obj_id, const char* name, size_t slen, char* buffer, bool indep)
501
{
502
  // Create datatype for a string
503
  hid_t datatype = H5Tcopy(H5T_C_S1);
156✔
504
  H5Tset_size(datatype, slen);
156✔
505
  // numpy uses null-padding when writing fixed-length strings
506
  H5Tset_strpad(datatype, H5T_STR_NULLPAD);
156✔
507

508
  // Read data into buffer
509
  read_dataset_lowlevel(obj_id, name, datatype, H5S_ALL, indep, buffer);
156✔
510

511
  // Free resources
512
  H5Tclose(datatype);
156✔
513
}
156✔
514

515
void read_complex(
1,104✔
516
  hid_t obj_id, const char* name, std::complex<double>* buffer, bool indep)
517
{
518
  // Create compound datatype for complex numbers
519
  struct complex_t {
520
    double re;
521
    double im;
522
  };
523
  complex_t tmp;
524
  hid_t complex_id = H5Tcreate(H5T_COMPOUND, sizeof tmp);
1,104✔
525
  H5Tinsert(complex_id, "r", HOFFSET(complex_t, re), H5T_NATIVE_DOUBLE);
1,104✔
526
  H5Tinsert(complex_id, "i", HOFFSET(complex_t, im), H5T_NATIVE_DOUBLE);
1,104✔
527

528
  // Read data
529
  read_dataset_lowlevel(obj_id, name, complex_id, H5S_ALL, indep, buffer);
1,104✔
530

531
  // Free resources
532
  H5Tclose(complex_id);
1,104✔
533
}
1,104✔
534

535
void read_tally_results(hid_t group_id, hsize_t n_filter, hsize_t n_score,
142✔
536
  hsize_t n_results, double* results)
537
{
538
  // Create dataspace for hyperslab in memory
539
  constexpr int ndim = 3;
142✔
540
  hsize_t dims[ndim] {n_filter, n_score, n_results};
142✔
541
  hsize_t start[ndim] {0, 0, 1};
142✔
542
  hsize_t count[ndim] {n_filter, n_score, n_results - 1};
142✔
543
  hid_t memspace = H5Screate_simple(ndim, dims, nullptr);
142✔
544
  H5Sselect_hyperslab(memspace, H5S_SELECT_SET, start, nullptr, count, nullptr);
142✔
545

546
  // Read the dataset
547
  read_dataset_lowlevel(
142✔
548
    group_id, "results", H5T_NATIVE_DOUBLE, memspace, false, results);
142✔
549

550
  // Free resources
551
  H5Sclose(memspace);
142✔
552
}
142✔
553

554
void write_attr(hid_t obj_id, int ndim, const hsize_t* dims, const char* name,
227,201✔
555
  hid_t mem_type_id, const void* buffer)
556
{
557
  // If array is given, create a simple dataspace. Otherwise, create a scalar
558
  // datascape.
559
  hid_t dspace;
560
  if (ndim > 0) {
227,201✔
561
    dspace = H5Screate_simple(ndim, dims, nullptr);
42,472✔
562
  } else {
563
    dspace = H5Screate(H5S_SCALAR);
184,729✔
564
  }
565

566
  // Create attribute and Write data
567
  hid_t attr =
568
    H5Acreate(obj_id, name, mem_type_id, dspace, H5P_DEFAULT, H5P_DEFAULT);
227,201✔
569
  H5Awrite(attr, mem_type_id, buffer);
227,201✔
570

571
  // Free resources
572
  H5Aclose(attr);
227,201✔
573
  H5Sclose(dspace);
227,201✔
574
}
227,201✔
575

UNCOV
576
void write_attr_double(hid_t obj_id, int ndim, const hsize_t* dims,
×
577
  const char* name, const double* buffer)
578
{
UNCOV
579
  write_attr(obj_id, ndim, dims, name, H5T_NATIVE_DOUBLE, buffer);
×
580
}
×
581

UNCOV
582
void write_attr_int(hid_t obj_id, int ndim, const hsize_t* dims,
×
583
  const char* name, const int* buffer)
584
{
UNCOV
585
  write_attr(obj_id, ndim, dims, name, H5T_NATIVE_INT, buffer);
×
586
}
×
587

588
void write_attr_string(hid_t obj_id, const char* name, const char* buffer)
35,343✔
589
{
590
  size_t n = strlen(buffer);
35,343✔
591
  if (n > 0) {
35,343✔
592
    // Set up appropriate datatype for a fixed-length string
593
    hid_t datatype = H5Tcopy(H5T_C_S1);
28,622✔
594
    H5Tset_size(datatype, n);
28,622✔
595

596
    write_attr(obj_id, 0, nullptr, name, datatype, buffer);
28,622✔
597

598
    // Free resources
599
    H5Tclose(datatype);
28,622✔
600
  }
601
}
35,343✔
602

603
void write_dataset_lowlevel(hid_t group_id, int ndim, const hsize_t* dims,
780,246✔
604
  const char* name, hid_t mem_type_id, hid_t mem_space_id, bool indep,
605
  const void* buffer)
606
{
607
  // If array is given, create a simple dataspace. Otherwise, create a scalar
608
  // datascape.
609
  hid_t dspace;
610
  if (ndim > 0) {
780,246✔
611
    dspace = H5Screate_simple(ndim, dims, nullptr);
243,779✔
612
  } else {
613
    dspace = H5Screate(H5S_SCALAR);
536,467✔
614
  }
615

616
  hid_t dset = H5Dcreate(
780,246✔
617
    group_id, name, mem_type_id, dspace, H5P_DEFAULT, H5P_DEFAULT, H5P_DEFAULT);
618

619
  if (using_mpio_device(group_id)) {
780,246✔
620
#ifdef PHDF5
621
    // Set up collective vs independent I/O
622
    auto data_xfer_mode = indep ? H5FD_MPIO_INDEPENDENT : H5FD_MPIO_COLLECTIVE;
×
623

624
    // Create dataset transfer property list
625
    hid_t plist = H5Pcreate(H5P_DATASET_XFER);
626
    H5Pset_dxpl_mpio(plist, data_xfer_mode);
627

628
    // Write data
629
    H5Dwrite(dset, mem_type_id, mem_space_id, H5S_ALL, plist, buffer);
630
    H5Pclose(plist);
631
#endif
632
  } else {
633
    H5Dwrite(dset, mem_type_id, mem_space_id, H5S_ALL, H5P_DEFAULT, buffer);
780,246✔
634
  }
635

636
  // Free resources
637
  H5Dclose(dset);
780,246✔
638
  H5Sclose(dspace);
780,246✔
639
}
780,246✔
640

UNCOV
641
void write_double(hid_t group_id, int ndim, const hsize_t* dims,
×
642
  const char* name, const double* buffer, bool indep)
643
{
UNCOV
644
  write_dataset_lowlevel(
×
645
    group_id, ndim, dims, name, H5T_NATIVE_DOUBLE, H5S_ALL, indep, buffer);
×
UNCOV
646
}
×
647

648
void write_int(hid_t group_id, int ndim, const hsize_t* dims, const char* name,
1,303✔
649
  const int* buffer, bool indep)
650
{
651
  write_dataset_lowlevel(
1,303✔
652
    group_id, ndim, dims, name, H5T_NATIVE_INT, H5S_ALL, indep, buffer);
1,303✔
653
}
1,303✔
654

UNCOV
655
void write_llong(hid_t group_id, int ndim, const hsize_t* dims,
×
656
  const char* name, const long long* buffer, bool indep)
657
{
UNCOV
658
  write_dataset_lowlevel(
×
659
    group_id, ndim, dims, name, H5T_NATIVE_LLONG, H5S_ALL, indep, buffer);
×
UNCOV
660
}
×
661

662
void write_string(hid_t group_id, int ndim, const hsize_t* dims, size_t slen,
347,300✔
663
  const char* name, const char* buffer, bool indep)
664
{
665
  if (slen > 0) {
347,300✔
666
    // Set up appropriate datatype for a fixed-length string
667
    hid_t datatype = H5Tcopy(H5T_C_S1);
312,111✔
668
    H5Tset_size(datatype, slen);
312,111✔
669

670
    write_dataset_lowlevel(
312,111✔
671
      group_id, ndim, dims, name, datatype, H5S_ALL, indep, buffer);
672

673
    // Free resources
674
    H5Tclose(datatype);
312,111✔
675
  }
676
}
347,300✔
677

678
void write_string(
290,337✔
679
  hid_t group_id, const char* name, const std::string& buffer, bool indep)
680
{
681
  write_string(
290,337✔
682
    group_id, 0, nullptr, buffer.length(), name, buffer.c_str(), indep);
683
}
290,337✔
684

685
void write_tally_results(hid_t group_id, hsize_t n_filter, hsize_t n_score,
19,106✔
686
  hsize_t n_results, const double* results)
687
{
688
  // Set dimensions of sum/sum_sq hyperslab to store
689
  constexpr int ndim = 3;
19,106✔
690
  hsize_t count[ndim] {n_filter, n_score, n_results - 1};
19,106✔
691

692
  // Set dimensions of results array
693
  hsize_t dims[ndim] {n_filter, n_score, n_results};
19,106✔
694
  hsize_t start[ndim] {0, 0, 1};
19,106✔
695
  hid_t memspace = H5Screate_simple(ndim, dims, nullptr);
19,106✔
696
  H5Sselect_hyperslab(memspace, H5S_SELECT_SET, start, nullptr, count, nullptr);
19,106✔
697

698
  // Create and write dataset
699
  write_dataset_lowlevel(group_id, ndim, count, "results", H5T_NATIVE_DOUBLE,
19,106✔
700
    memspace, false, results);
701

702
  // Free resources
703
  H5Sclose(memspace);
19,106✔
704
}
19,106✔
705

706
bool using_mpio_device(hid_t obj_id)
11,961,836✔
707
{
708
  // Determine file that this object is part of
709
  hid_t file_id = H5Iget_file_id(obj_id);
11,961,836✔
710

711
  // Get file access property list
712
  hid_t fapl_id = H5Fget_access_plist(file_id);
11,961,836✔
713

714
  // Get low-level driver identifier
715
  hid_t driver = H5Pget_driver(fapl_id);
11,961,836✔
716

717
  // Free resources
718
  H5Pclose(fapl_id);
11,961,836✔
719
  H5Fclose(file_id);
11,961,836✔
720

721
  return driver == H5FD_MPIO;
11,961,836✔
722
}
723

724
// Specializations of the H5TypeMap template struct
725
template<>
726
const hid_t H5TypeMap<bool>::type_id = H5T_NATIVE_INT8;
727
template<>
728
const hid_t H5TypeMap<int>::type_id = H5T_NATIVE_INT;
729
template<>
730
const hid_t H5TypeMap<unsigned long>::type_id = H5T_NATIVE_ULONG;
731
template<>
732
const hid_t H5TypeMap<unsigned long long>::type_id = H5T_NATIVE_ULLONG;
733
template<>
734
const hid_t H5TypeMap<unsigned int>::type_id = H5T_NATIVE_UINT;
735
template<>
736
const hid_t H5TypeMap<int64_t>::type_id = H5T_NATIVE_INT64;
737
template<>
738
const hid_t H5TypeMap<double>::type_id = H5T_NATIVE_DOUBLE;
739
template<>
740
const hid_t H5TypeMap<char>::type_id = H5T_NATIVE_CHAR;
741

742
} // 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