]> Creatis software - gdcm.git/blobdiff - src/gdcmDocument.cxx
upgrades for 4DSplitter
[gdcm.git] / src / gdcmDocument.cxx
index 1c50ab555b50df2015b30e3f8a369d96717e7408..284fe5e6fee561ab73a129a052ac542318acbb84 100644 (file)
@@ -3,8 +3,8 @@
   Program:   gdcm
   Module:    $RCSfile: gdcmDocument.cxx,v $
   Language:  C++
-  Date:      $Date: 2007/04/12 13:22:49 $
-  Version:   $Revision: 1.357 $
+  Date:      $Date: 2011/03/29 07:36:00 $
+  Version:   $Revision: 1.386 $
                                                                                 
   Copyright (c) CREATIS (Centre de Recherche et d'Applications en Traitement de
   l'Image). All rights reserved. See Doc/License.txt or
@@ -37,7 +37,7 @@
    #include <mem.h> // for memset
 #endif 
 
-namespace gdcm 
+namespace GDCM_NAME_SPACE 
 {
 //-----------------------------------------------------------------------------
 
@@ -50,7 +50,7 @@ const unsigned int Document::MAX_SIZE_LOAD_ELEMENT_VALUE = 0xfff; // 4096
 
 /**
  * \brief This default constructor neither loads nor parses the file. 
- *        You should then invoke \ref Document::Load.
+ *        You should then invoke Document::Load.
  *         
  */
 Document::Document() 
@@ -63,6 +63,7 @@ Document::Document()
    SwapCode = 1234;
    Filetype = ExplicitVR;
    CurrentOffsetPosition = 0;
+   OffsetOfPreviousParseDES =0;
    // Load will set it to true if sucessfull
    Group0002Parsed = false;
    IsDocumentAlreadyLoaded = false;
@@ -70,6 +71,8 @@ Document::Document()
    LoadMode = LD_ALL; // default : load everything, later
    
    SetFileName("");
+   changeFromUN=false;
+   UnexpectedEOF=false;
 }
 
 /**
@@ -159,6 +162,7 @@ bool Document::DoTheLoadingDocumentJob(  )
 
    // Recursive call.
    // Loading is done during parsing
+   OffsetOfPreviousParseDES = beg; 
    ParseDES( this, beg, lgt, false); // delim_mode is first defaulted to false
 
    if ( IsEmpty() )
@@ -244,7 +248,7 @@ bool Document::DoTheLoadingDocumentJob(  )
 
    // Force Loading some more elements if user asked to.
 
-   gdcm::DocEntry *d;
+   GDCM_NAME_SPACE::DocEntry *d;
    for (ListElements::iterator it = UserForceLoadList.begin();  
                                it != UserForceLoadList.end();
                              ++it)
@@ -279,10 +283,13 @@ bool Document::DoTheLoadingDocumentJob(  )
    //
    std::string RecCode;
    RecCode = GetEntryString(0x0008, 0x0010); // recognition code (RET)
-   if (RecCode == "ACRNEMA_LIBIDO_1.1" ||
-       RecCode == "CANRME_AILIBOD1_1." )  // for brain-damaged softwares
-                                          // with "little-endian strings"
+   
+
+   if(RecCode.find("ACRNEMA_LIBIDO") == 0 || // any version
+      RecCode.find("CANRME_AILIBOD") == 0)   // for brain-damaged softwares
+                                             // with "little-endian strings"
    {
+   
          Filetype = ACR_LIBIDO; 
          std::string rows    = GetEntryString(0x0028, 0x0010);
          std::string columns = GetEntryString(0x0028, 0x0011);
@@ -436,7 +443,7 @@ std::string Document::GetTransferSyntax()
    // happen at different stages): try loading and proceed with check...
    
    // Well ...
-   // (parsing and loading happen at the very same stage!) 
+   // (parsing and loading happen at the very same stage!)
    //LoadDocEntrySafe(entry); //JPRx
    if (DataEntry *dataEntry = dynamic_cast<DataEntry *>(entry) )
    {
@@ -526,11 +533,11 @@ uint32_t Document::SwapLong(uint32_t a)
 // save CPU time
          a=( ( a<<24)               | ((a<<8)  & 0x00ff0000) | 
              ((a>>8)  & 0x0000ff00) |  (a>>24)                );
-         break;   
+         break;
       case 3412 :
 //       a=( ((a<<16) & 0xffff0000) | ((a>>16) & 0x0000ffff) );
          a=( (a<<16)                | (a>>16)  );
-         break;  
+         break;
       case 2143 :
          a=( ((a<< 8) & 0xff00ff00) | ((a>>8) & 0x00ff00ff)  );
       break;
@@ -539,7 +546,7 @@ uint32_t Document::SwapLong(uint32_t a)
          a = 0;
    }
    return a;
-} 
+}
 
 /**
  * \brief   Swaps back the bytes of 8-byte long 'double' accordingly to
@@ -580,15 +587,15 @@ double Document::SwapDouble(double a)
 //
 // -----------------File I/O ---------------
 /**
- * \brief  Tries to open the file \ref Document::Filename and
+ * \brief  Tries to open the file Document::Filename and
  *         checks the preamble when existing,
  *         or if the file starts with an ACR-NEMA look-like element.
- * @return The FILE pointer on success, 0 on failure. 
+ * @return The FILE pointer on success, 0 on failure.
  */
 std::ifstream *Document::OpenFile()
 {
    HasDCMPreamble = false;
-   if (Filename.length() == 0) 
+   if (Filename.length() == 0)
    {
       return 0;
    }
@@ -603,13 +610,13 @@ std::ifstream *Document::OpenFile()
    if ( ! *Fp )
    {
    // Don't user gdcmErrorMacro :
-   // a spurious message will appear when you use, for instance 
+   // a spurious message will appear when you use, for instance
    // gdcm::FileHelper *fh = new gdcm::FileHelper( outputFileName );
    // to create outputFileName.
-   
-   // FIXME : if the upper comment is still usefull 
+
+   // FIXME : if the upper comment is still usefull
    //         --> the constructor is not so good ...
-   
+
       gdcmWarningMacro( "Cannot open file: " << Filename.c_str());
       delete Fp;
       Fp = 0;
@@ -617,7 +624,7 @@ std::ifstream *Document::OpenFile()
       //exit(1); // No function is allowed to leave the application instead
                  // of warning the caller
    }
+
    uint16_t zero = 0;
    Fp->read((char*)&zero, (size_t)2);
    if ( Fp->eof() )
@@ -625,7 +632,7 @@ std::ifstream *Document::OpenFile()
       CloseFile();
       return 0;
    }
+
    //-- DICOM --
    Fp->seekg(126L, std::ios::cur);  // Once per Document
    char dicm[4]; // = {' ',' ',' ',' '};
@@ -635,7 +642,7 @@ std::ifstream *Document::OpenFile()
       CloseFile();
       return 0;
    }
-   
+
    if ( memcmp(dicm, "DICM", 4) == 0 )
    {
       HasDCMPreamble = true;
@@ -645,13 +652,13 @@ std::ifstream *Document::OpenFile()
    //-- Broken ACR or DICOM (?) with no Preamble; may start with a Shadow Group --
    // FIXME : We cannot be sure the preable is only zeroes..
    //         (see ACUSON-24-YBR_FULL-RLE.dcm )
-   if ( 
+   if (
        zero == 0x0001 || zero == 0x0100 || zero == 0x0002 || zero == 0x0200 ||
        zero == 0x0003 || zero == 0x0300 || zero == 0x0004 || zero == 0x0400 ||
        zero == 0x0005 || zero == 0x0500 || zero == 0x0006 || zero == 0x0600 ||
        zero == 0x0007 || zero == 0x0700 || zero == 0x0008 || zero == 0x0800 ||
        zero == 0x0028 || 0x2800    // worse : some ACR-NEMA like files 
-                                   // start 00028 group ?!?      
+                                   // start 00028 group ?!?
        )
    {
       std::string msg = Util::Format(
@@ -660,19 +667,19 @@ std::ifstream *Document::OpenFile()
       gdcmWarningMacro( msg.c_str() );
       return Fp;
    }
-   
+
    // -- Neither ACR/No Preamble Dicom nor DICOMV3 file
    CloseFile();
    // Don't user Warning nor Error, not to pollute the output
    // while directory recursive parsing ...
    gdcmDebugMacro( "Neither ACR/No Preamble Dicom nor DICOMV3 file: "
-                      << Filename.c_str()); 
+                      << Filename.c_str());
    return 0;
 }
 
 /**
- * \brief closes the file  
- * @return  TRUE if the close was successfull 
+ * \brief closes the file
+ * @return  TRUE if the close was successfull
  */
 bool Document::CloseFile()
 {
@@ -686,12 +693,12 @@ bool Document::CloseFile()
 }
 
 /**
- * \brief Writes in a file all the Entries (Dicom Elements) 
+ * \brief Writes in a file all the Entries (Dicom Elements)
  * @param fp file pointer on an already open file (actually: Output File Stream)
- * @param filetype Type of the File to be written 
+ * @param filetype Type of the File to be written
  *          (ACR-NEMA, ExplicitVR, ImplicitVR)
  */
-void Document::WriteContent(std::ofstream *fp, FileType filetype)
+void Document::WriteContent(std::ofstream *fp, FileType filetype, bool, bool)
 {
    // Skip if user wants to write an ACR-NEMA file
 
@@ -719,8 +726,10 @@ void Document::WriteContent(std::ofstream *fp, FileType filetype)
     * --> was too much tricky / we were [in a hurry / too lazy]
     * --> We don't write the element 0x0000 (group length)
     */
-
-   ElementSet::WriteContent(fp, filetype); // This one is recursive
+ // This one is recursive
+ // false : outside MetaElements
+ // false : outside Sequence
+   ElementSet::WriteContent(fp, filetype, false, false);
 }
 
 // -----------------------------------------
@@ -849,7 +858,7 @@ void Document::LoadEntryBinArea(DataEntry *entry)
 //}
 
 /**
- * \brief   Compares two documents, according to \ref DicomDir rules
+ * \brief   Compares two documents, according to DicomDir rules
  * \warning Does NOT work with ACR-NEMA files
  * \todo    Find a trick to solve the pb (use RET fields ?)
  * @param   document to compare with current one
@@ -918,7 +927,7 @@ bool Document::operator<(Document &document)
 
 /**
  * \brief Reads a given length of bytes
- *       (in order to avoid to many CPU time consuming fread-s)
+ *       (in order to avoid to many CPU time-consuming fread-s)
  * @param l length to read 
  */
 void Document::ReadBegBuffer(size_t l)
@@ -1000,7 +1009,7 @@ uint32_t Document::ReadInt32()
 uint32_t Document::GetInt32()
 {
    uint32_t g = *((uint32_t*)PtrBegBuffer);
-   g = SwapLong(g); 
+   g = SwapLong(g);
    PtrBegBuffer+=4;
    return g;
 }
@@ -1008,7 +1017,7 @@ uint32_t Document::GetInt32()
 /**
  * \brief   Re-computes the length of the Dicom group 0002.
  */
-int Document::ComputeGroup0002Length( ) 
+int Document::ComputeGroup0002Length( )
 {
    uint16_t gr;
    VRKey vr;
@@ -1031,10 +1040,10 @@ int Document::ComputeGroup0002Length( )
             vr = entry->GetVR();
 
             //if ( (vr == "OB")||(vr == "OW")||(vr == "UT")||(vr == "SQ"))
-            // (no SQ, OW, UT in group 0x0002;)
+            // (no SQ, OW, OL, UT in group 0x0002;)
                if ( vr == "OB" ) 
                {
-                  // explicit VR AND (OB, OW, SQ, UT) : 4 more bytes
+                  // explicit VR AND (OB, OW, OL, SQ, UT, UN) : 4 more bytes
                   groupLength +=  4;
                }
             groupLength += 2 + 2 + 4 + entry->GetLength();   
@@ -1107,20 +1116,19 @@ void Document::ParseDES(DocEntrySet *set, long offset,
    bool delim_mode_intern = delim_mode;
    bool first = true;
    gdcmDebugMacro( "Enter in ParseDES, delim-mode " <<  delim_mode
-                     << " at offset " << std::hex << "0x(" << offset << ")" );    
+                     << " at offset " << std::hex << "0x(" << offset << ")" );
    while (true)
    {
    
    ///\todo FIXME : On 64 bits processors, tellg gives unexpected results after a while ?
-   ///              Probabely a bug in gdcm code somwhere (some memory erased ?)
+   ///              Probabely a bug in gdcm code somewhere (some memory erased ?)
 
 // Uncomment to track the bug
-/*   
-   if( Debug::GetDebugFlag() )   
+
+   if( Debug::GetDebugFlag() )
       std::cout << std::dec <<"(long)(Fp->tellg()) " << (long)(Fp->tellg()) // in Debug mode
                 << std::hex << " 0x(" <<(long)(Fp->tellg()) <<  ")" << std::endl;
- */ 
+
    // if ( !delim_mode && ((long)(Fp->tellg())-offset) >= l_max) // Once per DocEntry   
       if ( !delim_mode ) // 'and then' doesn't exist in C++ :-(
          if ( ((long)(Fp->tellg())-offset) >= l_max) // Once per DocEntry, when no delim mode
@@ -1137,7 +1145,7 @@ void Document::ParseDES(DocEntrySet *set, long offset,
       
       // Uncoment this cerr line to be able to 'follow' the DocEntries
       // when something *very* strange happens
-      if( Debug::GetDebugFlag() ) 
+      if( Debug::GetDebugFlag() )
          std::cerr<<newDocEntry->GetKey()<<" "<<newDocEntry->GetVR()<<std::endl;
 
        // an Item Starter found elsewhere but in the first position
@@ -1147,20 +1155,20 @@ void Document::ParseDES(DocEntrySet *set, long offset,
       if ( !first && newDocEntry->IsItemStarter() )
       {
          // Debug message within the method !
-         newDocEntry = Backtrack(newDocEntry);
+         newDocEntry = Backtrack(newDocEntry, set);
       }
       else
-      { 
-         PreviousDocEntry = newDocEntry; 
+      {
+         PreviousDocEntry = newDocEntry;
       }
+
       used = true;
       newDataEntry = dynamic_cast<DataEntry*>(newDocEntry);
 
-      if ( newDataEntry )  
+      if ( newDataEntry )
       {
          //////////////////////////// DataEntry
+
          //vr = newDocEntry->GetVR(); // useless ?
 
          if ( !set->AddEntry( newDataEntry ) )
@@ -1197,7 +1205,7 @@ void Document::ParseDES(DocEntrySet *set, long offset,
             }
          }
 
-         bool delimitor = newDataEntry->IsItemDelimitor(); 
+         bool delimitor = newDataEntry->IsItemDelimitor();
          bool outOfBounds = false;
          if (!delim_mode )
             if ( ((long)(Fp->tellg())-offset) >= l_max ) //Once per DataEntry when no delim mode
@@ -1221,7 +1229,7 @@ void Document::ParseDES(DocEntrySet *set, long offset,
       {
          /////////////////////// SeqEntry :  VR = "SQ"
 
-         unsigned long l = newDocEntry->GetReadLength();          
+         unsigned long l = newDocEntry->GetReadLength();
          if ( l != 0 ) // don't mess the delim_mode for 'zero-length sequence'
          {
             if ( l == 0xffffffff )
@@ -1282,11 +1290,11 @@ void Document::ParseDES(DocEntrySet *set, long offset,
                                << " at offset 0x(" << std::hex
                                << newDocEntry->GetOffset() << ")");
 
-            ParseSQ( newSeqEntry, 
-                     newDocEntry->GetOffset(),
-                     l, delim_mode_intern);
+            bool res = ParseSQ( newSeqEntry, 
+                         newDocEntry->GetOffset(),
+                         l, delim_mode_intern);
 
-            gdcmDebugMacro( "Exit from ParseSQ, delim " << delim_mode_intern);
+            gdcmDebugMacro( "Exit from ParseSQ, delim " << delim_mode_intern << " -->return : " << res);
          }
          if ( !set->AddEntry( newSeqEntry ) )
          {
@@ -1305,7 +1313,7 @@ void Document::ParseDES(DocEntrySet *set, long offset,
  
          if ( !delim_mode ) // 'and then' doesn't exist in C++ :-(
             if ( ((long)(Fp->tellg())-offset) >= l_max) // Once per SeqEntry when no delim mode
-     
+
          {
             if ( !used )
                newDocEntry->Delete();
@@ -1318,15 +1326,20 @@ void Document::ParseDES(DocEntrySet *set, long offset,
          newDocEntry->Delete();
       }
       first = false;
+      
+      if (UnexpectedEOF) // some terminator was missing
+      {
+         break;
+      }
    }                               // end While
    gdcmDebugMacro( "Exit from ParseDES, delim-mode " << delim_mode );
 }
 
 /**
  * \brief   Parses a Sequence ( SeqEntry after SeqEntry)
- * @return  parsed length for this level
+ * @return  false if expected fff0,e000 not found
  */ 
-void Document::ParseSQ( SeqEntry *seqEntry,
+bool Document::ParseSQ( SeqEntry *seqEntry,
                         long offset, long l_max, bool delim_mode)
 {
    int SQItemNumber = 0;
@@ -1340,8 +1353,10 @@ void Document::ParseSQ( SeqEntry *seqEntry,
 
       if ( !newDocEntry )
       {
+         // The most frequent is when a SQ terminator is missing (?!?)
          gdcmWarningMacro("in ParseSQ : should never get here!");
-         break;
+         UnexpectedEOF = true;
+         return false;
       }
       if ( delim_mode )
       {
@@ -1372,9 +1387,23 @@ void Document::ParseSQ( SeqEntry *seqEntry,
       {
          dlm_mod = false;
       }
-            
+
+      // avoid infinite loop when Bad assumption was made on illegal 'unknown length' UN //JPRx
+    
+      if (offsetStartCurrentSQItem <= OffsetOfPreviousParseDES)
+      {
+         gdcmWarningMacro("Bad assumption was made on illegal 'unknown length' UN!" << std::endl <<
+                          "OffsetOfPreviousParseDES " << std::hex << OffsetOfPreviousParseDES
+                           << " offsetStartCurrentSQItem " << offsetStartCurrentSQItem);
+         /// \todo when  "Bad assumption (SQ) on illegal 'unknown length' UN", Backtrack again + try OB      
+         return false; 
+      }
+      else 
+      {
+         OffsetOfPreviousParseDES = offsetStartCurrentSQItem;
+      }
+
       // fill up the current SQItem, starting at the beginning of fff0,e000
-      
       Fp->seekg(offsetStartCurrentSQItem, std::ios::beg);        // Once per SQItem
       ParseDES(itemSQ, offsetStartCurrentSQItem, l+8, dlm_mod);
       offsetStartCurrentSQItem = Fp->tellg();                    // Once per SQItem
@@ -1389,15 +1418,17 @@ void Document::ParseSQ( SeqEntry *seqEntry,
          break;
       }
    }
+   return true;
 }
 
 /**
  * \brief   When a private Sequence + Implicit VR is encountered
  *           we cannot guess it's a Sequence till we find the first
  *           Item Starter. We then backtrack to do the job.
- * @param   docEntry Item Starter that warned us 
+ * @param   docEntry Item Starter that warned us
+ * @param   set DocEntrySet (ElementSet/SQItem) the DocEntry will belong
  */
-DocEntry *Document::Backtrack(DocEntry *docEntry)
+DocEntry *Document::Backtrack(DocEntry *docEntry, DocEntrySet *set)
 {
    // delete the Item Starter, built erroneously out of any Sequence
    // it's not yet in the HTable/chained list
@@ -1412,7 +1443,8 @@ DocEntry *Document::Backtrack(DocEntry *docEntry)
    gdcmDebugMacro( "Backtrack :" << std::hex << group 
                                  << "|" << elem
                                  << " at offset 0x(" <<offset << ")" );
-   RemoveEntry( PreviousDocEntry );
+   
+   set->RemoveEntry( PreviousDocEntry );
 
    // forge the Seq Entry
    DocEntry *newEntry = NewSeqEntry(group, elem);
@@ -1422,7 +1454,7 @@ DocEntry *Document::Backtrack(DocEntry *docEntry)
    // Move back to the beginning of the Sequence
 
    Fp->seekg(offset, std::ios::beg); // Only for Shadow Implicit VR SQ
-   return newEntry;
+   return newEntry; // It will added where it has to be!
 }
 
 /**
@@ -1482,7 +1514,7 @@ void Document::LoadDocEntry(DocEntry *entry, bool forceLoad)
          return;
       }
    }
-   
+
    /// \todo: a method that *doesn't* load anything (maybe with MaxSizeLoadEntry=0 ?)
    ///       + a ForceLoad call on the +/- 20 'usefull' fields  
    ///       Allow user to tell the fields he wants to ForceLoad 
@@ -1502,20 +1534,30 @@ void Document::FindDocEntryLength( DocEntry *entry )
 {
    const VRKey &vr  = entry->GetVR();
    uint16_t length16;       
-   
    if ( Filetype == ExplicitVR && !entry->IsImplicitVR() ) 
    {
-      if ( vr == "OB" || vr == "OW" || vr == "SQ" || vr == "UT" 
-                                                           || vr == "UN" )
+
+   // WARNING :
+   //
+   // For some images, length of UN elements is coded on 2 bytes (instead of 4)
+   // There are *not* readable !
+   // You can make a quick and dirty patch, commenting out 
+   //| vr == "UN"
+   // in the following line.
+   // (the 'straight' images will no longer be readable ...)
+
+      if ( vr == "OB" || vr == "OW" || vr == "OL" || vr == "SQ" || vr == "UT" 
+                                                         || vr == "UN"  || changeFromUN == true)
       {
+         changeFromUN = false;
          // The following reserved two bytes (see PS 3.5-2003, section
          // "7.1.2 Data element structure with explicit vr", p 27) must be
          // skipped before proceeding on reading the length on 4 bytes.
 
-         //Fp->seekg( 2L, std::ios::cur); // Once per OW,OB,SQ DocEntry
-         uint32_t length32 = ReadInt32(); // Once per OW,OB,SQ DocEntry
+         //Fp->seekg( 2L, std::ios::cur); // Once per OB,OW,OL,UT,UN,SQ DocEntry
+         uint32_t length32 = ReadInt32(); // Once per OB,OW,OL,UT,UN,SQ DocEntry
          CurrentOffsetPosition+=4;
-         if ( (vr == "OB" || vr == "OW") && length32 == 0xffffffff ) 
+         if ( (vr == "OB" || vr == "OW" || vr == "OL") && length32 == 0xffffffff ) 
          {
             uint32_t lengthOB;
             try 
@@ -1549,11 +1591,9 @@ void Document::FindDocEntryLength( DocEntry *entry )
          FixDocEntryFoundLength(entry, length32); 
          return;
       }
-
       // Length is encoded on 2 bytes.
       //length16 = ReadInt16();
       length16 = GetInt16();
-
       // 0xffff means that we deal with 'No Length' Sequence 
       //        or 'No Length' SQItem
       if ( length16 == 0xffff) 
@@ -1603,7 +1643,8 @@ uint32_t Document::FindDocEntryLengthOBOrOW()
       uint16_t elem;
 
       try
-      {
+      {  ///\todo make sure there is never OL encoded pixel data!
+      
          //group = ReadInt16(); // Once per fragment (if any) of OB,OW DataElements
          //elem  = ReadInt16(); // Once per fragment (if any) of OB,OW DataElements 
          ReadBegBuffer(4); // Once per fragment (if any) of OB,OW DataElements
@@ -1622,8 +1663,8 @@ uint32_t Document::FindDocEntryLengthOBOrOW()
       {
          gdcmWarningMacro( 
               "Neither an Item tag nor a Sequence delimiter tag on :" 
-           << std::hex << group << " , " << elem 
-           << ")" );
+           << std::hex << group << "|" << elem << ") Pos. on entry was 0x(" <<positionOnEntry<< ") "
+            );
   
          Fp->seekg(positionOnEntry, std::ios::beg); // Once per fragment (if any) of OB,OW DataElements
          throw FormatUnexpected( 
@@ -1661,11 +1702,11 @@ VRKey Document::FindDocEntryVR()
    // Delimiters (0xfffe), are not explicit VR ... 
    if ( CurrentGroup == 0xfffe )
       return GDCM_VRUNKNOWN;
-         
-   //long positionOnEntry;     
-   //if( Debug::GetWarningFlag() ) 
+
+   //long positionOnEntry;
+   //if( Debug::GetWarningFlag() )
    //  positionOnEntry = Fp->tellg(); // Only in Warning Mode
-   
+
    // Warning: we believe this is explicit VR (Value Representation) because
    // we used a heuristic that found "UL" in the first tag and/or
    // 'Transfer Syntax' told us it is.
@@ -1681,26 +1722,17 @@ VRKey Document::FindDocEntryVR()
    //Fp->read(&(vr[0]),(size_t)2);
    vr[0] = *PtrBegBuffer++;
    vr[1] = *PtrBegBuffer++;
-   
+
    //if ( !CheckDocEntryVR(vr) ) // avoid useless function call
    if ( !Global::GetVR()->IsValidVR(vr) )
    {
-/*   
-//      std::cout << "================================================================Unknown VR" 
-               << std::hex << "0x(" 
-                        << (unsigned int)vr[0] << "|" << (unsigned int)vr[1] 
-                        << ")" << "for : " <<  CurrentGroup
-                        << " at offset : 0x(" << positionOnEntry << ")"
-                        << std::endl;
-*/
-      gdcmWarningMacro( "Unknown VR " << std::hex << "0x(" 
-                        << (unsigned int)vr[0] << "|" << (unsigned int)vr[1] 
-                        << ")"  
-                        << " at offset : 0x(" << CurrentOffsetPosition-4<< ") for group " << CurrentGroup
-                        );
+
+      gdcmWarningMacro( "Unknown VR " << vr.GetHexaRepresentation() << std::hex
+                        << " at offset : 0x(" << CurrentOffsetPosition-4
+                        << ") for group " << std::hex << CurrentGroup );
 
       //Fp->seekg(positionOnEntry, std::ios::beg); //JPRx
-      //Fp->seekg((long)-2, std::ios::cur);// only for unrecognized VR (?!?) 
+      //Fp->seekg((long)-2, std::ios::cur);// only for unrecognized VR (?!?)
                                          //see :MR_Philips_Intera_PrivateSequenceExplicitVR.dcm
       PtrBegBuffer-=2;
       return GDCM_VRUNKNOWN;
@@ -1737,7 +1769,7 @@ void Document::SkipDocEntry(DocEntry *entry)
 void Document::SkipToNextDocEntry(DocEntry *currentDocEntry) 
 {
    long l = currentDocEntry->GetReadLength();
-   if ( l == -1 ) // length = 0xffff shouldn't appear here ...
+   if ( (uint32_t) l == (uint32_t)-1 ) // length = 0xffff shouldn't appear here ...
                   // ... but PMS imagers happen !
       return;
    Fp->seekg((size_t)(currentDocEntry->GetOffset()), std::ios::beg); //FIXME :each DocEntry
@@ -1766,16 +1798,18 @@ void Document::FixDocEntryFoundLength(DocEntry *entry,
       entry->SetLength(0);
       return;  // return ASAP; don't waist time on useless tests
    }
-      
+
    uint16_t gr   = entry->GetGroup();
    uint16_t elem = entry->GetElement(); 
      
    if ( foundLength % 2)
    {
-      gdcmWarningMacro( "Warning : Tag with uneven length " << foundLength
-        <<  " in x(" << std::hex << gr << "," << elem <<")");
+      gdcmWarningMacro( "Warning : Tag (" << std::hex << gr << "|" << elem << ") with uneven length " 
+        << std::dec << foundLength << " 0x(" << std::hex << foundLength << ") "
+        //<< " at offset x(" << offset << ")"
+       );
    }
-      
+
    //////// Fix for some naughty General Electric images.
    // Allthough not recent many such GE corrupted images are still present
    // on Creatis hard disks. Hence this fix shall remain when such images
@@ -1797,7 +1831,7 @@ void Document::FixDocEntryFoundLength(DocEntry *entry,
    // 'Leonardo' source. Hence, one might consider commenting out the
    // following fix on efficiency reasons.
    else if ( gr == 0x0009 && ( elem == 0x1113 || elem == 0x1114 ) )
-   { 
+   {
    // Ideally we should check we are in Explicit and double check
    // that VR=UL... this is done properly in gdcm2
       if( foundLength == 6 )
@@ -1805,7 +1839,7 @@ void Document::FixDocEntryFoundLength(DocEntry *entry,
          gdcmWarningMacro( "Replacing Length from 6 into 4" );
          foundLength = 4;
          entry->SetReadLength(4); // a bug is to be fixed !
-      } 
+      }
       else if ( foundLength%4 )
       {
         gdcmErrorMacro( "This looks like to a buggy Siemens DICOM file."
@@ -1871,7 +1905,7 @@ bool Document::IsDocEntryAnInteger(DocEntry *entry)
          // message and proceed on parsing (while crossing fingers).
          long filePosition = Fp->tellg(); // Only when elem 0x0000 length is not 4 (?!?)
          (void)filePosition;
-         gdcmWarningMacro( "Erroneous Group Length element length  on : (" 
+         gdcmWarningMacro( "Erroneous Group Length element length  on : ("
            << std::hex << group << " , " << elem
            << ") -before- position x(" << filePosition << ")"
            << "lgt : " << length );
@@ -2060,10 +2094,9 @@ bool Document::CheckSwap()
          //  Find a trick to tell it the caller...
       
          s16 = *((uint16_t *)(deb));
+
          gdcmDebugMacro("not a DicomV3 nor a 'clean' ACR/NEMA;"
-                     << " (->despaired wild guesses !)"); 
-      
+                     << " (->despaired wild guesses !)");
          switch ( s16 )
          {
             case 0x0001 :
@@ -2076,7 +2109,9 @@ bool Document::CheckSwap()
             case 0x0008 :
             case 0x0028 :
                SwapCode = 1234;
-               Filetype = ACR;
+               // Brute hack to allow reading DICOM RT files
+               //Filetype = ACR;  // DICOM RT are *not* ACR-Nema files!
+               Filetype = ExplicitVR;
                return true;
             case 0x0100 :
             case 0x0200 :
@@ -2091,9 +2126,48 @@ bool Document::CheckSwap()
                Filetype = ACR;
                return true;
             default :
-               gdcmWarningMacro("ACR/NEMA unfound swap info (Hopeless !)");
-               Filetype = Unknown;
-               return false;
+
+               s16 = *((uint16_t *)(deb));
+               if (s16 != 0x0000)
+                   return false;
+               s16 = *((uint16_t *)(deb+2));
+
+               Fp->seekg ( 0L, std::ios::beg); // Once per Document
+               CurrentOffsetPosition = 0;
+               switch(s16)  // try an other trick!
+                            // -> to be able to decode 0029|1010 DataElement
+                            // -> and be not less cleaver than dcmdump ;-)
+               {
+                  case 0x0004 :
+                     SwapCode = 1234; 
+                     break;
+                  case 0x0400 :
+                     SwapCode = 3412;
+                     break;
+                  default:
+                     gdcmWarningMacro("ACR/NEMA unfound swap info (Hopeless !)");
+                     Filetype = Unknown;
+                     return false;
+               }
+               // Check if next 2 bytes are a VR
+               // Probabely something more time-consuming exists with std::string
+               const char VRvalues[] = "AEASATCSDADTFLFDISLOLTPNSHSLSSSTTMUIULUSUTOBOWOLOFATUNSQRT";
+               int nbVal = 29;
+               const char *pt = VRvalues;
+               for (int i=0;i<nbVal;i++)
+               {
+                  if(*(deb+4) == *pt++) {
+                    if(*(deb+5) == *pt++) {
+                       Filetype = ExplicitVR;
+                       return true;
+                    }
+                    else {
+                       pt++;
+                    }
+                 }
+              }
+              Filetype = ImplicitVR;
+              return true;
          }
    }
 }
@@ -2163,10 +2237,11 @@ DocEntry *Document::ReadNextDocEntry()
       // header parsing has to be considered as finished.
       return 0;
    }
-   
+
+   changeFromUN = false;
    CurrentGroup = GetInt16();
    CurrentElem  = GetInt16();
-   
+
    // In 'true DICOM' files Group 0002 is always little endian
    if ( HasDCMPreamble )
    {
@@ -2191,18 +2266,39 @@ DocEntry *Document::ReadNextDocEntry()
          realVR = "UL";
       }
 
-      // Commented out in order not to generate 'Shadow Groups' where some 
+      // Was commented out in order not to generate 'Shadow Groups' where some 
       // Data Elements are Explicit VR and some other ones Implicit VR
-      // (Stupid MatLab DICOM Reader couldn't read gdcm-written images)
-      /*
-      else if (CurrentGroup%2 == 1 &&  
-                               (CurrentElem >= 0x0010 && CurrentElem <=0x00ff ))
-      {  
-      // DICOM PS 3-5 7.8.1 a) states that those 
-      // (gggg-0010->00FF where gggg is odd) attributes have to be LO
-         realVR = "LO";
+      // -> Better we fix the problem at Write time
+     
+      else if (CurrentGroup%2 == 1 )
+      {
+         if (CurrentElem >= 0x0010 && CurrentElem <=0x00ff )
+         {
+            // DICOM PS 3-5 7.8.1 a) states that :
+            // Private Creator Data Elements numbered (gggg,0010-00FF) (gggg is odd)
+            // attributes have to be LO (Long String) and the VM shall be equal to 1
+            realVR = "LO";
+    
+            // Seems not to be true
+            // Still in gdcmtk, David Clunnie disagrees, Marco Eichelberg says it's OK ...
+            // We let it for a while? 
+            //(We should check length==4, for more security, but we don't have it yet !)
+         }
+         else if ( CurrentElem == 0x0001)
+         {
+            realVR = "UL"; // Private Group Length To End
+         }
+         else  // check the private dictionary for shadow elements when Implicit VR!
+         {
+            DictEntry *dictEntry = GetDictEntry(CurrentGroup,CurrentElem);
+            if ( dictEntry )
+            {
+               realVR = dictEntry->GetVR(); 
+               dictEntry->Unregister(); // GetDictEntry registered it
+            }   
+         }               
       }
-      */
+      
       else
       {
          DictEntry *dictEntry = GetDictEntry(CurrentGroup,CurrentElem);//only when ImplicitVR
@@ -2214,6 +2310,28 @@ DocEntry *Document::ReadNextDocEntry()
       }
    }
 
+   // if UN found, let's check the dictionary, and trust it!
+   // (maybe a private dictionary exists?)
+   else if (vr == "UN")
+   {
+      DictEntry *dictEntry = GetDictEntry(CurrentGroup,CurrentElem);
+      if ( dictEntry )
+      {
+         realVR = dictEntry->GetVR();
+         dictEntry->Unregister(); // GetDictEntry registered it
+
+         // for VR = "UN", length is always stored on 4 bytes.
+         // remember this info, in order not to crash later
+         changeFromUN=true;
+         /// \todo : fixme If inside a vr = "UN" DataElement (but SQ according to a private dictionnary)
+         ///         there is some more vr = "UN" DataElements, it will probabely fail.
+         ///         --> find a -non time consuming- trick to store changeFromUN info at DataElement level,
+         ///         not at the Document level.
+         /// --> ?!? JPR
+  
+      }   
+   }
+
    DocEntry *newEntry;
    //if ( Global::GetVR()->IsVROfSequence(realVR) )
    if (realVR == "SQ")
@@ -2232,14 +2350,14 @@ DocEntry *Document::ReadNextDocEntry()
       {
          // We thought this was explicit VR, but we end up with an
          // implicit VR tag. Let's backtrack.
-         //if ( newEntry->GetGroup() != 0xfffe ) 
+
+         //if ( newEntry->GetGroup() != 0xfffe )
          if (CurrentGroup != 0xfffe)
-         { 
+         {
             int offset = Fp->tellg();//Only when heuristic for Explicit/Implicit was wrong
 
             gdcmWarningMacro("Entry (" << newEntry->GetKey() << ") at x("
-                     <<  offset << ") should be Explicit VR");
+                     <<  std::hex << offset << ") should be Explicit VR");
           }
       }
       newEntry->SetImplicitVR();
@@ -2346,7 +2464,7 @@ void Document::HandleOutOfGroup0002(uint16_t &group, uint16_t &elem)
 
       if ( s == TS::ImplicitVRLittleEndian 
         ||
-          s == TS::ImplicitVRBigEndianPrivateGE  
+          s == TS::ImplicitVRBigEndianPrivateGE
          )
       {
          Filetype = ImplicitVR;
@@ -2359,10 +2477,10 @@ void Document::HandleOutOfGroup0002(uint16_t &group, uint16_t &elem)
       // to trust manufacturers.
       // (we often find 'Implicit VR' tag, 
       // even when Transfer Syntax tells us it's Explicit ...
-      
+
        // NEVER trust the meta elements!
        // (see what ezDICOM does ...)
-             
+
       /*
       if ( s ==  TS::ExplicitVRBigEndian )
       {
@@ -2383,34 +2501,34 @@ void Document::HandleOutOfGroup0002(uint16_t &group, uint16_t &elem)
        elem  = SwapShort(elem); 
        // not what we where told (by meta elements) !
        gdcmDebugMacro("Transfer Syntax Name = ["       
-                       << GetTransferSyntaxName() << "]" );         
+                       << GetTransferSyntaxName() << "]" );
     }
-      
+
       /// \todo  find a trick to warn user and stop processing
-            
+
       if ( s == TS::DeflatedExplicitVRLittleEndian)
       {
            gdcmWarningMacro("Transfer Syntax [" 
                         << GetTransferSyntaxName() << "] :"
                         << " not yet dealt with ");
-           return;       
+           return;
       }
-      
+
       // The following shouldn't occur very often
       // Let's check at the very end.
 
       if ( ts == GDCM_UNKNOWN )
       {
          gdcmDebugMacro("True DICOM File, with NO Transfer Syntax (?!) " );
-         return;      
+         return;
       }
-      
+
       if ( !Global::GetTS()->IsTransferSyntax(ts) )
       {
          gdcmWarningMacro("True DICOM File, with illegal Transfer Syntax: [" 
                           << ts << "]");
          return;
-      }      
+      }
 }
 
 //-----------------------------------------------------------------------------