forked from pressel/pycles
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDiagnosticVariables.pyx
193 lines (155 loc) · 8.26 KB
/
DiagnosticVariables.pyx
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
#!python
#cython: boundscheck=False
#cython: wraparound=False
#cython: initializedcheck=False
#cython: cdivision=True
cimport ParallelMPI
cimport Grid
from NetCDFIO cimport NetCDFIO_Stats
import numpy as np
cimport numpy as np
cimport ParallelMPI
from cpython.mem cimport PyMem_Malloc, PyMem_Realloc, PyMem_Free
cimport mpi4py.libmpi as mpi
cdef extern from "prognostic_variables.h":
void build_buffer(int nv, int dim, int s ,Grid.DimStruct *dims, double* values, double* buffer)
void buffer_to_values(int dim, int s, Grid.DimStruct *dims, double* values, double* buffer)
void set_bcs( int dim, int s, double bc_factor, Grid.DimStruct *dims, double* values)
cdef class DiagnosticVariables:
def __init__(self):
self.name_index = {}
self.index_name = []
self.units = {}
self.nice_name = {}
self.desc = {}
self.nv = 0
self.bc_type = np.array([],dtype=np.double,order='c')
self.name_index_2d = {}
self.units_2d = {}
self.nv_2d = 0
# keep track of which indices are associated with sedimentation velocity
self.sedv_index = np.array([],dtype=np.int,order='c')
self.nsedv = 0
cpdef add_variables(self, name, units, nice_name, desc, bc_type, ParallelMPI.ParallelMPI Pa):
self.name_index[name] = self.nv
self.index_name.append(name)
self.units[name] = units
#Add bc type to array
if bc_type == "sym":
self.bc_type = np.append(self.bc_type,[1.0])
elif bc_type =="asym":
self.bc_type = np.append(self.bc_type,[-1.0])
else:
Pa.root_print("Not a valid bc_type. Killing simulation now!")
Pa.kill()
if name[0:2] == 'w_':
self.sedv_index = np.append(self.sedv_index,self.nv)
self.nsedv += 1
self.nv = len(self.name_index.keys())
return
cpdef add_variables_2d(self, name, units):
self.name_index_2d[name] = self.nv_2d
self.units_2d[name] = units
self.nv_2d = len(self.name_index_2d.keys())
return
cdef void communicate_variable(self, Grid.Grid Gr, ParallelMPI.ParallelMPI PM, long nv):
cdef:
double* send_buffer
double* recv_buffer
long d, s
long var_shift, buffer_var_shift
long [:] shift = np.array([-1,1],dtype=np.int,order='c')
int ierr, source_rank, dest_rank
mpi.MPI_Status status
ierr = mpi.MPI_Comm_rank(PM.cart_comm_world,&source_rank)
var_shift = (nv) * Gr.dims.npg
for d in xrange(Gr.dims.dims):
buffer_var_shift = 0
#Allocate memory for send and recv buffers.
send_buffer = <double*> PyMem_Malloc(Gr.dims.nbuffer[d] * sizeof(double))
recv_buffer = <double*> PyMem_Malloc(Gr.dims.nbuffer[d] * sizeof(double))
for s in shift:
#Since we are only sending one variable at a time the first argument in buld_buffer should be 0
# let's clean this up later
build_buffer(0, d, s,&Gr.dims,&self.values[var_shift],&send_buffer[0])
#Determine the MPI shifts
ierr = mpi.MPI_Cart_shift(PM.cart_comm_world,d,s,&source_rank,&dest_rank)
#Do send and recv given shift
ierr = mpi.MPI_Sendrecv(&send_buffer[0],Gr.dims.nbuffer[d],mpi.MPI_DOUBLE,dest_rank,0,
&recv_buffer[0],Gr.dims.nbuffer[d],
mpi.MPI_DOUBLE,source_rank,0,PM.cart_comm_world,&status)
#If communicated values are to be used copy them into the correct location
if source_rank >= 0:
buffer_to_values(d, s,&Gr.dims,&self.values[var_shift],&recv_buffer[0])
#If communicated values are not be used, set numerical boundary consitions
else:
set_bcs(d,s,self.bc_type[nv],&Gr.dims,&self.values[var_shift])
#Important: Free memory associated with memory buffer to prevent memory leak
PyMem_Free(send_buffer)
PyMem_Free(recv_buffer)
return
cpdef get_variable_array(self,name,Grid.Grid Gr):
index = self.name_index[name]
view = np.array(self.values).view()
view.shape = (self.nv,Gr.dims.nlg[0],Gr.dims.nlg[1],Gr.dims.nlg[2])
return view[index,:,:,:]
cpdef val_nan(self,PA,message):
if np.isnan(self.values).any():
print('Nans found in Diagnostic Variables values')
print(message)
PA.kill()
return
cpdef initialize(self,Grid.Grid Gr, NetCDFIO_Stats NS, ParallelMPI.ParallelMPI Pa):
self.values = np.zeros((self.nv*Gr.dims.npg),dtype=np.double,order='c')
self.values_2d = np.zeros((self.nv_2d*Gr.dims.nlg[0]*Gr.dims.nlg[1]),dtype=np.double,order='c' )
#Add prognostic variables to Statistics IO
Pa.root_print('Setting up statistical output files for Diagnostic Variables')
for var_name in self.name_index.keys():
#Add mean profile
NS.add_profile(var_name+'_mean', Gr, Pa, units=self.units[var_name], nice_name = r'\overline{' + var_name + r'}', desc=var_name + ' hoizontal domain mean')
#Add mean of squares profile
NS.add_profile(var_name+'_mean2',Gr ,Pa, units = '('+self.units[var_name]+')^2',nice_name = r'\overline{' + var_name + r'^2}', desc = var_name + '^2 horizontal domain mean')
#Add mean of cubes profile
NS.add_profile(var_name+'_mean3', Gr, Pa, units = '('+self.units[var_name]+')^3',nice_name = r'\overline{' + var_name + r'^3}', desc = var_name + '^3 horizontal domain mean')
#Add max profile
NS.add_profile(var_name+'_max', Gr, Pa, units=self.units[var_name], nice_name = r'\max(' + var_name + r')', desc = var_name + ' horzontal domain maximum ')
#Add min profile
NS.add_profile(var_name+'_min', Gr, Pa, units=self.units[var_name], nice_name = r'\min(' + var_name + r')', desc = var_name + ' horzontal domain minimum ')
#Add max ts
NS.add_ts(var_name+'_max', Gr, Pa, units=self.units[var_name], nice_name = r'\max(' + var_name + r')', desc = var_name + ' domain maximum ')
#Add min ts
NS.add_ts(var_name+'_min',Gr , Pa, units=self.units[var_name], nice_name = r'\max(' + var_name + r')', desc = var_name + ' domain minimum ')
for var_name in self.name_index_2d.keys():
#Add mean profile
NS.add_ts(var_name+'_mean',Gr,Pa)
return
cpdef stats_io(self, Grid.Grid Gr, NetCDFIO_Stats NS, ParallelMPI.ParallelMPI Pa):
cdef:
int var_shift
double [:] tmp
double tmp2
for var_name in self.name_index.keys():
var_shift = self.get_varshift(Gr,var_name)
#Compute and write mean
tmp = Pa.HorizontalMean(Gr,&self.values[var_shift])
NS.write_profile(var_name + '_mean',tmp[Gr.dims.gw:-Gr.dims.gw],Pa)
#Compute and write mean of squres
tmp = Pa.HorizontalMeanofSquares(Gr,&self.values[var_shift],&self.values[var_shift])
NS.write_profile(var_name + '_mean2',tmp[Gr.dims.gw:-Gr.dims.gw],Pa)
#Compute and write mean of cubes
tmp = Pa.HorizontalMeanofCubes(Gr,&self.values[var_shift],&self.values[var_shift],&self.values[var_shift])
NS.write_profile(var_name + '_mean3',tmp[Gr.dims.gw:-Gr.dims.gw],Pa)
#Compute and write maxes
tmp = Pa.HorizontalMaximum(Gr,&self.values[var_shift])
NS.write_profile(var_name + '_max',tmp[Gr.dims.gw:-Gr.dims.gw],Pa)
NS.write_ts(var_name+'_max',np.amax(tmp[Gr.dims.gw:-Gr.dims.gw]),Pa)
#Compute and write mins
tmp = Pa.HorizontalMinimum(Gr,&self.values[var_shift])
NS.write_profile(var_name + '_min',tmp[Gr.dims.gw:-Gr.dims.gw],Pa)
NS.write_ts(var_name+'_min',np.amin(tmp[Gr.dims.gw:-Gr.dims.gw]),Pa)
for var_name in self.name_index_2d.keys():
var_shift = self.get_varshift_2d(Gr,var_name)
#Compute and write mean
tmp2 = Pa.HorizontalMeanSurface(Gr,&self.values_2d[var_shift])
NS.write_ts(var_name + '_mean',tmp2,Pa)
return