ITM Grid Service Library: Fortran 90
|
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