introduction of MED write/append with a lower major version of MED file format (hdf5 >= 1.10.2 needed)

This commit is contained in:
Paul RASCLE 2018-10-03 15:08:45 +02:00
parent 15fdc1b6fc
commit 9cef4666a1
6 changed files with 104 additions and 28 deletions

View File

@ -622,10 +622,13 @@ module SMESH
* the groups Group_On_All_Nodes, Group_On_All_Faces, ... ; * the groups Group_On_All_Nodes, Group_On_All_Faces, ... ;
* the typical use is auto_groups=false. * the typical use is auto_groups=false.
* - overwrite : boolean parameter for overwriting/not overwriting the file, if it exists * - overwrite : boolean parameter for overwriting/not overwriting the file, if it exists
* - minor : define the minor version of MED file format. * - version : define the version of MED file format, coded with major and minor digits (release digit not used)
* for instance med 3.2.1 is coded 3*10+2 = 32, med 4.0.0 is coded 4*10+0 = 40.
* The rules of compatibility to write a mesh in an older version than the current version
* depend on the current version. For instance, with med 4.0 it is possible to write/append
* med files in 4.0.0 (default format) or 3.2.1 or 3.3.1 formats.
* The minor must be between 0 and the current minor version of MED file library. * The minor must be between 0 and the current minor version of MED file library.
* If minor is equal to -1, the minor version is not changed (default). * If version is equal to -1, the version is not changed (default).
* The major version cannot be changed.
* - autoDimension : if @c true, a space dimension of a MED mesh can be either * - autoDimension : if @c true, a space dimension of a MED mesh can be either
* - 1D if all mesh nodes lie on OX coordinate axis, or * - 1D if all mesh nodes lie on OX coordinate axis, or
* - 2D if all mesh nodes lie on XOY coordinate plane, or * - 2D if all mesh nodes lie on XOY coordinate plane, or
@ -634,7 +637,7 @@ module SMESH
*/ */
void ExportMED( in string fileName, void ExportMED( in string fileName,
in boolean auto_groups, in boolean auto_groups,
in long minor, in long version,
in boolean overwrite, in boolean overwrite,
in boolean autoDimension) raises (SALOME::SALOME_Exception); in boolean autoDimension) raises (SALOME::SALOME_Exception);
@ -644,10 +647,13 @@ module SMESH
* - meshPart : a part of mesh to store * - meshPart : a part of mesh to store
* - fileName : name of the MED file * - fileName : name of the MED file
* - overwrite : boolean parameter for overwriting/not overwriting the file, if it exists * - overwrite : boolean parameter for overwriting/not overwriting the file, if it exists
* - minor : define the minor version (y, where version is x.y.z) of MED file format. * - version : define the version of MED file format, coded with major and minor digits (release digit not used)
* for instance med 3.2.1 is coded 3*10+2 = 32, med 4.0.0 is coded 4*10+0 = 40.
* The rules of compatibility to write a mesh in an older version than the current version
* depend on the current version. For instance, with med 4.0 it is possible to write/append
* med files in 4.0.0 (default format) or 3.2.1 or 3.3.1 formats.
* The minor must be between 0 and the current minor version of MED file library. * The minor must be between 0 and the current minor version of MED file library.
* If minor is equal to -1, the minor version is not changed (default). * If version is equal to -1, the version is not changed (default).
* The major version (x, where version is x.y.z) cannot be changed.
* - autoDimension : if @c True, a space dimension for export is defined by mesh * - autoDimension : if @c True, a space dimension for export is defined by mesh
* configuration; for example a planar mesh lying on XOY plane * configuration; for example a planar mesh lying on XOY plane
* will be exported as a mesh in 2D space. * will be exported as a mesh in 2D space.
@ -663,7 +669,7 @@ module SMESH
void ExportPartToMED( in SMESH_IDSource meshPart, void ExportPartToMED( in SMESH_IDSource meshPart,
in string fileName, in string fileName,
in boolean auto_groups, in boolean auto_groups,
in long minor, in long version,
in boolean overwrite, in boolean overwrite,
in boolean autoDimension, in boolean autoDimension,
in GEOM::ListOfFields fields, in GEOM::ListOfFields fields,
@ -681,6 +687,12 @@ module SMESH
*/ */
string GetVersionString(in long minor, in short nbDigits); string GetVersionString(in long minor, in short nbDigits);
/*!
* Return the list of med versions compatibles for write/append,
* encoded in 10*major+minor (for instance, code for med 3.2.1 is 32)
*/
long_array GetMEDVersionsCompatibleForAppend();
/*! /*!
* Export Mesh to different Formats * Export Mesh to different Formats
* (UNV supported version is I-DEAS 10) * (UNV supported version is I-DEAS 10)

View File

@ -41,6 +41,17 @@ extern "C"
#include <windows.h> #include <windows.h>
#endif #endif
#define MED_MAJOR_EXPECTED 4
#define MED_MINOR_EXPECTED 0
#if MED_MAJOR_NUM != MED_MAJOR_EXPECTED
#error "MED major version does not correspond to the expected version, fix the minor and major compatibility values in CheckCompatibility method (MED_VERSIONS_APPEND_COMPATIBLE) and set the correct expected version"
#endif
#if MED_MINOR_NUM != MED_MINOR_EXPECTED
#error "MED minor version does not correspond to the expected version, fix the minor and major compatibility values in CheckCompatibility method (MED_VERSIONS_APPEND_COMPATIBLE) and set the correct expected version"
#endif
#define MED_VERSIONS_APPEND_COMPATIBLE {40, 32, 33} // --- 10*major + minor (the 3rd digit, release, is not used here,
// med uses always the latest available)
// --- The first in the list should be the default: current version
namespace MED namespace MED
{ {
@ -61,25 +72,51 @@ namespace MED
#endif #endif
} }
/*!
* Return the list of med versions compatibles for write/append,
* encoded in 10*major+minor (for instance, code for med 3.2.1 is 32)
*/
std::vector<int> GetMEDVersionsAppendCompatible()
{
int mvok[] = MED_VERSIONS_APPEND_COMPATIBLE;
std::vector<int> MEDVersionsOK(mvok, mvok + sizeof(mvok)/sizeof(int));
return MEDVersionsOK;
}
/*!
* \brief: Check read or write(append) Compatibility of a med file
* \param [in] : fileName - the file to read or to append to
* \param [in] : isforAppend - when true, check if the med file version is OK to append a mesh,
* when false, check if the med file is readable.
*/
bool CheckCompatibility(const std::string& fileName, bool isForAppend) bool CheckCompatibility(const std::string& fileName, bool isForAppend)
{ {
bool ok = false; bool ok = false;
int medVersionsOK[] = MED_VERSIONS_APPEND_COMPATIBLE;
// check that file is accessible // check that file is accessible
if ( exists(fileName) ) { if ( exists(fileName) ) {
// check HDF5 && MED compatibility // check HDF5 && MED compatibility
med_bool hdfok, medok; med_bool hdfok, medok;
med_err r0 = MEDfileCompatibility(fileName.c_str(), &hdfok, &medok); med_err r0 = MEDfileCompatibility(fileName.c_str(), &hdfok, &medok);
//MESSAGE(r0 << " " << hdfok << " " << medok); MESSAGE(r0 << " " << hdfok << " " << medok);
if ( r0==0 && hdfok && medok ) { if ( r0==0 && hdfok && medok ) {
med_idt aFid = MEDfileOpen(fileName.c_str(), MED_ACC_RDONLY); med_idt aFid = MEDfileOpen(fileName.c_str(), MED_ACC_RDONLY);
if (aFid >= 0) { if (aFid >= 0) {
med_int major, minor, release; med_int major, minor, release;
med_err ret = MEDfileNumVersionRd(aFid, &major, &minor, &release); med_err ret = MEDfileNumVersionRd(aFid, &major, &minor, &release);
//MESSAGE(ret << " " << major << "." << minor << "." << release); MESSAGE(ret << " " << major << "." << minor << "." << release);
if (ret >= 0) { if (ret >= 0) {
bool isReadOnly = !isForAppend; bool isReadOnly = !isForAppend;
if ( isReadOnly || ((major == MED_MAJOR_NUM) && (minor == MED_MINOR_NUM))) if (isReadOnly)
ok = true; ok = true;
else {
int medVersion = 10*major + minor;
for (int ii=0; ii < sizeof(medVersionsOK)/sizeof(int); ii++)
if (medVersionsOK[ii] == medVersion) {
ok =true;
break;
}
}
} }
} }
MEDfileClose(aFid); MEDfileClose(aFid);

View File

@ -27,6 +27,7 @@
#include "MED_Wrapper.hxx" #include "MED_Wrapper.hxx"
#include <string> #include <string>
#include <vector>
namespace MED namespace MED
{ {
@ -39,6 +40,9 @@ namespace MED
MEDWRAPPER_EXPORT MEDWRAPPER_EXPORT
bool CheckCompatibility( const std::string& , bool isForAppend=false); bool CheckCompatibility( const std::string& , bool isForAppend=false);
MEDWRAPPER_EXPORT
std::vector<int> GetMEDVersionsAppendCompatible();
MEDWRAPPER_EXPORT MEDWRAPPER_EXPORT
PWrapper CrWrapperR( const std::string& ); PWrapper CrWrapperR( const std::string& );

View File

@ -747,35 +747,33 @@ namespace
QMap<QString, int> aFilterMap; QMap<QString, int> aFilterMap;
if ( isMED ) { if ( isMED ) {
//filters << QObject::tr( "MED_FILES_FILTER" ) + " (*.med)"; //filters << QObject::tr( "MED_FILES_FILTER" ) + " (*.med)";
QString vmed (aMesh->GetVersionString(-1, 2)); //QString vmed (aMesh->GetVersionString(-1, 2));
//MESSAGE("MED version: " << vmed.toStdString()); //MESSAGE("MED version: " << vmed.toStdString());
int minor = vmed.split(".").last().toInt(); SMESH::long_array_var mvok = aMesh->GetMEDVersionsCompatibleForAppend();
//MESSAGE("MED version minor: "<< minor); for ( int i = 0; i < mvok->length(); ++i )
//minor +=3; // TODO remove: test multiple minor
aFilterMap.insert( QObject::tr( "MED_VX_FILES_FILTER" ).arg( vmed ) + " (*.med)", minor );
for (int ii=0; ii<minor; ii++)
{ {
QString vs = aMesh->GetVersionString(ii, 2); int versionInt = mvok[i];
//std::ostringstream vss; // TODO remove: test multiple minor std::ostringstream vss;
//vss << "4."; // TODO remove: test multiple minor vss << versionInt/10;
//vss << ii; // TODO remove: test multiple minor vss << ".";
//vs = vss.str().c_str(); // TODO remove: test multiple minor vss << versionInt%10;
//MESSAGE("MED version: " << vs.toStdString()); QString vs = vss.str().c_str();
aFilterMap.insert( QObject::tr( "MED_VX_FILES_FILTER" ).arg( vs ) + " (*.med)", ii); MESSAGE("MED version: " << vs.toStdString());
aFilterMap.insert( QObject::tr( "MED_VX_FILES_FILTER" ).arg( vs ) + " (*.med)", i);
} }
} }
else { // isSAUV else { // isSAUV
aFilterMap.insert("All files (*)", -1 ); aFilterMap.insert("All files (*)", -1 );
aFilterMap.insert("SAUV files (*.sauv)", -1 ); aFilterMap.insert("SAUV files (*.sauv)", 0 );
aFilterMap.insert("SAUV files (*.sauve)", -1 ); aFilterMap.insert("SAUV files (*.sauve)", -1 );
} }
QStringList filters; QStringList filters;
QString aDefaultFilter;
QMap<QString, int>::const_iterator it = aFilterMap.begin(); QMap<QString, int>::const_iterator it = aFilterMap.begin();
QString aDefaultFilter = it.key();
for ( ; it != aFilterMap.end(); ++it ) { for ( ; it != aFilterMap.end(); ++it ) {
filters.push_back( it.key() ); filters.push_back( it.key() );
if (it.key() == 0) if (it.value() == 0) // explicit default for MED = current MED version
aDefaultFilter = it.key(); aDefaultFilter = it.key();
} }
QStringList checkBoxes; QStringList checkBoxes;
@ -790,6 +788,7 @@ namespace
new SalomeApp_CheckFileDlg ( SMESHGUI::desktop(), false, checkBoxes, true, true, wdgList ); new SalomeApp_CheckFileDlg ( SMESHGUI::desktop(), false, checkBoxes, true, true, wdgList );
fd->setWindowTitle( aTitle ); fd->setWindowTitle( aTitle );
fd->setNameFilters( filters ); fd->setNameFilters( filters );
fd->selectNameFilter( aDefaultFilter );
fd->SetChecked( toCreateGroups, 0 ); fd->SetChecked( toCreateGroups, 0 );
fd->SetChecked( toFindOutDim, 1 ); fd->SetChecked( toFindOutDim, 1 );
if ( !anInitialPath.isEmpty() ) if ( !anInitialPath.isEmpty() )

View File

@ -468,6 +468,23 @@ char* SMESH_Mesh_i::GetVersionString(CORBA::Long minor, CORBA::Short nbDigits)
return CORBA::string_dup( ver.c_str() ); return CORBA::string_dup( ver.c_str() );
} }
//================================================================================
/*!
* Return the list of med versions compatibles for write/append,
* encoded in 10*major+minor (for instance, code for med 3.2.1 is 32)
*/
//================================================================================
SMESH::long_array* SMESH_Mesh_i::GetMEDVersionsCompatibleForAppend()
{
SMESH::long_array_var aResult = new SMESH::long_array();
std::vector<int> mvok = MED::GetMEDVersionsAppendCompatible();
long nbver = mvok.size();
aResult->length( nbver );
for ( int i = 0; i < nbver; i++ )
aResult[i] = mvok[i];
return aResult._retn();
}
//============================================================================= //=============================================================================
/*! /*!
* ImportUNVFile * ImportUNVFile

View File

@ -225,11 +225,18 @@ public:
* Consider maximum group name length stored in MED file. * Consider maximum group name length stored in MED file.
*/ */
CORBA::Boolean HasDuplicatedGroupNamesMED(); CORBA::Boolean HasDuplicatedGroupNamesMED();
/*! /*!
* Return string representation of a MED file version comprising nbDigits * Return string representation of a MED file version comprising nbDigits
*/ */
char* GetVersionString(CORBA::Long minor, CORBA::Short nbDigits); char* GetVersionString(CORBA::Long minor, CORBA::Short nbDigits);
/*!
* Return the list of med versions compatibles for write/append,
* encoded in 10*major+minor (for instance, code for med 3.2.1 is 32)
*/
SMESH::long_array* GetMEDVersionsCompatibleForAppend();
void ExportMED( const char* file, void ExportMED( const char* file,
CORBA::Boolean auto_groups, CORBA::Boolean auto_groups,
CORBA::Long minor, CORBA::Long minor,