All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.finmath.montecarlo.interestrate.models.HullWhiteModel Maven / Gradle / Ivy

/*
 * (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.HashMap;
import java.util.List;
import java.util.Map;
import java.util.TreeMap;
import java.util.logging.Logger;

import net.finmath.exception.CalculationException;
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.montecarlo.RandomVariableFactory;
import net.finmath.montecarlo.RandomVariableFromArrayFactory;
import net.finmath.montecarlo.interestrate.CalibrationProduct;
import net.finmath.montecarlo.interestrate.LIBORModel;
import net.finmath.montecarlo.interestrate.ShortRateModel;
import net.finmath.montecarlo.interestrate.models.covariance.ShortRateVolatilityModel;
import net.finmath.montecarlo.interestrate.models.covariance.ShortRateVolatilityModelCalibrateable;
import net.finmath.montecarlo.interestrate.models.covariance.ShortRateVolatilityModelParametric;
import net.finmath.montecarlo.model.AbstractProcessModel;
import net.finmath.montecarlo.process.MonteCarloProcess;
import net.finmath.stochastic.RandomVariable;
import net.finmath.stochastic.Scalar;
import net.finmath.time.TimeDiscretization;

/**
 * Implements a Hull-White model with time dependent mean reversion speed and time dependent short rate volatility.
 *
 * 

* Model Dynamics *

* * The Hull-While model assumes the following dynamic for the short rate: * \[ d r(t) = ( \theta(t) - a(t) r(t) ) d t + \sigma(t) d W(t) \text{,} \quad r(t_{0}) = r_{0} \text{,} \] * where the function \( \theta \) determines the calibration to the initial forward curve, * \( a \) is the mean reversion and \( \sigma \) is the instantaneous volatility. * * The dynamic above is under the equivalent martingale measure corresponding to the numeraire * \[ N(t) = \exp\left( \int_0^t r(\tau) \mathrm{d}\tau \right) \text{.} \] * * The main task of this class is to provide the risk-neutral drift and the volatility to the numerical scheme (given the volatility model), simulating * \( r(t_{i}) \). The class then also provides and the corresponding numeraire and forward rates (LIBORs). * *

* Time Discrete Model *

* * Assuming piecewise constant coefficients (mean reversion speed \( a \) and short * rate volatility \( \sigma \) the class specifies the drift and factor loadings as * piecewise constant functions for an Euler-scheme. * The class provides the exact Euler step for the joint distribution of * \( (r,N) \), where \( r \) denotes the short rate and \( N \) denotes the * numeraire, following the scheme in ssrn.com/abstract=2737091. * * More specifically (assuming a constant mean reversion speed \( a \) for a moment), considering * \[ \Delta \bar{r}(t_{i}) = \frac{1}{t_{i+1}-t_{i}} \int_{t_{i}}^{t_{i+1}} d r(t) \] * we find from * \[ \exp(-a t) \ \left( \mathrm{d} \left( \exp(a t) r(t) \right) \right) \ = \ a r(t) + \mathrm{d} r(t) \ = \ \theta(t) \mathrm{d}t + \sigma(t) \mathrm{d}W(t) \] * that * \[ \exp(a t_{i+1}) r(t_{i+1}) - \exp(a t_{i}) r(t_{i}) \ = \ \int_{t_{i}}^{t_{i+1}} \left[ \exp(a t) \theta(t) \mathrm{d}t + \exp(a t) \sigma(t) \mathrm{d}W(t) \right] \] * that is * \[ r(t_{i+1}) - r(t_{i}) \ = \ -(1-\exp(-a (t_{i+1}-t_{i})) r(t_{i}) + \int_{t_{i}}^{t_{i+1}} \left[ \exp(-a (t_{i+1}-t)) \theta(t) \mathrm{d}t + \exp(-a (t_{i+1}-t)) \sigma(t) \mathrm{d}W(t) \right] \] * * Assuming piecewise constant \( \sigma \) and \( \theta \), being constant over \( (t_{i},t_{i}+\Delta t_{i}) \), we thus find * \[ r(t_{i+1}) - r(t_{i}) \ = \ \frac{1-\exp(-a \Delta t_{i})}{a \Delta t_{i}} \left( ( \theta(t_{i}) - a \bar{r}(t_{i})) \Delta t_{i} \right) + \sqrt{\frac{1-\exp(-2 a \Delta t_{i})}{2 a \Delta t_{i}}} \sigma(t_{i}) \Delta W(t_{i}) \] . * * In other words, the Euler scheme is exact if the mean reversion \( a \) is replaced by the effective mean reversion * \( \frac{1-\exp(-a \Delta t_{i})}{a \Delta t_{i}} a \) and the volatility is replaced by the * effective volatility \( \sqrt{\frac{1-\exp(-2 a \Delta t_{i})}{2 a \Delta t_{i}}} \sigma(t_{i}) \). * * In the calculations above the mean reversion speed is treated as a constants, but it is straight * forward to see that the same holds for piecewise constant mean reversion speeds, replacing * the expression \( a \ t \) by \( \int_{0}^t a(s) \mathrm{d}s \). * *

* Calibration *

* * The drift of the short rate is calibrated to the given forward curve using * \[ \theta(t) = \frac{\partial}{\partial T} f(0,t) + a(t) f(0,t) + \phi(t) \text{,} \] * where the function \( f \) denotes the instantanenous forward rate and * \( \phi(t) = \frac{1}{2} a \sigma^{2}(t) B(t)^{2} + \sigma^{2}(t) B(t) \frac{\partial}{\partial t} B(t) \) with \( B(t) = \frac{1-\exp(-a t)}{a} \). * *

* Volatility Model *

* * The Hull-White model is essentially equivalent to LIBOR Market Model where the forward rate normal volatility \( \sigma(t,T) \) is * given by * \[ \sigma(t,T_{i}) \ = \ (1 + L_{i}(t) (T_{i+1}-T_{i})) \sigma(t) \exp(-a (T_{i}-t)) \frac{1-\exp(-a (T_{i+1}-T_{i}))}{a (T_{i+1}-T_{i})} \] * (where \( \{ T_{i} \} \) is the forward rates tenor time discretization (note that this is the normal volatility, not the log-normal volatility) * (see ssrn.com/abstract=2737091 for details on the derivation). * Hence, we interpret both, short rate mean reversion speed and short rate volatility as part of the volatility model. * * The mean reversion speed and the short rate volatility have to be provided to this class via an object implementing * {@link net.finmath.montecarlo.interestrate.models.covariance.ShortRateVolatilityModel}. * * This implementation supports different method for the interpolation of the curves. * The property "isInterpolateDiscountFactorsOnLiborPeriodDiscretization" is a boolean. If true, the * given curves are used only at the discretization points given by liborPeriodDiscretization. * This implies that the model reports only a limited set of risk factors in the methods {@link HullWhiteModel#getModelParameters()}. * * @see net.finmath.montecarlo.interestrate.models.covariance.ShortRateVolatilityModel * @see ssrn.com/abstract=2737091 * * @author Christian Fries * @version 1.4 */ public class HullWhiteModel extends AbstractProcessModel implements ShortRateModel, LIBORModel, Serializable { private static final long serialVersionUID = 8677410149401310062L; private static final Logger logger = Logger.getLogger("net.finmath"); private final TimeDiscretization liborPeriodDiscretization; private String forwardCurveName; private final AnalyticModel analyticModel; private final ForwardCurve forwardRateCurve; private final DiscountCurve discountCurve; private final DiscountCurve discountCurveFromForwardCurve; private final RandomVariableFactory randomVariableFactory; private final ShortRateVolatilityModel volatilityModel; private final Map properties; private final boolean isInterpolateDiscountFactorsOnLiborPeriodDiscretization; /* * Cache */ private transient List numeraireDiscountFactors = new ArrayList<>(); private transient List numeraireDiscountFactorForwardRates = new ArrayList<>(); private transient List discountFactorFromForwardCurveCache = new ArrayList<>(); private transient List forwardRateCache = new ArrayList<>(); /** * Creates a Hull-White model which implements LIBORMarketModel. * * @param randomVariableFactory The factory to be used to construct random variables. * @param liborPeriodDiscretization The forward rate discretization to be used in the getLIBOR method. * @param analyticModel The analytic model to be used (currently not used, may be null). * @param forwardRateCurve The forward curve to be used (currently not used, - the model uses disocuntCurve only. * @param discountCurve The disocuntCurve (currently also used to determine the forward curve). * @param volatilityModel The volatility model specifying mean reversion and instantaneous volatility of the short rate. * @param properties A map specifying model properties. */ public HullWhiteModel( final RandomVariableFactory randomVariableFactory, final TimeDiscretization liborPeriodDiscretization, final AnalyticModel analyticModel, final ForwardCurve forwardRateCurve, final DiscountCurve discountCurve, final ShortRateVolatilityModel volatilityModel, final Map properties ) { this.randomVariableFactory = randomVariableFactory; this.liborPeriodDiscretization = liborPeriodDiscretization; this.analyticModel = analyticModel; this.forwardRateCurve = forwardRateCurve; this.discountCurve = discountCurve; this.volatilityModel = volatilityModel; this.properties = new HashMap<>(); if(properties != null) { for(final Map.Entry property : properties.entrySet()) { if(Serializable.class.isAssignableFrom(property.getValue().getClass())) { properties.put(property.getKey(), property.getValue()); } else { logger.warning("Ignored non serializable property under the key " + property.getKey() + ":" + property.getValue()); } } } isInterpolateDiscountFactorsOnLiborPeriodDiscretization = (Boolean) this.properties.getOrDefault("isInterpolateDiscountFactorsOnLiborPeriodDiscretization", Boolean.valueOf(true)); discountCurveFromForwardCurve = new DiscountCurveFromForwardCurve(forwardRateCurve); } /** * Creates a Hull-White model which implements LIBORMarketModel. * * @param liborPeriodDiscretization The forward rate discretization to be used in the getLIBOR method. * @param analyticModel The analytic model to be used (currently not used, may be null). * @param forwardRateCurve The forward curve to be used (currently not used, - the model uses disocuntCurve only. * @param discountCurve The disocuntCurve (currently also used to determine the forward curve). * @param volatilityModel The volatility model specifying mean reversion and instantaneous volatility of the short rate. * @param properties A map specifying model properties (currently not used, may be null). */ public HullWhiteModel( final TimeDiscretization liborPeriodDiscretization, final AnalyticModel analyticModel, final ForwardCurve forwardRateCurve, final DiscountCurve discountCurve, final ShortRateVolatilityModel volatilityModel, final Map properties ) { this(new RandomVariableFromArrayFactory(), liborPeriodDiscretization, analyticModel, forwardRateCurve, discountCurve, volatilityModel, properties); } /** * Creates a Hull-White model which implements LIBORMarketModel. * * @param randomVariableFactory The randomVariableFactory * @param liborPeriodDiscretization The forward rate discretization to be used in the getLIBOR method. * @param analyticModel The analytic model to be used (currently not used, may be null). * @param forwardRateCurve The forward curve to be used (currently not used, - the model uses disocuntCurve only. * @param discountCurve The disocuntCurve (currently also used to determine the forward curve). * @param volatilityModel The volatility model specifying mean reversion and instantaneous volatility of the short rate. * @param calibrationProducts The products to be used for calibration * @param properties The calibration properties * @return A (possibly calibrated) Hull White model. * @throws CalculationException Thrown if calibration fails. */ public static HullWhiteModel of( final RandomVariableFactory randomVariableFactory, final TimeDiscretization liborPeriodDiscretization, final AnalyticModel analyticModel, final ForwardCurve forwardRateCurve, final DiscountCurve discountCurve, final ShortRateVolatilityModel volatilityModel, final CalibrationProduct[] calibrationProducts, final Map properties ) throws CalculationException { final HullWhiteModel model = new HullWhiteModel(randomVariableFactory, liborPeriodDiscretization, analyticModel, forwardRateCurve, discountCurve, volatilityModel, properties); // Perform calibration, if data is given if(calibrationProducts != null && calibrationProducts.length > 0) { ShortRateVolatilityModelCalibrateable volatilityModelParametric = null; try { volatilityModelParametric = (ShortRateVolatilityModelCalibrateable)volatilityModel; } catch(final Exception e) { throw new ClassCastException("Calibration restricted to covariance models implementing HullWhiteModelCalibrateable."); } Map calibrationParameters = null; if(properties != null && properties.containsKey("calibrationParameters")) { calibrationParameters = (Map)properties.get("calibrationParameters"); } final ShortRateVolatilityModelCalibrateable volatilityModelCalibrated = volatilityModelParametric.getCloneCalibrated(model, calibrationProducts, calibrationParameters); final HullWhiteModel modelCalibrated = model.getCloneWithModifiedVolatilityModel(volatilityModelCalibrated); return modelCalibrated; } else { return model; } } @Override public LocalDateTime getReferenceDate() { return LocalDateTime.of(discountCurve.getReferenceDate(), LocalTime.of(0, 0)); } @Override public int getNumberOfComponents() { return 2; } @Override public int getNumberOfFactors() { return 1; } @Override public RandomVariable applyStateSpaceTransform(MonteCarloProcess process, int timeIndex, final int componentIndex, final RandomVariable randomVariable) { return randomVariable; } @Override public RandomVariable applyStateSpaceTransformInverse(MonteCarloProcess process, int timeIndex, final int componentIndex, final RandomVariable randomVariable) { return randomVariable; } @Override public RandomVariable[] getInitialState(MonteCarloProcess process) { // Initial value is zero - BrownianMotion serves as a factory here. final RandomVariable zero = getRandomVariableForConstant(0.0); return new RandomVariable[] { zero, zero }; } @Override public RandomVariable getNumeraire(MonteCarloProcess process, final double time) throws CalculationException { if(time < 0) { return randomVariableFactory.createRandomVariable(discountCurve.getDiscountFactor(analyticModel, time)); } if(time == process.getTime(0)) { // Initial value of numeraire is one - BrownianMotion serves as a factory here. final RandomVariable one = randomVariableFactory.createRandomVariable(1.0); return one; } final int timeIndex = process.getTimeIndex(time); if(timeIndex < 0) { /* * time is not part of the time discretization. */ // Find the time index prior to the current time (note: if time does not match a discretization point, we get a negative value, such that -index is next point). int previousTimeIndex = process.getTimeIndex(time); if(previousTimeIndex < 0) { previousTimeIndex = -previousTimeIndex-1; } previousTimeIndex--; final double previousTime = process.getTime(previousTimeIndex); final double nextTime = process.getTime(previousTimeIndex+1); // Log-linear interpolation return getNumeraire(process, previousTime).log().mult(nextTime-time) .add(getNumeraire(process, nextTime).log().mult(time-previousTime)) .div(nextTime-previousTime).exp(); } final RandomVariable logNum = process.getProcessValue(timeIndex, 1).add(getV(0,time).mult(0.5)); RandomVariable numeraireNormalized = logNum.exp(); // Control variate on zero bond numeraireNormalized = numeraireNormalized.mult(numeraireNormalized.invert().getAverage()); // Apply discount factor scaling RandomVariable discountFactor; if(discountCurve != null) { discountFactor = getDiscountFactor(process, time).div(getDiscountFactorFromForwardCurve(process, time).getAverage()).mult(getDiscountFactorFromForwardCurve(process, time)); // discountFactor = getDiscountFactor(time); } else { discountFactor = getDiscountFactorFromForwardCurve(process, time); } final RandomVariable numeraire = numeraireNormalized.div(discountFactor); return numeraire; } @Override public RandomVariable getForwardDiscountBond(final MonteCarloProcess process, final double time, final double maturity) throws CalculationException { final RandomVariable inverseForwardBondAsOfTime = getForwardRate(process, time, time, maturity).mult(maturity-time).add(1.0); final RandomVariable inverseForwardBondAsOfZero = getForwardRate(process, 0.0, time, maturity).mult(maturity-time).add(1.0); final RandomVariable forwardDiscountBondAsOfZero = getDiscountFactor(process, maturity).div(getDiscountFactor(process, time)); return forwardDiscountBondAsOfZero.mult(inverseForwardBondAsOfZero).div(inverseForwardBondAsOfTime); } @Override public RandomVariable[] getDrift(final MonteCarloProcess process, final int timeIndex, final RandomVariable[] realizationAtTimeIndex, final RandomVariable[] realizationPredictor) { final double time = process.getTime(timeIndex); final double timeNext = process.getTime(timeIndex+1); if(timeNext == time) { return new RandomVariable[] { null, null }; } int timeIndexVolatility = volatilityModel.getTimeDiscretization().getTimeIndex(time); if(timeIndexVolatility < 0) { timeIndexVolatility = -timeIndexVolatility-2; } final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndexVolatility); final RandomVariable driftShortRate = realizationAtTimeIndex[0].mult(meanReversion.mult(getB(time,timeNext).div(-1*(timeNext-time)))); final RandomVariable driftLogNumeraire = realizationAtTimeIndex[0].mult(getB(time,timeNext).div(timeNext-time)); return new RandomVariable[] { driftShortRate, driftLogNumeraire }; } @Override public RandomVariable[] getFactorLoading(final MonteCarloProcess process, final int timeIndex, final int componentIndex, final RandomVariable[] realizationAtTimeIndex) { final double time = process.getTime(timeIndex); final double timeNext = process.getTime(timeIndex+1); int timeIndexVolatility = volatilityModel.getTimeDiscretization().getTimeIndex(time); if(timeIndexVolatility < 0) { timeIndexVolatility = -timeIndexVolatility-2; } final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndexVolatility); final RandomVariable meanReversionTimesTime = meanReversion.mult(-2.0 * (timeNext-time)); // double scaling = Math.sqrt((1.0-Math.exp(-2.0 * meanReversion * (timeNext-time)))/(2.0 * meanReversion * (timeNext-time))); final RandomVariable scaling = meanReversionTimesTime.exp().sub(1.0).div(meanReversionTimesTime).sqrt(); final RandomVariable volatilityEffective = scaling.mult(volatilityModel.getVolatility(timeIndexVolatility)); RandomVariable factorLoading1, factorLoading2; if(componentIndex == 0) { // Factor loadings for the short rate driver. factorLoading1 = volatilityEffective; factorLoading2 = new Scalar(0.0); } else if(componentIndex == 1) { // Factor loadings for the numeraire driver. final RandomVariable volatilityLogNumeraire = getV(time,timeNext).div(timeNext-time).sqrt(); final RandomVariable rho = getDV(time,timeNext).div(timeNext-time).div(volatilityEffective.mult(volatilityLogNumeraire)); factorLoading1 = volatilityLogNumeraire.mult(rho); factorLoading2 = volatilityLogNumeraire.mult(rho.squared().sub(1).mult(-1).sqrt()); } else { throw new IllegalArgumentException(); } return new RandomVariable[] { factorLoading1, factorLoading2 }; } @Override public RandomVariable getRandomVariableForConstant(final double value) { return randomVariableFactory.createRandomVariable(value); } @Override public RandomVariable getForwardRate(final MonteCarloProcess process, final double time, final double periodStart, final double periodEnd) throws CalculationException { return getZeroCouponBond(process, time, periodStart).div(getZeroCouponBond(process, time, periodEnd)).sub(1.0).div(periodEnd-periodStart); } @Override public RandomVariable getLIBOR(final MonteCarloProcess process, final int timeIndex, final int liborIndex) throws CalculationException { return getZeroCouponBond(process, process.getTime(timeIndex), getLiborPeriod(liborIndex)).div(getZeroCouponBond(process, process.getTime(timeIndex), getLiborPeriod(liborIndex+1))).sub(1.0).div(getLiborPeriodDiscretization().getTimeStep(liborIndex)); } @Override public TimeDiscretization getLiborPeriodDiscretization() { return liborPeriodDiscretization; } @Override public int getNumberOfLibors() { return liborPeriodDiscretization.getNumberOfTimeSteps(); } @Override public double getLiborPeriod(final int timeIndex) { return liborPeriodDiscretization.getTime(timeIndex); } @Override public int getLiborPeriodIndex(final double time) { return liborPeriodDiscretization.getTimeIndex(time); } @Override public AnalyticModel getAnalyticModel() { return analyticModel; } @Override public DiscountCurve getDiscountCurve() { return discountCurve; } @Override public ForwardCurve getForwardRateCurve() { return forwardRateCurve; } @Override public LIBORModel getCloneWithModifiedData(final Map dataModified) { if(dataModified == null) { return new HullWhiteModel(randomVariableFactory, liborPeriodDiscretization, analyticModel, forwardRateCurve, discountCurve, volatilityModel, properties); } final RandomVariableFactory newRandomVariableFactory = (RandomVariableFactory) dataModified.getOrDefault("randomVariableFactory", randomVariableFactory); final ShortRateVolatilityModel newVolatilityModel = (ShortRateVolatilityModel) dataModified.getOrDefault("volatilityModel", volatilityModel); return new HullWhiteModel(newRandomVariableFactory, liborPeriodDiscretization, analyticModel, forwardRateCurve, discountCurve, newVolatilityModel, properties); } /** * Returns the "short rate" from timeIndex to timeIndex+1. * * @param timeIndex The time index (corresponding to {@link MonteCarloProcess#getTime(int)}). * @return The "short rate" from timeIndex to timeIndex+1. * @throws CalculationException Thrown if simulation failed. */ private RandomVariable getShortRate(final MonteCarloProcess process, final int timeIndex) throws CalculationException { final double time = process.getTime(timeIndex); final double timePrev = timeIndex > 0 ? process.getTime(timeIndex-1) : time; final double timeNext = process.getTime(timeIndex+1); final RandomVariable zeroRate = getZeroRateFromForwardCurve(process, time); //getDiscountFactorFromForwardCurve(time).div(getDiscountFactorFromForwardCurve(timeNext)).log().div(timeNext-time); final RandomVariable alpha = getDV(0, time); RandomVariable value = process.getProcessValue(timeIndex, 0); value = value.add(alpha); // value = value.sub(Math.log(value.exp().getAverage())); value = value.add(zeroRate); return value; } private RandomVariable getZeroCouponBond(final MonteCarloProcess process, final double time, final double maturity) throws CalculationException { final int timeIndex = process.getTimeIndex(time); if(timeIndex < 0) { final int timeIndexLo = -timeIndex-1-1; final double timeLo = process.getTime(timeIndexLo); return getZeroCouponBond(process, timeLo, maturity).div(getZeroCouponBond(process, timeLo, time)); } final RandomVariable shortRate = getShortRate(process, timeIndex); final RandomVariable A = getA(process, time, maturity); final RandomVariable B = getB(time, maturity); return shortRate.mult(B.mult(-1)).exp().mult(A); } /** * This is the shift alpha of the process, which essentially represents * the integrated drift of the short rate (without the interest rate curve related part). * * @param timeIndex Time index associated with the time discretization obtained from getProcess * @return The integrated drift (integrating from 0 to getTime(timeIndex)). */ private RandomVariable getIntegratedDriftAdjustment(final MonteCarloProcess process, final int timeIndex) { RandomVariable integratedDriftAdjustment = new Scalar(0.0); for(int i=1; i<=timeIndex; i++) { final double t = process.getTime(i-1); final double t2 = process.getTime(i); int timeIndexVolatilityModel = volatilityModel.getTimeDiscretization().getTimeIndex(t); if(timeIndexVolatilityModel < 0) { timeIndexVolatilityModel = -timeIndexVolatilityModel-2; // Get timeIndex corresponding to previous point } final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndexVolatilityModel); integratedDriftAdjustment = integratedDriftAdjustment.add(getShortRateConditionalVariance(0, t).mult(getB(t,t2))).sub(integratedDriftAdjustment.mult(meanReversion.mult(getB(t,t2)))); } return integratedDriftAdjustment; } /** * Returns A(t,T) where * \( A(t,T) = P(T)/P(t) \cdot exp(B(t,T) \cdot f(0,t) - \frac{1}{2} \phi(0,t) * B(t,T)^{2} ) \) * and * \( \phi(t,T) \) is the value calculated from integrating \( ( \sigma(s) exp(-\int_{s}^{T} a(\tau) \mathrm{d}\tau ) )^{2} \) with respect to s from t to T * in getShortRateConditionalVariance. * * @param time The parameter t. * @param maturity The parameter T. * @return The value A(t,T). */ private RandomVariable getA(final MonteCarloProcess process, final double time, final double maturity) { final int timeIndex = process.getTimeIndex(time); final double timeStep = process.getTimeDiscretization().getTimeStep(timeIndex); final RandomVariable zeroRate = getZeroRateFromForwardCurve(process, time); //getDiscountFactorFromForwardCurve(time).div(getDiscountFactorFromForwardCurve(timeNext)).log().div(timeNext-time); final RandomVariable forwardBond = getDiscountFactorFromForwardCurve(process, maturity).div(getDiscountFactorFromForwardCurve(process, time)).log(); final RandomVariable B = getB(time,maturity); final RandomVariable lnA = B.mult(zeroRate).sub(B.squared().mult(getShortRateConditionalVariance(0,time).div(2))).add(forwardBond); return lnA.exp(); } /** * Calculates \( \int_{t}^{T} a(s) \mathrm{d}s \), where \( a \) is the mean reversion parameter. * * @param time The parameter t. * @param maturity The parameter T. * @return The value of \( \int_{t}^{T} a(s) \mathrm{d}s \). */ private RandomVariable getMRTime(final double time, final double maturity) { int timeIndexStart = volatilityModel.getTimeDiscretization().getTimeIndex(time); if(timeIndexStart < 0) { timeIndexStart = -timeIndexStart-2; // Get timeIndex corresponding to previous point } int timeIndexEnd =volatilityModel.getTimeDiscretization().getTimeIndex(maturity); if(timeIndexEnd < 0) { timeIndexEnd = -timeIndexEnd-2; // Get timeIndex corresponding to previous point } RandomVariable integral = new Scalar(0.0); double timePrev = time; double timeNext; for(int timeIndex=timeIndexStart+1; timeIndex<=timeIndexEnd; timeIndex++) { timeNext = volatilityModel.getTimeDiscretization().getTime(timeIndex); final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndex-1); integral = integral.add(meanReversion.mult(timeNext-timePrev)); timePrev = timeNext; } timeNext = maturity; final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndexEnd); integral = integral.add(meanReversion.mult(timeNext-timePrev)); return integral; } /** * Calculates \( B(t,T) = \int_{t}^{T} \exp(-\int_{s}^{T} a(\tau) \mathrm{d}\tau) \mathrm{d}s \), where a is the mean reversion parameter. * For a constant \( a \) this results in \( \frac{1-\exp(-a (T-t)}{a} \), but the method also supports piecewise constant \( a \)'s. * * @param time The parameter t. * @param maturity The parameter T. * @return The value of B(t,T). */ private RandomVariable getB(final double time, final double maturity) { int timeIndexStart = volatilityModel.getTimeDiscretization().getTimeIndex(time); if(timeIndexStart < 0) { timeIndexStart = -timeIndexStart-2; // Get timeIndex corresponding to previous point } int timeIndexEnd =volatilityModel.getTimeDiscretization().getTimeIndex(maturity); if(timeIndexEnd < 0) { timeIndexEnd = -timeIndexEnd-2; // Get timeIndex corresponding to previous point } RandomVariable integral = new Scalar(0.0); double timePrev = time; double timeNext; for(int timeIndex=timeIndexStart+1; timeIndex<=timeIndexEnd; timeIndex++) { timeNext = volatilityModel.getTimeDiscretization().getTime(timeIndex); final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndex-1); integral = integral.add( getMRTime(timeNext,maturity).mult(-1.0).exp().sub( getMRTime(timePrev,maturity).mult(-1.0).exp()).div(meanReversion)); timePrev = timeNext; } final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndexEnd); timeNext = maturity; integral = integral.add( getMRTime(timeNext,maturity).mult(-1.0).exp().sub( getMRTime(timePrev,maturity).mult(-1.0).exp()).div(meanReversion)); return integral; } /** * Calculates the drift adjustment for the log numeraire, that is * \( * \int_{t}^{T} \sigma^{2}(s) B(s,T)^{2} \mathrm{d}s * \) where \( B(t,T) = \int_{t}^{T} \exp(-\int_{s}^{T} a(\tau) \mathrm{d}\tau) \mathrm{d}s \). * * @param time The parameter t in \( \int_{t}^{T} \sigma^{2}(s) B(s,T)^{2} \mathrm{d}s \) * @param maturity The parameter T in \( \int_{t}^{T} \sigma^{2}(s) B(s,T)^{2} \mathrm{d}s \) * @return The integral \( \int_{t}^{T} \sigma^{2}(s) B(s,T)^{2} \mathrm{d}s \). */ private RandomVariable getV(final double time, final double maturity) { if(time == maturity) { return new Scalar(0.0); } int timeIndexStart = volatilityModel.getTimeDiscretization().getTimeIndex(time); if(timeIndexStart < 0) { timeIndexStart = -timeIndexStart-2; // Get timeIndex corresponding to previous point } int timeIndexEnd =volatilityModel.getTimeDiscretization().getTimeIndex(maturity); if(timeIndexEnd < 0) { timeIndexEnd = -timeIndexEnd-2; // Get timeIndex corresponding to previous point } RandomVariable integral = new Scalar(0.0); double timePrev = time; double timeNext; RandomVariable expMRTimePrev = getMRTime(timePrev,maturity).mult(-1).exp(); for(int timeIndex=timeIndexStart+1; timeIndex<=timeIndexEnd; timeIndex++) { timeNext = volatilityModel.getTimeDiscretization().getTime(timeIndex); final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndex-1); final RandomVariable volatility = volatilityModel.getVolatility(timeIndex-1); final RandomVariable volatilityPerMeanReversionSquared = volatility.squared().div(meanReversion.squared()); final RandomVariable expMRTimeNext = getMRTime(timeNext,maturity).mult(-1).exp(); integral = integral.add(volatilityPerMeanReversionSquared.mult( expMRTimeNext.sub(expMRTimePrev).mult(-2).div(meanReversion) .add( expMRTimeNext.squared().sub(expMRTimePrev.squared()).div(meanReversion).div(2.0)) .add(timeNext-timePrev) )); timePrev = timeNext; expMRTimePrev = expMRTimeNext; } timeNext = maturity; final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndexEnd); final RandomVariable volatility = volatilityModel.getVolatility(timeIndexEnd); final RandomVariable volatilityPerMeanReversionSquared = volatility.squared().div(meanReversion.squared()); final RandomVariable expMRTimeNext = getMRTime(timeNext,maturity).mult(-1).exp(); integral = integral.add(volatilityPerMeanReversionSquared.mult( expMRTimeNext.sub(expMRTimePrev).mult(-2).div(meanReversion) .add( expMRTimeNext.squared().sub(expMRTimePrev.squared()).div(meanReversion).div(2.0)) .add(timeNext-timePrev) )); return integral; } private RandomVariable getDV(final double time, final double maturity) { if(time==maturity) { return new Scalar(0.0); } int timeIndexStart = volatilityModel.getTimeDiscretization().getTimeIndex(time); if(timeIndexStart < 0) { timeIndexStart = -timeIndexStart-2; // Get timeIndex corresponding to next point } int timeIndexEnd =volatilityModel.getTimeDiscretization().getTimeIndex(maturity); if(timeIndexEnd < 0) { timeIndexEnd = -timeIndexEnd-2; // Get timeIndex corresponding to previous point } RandomVariable integral = new Scalar(0.0); double timePrev = time; double timeNext; RandomVariable expMRTimePrev = getMRTime(timePrev,maturity).mult(-1).exp(); for(int timeIndex=timeIndexStart+1; timeIndex<=timeIndexEnd; timeIndex++) { timeNext = volatilityModel.getTimeDiscretization().getTime(timeIndex); final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndex-1); final RandomVariable volatility = volatilityModel.getVolatility(timeIndex-1); final RandomVariable volatilityPerMeanReversionSquared = volatility.squared().div(meanReversion.squared()); final RandomVariable expMRTimeNext = getMRTime(timeNext,maturity).mult(-1).exp(); integral = integral.add(volatilityPerMeanReversionSquared.mult( expMRTimeNext.sub(expMRTimePrev).add( expMRTimeNext.squared().sub(expMRTimePrev.squared()).div(-2.0) ) )); timePrev = timeNext; expMRTimePrev = expMRTimeNext; } timeNext = maturity; final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndexEnd); final RandomVariable volatility = volatilityModel.getVolatility(timeIndexEnd); final RandomVariable volatilityPerMeanReversionSquared = volatility.squared().div(meanReversion.squared()); final RandomVariable expMRTimeNext = getMRTime(timeNext,maturity).mult(-1).exp(); integral = integral.add(volatilityPerMeanReversionSquared.mult( expMRTimeNext.sub(expMRTimePrev).add( expMRTimeNext.squared().sub(expMRTimePrev.squared()).div(-2.0) ) )); return integral; } /** * Calculates the variance \( \mathop{Var}(r(t) \vert r(s) ) \), that is * \( * \int_{s}^{t} \sigma^{2}(\tau) \exp(-2 \cdot \int_{\tau}^{t} a(u) \mathrm{d}u ) \ \mathrm{d}\tau * \) where \( a \) is the meanReversion and \( \sigma \) is the short rate instantaneous volatility. * * @param time The parameter s in \( \int_{s}^{t} \sigma^{2}(\tau) \exp(-2 \cdot \int_{\tau}^{t} a(u) \mathrm{d}u ) \ \mathrm{d}\tau \) * @param maturity The parameter t in \( \int_{s}^{t} \sigma^{2}(\tau) \exp(-2 \cdot \int_{\tau}^{t} a(u) \mathrm{d}u ) \ \mathrm{d}\tau \) * @return The conditional variance of the short rate, \( \mathop{Var}(r(t) \vert r(s) ) \). */ public RandomVariable getShortRateConditionalVariance(final double time, final double maturity) { int timeIndexStart = volatilityModel.getTimeDiscretization().getTimeIndex(time); if(timeIndexStart < 0) { timeIndexStart = -timeIndexStart-2; // Get timeIndex corresponding to next point } int timeIndexEnd =volatilityModel.getTimeDiscretization().getTimeIndex(maturity); if(timeIndexEnd < 0) { timeIndexEnd = -timeIndexEnd-2; // Get timeIndex corresponding to previous point } RandomVariable integral = new Scalar(0.0); double timePrev = time; double timeNext; RandomVariable expMRTimePrev = getMRTime(timePrev,maturity).mult(-2).exp(); for(int timeIndex=timeIndexStart+1; timeIndex<=timeIndexEnd; timeIndex++) { timeNext = volatilityModel.getTimeDiscretization().getTime(timeIndex); final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndex-1); final RandomVariable volatility = volatilityModel.getVolatility(timeIndex-1); final RandomVariable volatilitySquaredPerMeanReversion = volatility.squared().div(meanReversion); final RandomVariable expMRTimeNext = getMRTime(timeNext,maturity).mult(-2).exp(); integral = integral.add(volatilitySquaredPerMeanReversion.mult(expMRTimeNext.sub(expMRTimePrev).div(2)) ); timePrev = timeNext; expMRTimePrev = expMRTimeNext; } timeNext = maturity; final RandomVariable meanReversion = volatilityModel.getMeanReversion(timeIndexEnd); final RandomVariable volatility = volatilityModel.getVolatility(timeIndexEnd); final RandomVariable volatilitySquaredPerMeanReversion = volatility.squared().div(meanReversion); final RandomVariable expMRTimeNext = getMRTime(timeNext,maturity).mult(-2).exp(); integral = integral.add(volatilitySquaredPerMeanReversion.mult(expMRTimeNext.sub(expMRTimePrev).div(2)) ); return integral; } public RandomVariable getIntegratedBondSquaredVolatility(final double time, final double maturity) { return getShortRateConditionalVariance(0, time).mult(getB(time,maturity).squared()); } @Override public HullWhiteModel getCloneWithModifiedVolatilityModel(final ShortRateVolatilityModel volatilityModel) { return new HullWhiteModel(randomVariableFactory, liborPeriodDiscretization, analyticModel, forwardRateCurve, discountCurve, volatilityModel, properties); } @Override public ShortRateVolatilityModel getVolatilityModel() { return volatilityModel; } @Override public Map getModelParameters() { // TODO Will remember last used process as a chache. final MonteCarloProcess process = null; final Map modelParameters = new TreeMap<>(); final TimeDiscretization timeDiscretizationForCurves = isInterpolateDiscountFactorsOnLiborPeriodDiscretization ? liborPeriodDiscretization : process.getTimeDiscretization(); // Add initial values for(int timeIndex=0; timeIndex= 0) { return getDiscountFactor(process, timeIndex); } else { // Interpolation final int timeIndexPrev = Math.min(-timeIndex-2, getLiborPeriodDiscretization().getNumberOfTimes()-2); final int timeIndexNext = timeIndexPrev+1; final double timePrev = timeDiscretizationForCurves.getTime(timeIndexPrev); final double timeNext = timeDiscretizationForCurves.getTime(timeIndexNext); final RandomVariable discountFactorPrev = getDiscountFactor(process, timeIndexPrev); final RandomVariable discountFactorNext = getDiscountFactor(process, timeIndexNext); return discountFactorPrev.mult(discountFactorNext.div(discountFactorPrev).pow((time-timePrev)/(timeNext-timePrev))); } } private RandomVariable getDiscountFactor(final MonteCarloProcess process, final int timeIndex) { final TimeDiscretization timeDiscretizationForCurves = isInterpolateDiscountFactorsOnLiborPeriodDiscretization ? liborPeriodDiscretization : process.getTimeDiscretization(); final double time = timeDiscretizationForCurves.getTime(timeIndex); synchronized(numeraireDiscountFactorForwardRates) { if(numeraireDiscountFactors.size() == 0) { final double dfInitial = discountCurve.getDiscountFactor(analyticModel, timeDiscretizationForCurves.getTime(0)); RandomVariable deterministicNumeraireAdjustment = randomVariableFactory.createRandomVariable(dfInitial); numeraireDiscountFactors.add(0, deterministicNumeraireAdjustment); for(int i=0; i= 0) { return getDiscountFactorFromForwardCurve(process, timeIndex); } else { final int timeIndexPrev = Math.min(-timeIndex-2, getLiborPeriodDiscretization().getNumberOfTimes()-2); final int timeIndexNext = timeIndexPrev+1; final double timePrev = timeDiscretizationForCurves.getTime(timeIndexPrev); final double timeNext = timeDiscretizationForCurves.getTime(timeIndexNext); final RandomVariable discountFactorPrev = getDiscountFactorFromForwardCurve(process, timeIndexPrev); final RandomVariable discountFactorNext = getDiscountFactorFromForwardCurve(process, timeIndexNext); return discountFactorPrev.mult(discountFactorNext.div(discountFactorPrev).pow((time-timePrev)/(timeNext-timePrev))); } } private RandomVariable getDiscountFactorFromForwardCurve(final MonteCarloProcess process, final int timeIndex) { synchronized(discountFactorFromForwardCurveCache) { if(discountFactorFromForwardCurveCache.size() <= timeIndex) { // Initialize cache final TimeDiscretization timeDiscretizationForCurves = isInterpolateDiscountFactorsOnLiborPeriodDiscretization ? liborPeriodDiscretization : process.getTimeDiscretization(); for(int i=discountFactorFromForwardCurveCache.size(); i<=timeIndex; i++) { RandomVariable dfAsRandomVariable; if(i == 0) { final double df = discountCurveFromForwardCurve.getDiscountFactor(analyticModel, timeDiscretizationForCurves.getTime(i)); dfAsRandomVariable = randomVariableFactory.createRandomVariable(df); } else { final RandomVariable dfPrevious = discountFactorFromForwardCurveCache.get(i-1); final RandomVariable forwardRate = getForwardRateInitialValue(process, i-1); dfAsRandomVariable = dfPrevious.div(forwardRate.mult(timeDiscretizationForCurves.getTimeStep(i-1)).add(1.0)); } discountFactorFromForwardCurveCache.add(dfAsRandomVariable); } } } return discountFactorFromForwardCurveCache.get(timeIndex); } private RandomVariable getForwardRateInitialValue(final MonteCarloProcess process, final int timeIndex) { synchronized(forwardRateCache) { if(forwardRateCache.size() <= timeIndex) { // Initialize cache final TimeDiscretization timeDiscretizationForCurves = isInterpolateDiscountFactorsOnLiborPeriodDiscretization ? liborPeriodDiscretization : process.getTimeDiscretization(); for(int i=forwardRateCache.size(); i<=timeIndex; i++) { final double dfPrev = discountCurveFromForwardCurve.getDiscountFactor(analyticModel, timeDiscretizationForCurves.getTime(i)); final double dfNext = discountCurveFromForwardCurve.getDiscountFactor(analyticModel, timeDiscretizationForCurves.getTime(i+1)); final RandomVariable forwardRate = randomVariableFactory.createRandomVariable((dfPrev / dfNext - 1.0) / timeDiscretizationForCurves.getTimeStep(i)); forwardRateCache.add(forwardRate); } } } return forwardRateCache.get(timeIndex); } private void readObject(final ObjectInputStream in) throws IOException, ClassNotFoundException { in.defaultReadObject(); /* * Init transient fields */ numeraireDiscountFactors = new ArrayList<>(); numeraireDiscountFactorForwardRates = new ArrayList<>(); discountFactorFromForwardCurveCache = new ArrayList<>(); forwardRateCache = new ArrayList<>(); } @Override public String toString() { return "HullWhiteModel [liborPeriodDiscretization=" + liborPeriodDiscretization + ", forwardCurveName=" + forwardCurveName + ", analyticModel=" + analyticModel + ", forwardRateCurve=" + forwardRateCurve + ", discountCurve=" + discountCurve + ", discountCurveFromForwardCurve=" + discountCurveFromForwardCurve + ", randomVariableFactory=" + randomVariableFactory + ", volatilityModel=" + volatilityModel + ", properties=" + properties + ", isInterpolateDiscountFactorsOnLiborPeriodDiscretization=" + isInterpolateDiscountFactorsOnLiborPeriodDiscretization + "]"; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy