X-Git-Url: https://git.creatis.insa-lyon.fr/pubgit/?a=blobdiff_plain;f=src%2FgdcmHeader.cxx;h=ebf321f2274d3bd122dba02e97794884db2340a5;hb=fdef18c217469cc507704db6d686ce195e82e4ce;hp=6572f6886cc65d17a4cd39cb0ad2f31e2d324a4e;hpb=fc4e699091cd861dc45f93ef3fb5530cc05ec2e2;p=gdcm.git diff --git a/src/gdcmHeader.cxx b/src/gdcmHeader.cxx index 6572f688..ebf321f2 100644 --- a/src/gdcmHeader.cxx +++ b/src/gdcmHeader.cxx @@ -1,4 +1,4 @@ -// $Header: /cvs/public/gdcm/src/Attic/gdcmHeader.cxx,v 1.111 2003/11/07 14:34:50 jpr Exp $ +// $Header: /cvs/public/gdcm/src/Attic/gdcmHeader.cxx,v 1.122 2004/01/13 11:13:08 regrain Exp $ #include "gdcmHeader.h" @@ -11,14 +11,20 @@ #include #endif #include // for isalpha -#include + +#ifdef GDCM_NO_ANSI_STRING_STREAM +# include +# define ostringstream ostrstream +# else +# include +#endif + #include "gdcmUtil.h" #include "gdcmTS.h" // Refer to gdcmHeader::CheckSwap() #define HEADER_LENGTH_TO_READ 256 // Refer to gdcmHeader::SetMaxSizeLoadElementValue() -//#define _MaxSizeLoadElementValue_ 1024 #define _MaxSizeLoadElementValue_ 4096 /** * \ingroup gdcmHeader @@ -38,7 +44,14 @@ void gdcmHeader::Initialise(void) { * @param InFilename * @param exception_on_error */ -gdcmHeader::gdcmHeader(const char *InFilename, bool exception_on_error) { +gdcmHeader::gdcmHeader(const char *InFilename, + bool exception_on_error, + bool enable_sequences ) { + if (enable_sequences) + enableSequences = 1; + else + enableSequences = 0; + SetMaxSizeLoadElementValue(_MaxSizeLoadElementValue_); filename = InFilename; Initialise(); @@ -65,7 +78,7 @@ gdcmHeader::gdcmHeader(bool exception_on_error) { * @param exception_on_error * @return */ - bool gdcmHeader::OpenFile(bool exception_on_error) +FILE *gdcmHeader::OpenFile(bool exception_on_error) throw(gdcmFileError) { fp=fopen(filename.c_str(),"rb"); if(exception_on_error) { @@ -79,20 +92,22 @@ gdcmHeader::gdcmHeader(bool exception_on_error) { //ACR -- or DICOM with no Preamble if( zero == 0x0008 || zero == 0x0800 || zero == 0x0002 || zero == 0x0200) - return true; + return(fp); + //DICOM fseek(fp, 126L, SEEK_CUR); char dicm[4]; fread(dicm, (size_t)4, (size_t)1, fp); if( memcmp(dicm, "DICM", 4) == 0 ) - return true; + return(fp); + fclose(fp); dbg.Verbose(0, "gdcmHeader::gdcmHeader not DICOM/ACR", filename.c_str()); } else { dbg.Verbose(0, "gdcmHeader::gdcmHeader cannot open file", filename.c_str()); } - return false; + return(NULL); } /** @@ -117,6 +132,8 @@ gdcmHeader::~gdcmHeader (void) { Dicts = (gdcmDictSet*)0; RefPubDict = (gdcmDict*)0; RefShaDict = (gdcmDict*)0; + + return; } @@ -627,17 +644,23 @@ bool gdcmHeader::IsDicomV3(void) { * applying this heuristic. */ void gdcmHeader::FixFoundLength(gdcmElValue * ElVal, guint32 FoundLength) { - if ( FoundLength == 0xffffffff) + + ElVal->SetReadLength(FoundLength); // will be updated only if a bug is found + + if ( FoundLength == 0xffffffff) { FoundLength = 0; + } // Sorry for the patch! // XMedCom did the trick to read some nasty GE images ... else if (FoundLength == 13) { // The following 'if' will be removed when there is no more // images on Creatis HDs with a 13 length for Manufacturer... - if ( (ElVal->GetGroup() != 0x0008) || (ElVal->GetElement() != 0x0070)) { + if ( (ElVal->GetGroup() != 0x0008) || + ( (ElVal->GetElement() != 0x0070) && (ElVal->GetElement() != 0x0080) ) ) { // end of remove area FoundLength =10; + ElVal->SetReadLength(10); // a bug is to be fixed } } // to fix some garbage 'Leonardo' Siemens images @@ -645,12 +668,14 @@ void gdcmHeader::FixFoundLength(gdcmElValue * ElVal, guint32 FoundLength) { else if ( (ElVal->GetGroup() == 0x0009) && ( (ElVal->GetElement() == 0x1113) || (ElVal->GetElement() == 0x1114) ) ){ - FoundLength =4; + FoundLength =4; + ElVal->SetReadLength(4); // a bug is to be fixed } // end of fix - // to try to 'go inside' SeQuences (with length), and not to ship them + // to try to 'go inside' SeQuences (with length), and not to skip them else if ( ElVal->GetVR() == "SQ") { + if (enableSequences) // only if the user does want to ! FoundLength =0; } @@ -661,7 +686,7 @@ void gdcmHeader::FixFoundLength(gdcmElValue * ElVal, guint32 FoundLength) { FoundLength =0; } - ElVal->SetLength(FoundLength); + ElVal->SetUsableLength(FoundLength); } /** @@ -724,7 +749,7 @@ void gdcmHeader::FixFoundLength(gdcmElValue * ElVal, guint32 FoundLength) { std::string vr = ElVal->GetVR(); guint16 length16; if( (element == 0x0010) && (group == 0x7fe0) ) { - dbg.SetDebug(0); + dbg.SetDebug(-1); dbg.Verbose(2, "gdcmHeader::FindLength: ", "we reached 7fe0 0010"); } @@ -744,7 +769,6 @@ void gdcmHeader::FixFoundLength(gdcmElValue * ElVal, guint32 FoundLength) { return; } FixFoundLength(ElVal, length32); - return; } @@ -768,7 +792,7 @@ void gdcmHeader::FixFoundLength(gdcmElValue * ElVal, guint32 FoundLength) { // appear when we find the first group with big endian encoding. This // is easy to detect since the length of a "Group Length" tag (the // ones with zero as element number) has to be of 4 (0x0004). When we - // encouter 1024 (0x0400) chances are the encoding changed and we + // encounter 1024 (0x0400) chances are the encoding changed and we // found a group with big endian encoding. // We shall use this second strategy. In order to make sure that we // can interpret the presence of an apparently big endian encoded @@ -821,6 +845,7 @@ void gdcmHeader::FixFoundLength(gdcmElValue * ElVal, guint32 FoundLength) { // not coexist in a Data Set and Data Sets nested within it".] // Length is on 4 bytes. FixFoundLength(ElVal, ReadInt32()); + return; } /** @@ -923,16 +948,16 @@ void gdcmHeader::LoadElementValue(gdcmElValue * ElVal) { // qui pourrait ne pas etre generalisable // Well, I'm expecting your code !!! - // to try to 'go inside' the SeQuences + // the test was commented out to 'go inside' the SeQuences // we don't any longer skip them ! // if( vr == "SQ" ) // SkipLoad = true; - // A sequence "contains" a set of Elements. + // A SeQuence "contains" a set of Elements. // (fffe e000) tells us an Element is beginning // (fffe e00d) tells us an Element just ended - // (fffe e0dd) tells us the current SQuence just ended + // (fffe e0dd) tells us the current SeQuence just ended if( group == 0xfffe ) SkipLoad = true; @@ -957,6 +982,7 @@ void gdcmHeader::LoadElementValue(gdcmElValue * ElVal) { s << "gdcm::NotLoaded."; s << " Address:" << (long)ElVal->GetOffset(); s << " Length:" << ElVal->GetLength(); + s << " x(" << std::hex << ElVal->GetLength() << ")"; ElVal->SetValue(s.str()); return; } @@ -964,9 +990,9 @@ void gdcmHeader::LoadElementValue(gdcmElValue * ElVal) { // When an integer is expected, read and convert the following two or // four bytes properly i.e. as an integer as opposed to a string. - // pour les elements de Value Multiplicity > 1 - // on aura en fait une serie d'entiers - // on devrait pouvoir faire + compact (?) + // Actually, elements with Value Multiplicity > 1 + // contain a set of integers (not a single one) + // Any compacter code suggested (?) if ( IsAnInteger(ElVal) ) { guint32 NewInt; @@ -996,6 +1022,9 @@ void gdcmHeader::LoadElementValue(gdcmElValue * ElVal) { } } } +#ifdef GDCM_NO_ANSI_STRING_STREAM + s << std::ends; // to avoid oddities on Solaris +#endif //GDCM_NO_ANSI_STRING_STREAM ElVal->SetValue(s.str()); return; } @@ -1133,7 +1162,6 @@ gdcmElValue* gdcmHeader::NewElValueByNumber(guint16 Group, guint16 Elem) { */ int gdcmHeader::ReplaceOrCreateByNumber(std::string Value, guint16 Group, guint16 Elem ) { - // TODO : FIXME JPRx // curieux, non ? // on (je) cree une Elvalue ne contenant pas de valeur @@ -1244,6 +1272,7 @@ gdcmElValue * gdcmHeader::ReadNextElement(void) { NewElVal = NewElValueByNumber(g, n); FindVR(NewElVal); FindLength(NewElVal); + if (errno == 1) { // Call it quits return (gdcmElValue *)0; @@ -1445,7 +1474,7 @@ std::string gdcmHeader::GetPubElValRepByNumber(guint16 group, guint16 element) { * \ingroup gdcmHeader * \brief Searches within the public dictionary for element value of * a given tag. - * @param TagName name of the researched element. + * @param TagName name of the searched element. * @return Corresponding element value when it exists, and the string * GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1462,7 +1491,7 @@ std::string gdcmHeader::GetPubElValByName(std::string TagName) { * to convert the string typed content to caller's native type * (think of C++ vs Python). The VR is actually of a higher level * of semantics than just the native C++ type. - * @param TagName name of the researched element. + * @param TagName name of the searched element. * @return Corresponding element value representation when it exists, * and the string GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1477,8 +1506,8 @@ std::string gdcmHeader::GetPubElValRepByName(std::string TagName) { * \ingroup gdcmHeader * \brief Searches within elements parsed with the SHADOW dictionary * for the element value of a given tag. - * @param group Group of the researched tag. - * @param element Element of the researched tag. + * @param group Group of the searched tag. + * @param element Element of the searched tag. * @return Corresponding element value representation when it exists, * and the string GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1495,8 +1524,8 @@ std::string gdcmHeader::GetShaElValByNumber(guint16 group, guint16 element) { * to convert the string typed content to caller's native type * (think of C++ vs Python). The VR is actually of a higher level * of semantics than just the native C++ type. - * @param group Group of the researched tag. - * @param element Element of the researched tag. + * @param group Group of the searched tag. + * @param element Element of the searched tag. * @return Corresponding element value representation when it exists, * and the string GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1511,7 +1540,7 @@ std::string gdcmHeader::GetShaElValRepByNumber(guint16 group, guint16 element) { * \ingroup gdcmHeader * \brief Searches within the elements parsed with the shadow dictionary * for an element value of given tag. - * @param TagName name of the researched element. + * @param TagName name of the searched element. * @return Corresponding element value when it exists, and the string * GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1528,7 +1557,7 @@ std::string gdcmHeader::GetShaElValByName(std::string TagName) { * to convert the string typed content to caller's native type * (think of C++ vs Python). The VR is actually of a higher level * of semantics than just the native C++ type. - * @param TagName name of the researched element. + * @param TagName name of the searched element. * @return Corresponding element value representation when it exists, * and the string GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1544,8 +1573,8 @@ std::string gdcmHeader::GetShaElValRepByName(std::string TagName) { * \brief Searches within elements parsed with the public dictionary * and then within the elements parsed with the shadow dictionary * for the element value of a given tag. - * @param group Group of the researched tag. - * @param element Element of the researched tag. + * @param group Group of the searched tag. + * @param element Element of the searched tag. * @return Corresponding element value representation when it exists, * and the string GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1566,8 +1595,8 @@ std::string gdcmHeader::GetElValByNumber(guint16 group, guint16 element) { * to convert the string typed content to caller's native type * (think of C++ vs Python). The VR is actually of a higher level * of semantics than just the native C++ type. - * @param group Group of the researched tag. - * @param element Element of the researched tag. + * @param group Group of the searched tag. + * @param element Element of the searched tag. * @return Corresponding element value representation when it exists, * and the string GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1583,7 +1612,7 @@ std::string gdcmHeader::GetElValRepByNumber(guint16 group, guint16 element) { * \brief Searches within elements parsed with the public dictionary * and then within the elements parsed with the shadow dictionary * for the element value of a given tag. - * @param TagName name of the researched element. + * @param TagName name of the searched element. * @return Corresponding element value when it exists, * and the string GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1604,7 +1633,7 @@ std::string gdcmHeader::GetElValByName(std::string TagName) { * to convert the string typed content to caller's native type * (think of C++ vs Python). The VR is actually of a higher level * of semantics than just the native C++ type. - * @param TagName name of the researched element. + * @param TagName name of the searched element. * @return Corresponding element value representation when it exists, * and the string GDCM_UNFOUND ("gdcm::Unfound") otherwise. */ @@ -1698,7 +1727,7 @@ void gdcmHeader::ParseHeader(bool exception_on_error) throw(gdcmFormatError) { rewind(fp); CheckSwap(); - while ( (newElValue = ReadNextElement()) ) { + while ( (newElValue = ReadNextElement()) ) { SkipElementValue(newElValue); PubElValSet.Add(newElValue); } @@ -1706,7 +1735,17 @@ void gdcmHeader::ParseHeader(bool exception_on_error) throw(gdcmFormatError) { /** * \ingroup gdcmHeader - * \brief This predicate, based on hopefully reasonnable heuristics, + * \brief + * @return + */ +FileType gdcmHeader::GetFileType(void) +{ + return(filetype); +} + +/** + * \ingroup gdcmHeader + * \brief This predicate, based on hopefully reasonable heuristics, * decides whether or not the current gdcmHeader was properly parsed * and contains the mandatory information for being considered as * a well formed and usable image. @@ -1765,11 +1804,22 @@ gdcmElValue* gdcmHeader::NewManualElValToPubDict(std::string NewTagName, * public tag based hash table. */ void gdcmHeader::LoadElements(void) { - rewind(fp); - TagElValueHT ht = PubElValSet.GetTagHt(); - for (TagElValueHT::iterator tag = ht.begin(); tag != ht.end(); ++tag) { - LoadElementValue(tag->second); - } + rewind(fp); + + // We don't use any longer the HashTable, since a lot a stuff is missing + // when SeQuences were encountered + // + //TagElValueHT ht = PubElValSet.GetTagHt(); + //for (TagElValueHT::iterator tag = ht.begin(); tag != ht.end(); ++tag) { + // LoadElementValue(tag->second); + //} + + for (ListTag::iterator i = GetListElem().begin(); + i != GetListElem().end(); + ++i){ + LoadElementValue(*i); + } + rewind(fp); // Load 'non string' values @@ -1829,8 +1879,7 @@ void gdcmHeader::PrintPubDict(std::ostream & os) { */ int gdcmHeader::Write(FILE * fp, FileType type) { - - // TODO : move the following lines (and a lot of others) + // TODO : move the following lines (and a lot of others, to be written) // to a future function CheckAndCorrectHeader if (type == ImplicitVR) { @@ -1876,16 +1925,12 @@ void * gdcmHeader::LoadElementVoidArea(guint16 Group, guint16 Elem) { int l=Element->GetLength(); void * a = malloc(l); if(!a) { - std::cout << "Big Broblem (LoadElementVoidArea, malloc) " - << std::hex << Group << " " << Elem << std::endl; return NULL; } /* int res = */ PubElValSet.SetVoidAreaByNumber(a, Group, Elem); // TODO check the result size_t l2 = fread(a, 1, l ,fp); if(l != l2) { - std::cout << "Big Broblem (LoadElementVoidArea, fread) " - << std::hex << Group << " " << Elem << std::endl; free(a); return NULL; } @@ -2107,8 +2152,6 @@ std::string gdcmHeader::GetPixelType(void) { if (Signed == "0") Signed = std::string("U"); else - -std::cout << "GetPixelType : " << BitsAlloc + Signed << std::endl; Signed = std::string("S"); return( BitsAlloc + Signed); @@ -2210,7 +2253,7 @@ int gdcmHeader::GetLUTNbits(void) { * @return Lookup Table RGBA */ -void * gdcmHeader::GetLUTRGBA(void) { +unsigned char * gdcmHeader::GetLUTRGBA(void) { // Not so easy : see // http://www.barre.nom.fr/medical/dicom2/limitations.html#Color%20Lookup%20Tables // and OT-PAL-8-face.dcm @@ -2220,9 +2263,7 @@ void * gdcmHeader::GetLUTRGBA(void) { if (gdcmHeader::GetPubElValByNumber(0x0028,0x0004) != "PALETTE COLOR ") { return NULL; } - - void * LutR,*LutG,*LutB; - int l; + int lengthR, debR, nbitsR; int lengthG, debG, nbitsG; int lengthB, debB, nbitsB; @@ -2335,3 +2376,32 @@ void * gdcmHeader::GetLUTRGBA(void) { return(LUTRGBA); } +///////////////////////////////////////////////////////////////// +/** + * \ingroup gdcmFile + * \brief Sets the Pixel Area size in the Header + * --> not-for-rats function + * + * \warning WARNING doit-etre etre publique ? + * TODO : y aurait il un inconvenient à fusionner ces 2 fonctions + * + * @param ImageDataSize new Pixel Area Size + * warning : nothing else is checked + */ + +void gdcmHeader::SetImageDataSize(size_t ImageDataSize) { + std::string content1; + char car[20]; + // Assumes ElValue (0x7fe0, 0x0010) exists ... + sprintf(car,"%d",ImageDataSize); + + gdcmElValue *a = GetElValueByNumber(0x7fe0, 0x0010); + a->SetLength(ImageDataSize); + + ImageDataSize+=8; + sprintf(car,"%d",ImageDataSize); + content1=car; + SetPubElValByNumber(content1, 0x7fe0, 0x0000); +} + +