-
Notifications
You must be signed in to change notification settings - Fork 18
Tutorial: Constraining Galacticus Parameters
This tutorial guides you through constraining the parameters of a Galacticus model to achieve a good fit to an observational dataset. Galacticus has built-in MCMC functionality which can be used for this purpose.
Here we will use a very simple example - we'll constrain a single parameter of a Galacticus model to obtain a match to a single point in the stellar mass-halo mass relation of Leauthaud et al. (2012). Much more complex cases are possible (multiple parameters, coupled parameters, multiple target datasets etc.), but this simple example will illustrate the key features.
For this tutorial we require Galacticus to be compiled with MPI parallelism. To do this:
make -j8 GALACTICUS_BUILD_OPTION=MPI Galacticus.exe
Note that you must have MPI installed for this to work.
Once compilation is completed, to run the tutorial model:
export OMP_NUM_THREADS=1
mpirun -np 4 Galacticus.exe parameters/tutorials/mcmcConfig.xml
The export OMP_NUM_THREADS=1
effectively switches off OpenMP parallelism (which we don't want to use for this tutorial). The mpirun -np 4
prefix command launches 4 parallel Galacticus processes which will communicate via MPI.
Expect this example to run for around 15 minutes - it should output something like this:
##
#### # #
# # # #
# ### # ### ### ### ## ### ## ## ##
# # # # # # # # # # # # # # #
# ### ### # ### # # # # # # #
# # # # # # # # # # # # # #
#### #### ### #### ### ## ### ### #### ##
© 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016,
2017, 2018, 2019, 2020
- Andrew Benson
1:M: -> Begin task: posterior sampling
1:M: Process 0001 [PID: 13486] is running on host 'node081'
2:M: -> Begin task: posterior sampling
2:M: Process 0002 [PID: 13487] is running on host 'node081'
3:M: -> Begin task: posterior sampling
3:M: Process 0003 [PID: 13488] is running on host 'node081'
0:M: -> Begin task: posterior sampling
0:M: Process 0000 [PID: 13485] is running on host 'node081'
0:M: -> Load balancing report
0:M: Not performing load balancing - missing work cost data
0:M: <- done
0:M: Chain 0 has logℒ= -0.8306E+00
1:M: Chain 1 has logℒ= -0.4578E+01
2:M: Chain 2 has logℒ= -0.3024E+02
3:M: Chain 3 has logℒ= -0.1131E+01
0:M: -> Node work done vs. expected:
0:M: Node 0000: work (actual/estimated) = 10.65 / -1.00
0:M: Node 0001: work (actual/estimated) = 8.99 / -1.00
0:M: Node 0002: work (actual/estimated) = 8.70 / -1.00
0:M: Node 0003: work (actual/estimated) = 8.07 / -1.00
0:M: <- done
0:M: -> Load balancing report
0:M: -> Chain redistribution:
0:M: Chain 0000 -> process/node 0000/0001 (work = 10.65)
0:M: Chain 0001 -> process/node 0001/0001 (work = 8.99)
0:M: Chain 0002 -> process/node 0002/0001 (work = 8.70)
0:M: Chain 0003 -> process/node 0003/0001 (work = 8.07)
0:M: <- done
0:M: -> Node work loads:
0:M: Node 0001: work = 36.40
0:M: <- done
0:M: <- done
0:M: Chain 0 has logℒ= -0.1154E+01
1:M: Chain 1 has logℒ= -0.3044E+02
2:M: Chain 2 has logℒ= -0.4176E+01
3:M: Chain 3 has logℒ= -0.3042E+02
0:M: -> Node work done vs. expected:
0:M: Node 0000: work (actual/estimated) = 8.01 / 10.65
0:M: Node 0001: work (actual/estimated) = 7.93 / 8.99
0:M: Node 0002: work (actual/estimated) = 8.71 / 8.70
0:M: Node 0003: work (actual/estimated) = 7.70 / 8.07
0:M: <- done
.
.
.
0:M: Converged after 10 steps
0:M: -> Load balancing report
0:M: -> Chain redistribution:
0:M: Chain 0000 -> process/node 0002/0001 (work = 7.84)
0:M: Chain 0001 -> process/node 0003/0001 (work = 7.81)
0:M: Chain 0002 -> process/node 0001/0001 (work = 8.14)
0:M: Chain 0003 -> process/node 0000/0001 (work = 9.10)
0:M: <- done
0:M: -> Node work loads:
0:M: Node 0001: work = 32.87
0:M: <- done
0:M: <- done
0:M: Chain 3 has logℒ= -0.8918E+00
1:M: Chain 2 has logℒ= -0.3917E+00
2:M: Chain 0 has logℒ= -0.9008E+00
3:M: Chain 1 has logℒ= -0.2899E+01
0:M: -> Node work done vs. expected:
0:M: Node 0000: work (actual/estimated) = 8.41 / 7.84
0:M: Node 0001: work (actual/estimated) = 8.42 / 7.81
0:M: Node 0002: work (actual/estimated) = 8.53 / 8.14
0:M: Node 0003: work (actual/estimated) = 8.55 / 9.10
0:M: <- done
.
.
.
0:M: -> Load balancing report
0:M: -> Chain redistribution:
0:M: Chain 0000 -> process/node 0001/0001 (work = 8.85)
0:M: Chain 0001 -> process/node 0000/0001 (work = 13.86)
0:M: Chain 0002 -> process/node 0002/0001 (work = 8.22)
0:M: Chain 0003 -> process/node 0003/0001 (work = 8.06)
0:M: <- done
0:M: -> Node work loads:
0:M: Node 0001: work = 39.00
0:M: <- done
0:M: <- done
0:M: Chain 1 has logℒ= -0.3850E+01
1:M: Chain 0 has logℒ= -0.1265E+02
2:M: Chain 2 has logℒ= -0.1516E-01
3:M: Chain 3 has logℒ= -0.3263E+02
0:M: -> Node work done vs. expected:
0:M: Node 0000: work (actual/estimated) = 7.64 / 8.85
0:M: Node 0001: work (actual/estimated) = 9.02 / 13.86
0:M: Node 0002: work (actual/estimated) = 9.03 / 8.22
0:M: Node 0003: work (actual/estimated) = 8.44 / 8.06
0:M: <- done
0:M: <- Done task: posterior sampling
2:M: <- Done task: posterior sampling
1:M: <- Done task: posterior sampling
3:M: <- Done task: posterior sampling
where we have cut out some of the output of chain likelihoods for brevity.
The output starts by reporting on where each MPI process is running and lists the PID for that process (which can be useful for debugging). Since we're running four MPI processes we get 4 MCMC chains, which begin to evaluate our model for different combinations of parameters. After each evaluation you'll see a report on the likelihood of the model for that chain:
0:M: Chain 0 has logℒ= -0.8306E+00
The simulation will continue evaluating models until it completes. In between this chain output you'll see various reports, for example a report on run time for each chain:
0:M: -> Node work done vs. expected:
0:M: Node 0000: work (actual/estimated) = 8.41 / 7.84
0:M: Node 0001: work (actual/estimated) = 8.42 / 7.81
0:M: Node 0002: work (actual/estimated) = 8.53 / 8.14
0:M: Node 0003: work (actual/estimated) = 8.55 / 9.10
0:M: <- done
which reports how much CPU time was spent on each process vs. how much was estimated would be used. This is used in load balancing, which is also reported:
0:M: -> Load balancing report
0:M: -> Chain redistribution:
0:M: Chain 0000 -> process/node 0002/0001 (work = 7.84)
0:M: Chain 0001 -> process/node 0003/0001 (work = 7.81)
0:M: Chain 0002 -> process/node 0001/0001 (work = 8.14)
0:M: Chain 0003 -> process/node 0000/0001 (work = 9.10)
0:M: <- done
0:M: -> Node work loads:
0:M: Node 0001: work = 32.87
0:M: <- done
0:M: <- done
which reports which chain was computed on which node and how much time total was used by each node.
You can view the complete input parameter file for this tutorial here. Here we'll focus on each section of the parameter file and understand what it does:
<taskMethod value="posteriorSample">
<initializeNodeClassHierarchy value="false"/>
</taskMethod>
We begin by specifying the "task" to perform - we choose posteriorSample
which causes Galacticus to run a posterior sampling simulation, which will generate a set of parameters sampled from the posterior distribution given some constraining datasets.
<posteriorSampleLikelihoodMethod value="galaxyPopulation">
<baseParametersFileName value="parameters/tutorials/mcmcBase.xml" />
<failedParametersFileName value="./failedParameters.xml"/>
<randomize value="false" />
<evolveForestsVerbosity value="0" />
</posteriorSampleLikelihoodMethod>
In this section we specify how the likelihood function for the model should be computed. Many different likelihood functions can be used, but here we use the galaxyPopulation
likelihood function - which causes Galacticus to run a model generating a population of galaxies, computed some observables from that population, and then find the likelihood of that model given some target datasets. We specify a baseParametersFileName
which gives the path to a parameter file describing the galaxy population model to run - this will be described in more detail below. We also specify a path for the output of failed parameter sets - sometimes while exploring the model parameter space a model will fail to complete - its parameters will be output to this file for further exploration. The randomize
option specifies whether or not to change the random number seed for the model on each evaluation. Finally, we set evolveForestsVerbosity
to 0 to switch off output of logging messages while running the galaxy population calculation (this just keeps the amount of output reasonable).
<!-- MCMC -->
<posteriorSampleSimulationMethod value="differentialEvolution">
<stepsMaximum value="1000" />
<acceptanceAverageCount value=" 10" />
<stateSwapCount value=" 100" />
<logFileRoot value="mcmcChains"/>
<reportCount value=" 10" />
<sampleOutliers value="false" />
<logFlushCount value=" 1" />
This next section controls the type of posterior sampling simulation that we want to perform. Here we choose to use the differential evolution MCMC algorithm (which runs multiple MCMC chains in parallel). Note that this section is not closed at this point - the sections below are all therefore subsections of this section.
<posteriorSampleStateMethod value="correlation">
<acceptedStateCount value="100"/>
</posteriorSampleStateMethod>
Next we specify the type of object used to store the state of each chain - here we choose a type which can store and compute the correlation structure of each chain's state.
<posteriorSampleStateInitializeMethod value="latinHypercube">
<maximinTrialCount value="100"/>
</posteriorSampleStateInitializeMethod>
Each MCMC chain must be assigned some initial position in the model parameter space. He we choose to assign those positions using a Latin hypercube design, in which we generate 100 trial Latin hypercubes and keep the hypercube with the maximum minimum distance between chains - this ensures a good coverage of the parameter space.
<posteriorSampleConvergenceMethod value="gelmanRubin">
<thresholdHatR value=" 1.30"/>
<burnCount value="10" />
<testCount value="10" />
<outlierCountMaximum value=" 1" />
<outlierSignificance value=" 0.95"/>
<outlierLogLikelihoodOffset value="60" />
<reportCount value=" 1" />
<logFileName value="mcmcConvergence.log"/>
</posteriorSampleConvergenceMethod>
To judge the convergence of our simulation we choose here to use the Gelman-Rubin statistic - specifying the threshold value for convergence to be declared, how many steps to burn before checking convergence, and how often to test convergence. There are also options to control detection of outlier chains.
<posteriorSampleStoppingCriterionMethod value="stepCount">
<stopAfterCount value="10"/>
</posteriorSampleStoppingCriterionMethod>
This option controls when the simulation will be stopped. Here we choose a simple example which stops the simulation after a specified number of steps post-convergence.
<posteriorSampleDffrntlEvltnRandomJumpMethod value="adaptive"/>
In generating proposed new states in the differential evolution MCMC method a random component is added to each chain to ensure that the parameter space is fully converged. This option choose to use the adaptive
method for this random component, which scales the size of the random proportion based on the current chain distribution.
<posteriorSampleDffrntlEvltnProposalSizeMethod value="adaptive" >
<gammaInitial value="0.500e+0"/>
<gammaAdjustFactor value="1.100e+0"/>
<gammaMinimum value="1.000e-4"/>
<gammaMaximum value="3.000e+0"/>
<acceptanceRateMinimum value="0.100e+0"/>
<acceptanceRateMaximum value="0.900e+0"/>
<updateCount value="10" />
</posteriorSampleDffrntlEvltnProposalSizeMethod>
In the differential evolution MCMC algorithm new proposed states are generated by taking some fraction, γ, of the difference of two randomly selected states and adding that to the current state. The choice of γ is controlled by the above section. We chose the adaptive
option which will adjust the size of γ to ensure that the acceptance rate of proposed states remains between a given minimum and maximum.
<!-- Feedback -->
<modelParameterMethod value="active">
<name value="nodeOperatorMethod::nodeOperatorMethod[2]::stellarFeedbackOutflowsMethod::stellarFeedbackOutflowsMethod::velocityCharacteristic"/>
<distributionFunction1DPrior value="uniform">
<limitLower value="25.0"/>
<limitUpper value="500.0"/>
</distributionFunction1DPrior>
<operatorUnaryMapper value="identity"/>
<distributionFunction1DPerturber value="cauchy">
<median value="0.0"/>
<scale value="1.0e-3"/>
</distributionFunction1DPerturber>
</modelParameterMethod>
Most importantly, we have to specify the model parameters that we wish to constrain. Here we use just one, but any number can be included. We choose an active
parameter (one which will be adjusted to obtain the best match to the target data). The name
elements specifies the parameter in the base parameter file to vary, as we will describe below. We must specify a prior for this function - here we choose a uniform prior between given lower and upper limits. We also specify a "mapping" for the parameter - here we use an identity mapping, but we could also use a logarithmic mapping such that the internal calculations would be performed with the log of the parameter value. Finally, we must specify a distribution function for the random perturbation added to each proposal generated for this parameter - here we use a Cauchy distribution which typically produces a small perturbation, but has a tail to arbitrarily large perturbations.
</posteriorSampleSimulationMethod>
Finally, we close the posteriodSampleSimulationMethod
section.
<!-- Random seed -->
<randomNumberGeneratorMethod value="GSL">
<seed value="219" />
<mpiRankOffset value="true"/>
</randomNumberGeneratorMethod>
One last thing - since the MCMC algorithm is a Monte Carlo algorithm it needs to generate random numbers. Here we choose a random number generator (we use the GSL standard algorithm), and explicitly request that the random number seeds be offset for each different MPI process rank - this ensures that each of our MPI processes generates a different sequence of random numbers.
In addition to the mcmcConfig.xml
parameter file that we examined above, a second parameter file is used in this tutorial. As explained above when discussing the posteriorSampleLikelihoodMethod
section, we supply a parameter file which defines the "base" model, which will then have its parameters varied to attempt to find a good fit to the target dataset. In this example that base parameter file is parameters/tutorials/mcmcBase.xml
. This is just a normal Galacticus parameter file, using the evolveForests
task, that you could run directly to generate a model if you wanted to. The only important feature of it is that it contains each parameter which will be varied as specified in the mcmcConfig.xml
file. In this case it contains a section:
<!-- Node evolution and physics -->
<nodeOperatorMethod value="multi">
<!-- Star formation options -->
<nodeOperatorMethod value="starFormationDisks" >
<luminositiesStellarInactive value="true"/>
</nodeOperatorMethod>
<nodeOperatorMethod value="starFormationSpheroids">
<luminositiesStellarInactive value="true"/>
</nodeOperatorMethod>
<!--Stellar feedback outflows-->
<nodeOperatorMethod value="stellarFeedbackDisks">
<stellarFeedbackOutflowsMethod value="rateLimit">
<timescaleOutflowFractionalMinimum value="0.001"/>
<stellarFeedbackOutflowsMethod value="powerLaw">
<velocityCharacteristic value="175.0"/> <!-- This is the parameter being varied -->
<exponent value=" 3.5"/>
</stellarFeedbackOutflowsMethod>
</stellarFeedbackOutflowsMethod>
</nodeOperatorMethod>
<nodeOperatorMethod value="stellarFeedbackSpheroids">
<stellarFeedbackOutflowsMethod value="rateLimit">
<timescaleOutflowFractionalMinimum value="0.001"/>
<stellarFeedbackOutflowsMethod value="powerLaw">
<velocityCharacteristic value=" 50.0"/>
<exponent value=" 3.5"/>
</stellarFeedbackOutflowsMethod>
</stellarFeedbackOutflowsMethod>
</nodeOperatorMethod>
</nodeOperatorMethod>
In the mcmcConfig.xml
file we referred to a parameter "nodeOperatorMethod::nodeOperatorMethod[2]::stellarFeedbackOutflowsMethod::stellarFeedbackOutflowsMethod::velocityCharacteristic
". This refers to the parameter indicated (by a comment) in the above - we first find the nodeOperatorMethod
element in mcmcBase.xml
. Next we look for the third nodeOperatorMethod
element within that element (remember that nodeOperatorMethod[2]
indexes the third nodeOperatorMethod
element as our indexing starts at 0). Next we find the stellarFeedbackOutflowsMethod
element inside that element, then the stellarFeedbackOutflowsMethod
inside that element, and finally locate the velocityCharacteristic
element inside that element. It is the value of this parameter which will be varied.
Of course, you can specify any other parameters in the base parameter file also - their values will be fixed throughout the MCMC simulation.
The other key part of the base parameter file is in defining the target datasets to be used to constrain the model. These make use of Galacticus' ability to compute predictions for observables as it runs. The base parameter file contains a section:
<!-- Analyses -->
<outputAnalysisMethod value="stellarVsHaloMassRelationLeauthaud2012" >
<redshiftInterval value="1" />
<computeScatter value="false" />
<systematicErrorPolynomialCoefficient value="0.0 0.0"/>
<likelihoodBin value="11" />
</outputAnalysisMethod>
This tells Galacticus to compute its prediction for the stellar mass-halo mass relation of Leauthaud et al. (2012) - specifically in the first redshift interval (Leauthaud et al. (2012) compute the relation in three redshift intervals), and to compute the likelihood of the model given this dataset using only halo mass bin number 11 (since for this simple tutorial we only compute halos falling within that bin). The computeScatter
option is set to false
such that we compute the mean of the relation - if it were instead set to true
the scatter in the relation would be computed instead. The systematicErrorPolynomialCoefficient
parameter allows for the possibility of including and constraining a model of observational systematic errors here, but we ignore it for now (setting the values of this parameter to zero).
-
Tutorials
- Introduction to Galacticus parameter files
- Dark matter halo mass function
- Warm dark matter halo mass function
- Power spectra
- Warm dark matter power spectra
- Dark matter only merger trees
- Subsampling of merger tree branches
- Dark matter only subhalo evolution
- Solving the excursion set problem
- Reionization calculations
- Instantaneous & Non-instantaneous recycling
- Computing Broadband Stellar Luminosities
- Postprocessing of stellar spectra
- Using N-body Merger Trees
- Generating Mock Catalogs with Lightcones
- Constraining Galacticus parameters
- Generating galaxy merger trees
-
How Galacticus works
- Structure Formation Flowchart
- Merger Tree Building Flowchart
- How Galacticus Evolves Halos and Galaxies
- Galaxy Physics Flowchart
- CGM Cooling Physics Flowchart
- Star Formation Physics Flowchart
- Outflow Physics Flowchart
- Galactic Structure Flowchart
- CGM Physics Flowchart
- SMBH Physics Flowchart
- Subhalo Evolution Flowchart
-
Contributing
- Coding conventions
- Coding tutorials
-
Reference models
- Benchmarks and validation scores
- Validation plots and data