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

net.finmath.singleswaprate.annuitymapping.BasicPiterbargAnnuityMapping Maven / Gradle / Ivy

package net.finmath.singleswaprate.annuitymapping;

import java.util.Arrays;

import net.finmath.marketdata.model.AnalyticModel;
import net.finmath.marketdata.model.volatilities.VolatilitySurface;
import net.finmath.marketdata.model.volatilities.VolatilitySurface.QuotingConvention;
import net.finmath.marketdata.products.SwapAnnuity;
import net.finmath.singleswaprate.model.VolatilityCubeModel;
import net.finmath.singleswaprate.model.volatilities.VolVolCube;
import net.finmath.singleswaprate.model.volatilities.VolatilityCube;
import net.finmath.singleswaprate.products.AnnuityDummyProduct;
import net.finmath.singleswaprate.products.NormalizingDummyProduct;
import net.finmath.time.Period;
import net.finmath.time.Schedule;
import net.finmath.time.ScheduleFromPeriods;

/**
 * Implements an annuity mapping following Vladimir Piterbarg's approach. This class does not take into account multi curve convexity adjustment.
 *
 * @author Christian Fries
 * @author Roland Bachl
 *
 */
public class BasicPiterbargAnnuityMapping implements AnnuityMapping {

	private static QuotingConvention quotingConvention = VolatilitySurface.QuotingConvention.VOLATILITYNORMAL;

	private final int numberOfPeriods;
	private final double[] periodLengths;

	private final double[] initialAnnuities;
	private final double[] initialSwapRates;

	private final double[] exponentialDriverMeans;
	private final double[] exponents;
	private final double[] denominators;

	private final double expectationCorrection;
	private final NormalizingFunction normalizer;

	/**
	 * Create the annuity mapping. When used without strike the volatilities are taken out of the cube at par swap rate.
	 *
	 * @param fixSchedule Fix leg schedule of the swap.
	 * @param floatSchedule Float leg schedule of the swap.
	 * @param model The model containing curve and cube.
	 * @param discountCurveName The name of the discount curve.
	 * @param volatilityCubeName The name of the volatility cube.
	 */
	public BasicPiterbargAnnuityMapping(final Schedule fixSchedule, final Schedule floatSchedule, final VolatilityCubeModel model, final String discountCurveName, final String volatilityCubeName) {
		this(fixSchedule, floatSchedule, Double.NaN, model, discountCurveName, volatilityCubeName);
	}

	/**
	 * Create the annuity mapping.
	 *
	 * @param fixSchedule Fix leg schedule of the swap.
	 * @param floatSchedule Float leg schedule of the swap.
	 * @param strike The strike of the product this annuity mapping is being created for.
	 * @param model The model containing curve and cube.
	 * @param discountCurveName The name of the discount curve.
	 * @param volatilityCubeName The name of the volatility cube.
	 */
	public BasicPiterbargAnnuityMapping(final Schedule fixSchedule, final Schedule floatSchedule, final double strike, final VolatilityCubeModel model,
			final String discountCurveName, final String volatilityCubeName) {
		this(fixSchedule, floatSchedule, strike, model, discountCurveName, volatilityCubeName, 0, 0, -1);
	}

	/**
	 * Create the annuity mapping.
	 *
	 * @param fixSchedule Fix leg schedule of the swap.
	 * @param floatSchedule Float leg schedule of the swap.
	 * @param strike The strike of the product this annuity mapping is being created for.
	 * @param model The model containing curve and cube.
	 * @param discountCurveName The name of the discount curve.
	 * @param volatilityCubeName The name of the volatility cube.
	 * @param lowerBound The lowest strike the Piterbarg annuity mapping may use during replication, when normalizing.
	 * @param upperBound The maximum strike the Piterbarg annuity mapping may use during replication, when normalizing.
	 * @param numberOfEvaluationPoints The number of points the replication may evaluate Piterbarg annuity mapping is normalizing.
	 */
	public BasicPiterbargAnnuityMapping(final Schedule fixSchedule, final Schedule floatSchedule, double strike, final VolatilityCubeModel model,
			final String discountCurveName, final String volatilityCubeName, final double lowerBound, final double upperBound, final int numberOfEvaluationPoints) {
		super();
		this.numberOfPeriods = fixSchedule.getNumberOfPeriods();

		final double maturity 	= fixSchedule.getPeriodStart(0);
		final double[] periodLengths	= new double[numberOfPeriods];
		final double[] periodEnds		= new double[numberOfPeriods];
		for(int index = 0; index < numberOfPeriods; index++) {
			periodLengths[index] 	= fixSchedule.getPeriodLength(index);
			periodEnds[index]		= fixSchedule.getPeriodEnd(index);
		}
		this.periodLengths 	= periodLengths;

		this.initialAnnuities = getAnnuities(fixSchedule, discountCurveName, model);
		this.initialSwapRates = getForwardSwapRates(fixSchedule, discountCurveName, model);

		final String volvolCubeName = "VolVolFrom" + volatilityCubeName;
		final VolatilityCube volvolCube = new VolVolCube(volvolCubeName, model.getVolatilityCube(volatilityCubeName).getReferenceDate(),
				volatilityCubeName, fixSchedule, initialSwapRates);

		if(Double.isNaN(strike)) {
			strike = initialSwapRates[numberOfPeriods-1];
		}

		//calculation of the various exp mu
		this.exponentialDriverMeans = findExponentialDriverMeans(periodEnds, maturity, strike, volvolCube, model);

		//calculation of arrays of the denominators and exponents of the individual summands in the function of the annuity mapping
		final double[] exponents    = new double[numberOfPeriods];
		final double[] denominators = new double[numberOfPeriods];
		Arrays.fill(denominators, 1.0);

		double currentSummand;
		double currentFactor;
		final double terminalVolvol = volvolCube.getValue(model, periodEnds[numberOfPeriods-1], maturity, strike, quotingConvention);

		for(int outerIndex = numberOfPeriods-1; outerIndex > -1; outerIndex--){

			currentSummand = volvolCube.getValue(model, periodEnds[outerIndex], maturity, strike, quotingConvention);
			currentFactor = (periodLengths[outerIndex] *initialSwapRates[outerIndex] +1.0) *exponentialDriverMeans[outerIndex];

			for(int innerIndex = 0; innerIndex <= outerIndex; innerIndex++){
				exponents[innerIndex]	 += currentSummand;
				denominators[innerIndex] *= currentFactor;
			}
			exponents[outerIndex] /= terminalVolvol;
		}
		this.exponents		= exponents;
		this.denominators 	= denominators;

		//calibrate the normalizing function to the measure of the annuity
		final NormalizingDummyProduct unscaledNormalizerDummy = new NormalizingDummyProduct(fixSchedule, floatSchedule, discountCurveName, null, volatilityCubeName,
				new ExponentialNormalizer(initialSwapRates[initialSwapRates.length-1], 1));
		if(numberOfEvaluationPoints > 0) {
			unscaledNormalizerDummy.setIntegrationParameters(lowerBound, upperBound, numberOfEvaluationPoints);
		}
		this.normalizer = new ExponentialNormalizer(initialSwapRates[initialSwapRates.length-1],
				1 / unscaledNormalizerDummy.getValue(fixSchedule.getFixing(0), model));

		//create a base mapping without correction of the expectation and calibrate the expectation correction
		final BasicPiterbargAnnuityMapping uncorrectedMapping = new BasicPiterbargAnnuityMapping(numberOfPeriods, periodLengths, initialAnnuities,
				initialSwapRates, exponentialDriverMeans, exponents, denominators, normalizer);
		final AnnuityDummyProduct uncorrectedAnnuityDummy = new AnnuityDummyProduct(fixSchedule, floatSchedule, discountCurveName, null, volatilityCubeName,
				uncorrectedMapping);
		if(numberOfEvaluationPoints > 0) {
			uncorrectedAnnuityDummy.setIntegrationParameters(lowerBound, upperBound, numberOfEvaluationPoints);
		}

		this.expectationCorrection = uncorrectedAnnuityDummy.getValue(fixSchedule.getFixing(0), model) - 1;
	}

	/**
	 * Private constructor to create a base copy which is used to determine the actual expectationCorrection.
	 *
	 * @param numberOfPeriods
	 * @param periodLengths
	 * @param initialAnnuities
	 * @param initialSwapRates
	 * @param exponentialDriverMeans
	 * @param exponents
	 * @param denominators
	 * @param normalizer
	 */
	private BasicPiterbargAnnuityMapping(final int numberOfPeriods, final double[] periodLengths, final double[] initialAnnuities,
			final double[] initialSwapRates, final double[] exponentialDriverMeans, final double[] exponents, final double[] denominators, final NormalizingFunction normalizer) {
		super();
		this.numberOfPeriods = numberOfPeriods;
		this.periodLengths = periodLengths;
		this.initialAnnuities = initialAnnuities;
		this.initialSwapRates = initialSwapRates;
		this.exponentialDriverMeans = exponentialDriverMeans;
		this.exponents = exponents;
		this.denominators = denominators;
		this.expectationCorrection = 0;
		this.normalizer = normalizer;
	}

	@Override
	public double getValue(final double swapRate) {

		double term  = periodLengths[numberOfPeriods-1] *swapRate +1.0;
		term 		/= periodLengths[numberOfPeriods-1] *initialSwapRates[numberOfPeriods -1] +1.0;
		term 		/= exponentialDriverMeans[numberOfPeriods -1];

		double value = 0.0;
		for(int index = 0; index < numberOfPeriods; index++){
			value += (Math.pow(term, -exponents[index])) *periodLengths[index] /denominators[index];
		}

		return initialAnnuities[numberOfPeriods-1]  /value - expectationCorrection * normalizer.getValue(swapRate);
	}

	@Override
	public double getFirstDerivative(final double swapRate) {

		double term  = periodLengths[numberOfPeriods-1] *swapRate +1.0;
		term 		/= periodLengths[numberOfPeriods-1] *initialSwapRates[numberOfPeriods -1] +1.0;
		term 		/= exponentialDriverMeans[numberOfPeriods -1];

		double value = 0.0;
		double innerDerivative = 0.0;
		for(int index = 0; index < numberOfPeriods; index++){
			value += (Math.pow(term, -exponents[index])) *periodLengths[index] /denominators[index];
			innerDerivative += (Math.pow(term, -exponents[index] -1)) *periodLengths[index]
					*-exponents[index] /denominators[index];
		}

		innerDerivative *= periodLengths[numberOfPeriods-1]  /exponentialDriverMeans[numberOfPeriods -1]
				/(periodLengths[numberOfPeriods-1] *initialSwapRates[numberOfPeriods -1] +1.0);

		return -initialAnnuities[numberOfPeriods-1] *innerDerivative /value /value - expectationCorrection * normalizer.getFirstDerivative(swapRate);
	}

	@Override
	public double getSecondDerivative(final double swapRate) {

		double term  = periodLengths[numberOfPeriods-1] *swapRate +1.0;
		term 		/= periodLengths[numberOfPeriods-1] *initialSwapRates[numberOfPeriods -1] +1.0;
		term 		/= exponentialDriverMeans[numberOfPeriods -1];

		double value = 0.0;
		double innerFirst = 0.0;
		double innerSecond = 0.0;
		for(int index = 0; index < numberOfPeriods; index++){
			value += (Math.pow(term, -exponents[0])) *periodLengths[index] /denominators[index];
			innerFirst +=(Math.pow(term, -exponents[index] -1)) *periodLengths[index]
					*-exponents[index] /denominators[index];
			innerSecond += (Math.pow(term, -exponents[index] -2)) *periodLengths[index]
					*exponents[index] *(exponents[index] +1.0) /denominators[index];
		}

		innerFirst *= periodLengths[numberOfPeriods-1] /(periodLengths[numberOfPeriods-1] *initialSwapRates[numberOfPeriods -1] +1.0)
				/exponentialDriverMeans[numberOfPeriods -1];

		innerSecond *=periodLengths[numberOfPeriods-1] *periodLengths[numberOfPeriods-1]
				/(periodLengths[numberOfPeriods-1] *initialSwapRates[numberOfPeriods -1] +1.0)
				/(periodLengths[numberOfPeriods-1] *initialSwapRates[numberOfPeriods -1] +1.0)
				/exponentialDriverMeans[numberOfPeriods -1] /exponentialDriverMeans[numberOfPeriods -1];

		return ((2.0 *innerFirst *innerFirst /value /value /value) - (innerSecond /value /value)) *initialAnnuities[numberOfPeriods-1]
				- expectationCorrection * normalizer.getSecondDerivative(swapRate);
	}

	/**
	 * Calculates the constants \(e^{\mu_j} \) for all sub tenors.
	 *
	 * @param timePoints
	 * @param maturity
	 * @param strike
	 * @param volvolCube
	 * @param model
	 * @return
	 */
	private double[] findExponentialDriverMeans(final double[] timePoints, final double maturity, final double strike, final VolatilityCube volvolCube,
			final VolatilityCubeModel model){

		final double[] exponentialMeans = new double[numberOfPeriods];
		final double[] volatilities = new double[numberOfPeriods];
		for(int index = 0; index < numberOfPeriods; index++){
			volatilities[index] = volvolCube.getValue(model, timePoints[index], maturity, strike, quotingConvention);
		}

		//initial ExponentialMean does not require the adjustment term
		exponentialMeans[0]  = Math.exp(maturity *0.5 *volatilities[0] *volatilities[0]);
		exponentialMeans[0] *= periodLengths[0] /initialAnnuities[0] /(periodLengths[0] *initialSwapRates[0] +1.0);

		double adjustment;

		//used to calculate the adjustment term in the loop
		final double[] volatilitySums = new double[numberOfPeriods];
		final double[] adjustmentSummands = new double[numberOfPeriods];
		for(int index = 0; index < numberOfPeriods; index++) {
			volatilitySums[index] = volatilities[0];
			adjustmentSummands[index] = 1.0;
		}

		for(int index = 1; index < numberOfPeriods; index++){

			//approximation adjustment roughly annuities[index-1]
			//			adjustment = initialAnnuities[index-1];

			// calculate adjustment
			adjustment = 0.0;
			for(int innerIndex = 0; innerIndex < index; innerIndex++){
				adjustmentSummands[innerIndex] /= exponentialMeans[index-1] *(periodLengths[index-1] *initialSwapRates[index-1] +1.0);
				volatilitySums[innerIndex] += volatilities[index];
			}
			for(int sumIndex = 0; sumIndex < index; sumIndex++){
				adjustment += periodLengths[sumIndex] *adjustmentSummands[sumIndex] *Math.exp(maturity *volatilitySums[sumIndex]
						*volatilitySums[sumIndex] /2);
			}

			//finalize exponentialMeans
			exponentialMeans[index]  = Math.exp(maturity *0.5 *volatilities[index] *volatilities[index]);
			exponentialMeans[index] *= periodLengths[index];
			exponentialMeans[index] += adjustment;
			exponentialMeans[index] /= initialAnnuities[index] *(periodLengths[index] *initialSwapRates[index] + 1.0);
		}
		return exponentialMeans;
	}

	/**
	 * Calculates initial annuities on all sub-tenors.
	 *
	 * @param schedule
	 * @param discountCurveName
	 * @param model
	 * @return
	 */
	private double[] getAnnuities(final Schedule schedule, final String discountCurveName, final AnalyticModel model){
		final double[] annuities = new double[schedule.getNumberOfPeriods()];
		for(int annuityIndex=0; annuityIndex < schedule.getNumberOfPeriods();annuityIndex++){
			final Period[] periods = new Period[annuityIndex +1];
			for(int periodIndex=0; periodIndex <= annuityIndex; periodIndex++){
				periods[periodIndex] = schedule.getPeriod(periodIndex);
			}
			final ScheduleFromPeriods partSchedule = new ScheduleFromPeriods(schedule.getReferenceDate(), schedule.getDaycountconvention(), periods);
			annuities[annuityIndex] = SwapAnnuity.getSwapAnnuity(schedule.getFixing(0), partSchedule, model.getDiscountCurve(discountCurveName), model);
		}
		return annuities;
	}

	/**
	 * Calculates initial swap rates on all sub-tenors, after annuities have been determined.
	 *
	 * @param schedule
	 * @param discountCurveName
	 * @param model
	 * @return
	 */
	private double[] getForwardSwapRates(final Schedule schedule, final String discountCurveName, final AnalyticModel model){
		final double[] swapRates = new double[schedule.getNumberOfPeriods()];
		final double discount = model.getDiscountCurve(discountCurveName).getDiscountFactor(model, schedule.getFixing(0));
		for(int swapRateIndex=0; swapRateIndex < schedule.getNumberOfPeriods();swapRateIndex++){
			swapRates[swapRateIndex] = 1 - model.getDiscountCurve(discountCurveName).getDiscountFactor(model, schedule.getPayment(swapRateIndex)) / discount;
			swapRates[swapRateIndex] /= initialAnnuities[swapRateIndex];
		}
		return swapRates;
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy