Skip to content
Snippets Groups Projects
tnl-view.h 17.1 KiB
Newer Older
/***************************************************************************
                          tnl-view.h  -  description
                             -------------------
    begin                : Jan 21, 2013
    copyright            : (C) 2013 by Tomas Oberhuber
    email                : tomas.oberhuber@fjfi.cvut.cz
 ***************************************************************************/

/***************************************************************************
 *                                                                         *
 *   This program is free software; you can redistribute it and/or modify  *
 *   it under the terms of the GNU General Public License as published by  *
 *   the Free Software Foundation; either version 2 of the License, or     *
 *   (at your option) any later version.                                   *
 *                                                                         *
 ***************************************************************************/

#ifndef TNL_VIEW_H_
#define TNL_VIEW_H_

Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
#include <cstdlib>
#include <core/mfilename.h>
#include <config/tnlParameterContainer.h>
#include <core/tnlString.h>
#include <core/vectors/tnlVector.h>
#include <core/vectors/tnlMultiVector.h>
#include <mesh/tnlGrid.h>
#include <functions/tnlMeshFunction.h>
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
using namespace std;
bool getOutputFileName( const tnlString& inputFileName,
                        const tnlString& outputFormat,
                        tnlString& outputFileName )
{
   outputFileName = inputFileName;
   RemoveFileExtension( outputFileName );
   if( outputFormat == "gnuplot" )
   {
      outputFileName += ".gplt";
      return true;
   }
   else
   {
      cerr << "Unknown file format " << outputFormat << ".";
      return false;
   }
}


template< typename MeshFunction >
bool writeMeshFunction( const typename MeshFunction::MeshType& mesh,
                        const tnlString& inputFileName,
                        const tnlParameterContainer& parameters  )
{
   MeshFunction function( mesh );
   if( ! function.load( inputFileName ) )
   {
      std::cerr << "Unable to load mesh function from a file " << inputFileName << "." << std::endl;
      return false;
   }

   int verbose = parameters. getParameter< int >( "verbose");
   tnlString outputFormat = parameters. getParameter< tnlString >( "output-format" );
   tnlString outputFileName;
   if( ! getOutputFileName( inputFileName,
                            outputFormat,
                            outputFileName ) )
      return false;
   if( verbose )
      cout << " writing to " << outputFileName << " ... " << flush;
   return function.write( outputFileName, outputFormat );
}

template< typename Mesh,
          int EntityDimensions,
          typename Real >
bool setMeshFunctionRealType( const Mesh& mesh,
                              const tnlString& inputFileName,
                              const tnlParameterContainer& parameters  )
{
   return writeMeshFunction< tnlMeshFunction< Mesh, EntityDimensions, Real > >( mesh, inputFileName, parameters );
}

template< typename Mesh,
          int EntityDimensions >
bool setMeshEntityType( const Mesh& mesh,
                        const tnlString& inputFileName,
                        const tnlList< tnlString >& parsedObjectType,
                        const tnlParameterContainer& parameters )
{
   if( parsedObjectType[ 3 ] == "float" )
      return setMeshFunctionRealType< Mesh, EntityDimensions, float >( mesh, inputFileName, parameters );
   if( parsedObjectType[ 3 ] == "double" )
      return setMeshFunctionRealType< Mesh, EntityDimensions, double >( mesh, inputFileName, parameters );
   if( parsedObjectType[ 3 ] == "long double" )
      return setMeshFunctionRealType< Mesh, EntityDimensions, long double >( mesh, inputFileName, parameters );
   std::cerr << "Unsupported arithmetics " << parsedObjectType[ 3 ] << " in mesh function " << inputFileName << std::endl;
   return false;
}

template< typename MeshReal,
          typename MeshIndex >
bool setMeshEntityDimensions( const tnlGrid< 1, MeshReal, tnlHost, MeshIndex >& mesh,
                              const tnlString& inputFileName,
                              const tnlList< tnlString >& parsedObjectType,
                              const tnlParameterContainer& parameters )
   typedef tnlGrid< 1, MeshReal, tnlHost, MeshIndex > Mesh;
   int meshEntityDimensions = atoi( parsedObjectType[ 2 ].getString() );
   switch( meshEntityDimensions )
      case 1:
         return setMeshEntityType< Mesh, 1 >( mesh, inputFileName, parsedObjectType, parameters );
         break;
      default:
         cerr << "Unsupported mesh functions entity dimensions count " << meshEntityDimensions << "." << endl;
         return false;
}

template< typename MeshReal,
          typename MeshIndex >
bool setMeshEntityDimensions( const tnlGrid< 2, MeshReal, tnlHost, MeshIndex >& mesh,
                              const tnlString& inputFileName,
                              const tnlList< tnlString >& parsedObjectType,
                              const tnlParameterContainer& parameters )
{
   typedef tnlGrid< 2, MeshReal, tnlHost, MeshIndex > Mesh;
   int meshEntityDimensions = atoi( parsedObjectType[ 2 ].getString() );
   switch( meshEntityDimensions )
   {
      case 1:
         return setMeshEntityType< Mesh, 1 >( mesh, inputFileName, parsedObjectType, parameters );
         break;
      case 2:
         return setMeshEntityType< Mesh, 2 >( mesh, inputFileName, parsedObjectType, parameters );
         break;
      default:
         cerr << "Unsupported mesh functions entity dimensions count " << meshEntityDimensions << "." << endl;
         return false;
   }
}

template< typename MeshReal,
          typename MeshIndex >
bool setMeshEntityDimensions( const tnlGrid< 3, MeshReal, tnlHost, MeshIndex >& mesh,
                              const tnlString& inputFileName,
                              const tnlList< tnlString >& parsedObjectType,
                              const tnlParameterContainer& parameters )
{
   typedef tnlGrid< 3, MeshReal, tnlHost, MeshIndex > Mesh;
   int meshEntityDimensions = atoi( parsedObjectType[ 2 ].getString() );
   switch( meshEntityDimensions )
   {
      case 1:
         return setMeshEntityType< Mesh, 1 >( mesh, inputFileName, parsedObjectType, parameters );
         break;
      case 2:
         return setMeshEntityType< Mesh, 2 >( mesh, inputFileName, parsedObjectType, parameters );
         break;
      case 3:
         return setMeshEntityType< Mesh, 3 >( mesh, inputFileName, parsedObjectType, parameters );
         break;
      default:
         cerr << "Unsupported mesh functions entity dimensions count " << meshEntityDimensions << "." << endl;
         return false;
   }
}

template< typename Mesh >
bool setMeshFunction( const Mesh& mesh,
                      const tnlString& inputFileName,
                      const tnlList< tnlString >& parsedObjectType,
                      const tnlParameterContainer& parameters )
{
   if( parsedObjectType[ 1 ] != mesh.getSerializationType() )
   {
      cerr << "Incompatible mesh type for the mesh function " << inputFileName << "." << endl;
      return false;
   }
   return setMeshEntityDimensions( mesh, inputFileName, parsedObjectType, parameters );
}

template< typename Mesh, typename Element, typename Real, typename Index, int Dimensions >
bool convertObject( const Mesh& mesh,
                    const tnlString& inputFileName,
                    const tnlList< tnlString >& parsedObjectType,
                    const tnlParameterContainer& parameters )
{
   int verbose = parameters. getParameter< int >( "verbose");
   tnlString outputFormat = parameters. getParameter< tnlString >( "output-format" );
   tnlString outputFileName;
   if( ! getOutputFileName( inputFileName,
                            outputFormat,
                            outputFileName ) )
      return false;
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   if( verbose )
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
      cout << " writing to " << outputFileName << " ... " << flush;
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   if( parsedObjectType[ 0 ] == "tnlSharedVector" ||
       parsedObjectType[ 0 ] == "tnlVector" )
   {
      tnlVector< Element, tnlHost, Index > vector;
      if( ! vector. load( inputFileName ) )
         return false;
      if( ! mesh. write( vector, outputFileName, outputFormat ) )
         return false;
   }

   if( parsedObjectType[ 0 ] == "tnlMultiVector" ||
       parsedObjectType[ 0 ] == "tnlSharedMultiVector" )
   {
      tnlMultiVector< Dimensions, Element, tnlHost, Index > multiVector;
      if( ! multiVector. load( inputFileName ) )
         return false;
      typedef tnlGrid< Dimensions, Real, tnlHost, Index > GridType;
      typedef typename GridType::VertexType VertexType;
      typedef typename GridType::CoordinatesType CoordinatesType;
      GridType grid;
      grid. setDomain( VertexType( 0.0 ), VertexType( 1.0 ) );
      grid. setDimensions( CoordinatesType( multiVector. getDimensions() ) );
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
      const Real spaceStep = grid. getSpaceSteps(). x();
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
      if( ! grid. write( multiVector, outputFileName, outputFormat ) )
         return false;
   }
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   return true;
template< typename Mesh, typename Element, typename Real, typename Index >
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
bool setDimensions( const Mesh& mesh,
                    const tnlString& inputFileName,
                    const tnlList< tnlString >& parsedObjectType,
                    const tnlParameterContainer& parameters )
{
   int dimensions( 0 );
   if( parsedObjectType[ 0 ] == "tnlMultiVector" ||
       parsedObjectType[ 0 ] == "tnlSharedMultiVector" )
      dimensions = atoi( parsedObjectType[ 1 ]. getString() );
   if( parsedObjectType[ 0 ] == "tnlVector" ||
       parsedObjectType[ 0 ] == "tnlSharedVector" )
      dimensions = 1;
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   switch( dimensions )
   {
      case 1:
         return convertObject< Mesh, Element, Real, Index, 1 >( mesh, inputFileName, parsedObjectType, parameters );
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
      case 2:
         return convertObject< Mesh, Element, Real, Index, 2 >( mesh, inputFileName, parsedObjectType, parameters );
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
      case 3:
         return convertObject< Mesh, Element, Real, Index, 3 >( mesh, inputFileName, parsedObjectType, parameters );
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   }
   cerr << "Cannot convert objects with " << dimensions << " dimensions." << endl;
   return false;
}

template< typename Mesh, typename Element, typename Real >
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
bool setIndexType( const Mesh& mesh,
                   const tnlString& inputFileName,
                   const tnlList< tnlString >& parsedObjectType,
                   const tnlParameterContainer& parameters )
{
   tnlString indexType;
   if( parsedObjectType[ 0 ] == "tnlMultiVector" ||
       parsedObjectType[ 0 ] == "tnlSharedMultiVector" )
      indexType = parsedObjectType[ 4 ];
   if( parsedObjectType[ 0 ] == "tnlSharedVector" ||
       parsedObjectType[ 0 ] == "tnlVector" )
      indexType = parsedObjectType[ 3 ];
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed

   if( indexType == "int" )
      return setDimensions< Mesh, Element, Real, int >( mesh, inputFileName, parsedObjectType, parameters );
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   if( indexType == "long-int" )
      return setDimensions< Mesh, Element, Real, long int >( mesh, inputFileName, parsedObjectType, parameters );
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   cerr << "Unknown index type " << indexType << "." << endl;
   return false;
}

template< typename Mesh >
bool setTupleType( const Mesh& mesh,
                   const tnlString& inputFileName,
                   const tnlList< tnlString >& parsedObjectType,
                   const tnlList< tnlString >& parsedElementType,
                   const tnlParameterContainer& parameters )
{
   int dimensions = atoi( parsedElementType[ 1 ].getString() );
   tnlString dataType = parsedElementType[ 2 ];
   if( dataType == "float" )
      switch( dimensions )
      {
         case 1:
            return setIndexType< Mesh, tnlStaticVector< 1, float >, float >( mesh, inputFileName, parsedObjectType, parameters );
            return setIndexType< Mesh, tnlStaticVector< 2, float >, float >( mesh, inputFileName, parsedObjectType, parameters );
            return setIndexType< Mesh, tnlStaticVector< 3, float >, float >( mesh, inputFileName, parsedObjectType, parameters );
            break;
      }
   if( dataType == "double" )
      switch( dimensions )
      {
         case 1:
            return setIndexType< Mesh, tnlStaticVector< 1, double >, double >( mesh, inputFileName, parsedObjectType, parameters );
            return setIndexType< Mesh, tnlStaticVector< 2, double >, double >( mesh, inputFileName, parsedObjectType, parameters );
            return setIndexType< Mesh, tnlStaticVector< 3, double >, double >( mesh, inputFileName, parsedObjectType, parameters );
            break;
      }
   if( dataType == "long double" )
      switch( dimensions )
      {
         case 1:
            return setIndexType< Mesh, tnlStaticVector< 1, long double >, long double >( mesh, inputFileName, parsedObjectType, parameters );
            return setIndexType< Mesh, tnlStaticVector< 2, long double >, long double >( mesh, inputFileName, parsedObjectType, parameters );
            return setIndexType< Mesh, tnlStaticVector< 3, long double >, long double >( mesh, inputFileName, parsedObjectType, parameters );
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
template< typename Mesh >
bool setElementType( const Mesh& mesh,
                     const tnlString& inputFileName,
                     const tnlList< tnlString >& parsedObjectType,
                     const tnlParameterContainer& parameters )
{
   tnlString elementType;
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   // TODO: Fix this even for arrays
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   if( parsedObjectType[ 0 ] == "tnlMultiVector" ||
       parsedObjectType[ 0 ] == "tnlSharedMultiVector" )
      elementType = parsedObjectType[ 2 ];
   if( parsedObjectType[ 0 ] == "tnlSharedVector" ||
       parsedObjectType[ 0 ] == "tnlVector" )
      elementType = parsedObjectType[ 1 ];

Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   if( elementType == "float" )
      return setIndexType< Mesh, float, float >( mesh, inputFileName, parsedObjectType, parameters );
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   if( elementType == "double" )
      return setIndexType< Mesh, double, double >( mesh, inputFileName, parsedObjectType, parameters );
   if( elementType == "long double" )
      return setIndexType< Mesh, long double, long double >( mesh, inputFileName, parsedObjectType, parameters );
   tnlList< tnlString > parsedElementType;
   if( ! parseObjectType( elementType, parsedElementType ) )
   {
      cerr << "Unable to parse object type " << elementType << "." << endl;
      return false;
   }
   if( parsedElementType[ 0 ] == "tnlStaticVector" )
      return setTupleType< Mesh >( mesh, inputFileName, parsedObjectType, parsedElementType, parameters );

Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   cerr << "Unknown element type " << elementType << "." << endl;
   return false;
}

template< typename Mesh >
bool processFiles( const tnlParameterContainer& parameters )
{
   int verbose = parameters. getParameter< int >( "verbose");
   tnlString meshFile = parameters. getParameter< tnlString >( "mesh" );
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
   Mesh mesh;
   if( meshFile != "" )
      if( ! mesh. load( meshFile ) )
      {
         cerr << "I am not able to load mesh from the file " << meshFile << "." << endl;
         return false;
      }
   mesh. writeMesh( "mesh.asy", "asymptote" );
   bool checkOutputFile = parameters. getParameter< bool >( "check-output-file" );
   tnlList< tnlString > inputFiles = parameters. getParameter< tnlList< tnlString > >( "input-files" );
   bool error( false );
//#ifdef HAVE_OPENMP
//#pragma omp parallel for
//#endif
   for( int i = 0; i < inputFiles. getSize(); i ++ )
   {
      if( verbose )
         cout << "Processing file " << inputFiles[ i ] << " ... " << flush;

      tnlString outputFormat = parameters. getParameter< tnlString >( "output-format" );
      tnlString outputFileName;
      if( ! getOutputFileName( inputFiles[ i ],
                               outputFormat,
                               outputFileName ) )
      if( checkOutputFile && fileExists( outputFileName ) )
      {
         if( verbose )
            cout << " file already exists. Skipping.            \r" << flush;
         continue;
      }

      tnlString objectType;
      if( ! getObjectType( inputFiles[ i ], objectType ) )
          cerr << "unknown object ... SKIPPING!" << endl;
      else
      {
         if( verbose )
            cout << objectType << " detected ... ";
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
         tnlList< tnlString > parsedObjectType;
         if( ! parseObjectType( objectType, parsedObjectType ) )
Tomáš Oberhuber's avatar
Tomáš Oberhuber committed
            cerr << "Unable to parse object type " << objectType << "." << endl;
            error = true;
            continue;
         if( parsedObjectType[ 0 ] == "tnlMultiVector" ||
             parsedObjectType[ 0 ] == "tnlSharedMultiVector" ||      
             parsedObjectType[ 0 ] == "tnlSharedVector" ||
             parsedObjectType[ 0 ] == "tnlVector" )
            setElementType< Mesh >( mesh, inputFiles[ i ], parsedObjectType, parameters );
         if( parsedObjectType[ 0 ] == "tnlMeshFunction" )
            setMeshFunction< Mesh >( mesh, inputFiles[ i ], parsedObjectType, parameters );
         if( verbose )
            cout << "[ OK ].  " << endl;

   if( verbose )
      cout << endl;