Damping and DIIS control.
ebcc.core.damping.BaseDamping(*args, options=None, **kwargs)
Bases: ABC
Base class for damping.
Initialise the damping object.
Source code in ebcc/core/damping.py
def __init__(self, *args: Any, options: Optional[_BaseOptions] = None, **kwargs: Any) -> None:
"""Initialise the damping object."""
self._arrays = {}
self._errors = {}
self._counter = 0
ebcc.core.damping.BaseDamping.__call__(array, error=None)
Apply damping to the array.
Parameters: |
|
---|
Returns: |
|
---|
Source code in ebcc/core/damping.py
def __call__(self, array: NDArray[T], error: Optional[NDArray[T]] = None) -> NDArray[T]:
"""Apply damping to the array.
Args:
array: The array to damp.
error: The error array.
Returns:
The damped array.
"""
self.push(array, error=error)
return self.extrapolate()
ebcc.core.damping.BaseDamping.push(array, error=None)
abstractmethod
Push the array and error into the damping object.
Parameters: |
|
---|
Source code in ebcc/core/damping.py
@abstractmethod
def push(self, array: NDArray[T], error: Optional[NDArray[T]] = None) -> None:
"""Push the array and error into the damping object.
Args:
array: The array to push.
error: The error array to push.
"""
pass
ebcc.core.damping.BaseDamping.extrapolate()
abstractmethod
Extrapolate the next array.
Returns: |
|
---|
Source code in ebcc/core/damping.py
@abstractmethod
def extrapolate(self) -> NDArray[T]:
"""Extrapolate the next array.
Returns:
The extrapolated array.
"""
pass
ebcc.core.damping.BaseDamping.reset()
Reset the damping object.
Source code in ebcc/core/damping.py
def reset(self) -> None:
"""Reset the damping object."""
self._arrays = {}
self._errors = {}
self._counter = 0
ebcc.core.damping.BaseDamping.__len__()
Get the number of arrays stored in the damping object.
Source code in ebcc/core/damping.py
def __len__(self) -> int:
"""Get the number of arrays stored in the damping object."""
return len(self._arrays)
ebcc.core.damping.NoDamping(*args, options=None, **kwargs)
Bases: BaseDamping
No damping.
Apply damping to the array.
Parameters: |
|
---|
Returns: |
|
---|
Source code in ebcc/core/damping.py
def __init__(self, *args: Any, options: Optional[_BaseOptions] = None, **kwargs: Any) -> None:
"""Apply damping to the array.
Args:
array: The array to damp.
error: The error array.
Returns:
The damped array.
"""
pass
ebcc.core.damping.NoDamping.__call__(array, error=None)
Apply damping to the array.
Parameters: |
|
---|
Returns: |
|
---|
Source code in ebcc/core/damping.py
def __call__(self, array: NDArray[T], error: Optional[NDArray[T]] = None) -> NDArray[T]:
"""Apply damping to the array.
Args:
array: The array to damp.
error: The error array.
Returns:
The damped array.
"""
return array
ebcc.core.damping.NoDamping.push(array, error=None)
Push the array and error into the damping object.
Parameters: |
|
---|
Source code in ebcc/core/damping.py
def push(self, array: NDArray[T], error: Optional[NDArray[T]] = None) -> None:
"""Push the array and error into the damping object.
Args:
array: The array to push.
error: The error array to push.
"""
raise NotImplementedError
ebcc.core.damping.NoDamping.extrapolate()
Extrapolate the next array.
Returns: |
|
---|
Source code in ebcc/core/damping.py
def extrapolate(self) -> NDArray[T]:
"""Extrapolate the next array.
Returns:
The extrapolated array.
"""
raise NotImplementedError
ebcc.core.damping.LinearDamping(*args, factor=None, options=None)
Bases: BaseDamping
Linear damping.
Initialise the damping object.
Parameters: |
|
---|
Source code in ebcc/core/damping.py
def __init__(
self, *args: Any, factor: Optional[float] = None, options: Optional[_BaseOptions] = None
) -> None:
"""Initialise the damping object.
Args:
factor: The damping factor. If `None`, use `options.damping`, if available. Otherwise,
use `0.5`.
options: The options object.
"""
super().__init__()
if factor is None:
factor = getattr(options, "damping", 0.5)
self.factor = factor
ebcc.core.damping.LinearDamping.push(array, error=None)
Push the array and error into the damping object.
Parameters: |
|
---|
Source code in ebcc/core/damping.py
def push(self, array: NDArray[T], error: Optional[NDArray[T]] = None) -> None:
"""Push the array and error into the damping object.
Args:
array: The array to push.
error: The error array to push.
"""
if len(self) == 2:
self._arrays.pop(min(self._arrays.keys()))
self._arrays[self._counter] = array
self._counter += 1
ebcc.core.damping.LinearDamping.extrapolate()
Extrapolate the next array.
Returns: |
|
---|
Source code in ebcc/core/damping.py
def extrapolate(self) -> NDArray[T]:
"""Extrapolate the next array.
Returns:
The extrapolated array.
"""
if len(self) < 2:
return next(iter(self._arrays.values()))
(_, previous), (_, current) = sorted(self._arrays.items())
return current * (1.0 - self.factor) + previous * self.factor
ebcc.core.damping.DIIS(space=None, min_space=None, factor=None, options=None)
Bases: BaseDamping
Direct inversion in the iterative subspace.
Initialize the DIIS object.
Parameters: |
|
---|
Source code in ebcc/core/damping.py
def __init__(
self,
space: Optional[int] = None,
min_space: Optional[int] = None,
factor: Optional[float] = None,
options: Optional[_BaseOptions] = None,
) -> None:
"""Initialize the DIIS object.
Args:
space: The number of vectors to store in the DIIS space. If `None`, use
`options.diis_space`, if available. Otherwise, use `6`.
min_space: The minimum number of vectors to store in the DIIS space. If `None`, use
`options.diis_min_space`, if available. Otherwise, use `1`.
factor: The damping factor. If `None`, use `options.damping`, if available. Otherwise,
use `0.5`.
options: The options object.
"""
super().__init__()
if space is None:
space = getattr(options, "diis_space", 6)
if min_space is None:
min_space = getattr(options, "diis_min_space", 1)
if factor is None:
factor = getattr(options, "damping", 0.5)
self.space = space
self.min_space = min_space
self.factor = factor
self._norm_cache: dict[tuple[int, int], T] = {}
ebcc.core.damping.DIIS.push(array, error=None)
Push the array and error into the damping object.
Parameters: |
|
---|
Source code in ebcc/core/damping.py
def push(self, array: NDArray[T], error: Optional[NDArray[T]] = None) -> None:
"""Push the array and error into the damping object.
Args:
array: The array to push.
error: The error array to push.
"""
# Get the error if not provided
if error is None and -1 in self._arrays:
error = array - self._arrays[-1]
elif error is None or len(self) == 0:
self._arrays[-1] = array
return
# Push the array and error into the DIIS subspace
self._arrays[self._counter] = array
self._errors[self._counter] = error
self._counter += 1
# Remove an array if the space is exceeded
if len(self) > self.space:
errors = {
counter: self._error_norm(counter, counter) for counter in self._errors.keys()
}
counter = max(errors, key=errors.__getitem__) # type: ignore[arg-type]
self._arrays.pop(counter)
self._errors.pop(counter)
ebcc.core.damping.DIIS.extrapolate()
Extrapolate the next array.
Returns: |
|
---|
Source code in ebcc/core/damping.py
def extrapolate(self) -> NDArray[T]:
"""Extrapolate the next array.
Returns:
The extrapolated array.
"""
# Return the last array if the space is less than the minimum space
counters = sorted(self._errors.keys())
size = len(counters)
if size < self.min_space:
return self._arrays[-1]
# Get the backend
if USE_BACKEND:
backend = np
else:
backend = numpy
# Build the error matrix
errors = backend.array(
[
[self._error_norm(counter_i, counter_j) for counter_j in counters]
for counter_i in counters
]
)
zeros = backend.zeros((1, 1), dtype=errors.dtype)
ones = backend.ones((size, 1), dtype=errors.dtype)
matrix = backend.block([[errors, -ones], [-backend.transpose(ones), zeros]])
# Build the right-hand side
zeros = backend.zeros((size,), dtype=errors.dtype)
ones = backend.ones((1,), dtype=errors.dtype)
residual = backend.block([zeros, -ones])
# Solve the linear problem
try:
c = backend.linalg.solve(matrix, residual)
except Exception:
w, v = backend.linalg.eigh(matrix)
if backend.any(backend.abs(w) < 1e-14):
# avoiding fancy indexing for compatibility
for i in range(size + 1):
if backend.abs(w[i]) < 1e-14:
_put(
v,
backend.ix_(backend.arange(size + 1), backend.array([i])),
backend.zeros_like(v[:, i]),
)
c = util.einsum("pi,qi,i,q->p", v, np.conj(v), w**-1.0, residual)
# Construct the new array
array = np.zeros_like(self._arrays[next(iter(self._arrays))])
for counter, coefficient in zip(counters, c):
array += self._arrays[counter] * coefficient
error = np.zeros_like(self._errors[next(iter(self._errors))])
for counter, coefficient in zip(counters, c):
error += self._errors[counter] * coefficient
# Apply the damping factor
if self.factor:
array = array * (1.0 - self.factor) + self._arrays[-1] * self.factor
# Replace the previous array with the extrapolated array
self._arrays[-1] = array
self._arrays[self._counter] = array
self._errors[self._counter] = error
return array
ebcc.core.damping.DIIS.reset()
Reset the damping object.
Source code in ebcc/core/damping.py
def reset(self) -> None:
"""Reset the damping object."""
super().reset()
self._norm_cache = {}