13#include <camp/aero_phase_solver.h>
14#include <camp/aero_reps.h>
15#include <camp/camp_solver.h>
18#define TEMPERATURE_K_ env_data[0]
19#define PRESSURE_PA_ env_data[1]
21#define UPDATE_NUMBER 0
23#define NUM_LAYERS_ int_data[0]
24#define AERO_REP_ID_ int_data[1]
25#define MAX_PARTICLES_ int_data[2]
26#define PARTICLE_STATE_SIZE_ int_data[3]
27#define NUMBER_CONC_(x) aero_rep_env_data[x]
28#define NUM_INT_PROP_ 4
29#define NUM_FLOAT_PROP_ 0
30#define NUM_ENV_PARAM_ MAX_PARTICLES_
31#define LAYER_PHASE_START_(l) (int_data[NUM_INT_PROP_+l]-1)
32#define LAYER_PHASE_END_(l) (int_data[NUM_INT_PROP_+NUM_LAYERS_+l]-1)
33#define TOTAL_NUM_PHASES_ (LAYER_PHASE_END_(NUM_LAYERS_-1)-LAYER_PHASE_START_(0)+1)
34#define NUM_PHASES_(l) (LAYER_PHASE_END_(l)-LAYER_PHASE_START_(l)+1)
35#define PHASE_STATE_ID_(l,p) (int_data[NUM_INT_PROP_+2*NUM_LAYERS_+LAYER_PHASE_START_(l)+p]-1)
36#define PHASE_MODEL_DATA_ID_(l,p) (int_data[NUM_INT_PROP_+2*NUM_LAYERS_+TOTAL_NUM_PHASES_+LAYER_PHASE_START_(l)+p]-1)
37#define PHASE_NUM_JAC_ELEM_(l,p) int_data[NUM_INT_PROP_+2*NUM_LAYERS_+2*TOTAL_NUM_PHASES_+LAYER_PHASE_START_(l)+p]
54 int *aero_rep_int_data,
55 double *aero_rep_float_data,
57 int *int_data = aero_rep_int_data;
58 double *float_data = aero_rep_float_data;
64 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
65 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
87 double *aero_rep_float_data,
89 int *int_data = aero_rep_int_data;
90 double *float_data = aero_rep_float_data;
109 int *aero_rep_int_data,
110 double *aero_rep_float_data,
111 double *aero_rep_env_data) {
112 int *int_data = aero_rep_int_data;
113 double *float_data = aero_rep_float_data;
114 double *env_data = model_data->grid_cell_env;
130 int *aero_rep_int_data,
131 double *aero_rep_float_data,
132 double *aero_rep_env_data) {
133 int *int_data = aero_rep_int_data;
134 double *float_data = aero_rep_float_data;
154 ModelData *model_data,
int aero_phase_idx,
double *layer_radius,
155 double *partial_deriv,
int *aero_rep_int_data,
double *aero_rep_float_data,
156 double *aero_rep_env_data) {
158 int *int_data = aero_rep_int_data;
159 double *float_data = aero_rep_float_data;
161 double *curr_partial = NULL;
162 int aero_phase_idx_temp = aero_phase_idx;
165 int i_layer_radius = -1;
166 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
169 i_layer_radius = i_layer;
175 if (partial_deriv) curr_partial = partial_deriv;
176 for (
int i_layer = 0; i_layer <= i_layer_radius; ++i_layer) {
177 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
178 double *state = (
double *)(model_data->grid_cell_state);
182 state, &(volume), curr_partial);
184 *layer_radius += volume;
187 *layer_radius = pow(((*layer_radius) * 3.0 / 4.0 / 3.14159265359), 1.0 / 3.0);
188 if (!partial_deriv)
return;
189 for (
int i_layer = 0; i_layer <= i_layer_radius; ++i_layer) {
190 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
193 1.0 / 4.0 / 3.14159265359 * pow(*layer_radius, -2.0) * (*partial_deriv);
217 ModelData *model_data,
int aero_phase_idx,
double *radius,
218 double *partial_deriv,
int *aero_rep_int_data,
double *aero_rep_float_data,
219 double *aero_rep_env_data) {
221 int *int_data = aero_rep_int_data;
222 double *float_data = aero_rep_float_data;
223 double *curr_partial = NULL;
228 aero_phase_idx += offset;
256 ModelData *model_data,
int aero_phase_idx,
double *phase_volume,
257 double *partial_deriv,
int *aero_rep_int_data,
double *aero_rep_float_data,
258 double *aero_rep_env_data) {
260 int *int_data = aero_rep_int_data;
261 double *float_data = aero_rep_float_data;
263 double *curr_partial = NULL;
264 int aero_phase_idx_temp = aero_phase_idx;
267 int i_layer_phase = -1;
268 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
271 i_layer_phase = i_layer;
276 if (i_layer_phase == -1) {
277 printf(
"\n\nERROR: aero_rep_single_particle_get_phase_volume__m3_m3: Could not determine i_layer_phase ");
278 printf(
"for aero_phase_idx=%d.\n\n", aero_phase_idx);
282 int total_phases_previous_layers = 0;
283 for (
int i_layer = 0; i_layer < i_layer_phase; ++i_layer) {
284 total_phases_previous_layers +=
NUM_PHASES_(i_layer);
286 int i_phase = aero_phase_idx_temp - total_phases_previous_layers;
288 if (partial_deriv) curr_partial = partial_deriv;
289 double *state = (
double *)(model_data->grid_cell_state);
292 state, phase_volume, curr_partial);
319 ModelData *model_data,
int aero_phase_idx_first,
int aero_phase_idx_second,
320 double *surface_area,
double *partial_deriv,
321 int *aero_rep_int_data,
double *aero_rep_float_data,
double *aero_rep_env_data) {
323 int *int_data = aero_rep_int_data;
324 double *float_data = aero_rep_float_data;
325 double *curr_partial = NULL;
326 int layer_first = -1;
327 int layer_second = -1;
328 int layer_interface = -1;
329 int phase_model_data_id_first = -1;
330 int phase_model_data_id_second = -1;
337 int i_phase_count = 0;
338 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
339 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
342 i_phase_count == aero_phase_idx_first) {
343 layer_first = i_layer;
347 i_phase_count == aero_phase_idx_second) {
348 layer_second = i_layer;
355 layer_interface = layer_first > layer_second ? layer_second : layer_first;
361 double interface_volume = 0.0;
362 double total_volume_layer_first = 0.0;
363 double total_volume_layer_second = 0.0;
364 double volume_phase_first = 0.0;
365 double volume_phase_second = 0.0;
367 if (partial_deriv) curr_partial = partial_deriv;
368 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
369 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
370 double *state = (
double *)(model_data->grid_cell_state);
374 state, &(volume), curr_partial);
375 if (i_layer == layer_first) total_volume_layer_first += volume;
376 if (i_phase_count == aero_phase_idx_first &&
378 phase_model_data_id_first) volume_phase_first = volume;
379 if (i_layer == layer_second) total_volume_layer_second += volume;
380 if (i_phase_count == aero_phase_idx_second &&
382 phase_model_data_id_second) volume_phase_second = volume;
383 if (i_layer <= layer_interface) interface_volume += volume;
393 double f_first = volume_phase_first / total_volume_layer_first;
394 double f_second = volume_phase_second / total_volume_layer_second;
395 radius = pow(((interface_volume) * 3.0 / 4.0 / M_PI), 1.0 / 3.0);
396 *surface_area = f_first * f_second * 4 * M_PI * pow(radius, 2.0);
400 if (!partial_deriv)
return;
402 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
403 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
404 double *state = (
double *)(model_data->grid_cell_state);
408 state, &(volume_phase), NULL);
411 if (i_layer == layer_first && i_phase_count == aero_phase_idx_first) {
413 (((total_volume_layer_first - volume_phase_first) *
414 pow(total_volume_layer_first, -2.0) * f_second * (*surface_area)) +
415 2.0 * f_first * f_second * pow(radius, -1.0)) * (*partial_deriv);
419 else if (i_layer == layer_first && i_phase_count != aero_phase_idx_first) {
421 (((-1 * volume_phase) * pow(total_volume_layer_first, -2.0) *
422 f_second * (*surface_area)) + 2.0 * f_first * f_second *
423 pow(radius, -1.0)) * (*partial_deriv);
427 else if (i_layer == layer_second && i_phase_count == aero_phase_idx_second) {
429 (((total_volume_layer_second - volume_phase_second) *
430 pow(total_volume_layer_second, -2.0) * f_first * (*surface_area)) +
431 2.0 * f_first * f_second * pow(radius, -1.0)) * (*partial_deriv);
435 else if (i_layer == layer_second && i_phase_count != aero_phase_idx_second) {
437 (((-1 * volume_phase) * pow(total_volume_layer_second, -2.0) *
438 f_first * (*surface_area)) + 2.0 * f_first * f_second *
439 pow(radius, -1.0)) * (*partial_deriv);
443 else if (i_layer != layer_first && i_layer != layer_second) {
444 *(partial_deriv++) = ZERO;
447 printf(
"\n\nERROR No conditions met for surface area partial derivative.\n\n");
472 ModelData *model_data,
int aero_phase_idx,
double *layer_thickness,
473 double *partial_deriv,
int *aero_rep_int_data,
double *aero_rep_float_data,
474 double *aero_rep_env_data) {
476 int *int_data = aero_rep_int_data;
477 double *float_data = aero_rep_float_data;
479 double radius_inner, radius_outer;
481 int aero_phase_idx_temp = aero_phase_idx;
485 double *jac_inner = NULL;
486 double *jac_outer = NULL;
489 jac_inner = (
double *)calloc(jac_size,
sizeof(
double));
490 jac_outer = (
double *)calloc(jac_size,
sizeof(
double));
493 int i_layer_inner = -1;
494 int i_layer_outer = -1;
495 int i_phase_inner = -1;
496 int i_phase_outer = -1;
497 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
500 i_layer_outer = i_layer;
501 if (i_layer - 1 >= 0 ) {
502 i_layer_inner = i_layer - 1;
503 }
else if (i_layer - 1 < 0 ) {
504 i_layer_inner = i_layer;
509 int aero_phase_idx_inner = -1;
510 if (i_layer_inner == i_layer_outer) {
511 aero_phase_idx_inner = aero_phase_idx;
513 aero_phase_idx_inner = aero_phase_idx - (offset+1);
527 aero_phase_idx_inner,
534 if (i_layer_inner == i_layer_outer) {
535 *layer_thickness = radius_inner;
537 *layer_thickness = radius_outer - radius_inner;
541 for (
int i = 0; i < jac_size; ++i) {
542 partial_deriv[i] = jac_outer[i] - jac_inner[i];
574 ModelData *model_data,
int aero_phase_idx,
double *number_conc,
575 double *partial_deriv,
int *aero_rep_int_data,
double *aero_rep_float_data,
576 double *aero_rep_env_data) {
579 int *int_data = aero_rep_int_data;
580 double *float_data = aero_rep_float_data;
586 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
587 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
589 *(partial_deriv++) = ZERO;
613 int *aero_rep_int_data,
614 double *aero_rep_float_data,
615 double *aero_rep_env_data) {
616 int *int_data = aero_rep_int_data;
617 double *float_data = aero_rep_float_data;
644 ModelData *model_data,
int aero_phase_idx,
double *aero_phase_mass,
645 double *partial_deriv,
int *aero_rep_int_data,
double *aero_rep_float_data,
646 double *aero_rep_env_data) {
648 int *int_data = aero_rep_int_data;
649 double *float_data = aero_rep_float_data;
653 int i_total_phase = 0;
654 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
655 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
656 if ( i_total_phase == aero_phase_idx ) {
657 double *state = (
double *)(model_data->grid_cell_state);
661 state, aero_phase_mass, &mw, partial_deriv, NULL);
663 }
else if (partial_deriv) {
665 *(partial_deriv++) = ZERO;
693 ModelData *model_data,
int aero_phase_idx,
double *aero_phase_avg_MW,
694 double *partial_deriv,
int *aero_rep_int_data,
double *aero_rep_float_data,
695 double *aero_rep_env_data) {
697 int *int_data = aero_rep_int_data;
698 double *float_data = aero_rep_float_data;
702 int i_total_phase = 0;
703 for (
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer) {
704 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
705 if ( i_total_phase == aero_phase_idx ) {
706 double *state = (
double *)(model_data->grid_cell_state);
710 state, &mass, aero_phase_avg_MW, NULL, partial_deriv);
712 }
else if (partial_deriv) {
714 *(partial_deriv++) = ZERO;
745 int *aero_rep_int_data,
746 double *aero_rep_float_data,
747 double *aero_rep_env_data) {
748 int *int_data = aero_rep_int_data;
749 double *float_data = aero_rep_float_data;
751 int *aero_rep_id = (
int *)update_data;
752 int *update_type = (
int *)&(aero_rep_id[1]);
753 int *particle_id = (
int *)&(update_type[1]);
754 double *new_value = (
double *)&(update_type[2]);
776 double *aero_rep_float_data) {
777 int *int_data = aero_rep_int_data;
778 double *float_data = aero_rep_float_data;
780 printf(
"\n\nSingle particle aerosol representation\n");
782 printf(
"\nAerosol representation id: %d",
AERO_REP_ID_);
785 for(
int i_layer = 0; i_layer <
NUM_LAYERS_; ++i_layer){
786 printf(
"\nLayer: %d", i_layer);
788 printf(
"\n Number of phases: %d",
NUM_PHASES_(i_layer));
789 printf(
"\n\n - Phases -");
790 for (
int i_phase = 0; i_phase <
NUM_PHASES_(i_layer); ++i_phase) {
791 printf(
"\n state id: %d model data id: %d num Jac elements: %d",
796 printf(
"\n\nEnd single particle aerosol representation\n");
806 int *update_data = (
int *)malloc(3 *
sizeof(
int) +
sizeof(double));
807 if (update_data == NULL) {
808 printf(
"\n\nERROR allocating space for number update data\n\n");
811 return (
void *)update_data;
824 double number_conc) {
825 int *new_aero_rep_id = (
int *)update_data;
826 int *update_type = (
int *)&(new_aero_rep_id[1]);
827 int *new_particle_id = (
int *)&(update_type[1]);
828 double *new_number_conc = (
double *)&(update_type[2]);
829 *new_aero_rep_id = aero_rep_id;
831 *new_particle_id = particle_id;
832 *new_number_conc = number_conc;
void aero_phase_get_volume__m3_m3(ModelData *model_data, int aero_phase_idx, double *state_var, double *volume, double *jac_elem)
Get the volume of an aerosol phase.
int aero_phase_get_used_jac_elem(ModelData *model_data, int aero_phase_idx, int state_var_id, bool *jac_struct)
Flag Jacobian elements used in calculations of mass and volume.
void aero_phase_get_mass__kg_m3(ModelData *model_data, int aero_phase_idx, double *state_var, double *mass, double *MW, double *jac_elem_mass, double *jac_elem_MW)
Get the mass and average MW in an aerosol phase.
int aero_rep_single_particle_get_used_jac_elem(ModelData *model_data, int aero_phase_idx, int *aero_rep_int_data, double *aero_rep_float_data, bool *jac_struct)
Flag Jacobian elements used in calcualtions of mass and volume.
void aero_rep_single_particle_get_aero_phase_avg_MW__kg_mol(ModelData *model_data, int aero_phase_idx, double *aero_phase_avg_MW, double *partial_deriv, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Get the average molecular weight in an aerosol phase ( )
void aero_rep_single_particle_get_effective_radius__m(ModelData *model_data, int aero_phase_idx, double *radius, double *partial_deriv, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Get the effective particle radius (m) Finds the radius of the largest layer in specified particle.
#define PARTICLE_STATE_SIZE_
void aero_rep_single_particle_get_aero_conc_type(int aero_phase_idx, int *aero_conc_type, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Get the type of aerosol concentration used.
void * aero_rep_single_particle_create_number_update_data()
Create update data for new particle number.
#define PHASE_NUM_JAC_ELEM_(l, p)
void aero_rep_single_particle_get_interface_surface_area__m2(ModelData *model_data, int aero_phase_idx_first, int aero_phase_idx_second, double *surface_area, double *partial_deriv, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Get the surface area of specified particle layer (m)
void aero_rep_single_particle_print(int *aero_rep_int_data, double *aero_rep_float_data)
Print the Single Particle reaction parameters.
#define LAYER_PHASE_START_(l)
#define PHASE_MODEL_DATA_ID_(l, p)
#define TOTAL_NUM_PHASES_
#define PHASE_STATE_ID_(l, p)
void aero_rep_single_particle_get_phase_volume__m3_m3(ModelData *model_data, int aero_phase_idx, double *phase_volume, double *partial_deriv, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Get the volume of a specified phase in the corresponding layer.
void aero_rep_single_particle_update_env_state(ModelData *model_data, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Update aerosol representation data for new environmental conditions.
void aero_rep_single_particle_get_layer_thickness__m(ModelData *model_data, int aero_phase_idx, double *layer_thickness, double *partial_deriv, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Get the thickness of a particle layer (m)
bool aero_rep_single_particle_update_data(void *update_data, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Update aerosol representation data.
void aero_rep_single_particle_get_dependencies(int *aero_rep_int_data, double *aero_rep_float_data, bool *state_flags)
Flag elements on the state array used by this aerosol representation.
void aero_rep_single_particle_get_aero_phase_mass__kg_m3(ModelData *model_data, int aero_phase_idx, double *aero_phase_mass, double *partial_deriv, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Get the total mass in an aerosol phase ( )
#define LAYER_PHASE_END_(l)
void aero_rep_single_particle_get_effective_layer_radius__m(ModelData *model_data, int aero_phase_idx, double *layer_radius, double *partial_deriv, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Get the effective radius of a specified layer (m)
void aero_rep_single_particle_get_number_conc__n_m3(ModelData *model_data, int aero_phase_idx, double *number_conc, double *partial_deriv, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Get the particle number concentration ( )
void aero_rep_single_particle_set_number_update_data__n_m3(void *update_data, int aero_rep_id, int particle_id, double number_conc)
Set number update data (#/m3)
void aero_rep_single_particle_update_state(ModelData *model_data, int *aero_rep_int_data, double *aero_rep_float_data, double *aero_rep_env_data)
Update aerosol representation data for a new state.