#include "sopnamsp.h" #include "machdefs.h" #include "segdatablock.h" #include "fitsinoutfile.h" #include #include #include string FitsTypes::ImageTypeToTypeString(int ityp) { switch (ityp) { case BYTE_IMG : return "uint_1"; break; case SHORT_IMG : return "int_2"; break; case LONG_IMG : return "int_4"; break; case FLOAT_IMG : return "r_4"; break; case DOUBLE_IMG : return "r_8"; break; default: return "???" ; break; } return ""; } string FitsTypes::DataTypeToTypeString(int ityp) { switch (ityp) { case TBYTE : return "uint_1"; break; case TSHORT : return "int_2"; break; case TUSHORT : return "uint_2"; break; case TINT : if (sizeof(int) == 4) return "int_4"; else if (sizeof(int) == 8) return "int_8"; else if (sizeof(int) == 2) return "int_2"; break; case TUINT : if (sizeof(int) == 4) return "uint_4"; else if (sizeof(int) == 8) return "uint_8"; else if (sizeof(int) == 2) return "uint_2"; break; case TLONG : if (sizeof(long) == 4) return "int_4"; else if (sizeof(long) == 8) return "int_8"; else if (sizeof(long) == 2) return "int_2"; break; case TULONG : if (sizeof(long) == 4) return "uint_4"; else if (sizeof(long) == 8) return "uint_8"; else if (sizeof(long) == 2) return "uint_2"; break; #ifdef TLONGLONG case TLONGLONG : return "int_8"; break; #endif case TFLOAT : return "r_4"; break; case TDOUBLE : return "r_8"; break; case TCOMPLEX : return "complex< r_4 >"; break; case TDBLCOMPLEX : return "complex< r_8 >"; break; case TSTRING : return "string"; break; default: return "???" ; break; } return ""; } /*! \class SOPHYA::FitsInOutFile \ingroup FitsIOServer \brief Wrapper class for cfitsio library functions */ /*-- Methode --*/ //! Default constructor - The file should be opened subsequently using Open FitsInOutFile::FitsInOutFile() { fptr_ = NULL; mode_ = Fits_RO; SetDef_BinTable(); SetDef_StrColWidth(); } /*-- Methode --*/ //! Constructor with specification of file name and access mode FitsInOutFile::FitsInOutFile(string const & name, FitsIOMode mode) { // cout << " DBG - FitsInOutFile(string name= " << name << ")" << endl; fptr_ = NULL; ownfptr = true; SetDef_BinTable(); SetDef_StrColWidth(); Open(name.c_str(), mode); } /*-- Methode --*/ //! Constructor with specification of file name and access mode FitsInOutFile::FitsInOutFile(const char * name, FitsIOMode mode) { // cout << " DBG - FitsInOutFile(char* name= " << name << ")" << endl; fptr_ = NULL; ownfptr = true; SetDef_BinTable(); SetDef_StrColWidth(); Open(name, mode); } /*! \brief Copy constructor \warning The fits file pointer is owned by the original FitsInOutFile object and should not be closed as long as the new object is being used. */ FitsInOutFile::FitsInOutFile(FitsInOutFile const& fios) { fptr_ = fios.fptr_; fname_ = fios.fname_; mode_ = fios.mode_; ownfptr = false; SetDef_BinTable(); SetDef_StrColWidth(); } /* -- Fonction utilitaire pour verifier fichier ouvert (pointeur non null) -- */ static inline void CheckFitsPtr(fitsfile* fptr) { if (fptr == NULL) throw FitsIOException("FitsInOutFile/Error - NULL fitsfile pointer "); return; } /* -- Fonction utilitaire pour verifier le code d'erreur fitsio -- */ static inline bool FitsCheckStatus(int st, char * emsg = NULL) { if (st) { fits_report_error(stderr, st); if (emsg) { char buff[FLEN_ERRMSG]; fits_get_errstatus(st, buff); string msg = emsg; msg += buff; throw FitsIOException(msg); } else return true; } else return false; } /*-- Methode --*/ //! Destructor - Closes the fits file (if opened) FitsInOutFile::~FitsInOutFile() { Close(); } /*-- Methode --*/ //! Opens the named fits file (see cfitsio routines fits_open_file and fits_create_file) void FitsInOutFile::Open(const char* name, FitsIOMode mode) { if (fptr_ != NULL) throw FitsIOException("FitsInOutFile::Open() /Error - file already opened "); int status = 0; fptr_ = NULL; switch ( mode ) { case Fits_RO : fits_open_file(&fptr_, name, READONLY, &status); break; case Fits_RW : fits_open_file(&fptr_, name, READWRITE, &status); break; case Fits_Create : fits_create_file(&fptr_, name, &status); break; } FitsCheckStatus(status, "FitsInOutFile::Open() Error: "); fname_ = name; mode_ = mode; ownfptr = true; return; } /*-- Methode --*/ //! Opens the named fits file (see cfitsio routines fits_open_file and fits_create_file) void FitsInOutFile::Close() { if (ownfptr == false) return; if (fptr_ == NULL) return; int status = 0; if (mode_ == Fits_Create) { status = 0; int hdutyp; fits_movabs_hdu(FitsPtr() , 1, &hdutyp, &status); status = 0; float sfv = Version(); fits_write_key(FitsPtr(), TFLOAT, "SOPHYAFV", &sfv, "SOPHYA FitsIOServer module version", &status); fits_write_date(FitsPtr(), &status); status = 0; fits_write_comment(FitsPtr(), "------------- SOPHYA (http://www.sophya.org) -------------", &status); fits_write_comment(FitsPtr(), " (C) LAL/IN2P3-CNRS Orsay , (C) DAPNIA/CEA Saclay (FRANCE)", &status); fits_write_comment(FitsPtr(), "-----------------------------------------------------------", &status); } MoveAbsToHDU(1); status = 0; if (fptr_) fits_close_file(fptr_, &status); if (status) { cerr << " FitsInOutFile::Close - Error closing fits file !" << endl; fits_report_error(stderr, status); } } /*! \brief Closes the current fits file and uses \b fios file for subsequent operations. \warning The fits file pointer is owned by the original FitsInOutFile object and should not be closed as long as the current object (this) is being used. */ void FitsInOutFile::ShareFitsPtr(FitsInOutFile const& fios) { Close(); fptr_ = fios.fptr_; fname_ = fios.fname_; mode_ = fios.mode_; ownfptr = false; } /*-- Methode --*/ float FitsInOutFile::cfitsioVersion() { float ver; fits_get_version(&ver); return ver; } /*-- Methode --*/ int FitsInOutFile::NbHDUs() const { int status = 0; int nbhdu = 0; fits_get_num_hdus(FitsPtr() , &nbhdu, &status); FitsCheckStatus(status, "FitsInOutFile::NbHDUs() Error: "); return nbhdu; } /*-- Methode --*/ int FitsInOutFile::CurrentHDU() const { int status = 0; int curhdu = 0; fits_get_hdu_num(FitsPtr() , &curhdu); return curhdu; } /*-- Methode --*/ int FitsInOutFile::CurrentHDUType() const { int status = 0; int hdutyp = 0; fits_get_hdu_type(FitsPtr() , &hdutyp, &status); FitsCheckStatus(status, "FitsInOutFile::CurrentHDUType() Error: "); return hdutyp; } /*-- Methode --*/ string FitsInOutFile::CurrentHDUTypeStr() const { int status = 0; int hdutyp = 0; fits_get_hdu_type(FitsPtr() , &hdutyp, &status); if ( FitsCheckStatus(status, "FitsInOutFile::CurrentHDUTypeStr() Error: ") ) return "Unknown"; else { if (hdutyp == IMAGE_HDU) return "IMAGE_HDU"; else if (hdutyp == BINARY_TBL) return "BINARY_TBL"; else if (hdutyp == ASCII_TBL) return "ASCII_TBL"; else return "Unknown"; } } /*-- Methode --*/ int FitsInOutFile::MoveAbsToHDU(int hdunum) { int status = 0; int hdutyp = 0; fits_movabs_hdu(FitsPtr() , hdunum, &hdutyp, &status); FitsCheckStatus(status, "FitsInOutFile::MoveAbsToHDU Error: "); return hdutyp; } /*-- Methode --*/ int FitsInOutFile::MoveRelToHDU(int hdunum) { int status = 0; int hdutyp = 0; fits_movrel_hdu(FitsPtr() , hdunum, &hdutyp, &status); FitsCheckStatus(status, "FitsInOutFile::MoveRelToHDU Error: "); return hdutyp; } /*-- Methode --*/ int FitsInOutFile::MoveToNextHDU() { if (CurrentHDU() < NbHDUs()) return MoveRelToHDU(1); else return -1; } /*-- Methode --*/ void FitsInOutFile::CreateImageHDU(int bitpix, int naxis, long* naxes) { int status = 0; fits_create_img(fptr_, bitpix, naxis, naxes, &status); FitsCheckStatus(status,"FitsInOutFile::CreateImageHDU() Error: "); return; } /*-- Methode --*/ /*! See cfitsio function fits_get_img_param() for more information naxis : input=max naxes dimension / out=image dimension Rc : return the image type (bitpix) */ int FitsInOutFile::GetImageHDUInfo(int& naxis, long* naxes) const { int status = 0; int maxdim = naxis; int bitpix = 0; fits_get_img_param(fptr_, maxdim, &bitpix, &naxis, naxes, &status); FitsCheckStatus(status, "FitsInOutFile::GetImageHDUInfo() Error: "); return bitpix; } /*-- Methode --*/ long FitsInOutFile::GetNbRows() const { int status = 0; long nbrow = 0; fits_get_num_rows(FitsPtr() , &nbrow, &status); FitsCheckStatus(status, "FitsInOutFile::GetNbRows() Error: " ); return nbrow; } /*-- Methode --*/ int FitsInOutFile::GetNbCols() const { int status = 0; int nbcol = 0; fits_get_num_cols(FitsPtr() , &nbcol, &status); FitsCheckStatus(status, "FitsInOutFile::GetNbCols() Error: "); return nbcol; } /*-- Methode --*/ /*! Create a binary or ascii table - See cfitsio routine fits_create_tbl for more information. \param extname : extension name. NextExtensionName() will be used if extname == NULL or extname == "". \param ncols : Number of columns \param ttype : Column names \param tform : Column data types J / V / K / E / D ... \param tunit : Column units */ void FitsInOutFile::CreateTable(int tbltyp, const char * extname, int ncols, char * ttype[], char * tform[], char * tunit[], long ininr) { int status = 0; char * extn; if ( (extname != NULL) && (extname[0] != '\0') ) extn = const_cast(extname); else extn = const_cast(next_extname_.c_str()); fits_create_tbl(FitsPtr(), tbltyp, ininr, ncols, ttype, tform, tunit, extn, &status); next_extname_ = ""; FitsCheckStatus(status, "FitsInOutFile::CreateTable() Error: "); return; } /*-- Methode --*/ /*! Create a binary or ascii table - See cfitsio routine fits_create_tbl for more information. number of columns specified by colnames.size() \param extname : extension name \param colnames : Column names \param tform : Column data types J / V / K / E / D ... \param tunit : Column units */ void FitsInOutFile::CreateTable(int tbltyp, const string & extname, const vector & colnames, const vector & tform, const vector & tunit, long ininr) { if ( (colnames.size() != tform.size() ) || (colnames.size() != tunit.size() ) ) throw SzMismatchError("FitsInOutFile::CreateTable(): different sizes for colnames,tform,tunit"); // On utilise les SegDataBlock pour eviter d'avoir a gerer les new/delete // en plus avec les exceptions ... size_t kk; int ncols = colnames.size(); SegDataBlock colnm(colnames.size(), 1); for(kk=0; kk tfm(tform.size(), 1); for(kk=0; kk tun(tunit.size(), 1); for(kk=0; kk(extname.c_str()), ncols, const_cast(colnm.GetSegment(0)), const_cast(tfm.GetSegment(0)), const_cast(tun.GetSegment(0)), ininr); } /*-- Methode --*/ /*! Return number of columns in table (See fits_get_colname and fits_get_coltype for more information) \param colnames : Column names \param coltypes : Column data types ( TSTRING / TSHORT / TFLOAT / ... ) \param repcnt : Repeat count (for columns with vector data) \param width : The width (in bytes) of a single element in a column */ long FitsInOutFile::GetColInfo(vector & colnames, vector & coltypes, vector & repcnt, vector & width) { int status = 0; colnames.clear(); coltypes.clear(); width.clear(); repcnt.clear(); int colnum, typecode; long repeat, colw; int ncols = 0; char colname[128]; // longueur max d'un nom de colonne while (status != COL_NOT_FOUND) { fits_get_colname(FitsPtr(), CASEINSEN, "*", colname, &colnum, &status); if (status == COL_NOT_FOUND) break; if ( (status != COL_NOT_UNIQUE) && (status != 0) ) { char buff[32]; fits_get_errstatus(status, buff); string msg = "FitsInOutFile::GetColInfo() Error(1): " ; msg += buff; throw FitsIOException(msg); } int sta2 = 0; fits_get_coltype(FitsPtr(), colnum, &typecode, &repeat, &colw, &sta2); FitsCheckStatus(sta2, "FitsInOutFile::GetColInfo() Error(2): "); colnames.push_back(colname); coltypes.push_back(typecode); repcnt.push_back(repeat); width.push_back(colw); if (status == 0) break; } return colnames.size(); } /*-- Methode --*/ void FitsInOutFile::InsertColumn(int numcol, const char* colname, const char* fmt) { int status = 0; fits_insert_col(FitsPtr(), numcol, const_cast(colname), const_cast(fmt), &status); FitsCheckStatus(status, "FitsInOutFile::AddColumn() Error: "); return; } /*-- Methode --*/ /*! Return the value associated to the keyword \b key in the header as a string. If the keyword is not found in the fits header, an empty string is returned and the \b nosk flag is set to true. */ string FitsInOutFile::KeyValue(string const & key, bool& nosk) { nosk = false; int status = 0; char value[FLEN_VALUE], comm[FLEN_COMMENT]; fits_read_key(FitsPtr(), TSTRING, const_cast(key.c_str()), value, comm, &status); if (status == KEY_NO_EXIST) { nosk = true; return ""; } FitsCheckStatus(status, "FitsInOutFile::KeyValue() Error: "); return value; } /*-- Methode --*/ int FitsInOutFile::GetHeaderRecords(DVList& dvl) { int status = 0; int nkeys = 0; fits_get_hdrspace(FitsPtr(), &nkeys, NULL, &status); FitsCheckStatus(status, "FitsInOutFile::GetHeaderRecords() Error(1): "); char record[FLEN_CARD], value[FLEN_VALUE], comm[FLEN_COMMENT]; string comment; int nok = 0; for(int kk=1; kk<=nkeys; kk++) { status = 0; fits_read_record(FitsPtr(), kk, record, &status); FitsCheckStatus(status, "FitsInOutFile::GetHeaderRecords() Error(2): "); int kclas = fits_get_keyclass(record); if ( (kclas == TYP_STRUC_KEY) || (kclas == TYP_NULL_KEY) ) continue; int len = 0; status = 0; fits_get_keyname(record, value, &len, &status); if (status) continue; string keyname = value; status = 0; fits_parse_value(record, value, comm, &status); if (status) continue; if (kclas == TYP_COMM_KEY) { if (comment.length() > 0) comment += '\n'; comment += value; continue; } char ktyp; status = 0; fits_get_keytype(value, &ktyp, &status); if (status) continue; switch (ktyp) { case 'C' : case 'L' : // Il faudra traiter le cas des nb complexes case 'X' : // idem pour les valeurs logiques dvl.SetS(keyname, value); break; case 'I' : dvl.SetI(keyname, atoi(value)); break; case 'F' : dvl.SetD(keyname, atof(value)); break; } if (strlen(comm) > 0) { string scom = comm; dvl.SetComment(keyname, scom); } nok++; } if (comment.length() > 0) dvl.Comment() = comment; return nok; } /*-- Methode --*/ void FitsInOutFile::WriteKey(const char * kname, MuTyV const & mtv, const char *comment) { CheckFitsPtr(FitsPtr()); char keyname[FLEN_KEYWORD], comm[FLEN_COMMENT], sval[FLEN_VALUE]; long lval; double dval; string s; keyname[0] = '\0'; strncpy(keyname, kname, FLEN_KEYWORD); keyname[FLEN_KEYWORD-1] = '\0'; comm[0] = '\0'; if (comm != NULL) strncpy(comm, comment, FLEN_COMMENT); comm[FLEN_COMMENT-1] = '\0'; int status = 0; switch (mtv.Type()) { case MuTyV::MTVInteger : lval = mtv.GetIntPart(); fits_write_key(FitsPtr(), TLONG, keyname, &lval, comm, &status); break; case MuTyV::MTVFloat : dval = mtv.GetRealPart(); fits_write_key(FitsPtr(), TDOUBLE, keyname, &dval, comm, &status); break; case MuTyV::MTVString : strncpy(sval, mtv.GetStringPointer()->c_str(), FLEN_VALUE); keyname[FLEN_VALUE-1] = '\0'; fits_write_key(FitsPtr(), TSTRING, keyname, sval, comm, &status); break; default : s = (string)mtv; strncpy(sval, s.c_str(), FLEN_VALUE); keyname[FLEN_VALUE-1] = '\0'; fits_write_key(FitsPtr(), TSTRING, keyname, sval, comm, &status); break; } FitsCheckStatus(status, "FitsInOutFile::WriteKey() Error: "); return; } /*-- Methode --*/ int FitsInOutFile::WriteHeaderRecords(DVList & dvl) { CheckFitsPtr(FitsPtr()); int status = 0; DVList::ValList::const_iterator it; for(it = dvl.Begin(); it != dvl.End(); it++) WriteKey( (*it).first, (*it).second.elval, (*it).second.elcomm); // Ecriture commentaires return 0; } /*-- Methode --*/ void FitsInOutFile::Print(ostream& os, int lev) const { string mode; if (mode_ == Fits_Create) mode = "Create"; else if (mode_ == Fits_RO) mode = "ReadOnly"; else mode = "ReadWrite"; os << " FitsInOutFile(FileName= " << fname_ << " Mode=" << mode << ") fitsioVers= " << cfitsioVersion() << endl; os << " TotalNumberHDU= " << NbHDUs() << " CurrentHDU: Num= " << CurrentHDU() << " Type= " << CurrentHDUTypeStr() << endl; return; }