1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-04 18:06:22 +03:00
occt/src/PCDM/PCDM_ReadWriter.cxx
mpv d5c71e2057 0031918: Application Framework - New binary format for fast reading part of OCAF document
Implementation of new format for quick reading and writing parts of the documents (sub-set of labels and sub-set of attributes). It consists in writing shapes and all their contents right in the TNaming_NamedShape attribute placement and skipping the shape section. New format 12 for Binary file types is assigned to this version.

Added PCDM_ReaderFilter class that could be used in Open methods of TDocStd_Application. If it is defined, it allows to read:
- into already opened document in append mode AppendMode_Protect (do not overwrite existing attributes) or AppendMode_Overwrite
- only specified sub-trees of the document using AddPath (const TCollection_AsciiString& theEntryToRead)
- only specified attributes using AddRead (const TCollection_AsciiString& theRead) where theRead could be "TDataStd_Name", for example
- to skip specified attributes read using AddSkipped (const TCollection_AsciiString& theSkipped) where theSkipped could be "TDF_Reference", for example

The current limitations:
- only in Bin format
- if shapes have in the document shared topology, loaded in "append" mode in different "load" operations, they will have no shared topology anymore

Modifications:
BinLDrivers and BinDrivers packages - modifications related to the quick part tree format flag usage, skipping shape section writing and adding labels sizes into the document to be able to pass labels during the reading quickly.
BinObjMgt_Persistent and BinObjMgt_Position - to add possibility to write directly into the stream some data just after the attribute. Before this record a data-size is recorded.
BinMXCAFDoc package modifications to write BinMXCAFDoc_LocationDriver location in the same way as shapes write location data right after the attribute (empty) data in this new format.
BinTools package: creation of ShapeReader and ShapeWriter classes with same root class ShapeSetBase with ShapeSet class. These classes allows to write/read shapes directly to the stream. If some object is already in the stream, write a reference - relative position of the duplicated object. The old format of documents is still supported by Bin_ToolsShapeSet class.
PCDM_ReaderFilter - Allows the user to create a reading filter. It contains algorithm to browse labels tree quickly, without usage of referencing by entry-strings.
TDocStd, CDF and some other packages are changed for supporting reading filters API and options.

Tests, documentation and upgrade information are also added for both issues: 31839 and 31918 related to this commit.
2021-07-09 19:13:59 +03:00

237 lines
7.8 KiB
C++

// Created on: 1997-12-09
// Created by: Jean-Louis Frenkel
// Copyright (c) 1997-1999 Matra Datavision
// Copyright (c) 1999-2014 OPEN CASCADE SAS
//
// This file is part of Open CASCADE Technology software library.
//
// This library is free software; you can redistribute it and/or modify it under
// the terms of the GNU Lesser General Public License version 2.1 as published
// by the Free Software Foundation, with special exception defined in the file
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
// distribution for complete text of the license and disclaimer of any warranty.
//
// Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement.
#include <CDM_Document.hxx>
#include <PCDM.hxx>
#include <PCDM_DOMHeaderParser.hxx>
#include <PCDM_ReadWriter.hxx>
#include <PCDM_ReadWriter_1.hxx>
#include <Standard_ErrorHandler.hxx>
#include <Standard_Type.hxx>
#include <Storage_BaseDriver.hxx>
#include <Storage_Data.hxx>
#include <Storage_HeaderData.hxx>
#include <Storage_TypeData.hxx>
#include <TCollection_AsciiString.hxx>
#include <TCollection_ExtendedString.hxx>
#include <TColStd_HSequenceOfAsciiString.hxx>
#include <UTL.hxx>
IMPLEMENT_STANDARD_RTTIEXT(PCDM_ReadWriter,Standard_Transient)
#define FILE_FORMAT "FILE_FORMAT: "
static TCollection_ExtendedString TryXmlDriverType
(const TCollection_AsciiString& theFileName);
static TCollection_ExtendedString TryXmlDriverType (Standard_IStream& theIStream);
//=======================================================================
//function : Open
//purpose :
//=======================================================================
void PCDM_ReadWriter::Open (const Handle(Storage_BaseDriver)& aDriver,
const TCollection_ExtendedString& aFileName,
const Storage_OpenMode aMode)
{
Storage_Error error = UTL::OpenFile(aDriver,aFileName,aMode);
if(error != Storage_VSOk) {
Standard_SStream aMsg; aMsg << "could not open the file: ";
aMsg << aFileName;
switch (error) {
case Storage_VSOpenError: aMsg << "; file was not found or permission denied"; break;
case Storage_VSAlreadyOpen: aMsg<< "; file was already opened";
default:
break;
}
aMsg << (char)0;
throw Standard_Failure(aMsg.str().c_str());
}
}
//=======================================================================
//function : Reader
//purpose :
//=======================================================================
Handle(PCDM_ReadWriter) PCDM_ReadWriter::Reader (const TCollection_ExtendedString&)
{
return (new PCDM_ReadWriter_1);
}
//=======================================================================
//function : Writer
//purpose :
//=======================================================================
Handle(PCDM_ReadWriter) PCDM_ReadWriter::Writer()
{
return (new PCDM_ReadWriter_1);
}
//=======================================================================
//function : WriteFileFormat
//purpose :
//=======================================================================
void PCDM_ReadWriter::WriteFileFormat (const Handle(Storage_Data)& aData,
const Handle(CDM_Document)& aDocument)
{
TCollection_AsciiString ligne(FILE_FORMAT);
ligne += TCollection_AsciiString(aDocument->StorageFormat(),'?');
aData->AddToUserInfo(ligne);
}
//=======================================================================
//function : FileFormat
//purpose :
//=======================================================================
TCollection_ExtendedString PCDM_ReadWriter::FileFormat
(const TCollection_ExtendedString& aFileName)
{
TCollection_ExtendedString theFormat;
Handle(Storage_BaseDriver) theFileDriver;
// conversion to UTF-8 is done inside
TCollection_AsciiString theFileName (aFileName);
if (PCDM::FileDriverType (theFileName, theFileDriver) == PCDM_TOFD_Unknown)
return ::TryXmlDriverType (theFileName);
Standard_Boolean theFileIsOpen(Standard_False);
try {
OCC_CATCH_SIGNALS
Open(theFileDriver,aFileName,Storage_VSRead);
theFileIsOpen=Standard_True;
Storage_HeaderData hd;
hd.Read (theFileDriver);
const TColStd_SequenceOfAsciiString &refUserInfo = hd.UserInfo();
Standard_Boolean found=Standard_False;
for (Standard_Integer i =1; !found && i<= refUserInfo.Length() ; i++) {
if(refUserInfo(i).Search(FILE_FORMAT) != -1) {
found=Standard_True;
theFormat=TCollection_ExtendedString(refUserInfo(i).Token(" ",2).ToCString(),
Standard_True);
}
}
if (!found)
{
Storage_TypeData td;
td.Read (theFileDriver);
theFormat = td.Types()->Value(1);
}
}
catch (Standard_Failure const&) {}
if(theFileIsOpen)
{
theFileDriver->Close();
}
return theFormat;
}
//=======================================================================
//function : FileFormat
//purpose :
//=======================================================================
TCollection_ExtendedString PCDM_ReadWriter::FileFormat (Standard_IStream& theIStream, Handle(Storage_Data)& theData)
{
TCollection_ExtendedString aFormat;
Handle(Storage_BaseDriver) aFileDriver;
if (PCDM::FileDriverType (theIStream, aFileDriver) == PCDM_TOFD_XmlFile)
{
return ::TryXmlDriverType (theIStream);
}
if (!aFileDriver)
{
// type is not recognized, return empty string
return aFormat;
}
aFileDriver->ReadCompleteInfo (theIStream, theData);
for (Standard_Integer i = 1; i <= theData->HeaderData()->UserInfo().Length(); i++)
{
const TCollection_AsciiString& aLine = theData->HeaderData()->UserInfo().Value(i);
if(aLine.Search (FILE_FORMAT) != -1)
{
aFormat = TCollection_ExtendedString (aLine.Token(" ",2).ToCString(), Standard_True);
}
}
return aFormat;
}
//=======================================================================
//function : ::TryXmlDriverType
//purpose : called from FileFormat()
//=======================================================================
static TCollection_ExtendedString TryXmlDriverType
(const TCollection_AsciiString& theFileName)
{
TCollection_ExtendedString theFormat;
PCDM_DOMHeaderParser aParser;
const char * aDocumentElementName = "document";
aParser.SetStartElementName (Standard_CString(aDocumentElementName));
// Parse the file; if there is no error or an error appears before retrieval
// of the DocumentElement, the XML format cannot be defined
if (aParser.parse (theFileName.ToCString()))
{
LDOM_Element anElement = aParser.GetElement();
if (anElement.getTagName().equals (LDOMString(aDocumentElementName)))
theFormat = anElement.getAttribute ("format");
}
return theFormat;
}
//=======================================================================
//function : ::TryXmlDriverType
//purpose : called from FileFormat()
//=======================================================================
static TCollection_ExtendedString TryXmlDriverType (Standard_IStream& theIStream)
{
TCollection_ExtendedString theFormat;
PCDM_DOMHeaderParser aParser;
const char * aDocumentElementName = "document";
aParser.SetStartElementName (Standard_CString(aDocumentElementName));
if (theIStream.good())
{
// Parse the file; if there is no error or an error appears before retrieval
// of the DocumentElement, the XML format cannot be defined
if (aParser.parse (theIStream, Standard_True))
{
LDOM_Element anElement = aParser.GetElement();
if (anElement.getTagName().equals (LDOMString(aDocumentElementName)))
theFormat = anElement.getAttribute ("format");
}
}
return theFormat;
}