MADNESS
version 0.9
|
The source is here.
We wish to solve Poisson's equation within a non-homogeneous medium, i.e., in which the permittivity is not constant.
Expanding and rearranging yields,
We can interpret as the effective free charge density and as the induced surface charge density. Assuming free-space boundary conditions at long range we can invert the Laplacian by convolution with the free-space Green's function that is . Note that MADNESS provides convolution with so you have to keep track of the yourself.
Thus, our equation becomes (deliberately written in the form of a fixed point iteration — given a guess for we can compute the r.h.s. and directly obtain a new value for that hopefully is closer to the solution)
where
and
Let's solve a problem to which we know the exact answer — a point charge at the center of a sphere . Inside the sphere the permittivity is and outside it is . The exact solution is
The surface charge density integrated over the suface has the value
To implement the problem in MADNESS we want the permittivity defined as a function over all space, so we define a characteristic function (or mask) that is defined to be one inside the sphere and zero outside
where is the Heaviside step function. Hence, we have
To smooth the discontinuity we replace the Heaviside step function with
where is the effective width of the step (0.2 in the code). Similarly, the point charge is replaced by a suitably normalized Gaussian with a an exponent sufficiently large as to appear as a point charge (delta function) on the scale of interest (we pick )
Starting from an initial guess we could try to iterate our equation for but, sadly, this does not converge reliably. The simplest approach is to introduce some damping or step restriction. With indicating the iteration number, we write
with . This works (for sufficiently small ) and is implemented in the code.
A more robust approach is to use a solver that exploits information from previous iterations (the Krylov subspace) to estimate the optimal direction and length of the step to take. Such a solver is provided by nonlinsol.h. Each iteration we pass the current solution and corresponding residual to the solver and it provides the next trial vector.
In the code you can switch between using the solver or simple iteration by changing the value of USE_SOLVER
.
One final point is how to compute the reciprocal of the permittivity. This operation is not provided explicity by MADNESS, in part because even functions that are analytically never zero, might be (nearly) zero due to numerical truncation. Handling this issue correctly is problem specific. More generally, we want to compute a function-of-a-function. If is a MADNESS function that takes a d-dimensional vector as its argument (in this examaple ) and is computable function that takes a scalar argument, we want to compute the new MADNESS function
There are various ways to do this. The simplest is employed here — we define a function (reciprocal()
) that is passed into the unaryop()
method that modifies the function in-place. This simple approach is justified here since we know our input function is never zero even due to numerical noise, and the output function is about as smooth as the input. If it were not, we might have to refine the input function to obtain the desired precision.