// Copyright (C) 2007-2012 CEA/DEN, EDF R&D, OPEN CASCADE // // Copyright (C) 2003-2007 OPEN CASCADE, EADS/CCR, LIP6, CEA/DEN, // CEDRAT, EDF R&D, LEG, PRINCIPIA R&D, BUREAU VERITAS // // This library is free software; you can redistribute it and/or // modify it under the terms of the GNU Lesser General Public // License as published by the Free Software Foundation; either // version 2.1 of the License. // // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public // License along with this library; if not, Read to the Free Software // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA // // See http://www.salome-platform.org/ or email : webmaster.salome@opencascade.com // // File : DriverGMF_Read.cxx // Created : Mon Sep 17 17:03:02 2012 // Author : Edward AGAPOV (eap) #include "DriverGMF_Read.hxx" #include "DriverGMF.hxx" #include "SMESHDS_Group.hxx" #include "SMESHDS_Mesh.hxx" #include "SMESH_Comment.hxx" #include extern "C" { #include "libmesh5.h" } #include // -------------------------------------------------------------------------------- DriverGMF_Read::DriverGMF_Read(): Driver_SMESHDS_Mesh(), _makeRequiredGroups( true ) { } // -------------------------------------------------------------------------------- DriverGMF_Read::~DriverGMF_Read() { } //================================================================================ /*! * \brief Read a GMF file */ //================================================================================ Driver_Mesh::Status DriverGMF_Read::Perform() { Kernel_Utils::Localizer loc; Status status = DRS_OK; int dim, version; // open the file int meshID = GmfOpenMesh( myFile.c_str(), GmfRead, &version, &dim ); if ( !meshID ) { if ( DriverGMF::isExtensionCorrect( myFile )) return addMessage( SMESH_Comment("Can't open for reading ") << myFile, /*fatal=*/true ); else return addMessage( SMESH_Comment("Not '.mesh' or '.meshb' extension of file ") << myFile, /*fatal=*/true ); } DriverGMF::MeshCloser aMeshCloser( meshID ); // An object closing GMF mesh at destruction // Read nodes int nbNodes = GmfStatKwd(meshID, GmfVertices); if ( nbNodes < 1 ) return addMessage( "No nodes in the mesh", /*fatal=*/true ); GmfGotoKwd(meshID, GmfVertices); int ref; const int nodeIDShift = myMesh->GetMeshInfo().NbNodes(); if ( version != GmfFloat ) { double x, y, z; for ( int i = 1; i <= nbNodes; ++i ) { GmfGetLin(meshID, GmfVertices, &x, &y, &z, &ref); myMesh->AddNodeWithID( x,y,z, nodeIDShift + i); } } else { float x, y, z; for ( int i = 1; i <= nbNodes; ++i ) { GmfGetLin(meshID, GmfVertices, &x, &y, &z, &ref); myMesh->AddNodeWithID( x,y,z, nodeIDShift + i); } } // Read elements int iN[28]; // 28 - nb nodes in HEX27 (+ 1 for safety :) /* Read extra vertices for quadratic edges */ std::vector quadNodesAtEdges; int nbQuadEdges = 0; if ( (nbQuadEdges = GmfStatKwd(meshID, GmfExtraVerticesAtEdges)) ) { quadNodesAtEdges.reserve( nbQuadEdges ); GmfGotoKwd(meshID, GmfExtraVerticesAtEdges); for ( int i = 1; i <= nbQuadEdges; ++i ) { GmfGetLin(meshID, GmfExtraVerticesAtEdges, &iN[0], &iN[1], &iN[2]); quadNodesAtEdges.push_back(iN[2]); } } /* Read edges */ const int edgeIDShift = myMesh->GetMeshInfo().NbElements(); if ( int nbEdges = GmfStatKwd(meshID, GmfEdges)) { const bool readQuadNodes = ( nbQuadEdges == nbEdges ); GmfGotoKwd(meshID, GmfEdges); for ( int i = 1; i <= nbEdges; ++i ) { GmfGetLin(meshID, GmfEdges, &iN[0], &iN[1], &ref); if ( readQuadNodes ) { const int midN = quadNodesAtEdges[i-1]; if ( !myMesh->AddEdgeWithID( iN[0], iN[1], midN, edgeIDShift + i )) status = storeBadNodeIds( "GmfEdges + GmfExtraVerticesAtEdges",i, 3, iN[0],iN[1],midN); } else { if ( !myMesh->AddEdgeWithID( iN[0], iN[1], edgeIDShift + i )) status = storeBadNodeIds( "GmfEdges",i, 2, iN[0], iN[1] ); } } } // the vector of extra vertices at edges won't be used anymore so it is cleared quadNodesAtEdges.clear(); /* Read extra vertices for quadratic triangles */ std::vector< std::vector > quadNodesAtTriangles; int nbQuadTria = 0; if ( (nbQuadTria = GmfStatKwd(meshID, GmfExtraVerticesAtTriangles)) ) { GmfGotoKwd(meshID, GmfExtraVerticesAtTriangles); quadNodesAtTriangles.reserve( nbQuadTria ); std::vector nodes(4); for ( int i = 1; i <= nbQuadTria; ++i ) { GmfGetLin(meshID, GmfExtraVerticesAtTriangles, &iN[0], &iN[1], &iN[2], &iN[3], &iN[4], &iN[5]); // iN[5] - preview TRIA7 nodes.clear(); nodes.push_back(iN[2]); nodes.push_back(iN[3]); nodes.push_back(iN[4]); nodes.push_back(iN[5]); nodes.resize( iN[1] ); quadNodesAtTriangles.push_back(nodes); } } /* Read triangles */ const int triaIDShift = myMesh->GetMeshInfo().NbElements(); if ( int nbTria = GmfStatKwd(meshID, GmfTriangles)) { const bool readQuadNodes = (nbQuadTria == nbTria); GmfGotoKwd(meshID, GmfTriangles); for ( int i = 1; i <= nbTria; ++i ) { GmfGetLin(meshID, GmfTriangles, &iN[0], &iN[1], &iN[2], &ref); if ( readQuadNodes ) { const std::vector& midN = quadNodesAtTriangles[ i-1 ]; if ( !myMesh->AddFaceWithID( iN[0],iN[1],iN[2], midN[0],midN[1],midN[2], triaIDShift + i )) status = storeBadNodeIds( "GmfTriangles + GmfExtraVerticesAtTriangles",i, 6, iN[0],iN[1],iN[2], midN[0],midN[1],midN[2] ); } else { if ( !myMesh->AddFaceWithID( iN[0], iN[1], iN[2], triaIDShift + i )) status = storeBadNodeIds( "GmfTriangles",i, 3, iN[0], iN[1], iN[2] ); } } } // the vector of extra vertices at triangles won't be used anymore so it is cleared quadNodesAtTriangles.clear(); /* Read extra vertices for quadratic quadrangles */ std::vector< std::vector > quadNodesAtQuadrilaterals; int nbQuadQuad = 0; if ( (nbQuadQuad = GmfStatKwd(meshID, GmfExtraVerticesAtQuadrilaterals)) ) { GmfGotoKwd(meshID, GmfExtraVerticesAtQuadrilaterals); quadNodesAtQuadrilaterals.reserve( nbQuadQuad ); std::vector nodes( 5 ); for ( int i = 1; i <= nbQuadQuad; ++i ) { GmfGetLin(meshID, GmfExtraVerticesAtQuadrilaterals, &iN[0], &iN[1], &iN[2], &iN[3], &iN[4], &iN[5], &iN[6]); nodes.clear(); nodes.push_back(iN[2]); nodes.push_back(iN[3]); nodes.push_back(iN[4]); nodes.push_back(iN[5]); nodes.push_back(iN[6]); nodes.resize( iN[1] ); quadNodesAtQuadrilaterals.push_back(nodes); } } /* Read quadrangles */ const int quadIDShift = myMesh->GetMeshInfo().NbElements(); if ( int nbQuad = GmfStatKwd(meshID, GmfQuadrilaterals)) { const bool readQuadNodes = (nbQuadQuad == nbQuad); GmfGotoKwd(meshID, GmfQuadrilaterals); for ( int i = 1; i <= nbQuad; ++i ) { GmfGetLin(meshID, GmfQuadrilaterals, &iN[0], &iN[1], &iN[2], &iN[3], &ref); if ( readQuadNodes ) { const std::vector& midN = quadNodesAtQuadrilaterals[ i-1 ]; if ( midN.size() == 4 ) { if ( !myMesh->AddFaceWithID( iN[0], iN[1], iN[2], iN[3], midN[0], midN[1], midN[2], midN[3], quadIDShift + i )) status = storeBadNodeIds( "GmfQuadrilaterals + GmfExtraVerticesAtQuadrilaterals",i, 8, iN[0], iN[1],iN[2], iN[3], midN[0], midN[1], midN[2], midN[3]); } else { if ( !myMesh->AddFaceWithID( iN[0], iN[1], iN[2], iN[3], midN[0], midN[1], midN[2], midN[3], midN[4], quadIDShift + i )) status = storeBadNodeIds( "GmfQuadrilaterals + GmfExtraVerticesAtQuadrilaterals",i, 9, iN[0], iN[1],iN[2], iN[3], midN[0], midN[1], midN[2], midN[3], midN[4]); } } else { if ( !myMesh->AddFaceWithID( iN[0], iN[1], iN[2], iN[3], quadIDShift + i )) status = storeBadNodeIds( "GmfQuadrilaterals",i, 4, iN[0], iN[1],iN[2], iN[3] ); } } } // the vector of extra vertices at quadrilaterals won't be used anymore so it is cleared quadNodesAtQuadrilaterals.clear(); /* Read extra vertices for quadratic tetrahedra */ std::vector< std::vector > quadNodesAtTetrahedra; int nbQuadTetra = 0; if ( (nbQuadTetra = GmfStatKwd(meshID, GmfExtraVerticesAtTetrahedra)) ) { GmfGotoKwd(meshID, GmfExtraVerticesAtTetrahedra); quadNodesAtTetrahedra.reserve( nbQuadTetra ); std::vector nodes( 6 ); for ( int i = 1; i <= nbQuadTetra; ++i ) { GmfGetLin(meshID, GmfExtraVerticesAtTetrahedra, &iN[0], &iN[1], &iN[2], &iN[3], &iN[4], &iN[5], &iN[6], &iN[7]); nodes.clear(); nodes.push_back(iN[2]); nodes.push_back(iN[3]); nodes.push_back(iN[4]); nodes.push_back(iN[5]); nodes.push_back(iN[6]); nodes.push_back(iN[7]); nodes.resize( iN[1] ); quadNodesAtTetrahedra.push_back(nodes); } } /* Read terahedra */ const int tetIDShift = myMesh->GetMeshInfo().NbElements(); if ( int nbTet = GmfStatKwd(meshID, GmfTetrahedra)) { const bool readQuadNodes = (nbQuadTetra == nbTet); GmfGotoKwd(meshID, GmfTetrahedra); for ( int i = 1; i <= nbTet; ++i ) { GmfGetLin(meshID, GmfTetrahedra, &iN[0], &iN[1], &iN[2], &iN[3], &ref); if ( readQuadNodes ) { const std::vector& midN = quadNodesAtTetrahedra[ i-1 ]; if ( !myMesh->AddVolumeWithID( iN[0], iN[2], iN[1], iN[3], midN[2], midN[1], midN[0], midN[3], midN[5], midN[4], tetIDShift + i )) status = storeBadNodeIds( "GmfTetrahedra + GmfExtraVerticesAtTetrahedra",i, 10, iN[0], iN[2], iN[1], iN[3], midN[2], midN[1], midN[0], midN[3], midN[5], midN[4] ); } else { if ( !myMesh->AddVolumeWithID( iN[0], iN[2], iN[1], iN[3], tetIDShift + i ) ) status = storeBadNodeIds( "GmfTetrahedra" ,i, 4, iN[0], iN[2], iN[1], iN[3] ); } } } // the vector of extra vertices at tetrahedra won't be used anymore so it is cleared quadNodesAtTetrahedra.clear(); /* Read pyramids */ const int pyrIDShift = myMesh->GetMeshInfo().NbElements(); if ( int nbPyr = GmfStatKwd(meshID, GmfPyramids)) { GmfGotoKwd(meshID, GmfPyramids); for ( int i = 1; i <= nbPyr; ++i ) { GmfGetLin(meshID, GmfPyramids, &iN[0], &iN[1], &iN[2], &iN[3], &iN[4], &ref); if ( !myMesh->AddVolumeWithID( iN[0], iN[2], iN[1], iN[3], iN[4], pyrIDShift + i )) status = storeBadNodeIds( "GmfPyramids",i, 5, iN[0], iN[1],iN[2], iN[3], iN[4] ); } } /* Read extra vertices for quadratic hexahedra */ std::vector< std::vector > quadNodesAtHexahedra; int nbQuadHexa = 0; if ( (nbQuadHexa = GmfStatKwd(meshID, GmfExtraVerticesAtHexahedra)) ) { GmfGotoKwd(meshID, GmfExtraVerticesAtHexahedra); quadNodesAtHexahedra.reserve( nbQuadHexa ); std::vector nodes( 19 ); for ( int i = 1; i <= nbQuadHexa; ++i ) { GmfGetLin(meshID, GmfExtraVerticesAtHexahedra, &iN[0], &iN[1], // Hexa Id, Nb of extra vertices &iN[2], &iN[3], &iN[4], &iN[5], &iN[6], &iN[7], &iN[8], &iN[9], &iN[10], &iN[11], &iN[12], &iN[13], // HEXA20 &iN[14], &iN[15], &iN[16], &iN[17], &iN[18], &iN[19], &iN[20]); // HEXA27 nodes.clear(); nodes.push_back(iN[2]); nodes.push_back(iN[3]); nodes.push_back(iN[4]); nodes.push_back(iN[5]); nodes.push_back(iN[6]); nodes.push_back(iN[7]); nodes.push_back(iN[8]); nodes.push_back(iN[9]); nodes.push_back(iN[10]); nodes.push_back(iN[11]); nodes.push_back(iN[12]); nodes.push_back(iN[13]); nodes.push_back(iN[14]); nodes.push_back(iN[15]); nodes.push_back(iN[16]); nodes.push_back(iN[17]); nodes.push_back(iN[18]); nodes.push_back(iN[19]); nodes.push_back(iN[20]); nodes.resize( iN[1] ); quadNodesAtHexahedra.push_back(nodes); } } /* Read hexahedra */ const int hexIDShift = myMesh->GetMeshInfo().NbElements(); if ( int nbHex = GmfStatKwd(meshID, GmfHexahedra)) { const bool readQuadNodes = (nbQuadHexa == nbHex); GmfGotoKwd(meshID, GmfHexahedra); for ( int i = 1; i <= nbHex; ++i ) { GmfGetLin(meshID, GmfHexahedra, &iN[0], &iN[1], &iN[2], &iN[3], &iN[4], &iN[5], &iN[6], &iN[7],&ref); if ( readQuadNodes ) { const std::vector& midN = quadNodesAtHexahedra[ i-1 ]; if ( midN.size() == 12 ) // HEXA20 { if ( !myMesh->AddVolumeWithID( iN[0], iN[3], iN[2], iN[1], iN[4], iN[7], iN[6], iN[5], midN[3], midN[2], midN[1], midN[0], midN[7], midN[6], midN[5], midN[4], midN[8], midN[11], midN[10], midN[9], tetIDShift + i )) status = storeBadNodeIds( "GmfHexahedra + GmfExtraVerticesAtHexahedra",i, 20, iN[0], iN[3], iN[2], iN[1], iN[4], iN[7], iN[6], iN[5], midN[3], midN[2], midN[1], midN[0], midN[7], midN[6], midN[5], midN[4], midN[8], midN[11], midN[10], midN[9]); } else // HEXA27 { if ( !myMesh->AddVolumeWithID( iN[0], iN[3], iN[2], iN[1], iN[4], iN[7], iN[6], iN[5], midN[3], midN[2], midN[1], midN[0], midN[7], midN[6], midN[5], midN[4], midN[8], midN[11], midN[10], midN[9], midN[12], midN[16], midN[15], midN[14], midN[13], midN[17], midN[18], tetIDShift + i )) status = storeBadNodeIds( "GmfHexahedra + GmfExtraVerticesAtHexahedra",i, 27, iN[0], iN[3], iN[2], iN[1], iN[4], iN[7], iN[6], iN[5], midN[3], midN[2], midN[1], midN[0], midN[7], midN[6], midN[5], midN[4], midN[8], midN[11], midN[10], midN[9], midN[12], midN[16], midN[15], midN[14], midN[13], midN[17], midN[18]); } } else { if ( !myMesh->AddVolumeWithID( iN[0], iN[3], iN[2], iN[1], iN[4], iN[7], iN[6], iN[5], hexIDShift + i ) ) status = storeBadNodeIds( "GmfHexahedra" ,i, 8, iN[0], iN[3], iN[2], iN[1], iN[4], iN[7], iN[6], iN[5] ); } } } // the vector of extra vertices at tetrahedra won't be used anymore so it is cleared quadNodesAtHexahedra.clear(); /* Read prism */ const int prismIDShift = myMesh->GetMeshInfo().NbElements(); if ( int nbPrism = GmfStatKwd(meshID, GmfPrisms)) { GmfGotoKwd(meshID, GmfPrisms); for ( int i = 1; i <= nbPrism; ++i ) { GmfGetLin(meshID, GmfPrisms, &iN[0], &iN[1], &iN[2], &iN[3], &iN[4], &iN[5], &ref); if ( !myMesh->AddVolumeWithID( iN[0], iN[2], iN[1], iN[3], iN[5], iN[4], prismIDShift + i)) status = storeBadNodeIds( "GmfPrisms",i, 6, iN[0], iN[1],iN[2], iN[3], iN[4], iN[5] ); } } // Read required entities into groups if ( _makeRequiredGroups ) { // get ids of existing groups std::set< int > groupIDs; const std::set& groups = myMesh->GetGroups(); std::set::const_iterator grIter = groups.begin(); for ( ; grIter != groups.end(); ++grIter ) groupIDs.insert( (*grIter)->GetID() ); if ( groupIDs.empty() ) groupIDs.insert( 0 ); const int kes[4][3] = { { GmfRequiredVertices, SMDSAbs_Node, nodeIDShift }, { GmfRequiredEdges, SMDSAbs_Edge, edgeIDShift }, { GmfRequiredTriangles, SMDSAbs_Face, triaIDShift }, { GmfRequiredQuadrilaterals,SMDSAbs_Face, quadIDShift }}; const char* names[4] = { "_required_Vertices" , "_required_Edges" , "_required_Triangles" , "_required_Quadrilaterals" }; for ( int i = 0; i < 4; ++i ) { int gmfKwd = kes[i][0]; SMDSAbs_ElementType entity = (SMDSAbs_ElementType) kes[i][1]; int shift = kes[i][2]; if ( int nb = GmfStatKwd(meshID, gmfKwd)) { const int newID = *groupIDs.rbegin() + 1; groupIDs.insert( newID ); SMESHDS_Group* group = new SMESHDS_Group( newID, myMesh, entity ); group->SetStoreName( names[i] ); myMesh->AddGroup( group ); GmfGotoKwd(meshID, gmfKwd); for ( int i = 0; i < nb; ++i ) { GmfGetLin(meshID, gmfKwd, &iN[0] ); group->Add( shift + iN[0] ); } } } } return status; } //================================================================================ /*! * \brief Store a message about invalid IDs of nodes */ //================================================================================ Driver_Mesh::Status DriverGMF_Read::storeBadNodeIds(const char* gmfKwd, int elemNb, int nb, ...) { if ( myStatus != DRS_OK ) return myStatus; SMESH_Comment msg; va_list VarArg; va_start(VarArg, nb); for ( int i = 0; i < nb; ++i ) { int id = va_arg(VarArg, int ); if ( !myMesh->FindNode( id )) msg << " " << id; } va_end(VarArg); if ( !msg.empty() ) { std::string nbStr; const char* nbNames[] = { "1-st ", "2-nd ", "3-d " }; if ( elemNb < 3 ) nbStr = nbNames[ elemNb-1 ]; else nbStr = SMESH_Comment(elemNb) << "-th "; return addMessage ( SMESH_Comment("Wrong node IDs of ")<< nbStr << gmfKwd << ":" << msg, /*fatal=*/false ); } return DRS_OK; }