ITM Grid Service Library: Fortran 90

src/service/itm_grid_simplex.f90

Go to the documentation of this file.
00001 module itm_grid_simplex
00002 
00003   !> Service routines for grids consisting out of simplexes (nodes, triangles, tetrahedra...)
00004 
00005   use itm_types
00006   use itm_assert
00007   use itm_combinations
00008 
00009   use itm_grid_common
00010   use itm_grid_object
00011   use itm_grid_access
00012   use itm_grid_subgrid
00013 
00014   use itm_grid_simplex_dict
00015 
00016   implicit none
00017 
00018   !> Definition of default subgrids
00019   integer, parameter :: GRID_SIMPLEX_SUBGRID_0D = 1 ! all 0d objects
00020   integer, parameter :: GRID_SIMPLEX_SUBGRID_1D = 2 ! all 1d objects
00021   integer, parameter :: GRID_SIMPLEX_SUBGRID_2D = 3 ! all 2d objects
00022   integer, parameter :: GRID_SIMPLEX_SUBGRID_3D = 4 ! all 3d objects
00023 
00024   ! Same as above, with human-readable names
00025   integer, parameter :: GRID_SIMPLEX_NODES     = GRID_SIMPLEX_SUBGRID_0D ! all 0d objects
00026   integer, parameter :: GRID_SIMPLEX_EDGES     = GRID_SIMPLEX_SUBGRID_1D ! all 1d objects
00027   integer, parameter :: GRID_SIMPLEX_TRIANGLES = GRID_SIMPLEX_SUBGRID_2D ! all 2d objects  
00028   integer, parameter :: GRID_SIMPLEX_TETRAEDER = GRID_SIMPLEX_SUBGRID_3D ! all 3d objects 
00029 
00030 contains
00031 
00032   !> Write a n-dimensional grid consisting of simplexes (0d: nodes, 1d: edges, 2d: triangles, 3d: tetrahedra)
00033   !> into a grid descriptor, as well as the default subgrids for objects of all dimensions.
00034   !>
00035   !> The dimension n of the grid is taken as size(coordtype).
00036   !> 
00037   !> @param grid Grid descriptor to fill
00038   !> @param coordtype Dimension(n). Defines coordinate types / labels for the 
00039   !>                  individual spaces axes. See the 
00040   !>                  constants defined in itm_grid.f90 (COORDTYPE_*)
00041   !> @param nodes     The positions of the grid nodes in space (with axes as defined in coordtype). 
00042   !>                  First dimension is node index, second coordinate index, 
00043   !>                  i.e. nodes(i,:) is the position vector of node i.
00044   !> @param simplexes Node index lists for the invidiual n-dimensional cells/simplexes in the grid. 
00045   !>                  First dimension is simplex index, second dimension is node index (as referring to the nodes array),
00046   !>                  i.e. simplices(i,:) is the vector of node indices for simplex i.
00047   !>                  Grid node coordinates in the individual dimensions. 
00048   !> @param id        Name/Identifier of the grid.
00049   !> @param createSubgrids Optional flag controlling whether default subgrids
00050   !>                  are created. Default is .true.
00051   subroutine gridSetupSimplex( grid, coordtype, nodes, simplexes, id, createSubgrids )
00052     type(type_complexgrid), intent(out) :: grid 
00053     integer, dimension(:), intent(in) :: coordtype
00054     real(R8), dimension(:, :), intent(in) :: nodes
00055     integer, dimension(:, :), intent(in) :: simplexes
00056     character(*), intent(in), optional :: id
00057     logical, intent(in), optional :: createSubgrids
00058     
00059     ! internal
00060     integer :: idim, ndim, iSg, iObj
00061     type(GridObject) :: obj
00062     integer, allocatable :: classes(:,:)
00063     logical :: lCreateSubgrids = .true.
00064 
00065     character(9), parameter :: genId(0:3) = (/ 'nodes    ', 'edges    ', 'triangles', 'simplexes' /)
00066 
00067     if (present(createSubgrids)) lCreateSubgrids = createSubgrids
00068 
00069     call assert( size( coordtype ) == size( nodes, 2 ), &
00070          & "gridSetupSimplex: size of coordtype and nodes don't match" )
00071     call assert( size(coordtype) == size(simplexes, 2) - 1, &
00072          & "gridSetupSimplex: shape of simplexes is inconsistent with shape of coordtypes" )
00073 
00074     ndim = size(coordtype)
00075     allocate( grid % spaces(1) )
00076 
00077     ! ... fill in the grid data
00078     call gridSetupSimplexSpace( grid % spaces(1), coordtype, nodes, simplexes )
00079 
00080     ! Create subgrids 
00081     if (lCreateSubgrids) then
00082         call gridCreateDefaultSubGrids(grid, id)
00083     end if
00084 
00085   end subroutine gridSetupSimplex
00086 
00087 
00088   !> Set up a 1d space consisting of simplices. 
00089   subroutine gridSetupSimplexSpace( space, coordtype, nodes, simplexes )
00090     type(type_complexgrid_space), intent(inout) :: space !> The space descriptor to fill
00091     integer, dimension(:), intent(in) :: coordtype !> The coordinate type of the space
00092     real(R8), dimension(:, :), intent(in) :: nodes
00093     integer, dimension(:, :), intent(in) :: simplexes
00094    
00095     ! internal
00096     integer :: ndim, nnode, nsim
00097     integer :: nparents(size(nodes, 1))
00098     integer, allocatable :: children(:,:), simplex(:)
00099     type(SimplexDict) :: dictN, dictC
00100 
00101     integer :: isim, inode, ichild, idim
00102     integer :: MAX_ENTRIES = 10
00103 
00104     type(SimplexDictIterator) :: iter
00105 
00106     
00107     ndim = size( coordtype )  ! Simplex of dimension ndim has ndim+1 nodes
00108     nnode = size( nodes, 1 )
00109     nsim = size( simplexes, 1 )
00110     call assert( ndim == size(simplexes, 2) - 1 )
00111 
00112     ! Set coordinate types
00113     ! TODO: support multiple geometries
00114     allocate( space % coordtype(ndim, 1) )
00115     space % coordtype(:, 1) = coordtype
00116 
00117     ! Allocate object definition arrays
00118     allocate( space % objects(NDIM + 1) )
00119 
00120     ! Set up node information
00121     ! Geometry: normal geometry representation, third dimension unused.
00122     ! TODO: support multiple geometries
00123     allocate( space % objects(1) % geo(size(nodes, 1), ndim, 1, 1) )
00124     space % objects(1) % geo(:, :, 1, 1) = nodes
00125     ! space % nodes % xpoints, altgeo, alias: unused for this grid
00126 
00127     ! Do the ndim-simplices as one-offs
00128     
00129     ! Populate dictionary of ndim-1-simplices.
00130     ! We have to set up the entire dictionary first to have the correct global simplex indices.
00131     call createSimplexDict( dictC, ndim-1, nnode, MAX_ENTRIES )
00132     allocate(children(ndim+1, ndim))
00133     do isim = 1, nsim
00134         children = listChildren(simplexes(isim,:))
00135         do ichild = 1, size(children, 1)
00136             call insertSimplex( dictC, sortSimplex(children(ichild, :)) )
00137         end do
00138     end do
00139 
00140     ! Fill in object boundaries
00141 
00142     ! Boundary array
00143     allocate( space % objects(ndim+1) % boundary( nsim, ndim+1 ) )
00144 
00145     ! First set all to undefined
00146     space % objects(ndim+1) % boundary = GRID_UNDEFINED
00147     do isim = 1, nsim
00148         children = listChildren(simplexes(isim,:))
00149         do ichild = 1, size(children, 1)
00150             space % objects(ndim+1) % boundary(isim, ichild) &
00151                 & = simplexIndex(dictC, sortSimplex(children(ichild, :)))
00152         end do
00153     end do
00154     deallocate(children)
00155 
00156     ! From here on, do lower-dimensional objects in a standard way
00157     do idim = ndim - 1, 1 , -1
00158         write (*,*) "Writing simplexes of dimension", idim
00159         allocate(simplex(idim+1), children(idim+1, idim))
00160 
00161         dictN = dictC
00162         call createSimplexDict( dictC, idim-1, nnode, MAX_ENTRIES )
00163         call resetIterator(iter)
00164         do 
00165             call nextSimplex(dictN, iter, simplex)           
00166             if (simplexUndefined(simplex)) exit
00167 
00168             children = listChildren(simplex)
00169             do ichild = 1, size(children, 1)
00170                 call insertSimplex( dictC, sortSimplex(children(ichild, :)) )
00171             end do
00172         end do
00173 
00174         allocate( space % objects(idim+1) % boundary( countSimplexes(dictN), idim+1 ) )
00175         call resetIterator(iter)
00176         isim = 0
00177         do 
00178             call nextSimplex(dictN, iter, simplex)
00179             if (simplexUndefined(simplex)) exit
00180 
00181             isim = isim + 1
00182             children = listChildren(simplex)
00183             do ichild = 1, size(children, 1)               
00184                 space % objects(idim+1) % boundary(isim, ichild) &
00185                     & = simplexIndex(dictC, sortSimplex(children(ichild, :)))                
00186             end do
00187         end do    
00188 
00189         deallocate(simplex, children)
00190     end do
00191 
00192   end subroutine gridSetupSimplexSpace
00193 
00194 
00195   subroutine writeSimplexes(simplexes)
00196     integer, intent(in) :: simplexes(:,:)
00197 
00198     ! internal
00199     integer :: i
00200 
00201     do i = 1, size(simplexes, 1)
00202         write (*,*) "(", simplexes(i,:), ")"
00203     end do
00204 
00205   end subroutine writeSimplexes
00206 
00207   
00208 end module itm_grid_simplex
 All Classes Namespaces Files Functions Variables