-
Notifications
You must be signed in to change notification settings - Fork 11
/
fpsolve.py
222 lines (167 loc) · 7.36 KB
/
fpsolve.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
"""
Package to solve Fokker-Planck equations
* Steady-state Fourier-space solver for the PDF
* Adjoint finite difference solver for first/second moments
(Only 1D and 2D implemented so far)
Jared Callaham (2020)
"""
import numpy as np
from numpy.fft import fft, fftn, fftfreq, ifftn
from scipy import linalg, sparse
class SteadyFP:
"""
Solver object for steady-state Fokker-Planck equation
Initializing this independently avoids having to re-initialize all of the indexing arrays
for repeated loops with different drift and diffusion
Jared Callaham (2020)
"""
def __init__(self, N, dx):
"""
ndim - number of dimensions
N - array of ndim ints: grid resolution N[0] x N[1] x ... x N[ndim-1]
dx - grid spacing (array of floats)
"""
if isinstance(N, int):
self.ndim = 1
else:
self.ndim = len(N)
self.N = N
self.dx = dx
# Set up indexing matrices for ndim=1, 2
if self.ndim == 1:
self.k = 2*np.pi*fftfreq(N, dx)
self.idx = np.zeros((self.N, self.N), dtype=np.int32)
for i in range(self.N):
self.idx[i, :] = i-np.arange(N)
elif self.ndim == 2:
# Fourier frequencies
self.k = [2*np.pi*fftfreq(N[i], dx[i]) for i in range(self.ndim)]
self.idx = np.zeros((2, self.N[0], self.N[1], self.N[0], self.N[1]), dtype=np.int32)
for m in range(N[0]):
for n in range(N[1]):
self.idx[0, m, n, :, :] = m-np.tile(np.arange(N[0]), [N[1], 1]).T
self.idx[1, m, n, :, :] = n-np.tile(np.arange(N[1]), [N[0], 1])
else:
print("WARNING: NOT IMPLEMENTED FOR HIGHER DIMENSIONS")
self.A = None # Need to initialize with precompute_operator
def precompute_operator(self, f, a):
"""
f - array of drift coefficients on domain (ndim x N[0] x N[1] x ... x N[ndim])
a - array of diffusion coefficients on domain (ndim x N[0] x N[1] x ... x N[ndim])
NOTE: To generalize to covariate noise, would need to add a dimension to a
"""
if self.ndim == 1:
f_hat = self.dx*fftn(f)
a_hat = self.dx*fftn(a)
# Set up spectral projection operator
self.A = np.einsum('i,ij->ij', -1j*self.k, f_hat[self.idx]) \
+ np.einsum('i,ij->ij', -self.k**2, a_hat[self.idx])
if self.ndim == 2:
# Initialize Fourier transformed coefficients
f_hat = np.zeros(np.append([self.ndim], self.N), dtype=np.complex64)
a_hat = np.zeros(f_hat.shape, dtype=np.complex64)
for i in range(self.ndim):
f_hat[i] = np.prod(self.dx)*fftn(f[i])
a_hat[i] = np.prod(self.dx)*fftn(a[i])
self.A = -1j*np.einsum('i,ijkl->ijkl', self.k[0], f_hat[0, self.idx[0], self.idx[1]]) \
-1j*np.einsum('j,ijkl->ijkl', self.k[1], f_hat[1, self.idx[0], self.idx[1]]) \
-np.einsum('i,ijkl->ijkl', self.k[0]**2, a_hat[0, self.idx[0], self.idx[1]]) \
-np.einsum('j,ijkl->ijkl', self.k[1]**2, a_hat[1, self.idx[0], self.idx[1]])
self.A = np.reshape(self.A, (np.prod(self.N), np.prod(self.N)))
def solve(self, f, a):
"""
Solve Fokker-Planck equation from input drift coefficients
"""
self.precompute_operator(f, a)
q_hat = np.linalg.lstsq(self.A[1:, 1:], -self.A[1:, 0], rcond=1e-6)[0]
q_hat = np.append([1], q_hat)
return np.real(ifftn( np.reshape(q_hat, self.N) ))/np.prod(self.dx)
class AdjFP:
"""
Solver object for adjoint Fokker-Planck equation
Jared Callaham (2020)
"""
# 1D derivative operators
@staticmethod
def derivs1d(x):
N = len(x)
dx = x[1]-x[0]
one = np.ones((N))
# First derivative
Dx = sparse.diags([one, -one], [1, -1], shape=(N, N))
Dx = sparse.lil_matrix(Dx)
# Forward/backwards difference at boundaries
Dx[0, :3] = [-3, 4, -1]
Dx[-1, -3:] = [1, -4, 3]
Dx = sparse.csr_matrix(Dx)/(2*dx)
# Second derivative
Dxx = sparse.diags([one, -2*one, one], [1, 0, -1], shape=(N, N))
Dxx = sparse.lil_matrix(Dxx)
# Forwards/backwards differences (second-order accurate)
Dxx[-1, -4:] = [1.25, -2.75, 1.75, -.25]
Dxx[0, :4] = [-.25, 1.75, -2.75, 1.25]
Dxx = sparse.csr_matrix(Dxx)/(dx**2)
return Dx, Dxx
@staticmethod
def derivs2d(x, y):
hx, hy = x[1]-x[0], y[1]-y[0]
Nx, Ny = len(x), len(y)
Dy = sparse.diags( [-1, 1], [-1, 1], shape=(Ny, Ny) ).toarray()
# Second-order forward/backwards at boundaries
Dy[0, :3] = np.array([-3, 4, -1])
Dy[-1, -3:] = np.array([1, -4, 3])
# Repeat for each x-location
Dy = linalg.block_diag(*Dy.reshape(1, Ny, Ny).repeat(Nx,axis=0))/(2*hy)
Dy = sparse.csr_matrix(Dy)
Dx = sparse.diags( [-1, 1], [-Ny, Ny], shape=(Nx*Ny, Nx*Ny)).toarray()
# Second-order forwards/backwards at boundaries
for i in range(Ny):
Dx[i, i] = -3
Dx[i, Ny+i] = 4
Dx[i, 2*Ny+i] = -1
Dx[-(i+1), -(i+1)] = 3
Dx[-(i+1), -(Ny+i+1)] = -4
Dx[-(i+1), -(2*Ny+i+1)] = 1
Dx = sparse.csr_matrix(Dx)/(2*hx)
Dxx = sparse.csr_matrix(Dx @ Dx)
Dyy = sparse.csr_matrix(Dy @ Dy)
return Dx, Dy, Dxx, Dyy
def __init__(self, x, ndim=1):
"""
x - uniform grid (array of floats)
"""
self.ndim = ndim
if self.ndim == 1:
self.N = [len(x)]
self.dx = [x[1]-x[0]]
self.x = [x]
self.Dx, self.Dxx = AdjFP.derivs1d(x)
self.precompute_operator = self.operator1d
else:
self.x = x
self.N = [len(x[i]) for i in range(len(x))]
self.dx = [x[i][1]-x[i][0] for i in range(len(x))]
self.Dx, self.Dy, self.Dxx, self.Dyy = AdjFP.derivs2d(*x)
self.precompute_operator = self.operator2d
self.XX = np.meshgrid(*self.x, indexing='ij')
self.precompute_moments()
def precompute_moments(self):
self.m1 = np.zeros([self.ndim, np.prod(self.N), np.prod(self.N)])
self.m2 = np.zeros([self.ndim, np.prod(self.N), np.prod(self.N)])
for d in range(self.ndim):
for i in range(np.prod(self.N)):
self.m1[d, i, :] = self.XX[d].flatten() - self.XX[d].flatten()[i]
self.m2[d, i, :] = (self.XX[d].flatten() - self.XX[d].flatten()[i])**2
def operator1d(self, f, a):
self.L = sparse.diags(f) @ self.Dx + sparse.diags(a) @ self.Dxx
def operator2d(self, f, a):
self.L = sparse.diags(f[0]) @ self.Dx + sparse.diags(f[1]) @ self.Dy + \
sparse.diags(a[0]) @ self.Dxx + sparse.diags(a[1]) @ self.Dyy
def solve(self, tau, d=0):
if self.L is None:
print("Need to initialize operator")
return None
L_tau = linalg.expm(self.L.todense()*tau)
f_tau = np.einsum('ij,ij->i', L_tau, self.m1[d])/tau
a_tau = np.einsum('ij,ij->i', L_tau, self.m2[d])/(2*tau)
return f_tau, a_tau