cosmotool/src/yorick_nc4.cpp

238 lines
5.5 KiB
C++
Raw Normal View History

2012-09-16 00:53:47 +02:00
#include "ctool_netcdf_ver.hpp"
2008-12-02 18:00:28 +01:00
#include "config.hpp"
2012-09-16 00:53:47 +02:00
#include <netcdf>
2012-09-19 15:15:09 +02:00
using namespace netCDF;
2008-12-02 18:00:28 +01:00
#include <fstream>
#include "yorick.hpp"
#include <assert.h>
using namespace CosmoTool;
using namespace std;
class NetCDF_handle
{
public:
NcFile *outFile;
2012-09-19 15:15:09 +02:00
NcVar curVar;
vector<size_t> curPos;
vector<size_t> counts;
vector<NcDim> dimList;
2008-12-02 18:00:28 +01:00
uint32_t rank;
2012-09-19 15:15:09 +02:00
NetCDF_handle(NcFile *f, NcVar v, vector<NcDim>& dimList, uint32_t rank);
2008-12-02 18:00:28 +01:00
virtual ~NetCDF_handle();
};
2012-09-19 15:15:09 +02:00
NetCDF_handle::NetCDF_handle(NcFile *f, NcVar v, vector<NcDim>& dimList, uint32_t rank)
2008-12-02 18:00:28 +01:00
{
this->outFile = f;
this->curVar = v;
this->dimList = dimList;
this->rank = rank;
2012-09-19 15:15:09 +02:00
this->counts.resize(rank);
this->curPos.resize(rank);
2008-12-02 18:00:28 +01:00
for (long i = 0; i < rank; i++)
this->curPos[i] = 0;
for (long i = 0; i < rank; i++)
this->counts[i] = 1;
}
NetCDF_handle::~NetCDF_handle()
{
delete outFile;
}
template<typename T>
class InputGenCDF: public NetCDF_handle, public ProgressiveInputImpl<T>
{
public:
2012-09-19 15:15:09 +02:00
InputGenCDF(NcFile *f, NcVar v, vector<NcDim>& dimList, uint32_t rank)
2008-12-02 18:00:28 +01:00
: NetCDF_handle(f,v,dimList,rank)
{}
virtual ~InputGenCDF() {}
virtual T read()
{
T a;
2012-09-19 15:15:09 +02:00
curVar.getVar(curPos, counts, &a);
2008-12-02 18:00:28 +01:00
curPos[rank-1]++;
for (long i = rank-1; i >= 1; i--)
{
2012-09-19 15:15:09 +02:00
if (curPos[i] == dimList[i].getSize())
2008-12-02 18:00:28 +01:00
{
curPos[i-1]++;
curPos[i] = 0;
}
}
return a;
}
virtual void seek(uint32_t *pos)
{
for (long i = rank-1; i >= 0; i--)
curPos[i] = pos[rank-1-i];
}
};
template<typename T>
class OutputGenCDF: public NetCDF_handle, public ProgressiveOutputImpl<T>
{
public:
2012-09-19 15:15:09 +02:00
OutputGenCDF(NcFile *f, NcVar v, vector<NcDim>& dimList, uint32_t rank)
2008-12-02 18:00:28 +01:00
: NetCDF_handle(f,v,dimList,rank)
{}
virtual ~OutputGenCDF() {}
virtual void put(T a)
{
2012-09-19 15:15:09 +02:00
curVar.putVar(curPos, counts, &a);
2008-12-02 18:00:28 +01:00
curPos[rank-1]++;
for (long i = rank-1; i >= 1; i--)
{
2012-09-19 15:15:09 +02:00
if (curPos[i] == dimList[i].getSize())
2008-12-02 18:00:28 +01:00
{
curPos[i-1]++;
curPos[i] = 0;
}
}
}
};
2012-09-19 15:15:09 +02:00
template<typename T> NcType& get_NetCDF_type();
2011-05-31 16:08:44 +02:00
2012-09-19 15:15:09 +02:00
#define IMPL_TYPE(T,ncT) \
template<> \
NcType& get_NetCDF_type<T>() \
{ \
return ncT; \
}
2008-12-02 18:00:28 +01:00
2012-09-19 15:15:09 +02:00
IMPL_TYPE(int,ncInt);
IMPL_TYPE(unsigned int,ncInt);
IMPL_TYPE(double,ncDouble);
IMPL_TYPE(float,ncFloat);
2008-12-02 18:00:28 +01:00
namespace CosmoTool {
template<typename T>
ProgressiveOutput<T>
ProgressiveOutput<T>::saveArrayProgressive(const std::string& fname, uint32_t *dimList,
2008-12-02 18:00:28 +01:00
uint32_t rank)
{
2012-09-19 15:15:09 +02:00
NcFile *f = new NcFile(fname, NcFile::replace);
2008-12-02 18:00:28 +01:00
2012-09-19 15:15:09 +02:00
vector<NcDim> dimArray;
2008-12-02 18:00:28 +01:00
for (uint32_t i = 0; i < rank; i++)
{
char dimName[255];
sprintf(dimName, "dim%d", i);
2012-09-19 15:15:09 +02:00
dimArray.push_back(f->addDim(dimName, dimList[rank-1-i]));
2008-12-02 18:00:28 +01:00
}
2012-09-19 15:15:09 +02:00
NcVar v = f->addVar("array", get_NetCDF_type<T>(), dimArray);
2008-12-02 18:00:28 +01:00
2012-09-19 15:15:09 +02:00
vector<NcDim> ldimList;
2008-12-02 18:00:28 +01:00
for (uint32_t i = 0; i < rank; i++)
2012-09-19 15:15:09 +02:00
ldimList.push_back(dimArray[rank-1-i]);
2008-12-02 18:00:28 +01:00
OutputGenCDF<T> *impl = new OutputGenCDF<T>(f, v, ldimList, rank);
return ProgressiveOutput<T>(impl);
}
template<typename T>
ProgressiveInput<T>
ProgressiveInput<T>::loadArrayProgressive(const std::string& fname, uint32_t *&dimList,
2008-12-02 18:00:28 +01:00
uint32_t& rank)
{
2012-09-19 15:15:09 +02:00
NcFile *f = new NcFile(fname, NcFile::read);
2008-12-02 18:00:28 +01:00
2012-09-19 15:15:09 +02:00
NcVar v = f->getVar("array");
2008-12-02 18:00:28 +01:00
2012-09-19 15:15:09 +02:00
rank = v.getDimCount();
vector<NcDim> vdimlist = v.getDims();
2008-12-02 18:00:28 +01:00
dimList = new uint32_t[rank];
for (uint32_t i = 0; i < rank; i++)
{
2012-09-19 15:15:09 +02:00
dimList[rank-i-1] = vdimlist[i].getSize();
2008-12-02 18:00:28 +01:00
}
2012-09-19 15:15:09 +02:00
InputGenCDF<T> *impl = new InputGenCDF<T>(f, v, vdimlist, rank);
2008-12-02 18:00:28 +01:00
return ProgressiveInput<T>(impl);
}
template<typename T>
void saveArray(const std::string& fname,
2008-12-02 18:00:28 +01:00
T *array, uint32_t *dimList, uint32_t rank)
{
2012-09-19 15:15:09 +02:00
NcFile f(fname.c_str(), NcFile::replace);
2008-12-02 18:00:28 +01:00
2012-09-19 15:15:09 +02:00
vector<NcDim> dimArray;
2008-12-02 18:00:28 +01:00
for (uint32_t i = 0; i < rank; i++)
{
char dimName[255];
sprintf(dimName, "dim%d", i);
2012-09-19 15:15:09 +02:00
dimArray.push_back(f.addDim(dimName, dimList[i]));
2008-12-02 18:00:28 +01:00
}
2012-09-19 15:15:09 +02:00
NcVar v = f.addVar("array", get_NetCDF_type<T>(), dimArray);
2008-12-02 18:00:28 +01:00
2012-09-19 15:15:09 +02:00
v.putVar(array);
2008-12-02 18:00:28 +01:00
}
template<typename T>
void loadArray(const std::string& fname,
2008-12-02 18:00:28 +01:00
T*&array, uint32_t *&dimList, uint32_t& rank)
2010-03-02 17:59:41 +01:00
throw (NoSuchFileException)
2008-12-02 18:00:28 +01:00
{
2012-09-19 15:15:09 +02:00
NcFile f(fname.c_str(), NcFile::read);
2008-12-02 18:00:28 +01:00
2012-09-19 15:15:09 +02:00
//if (!f.is_valid())
// throw NoSuchFileException(fname);
2008-12-02 18:00:28 +01:00
2012-09-19 15:15:09 +02:00
NcVar v = f.getVar("array");
vector<NcDim> dims = v.getDims();
rank = v.getDimCount();
2008-12-02 18:00:28 +01:00
uint32_t fullSize = 1;
dimList = new uint32_t[rank];
for (int i = 0; i < rank; i++)
{
2012-09-19 15:15:09 +02:00
dimList[i] = dims[i].getSize();
fullSize *= dimList[i];
2008-12-02 18:00:28 +01:00
}
if (fullSize != 0) {
array = new T[fullSize];
2012-09-19 15:15:09 +02:00
v.getVar(array);
}
2008-12-02 18:00:28 +01:00
}
template class ProgressiveInput<int>;
template class ProgressiveInput<float>;
template class ProgressiveInput<double>;
template class ProgressiveOutput<int>;
template class ProgressiveOutput<float>;
template class ProgressiveOutput<double>;
template void loadArray<int>(const std::string& fname,
2008-12-02 18:00:28 +01:00
int*& array, uint32_t *&dimList, uint32_t& rank);
template void loadArray<float>(const std::string& fname,
2008-12-02 18:00:28 +01:00
float*& array, uint32_t *&dimList, uint32_t& rank);
template void loadArray<double>(const std::string& fname,
2008-12-02 18:00:28 +01:00
double*& array, uint32_t *&dimList, uint32_t& rank);
template void saveArray<int>(const std::string& fname,
2008-12-02 18:00:28 +01:00
int *array, uint32_t *dimList, uint32_t rank);
template void saveArray<float>(const std::string& fname,
2008-12-02 18:00:28 +01:00
float *array, uint32_t *dimList, uint32_t rank);
template void saveArray<double>(const std::string& fname,
2008-12-02 18:00:28 +01:00
double *array, uint32_t *dimList, uint32_t rank);
}