/*******************************************************************************
* This file is part of SWIFT.
* Copyright (c) 2016 Tom Theuns (tom.theuns@durham.ac.uk)
* Matthieu Schaller (schaller@strw.leidenuniv.nl)
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published
* by the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 Lesser General Public License
* along with this program. If not, see .
*
******************************************************************************/
/* Config parameters. */
#include
/* This object's header. */
#include "physical_constants.h"
/* Local headers. */
#include "common_io.h"
#include "error.h"
#include "physical_constants_cgs.h"
#include "restart.h"
/**
* @brief Converts physical constants to the internal unit system
*
* Some constants can be overwritten by the YAML file values. If the
* param argument is NULL, no overwriting is done.
*
* @param us The current internal system of units.
* @param params The parsed parameter file.
* @param internal_const The physical constants to initialize.
*/
void phys_const_init(const struct unit_system *us, struct swift_params *params,
struct phys_const *internal_const) {
/* Units are declared as {U_M, U_L, U_t, U_I, U_T} */
const float dimension_G[5] = {-1, 3, -2, 0, 0}; /* [g^-1 cm^3 s^-2] */
internal_const->const_newton_G =
const_newton_G_cgs / units_general_cgs_conversion_factor(us, dimension_G);
/* Overwrite G if present in the file */
if (params != NULL) {
internal_const->const_newton_G = parser_get_opt_param_double(
params, "PhysicalConstants:G", internal_const->const_newton_G);
}
const float dimension_c[5] = {0, 1, -1, 0, 0}; /* [cm s^-1] */
internal_const->const_speed_light_c =
const_speed_light_c_cgs /
units_general_cgs_conversion_factor(us, dimension_c);
const float dimension_h[5] = {1, 2, -1, 0, 0}; /* [g cm^2 s^-1] */
internal_const->const_planck_h =
const_planck_h_cgs / units_general_cgs_conversion_factor(us, dimension_h);
internal_const->const_planck_hbar =
const_planck_hbar_cgs /
units_general_cgs_conversion_factor(us, dimension_h);
const float dimension_k[5] = {1, 2, -2, 0, -1}; /* [g cm^2 s^-2 K^-1] */
internal_const->const_boltzmann_k =
const_boltzmann_k_cgs /
units_general_cgs_conversion_factor(us, dimension_k);
const float dimension_Na[5] = {0, 0, 0, 0, 0}; /* [ - ] */
internal_const->const_avogadro_number =
const_avogadro_number_cgs /
units_general_cgs_conversion_factor(us, dimension_Na);
const float dimension_thomson[5] = {0, 2, 0, 0, 0}; /* [cm^2] */
internal_const->const_thomson_cross_section =
const_thomson_cross_section_cgs /
units_general_cgs_conversion_factor(us, dimension_thomson);
const float dimension_stefan[5] = {1, 0, -3, 0, -4}; /* [g s^-3 K^-4] */
internal_const->const_stefan_boltzmann =
const_stefan_boltzmann_cgs /
units_general_cgs_conversion_factor(us, dimension_stefan);
const float dimension_ev[5] = {1, 2, -2, 0, 0}; /* [g cm^2 s^-2] */
internal_const->const_electron_volt =
const_electron_volt_cgs /
units_general_cgs_conversion_factor(us, dimension_ev);
const float dimension_charge[5] = {0, 0, 1, 1, 0}; /* [A s] */
internal_const->const_electron_charge =
const_electron_charge_cgs /
units_general_cgs_conversion_factor(us, dimension_charge);
const float dimension_permea[5] = {1, 1, -2, -2, 0}; /* [g cm A^-2 s^-2] */
internal_const->const_vacuum_permeability =
const_vacuum_permeability_cgs /
units_general_cgs_conversion_factor(us, dimension_permea);
/* Overwrite mu0 if present in the file */
if (params != NULL) {
internal_const->const_vacuum_permeability =
parser_get_opt_param_double(params, "PhysicalConstants:mu_0",
internal_const->const_vacuum_permeability);
}
const float dimension_mass[5] = {1, 0, 0, 0, 0}; /* [g] */
internal_const->const_electron_mass =
const_electron_mass_cgs /
units_general_cgs_conversion_factor(us, dimension_mass);
internal_const->const_proton_mass =
const_proton_mass_cgs /
units_general_cgs_conversion_factor(us, dimension_mass);
internal_const->const_solar_mass =
const_solar_mass_cgs /
units_general_cgs_conversion_factor(us, dimension_mass);
internal_const->const_earth_mass =
const_earth_mass_cgs /
units_general_cgs_conversion_factor(us, dimension_mass);
const float dimension_time[5] = {0, 0, 1, 0, 0}; /* [s] */
internal_const->const_year =
const_year_cgs / units_general_cgs_conversion_factor(us, dimension_time);
const float dimension_length[5] = {0, 1, 0, 0, 0}; /* [cm] */
internal_const->const_astronomical_unit =
const_astronomical_unit_cgs /
units_general_cgs_conversion_factor(us, dimension_length);
internal_const->const_parsec =
const_parsec_cgs /
units_general_cgs_conversion_factor(us, dimension_length);
internal_const->const_light_year =
const_light_year_cgs /
units_general_cgs_conversion_factor(us, dimension_length);
internal_const->const_solar_radius =
const_solar_radius_cgs /
units_general_cgs_conversion_factor(us, dimension_length);
internal_const->const_earth_radius =
const_earth_radius_cgs /
units_general_cgs_conversion_factor(us, dimension_length);
const float dimension_power[5] = {1, 2, -3, 0, 0}; /* [g cm^2 s^-3] */
internal_const->const_solar_luminosity =
const_solar_luminosity_cgs /
units_general_cgs_conversion_factor(us, dimension_power);
const float dimension_temperature[5] = {0, 0, 0, 0, 1}; /* [K] */
internal_const->const_T_CMB_0 =
const_T_CMB_0_cgs /
units_general_cgs_conversion_factor(us, dimension_temperature);
const float dimension_Yp[5] = {0, 0, 0, 0, 0}; /* [ - ] */
internal_const->const_primordial_He_fraction =
const_primordial_He_fraction_cgs /
units_general_cgs_conversion_factor(us, dimension_Yp);
const float dimension_reduced_hubble[5] = {0, 0, -1, 0, 0}; /* [s^-1] */
internal_const->const_reduced_hubble =
const_reduced_hubble_cgs /
units_general_cgs_conversion_factor(us, dimension_reduced_hubble);
const float dimension_alphaB[5] = {0, 3, -1, 0, 0}; /* [cm^3 s^-1] */
internal_const->const_caseb_recomb =
const_caseb_recomb_cgs /
units_general_cgs_conversion_factor(us, dimension_alphaB);
}
/**
* @brief Print the value of the physical constants to stdout.
*
* @param internal_const The constants in the internal unit system.
*/
void phys_const_print(const struct phys_const *internal_const) {
message("%25s = %e", "Gravitational constant",
internal_const->const_newton_G);
message("%25s = %e", "Speed of light", internal_const->const_speed_light_c);
message("%25s = %e", "Planck constant", internal_const->const_planck_h);
message("%25s = %e", "Boltzmann constant", internal_const->const_boltzmann_k);
message("%25s = %e", "Thomson cross-section",
internal_const->const_thomson_cross_section);
message("%25s = %e", "Electron-Volt", internal_const->const_electron_volt);
message("%25s = %e", "Vacuum permeability",
internal_const->const_vacuum_permeability);
message("%25s = %e", "Proton mass", internal_const->const_proton_mass);
message("%25s = %e", "Year", internal_const->const_year);
message("%25s = %e", "Parsec", internal_const->const_parsec);
message("%25s = %e", "Astronomical Unit",
internal_const->const_astronomical_unit);
message("%25s = %e", "Earth radius", internal_const->const_earth_radius);
message("%25s = %e", "Solar mass", internal_const->const_solar_mass);
message("%25s = %e", "Solar luminosity",
internal_const->const_solar_luminosity);
message("%25s = %e", "H_0 / h = 100 km/s/Mpc",
internal_const->const_reduced_hubble);
message("%25s = %e", "T_CMB0", internal_const->const_T_CMB_0);
}
#if defined(HAVE_HDF5)
/**
* @brief Write the physical constants to given HDF5 file.
*
* We write the constants both in the CGS and internal systems of units.
*
* @param h_file The opened hdf5 file.
* @param p The physical constants.
*/
void phys_const_print_snapshot(hid_t h_file, const struct phys_const *p) {
const hid_t h_grp = H5Gcreate(h_file, "/PhysicalConstants", H5P_DEFAULT,
H5P_DEFAULT, H5P_DEFAULT);
if (h_grp < 0) error("Error while creating physical constants group");
#ifdef SWIFT_USE_GADGET2_PHYSICAL_CONSTANTS
io_write_attribute_s(h_grp, "Constants choice", "Gadget-2 defaults");
#else
io_write_attribute_s(h_grp, "Constants choice", "SWIFT defaults (PDG 2019)");
#endif
/* Start by writing all the constants in CGS */
const hid_t h_grp_cgs =
H5Gcreate(h_grp, "CGS", H5P_DEFAULT, H5P_DEFAULT, H5P_DEFAULT);
if (h_grp_cgs < 0) error("Error while creating CGS group");
io_write_attribute_d(h_grp_cgs, "newton_G", const_newton_G_cgs);
io_write_attribute_d(h_grp_cgs, "speed_light_c", const_speed_light_c_cgs);
io_write_attribute_d(h_grp_cgs, "planck_h", const_planck_h_cgs);
io_write_attribute_d(h_grp_cgs, "planck_hbar", const_planck_hbar_cgs);
io_write_attribute_d(h_grp_cgs, "boltzmann_k", const_boltzmann_k_cgs);
io_write_attribute_d(h_grp_cgs, "avogadro_number", const_avogadro_number_cgs);
io_write_attribute_d(h_grp_cgs, "thomson_cross_section",
const_thomson_cross_section_cgs);
io_write_attribute_d(h_grp_cgs, "stefan_boltzmann",
const_stefan_boltzmann_cgs);
io_write_attribute_d(h_grp_cgs, "electron_charge", const_electron_charge_cgs);
io_write_attribute_d(h_grp_cgs, "electron_volt", const_electron_volt_cgs);
io_write_attribute_d(h_grp_cgs, "electron_mass", const_electron_mass_cgs);
io_write_attribute_d(h_grp_cgs, "vacuum_permeability",
const_vacuum_permeability_cgs);
io_write_attribute_d(h_grp_cgs, "proton_mass", const_proton_mass_cgs);
io_write_attribute_d(h_grp_cgs, "year", const_year_cgs);
io_write_attribute_d(h_grp_cgs, "astronomical_unit",
const_astronomical_unit_cgs);
io_write_attribute_d(h_grp_cgs, "parsec", const_parsec_cgs);
io_write_attribute_d(h_grp_cgs, "light_year", const_light_year_cgs);
io_write_attribute_d(h_grp_cgs, "solar_radius", const_solar_radius_cgs);
io_write_attribute_d(h_grp_cgs, "earth_radius", const_earth_radius_cgs);
io_write_attribute_d(h_grp_cgs, "solar_mass", const_solar_mass_cgs);
io_write_attribute_d(h_grp_cgs, "earth_mass", const_earth_mass_cgs);
io_write_attribute_d(h_grp_cgs, "solar_luminosity",
const_solar_luminosity_cgs);
io_write_attribute_d(h_grp_cgs, "T_CMB_0", const_T_CMB_0_cgs);
io_write_attribute_d(h_grp_cgs, "primordial_He_fraction",
const_primordial_He_fraction_cgs);
io_write_attribute_d(h_grp_cgs, "reduced_hubble", const_reduced_hubble_cgs);
io_write_attribute_d(h_grp_cgs, "caseb_recomb", const_caseb_recomb_cgs);
H5Gclose(h_grp_cgs);
/* Now write them in internal units */
const hid_t h_grp_int = H5Gcreate1(h_grp, "InternalUnits", 0);
if (h_grp_int < 0) error("Error while creating internal units group");
io_write_attribute_d(h_grp_int, "newton_G", p->const_newton_G);
io_write_attribute_d(h_grp_int, "speed_light_c", p->const_speed_light_c);
io_write_attribute_d(h_grp_int, "planck_h", p->const_planck_h);
io_write_attribute_d(h_grp_int, "planck_hbar", p->const_planck_hbar);
io_write_attribute_d(h_grp_int, "boltzmann_k", p->const_boltzmann_k);
io_write_attribute_d(h_grp_int, "avogadro_number", p->const_avogadro_number);
io_write_attribute_d(h_grp_int, "thomson_cross_section",
p->const_thomson_cross_section);
io_write_attribute_d(h_grp_int, "stefan_boltzmann",
p->const_stefan_boltzmann);
io_write_attribute_d(h_grp_int, "electron_charge", p->const_electron_charge);
io_write_attribute_d(h_grp_int, "electron_volt", p->const_electron_volt);
io_write_attribute_d(h_grp_int, "electron_mass", p->const_electron_mass);
io_write_attribute_d(h_grp_int, "vacuum_permeability",
p->const_vacuum_permeability);
io_write_attribute_d(h_grp_int, "proton_mass", p->const_proton_mass);
io_write_attribute_d(h_grp_int, "year", p->const_year);
io_write_attribute_d(h_grp_int, "astronomical_unit",
p->const_astronomical_unit);
io_write_attribute_d(h_grp_int, "parsec", p->const_parsec);
io_write_attribute_d(h_grp_int, "light_year", p->const_light_year);
io_write_attribute_d(h_grp_int, "solar_radius", p->const_solar_radius);
io_write_attribute_d(h_grp_int, "earth_radius", p->const_earth_radius);
io_write_attribute_d(h_grp_int, "solar_mass", p->const_solar_mass);
io_write_attribute_d(h_grp_int, "earth_mass", p->const_earth_mass);
io_write_attribute_d(h_grp_int, "solar_luminosity",
p->const_solar_luminosity);
io_write_attribute_d(h_grp_int, "T_CMB_0", p->const_T_CMB_0);
io_write_attribute_d(h_grp_int, "primordial_He_fraction",
p->const_primordial_He_fraction);
io_write_attribute_d(h_grp_int, "reduced_hubble", p->const_reduced_hubble);
io_write_attribute_d(h_grp_int, "caseb_recomb", p->const_caseb_recomb);
H5Gclose(h_grp_int);
H5Gclose(h_grp);
}
#endif
/**
* @brief Write a phys_const struct to the given FILE as a stream of bytes.
*
* @param internal_const the struct
* @param stream the file stream
*/
void phys_const_struct_dump(const struct phys_const *internal_const,
FILE *stream) {
restart_write_blocks((void *)internal_const, sizeof(struct phys_const), 1,
stream, "physconst", "phys_const params");
}
/**
* @brief Restore a phys_const struct from the given FILE as a stream of
* bytes.
*
* @param internal_const the struct
* @param stream the file stream
*/
void phys_const_struct_restore(const struct phys_const *internal_const,
FILE *stream) {
restart_read_blocks((void *)internal_const, sizeof(struct phys_const), 1,
stream, NULL, "phys_const params");
}