net.finmath.montecarlo.interestrate.models.LIBORMarketModelWithTenorRefinement 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.util.ArrayList; import java.util.Arrays; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import net.finmath.exception.CalculationException; import net.finmath.marketdata.model.AnalyticModel; import net.finmath.marketdata.model.curves.DiscountCurve; import net.finmath.marketdata.model.curves.ForwardCurve; import net.finmath.montecarlo.interestrate.CalibrationProduct; import net.finmath.montecarlo.interestrate.TermStructureModel; import net.finmath.montecarlo.interestrate.models.covariance.TermStructureCovarianceModelInterface; import net.finmath.montecarlo.interestrate.models.covariance.TermStructureCovarianceModelParametric; import net.finmath.montecarlo.model.AbstractProcessModel; import net.finmath.montecarlo.process.MonteCarloProcess; import net.finmath.stochastic.RandomVariable; import net.finmath.time.TimeDiscretization; import net.finmath.time.TimeDiscretizationFromArray; /** * Implements a discretized Heath-Jarrow-Morton model / LIBOR market model with dynamic tenor refinement, see * https://papers.ssrn.com/sol3/papers.cfm?abstract_id=2884699. * *
or
* In its default case the class specifies a multi-factor LIBOR market model, that is * \( L_{j} = \frac{1}{T_{j+1}-T_{j}} ( exp(Y_{j}) - 1 ) \), where * \[ * dY_{j} = \mu_{j} dt + \lambda_{1,j} dW_{1} + \ldots + \lambda_{m,j} dW_{m} * \] *
* The model uses anAbstractLIBORCovarianceModel
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 *
* 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 (through a local volatility model). *
* The simulation is performed under spot measure, that is, the numeraire * is \( N(T_{i}) = \prod_{j=0}^{i-1} (1 + L(T_{j},T_{j+1};T_{j}) (T_{j+1}-T_{j})) \). * * The mapproperties
allows to configure the model. The following keys may be used: **
*- *
*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. * * @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 https://papers.ssrn.com/sol3/papers.cfm?abstract_id=2884699 */ public class LIBORMarketModelWithTenorRefinement extends AbstractProcessModel implements TermStructureModel { public enum Driftapproximation { EULER, LINE_INTEGRAL, PREDICTOR_CORRECTOR } private final TimeDiscretization[] liborPeriodDiscretizations; private final Integer[] numberOfDiscretizationIntervalls; private String forwardCurveName; private AnalyticModel curveModel; private ForwardCurve forwardRateCurve; private DiscountCurve discountCurve; private TermStructureCovarianceModelInterface covarianceModel; // Cache for the numeraires, needs to be invalidated if process changes private final ConcurrentHashMapnumeraires; private MonteCarloProcess numerairesProcess = null; /** * Creates a model for given covariance. * * Creates a discretized Heath-Jarrow-Morton model / LIBOR market model with dynamic tenor refinement, see * https://papers.ssrn.com/sol3/papers.cfm?abstract_id=2884699. *
* 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 liborPeriodDiscretizations A vector of tenor discretizations of the interest rate curve into forward rates (tenor structure), finest first. * @param numberOfDiscretizationIntervalls A vector of number of periods to be taken from the liborPeriodDiscretizations. * @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 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- *
*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 LIBORMarketModelWithTenorRefinement( TimeDiscretization[] liborPeriodDiscretizations, Integer[] numberOfDiscretizationIntervalls, AnalyticModel analyticModel, ForwardCurve forwardRateCurve, DiscountCurve discountCurve, TermStructureCovarianceModelInterface covarianceModel, CalibrationProduct[] calibrationProducts, Mapproperties ) throws CalculationException { Map calibrationParameters = null; if(properties != null && properties.containsKey("calibrationParameters")) { calibrationParameters = (Map )properties.get("calibrationParameters"); } this.liborPeriodDiscretizations = liborPeriodDiscretizations; this.numberOfDiscretizationIntervalls = numberOfDiscretizationIntervalls; this.curveModel = analyticModel; this.forwardRateCurve = forwardRateCurve; this.discountCurve = discountCurve; this.covarianceModel = covarianceModel; // Perform calibration, if data is given if(calibrationProducts != null && calibrationProducts.length > 0) { TermStructureCovarianceModelParametric covarianceModelParametric = null; try { covarianceModelParametric = (TermStructureCovarianceModelParametric)covarianceModel; } catch(Exception e) { throw new ClassCastException("Calibration is currently restricted to parametric covariance models (TermStructureCovarianceModelParametricInterface)."); } this.covarianceModel = covarianceModelParametric.getCloneCalibrated(this, calibrationProducts, calibrationParameters); } numeraires = new ConcurrentHashMap<>(); } /** * Return the numeraire at a given time. * * The numeraire is provided for interpolated points. If requested on points which are not * part of the tenor discretization, the numeraire uses a linear interpolation of the reciprocal * value. See ISBN 0470047224 for details. * * @param time Time time 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 { int timeIndex = liborPeriodDiscretizations[0].getTimeIndex(time); TimeDiscretization liborPeriodDiscretization = liborPeriodDiscretizations[0]; if(timeIndex < 0) { // Interpolation of Numeraire: log linear interpolation. int upperIndex = -timeIndex-1; int lowerIndex = upperIndex-1; if(lowerIndex < 0) { throw new IllegalArgumentException("Numeraire requested for time " + time + ". Unsupported"); } double alpha = (time-liborPeriodDiscretization.getTime(lowerIndex)) / (liborPeriodDiscretization.getTime(upperIndex) - liborPeriodDiscretization.getTime(lowerIndex)); RandomVariable numeraire = getNumeraire(liborPeriodDiscretization.getTime(upperIndex)).log().mult(alpha).add(getNumeraire(liborPeriodDiscretization.getTime(lowerIndex)).log().mult(1.0-alpha)).exp(); /* * Adjust for discounting, i.e. funding or collateralization */ if(discountCurve != null) { // This includes a control for zero bonds double deterministicNumeraireAdjustment = numeraire.invert().getAverage() / discountCurve.getDiscountFactor(curveModel, time); numeraire = numeraire.mult(deterministicNumeraireAdjustment); } return numeraire; } /* * Calculate the numeraire, when time is part of liborPeriodDiscretization */ /* * Check if numeraire cache is values (i.e. process did not change) */ if(getProcess() != numerairesProcess) { numeraires.clear(); numerairesProcess = getProcess(); } /* * Check if numeraire is part of the cache */ RandomVariable numeraire = numeraires.get(timeIndex); if(numeraire == null) { /* * Calculate the numeraire for timeIndex */ if(timeIndex == 0) { numeraire = getProcess().getStochasticDriver().getRandomVariableForConstant(1.0); } else { // Initialize to previous numeraire numeraire = getNumeraire(liborPeriodDiscretizations[0].getTime(timeIndex-1)); double periodStart = liborPeriodDiscretizations[0].getTime(timeIndex-1); double periodEnd = liborPeriodDiscretizations[0].getTime(timeIndex); RandomVariable libor = getLIBOR(periodStart, periodStart, periodEnd); numeraire = numeraire.accrue(libor, periodEnd-periodStart); } // Cache the numeraire numeraires.put(timeIndex, numeraire); } /* * Adjust for discounting, i.e. funding or collateralization */ if(discountCurve != null) { // This includes a control for zero bonds double deterministicNumeraireAdjustment = numeraire.invert().getAverage() / discountCurve.getDiscountFactor(curveModel, time); numeraire = numeraire.mult(deterministicNumeraireAdjustment); } return numeraire; } @Override public RandomVariable[] getInitialState() { RandomVariable[] initialStateRandomVariable = new RandomVariable[getNumberOfComponents()]; for(int componentIndex=0; componentIndexMeasure.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.LIBORMarketModelWithTenorRefinement#getNumeraire(double) The calculation of the drift is consistent with the calculation of the numeraire ingetNumeraire
. * @see net.finmath.montecarlo.interestrate.models.LIBORMarketModelWithTenorRefinement#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); double timeStep = getTimeDiscretization().getTimeStep(timeIndex); double timeNext = getTime(timeIndex+1); RandomVariable zero = getProcess().getStochasticDriver().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=0; componentIndex0) { return weight2 / covarianceModel.getScaledTenorTime(periodStart, periodEnd) - weight1 / covarianceModel.getScaledTenorTime(periodStartPrevious, periodEndPrevious); } else { return weight2 / covarianceModel.getScaledTenorTime(periodStart, periodEnd); } } @Override public RandomVariable[] getFactorLoading(int timeIndex, int componentIndex, RandomVariable[] realizationAtTimeIndex) { RandomVariable zero = getProcess().getStochasticDriver().getRandomVariableForConstant(0.0); if(componentIndex < getNumberOfLibors()) { TimeDiscretization liborPeriodDiscretization = getLiborPeriodDiscretization(getTime(timeIndex)); TimeDiscretization liborPeriodDiscretizationNext = getLiborPeriodDiscretization(getTime(timeIndex+1)); double periodStart = liborPeriodDiscretizationNext.getTime(componentIndex); double periodEnd = liborPeriodDiscretizationNext.getTime(componentIndex+1); RandomVariable[] factorLoadingVector = covarianceModel.getFactorLoading(getTime(timeIndex), periodStart, periodEnd, liborPeriodDiscretization, realizationAtTimeIndex, this); return factorLoadingVector; } else { RandomVariable[] zeros = new RandomVariable[getProcess().getStochasticDriver().getNumberOfFactors()]; Arrays.fill(zeros, zero); return zeros; } } @Override public RandomVariable applyStateSpaceTransform(int componentIndex, RandomVariable randomVariable) { return randomVariable; } @Override public RandomVariable applyStateSpaceTransformInverse(int componentIndex, RandomVariable randomVariable) { return randomVariable; } /* (non-Javadoc) * @see net.finmath.montecarlo.model.ProcessModel#getRandomVariableForConstant(double) */ @Override public RandomVariable getRandomVariableForConstant(double value) { return getProcess().getStochasticDriver().getRandomVariableForConstant(value); } private TimeDiscretization getLiborPeriodDiscretization(double time) { ArrayList tenorTimes = new ArrayList<>(); double firstTime = liborPeriodDiscretizations[0].getTime(liborPeriodDiscretizations[0].getTimeIndexNearestLessOrEqual(time)); double lastTime = firstTime; tenorTimes.add(firstTime); for(int discretizationLevelIndex = 0; discretizationLevelIndex = liborPeriodDiscretizations[discretizationLevelIndex].getNumberOfTimes()) { break; } lastTime = liborPeriodDiscretizations[discretizationLevelIndex].getTime(tentorIntervallStartIndex+tenorIntervall); // round to liborPeriodDiscretizations[0] lastTime = liborPeriodDiscretizations[0].getTime(liborPeriodDiscretizations[0].getTimeIndexNearestLessOrEqual(lastTime)); tenorTimes.add(lastTime); } } return new TimeDiscretizationFromArray(tenorTimes); } public RandomVariable getStateVariableForPeriod(TimeDiscretization liborPeriodDiscretization, RandomVariable[] stateVariables, double periodStart, double periodEnd) { int periodStartIndex = liborPeriodDiscretization.getTimeIndex(periodStart); int periodEndIndex = liborPeriodDiscretization.getTimeIndex(periodEnd); RandomVariable stateVariableSum = this.getProcess().getStochasticDriver().getRandomVariableForConstant(0.0); if(periodStartIndex < 0) { periodStartIndex = -periodStartIndex-1; if(periodStartIndex >= liborPeriodDiscretization.getNumberOfTimes()) { throw new IllegalArgumentException(); } RandomVariable stateVariable = stateVariables[periodStartIndex-1]; double shortPeriodEnd = liborPeriodDiscretization.getTime(periodStartIndex); double tenorRefinementWeight = getWeightForTenorRefinement(liborPeriodDiscretization.getTime(periodStartIndex-1), shortPeriodEnd, periodStart, shortPeriodEnd); RandomVariable integratedVariance = stateVariables[getNumberOfLibors()+periodStartIndex-1]; double tenor = covarianceModel.getScaledTenorTime(periodStart, shortPeriodEnd); stateVariableSum = stateVariableSum.addProduct(stateVariable.addProduct(integratedVariance, tenorRefinementWeight), tenor); } if(periodEndIndex < 0) { periodEndIndex = -periodEndIndex-1; RandomVariable stateVariable = stateVariables[periodEndIndex-1]; double shortPeriodStart = liborPeriodDiscretization.getTime(periodEndIndex-1); double tenorRefinementWeight = getWeightForTenorRefinement(shortPeriodStart, liborPeriodDiscretization.getTime(periodEndIndex), shortPeriodStart, periodEnd); RandomVariable integratedVariance = stateVariables[getNumberOfLibors()+periodEndIndex-1]; double tenor = covarianceModel.getScaledTenorTime(shortPeriodStart, periodEnd); stateVariableSum = stateVariableSum.addProduct(stateVariable.addProduct(integratedVariance, tenorRefinementWeight), tenor); periodEndIndex--; } for(int periodIndex = periodStartIndex; periodIndex = liborPeriodDiscretization.getNumberOfTimes()) { throw new IllegalArgumentException(); } RandomVariable stateVariable = getProcessValue(timeIndex, periodStartIndex-1); double shortPeriodEnd = liborPeriodDiscretization.getTime(periodStartIndex); double tenorRefinementWeight = getWeightForTenorRefinement(liborPeriodDiscretization.getTime(periodStartIndex-1), shortPeriodEnd, periodStart, shortPeriodEnd); RandomVariable integratedVariance = getIntegratedVariance(timeIndex, liborPeriodDiscretization.getTime(periodStartIndex-1), liborPeriodDiscretization.getTime(periodStartIndex)); stateVariableSum = stateVariableSum.addProduct(stateVariable.addProduct(integratedVariance, tenorRefinementWeight), covarianceModel.getScaledTenorTime(periodStart, shortPeriodEnd)); } if(periodEndIndex < 0) { periodEndIndex = -periodEndIndex-1; RandomVariable stateVariable = getProcessValue(timeIndex, periodEndIndex-1); double shortPeriodStart = liborPeriodDiscretization.getTime(periodEndIndex-1); double tenorRefinementWeight = getWeightForTenorRefinement(shortPeriodStart, liborPeriodDiscretization.getTime(periodEndIndex), shortPeriodStart, periodEnd); RandomVariable integratedVariance = getIntegratedVariance(timeIndex, liborPeriodDiscretization.getTime(periodEndIndex-1), liborPeriodDiscretization.getTime(periodEndIndex)); stateVariableSum = stateVariableSum.addProduct(stateVariable.addProduct(integratedVariance, tenorRefinementWeight), covarianceModel.getScaledTenorTime(shortPeriodStart,periodEnd)); periodEndIndex--; } for(int periodIndex = periodStartIndex; periodIndex properties = new HashMap (); properties.put("measure", measure.name()); properties.put("stateSpace", stateSpace.name()); return new LIBORMarketModelWithTenorRefinement(getLiborPeriodDiscretization(), getForwardRateCurve(), getDiscountCurve(), 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; } @Override public TermStructureModel getCloneWithModifiedData(Map dataModified) throws CalculationException { CalibrationProduct[] calibrationItems = null; Map properties = null; TermStructureCovarianceModelInterface covarianceModel = this.covarianceModel; if(dataModified.containsKey("covarianceModel")) { covarianceModel = (TermStructureCovarianceModelInterface)dataModified.get("covarianceModel"); } return new LIBORMarketModelWithTenorRefinement(liborPeriodDiscretizations, numberOfDiscretizationIntervalls, curveModel, forwardRateCurve, discountCurve, covarianceModel, calibrationItems, properties); } /** * Returns the term structure covariance model. * * @return the term structure covariance model. */ public TermStructureCovarianceModelInterface getCovarianceModel() { return covarianceModel; } }
© 2015 - 2025 Weber Informatics LLC | Privacy Policy