-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathmpfa_subgrid.m
115 lines (101 loc) · 4.45 KB
/
mpfa_subgrid.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
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
function out = mpfa_subgrid(G,rock,mode,varargin)
% Discretize Darcy's law by a multipoint flux approximation.
%
% This wrapper allows for discretizaiton of parts of the domain, as opposed
% to the full discretization computed by mpfa.m
% So far, the only version implemented handles the case where the
% computation in memory bound (see comment in mpfa.m). A similar approach
% could be used if the permeability or geometry has changed in part of the
% domain, however, at the moment this has not been implemented.
%
% Parameters:
% G - mrst grid structure, see http://www.sintef.no/projectweb/mrst/
% To get an overview of the data format, type 'help grid_structure'
% rock - mrst rock structure. Should contain a field perm. To get an
% overview of the data format, type 'help makeRock'
% mode - basically a parameter describing the reason for calling this
% function and not mpfa.m. At the moment, this parameter defaults to 1,
% which is the memory bound case
%
% Optional parametrs, defined as keyword - value pairs:
% 'eta' - Location of continuity point on the half edges, defined
% according to Aavatsmark 2002. Between 0 and 1. Default value is 0, for
% simplices, the value should be 1/3 (will give symmetric method, see
% Klausen, Radu Eigestad 2008).
% bc - boundary conditions, as defined by the MRST function addBC. If none
% are provided, homogeneous Neumann conditions will be assigned.
% invertBlocks - method for inverting block diagonal systems. Should be
% either 'matlab' or 'mex'. The former is pure matlab, which will be
% slow for large problems. Mex is substantially faster, but requires
% that matlab has access to a mex / c compiler.
% maxMemory - maximum memory assigned to storing the inverse gradient
% operator in mpfa. Note that although this gives a substantial part of
% the total memory consumption, in particular for large grids, maxMemory
% does NOT control total memory usage.
%
%{
Copyright 2015-2016, University of Bergen.
This file is part of FVBiot.
FVBiot is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
FVBiot is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this file. If not, see <http://www.gnu.org/licenses/>.
%}
opt = struct('maxMemory',100e6,...
'invertBlocks','mex',...
'eta',0, ...
'bc', []);
opt = merge_options(opt,varargin{:});
% So far, only a wrapper for the memory
mode = 1;
switch mode
case 1
out.F = sparse(G.faces.num,G.cells.num);
out.boundFlux = sparse(G.faces.num,G.faces.num);
% Memory bound
[cno,nno, ~, fno, subfno] = createSubcellMapping(G);
[~, ind] = unique(subfno);
hf2f = sparse(fno(ind),subfno(ind),1);
% Estimated size of each system
blocksz = accumarray(nno(ind),1);
neq = G.griddim * blocksz;
bnds = cumsum(neq.^2);
lower = 1;
mem = opt.maxMemory;
threshold = mem;
iter = 0;
while lower < G.nodes.num
upper = find(bnds < threshold,1,'last');
if upper == lower
error('You need to spend sufficient memory for at least one block')
end
nodes = lower : upper;
d = mpfa(G,rock,nodes,'returnHalfEdgeValues',1,'invertBlocks',opt.invertBlocks,...
'eta', opt.eta, 'bc', opt.bc);
if lower == 1
out.F = hf2f * d.F;
out.boundFlux = hf2f * d.boundFlux;
else
out.F = out.F + hf2f * d.F;
out.boundFlux = out.boundFlux + hf2f * d.boundFlux;
end
lower = upper + 1;
threshold = threshold + mem;
iter = iter + 1;
end
disp(['Finished mpfa reduced memory in ' num2str(iter) ' iterations'])
out.F = out.F;
out.div = scalarDivergence(G);
out.A = out.div * out.F;
out.boundFlux = out.boundFlux;
case 2
error('Partial update of discretization not yet implemented')
otherwise
error(['Unknown mode ' mode])
end