from abc import ABCMeta, abstractmethod
from functools import partial
from typing import Callable, List, Tuple
import numpy as np
import scipy.constants as cn
import scipy.fft
[docs]
def poisson(
rho: np.ndarray,
dx: float,
boundary_types: List[str] = ["periodic", "periodic", "periodic"],
boundary_values: Tuple[Tuple[float]] = [(0.0, 0.0), (0.0, 0.0), (0.0, 0.0)],
btypes: str = None,
epsilon_0=cn.epsilon_0,
):
"""Solve Poisson's equation with FFT.
Parameters
----------
rho : np.ndarray
3-dimentional array of the charge density [C/m^3]. The shape is (nz+1, ny+1, nx+1).
boundary_types : List[str] of {'periodic', 'dirichlet', 'neumann'},
the boundary condition types, by default ['periodic', 'periodic', 'periodic']
boundary_values : List[Tuple[float]]
the boundary values [(x-lower, x-upper), (y-lower, y-upper), (z-lower, z-upper)],
by default [(0., 0.), (0., 0.), (0., 0.)]
btypes : str
string consisting of prefixes of boundary conditions, by default None.
If this is set, it takes precedence over boundary_types.
dx : float, optional
the grid width [m], by default 1.0
epsilon_0 : _type_, optional
the electric constant (vacuum permittivity) [F/m], by default cn.epsilon_0
Returns
-------
np.ndarray
3-dimentional of the potential [V].
"""
# If a boundary condition is specified in abbreviated form by btypes, revert to the original notation.
if btypes:
btypes_dict = {
"p": "periodic",
"d": "dirichlet",
"n": "neumann",
}
boundary_types = [btypes_dict[btype] for btype in btypes]
POISSON_BOUNDARIES = {
"periodic": PeriodicPoissonBoundary,
"dirichlet": DirichletPoissonBoundary,
"neumann": NeumannPoissonBoundary,
}
# [x-boundary, y-boundary, z-boundary]
boundaries: List[PoissonBoundary] = [
POISSON_BOUNDARIES[_type](2 - i, boundary_values[i])
for i, _type in enumerate(boundary_types)
]
rho_target = rho[
tuple(boundary.get_target_slice() for boundary in reversed(boundaries))
].copy()
# Poisson's equation: dphi/dx^2 = -rho/epsilon_0
rho_target = -rho_target / epsilon_0 * dx * dx
# Transpose boundary values.
for boundary in boundaries:
boundary.transpose_boundary_values(rho_target, dx)
# Create a FFT-solver with 3d data.
forwards = [boundary.fft_forward for boundary in boundaries]
backwards = [boundary.fft_backward for boundary in boundaries]
fft3d = FFT3d(forwards, backwards)
# FFT forward.
rhok = fft3d.forward(rho_target)
# Caluculate a modified wave number.
modified_wave_number = np.zeros_like(rhok, dtype=float)
nz, ny, nx = np.array(rho.shape) - 1
for kx in range(rhok.shape[2]):
modified_wave_number[:, :, kx] += boundaries[0].modified_wave_number(kx, nx)
for ky in range(rhok.shape[1]):
modified_wave_number[:, ky, :] += boundaries[1].modified_wave_number(ky, ny)
for kz in range(rhok.shape[0]):
modified_wave_number[kz, :, :] += boundaries[2].modified_wave_number(kz, nz)
# Solve the equation in the wavenumber domain
phik = rhok / modified_wave_number
# When all boundary conditions are periodic|neumann boundaries,
# there is no reference for the potential and it is not uniquely determined,
# so the average is set to zero.
if all([_type in ("periodic", "neumann") for _type in boundary_types]):
phik[0, 0, 0] = 0.0
# FFT backward
_phi = fft3d.backward(phik)
# Create an array of the same shape as the input rho array.
phi = np.zeros_like(rho)
phi[tuple(boundary.get_target_slice() for boundary in reversed(boundaries))] = _phi
# In the above, the operation was performed on the array excluding the boundary values,
# so the boundary values are substituted here.
for boundary in boundaries:
boundary.correct_boundary_values(phi)
return phi
[docs]
class FFT3d:
"""FFT3d クラス。
"""
def __init__(
self,
forwards: List[Callable[[np.ndarray], np.ndarray]],
backwards: List[Callable[[np.ndarray], np.ndarray]],
):
"""インスタンスを初期化する。
Parameters
----------
forwards : List[Callable[[np.ndarray], np.ndarray]]
コールバック関数です。
backwards : List[Callable[[np.ndarray], np.ndarray]]
コールバック関数です。
"""
self.__forwards = forwards
self.__backwards = backwards
[docs]
def forward(self, data3d: np.ndarray) -> np.ndarray:
"""順変換を適用する。
Parameters
----------
data3d : np.ndarray
3 次元データ。
Returns
-------
np.ndarray
処理結果です。
"""
result3d = data3d
result3d = self.__forwards[2](result3d, axis=0, norm="ortho")
result3d = self.__forwards[1](result3d, axis=1, norm="ortho")
result3d = self.__forwards[0](result3d, axis=2, norm="ortho")
return result3d
[docs]
def backward(self, data3d: np.ndarray) -> np.ndarray:
"""逆変換を適用する。
Parameters
----------
data3d : np.ndarray
3 次元データ。
Returns
-------
np.ndarray
処理結果です。
"""
result3d = data3d
result3d = self.__backwards[2](result3d, axis=0, norm="ortho")
result3d = self.__backwards[1](result3d, axis=1, norm="ortho")
result3d = self.__backwards[0](result3d, axis=2, norm="ortho")
return result3d
[docs]
class PoissonBoundary(metaclass=ABCMeta):
"""PoissonBoundary クラス。
"""
def __init__(self, axis: int, boundary_values: Tuple[float] = (0.0, 0.0)):
"""インスタンスを初期化する。
Parameters
----------
axis : int
対象軸。
boundary_values : Tuple[float], optional
対象軸の両端境界値 `(lower, upper)` です。
"""
self.__axis = axis
self.__boundary_values = boundary_values
@property
def axis(self) -> int:
"""対象軸の情報を返す。
Returns
-------
int
件数または index を返します。
"""
return self.__axis
@property
def boundary_values(self) -> Tuple[float]:
"""境界値 `(lower, upper)` を返す。
Returns
-------
Tuple[float]
処理結果です。
"""
return self.__boundary_values
@property
@abstractmethod
def fft_forward(self) -> Callable[[np.ndarray], np.ndarray]:
"""境界条件に対応した順方向 FFT 関数を返す。
Returns
-------
Callable[[np.ndarray], np.ndarray]
処理結果です。
"""
pass
@property
@abstractmethod
def fft_backward(self) -> Callable[[np.ndarray], np.ndarray]:
"""境界条件に対応した逆方向 FFT 関数を返す。
Returns
-------
Callable[[np.ndarray], np.ndarray]
処理結果です。
"""
pass
[docs]
@abstractmethod
def get_target_slice(self) -> slice:
"""値を取得する。
Returns
-------
slice
解く対象軸に対応するスライスです。
"""
pass
[docs]
@abstractmethod
def modified_wave_number(self, k: int, n: int) -> float:
"""境界条件に応じた修正波数を計算する。
Parameters
----------
k : int
波数 index です。
n : int
サンプル数または格子点数です。
Returns
-------
float
処理結果です。
"""
pass
[docs]
@abstractmethod
def transpose_boundary_values(self, rho_target: np.ndarray, dx: float) -> None:
"""boundary values を転置処理を行う。
Parameters
----------
rho_target : np.ndarray
解く対象の電荷密度配列です。
dx : float
x 方向の格子間隔です。
Returns
-------
None
戻り値はありません。
"""
pass
[docs]
@abstractmethod
def correct_boundary_values(self, phi: np.ndarray) -> None:
"""boundary values を補正する。
Parameters
----------
phi : np.ndarray
電位配列です。
Returns
-------
None
戻り値はありません。
"""
pass
def _get_slices_at(self, index_axis) -> Tuple[slice]:
"""slices at を取得する。
Parameters
----------
index_axis : object
`self.axis` に対応する固定 index(例: `0`, `-1`)です。
Returns
-------
Tuple[slice]
処理結果です。
"""
return tuple(index_axis if i == self.axis else slice(None) for i in range(3))
[docs]
class PeriodicPoissonBoundary(PoissonBoundary):
"""PeriodicPoissonBoundary クラス。
"""
@property
def fft_forward(self) -> Callable[[np.ndarray], np.ndarray]:
"""境界条件に対応した順方向 FFT 関数を返す。
Returns
-------
Callable[[np.ndarray], np.ndarray]
処理結果です。
"""
return scipy.fft.fft
@property
def fft_backward(self) -> Callable[[np.ndarray], np.ndarray]:
"""境界条件に対応した逆方向 FFT 関数を返す。
Returns
-------
Callable[[np.ndarray], np.ndarray]
処理結果です。
"""
return scipy.fft.ifft
[docs]
def get_target_slice(self) -> slice:
"""値を取得する。
Returns
-------
slice
処理結果です。
"""
return slice(0, -1)
[docs]
def modified_wave_number(self, k: int, n: int) -> float:
"""境界条件に応じた修正波数を計算する。
Parameters
----------
k : int
波数 index です。
n : int
サンプル数または格子点数です。
Returns
-------
float
処理結果です。
"""
if k <= int(n / 2):
wn = 2.0 * np.sin(np.pi * k / float(n))
else:
wn = 2.0 * np.sin(np.pi * (n - k) / float(n))
wn = -wn * wn
return wn
[docs]
def transpose_boundary_values(self, rho_target: np.ndarray, dx: float) -> None:
"""boundary values を転置処理を行う。
Parameters
----------
rho_target : np.ndarray
解く対象の電荷密度配列です。
dx : float
x 方向の格子間隔です。
Returns
-------
None
戻り値はありません。
"""
pass
[docs]
def correct_boundary_values(self, phi: np.ndarray) -> None:
"""boundary values を補正する。
Parameters
----------
phi : np.ndarray
電位配列です。
Returns
-------
None
戻り値はありません。
"""
phi[self._get_slices_at(-1)] = phi[self._get_slices_at(0)]
[docs]
class DirichletPoissonBoundary(PoissonBoundary):
"""DirichletPoissonBoundary クラス。
"""
@property
def fft_forward(self) -> Callable[[np.ndarray], np.ndarray]:
"""境界条件に対応した順方向 FFT 関数を返す。
Returns
-------
Callable[[np.ndarray], np.ndarray]
処理結果です。
"""
return partial(scipy.fft.dst, type=1)
@property
def fft_backward(self) -> Callable[[np.ndarray], np.ndarray]:
"""境界条件に対応した逆方向 FFT 関数を返す。
Returns
-------
Callable[[np.ndarray], np.ndarray]
処理結果です。
"""
return partial(scipy.fft.idst, type=1)
[docs]
def get_target_slice(self) -> slice:
"""値を取得する。
Returns
-------
slice
処理結果です。
"""
return slice(1, -1)
[docs]
def modified_wave_number(self, k: int, n: int) -> float:
"""境界条件に応じた修正波数を計算する。
Parameters
----------
k : int
波数 index です。
n : int
サンプル数または格子点数です。
Returns
-------
float
処理結果です。
"""
wn = 2.0 * (np.cos(np.pi * (k + 1) / float(n + 1)) - 1.0)
return wn
[docs]
def transpose_boundary_values(self, rho_target: np.ndarray, dx: float) -> None:
"""boundary values を転置処理を行う。
Parameters
----------
rho_target : np.ndarray
解く対象の電荷密度配列です。
dx : float
x 方向の格子間隔です。
Returns
-------
None
戻り値はありません。
"""
rho_target[self._get_slices_at(0)] = (
rho_target[self._get_slices_at(0)] - self.boundary_values[0]
)
rho_target[self._get_slices_at(-1)] = (
rho_target[self._get_slices_at(-1)] - self.boundary_values[1]
)
[docs]
def correct_boundary_values(self, phi: np.ndarray) -> None:
"""boundary values を補正する。
Parameters
----------
phi : np.ndarray
電位配列です。
Returns
-------
None
戻り値はありません。
"""
phi[self._get_slices_at(0)] = self.boundary_values[0]
phi[self._get_slices_at(-1)] = self.boundary_values[1]
[docs]
class NeumannPoissonBoundary(PoissonBoundary):
"""NeumannPoissonBoundary クラス。
"""
@property
def fft_forward(self) -> Callable[[np.ndarray], np.ndarray]:
"""境界条件に対応した順方向 FFT 関数を返す。
Returns
-------
Callable[[np.ndarray], np.ndarray]
処理結果です。
"""
return partial(scipy.fft.dct, type=1, orthogonalize=False)
@property
def fft_backward(self) -> Callable[[np.ndarray], np.ndarray]:
"""境界条件に対応した逆方向 FFT 関数を返す。
Returns
-------
Callable[[np.ndarray], np.ndarray]
処理結果です。
"""
return partial(scipy.fft.idct, type=1, orthogonalize=False)
[docs]
def get_target_slice(self) -> slice:
"""値を取得する。
Returns
-------
slice
処理結果です。
"""
return slice(None, None)
[docs]
def modified_wave_number(self, k: int, n: int) -> float:
"""境界条件に応じた修正波数を計算する。
Parameters
----------
k : int
波数 index です。
n : int
サンプル数または格子点数です。
Returns
-------
float
処理結果です。
"""
wn = 2.0 * (np.cos(np.pi * (k) / float(n)) - 1.0)
return wn
[docs]
def transpose_boundary_values(self, rho_target: np.ndarray, dx: float) -> None:
"""boundary values を転置処理を行う。
Parameters
----------
rho_target : np.ndarray
解く対象の電荷密度配列です。
dx : float
x 方向の格子間隔です。
Returns
-------
None
戻り値はありません。
"""
rho_target[self._get_slices_at(0)] = (
rho_target[self._get_slices_at(0)] - self.boundary_values[0] * dx
)
rho_target[self._get_slices_at(-1)] = (
rho_target[self._get_slices_at(-1)] + self.boundary_values[1] * dx
)
[docs]
def correct_boundary_values(self, phi: np.ndarray) -> None:
"""boundary values を補正する。
Parameters
----------
phi : np.ndarray
電位配列です。
Returns
-------
None
戻り値はありません。
"""
pass