Skip to content

Commit

Permalink
Separate Gmsh output format saving such that it can be used within li…
Browse files Browse the repository at this point in the history
…brary too for adaptive mesh refinement.

Add a Gmsh alternative to the default file I/O based meshing strategy.
  • Loading branch information
raback committed Oct 21, 2024
1 parent 51e1a69 commit b47701c
Show file tree
Hide file tree
Showing 9 changed files with 761 additions and 726 deletions.
175 changes: 108 additions & 67 deletions fem/src/Adaptive.F90
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,8 @@ MODULE Adaptive
USE LoadMod
USE MeshUtils
USE MeshRemeshing

USE SaveUtils

IMPLICIT NONE


Expand Down Expand Up @@ -1279,114 +1280,154 @@ FUNCTION External_ReMesh( RefMesh, ErrorLimit, HValue, NodalError, &
TYPE(Mesh_t), POINTER :: NewMesh, RefMesh
!------------------------------------------------------------------------------
TYPE(Mesh_t), POINTER :: Mesh
INTEGER :: i,j,k,n
INTEGER :: i,j,k,n,dim
REAL(KIND=dp) :: Lambda
REAL(KIND=dp), POINTER :: HValueF(:)
CHARACTER(:), ALLOCATABLE :: MeshCommand, Name, MeshInputFile
LOGICAL :: GmshFormat
!------------------------------------------------------------------------------

OPEN( 11, STATUS='UNKNOWN', FILE='bgmesh' )
WRITE( 11,* ) COUNT( NodalError > 100*AEPS )
dim = CoordinateSystemDimension()

! Create a temporal field that includes the desired mesh density where Hvalue has been computed.
! This is in terms of desired mesh density, currently -1 value is given if the nodal error is zero
! implying that nothing is computed here.
ALLOCATE(HvalueF(SIZE(HValue)))
HValueF = -1.0_dp
DO i=1,RefMesh % NumberOfNodes
IF ( NodalError(i) > 100*AEPS ) THEN
Lambda = ( ErrorLimit / NodalError(i) ) ** ( 1.0d0 / hConvergence(i) )

IF ( RefMesh % AdaptiveDepth < 1 ) THEN
Lambda = HValue(i) * MAX( MIN( Lambda, 1.33d0), 0.75d0)
ELSE
Lambda = HValue(i) * MAX(MIN(Lambda, MaxChange), 1.0d0/MaxChange)
END IF

IF( .NOT.Coarsening ) Lambda = MIN( Lambda, Hvalue(i) )

IF ( maxH > 0 ) Lambda = MIN( Lambda, maxH )
IF ( minH > 0 ) Lambda = MAX( Lambda, minH )
IF ( NodalError(i) > 100*AEPS ) THEN
Lambda = ( ErrorLimit / NodalError(i) ) ** ( 1.0d0 / hConvergence(i) )
IF ( RefMesh % AdaptiveDepth < 1 ) THEN
Lambda = HValue(i) * MAX( MIN( Lambda, 1.33d0), 0.75d0)
ELSE
Lambda = HValue(i) * MAX(MIN(Lambda, MaxChange), 1.0d0/MaxChange)
END IF
IF( .NOT.Coarsening ) Lambda = MIN( Lambda, Hvalue(i) )

IF ( CoordinateSystemDimension() == 2 ) THEN
WRITE(11,'(3e23.15)') RefMesh % Nodes % x(i), &
RefMesh % Nodes % y(i), Lambda
ELSE
WRITE(11,'(4e23.15)') RefMesh % Nodes % x(i), &
RefMesh % Nodes % y(i), &
RefMesh % Nodes % z(i), Lambda
END IF
ELSE
IF ( CoordinateSystemDimension() == 2 ) THEN
WRITE(11,'(3e23.15)') RefMesh % Nodes % x(i), &
RefMesh % Nodes % y(i), HValue(i)
ELSE
WRITE(11,'(4e23.15)') RefMesh % Nodes % x(i), &
RefMesh % Nodes % y(i), &
RefMesh % Nodes % z(i), HValue(i)
END IF
END IF
IF ( maxH > 0 ) Lambda = MIN( Lambda, maxH )
IF ( minH > 0 ) Lambda = MAX( Lambda, minH )
HValueF(i) = Lambda
END IF
END DO

WRITE(11,*) 0
CLOSE(11)

! Save the current mesh in Elmer mesh format
Path = ListGetString( Params, 'Adaptive Mesh Name', Found )
IF ( .NOT. Found ) Path = 'RefinedMesh'

i = RefMesh % AdaptiveDepth + 1
nLen = LEN_TRIM(Path)
Path = Path(1:nlen) // I2S(i)

IF ( .NOT. Found ) THEN
i = RefMesh % AdaptiveDepth + 1
Path = 'RefinedMesh'//I2S(i)
END IF

nLen = LEN_TRIM(OutputPath)
IF ( nlen > 0 ) THEN
Path = OutputPath(1:nlen) // '/' // TRIM(Path)
Path = OutputPath(1:nlen) // '/' // TRIM(Path)
ELSE
Path = TRIM(Path)
Path = TRIM(Path)
END IF

GmshFormat = ListGetLogical( Params,'Adaptive Remesh Use Gmsh', Found )

IF( GmshFormat ) THEN
CALL Info( Caller,'Saving background mesh density in gmsh 2.0 (.msh) format' )

! A cludge to change the pointer and save results in Gmsh format.
BLOCK
REAL(KIND=dp), POINTER :: PtoHvalue(:)
TYPE(Variable_t), POINTER :: HVar
HVar => VariableGet( RefMesh % Variables,'Hvalue')
pToHvalue => HVar % Values
HVar % Values => HvalueF
CALL ListAddString(Solver % Values,'Scalar Field 1','Hvalue')
CALL ListAddLogical(Solver % Values,'File Append',.FALSE.)
CALL ListAddLogical(Solver % Values,'Alter Topology',.TRUE.)
CALL ListAddNewString(Solver % Values, 'Output File Name', 'gmsh_bgmesh.msh')
CALL SaveGmshOutput( Model,Solver,0.0_dp,.FALSE.)
HVar % Values => PtoHvalue
END BLOCK

ELSE
CALL Info( Caller,'Saving background mesh density in point cloud format' )

OPEN( 11, STATUS='UNKNOWN', FILE='bgmesh.nodes' )
WRITE( 11,* ) COUNT( HValueF > 0.0_dp )
DO i=1,RefMesh % NumberOfNodes
IF(.NOT. (HValueF(i) > 0.0_dp )) CYCLE
IF (dim == 2 ) THEN
WRITE(11,'(3e23.15)') RefMesh % Nodes % x(i), &
RefMesh % Nodes % y(i), HValueF(i)
ELSE
WRITE(11,'(4e23.15)') RefMesh % Nodes % x(i), &
RefMesh % Nodes % y(i), &
RefMesh % Nodes % z(i), HValueF(i)
END IF
END DO
WRITE(11,*) 0
CLOSE(11)

CALL MakeDirectory( TRIM(Path) // CHAR(0) )
CALL WriteMeshToDisk( RefMesh, Path )
CALL MakeDirectory( TRIM(Path) // CHAR(0) )
CALL WriteMeshToDisk( RefMesh, Path )
END IF

Mesh => RefMesh
DO WHILE( ASSOCIATED( Mesh ) )
IF ( Mesh % AdaptiveDepth == 0 ) EXIT
Mesh => Mesh % Parent
IF ( Mesh % AdaptiveDepth == 0 ) EXIT
Mesh => Mesh % Parent
END DO

MeshInputFile = ListGetString( Params, 'Mesh Input File', Found )

IF ( .NOT. Found ) THEN
MeshInputFile = ListGetString( Model % Simulation, 'Mesh Input File' )
END IF
MeshCommand = ListGetString( Solver % Values,'Mesh Command',Found)
IF(.NOT. Found ) THEN
IF( GmshFormat ) THEN
CALL Fatal('ReMesh','For now, provide "Mesh Command" for Gmsh meshing!')
END IF

MeshCommand = TRIM(OutputPath) // '/' // TRIM(Mesh % Name) // '/' // &
TRIM( MeshInputFile )
MeshInputFile = ListGetString( Params, 'Mesh Input File', Found )
IF ( .NOT. Found ) THEN
MeshInputFile = ListGetString( Model % Simulation, 'Mesh Input File' )
END IF

SELECT CASE( CoordinateSystemDimension() )
CASE(2)
MeshCommand = 'Mesh2D ' // TRIM(MeshCommand) // ' ' // &
TRIM(Path) // ' --bgmesh=bgmesh'
MeshCommand = TRIM(OutputPath) // '/' // TRIM(Mesh % Name) // '/' // &
TRIM( MeshInputFile )

CASE(3)
MeshCommand = 'Mesh3D ' // TRIM(MeshCommand) // ' ' // &
TRIM(Path) // ' bgmesh'
END SELECT
SELECT CASE( dim )
CASE(2)
! Legacy mesh generator from the Elmer suite.
MeshCommand = 'Mesh2D '//TRIM(MeshCommand)//' '//TRIM(Path)// ' --bgmesh=bgmesh.nodes'
CASE(3)
MeshCommand = 'Mesh3D '//TRIM(MeshCommand)//' '//TRIM(Path)//' bgmesh.nodes'
END SELECT
END IF

CALL Info('ReMesh','System command: '//TRIM(MeshCommand),Level=10)
! Remeshing command.
CALL Info('ReMesh','Meshing command: '//TRIM(MeshCommand),Level=10)
CALL SystemCommand( MeshCommand )

! Check if also conversion command is given.
MeshCommand = ListGetString( Solver % Values,'Mesh Conversion Command',Found)
IF( Found ) THEN
CALL Info('ReMesh','Conversion command: '//TRIM(MeshCommand),Level=10)
CALL SystemCommand( MeshCommand )
END IF

! Read the new mesh.
NewMesh => LoadMesh2( Model, OutPutPath, Path, .FALSE., 1, 0 )

! Loading Gebhart factors is more or less obsolite.
IF ( Solver % Variable % Name == 'temperature' ) THEN
Name = ListGetString( Model % Simulation, 'Gebhart Factors', Found )
IF ( Found ) THEN
MeshCommand = 'View ' // TRIM(OutputPath) // &
'/' // TRIM(Mesh % Name) // ' ' // TRIM(Path)

CALL SystemCommand( MeshCommand )

Name = TRIM(OutputPath) // '/' // &
TRIM(Mesh % Name) // '/' // TRIM(Name)

TRIM(Mesh % Name) // '/' // TRIM(Name)
CALL LoadGebhartFactors( NewMesh, TRIM(Name) )
END IF
END IF

DEALLOCATE(HvalueF)

!------------------------------------------------------------------------------
END FUNCTION External_ReMesh
!------------------------------------------------------------------------------
Expand Down
1 change: 1 addition & 0 deletions fem/src/ParticleUtils.F90
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ MODULE ParticleUtils
USE Lists
USE MeshUtils
USE GeneralUtils
USE SaveUtils

IMPLICIT NONE

Expand Down
Loading

0 comments on commit b47701c

Please sign in to comment.