gridlock/gridlock/grid.py

159 lines
5.6 KiB
Python
Raw Normal View History

2024-07-29 01:37:58 -07:00
from typing import ClassVar, Self
from collections.abc import Callable, Sequence
2016-03-15 20:07:07 -07:00
2022-10-18 19:44:30 -07:00
import numpy
from numpy.typing import NDArray, ArrayLike
2016-03-15 20:07:07 -07:00
import pickle
import warnings
2016-07-03 03:13:20 -07:00
import copy
2016-03-15 20:07:07 -07:00
from . import GridError
2024-07-29 01:37:58 -07:00
from .draw import GridDrawMixin
from .read import GridReadMixin
from .position import GridPosMixin
2016-03-15 20:07:07 -07:00
2017-10-10 16:42:11 -07:00
2022-10-18 19:44:30 -07:00
foreground_callable_type = Callable[[NDArray, NDArray, NDArray], NDArray]
2016-03-15 20:07:07 -07:00
2024-07-29 01:37:58 -07:00
class Grid(GridDrawMixin, GridReadMixin, GridPosMixin):
2016-03-15 20:07:07 -07:00
"""
Simulation grid metadata for finite-difference simulations.
Can be used to generate non-uniform rectangular grids (the entire grid
2016-03-15 20:07:07 -07:00
is generated based on the coordinates of the boundary points). Also does
straightforward natural <-> grid unit conversion.
This class handles data describing the grid, and should be paired with a
(separate) ndarray that contains the actual data in each cell. The `allocate()`
method can be used to create this ndarray.
The resulting `cell_data[i, a, b, c]` should correspond to the value in the
`i`-th grid, in the cell centered around
```
2017-09-25 00:41:00 -07:00
(xyz[0][a] + dxyz[0][a] * shifts[i, 0],
xyz[1][b] + dxyz[1][b] * shifts[i, 1],
xyz[2][c] + dxyz[2][c] * shifts[i, 2]).
```
You can get raw edge coordinates (`exyz`),
center coordinates (`xyz`),
cell sizes (`dxyz`),
2016-03-15 20:07:07 -07:00
from the properties named as above, or get them for a given grid by using the
`self.shifted_*xyz(which_shifts)` functions.
2017-09-25 00:43:13 -07:00
The sizes of adjacent cells are taken into account when applying shifts. The
total shift for each edge is chosen using `(shift * dx_of_cell_being_moved_through)`.
2016-03-15 20:07:07 -07:00
It is tricky to determine the size of the right-most cell after shifting,
since its right boundary should shift by `shifts[i][a] * dxyz[a][dxyz[a].size]`,
2016-03-15 20:07:07 -07:00
where the dxyz element refers to a cell that does not exist.
Because of this, we either assume this 'ghost' cell is the same size as the last
real cell, or, if `self.periodic[a]` is set to `True`, the same size as the first cell.
2016-03-15 20:07:07 -07:00
"""
2024-07-18 00:17:20 -07:00
exyz: list[NDArray]
2021-01-08 22:58:36 -08:00
"""Cell edges. Monotonically increasing without duplicates."""
2016-03-15 20:07:07 -07:00
2024-07-18 00:17:20 -07:00
periodic: list[bool]
2021-01-08 22:58:36 -08:00
"""For each axis, determines how far the rightmost boundary gets shifted. """
2022-10-18 19:44:30 -07:00
shifts: NDArray
2021-01-08 22:58:36 -08:00
"""Offsets `[[x0, y0, z0], [x1, y1, z1], ...]` for grid `0,1,...`"""
2022-10-18 19:44:30 -07:00
Yee_Shifts_E: ClassVar[NDArray] = 0.5 * numpy.array([
[1, 0, 0],
[0, 1, 0],
[0, 0, 1],
], dtype=float)
2019-11-27 19:31:48 -08:00
"""Default shifts for Yee grid E-field"""
2022-10-18 19:44:30 -07:00
Yee_Shifts_H: ClassVar[NDArray] = 0.5 * numpy.array([
[0, 1, 1],
[1, 0, 1],
[1, 1, 0],
], dtype=float)
2019-11-27 19:31:48 -08:00
"""Default shifts for Yee grid H-field"""
2016-03-15 20:07:07 -07:00
2022-10-18 19:44:30 -07:00
def __init__(
self,
pixel_edge_coordinates: Sequence[ArrayLike],
shifts: ArrayLike = Yee_Shifts_E,
2024-07-18 00:17:20 -07:00
periodic: bool | Sequence[bool] = False,
2022-10-18 19:44:30 -07:00
) -> None:
2016-03-15 20:07:07 -07:00
"""
Args:
pixel_edge_coordinates: 3-element list of (ndarrays or lists) specifying the
coordinates of the pixel edges in each dimensions
(ie, `[[x0, x1, x2,...], [y0,...], [z0,...]]` where the first pixel has x-edges x=`x0` and
x=`x1`, the second has edges x=`x1` and x=`x2`, etc.)
shifts: Nx3 array containing `[x, y, z]` offsets for each of N grids.
E-field Yee shifts are used by default.
periodic: Specifies how the sizes of edge cells are calculated; see main class
documentation. List of 3 bool, or a single bool that gets broadcast. Default `False`.
Raises:
`GridError` on invalid input
2016-03-15 20:07:07 -07:00
"""
2024-07-29 01:37:58 -07:00
edge_arrs = [numpy.array(cc, copy=False) for cc in pixel_edge_coordinates]
self.exyz = [numpy.unique(edges) for edges in edge_arrs]
2021-01-08 22:58:36 -08:00
self.shifts = numpy.array(shifts, dtype=float)
2019-11-27 19:32:53 -08:00
2016-03-15 20:07:07 -07:00
for i in range(3):
2024-07-29 01:37:58 -07:00
if self.exyz[i].size != edge_arrs[i].size:
2021-10-24 19:05:48 -07:00
warnings.warn(f'Dimension {i} had duplicate edge coordinates', stacklevel=2)
2016-03-15 20:07:07 -07:00
2021-01-08 22:58:36 -08:00
if isinstance(periodic, bool):
self.periodic = [periodic] * 3
else:
self.periodic = list(periodic)
2016-03-15 20:07:07 -07:00
if len(self.shifts.shape) != 2:
raise GridError('Misshapen shifts: shifts must have two axes! '
2021-10-24 19:05:48 -07:00
f' The given shifts has shape {self.shifts.shape}')
2016-03-15 20:07:07 -07:00
if self.shifts.shape[1] != 3:
raise GridError('Misshapen shifts; second axis size should be 3,'
2021-10-24 19:05:48 -07:00
f' shape is {self.shifts.shape}')
2016-03-15 20:07:07 -07:00
2017-09-25 00:43:26 -07:00
if (numpy.abs(self.shifts) > 1).any():
raise GridError('Only shifts in the range [-1, 1] are currently supported')
if (self.shifts < 0).any():
# TODO: Test negative shifts
2019-04-07 17:18:57 -07:00
warnings.warn('Negative shifts are still experimental and mostly untested, be careful!', stacklevel=2)
2017-09-25 00:43:26 -07:00
2016-03-15 20:07:07 -07:00
@staticmethod
def load(filename: str) -> 'Grid':
"""
Load a grid from a file
Args:
filename: Filename to load from.
2016-03-15 20:07:07 -07:00
"""
with open(filename, 'rb') as f:
tmp_dict = pickle.load(f)
g = Grid([[-1, 1]] * 3)
g.__dict__.update(tmp_dict)
return g
2024-07-18 00:17:20 -07:00
def save(self, filename: str) -> Self:
2016-03-15 20:07:07 -07:00
"""
Save to file.
Args:
filename: Filename to save to.
2021-10-24 19:06:27 -07:00
Returns:
self
2016-03-15 20:07:07 -07:00
"""
with open(filename, 'wb') as f:
pickle.dump(self.__dict__, f, protocol=2)
2021-10-24 19:06:27 -07:00
return self
2016-03-15 20:07:07 -07:00
2024-07-18 00:17:20 -07:00
def copy(self) -> Self:
2016-07-03 03:13:20 -07:00
"""
Returns:
Deep copy of the grid.
2016-07-03 03:13:20 -07:00
"""
return copy.deepcopy(self)