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

net.finmath.montecarlo.interestrate.models.funding.FundingCapacityWithMemory Maven / Gradle / Ivy

package net.finmath.montecarlo.interestrate.models.funding;

import java.util.Map;
import java.util.Set;
import java.util.SortedMap;

import net.finmath.exception.CalculationException;
import net.finmath.montecarlo.interestrate.TermStructureMonteCarloSimulationModel;
import net.finmath.montecarlo.interestrate.products.components.AbstractProductComponent;
import net.finmath.stochastic.RandomVariable;
import net.finmath.stochastic.Scalar;

/**
 * Models the notional dependent survival probability and default compensation
 * of a funding capacity (funding provider)
 * using a piecewise constant function for the instantaneous survival probability.
 *
 * The piecewise constant instantaneous survival probability has to be provided
 * by a SortedMap<Double, Double> instantaneouseSurvivalProbability.
 *
 * This map defines the mapping \( x_{i} \mapsto q_{i} \). Defining
 * \[ q(x) = q_{i} \text{\ for\ } x \in (x_{i-1}-x_{i}] \] the
 * getDefaultFactors method of this class calculates
 * for a given argument \( (t,x) \):
 * 
*
* the effective survival probability *
*
* \[ \frac{1}{x} \int_{a}^{a+x} q(\xi) \mathrm{d}\xi \], * * where a denotes the current level of fund provided by this capacity, and *
*
* the effective default compensation factor R, such that *
*
* \[ \frac{1}{x} \int_{a}^{a+R x} q(\xi) \mathrm{d}\xi \ = \ 1 \], *
*
* * Important: * *
    *
  • * Since the class keeps track of past fundings * used, it is mandatory that the factors are calculated in * time-sequential order. *
  • *
  • * The map instantaneouseSurvivalProbability \( x_{i} \mapsto q_{i} \) * defines the survival probability for \( (x_{i-1},x_{i} \). * * For funding above the last discretization point \( x_{n-1} \) a value of \( q_{n} = 0 \) * is used. * Hence, to avoid this extrapolation, set a very large value of \( x_{n-1} \), e.g. * Double.MAX_VALLUE *
  • *
* * @author Christian Fries */ public class FundingCapacityWithMemory extends AbstractProductComponent implements FundingCapacity { private static final long serialVersionUID = 6863200178588875665L; private final SortedMap instantaneousSurvivalProbability; private Double currentTime; private RandomVariable currentCapacity; public FundingCapacityWithMemory(String currency, RandomVariable intialCapacity, SortedMap instantaneouseSurvivalProbability) { super(currency); this.currentTime = 0.0; this.currentCapacity = intialCapacity; this.instantaneousSurvivalProbability = instantaneouseSurvivalProbability; } /** * Apply a new funding requirement to this funding capacity * and return the associated DefaultFactors. * * @param time The time at which the funding is required. * @param fundingRequirement The required funding. * @return A DefaultFactors that reflects the amount that has to be contracted to secure the funding. */ @Override public DefaultFactors getDefaultFactors(double time, RandomVariable fundingRequirement) { /* * Determine integral bounds (synchronized for thread safety) */ RandomVariable fundingIntervalLeft, fundingIntervalRight; synchronized (currentTime) { if(time < currentTime) { throw new IllegalStateException("The method must be called in time-successive order."); } currentTime = time; /* * The fundingRequirement may be negative, in which case funding is returned to the provider. * We first calculate the lower and upper integral bounds from the fundingRequirement. * The integral calculated is always positive, since we require only the factor. */ final RandomVariable newCapacity = currentCapacity.add(fundingRequirement); fundingIntervalLeft = currentCapacity.cap(newCapacity); // min(current,new) fundingIntervalRight = currentCapacity.floor(newCapacity); // max(current,new) currentCapacity = newCapacity; } RandomVariable integratedSurvivalProbability = new Scalar(0.0); RandomVariable integratedDefaultCompensation = new Scalar(0.0); double previousFundingLevel = -Double.MAX_VALUE; double previousProvidedLevel = -Double.MAX_VALUE; for(final Map.Entry entry : instantaneousSurvivalProbability.entrySet()) { final double fundingLevel = entry.getKey(); final double survivalProbability = entry.getValue(); final double providedLevel = Math.max(previousProvidedLevel,0) + (fundingLevel-Math.max(previousFundingLevel,0)) * survivalProbability; integratedDefaultCompensation = integratedDefaultCompensation.add( fundingIntervalRight.cap(providedLevel) .sub(fundingIntervalLeft.floor(previousProvidedLevel)) .floor(0.0) .div(survivalProbability)); integratedSurvivalProbability = integratedSurvivalProbability.add( fundingIntervalRight.cap(fundingLevel) .sub(fundingIntervalLeft.floor(previousFundingLevel)) .floor(0.0) .mult(survivalProbability)); previousFundingLevel = fundingLevel; previousProvidedLevel = providedLevel; } // The cap is used to map to avoid 0*infty to zero. final RandomVariable oneOverFundingAmount = fundingIntervalRight.sub(fundingIntervalLeft).invert().cap(Double.MAX_VALUE); integratedSurvivalProbability = integratedSurvivalProbability.mult(oneOverFundingAmount); integratedDefaultCompensation = integratedDefaultCompensation.mult(oneOverFundingAmount); return new DefaultFactors(integratedSurvivalProbability, integratedDefaultCompensation); } @Deprecated public RandomVariable getDefaultCompensationForRequiredFunding(double time, RandomVariable fundingRequirement) { RandomVariable fundingIntervalLeft, fundingIntervalRight; synchronized (currentTime) { if(time < currentTime) { throw new IllegalStateException("The method must be called in time-successive order."); } currentTime = time; /* * The fundingRequirement may be negative, in which case funding is retured to the provides. * We first calculate the lower and upper integral bounds from the fundingRequirement. * The integral calculated is always positive, the correct sign of the integral will be checked later. */ final RandomVariable newCapacity = currentCapacity.add(fundingRequirement); fundingIntervalLeft = currentCapacity.cap(newCapacity); // min(current,new) fundingIntervalRight = currentCapacity.floor(newCapacity); // max(current,new) currentCapacity = newCapacity; } RandomVariable integratedSurvivalProbability = new Scalar(0.0); double previousFundingLevel = -Double.MAX_VALUE; final double previousProvidedLevel = -Double.MAX_VALUE; for(final Map.Entry entry : instantaneousSurvivalProbability.entrySet()) { final double fundingLevel = entry.getKey(); final double survivalProbability = entry.getValue(); final double providedLevel = Math.max(previousProvidedLevel,0) + (fundingLevel-Math.max(previousFundingLevel,0)) * survivalProbability; integratedSurvivalProbability = integratedSurvivalProbability.add( fundingIntervalRight.cap(providedLevel) .sub(fundingIntervalLeft.floor(previousProvidedLevel)) .floor(0.0) .div(survivalProbability)); previousFundingLevel = fundingLevel; } integratedSurvivalProbability = integratedSurvivalProbability.div(fundingIntervalRight.sub(fundingIntervalLeft)); return integratedSurvivalProbability; } @Deprecated public RandomVariable getSurvivalProbabilityRequiredFunding(double time, RandomVariable fundingRequirement) { RandomVariable fundingIntervalLeft, fundingIntervalRight; synchronized (currentTime) { if(time < currentTime) { throw new IllegalStateException("The method getSurvivalProbabilityRequiredFunding must be called in successive order."); } currentTime = time; /* * The fundingRequirement may be negative, in which case funding is retured to the provides. * We first calculate the lower and upper integral bounds from the fundingRequirement. * The integral calculated is always positive, the correct sign of the integral will be checked later. */ final RandomVariable newCapacity = currentCapacity.add(fundingRequirement); fundingIntervalLeft = currentCapacity.cap(newCapacity); // min(current,new) fundingIntervalRight = currentCapacity.floor(newCapacity); // max(current,new) currentCapacity = newCapacity; } RandomVariable integratedSurvivalProbability = new Scalar(0.0); double previousFundingLevel = -Double.MAX_VALUE; for(final Map.Entry entry : instantaneousSurvivalProbability.entrySet()) { final double fundingLevel = entry.getKey(); final double survivalProbability = entry.getValue(); integratedSurvivalProbability = integratedSurvivalProbability.add( fundingIntervalRight.cap(fundingLevel) .sub(fundingIntervalLeft.floor(previousFundingLevel)) .floor(0.0) .mult(survivalProbability)); previousFundingLevel = fundingLevel; } integratedSurvivalProbability = integratedSurvivalProbability.div(fundingIntervalRight.sub(fundingIntervalLeft)); return integratedSurvivalProbability; } public RandomVariable getCurrentFundingLevel() { return currentCapacity; } @Override public Set queryUnderlyings() { return null; } @Override public RandomVariable getValue(double evaluationTime, TermStructureMonteCarloSimulationModel model) throws CalculationException { throw new UnsupportedOperationException(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy