Loading [MathJax]/extensions/tex2jax.js
PISM, A Parallel Ice Sheet Model 2.2.2-d6b3a29ca committed by Constantine Khrulev on 2025-03-28
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
PNCFile.cc
Go to the documentation of this file.
1// Copyright (C) 2012, 2013, 2014, 2015, 2016, 2017, 2019, 2020, 2021, 2023, 2024 PISM Authors
2//
3// This file is part of PISM.
4//
5// PISM is free software; you can redistribute it and/or modify it under the
6// terms of the GNU General Public License as published by the Free Software
7// Foundation; either version 3 of the License, or (at your option) any later
8// version.
9//
10// PISM is distributed in the hope that it will be useful, but WITHOUT ANY
11// WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
12// FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
13// details.
14//
15// You should have received a copy of the GNU General Public License
16// along with PISM; if not, write to the Free Software
17// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18
19#include <pnetcdf.h>
20#include <sstream>
21#include <cstring> // memset
22
23#include "pism/util/io/PNCFile.hh"
24
25#include "pism/util/io/pism_type_conversion.hh" // has to go after pnetcdf.h
26
27#include "pism/util/error_handling.hh"
28#include "pism/util/pism_utilities.hh"
29
30namespace pism {
31namespace io {
32
34 : NCFile(c) {
35 MPI_Info_create(&m_mpi_info);
36}
37
38
40 MPI_Info_free(&m_mpi_info);
41}
42
43static void check(const ErrorLocation &where, int return_code) {
44 if (return_code != NC_NOERR) {
45 throw RuntimeError(where, ncmpi_strerror(return_code));
46 }
47}
48
49void PNCFile::open_impl(const std::string &fname, io::Mode mode) {
50 int stat;
51
52 init_hints();
53
54 int open_mode = mode == io::PISM_READONLY ? NC_NOWRITE : NC_WRITE;
55
56 stat = ncmpi_open(m_com, fname.c_str(), open_mode, m_mpi_info, &m_file_id);
58}
59
60
61void PNCFile::create_impl(const std::string &fname) {
62 int stat;
63
64 init_hints();
65
66 stat = ncmpi_create(m_com, fname.c_str(), NC_CLOBBER | NC_64BIT_DATA, m_mpi_info, &m_file_id);
68}
69
70void PNCFile::sync_impl() const {
71
72 int stat = ncmpi_sync(m_file_id);
74}
75
76
78 int stat = ncmpi_close(m_file_id);
80
81 m_file_id = -1;
82}
83
84
86
87 int stat = ncmpi_enddef(m_file_id);
89}
90
91
92void PNCFile::redef_impl() const {
93
94 int stat = ncmpi_redef(m_file_id);
96}
97
98
99void PNCFile::def_dim_impl(const std::string &name, size_t length) const {
100 int dimid = 0, stat;
101
102 stat = ncmpi_def_dim(m_file_id, name.c_str(), length, &dimid);
104}
105
106
107void PNCFile::inq_dimid_impl(const std::string &dimension_name, bool &exists) const {
108 int tmp, stat;
109
110 stat = ncmpi_inq_dimid(m_file_id, dimension_name.c_str(), &tmp);
111
112 if (stat == NC_NOERR) {
113 exists = true;
114 } else {
115 exists = false;
116 }
117}
118
119
120void PNCFile::inq_dimlen_impl(const std::string &dimension_name, unsigned int &result) const {
121 int stat, dimid = -1;
122 MPI_Offset len;
123
124 stat = ncmpi_inq_dimid(m_file_id, dimension_name.c_str(), &dimid);
126
127 stat = ncmpi_inq_dimlen(m_file_id, dimid, &len);
129
130 result = static_cast<unsigned int>(len);
131}
132
133
134void PNCFile::inq_unlimdim_impl(std::string &result) const {
135 int stat = NC_NOERR, dimid = -1;
136 char dimname[NC_MAX_NAME];
137
138 stat = ncmpi_inq_unlimdim(m_file_id, &dimid);
140
141 if (dimid == -1) {
142 result.clear();
143 } else {
144 stat = ncmpi_inq_dimname(m_file_id, dimid, dimname);
146
147 result = dimname;
148 }
149}
150
151void PNCFile::def_var_impl(const std::string &name, io::Type nctype,
152 const std::vector<std::string> &dims) const {
153 std::vector<int> dimids;
154 int stat, varid;
155
156 for (auto d : dims) {
157 int dimid = -1;
158 stat = ncmpi_inq_dimid(m_file_id, d.c_str(), &dimid);
160 dimids.push_back(dimid);
161 }
162
163 stat = ncmpi_def_var(m_file_id, name.c_str(), pism_type_to_nc_type(nctype),
164 static_cast<int>(dims.size()), dimids.data(), &varid);
166}
167
168
169void PNCFile::get_vara_double_impl(const std::string &variable_name,
170 const std::vector<unsigned int> &start,
171 const std::vector<unsigned int> &count, double *ip) const {
172 return this->get_var_double(variable_name, start, count, ip);
173}
174
175
176void PNCFile::put_vara_double_impl(const std::string &variable_name,
177 const std::vector<unsigned int> &start,
178 const std::vector<unsigned int> &count, const double *op) const {
179 int stat, varid, ndims = static_cast<int>(start.size());
180
181 std::vector<MPI_Offset> nc_start(ndims), nc_count(ndims);
182
183 stat = ncmpi_inq_varid(m_file_id, variable_name.c_str(), &varid);
185
186 for (int j = 0; j < ndims; ++j) {
187 nc_start[j] = start[j];
188 nc_count[j] = count[j];
189 }
190
191 stat = ncmpi_put_vara_double_all(m_file_id, varid, nc_start.data(), nc_count.data(), op);
193}
194
195void PNCFile::inq_nvars_impl(int &result) const {
196 int stat;
197
198 stat = ncmpi_inq_nvars(m_file_id, &result);
200}
201
202
203void PNCFile::inq_vardimid_impl(const std::string &variable_name,
204 std::vector<std::string> &result) const {
205 int stat, ndims, varid = -1;
206 std::vector<int> dimids;
207
208 stat = ncmpi_inq_varid(m_file_id, variable_name.c_str(), &varid);
210
211 stat = ncmpi_inq_varndims(m_file_id, varid, &ndims);
213
214 if (ndims == 0) {
215 result.clear();
216 return;
217 }
218
219 result.resize(ndims);
220 dimids.resize(ndims);
221
222 stat = ncmpi_inq_vardimid(m_file_id, varid, dimids.data());
224
225 for (int k = 0; k < ndims; ++k) {
226 char name[NC_MAX_NAME];
227 memset(name, 0, NC_MAX_NAME);
228
229 stat = ncmpi_inq_dimname(m_file_id, dimids[k], name);
231
232 result[k] = name;
233 }
234}
235
236int PNCFile::get_varid(const std::string &variable_name) const {
237 if (variable_name == "PISM_GLOBAL") {
238 return NC_GLOBAL;
239 }
240
241 int varid = -1;
242 int stat = ncmpi_inq_varid(m_file_id, variable_name.c_str(), &varid);
244
245 return varid;
246}
247
248void PNCFile::inq_varnatts_impl(const std::string &variable_name, int &result) const {
249
250 int stat = ncmpi_inq_varnatts(m_file_id, get_varid(variable_name), &result);
252}
253
254
255void PNCFile::inq_varid_impl(const std::string &variable_name, bool &exists) const {
256 int stat, flag = -1;
257
258 stat = ncmpi_inq_varid(m_file_id, variable_name.c_str(), &flag);
259
260 exists = (stat == NC_NOERR);
261}
262
263
264void PNCFile::inq_varname_impl(unsigned int j, std::string &result) const {
265 int stat;
266 char varname[NC_MAX_NAME];
267 memset(varname, 0, NC_MAX_NAME);
268
269 stat = ncmpi_inq_varname(m_file_id, j, varname);
271
272 result = varname;
273}
274
275void PNCFile::get_att_double_impl(const std::string &variable_name, const std::string &att_name,
276 std::vector<double> &result) const {
277 int len, varid = get_varid(variable_name);
278 MPI_Offset attlen = 0;
279
280 // Read the attribute length:
281 int stat = ncmpi_inq_attlen(m_file_id, varid, att_name.c_str(), &attlen);
282
283 if (stat == NC_NOERR) {
284 len = static_cast<int>(attlen);
285 } else if (stat == NC_ENOTATT) {
286 len = 0;
287 } else {
289 len = 0;
290 }
291
292 if (len == 0) {
293 result.clear();
294 return;
295 }
296
297 result.resize(len);
298
299 // Now read data and broadcast stat to see if we succeeded:
300 stat = ncmpi_get_att_double(m_file_id, varid, att_name.c_str(), result.data());
302
303 // On error, print a message and stop.
304 if (stat != NC_NOERR) {
305 fprintf(stderr, "Error reading the %s attribute; (varid %d, NetCDF error %s)", att_name.c_str(),
306 varid, ncmpi_strerror(stat));
307 }
308}
309
310
311void PNCFile::get_att_text_impl(const std::string &variable_name, const std::string &att_name,
312 std::string &result) const {
313 int varid = get_varid(variable_name);
314
315 // Read the attribute length:
316 MPI_Offset attlen = 0;
317
318 int stat = ncmpi_inq_attlen(m_file_id, varid, att_name.c_str(), &attlen);
319 int len = (stat == NC_NOERR) ? static_cast<int>(attlen) : 0;
320
321 // Allocate some memory or set result to NULL and return:
322 if (len == 0) {
323 result.clear();
324 return;
325 }
326
327 std::vector<char> str(len + 1, 0);
328
329 // Now read the string and see if we succeeded:
330 stat = ncmpi_get_att_text(m_file_id, varid, att_name.c_str(), str.data());
332
333 result = str.data();
334}
335
336void PNCFile::del_att_impl(const std::string &variable_name, const std::string &att_name) const {
337 int stat = ncmpi_del_att(m_file_id, get_varid(variable_name), att_name.c_str());
339}
340
341void PNCFile::put_att_double_impl(const std::string &variable_name, const std::string &att_name,
342 io::Type nctype, const std::vector<double> &data) const {
343
344 int stat = ncmpi_put_att_double(m_file_id, get_varid(variable_name), att_name.c_str(),
345 pism_type_to_nc_type(nctype), data.size(), data.data());
347}
348
349
350void PNCFile::put_att_text_impl(const std::string &variable_name, const std::string &att_name,
351 const std::string &value) const {
352 int stat = ncmpi_put_att_text(m_file_id, get_varid(variable_name), att_name.c_str(), value.size(),
353 value.c_str());
355}
356
357
358void PNCFile::inq_attname_impl(const std::string &variable_name, unsigned int n,
359 std::string &result) const {
360 int stat;
361 char name[NC_MAX_NAME];
362 memset(name, 0, NC_MAX_NAME);
363
364 int varid = get_varid(variable_name);
365
366 stat = ncmpi_inq_attname(m_file_id, varid, n, name);
368
369 result = name;
370}
371
372
373void PNCFile::inq_atttype_impl(const std::string &variable_name, const std::string &att_name,
374 io::Type &result) const {
375 int varid = get_varid(variable_name);
376
377 nc_type tmp = NC_NAT;
378 int stat = ncmpi_inq_atttype(m_file_id, varid, att_name.c_str(), &tmp);
379 if (stat == NC_ENOTATT) {
380 tmp = NC_NAT;
381 } else {
383 }
384
385 result = nc_type_to_pism_type(tmp);
386}
387
388
389void PNCFile::set_fill_impl(int fillmode, int &old_modep) const {
390 int stat = ncmpi_set_fill(m_file_id, fillmode, &old_modep);
392}
393
394
395void PNCFile::get_var_double(const std::string &variable_name,
396 const std::vector<unsigned int> &start,
397 const std::vector<unsigned int> &count, double *ip) const {
398 int stat, varid, ndims = static_cast<int>(start.size());
399
400 std::vector<MPI_Offset> nc_start(ndims), nc_count(ndims);
401
402 stat = ncmpi_inq_varid(m_file_id, variable_name.c_str(), &varid);
404
405 for (int j = 0; j < ndims; ++j) {
406 nc_start[j] = start[j];
407 nc_count[j] = count[j];
408 }
409
410 stat = ncmpi_get_vara_double_all(m_file_id, varid, nc_start.data(), nc_count.data(), ip);
412}
413
415
416 for (auto hint : m_mpi_io_hints) {
417 auto words = split(hint, ':');
418
419 if (words.size() == 2) {
420 // printf("Setting MPI I/O hint \"%s\" to \"%s\"...\n",
421 // words[0].c_str(), words[1].c_str());
422
423 MPI_Info_set(m_mpi_info,
424 const_cast<char*>(words[0].c_str()),
425 const_cast<char*>(words[1].c_str()));
426 } else {
427 int rank = 0;
428 MPI_Comm_rank(m_com, &rank);
429 if (rank == 0) {
430 printf("PISM WARNING: invalid MPI I/O hint: %s. Ignoring it...\n",
431 hint.c_str());
432 }
433 }
434 }
435}
436
438 (void) level;
439 // NetCDF-3 does not support compression.
440}
441
442} // end of namespace io
443} // end of namespace pism
MPI_Comm m_com
Definition NCFile.hh:221
The PISM wrapper for a subset of the NetCDF C API.
Definition NCFile.hh:59
void def_dim_impl(const std::string &name, size_t length) const
Definition PNCFile.cc:99
PNCFile(MPI_Comm com)
Definition PNCFile.cc:33
void inq_unlimdim_impl(std::string &result) const
Definition PNCFile.cc:134
void get_vara_double_impl(const std::string &variable_name, const std::vector< unsigned int > &start, const std::vector< unsigned int > &count, double *ip) const
Definition PNCFile.cc:169
void inq_varid_impl(const std::string &variable_name, bool &exists) const
Definition PNCFile.cc:255
void close_impl()
Definition PNCFile.cc:77
void inq_varname_impl(unsigned int j, std::string &result) const
Definition PNCFile.cc:264
void inq_dimid_impl(const std::string &dimension_name, bool &exists) const
Definition PNCFile.cc:107
void inq_nvars_impl(int &result) const
Definition PNCFile.cc:195
void del_att_impl(const std::string &variable_name, const std::string &att_name) const
Definition PNCFile.cc:336
void enddef_impl() const
Definition PNCFile.cc:85
void put_vara_double_impl(const std::string &variable_name, const std::vector< unsigned int > &start, const std::vector< unsigned int > &count, const double *op) const
Definition PNCFile.cc:176
void inq_varnatts_impl(const std::string &variable_name, int &result) const
Definition PNCFile.cc:248
void inq_dimlen_impl(const std::string &dimension_name, unsigned int &result) const
Definition PNCFile.cc:120
std::vector< std::string > m_mpi_io_hints
Definition PNCFile.hh:112
void sync_impl() const
Definition PNCFile.cc:70
void put_att_double_impl(const std::string &variable_name, const std::string &att_name, io::Type xtype, const std::vector< double > &data) const
Definition PNCFile.cc:341
void inq_attname_impl(const std::string &variable_name, unsigned int n, std::string &result) const
Definition PNCFile.cc:358
void get_var_double(const std::string &variable_name, const std::vector< unsigned int > &start, const std::vector< unsigned int > &count, double *ip) const
Definition PNCFile.cc:395
void inq_atttype_impl(const std::string &variable_name, const std::string &att_name, io::Type &result) const
Definition PNCFile.cc:373
virtual ~PNCFile()
Definition PNCFile.cc:39
void inq_vardimid_impl(const std::string &variable_name, std::vector< std::string > &result) const
Definition PNCFile.cc:203
void get_att_text_impl(const std::string &variable_name, const std::string &att_name, std::string &result) const
Definition PNCFile.cc:311
MPI_Info m_mpi_info
Definition PNCFile.hh:121
void redef_impl() const
Definition PNCFile.cc:92
void set_fill_impl(int fillmode, int &old_modep) const
Definition PNCFile.cc:389
void set_compression_level_impl(int level) const
Definition PNCFile.cc:437
void create_impl(const std::string &filename)
Definition PNCFile.cc:61
void get_att_double_impl(const std::string &variable_name, const std::string &att_name, std::vector< double > &result) const
Definition PNCFile.cc:275
int get_varid(const std::string &variable_name) const
Definition PNCFile.cc:236
void open_impl(const std::string &filename, io::Mode mode)
Definition PNCFile.cc:49
void put_att_text_impl(const std::string &variable_name, const std::string &att_name, const std::string &value) const
Definition PNCFile.cc:350
void def_var_impl(const std::string &name, io::Type nctype, const std::vector< std::string > &dims) const
Definition PNCFile.cc:151
#define PISM_ERROR_LOCATION
#define n
Definition exactTestM.c:37
@ PISM_READONLY
open an existing file for reading only
Definition IO_Flags.hh:68
static void check(const ErrorLocation &where, int return_code)
Prints an error message; for debugging.
Definition NC4_Par.cc:36
std::string printf(const char *format,...)
static const double k
Definition exactTestP.cc:42
static pism::io::Type nc_type_to_pism_type(int input)
static nc_type pism_type_to_nc_type(pism::io::Type input)
std::vector< std::string > split(const std::string &input, char separator)
Transform a separator-separated list (a string) into a vector of strings.
int count
Definition test_cube.c:16