This repository has been archived by the owner on Jun 10, 2020. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathdiscretize_univariate_diffusion.m
64 lines (55 loc) · 3.99 KB
/
discretize_univariate_diffusion.m
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
% Modification of Ben Moll's: http://www.princeton.edu/~moll/HACTproject/option_simple_LCP.m
%For algebra and equation numbers, see the 'operator_discretization_finite_differences.pdf'
%This function takes a grid on [x_min, x_max] and discretizing a general diffusion defined by the following SDE
%d x_t = mu(x_t)dt + sigma(x_t)^2 dW_t
%Subject to reflecting barrier at x_min and x_max
%Pass in the vector of the grid x, and the vectors of mu and sigma_2 at the nodes, and returns a sparse discretized operator.
function [A, Delta_p, Delta_m] = discretize_univariate_diffusion(x, mu, sigma_2, check_absorbing_states)
if nargin < 4
check_absorbing_states = true;
end
I = length(x); %number of grid variables for x
%Check if the grid is uniform
tol = 1E-10; %Tolerance for seeing if the grid is uniform
Delta_p = [diff(x)' (x(I)-x(I-1))]'; %(35) Find distances between grid points.
Delta_m = [x(2)-x(1) diff(x)']'; % %(34) \Delta_{i, -}
if(check_absorbing_states) %In some circumstances, such as in optimal stopping problems, we can ignore these issues.
assert(sigma_2(1) > 0 || mu(1) >= 0, 'Cannot jointly have both sigma = 0 or mu < 0 at x_min, or an absorbing state');
assert(sigma_2(end) > 0 || mu(end) <= 0, 'Cannot jointly have both sigma = 0 or mu > 0 at x_max, or an absorbing state');
end
if(abs(min(Delta_p) - max(Delta_p)) < tol) %i.e. a uniform grid within tolerance
Delta = x(2)-x(1); % (1)
Delta_2 = Delta^2; %Just squaring the Delta for the second order terms in the finite differences.
%% Construct sparse A matrix with uniform grid
mu_m = min(mu,0); %General notation of plus/minus.
mu_p = max(mu,0);
X = - mu_m/Delta + sigma_2/(2*Delta_2); % (7)
Y = - mu_p/Delta + mu_m/Delta - sigma_2/Delta_2; % (8)
Z = mu_p/Delta + sigma_2/(2*Delta_2); %(9)
%Creates a tri-diagonal matrix. See the sparse matrix tricks documented below
A = spdiags([[X(2:I); NaN] Y [NaN; Z(1:I - 1)]], [-1 0 1], I,I);% (10) interior is correct. Corners will require adjustment
%Manually adjust the boundary values at the corners.
A(1,1) = Y(1) + X(1); %Reflecting barrier, (10) and (5)
A(I,I) = Y(I) + Z(I); %Reflecting barrier, (10) and (6)
else
%% Construct sparse A matrix with non-uniform gird
%For non-uniform grid, \Delta_{i, +}=x_{i+1} - x_{i} and \Delta_{i, -}=x_{i} - x_{i-1}
mu_m = min(mu,0); %General notation of plus/minus.
mu_p = max(mu,0);
X = - mu_m./Delta_m + sigma_2 ./(Delta_m.*(Delta_p + Delta_m)); %(31)
Y = - mu_p./Delta_p + mu_m./Delta_m - sigma_2./(Delta_p .* Delta_m); % (32)
Z = mu_p./Delta_p + sigma_2 ./ (Delta_p.*(Delta_p + Delta_m)); % (33)
%Creates a tri-diagonal matrix. See the sparse matrix tricks documented below
A = spdiags([[X(2:I); NaN] Y [NaN; Z(1:I - 1)]], [-1 0 1], I,I);% (36) interior is the same as one for uniform grid case. Corners will require adjustment
%Manually adjust the boundary values at the corners.
A(1,1) = Y(1) + X(1); %Reflecting barrier, top corner of (36)
A(I,I) = Y(I) + Z(I); %Reflecting barrier, bottom corner of (36)
end
end
%Sparse matrix trick: spdiags takes vector(s) and offset(s). It returns the vector(s) in sparse a diagonal matrix where the diagonal is offset by the other argument.
%For example:
% norm(spdiags([1;2;3], 0, 3, 3) - diag([1 2 3]), Inf) % on the true diagonal, offset 0.
% norm(spdiags([2;3;9999], -1, 3, 3)- [0 0 0; 2 0 0; 0 3 0], Inf) %on the diagonal below. Note that the last element is skipped since only 2 points on off diagonal.
% norm(spdiags([9999;2;3], 1, 3, 3)- [0 2 0; 0 0 3; 0 0 0], Inf) %on the diagonal above. Note that the first element is skipped since only 2 points on off diagonal.
%Alternatively this can be done in a single operation to form a tridiagonal matrix by stacking up the arrays, where the 2nd argument is a list of the offsets to apply the columns to)
%Can add them as sparse matrices. For example, the above code is equivalent to %A = spdiags(Y, 0, I, I) + spdiags(X(2:I),-1, I, I) + spdiags([0;Z(1:I-1)], 1, I, I);