From 0435edfe540876baf3daae80b9cefe194ca36c3d Mon Sep 17 00:00:00 2001 From: ichesnok Date: Tue, 12 Sep 2023 11:58:55 +0100 Subject: [PATCH] 0033474: Data Exchange - Implement stream reading into RWMesh interface Stream usage as parameter --- src/RWGltf/RWGltf_CafReader.cxx | 35 +++++++++--------- src/RWGltf/RWGltf_CafReader.hxx | 3 +- src/RWMesh/RWMesh_CafReader.cxx | 16 ++++++++- src/RWMesh/RWMesh_CafReader.hxx | 60 ++++++++++++++++++++++++++----- src/RWObj/RWObj_CafReader.cxx | 7 ++-- src/RWObj/RWObj_CafReader.hxx | 3 +- src/RWObj/RWObj_Reader.cxx | 44 +++++------------------ src/RWObj/RWObj_Reader.hxx | 46 ++++++++++++++++++------ src/VrmlAPI/VrmlAPI_CafReader.cxx | 9 +++-- src/VrmlAPI/VrmlAPI_CafReader.hxx | 10 +++--- src/VrmlData/VrmlData_Scene.cxx | 4 +++ 11 files changed, 149 insertions(+), 88 deletions(-) diff --git a/src/RWGltf/RWGltf_CafReader.cxx b/src/RWGltf/RWGltf_CafReader.cxx index e50e9426dc..6db997df3d 100644 --- a/src/RWGltf/RWGltf_CafReader.cxx +++ b/src/RWGltf/RWGltf_CafReader.cxx @@ -182,16 +182,15 @@ RWGltf_CafReader::RWGltf_CafReader() // Function : performMesh // Purpose : //================================================================ -Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& theFile, +Standard_Boolean RWGltf_CafReader::performMesh (std::istream& theStream, + const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe) { - Message_ProgressScope aPSentry (theProgress, "Reading glTF", 2); + Message_ProgressScope aPSentry(theProgress, "Reading glTF", 2); aPSentry.Show(); - const Handle(OSD_FileSystem)& aFileSystem = OSD_FileSystem::DefaultFileSystem(); - std::shared_ptr aFile = aFileSystem->OpenIStream (theFile, std::ios::in | std::ios::binary); - if (aFile.get() == NULL || !aFile->good()) + if (!theStream.good()) { Message::SendFail (TCollection_AsciiString ("File '") + theFile + "' is not found"); return false; @@ -199,7 +198,7 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t bool isBinaryFile = false; char aGlbHeader[12] = {}; - aFile->read (aGlbHeader, sizeof (aGlbHeader)); + theStream.read (aGlbHeader, sizeof (aGlbHeader)); int64_t aBinBodyOffset = 0; int64_t aBinBodyLen = 0; int64_t aJsonBodyOffset = 0; @@ -218,7 +217,7 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t } char aHeader1[8] = {}; - aFile->read (aHeader1, sizeof (aHeader1)); + theStream.read (aHeader1, sizeof (aHeader1)); const uint32_t* aSceneLen = (const uint32_t* )(aHeader1 + 0); const uint32_t* aSceneFormat = (const uint32_t* )(aHeader1 + 4); @@ -240,16 +239,16 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t Message::SendWarning (TCollection_AsciiString ("File '") + theFile + "' is written using unknown version " + int(*aVer)); } - for (int aChunkIter = 0; !aFile->eof() && aChunkIter < 2; ++aChunkIter) + for (int aChunkIter = 0; !theStream.eof() && aChunkIter < 2; ++aChunkIter) { char aChunkHeader2[8] = {}; - if (int64_t (aFile->tellg()) + int64_t (sizeof (aChunkHeader2)) > int64_t (*aLen)) + if (int64_t (theStream.tellg()) + int64_t (sizeof (aChunkHeader2)) > int64_t (*aLen)) { break; } - aFile->read (aChunkHeader2, sizeof (aChunkHeader2)); - if (!aFile->good()) + theStream.read (aChunkHeader2, sizeof (aChunkHeader2)); + if (!theStream.good()) { Message::SendFail (TCollection_AsciiString ("File '") + theFile + "' is written using unsupported format"); return false; @@ -259,26 +258,26 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t const uint32_t* aChunkType = (const uint32_t* )(aChunkHeader2 + 4); if (*aChunkType == 0x4E4F534A) { - aJsonBodyOffset = int64_t (aFile->tellg()); + aJsonBodyOffset = int64_t (theStream.tellg()); aJsonBodyLen = int64_t (*aChunkLen); } else if (*aChunkType == 0x004E4942) { - aBinBodyOffset = int64_t (aFile->tellg()); + aBinBodyOffset = int64_t (theStream.tellg()); aBinBodyLen = int64_t (*aChunkLen); } if (*aChunkLen != 0) { - aFile->seekg (*aChunkLen, std::ios_base::cur); + theStream.seekg (*aChunkLen, std::ios_base::cur); } } - aFile->seekg ((std::streamoff )aJsonBodyOffset, std::ios_base::beg); + theStream.seekg ((std::streamoff )aJsonBodyOffset, std::ios_base::beg); } } else { - aFile->seekg (0, std::ios_base::beg); + theStream.seekg (0, std::ios_base::beg); } TCollection_AsciiString anErrPrefix = TCollection_AsciiString ("File '") + theFile + "' defines invalid glTF!\n"; @@ -303,10 +302,10 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t #ifdef HAVE_RAPIDJSON rapidjson::ParseResult aRes; - rapidjson::IStreamWrapper aFileStream (*aFile); + rapidjson::IStreamWrapper aFileStream (theStream); if (isBinaryFile) { - aRes = aDoc.ParseStream, rapidjson::IStreamWrapper> (aFileStream); + aRes = aDoc.ParseStream, rapidjson::IStreamWrapper>(aFileStream); } else { diff --git a/src/RWGltf/RWGltf_CafReader.hxx b/src/RWGltf/RWGltf_CafReader.hxx index 8b2d811649..905e59958a 100644 --- a/src/RWGltf/RWGltf_CafReader.hxx +++ b/src/RWGltf/RWGltf_CafReader.hxx @@ -83,7 +83,8 @@ public: protected: //! Read the mesh from specified file. - Standard_EXPORT virtual Standard_Boolean performMesh (const TCollection_AsciiString& theFile, + Standard_EXPORT virtual Standard_Boolean performMesh (std::istream& theStream, + const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe) Standard_OVERRIDE; diff --git a/src/RWMesh/RWMesh_CafReader.cxx b/src/RWMesh/RWMesh_CafReader.cxx index 4164e4496f..d1086ff15b 100644 --- a/src/RWMesh/RWMesh_CafReader.cxx +++ b/src/RWMesh/RWMesh_CafReader.cxx @@ -98,6 +98,20 @@ TopoDS_Shape RWMesh_CafReader::SingleShape() const Standard_Boolean RWMesh_CafReader::perform (const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe) +{ + std::ifstream aStream; + OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary); + return perform(aStream, theFile, theProgress, theToProbe); +} + +// ======================================================================= +// function : perform +// purpose : +// ======================================================================= +Standard_Boolean RWMesh_CafReader::perform (std::istream& theStream, + const TCollection_AsciiString& theFile, + const Message_ProgressRange& theProgress, + const Standard_Boolean theToProbe) { Standard_Integer aNewRootsLower = 1; if (!myXdeDoc.IsNull()) @@ -109,7 +123,7 @@ Standard_Boolean RWMesh_CafReader::perform (const TCollection_AsciiString& theFi OSD_Timer aLoadingTimer; aLoadingTimer.Start(); - const Standard_Boolean isDone = performMesh (theFile, theProgress, theToProbe); + const Standard_Boolean isDone = performMesh (theStream, theFile, theProgress, theToProbe); if (theToProbe || theProgress.UserBreak()) { return isDone; diff --git a/src/RWMesh/RWMesh_CafReader.hxx b/src/RWMesh/RWMesh_CafReader.hxx index 3be45e7d21..5e93d60b3e 100644 --- a/src/RWMesh/RWMesh_CafReader.hxx +++ b/src/RWMesh/RWMesh_CafReader.hxx @@ -16,6 +16,7 @@ #define _RWMesh_CafReader_HeaderFile #include +#include #include #include #include @@ -148,12 +149,22 @@ public: public: - //! Read the data from specified file. + //! Open stream and pass it to Perform method. //! The Document instance should be set beforehand. bool Perform (const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress) { - return perform (theFile, theProgress, Standard_False); + std::ifstream aStream; + OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary); + return Perform(aStream, theProgress, theFile); + } + + //! Read the data from specified file. + bool Perform (std::istream& theStream, + const Message_ProgressRange& theProgress, + const TCollection_AsciiString& theFile = "") + { + return perform(theStream, theFile, theProgress, Standard_False); } //! Return extended status flags. @@ -171,19 +182,28 @@ public: //! Return metadata map. const TColStd_IndexedDataMapOfStringString& Metadata() const { return myMetadata; } - //! Read the header data from specified file without reading entire model. - //! The main purpose is collecting metadata and external references - for copying model into a new location, for example. - //! Can be NOT implemented (unsupported by format / reader). + //! Open stream and pass it to ProbeHeader method. Standard_Boolean ProbeHeader (const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress = Message_ProgressRange()) { - return perform (theFile, theProgress, Standard_True); + std::ifstream aStream; + OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary); + return ProbeHeader (aStream, theFile, theProgress); + } + + //! Read the header data from specified file without reading entire model. + //! The main purpose is collecting metadata and external references - for copying model into a new location, for example. + //! Can be NOT implemented (unsupported by format / reader). + Standard_Boolean ProbeHeader (std::istream& theStream, + const TCollection_AsciiString& theFile = "", + const Message_ProgressRange& theProgress = Message_ProgressRange()) + { + return perform(theStream, theFile, theProgress, Standard_True); } protected: - //! Read the data from specified file. - //! Default implementation calls performMesh() and fills XDE document from collected shapes. + //! Open stream and pass it to Perform method. //! @param theFile file to read //! @param optional progress indicator //! @param theToProbe flag indicating that mesh data should be skipped and only basing information to be read @@ -191,8 +211,30 @@ protected: const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe); - //! Read the mesh from specified file - interface to be implemented by sub-classes. + //! Read the data from specified file. + //! Default implementation calls performMesh() and fills XDE document from collected shapes. + //! @param theStream input stream + //! @param theFile path of additional files + //! @param optional progress indicator + //! @param theToProbe flag indicating that mesh data should be skipped and only basing information to be read + Standard_EXPORT virtual Standard_Boolean perform (std::istream& theStream, + const TCollection_AsciiString& theFile, + const Message_ProgressRange& theProgress, + const Standard_Boolean theToProbe); + + //! Read the mesh from specified file Standard_EXPORT virtual Standard_Boolean performMesh (const TCollection_AsciiString& theFile, + const Message_ProgressRange& theProgress, + const Standard_Boolean theToProbe) + { + std::ifstream aStream; + OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary); + return performMesh(aStream, theFile, theProgress, theToProbe); + } + + //! Read the mesh from specified file - interface to be implemented by sub-classes. + Standard_EXPORT virtual Standard_Boolean performMesh (std::istream& theStream, + const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe) = 0; diff --git a/src/RWObj/RWObj_CafReader.cxx b/src/RWObj/RWObj_CafReader.cxx index 84d376d83a..83cf2738f9 100644 --- a/src/RWObj/RWObj_CafReader.cxx +++ b/src/RWObj/RWObj_CafReader.cxx @@ -94,7 +94,8 @@ Handle(RWObj_TriangulationReader) RWObj_CafReader::createReaderContext() // Function : performMesh // Purpose : //================================================================ -Standard_Boolean RWObj_CafReader::performMesh (const TCollection_AsciiString& theFile, +Standard_Boolean RWObj_CafReader::performMesh (std::istream& theStream, + const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe) { @@ -107,11 +108,11 @@ Standard_Boolean RWObj_CafReader::performMesh (const TCollection_AsciiString& th Standard_Boolean isDone = Standard_False; if (theToProbe) { - isDone = aCtx->Probe (theFile.ToCString(), theProgress); + isDone = aCtx->Probe (theStream, theFile, theProgress); } else { - isDone = aCtx->Read (theFile.ToCString(), theProgress); + isDone = aCtx->Read (theStream, theFile, theProgress); } if (!aCtx->FileComments().IsEmpty()) { diff --git a/src/RWObj/RWObj_CafReader.hxx b/src/RWObj/RWObj_CafReader.hxx index 581f35917f..45772d064f 100644 --- a/src/RWObj/RWObj_CafReader.hxx +++ b/src/RWObj/RWObj_CafReader.hxx @@ -35,7 +35,8 @@ public: protected: //! Read the mesh from specified file. - Standard_EXPORT virtual Standard_Boolean performMesh (const TCollection_AsciiString& theFile, + Standard_EXPORT virtual Standard_Boolean performMesh (std::istream& theStream, + const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe) Standard_OVERRIDE; diff --git a/src/RWObj/RWObj_Reader.cxx b/src/RWObj/RWObj_Reader.cxx index 5a5b8c1a88..a3a4f4d27f 100644 --- a/src/RWObj/RWObj_Reader.cxx +++ b/src/RWObj/RWObj_Reader.cxx @@ -47,35 +47,6 @@ namespace // The length of buffer to read (in bytes) static const size_t THE_BUFFER_SIZE = 4 * 1024; - //! Simple wrapper. - struct RWObj_ReaderFile - { - FILE* File; - int64_t FileLen; - - //! Constructor opening the file. - RWObj_ReaderFile (const TCollection_AsciiString& theFile) - : File (OSD_OpenFile (theFile.ToCString(), "rb")), - FileLen (0) - { - if (this->File != NULL) - { - // determine length of file - ::fseek64 (this->File, 0, SEEK_END); - FileLen = ::ftell64 (this->File); - ::fseek64 (this->File, 0, SEEK_SET); - } - } - - //! Destructor closing the file. - ~RWObj_ReaderFile() - { - if (File != NULL) - { - ::fclose (File); - } - } - }; //! Return TRUE if given polygon has clockwise node order. static bool isClockwisePolygon (const Handle(BRepMesh_DataStructureOfDelaun)& theMesh, @@ -115,7 +86,8 @@ RWObj_Reader::RWObj_Reader() // Function : read // Purpose : // ================================================================ -Standard_Boolean RWObj_Reader::read (const TCollection_AsciiString& theFile, +Standard_Boolean RWObj_Reader::read (std::istream& theStream, + const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe) { @@ -140,15 +112,16 @@ Standard_Boolean RWObj_Reader::read (const TCollection_AsciiString& theFile, myCurrElem.resize (1024, -1); Standard_CLocaleSentry aLocaleSentry; - RWObj_ReaderFile aFile (theFile); - if (aFile.File == NULL) + if (!theStream.good()) { Message::SendFail (TCollection_AsciiString ("Error: file '") + theFile + "' is not found"); return Standard_False; } // determine length of file - const int64_t aFileLen = aFile.FileLen; + theStream.seekg(0, theStream.end); + const int64_t aFileLen = theStream.tellg(); + theStream.seekg(0, theStream.beg); if (aFileLen <= 0L) { Message::SendFail (TCollection_AsciiString ("Error: file '") + theFile + "' is empty"); @@ -158,12 +131,11 @@ Standard_Boolean RWObj_Reader::read (const TCollection_AsciiString& theFile, Standard_ReadLineBuffer aBuffer (THE_BUFFER_SIZE); aBuffer.SetMultilineMode (true); - const Standard_Integer aNbMiBTotal = Standard_Integer(aFileLen / (1024 * 1024)); + const Standard_Integer aNbMiBTotal = Standard_Integer(aFileLen / (1024 * 1024)); Standard_Integer aNbMiBPassed = 0; Message_ProgressScope aPS (theProgress, "Reading text OBJ file", aNbMiBTotal); OSD_Timer aTimer; aTimer.Start(); - bool isStart = true; int64_t aPosition = 0; size_t aLineLen = 0; @@ -171,7 +143,7 @@ Standard_Boolean RWObj_Reader::read (const TCollection_AsciiString& theFile, const char* aLine = NULL; for (;;) { - aLine = aBuffer.ReadLine (aFile.File, aLineLen, aReadBytes); + aLine = aBuffer.ReadLine (theStream, aLineLen, aReadBytes); if (aLine == NULL) { break; diff --git a/src/RWObj/RWObj_Reader.hxx b/src/RWObj/RWObj_Reader.hxx index 4b91d68bd2..d652a0ef8e 100644 --- a/src/RWObj/RWObj_Reader.hxx +++ b/src/RWObj/RWObj_Reader.hxx @@ -23,7 +23,7 @@ #include #include #include - +#include #include #include #include @@ -48,17 +48,27 @@ public: //! Empty constructor. Standard_EXPORT RWObj_Reader(); - //! Reads data from OBJ file. - //! Unicode paths can be given in UTF-8 encoding. - //! Returns true if success, false on error or user break. + //! Open stream and pass it to Read method + //! Returns true if success, false on error. Standard_Boolean Read (const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress) { - return read (theFile, theProgress, Standard_False); + std::ifstream aStream; + OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary); + return Read(aStream, theFile, theProgress); } - //! Probe data from OBJ file (comments, external references) without actually reading mesh data. - //! Although mesh data will not be collected, the full file content will be parsed, due to OBJ format limitations. + //! Reads data from OBJ file. + //! Unicode paths can be given in UTF-8 encoding. + //! Returns true if success, false on error or user break. + Standard_Boolean Read (std::istream& theStream, + const TCollection_AsciiString& theFile, + const Message_ProgressRange& theProgress) + { + return read(theStream, theFile, theProgress, Standard_False); + } + + //! Open stream and pass it to Probe method. //! @param theFile path to the file //! @param theProgress progress indicator //! @return TRUE if success, FALSE on error or user break. @@ -66,7 +76,23 @@ public: Standard_Boolean Probe (const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress) { - return read (theFile, theProgress, Standard_True); + std::ifstream aStream; + OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary); + return Probe(aStream, theFile, theProgress); + } + + //! Probe data from OBJ file (comments, external references) without actually reading mesh data. + //! Although mesh data will not be collected, the full file content will be parsed, due to OBJ format limitations. + //! @param theStream input stream + //! @param theFile path to the file + //! @param theProgress progress indicator + //! @return TRUE if success, FALSE on error or user break. + //! @sa FileComments(), ExternalFiles(), NbProbeNodes(), NbProbeElems(). + Standard_Boolean Probe (std::istream& theStream, + const TCollection_AsciiString& theFile, + const Message_ProgressRange& theProgress) + { + return read(theStream, theFile, theProgress, Standard_True); } //! Returns file comments (lines starting with # at the beginning of file). @@ -107,7 +133,8 @@ protected: //! Reads data from OBJ file. //! Unicode paths can be given in UTF-8 encoding. //! Returns true if success, false on error or user break. - Standard_EXPORT Standard_Boolean read (const TCollection_AsciiString& theFile, + Standard_EXPORT Standard_Boolean read (std::istream& theStream, + const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe); @@ -355,7 +382,6 @@ protected: RWObj_SubMesh myActiveSubMesh; //!< active sub-mesh definition std::vector myCurrElem; //!< indices for the current element - }; #endif // _RWObj_Reader_HeaderFile diff --git a/src/VrmlAPI/VrmlAPI_CafReader.cxx b/src/VrmlAPI/VrmlAPI_CafReader.cxx index 523442da81..963ecce06a 100644 --- a/src/VrmlAPI/VrmlAPI_CafReader.cxx +++ b/src/VrmlAPI/VrmlAPI_CafReader.cxx @@ -89,14 +89,13 @@ namespace // function : performMesh // purpose : //======================================================================= -bool VrmlAPI_CafReader::performMesh(const TCollection_AsciiString& theFile, +bool VrmlAPI_CafReader::performMesh(std::istream& theStream, + const TCollection_AsciiString& theFile, const Message_ProgressRange& theProgress, const Standard_Boolean theToProbe) { (void)theProgress; - Handle(OSD_FileSystem) aFile = OSD_FileSystem::DefaultFileSystem(); - std::shared_ptr aFileStream = aFile->OpenIStream(theFile, std::ios::in | std::ios::binary); - if (aFileStream.get() == nullptr || !aFileStream->good()) + if (!theStream.good()) { Message::SendFail() << "Error in VrmlAPI_CafReader: file '" << theFile << "' is not found"; return false; @@ -115,7 +114,7 @@ bool VrmlAPI_CafReader::performMesh(const TCollection_AsciiString& theFile, VrmlData_Scene aScene; aScene.SetLinearScale(FileLengthUnit()); aScene.SetVrmlDir(aFolder); - aScene << *aFileStream; + aScene << theStream; VrmlData_DataMapOfShapeAppearance aShapeAppMap; TopoDS_Shape aShape = aScene.GetShape(aShapeAppMap); diff --git a/src/VrmlAPI/VrmlAPI_CafReader.hxx b/src/VrmlAPI/VrmlAPI_CafReader.hxx index 87af90d064..eac8397c0d 100644 --- a/src/VrmlAPI/VrmlAPI_CafReader.hxx +++ b/src/VrmlAPI/VrmlAPI_CafReader.hxx @@ -24,13 +24,15 @@ class VrmlAPI_CafReader : public RWMesh_CafReader protected: //! Read the mesh data from specified file. - //! @param theFile file to read + //! @param theStream input stream + //! @param theFile path of additional files //! @param theProgress progress indicator //! @param theToProbe flag for probing file without complete reading. Not supported. //! @return false when theToProbe is set to true or reading has completed with error. - Standard_EXPORT virtual Standard_Boolean performMesh(const TCollection_AsciiString& theFile, - const Message_ProgressRange& theProgress, - const Standard_Boolean theToProbe) Standard_OVERRIDE; + Standard_EXPORT virtual Standard_Boolean performMesh (std::istream& theStream, + const TCollection_AsciiString& theFile, + const Message_ProgressRange& theProgress, + const Standard_Boolean theToProbe) Standard_OVERRIDE; }; diff --git a/src/VrmlData/VrmlData_Scene.cxx b/src/VrmlData/VrmlData_Scene.cxx index 538a35c76e..01adb4d2b0 100644 --- a/src/VrmlData/VrmlData_Scene.cxx +++ b/src/VrmlData/VrmlData_Scene.cxx @@ -170,6 +170,10 @@ Standard_OStream& operator << (Standard_OStream& theOutput, void VrmlData_Scene::SetVrmlDir (const TCollection_ExtendedString& theDir) { TCollection_ExtendedString& aDir = myVrmlDir.Append (theDir); + if (aDir.IsEmpty()) + { + return; + } const Standard_ExtCharacter aTerminator = aDir.Value(aDir.Length()); if (aTerminator != Standard_ExtCharacter('\\') && aTerminator != Standard_ExtCharacter('/'))