net.finmath.montecarlo.interestrate.models.LIBORMarketModelFromCovarianceModel Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of finmath-lib Show documentation
Show all versions of finmath-lib Show documentation
finmath lib is a Mathematical Finance Library in Java.
It provides algorithms and methodologies related to mathematical finance.
/* * (c) Copyright Christian P. Fries, Germany. Contact: [email protected]. * * Created on 09.02.2004 */ package net.finmath.montecarlo.interestrate.models; import java.io.IOException; import java.io.ObjectInputStream; import java.io.Serializable; import java.time.LocalDateTime; import java.time.LocalTime; import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.Map; import java.util.Map.Entry; import java.util.TreeMap; import java.util.concurrent.ConcurrentHashMap; import net.finmath.exception.CalculationException; import net.finmath.functions.AnalyticFormulas; import net.finmath.marketdata.model.AnalyticModel; import net.finmath.marketdata.model.curves.DiscountCurve; import net.finmath.marketdata.model.curves.DiscountCurveFromForwardCurve; import net.finmath.marketdata.model.curves.ForwardCurve; import net.finmath.marketdata.model.volatilities.AbstractSwaptionMarketData; import net.finmath.marketdata.products.Swap; import net.finmath.marketdata.products.SwapAnnuity; import net.finmath.montecarlo.AbstractRandomVariableFactory; import net.finmath.montecarlo.RandomVariableFactory; import net.finmath.montecarlo.interestrate.CalibrationProduct; import net.finmath.montecarlo.interestrate.LIBORMarketModel; import net.finmath.montecarlo.interestrate.models.covariance.AbstractLIBORCovarianceModelParametric; import net.finmath.montecarlo.interestrate.models.covariance.LIBORCovarianceModel; import net.finmath.montecarlo.interestrate.models.covariance.LIBORCovarianceModelCalibrateable; import net.finmath.montecarlo.interestrate.products.AbstractLIBORMonteCarloProduct; import net.finmath.montecarlo.interestrate.products.SwaptionAnalyticApproximation; import net.finmath.montecarlo.interestrate.products.SwaptionSimple; import net.finmath.montecarlo.model.AbstractProcessModel; import net.finmath.montecarlo.process.MonteCarloProcess; import net.finmath.stochastic.RandomVariable; import net.finmath.time.RegularSchedule; import net.finmath.time.Schedule; import net.finmath.time.TimeDiscretization; import net.finmath.time.TimeDiscretizationFromArray; /** * Implements a (generalized) LIBOR market model with generic covariance structure (lognormal, normal, displaced or stochastic volatility) * with some drift approximation methods. *
or
* In its default case the class specifies a multi-factor LIBOR market model in its log-normal formulation, that is * Lj = exp(Yj) where * \[ * dY_{j} = \mu_{j} dt + \lambda_{1,j} dW_{1} + \ldots + \lambda_{m,j} dW_{m} * \] *
* The model uses an {@link net.finmath.montecarlo.interestrate.models.covariance.LIBORCovarianceModel} for the specification of * (λ1,j,...,λm,j) as a covariance model. * See {@link net.finmath.montecarlo.model.ProcessModel} for details on the implemented interface *
* However, the class is more general: **
* *- * The model may be log-normal or normal specification with a given local volatility. *
*- * The class implements different measure(drift) / numeraire pairs: terminal measure and spot measure. *
*- * The class allows to configure a discounting curve (e.g. for "OIS discounting") using a simple deterministic zero spread. * In this case, the numeraire \( N(t) \) is adjusted by \( \exp( \int_0^t -\lambda(\tau) d\tau ) \). *
*
* The class specifies a LIBOR market model, that is * Lj = f(Yj) where **
* and *- * f is f(x) = exp(x) (default, log-normal LIBOR Market Model) or *
*- * f is f(x) = x (normal model, used if
*property.set("stateSpace","NORMAL"))
*
* dYj = μj dt + λ1,j dW1 + ... + λm,j dWm
*
* see {@link net.finmath.montecarlo.model.ProcessModel} for details on the implemented interface. *
* The model uses anAbstractLIBORCovarianceModel
as a covariance model. * If the covariance model is of typeAbstractLIBORCovarianceModelParametric
* a calibration to swaptions can be performed. *
* Note that λ may still depend on L, hence generating a log-normal dynamic for L even * if the stateSpace property has been set to NORMAL. *
* * The mapproperties
allows to configure the model. The following keys may be used: **
*- *
*measure
: Possible values: **
*- *
*SPOT
: Simulate under spot measure. In this case, the single curve numeraire * is \( N(T_{i}) = \prod_{j=0}^{i-1} (1 + L(T_{j},T_{j+1};T_{j}) (T_{j+1}-T_{j})) \). *- *
*TERMINAL
: Simulate under terminal measure. In this case, the single curve numeraire * is \( N(T_{i}) = P(T_{n};T_{i}) = \prod_{j=i}^{n-1} (1 + L(T_{j},T_{j+1};T_{i}) (T_{j+1}-T_{j}))^{-1} \). *- *
*stateSpace
: Possible values: **
*- *
*LOGNORMAL
: The state space transform is set to exp, i.e., L = exp(Y). When the covariance model is deterministic, then this is the classical lognormal LIBOR market model. Note that the covariance model may still provide a local volatility function. *- *
*NORMAL
: The state space transform is set to identity, i.e., L = Y. When the covariance model is deterministic, then this is a normal LIBOR market model. Note that the covariance model may still provide a local volatility function. *- *
*liborCap
: An optionalDouble
value applied as a cap to the LIBOR rates. * May be used to limit the simulated valued to prevent values attaining POSITIVE_INFINITY and * numerical problems. To disable the cap, setliborCap
toDouble.POSITIVE_INFINITY
. *
* The main task of this class is to calculate the risk-neutral drift and the * corresponding numeraire given the covariance model. * * The calibration of the covariance structure is not part of this class. For the calibration * of parametric models of the instantaneous covariance see * {@link net.finmath.montecarlo.interestrate.models.covariance.AbstractLIBORCovarianceModelParametric#getCloneCalibrated(LIBORMarketModel, CalibrationProduct[], Map)}. * * @author Christian Fries * @version 1.2 * @see net.finmath.montecarlo.process.MonteCarloProcess The interface for numerical schemes. * @see net.finmath.montecarlo.model.ProcessModel The interface for models provinding parameters to numerical schemes. * @see net.finmath.montecarlo.interestrate.models.covariance.AbstractLIBORCovarianceModel The abstract covariance model plug ins. * @see net.finmath.montecarlo.interestrate.models.covariance.AbstractLIBORCovarianceModelParametric A parametic covariance model including a generic calibration algorithm. */ public class LIBORMarketModelFromCovarianceModel extends AbstractProcessModel implements LIBORMarketModel, Serializable { private static final long serialVersionUID = 4166077559001066615L; public enum Measure { SPOT, TERMINAL } public enum StateSpace { NORMAL, LOGNORMAL } public enum Driftapproximation { EULER, LINE_INTEGRAL, PREDICTOR_CORRECTOR } public enum InterpolationMethod { LINEAR, LOG_LINEAR_UNCORRECTED, LOG_LINEAR_CORRECTED } private final TimeDiscretization liborPeriodDiscretization; private String forwardCurveName; private AnalyticModel curveModel; private ForwardCurve forwardRateCurve; private DiscountCurve discountCurve; private final AbstractRandomVariableFactory randomVariableFactory; private LIBORCovarianceModel covarianceModel; private AbstractSwaptionMarketData swaptionMarketData; private Driftapproximation driftApproximationMethod = Driftapproximation.EULER; private Measure measure = Measure.SPOT; private StateSpace stateSpace = StateSpace.LOGNORMAL; private InterpolationMethod interpolationMethod = InterpolationMethod.LOG_LINEAR_UNCORRECTED; private double liborCap = 1E5; // This is a cache of the integrated covariance. private double[][][] integratedLIBORCovariance; private transient Object integratedLIBORCovarianceLazyInitLock = new Object(); // Cache for the numeraires, needs to be invalidated if process changes - move out of the object (to process?) private final ConcurrentHashMapnumeraires; private final ConcurrentHashMap numeraireAdjustments; private transient MonteCarloProcess numerairesProcess = null; /** * Creates a LIBOR Market Model for given covariance. * * The map properties
allows to configure the model. The following keys may be used: **
* * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param analyticModel The associated analytic model of this model (containing the associated market data objects like curve). * @param forwardRateCurve The initial values for the forward rates. * @param discountCurve The discount curve to use. This will create an LMM model with a deterministic zero-spread discounting adjustment. * @param randomVariableFactory The random variable factory used to create the inital values of the model. * @param covarianceModel The covariance model to use. * @param properties Key value map specifying properties like- *
*measure
: Possible values: **
*- *
*SPOT
(String
): Simulate under spot measure. *- *
*TERMINAL
(String
): Simulate under terminal measure. *- *
*stateSpace
: Possible values: **
*- *
*LOGNORMAL
(String
): Simulate L = exp(Y). *- *
*NORMAL
(String
): Simulate L = Y. *- *
*liborCap
: An optionalDouble
value applied as a cap to the LIBOR rates. * May be used to limit the simulated valued to prevent values attaining POSITIVE_INFINITY and * numerical problems. To disable the cap, setliborCap
toDouble.POSITIVE_INFINITY
. *- *
*calibrationParameters
: Possible values: **
*- *
*Map<String,Object>
a parameter map with the following key/value pairs: **
*- *
*accuracy
:Double
specifying the required solver accuracy. *- *
*maxIterations
:Integer
specifying the maximum iterations for the solver. *measure
andstateSpace
. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. */ public LIBORMarketModelFromCovarianceModel( TimeDiscretization liborPeriodDiscretization, AnalyticModel analyticModel, ForwardCurve forwardRateCurve, DiscountCurve discountCurve, AbstractRandomVariableFactory randomVariableFactory, LIBORCovarianceModel covarianceModel, Mapproperties ) throws CalculationException { // Set some properties if(properties != null && properties.containsKey("measure")) { measure = Measure.valueOf(((String)properties.get("measure")).toUpperCase()); } if(properties != null && properties.containsKey("stateSpace")) { stateSpace = StateSpace.valueOf(((String)properties.get("stateSpace")).toUpperCase()); } if(properties != null && properties.containsKey("interpolationMethod")) { interpolationMethod = InterpolationMethod.valueOf(((String)properties.get("interpolationMethod")).toUpperCase()); } if(properties != null && properties.containsKey("liborCap")) { liborCap = (Double)properties.get("liborCap"); } Map calibrationParameters = null; if(properties != null && properties.containsKey("calibrationParameters")) { calibrationParameters = (Map )properties.get("calibrationParameters"); } this.liborPeriodDiscretization = liborPeriodDiscretization; this.curveModel = analyticModel; this.forwardRateCurve = forwardRateCurve; this.discountCurve = discountCurve; this.randomVariableFactory = randomVariableFactory; this.covarianceModel = covarianceModel; numeraires = new ConcurrentHashMap<>(liborPeriodDiscretization.getNumberOfTimes()); numeraireAdjustments = new ConcurrentHashMap<>(liborPeriodDiscretization.getNumberOfTimes()); } /** * Creates a LIBOR Market Model for given covariance with a calibration (if calibration items are given). *
* If calibrationItems in non-empty and the covariance model is a parametric model, * the covariance will be replaced by a calibrate version of the same model, i.e., * the LIBOR Market Model will be calibrated. Note: Calibration is not lazy. *
* The mapproperties
allows to configure the model. The following keys may be used: **
* * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param analyticModel The associated analytic model of this model (containing the associated market data objects like curve). * @param forwardRateCurve The initial values for the forward rates. * @param discountCurve The discount curve to use. This will create an LMM model with a deterministic zero-spread discounting adjustment. * @param randomVariableFactory The random variable factory used to create the inital values of the model. * @param covarianceModel The covariance model to use. * @param calibrationProducts The vector of calibration items (a union of a product, target value and weight) for the objective function sum weight(i) * (modelValue(i)-targetValue(i). * @param properties Key value map specifying properties like- *
*measure
: Possible values: **
*- *
*SPOT
(String
): Simulate under spot measure. *- *
*TERMINAL
(String
): Simulate under terminal measure. *- *
*stateSpace
: Possible values: **
*- *
*LOGNORMAL
(String
): Simulate L = exp(Y). *- *
*NORMAL
(String
): Simulate L = Y. *- *
*liborCap
: An optionalDouble
value applied as a cap to the LIBOR rates. * May be used to limit the simulated valued to prevent values attaining POSITIVE_INFINITY and * numerical problems. To disable the cap, setliborCap
toDouble.POSITIVE_INFINITY
. *- *
*calibrationParameters
: Possible values: **
*- *
*Map<String,Object>
a parameter map with the following key/value pairs: **
*- *
*accuracy
:Double
specifying the required solver accuracy. *- *
*maxIterations
:Integer
specifying the maximum iterations for the solver. *measure
andstateSpace
. * @return A new instance of LIBORMarketModelFromCovarianceModel, possibly calibrated. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. */ public static LIBORMarketModelFromCovarianceModel of( TimeDiscretization liborPeriodDiscretization, AnalyticModel analyticModel, ForwardCurve forwardRateCurve, DiscountCurve discountCurve, AbstractRandomVariableFactory randomVariableFactory, LIBORCovarianceModel covarianceModel, CalibrationProduct[] calibrationProducts, Mapproperties ) throws CalculationException { LIBORMarketModelFromCovarianceModel model = new LIBORMarketModelFromCovarianceModel(liborPeriodDiscretization, analyticModel, forwardRateCurve, discountCurve, randomVariableFactory, covarianceModel, properties); // Perform calibration, if data is given if(calibrationProducts != null && calibrationProducts.length > 0) { Map calibrationParameters = null; if(properties != null && properties.containsKey("calibrationParameters")) { calibrationParameters = (Map )properties.get("calibrationParameters"); } LIBORCovarianceModelCalibrateable covarianceModelParametric = null; try { covarianceModelParametric = (LIBORCovarianceModelCalibrateable)covarianceModel; } catch(Exception e) { throw new ClassCastException("Calibration restricted to covariance models implementing LIBORCovarianceModelCalibrateable."); } LIBORCovarianceModel covarianceModelCalibrated = covarianceModelParametric.getCloneCalibrated(model, calibrationProducts, calibrationParameters); LIBORMarketModelFromCovarianceModel modelCalibrated = model.getCloneWithModifiedCovarianceModel(covarianceModelCalibrated); return modelCalibrated; } else { return model; } } /** * Creates a LIBOR Market Model for given covariance. *
* If calibrationItems in non-empty and the covariance model is a parametric model, * the covariance will be replaced by a calibrate version of the same model, i.e., * the LIBOR Market Model will be calibrated. *
* The mapproperties
allows to configure the model. The following keys may be used: **
* * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param analyticModel The associated analytic model of this model (containing the associated market data objects like curve). * @param forwardRateCurve The initial values for the forward rates. * @param discountCurve The discount curve to use. This will create an LMM model with a deterministic zero-spread discounting adjustment. * @param randomVariableFactory The random variable factory used to create the inital values of the model. * @param covarianceModel The covariance model to use. * @param calibrationProducts The vector of calibration items (a union of a product, target value and weight) for the objective function sum weight(i) * (modelValue(i)-targetValue(i). * @param properties Key value map specifying properties like- *
*measure
: Possible values: **
*- *
*SPOT
(String
): Simulate under spot measure. *- *
*TERMINAL
(String
): Simulate under terminal measure. *- *
*stateSpace
: Possible values: **
*- *
*LOGNORMAL
(String
): Simulate L = exp(Y). *- *
*NORMAL
(String
): Simulate L = Y. *- *
*liborCap
: An optionalDouble
value applied as a cap to the LIBOR rates. * May be used to limit the simulated valued to prevent values attaining POSITIVE_INFINITY and * numerical problems. To disable the cap, setliborCap
toDouble.POSITIVE_INFINITY
. *- *
*calibrationParameters
: Possible values: **
*- *
*Map<String,Object>
a parameter map with the following key/value pairs: **
*- *
*accuracy
:Double
specifying the required solver accuracy. *- *
*maxIterations
:Integer
specifying the maximum iterations for the solver. *measure
andstateSpace
. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. * @deprecated Use LIBORMarketModelFromCovarianceModel.of() instead. */ public LIBORMarketModelFromCovarianceModel( TimeDiscretization liborPeriodDiscretization, AnalyticModel analyticModel, ForwardCurve forwardRateCurve, DiscountCurve discountCurve, AbstractRandomVariableFactory randomVariableFactory, LIBORCovarianceModel covarianceModel, CalibrationProduct[] calibrationProducts, Mapproperties ) throws CalculationException { // Set some properties if(properties != null && properties.containsKey("measure")) { measure = Measure.valueOf(((String)properties.get("measure")).toUpperCase()); } if(properties != null && properties.containsKey("stateSpace")) { stateSpace = StateSpace.valueOf(((String)properties.get("stateSpace")).toUpperCase()); } if(properties != null && properties.containsKey("interpolationMethod")) { interpolationMethod = InterpolationMethod.valueOf(((String)properties.get("interpolationMethod")).toUpperCase()); } if(properties != null && properties.containsKey("liborCap")) { liborCap = (Double)properties.get("liborCap"); } Map calibrationParameters = null; if(properties != null && properties.containsKey("calibrationParameters")) { calibrationParameters = (Map )properties.get("calibrationParameters"); } this.liborPeriodDiscretization = liborPeriodDiscretization; this.curveModel = analyticModel; this.forwardRateCurve = forwardRateCurve; this.discountCurve = discountCurve; this.randomVariableFactory = randomVariableFactory; // Perform calibration, if data is given if(calibrationProducts != null && calibrationProducts.length > 0) { LIBORCovarianceModelCalibrateable covarianceModelParametric = null; try { covarianceModelParametric = (LIBORCovarianceModelCalibrateable)covarianceModel; } catch(Exception e) { throw new ClassCastException("Calibration restricted to covariance models implementing LIBORCovarianceModelCalibrateable."); } this.covarianceModel = (LIBORCovarianceModel) covarianceModelParametric.getCloneCalibrated(this, calibrationProducts, calibrationParameters); } else { this.covarianceModel = covarianceModel; } numeraires = new ConcurrentHashMap<>(liborPeriodDiscretization.getNumberOfTimes()); numeraireAdjustments = new ConcurrentHashMap<>(liborPeriodDiscretization.getNumberOfTimes()); } /** * Creates a LIBOR Market Model for given covariance. *
* If calibrationItems in non-empty and the covariance model is a parametric model, * the covariance will be replaced by a calibrate version of the same model, i.e., * the LIBOR Market Model will be calibrated. *
* The mapproperties
allows to configure the model. The following keys may be used: **
* * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param analyticModel The associated analytic model of this model (containing the associated market data objects like curve). * @param forwardRateCurve The initial values for the forward rates. * @param discountCurve The discount curve to use. This will create an LMM model with a deterministic zero-spread discounting adjustment. * @param covarianceModel The covariance model to use. * @param calibrationItems The vector of calibration items (a union of a product, target value and weight) for the objective function sum weight(i) * (modelValue(i)-targetValue(i). * @param properties Key value map specifying properties like- *
*measure
: Possible values: **
*- *
*SPOT
(String
): Simulate under spot measure. *- *
*TERMINAL
(String
): Simulate under terminal measure. *- *
*stateSpace
: Possible values: **
*- *
*LOGNORMAL
(String
): Simulate L = exp(Y). *- *
*NORMAL
(String
): Simulate L = Y. *- *
*liborCap
: An optionalDouble
value applied as a cap to the LIBOR rates. * May be used to limit the simulated valued to prevent values attaining POSITIVE_INFINITY and * numerical problems. To disable the cap, setliborCap
toDouble.POSITIVE_INFINITY
. *- *
*calibrationParameters
: Possible values: **
*- *
*Map<String,Object>
a parameter map with the following key/value pairs: **
*- *
*accuracy
:Double
specifying the required solver accuracy. *- *
*maxIterations
:Integer
specifying the maximum iterations for the solver. *measure
andstateSpace
. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. * @deprecated Use LIBORMarketModelFromCovarianceModel.of() instead. */ public LIBORMarketModelFromCovarianceModel( TimeDiscretization liborPeriodDiscretization, AnalyticModel analyticModel, ForwardCurve forwardRateCurve, DiscountCurve discountCurve, LIBORCovarianceModel covarianceModel, CalibrationProduct[] calibrationItems, Mapproperties ) throws CalculationException { this(liborPeriodDiscretization, analyticModel, forwardRateCurve, discountCurve, new RandomVariableFactory(), covarianceModel, calibrationItems, properties); } /** * Creates a LIBOR Market Model for given covariance. * * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param forwardRateCurve The initial values for the forward rates. * @param covarianceModel The covariance model to use. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via the cause()
method. */ public LIBORMarketModelFromCovarianceModel( TimeDiscretization liborPeriodDiscretization, ForwardCurve forwardRateCurve, LIBORCovarianceModel covarianceModel ) throws CalculationException { this(liborPeriodDiscretization, forwardRateCurve, new DiscountCurveFromForwardCurve(forwardRateCurve), covarianceModel, new CalibrationProduct[0], null); } /** * Creates a LIBOR Market Model for given covariance. * * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param forwardRateCurve The initial values for the forward rates. * @param discountCurve The discount curve to use. This will create an LMM model with a deterministic zero-spread discounting adjustment. * @param covarianceModel The covariance model to use. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. */ public LIBORMarketModelFromCovarianceModel( TimeDiscretization liborPeriodDiscretization, ForwardCurve forwardRateCurve, DiscountCurve discountCurve, LIBORCovarianceModel covarianceModel ) throws CalculationException { this(liborPeriodDiscretization, forwardRateCurve, discountCurve, covarianceModel, new CalibrationProduct[0], null); } /** * Creates a LIBOR Market Model using a given covariance model and calibrating this model * to given swaption volatility data. * * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param forwardRateCurve The initial values for the forward rates. * @param covarianceModel The covariance model to use. * @param swaptionMarketData The set of swaption values to calibrate to. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. */ public LIBORMarketModelFromCovarianceModel( TimeDiscretization liborPeriodDiscretization, ForwardCurve forwardRateCurve, LIBORCovarianceModel covarianceModel, AbstractSwaptionMarketData swaptionMarketData ) throws CalculationException { this(liborPeriodDiscretization, forwardRateCurve, new DiscountCurveFromForwardCurve(forwardRateCurve), covarianceModel, swaptionMarketData, null); } /** * Creates a LIBOR Market Model for given covariance. * * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param forwardRateCurve The initial values for the forward rates. * @param discountCurve The discount curve to use. This will create an LMM model with a deterministic zero-spread discounting adjustment. * @param covarianceModel The covariance model to use. * @param swaptionMarketData The set of swaption values to calibrate to. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. */ public LIBORMarketModelFromCovarianceModel( TimeDiscretization liborPeriodDiscretization, ForwardCurve forwardRateCurve, DiscountCurve discountCurve, LIBORCovarianceModel covarianceModel, AbstractSwaptionMarketData swaptionMarketData ) throws CalculationException { this(liborPeriodDiscretization, forwardRateCurve, discountCurve, covarianceModel, swaptionMarketData, null); } /** * Creates a LIBOR Market Model for given covariance. * * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param forwardRateCurve The initial values for the forward rates. * @param discountCurve The discount curve to use. This will create an LMM model with a deterministic zero-spread discounting adjustment. * @param covarianceModel The covariance model to use. * @param swaptionMarketData The set of swaption values to calibrate to. * @param properties Key value map specifying properties likemeasure
andstateSpace
. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. */ public LIBORMarketModelFromCovarianceModel( TimeDiscretization liborPeriodDiscretization, ForwardCurve forwardRateCurve, DiscountCurve discountCurve, LIBORCovarianceModel covarianceModel, AbstractSwaptionMarketData swaptionMarketData, Mapproperties ) throws CalculationException { this( liborPeriodDiscretization, forwardRateCurve, discountCurve, covarianceModel, getCalibrationItems( liborPeriodDiscretization, forwardRateCurve, swaptionMarketData, // Condition under which we use analytic approximation (properties == null || properties.get("stateSpace") == null || ((String)properties.get("stateSpace")).toUpperCase().equals(StateSpace.LOGNORMAL.name())) && AbstractLIBORCovarianceModelParametric.class.isAssignableFrom(covarianceModel.getClass()) ), properties ); } /** * Creates a LIBOR Market Model for given covariance. *
* If calibrationItems in non-empty and the covariance model is a parametric model, * the covariance will be replaced by a calibrate version of the same model, i.e., * the LIBOR Market Model will be calibrated. *
* The mapproperties
allows to configure the model. The following keys may be used: **
* * @param liborPeriodDiscretization The discretization of the interest rate curve into forward rates (tenor structure). * @param forwardRateCurve The initial values for the forward rates. * @param discountCurve The discount curve to use. This will create an LMM model with a deterministic zero-spread discounting adjustment. * @param covarianceModel The covariance model to use. * @param calibrationItems The vector of calibration items (a union of a product, target value and weight) for the objective function sum weight(i) * (modelValue(i)-targetValue(i). * @param properties Key value map specifying properties like- *
*measure
: Possible values: **
*- *
*SPOT
(String
): Simulate under spot measure. *- *
*TERMINAL
(String
): Simulate under terminal measure. *- *
*stateSpace
: Possible values: **
*- *
*LOGNORMAL
(String
): Simulate L = exp(Y). *- *
*NORMAL
(String
): Simulate L = Y. *- *
*calibrationParameters
: Possible values: **
*- *
*Map<String,Object>
a parameter map with the following key/value pairs: **
*- *
*accuracy
:Double
specifying the required solver accuracy. *- *
*maxIterations
:Integer
specifying the maximum iterations for the solver. *measure
andstateSpace
. * @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. * @deprecated Use LIBORMarketModelFromCovarianceModel.of() instead. */ public LIBORMarketModelFromCovarianceModel( TimeDiscretization liborPeriodDiscretization, ForwardCurve forwardRateCurve, DiscountCurve discountCurve, LIBORCovarianceModel covarianceModel, CalibrationProduct[] calibrationItems, Mapproperties ) throws CalculationException { this(liborPeriodDiscretization, null, forwardRateCurve, discountCurve, covarianceModel, calibrationItems, properties); } private static CalibrationProduct[] getCalibrationItems(TimeDiscretization liborPeriodDiscretization, ForwardCurve forwardCurve, AbstractSwaptionMarketData swaptionMarketData, boolean isUseAnalyticApproximation) { if(swaptionMarketData == null) { return null; } TimeDiscretization optionMaturities = swaptionMarketData.getOptionMaturities(); TimeDiscretization tenor = swaptionMarketData.getTenor(); double swapPeriodLength = swaptionMarketData.getSwapPeriodLength(); ArrayList calibrationProducts = new ArrayList<>(); for(int exerciseIndex=0; exerciseIndex<=optionMaturities.getNumberOfTimeSteps(); exerciseIndex++) { for(int tenorIndex=0; tenorIndex<=tenor.getNumberOfTimeSteps()-exerciseIndex; tenorIndex++) { // Create a swaption double exerciseDate = optionMaturities.getTime(exerciseIndex); double swapLength = tenor.getTime(tenorIndex); if(liborPeriodDiscretization.getTimeIndex(exerciseDate) < 0) { continue; } if(liborPeriodDiscretization.getTimeIndex(exerciseDate+swapLength) <= liborPeriodDiscretization.getTimeIndex(exerciseDate)) { continue; } int numberOfPeriods = (int)(swapLength / swapPeriodLength); double[] fixingDates = new double[numberOfPeriods]; double[] paymentDates = new double[numberOfPeriods]; double[] swapTenorTimes = new double[numberOfPeriods+1]; for(int periodStartIndex=0; periodStartIndex t for which the numeraire should be returned N(t). * @return The numeraire at the specified time as RandomVariable
* @throws net.finmath.exception.CalculationException Thrown if the valuation fails, specific cause may be available via thecause()
method. */ @Override public RandomVariable getNumeraire(double time) throws CalculationException { RandomVariable numeraire = getNumerairetUnAdjusted(time); /* * Adjust for discounting, i.e. funding or collateralization */ if (discountCurve != null) { synchronized (numeraires) { /* * Check if numeraire cache is valid (i.e. process did not change) */ if (getProcess() != numerairesProcess) { numeraires.clear(); numeraireAdjustments.clear(); numerairesProcess = getProcess(); } RandomVariable deterministicNumeraireAdjustment = numeraireAdjustments.get(time); if(deterministicNumeraireAdjustment == null) { // This includes a control for zero bonds deterministicNumeraireAdjustment = numeraire.invert().average().div(discountCurve.getDiscountFactor(curveModel, time)); numeraireAdjustments.put(time, deterministicNumeraireAdjustment); } numeraire = numeraire.mult(deterministicNumeraireAdjustment); } } return numeraire; } protected RandomVariable getNumerairetUnAdjusted(double time) throws CalculationException { /* * Check if numeraire is on LIBOR time grid */ int liborTimeIndex = getLiborPeriodIndex(time); RandomVariable numeraireUnadjusted; if (liborTimeIndex < 0) { /* * Interpolation of Numeraire: use already interpolated short Libor */ int upperIndex = -liborTimeIndex - 1; int lowerIndex = upperIndex - 1; if (lowerIndex < 0) { throw new IllegalArgumentException("Numeraire requested for time " + time + ". Unsupported"); } if (measure == Measure.TERMINAL) { /* * Due to time < T_{timeIndex+1} loop is needed. */ numeraireUnadjusted = getRandomVariableForConstant(1.0); for (int liborIndex = upperIndex; liborIndex <= liborPeriodDiscretization.getNumberOfTimeSteps() - 1; liborIndex++) { RandomVariable libor = getLIBOR(getTimeIndex(Math.min(time, liborPeriodDiscretization.getTime(liborIndex))), liborIndex); double periodLength = liborPeriodDiscretization.getTimeStep(liborIndex); numeraireUnadjusted = numeraireUnadjusted.discount(libor, periodLength); } } else if (measure == Measure.SPOT) { numeraireUnadjusted = getNumerairetUnAdjusted(getLiborPeriod(upperIndex)); } else { throw new CalculationException("Numeraire not implemented for specified measure."); } /* * Multiply with short period bond */ numeraireUnadjusted = numeraireUnadjusted .discount(getLIBOR(time, time, getLiborPeriod(upperIndex)), getLiborPeriod(upperIndex) - time); return numeraireUnadjusted; } else { /* * Calculate the numeraire, when time is part of liborPeriodDiscretization */ return getNumerairetUnAdjustedAtLIBORIndex(liborTimeIndex); } } protected RandomVariable getNumerairetUnAdjustedAtLIBORIndex(int liborTimeIndex) throws CalculationException { /* * synchronize lazy init cache */ synchronized(numeraires) { /* * Check if numeraire cache is valid (i.e. process did not change) */ if (getProcess() != numerairesProcess) { numeraires.clear(); numeraireAdjustments.clear(); numerairesProcess = getProcess(); } /* * Check if numeraire is part of the cache */ RandomVariable numeraireUnadjusted = numeraires.get(liborTimeIndex); if (numeraireUnadjusted == null) { if (measure == Measure.TERMINAL) { int timeIndex = getTimeIndex(liborPeriodDiscretization.getTime(liborTimeIndex)); if(timeIndex < 0) timeIndex = -timeIndex -1; // Initialize to 1.0 numeraireUnadjusted = getRandomVariableForConstant(1.0); /* * Due to time < T_{timeIndex+1} loop is needed. */ for (int liborIndex = liborTimeIndex; liborIndex <= liborPeriodDiscretization.getNumberOfTimeSteps() - 1; liborIndex++) { RandomVariable libor = getLIBOR(timeIndex, liborIndex); double periodLength = liborPeriodDiscretization.getTimeStep(liborIndex); numeraireUnadjusted = numeraireUnadjusted.discount(libor, periodLength); } } else if (measure == Measure.SPOT) { /* * If numeraire is not N(0), multiply (1 + L(Ti-1)*dt) on N(Ti-1) */ if (liborTimeIndex != 0) { int timeIndex = getTimeIndex(liborPeriodDiscretization.getTime(liborTimeIndex-1)); if(timeIndex < 0) timeIndex = -timeIndex -1; double periodLength = liborPeriodDiscretization.getTimeStep(liborTimeIndex - 1); RandomVariable libor = getLIBOR(timeIndex, liborTimeIndex - 1); numeraireUnadjusted = getNumerairetUnAdjustedAtLIBORIndex(liborTimeIndex - 1).accrue(libor, periodLength); } else { numeraireUnadjusted = getRandomVariableForConstant(1.0); } } else { throw new CalculationException("Numeraire not implemented for specified measure."); } numeraires.put(liborTimeIndex, numeraireUnadjusted); } return numeraireUnadjusted; } } public MapgetNumeraireAdjustments() { return Collections.unmodifiableMap(numeraireAdjustments); } @Override public RandomVariable[] getInitialState() { double[] liborInitialStates = new double[liborPeriodDiscretization.getNumberOfTimeSteps()]; for(int timeIndex=0; timeIndex Measure.SPOT Measure.TERMINAL
- depending how the * model object was constructed. ForMeasure.TERMINAL
the j-th entry of the return value is the random variable * \[ * \mu_{j}^{\mathbb{Q}^{P(T_{n})}}(t) \ = \ - \mathop{\sum_{l\geq j+1}}_{l\leq n-1} \frac{\delta_{l}}{1+\delta_{l} L_{l}(t)} (\lambda_{j}(t) \cdot \lambda_{l}(t)) * \] * and forMeasure.SPOT
the j-th entry of the return value is the random variable * \[ * \mu_{j}^{\mathbb{Q}^{N}}(t) \ = \ \sum_{m(t) < l\leq j} \frac{\delta_{l}}{1+\delta_{l} L_{l}(t)} (\lambda_{j}(t) \cdot \lambda_{l}(t)) * \] * where \( \lambda_{j} \) is the vector for factor loadings for the j-th component of the stochastic process (that is, the diffusion part is * \( \sum_{k=1}^m \lambda_{j,k} \mathrm{d}W_{k} \)). * * Note: The scalar product of the factor loadings determines the instantaneous covariance. If the model is written in log-coordinates (using exp as a state space transform), we find * \(\lambda_{j} \cdot \lambda_{l} = \sum_{k=1}^m \lambda_{j,k} \lambda_{l,k} = \sigma_{j} \sigma_{l} \rho_{j,l} \). * If the model is written without a state space transformation (in its orignial coordinates) then \(\lambda_{j} \cdot \lambda_{l} = \sum_{k=1}^m \lambda_{j,k} \lambda_{l,k} = L_{j} L_{l} \sigma_{j} \sigma_{l} \rho_{j,l} \). * * * @see net.finmath.montecarlo.interestrate.models.LIBORMarketModelFromCovarianceModel#getNumeraire(double) The calculation of the drift is consistent with the calculation of the numeraire ingetNumeraire
. * @see net.finmath.montecarlo.interestrate.models.LIBORMarketModelFromCovarianceModel#getFactorLoading(int, int, RandomVariable[]) The factor loading \( \lambda_{j,k} \). * * @param timeIndex Time index i for which the drift should be returned μ(ti). * @param realizationAtTimeIndex Time current forward rate vector at time index i which should be used in the calculation. * @return The drift vector μ(ti) asRandomVariableFromDoubleArray[]
*/ @Override public RandomVariable[] getDrift(int timeIndex, RandomVariable[] realizationAtTimeIndex, RandomVariable[] realizationPredictor) { double time = getTime(timeIndex); int firstLiborIndex = this.getLiborPeriodIndex(time)+1; if(firstLiborIndex<0) { firstLiborIndex = -firstLiborIndex-1 + 1; } RandomVariable zero = getRandomVariableForConstant(0.0); // Allocate drift vector and initialize to zero (will be used to sum up drift components) RandomVariable[] drift = new RandomVariable[getNumberOfComponents()]; for(int componentIndex=firstLiborIndex; componentIndex=firstLiborIndex; componentIndex--) { double periodLength = liborPeriodDiscretization.getTimeStep(componentIndex); RandomVariable libor = realizationAtTimeIndex[componentIndex]; RandomVariable oneStepMeasureTransform = getRandomVariableForConstant(periodLength).discount(libor, periodLength); if(stateSpace == StateSpace.LOGNORMAL) { oneStepMeasureTransform = oneStepMeasureTransform.mult(libor); } RandomVariable[] factorLoading = getFactorLoading(timeIndex, componentIndex, realizationAtTimeIndex); for(int factorIndex=0; factorIndex = liborPeriodDiscretization.getNumberOfTimes() || timeIndex < 0) { throw new ArrayIndexOutOfBoundsException("Index for LIBOR period discretization out of bounds: " + timeIndex + "."); } return liborPeriodDiscretization.getTime(timeIndex); } @Override public int getLiborPeriodIndex(double time) { return liborPeriodDiscretization.getTimeIndex(time); } @Override public TimeDiscretization getLiborPeriodDiscretization() { return liborPeriodDiscretization; } /** * @return Returns the measure. See {@link InterpolationMethod}. */ public InterpolationMethod getInterpolationMethod() { return interpolationMethod; } /** * @return Returns the measure. See {@link Measure}. */ public Measure getMeasure() { return measure; } @Override public double[][][] getIntegratedLIBORCovariance() { synchronized (integratedLIBORCovarianceLazyInitLock) { if(integratedLIBORCovariance == null) { TimeDiscretization liborPeriodDiscretization = getLiborPeriodDiscretization(); TimeDiscretization simulationTimeDiscretization = getTimeDiscretization(); integratedLIBORCovariance = new double[simulationTimeDiscretization.getNumberOfTimeSteps()][liborPeriodDiscretization.getNumberOfTimeSteps()][liborPeriodDiscretization.getNumberOfTimeSteps()]; for(int timeIndex = 0; timeIndex < simulationTimeDiscretization.getNumberOfTimeSteps(); timeIndex++) { double dt = simulationTimeDiscretization.getTime(timeIndex+1) - simulationTimeDiscretization.getTime(timeIndex); RandomVariable[][] factorLoadings = new RandomVariable[liborPeriodDiscretization.getNumberOfTimeSteps()][]; // Prefetch factor loadings for(int componentIndex = 0; componentIndex < liborPeriodDiscretization.getNumberOfTimeSteps(); componentIndex++) { factorLoadings[componentIndex] = getFactorLoading(timeIndex, componentIndex, null); } for(int componentIndex1 = 0; componentIndex1 < liborPeriodDiscretization.getNumberOfTimeSteps(); componentIndex1++) { RandomVariable[] factorLoadingOfComponent1 = factorLoadings[componentIndex1]; // Sum the libor cross terms (use symmetry) for(int componentIndex2 = componentIndex1; componentIndex2 < liborPeriodDiscretization.getNumberOfTimeSteps(); componentIndex2++) { double integratedLIBORCovarianceValue = 0.0; if(getLiborPeriod(componentIndex1) > getTime(timeIndex)) { RandomVariable[] factorLoadingOfComponent2 = factorLoadings[componentIndex2]; for(int factorIndex = 0; factorIndex < getNumberOfFactors(); factorIndex++) { integratedLIBORCovarianceValue += factorLoadingOfComponent1[factorIndex].get(0) * factorLoadingOfComponent2[factorIndex].get(0) * dt; } } integratedLIBORCovariance[timeIndex][componentIndex1][componentIndex2] = integratedLIBORCovarianceValue; } } } // Integrate over time (i.e. sum up). for(int timeIndex = 1; timeIndex < simulationTimeDiscretization.getNumberOfTimeSteps(); timeIndex++) { double[][] prevIntegratedLIBORCovariance = integratedLIBORCovariance[timeIndex-1]; double[][] thisIntegratedLIBORCovariance = integratedLIBORCovariance[timeIndex]; for(int componentIndex1 = 0; componentIndex1 < liborPeriodDiscretization.getNumberOfTimeSteps(); componentIndex1++) { for(int componentIndex2 = componentIndex1; componentIndex2 < liborPeriodDiscretization.getNumberOfTimeSteps(); componentIndex2++) { thisIntegratedLIBORCovariance[componentIndex1][componentIndex2] = prevIntegratedLIBORCovariance[componentIndex1][componentIndex2] + thisIntegratedLIBORCovariance[componentIndex1][componentIndex2]; thisIntegratedLIBORCovariance[componentIndex2][componentIndex1] = thisIntegratedLIBORCovariance[componentIndex1][componentIndex2]; } } } } } return integratedLIBORCovariance; } @Override public Object clone() { try { Map properties = new HashMap<>(); properties.put("measure", measure.name()); properties.put("stateSpace", stateSpace.name()); return new LIBORMarketModelFromCovarianceModel(getLiborPeriodDiscretization(), getAnalyticModel(), getForwardRateCurve(), getDiscountCurve(), randomVariableFactory, covarianceModel, new CalibrationProduct[0], properties); } catch (CalculationException e) { return null; } } @Override public AnalyticModel getAnalyticModel() { return curveModel; } @Override public DiscountCurve getDiscountCurve() { return discountCurve; } @Override public ForwardCurve getForwardRateCurve() { return forwardRateCurve; } /** * Return the swaption market data used for calibration (if any, may be null). * * @return The swaption market data used for calibration (if any, may be null). */ public AbstractSwaptionMarketData getSwaptionMarketData() { return swaptionMarketData; } @Override public LIBORCovarianceModel getCovarianceModel() { return covarianceModel; } /** * @param covarianceModel A covariance model * @return A new LIBORMarketModelFromCovarianceModel
using the specified covariance model. */ @Override public LIBORMarketModelFromCovarianceModel getCloneWithModifiedCovarianceModel(LIBORCovarianceModel covarianceModel) { LIBORMarketModelFromCovarianceModel model = (LIBORMarketModelFromCovarianceModel)this.clone(); model.covarianceModel = covarianceModel; return model; } @Override public LIBORMarketModelFromCovarianceModel getCloneWithModifiedData(MapdataModified) throws CalculationException { TimeDiscretization liborPeriodDiscretization = this.liborPeriodDiscretization; AnalyticModel analyticModel = this.curveModel; ForwardCurve forwardRateCurve = this.forwardRateCurve; DiscountCurve discountCurve = this.discountCurve; LIBORCovarianceModel covarianceModel = this.covarianceModel; AbstractSwaptionMarketData swaptionMarketData = null; // No recalibration, unless new swaption data is specified Map properties = new HashMap<>(); properties.put("measure", measure.name()); properties.put("stateSpace", stateSpace.name()); if(dataModified != null && dataModified.containsKey("liborPeriodDiscretization")) { liborPeriodDiscretization = (TimeDiscretization)dataModified.get("liborPeriodDiscretization"); } if(dataModified != null && dataModified.containsKey("forwardRateCurve")) { forwardRateCurve = (ForwardCurve)dataModified.get("forwardRateCurve"); } if(dataModified != null && dataModified.containsKey("discountCurve")) { discountCurve = (DiscountCurve)dataModified.get("discountCurve"); } if(dataModified != null && dataModified.containsKey("forwardRateShift")) { throw new RuntimeException("Forward rate shift clone currently disabled."); } if(dataModified != null && dataModified.containsKey("covarianceModel")) { covarianceModel = (LIBORCovarianceModel)dataModified.get("covarianceModel"); } if(dataModified != null && dataModified.containsKey("swaptionMarketData")) { swaptionMarketData = (AbstractSwaptionMarketData)dataModified.get("swaptionMarketData"); } LIBORMarketModelFromCovarianceModel newModel = new LIBORMarketModelFromCovarianceModel(liborPeriodDiscretization, curveModel, forwardRateCurve, discountCurve, randomVariableFactory, covarianceModel, properties); newModel.curveModel = analyticModel; return newModel; } @Override public Map getModelParameters() { Map modelParameters = new TreeMap<>(); // Add initial values for(int liborIndex=0; liborIndex numeraireAdjustment : numeraireAdjustments.entrySet()) { modelParameters.put("NUMERAIREADJUSTMENT("+ numeraireAdjustment.getKey() + ")", numeraireAdjustment.getValue()); } return modelParameters; } private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { in.defaultReadObject(); /* * Init transient fields */ integratedLIBORCovarianceLazyInitLock = new Object(); } @Override public String toString() { return "LIBORMarketModelFromCovarianceModel [liborPeriodDiscretization=" + liborPeriodDiscretization + ", forwardCurveName=" + forwardCurveName + ", curveModel=" + curveModel + ", forwardRateCurve=" + forwardRateCurve + ", discountCurve=" + discountCurve + ", covarianceModel=" + covarianceModel + ", driftApproximationMethod=" + driftApproximationMethod + ", measure=" + measure + ", stateSpace=" + stateSpace + "]"; } }
© 2015 - 2025 Weber Informatics LLC | Privacy Policy