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

net.finmath.equities.models.SviVolatilitySmile Maven / Gradle / Ivy

package net.finmath.equities.models;

import java.time.LocalDate;
import java.util.ArrayList;
import java.util.Collections;
import java.util.stream.Collectors;

/**
 * Class that implements the smile-specific parts of the SVI volatility parametrization
 * from Gatheral's 2013 paper.
 *
 * @author Andreas Grotz
 */

public class SviVolatilitySmile {

	private final double a;
	private final double b;
	private final double rho;
	private final double m;
	private final double sigma;
	private final LocalDate smileDate;

	public SviVolatilitySmile(LocalDate date, double a, double b, double rho, double m, double sigma) {
		this.a = a;
		this.b = b;
		this.rho = rho;
		this.m = m;
		this.sigma = sigma;
		smileDate = date;
		validate();
	}

	public static double sviTotalVariance(
			double logStrike, double a, double b, double rho, double m, double sigma)
	{
		final var kShifted = logStrike - m;
		return a + b * (rho * kShifted + Math.sqrt(kShifted * kShifted + sigma * sigma));
	}

	public static double sviVolatility(
			double logStrike, double a, double b, double rho, double m, double sigma, double ttm)
	{
		return Math.sqrt(sviTotalVariance(logStrike, a, b, rho, m, sigma) / ttm);
	}

	public static double[] sviInitialGuess(ArrayList logStrikes, ArrayList totalVariances)
	{
		// Use the Jump Wing parametrization from Gatheral's 2013 paper to derive an initial guess
		final var nPoints = logStrikes.size();
		assert nPoints >= 5 : "An initial guess for SVI is not sensible with less than 5 points.";
		final var minIndex = totalVariances.indexOf(Collections.min(totalVariances));
		final var k0 = logStrikes.get(minIndex);
		if(k0 == 0.0)
		{
			final var atmIndex = logStrikes.indexOf(0.0);
			final var w = totalVariances.get(atmIndex);
			final var d2wdk2 = 2 * (totalVariances.get(atmIndex + 1) * totalVariances.get(atmIndex - 1) - 2 * w)
					/ (Math.pow(logStrikes.get(atmIndex + 1), 2) + Math.pow(logStrikes.get(atmIndex - 1), 2));
			final var c = (totalVariances.get(nPoints - 1) - totalVariances.get(nPoints - 2))
					/ (logStrikes.get(nPoints - 1) - logStrikes.get(nPoints - 2));
			final var p = (totalVariances.get(0) - totalVariances.get(1))
					/ (logStrikes.get(1) - logStrikes.get(0));
			final var b = 0.5 * (c + p);
			final var rho = 1 - 2 * p / (c + p);
			final var m = b * (1 - rho * rho) * Math.abs(rho) / d2wdk2;
			final var sigma = m * Math.sqrt(1 - rho * rho) / rho;
			final var a = w - b * sigma * Math.sqrt(1 - rho * rho);
			return new double[] {a, b, rho, m, sigma};
		}
		else
		{
			final var wMin = totalVariances.get(minIndex);
			double w, dwdk;
			if (logStrikes.contains(0.0))
			{
				final var atmIndex = logStrikes.indexOf(0.0);
				w = totalVariances.get(atmIndex);
				dwdk = 0.5 * ((totalVariances.get(atmIndex + 1) - w) / (logStrikes.get(atmIndex + 1))
						+ (totalVariances.get(atmIndex - 1) - w) / (logStrikes.get(atmIndex - 1)));
			}
			else
			{
				final var maxNegIndex = logStrikes.indexOf(
						Collections.max(logStrikes.stream().filter(s -> s < 0.0).collect(Collectors.toList())));
				dwdk = (totalVariances.get(maxNegIndex + 1) - totalVariances.get(maxNegIndex))
						/ (logStrikes.get(maxNegIndex + 1) - logStrikes.get(maxNegIndex));
				w = totalVariances.get(maxNegIndex) - dwdk * logStrikes.get(maxNegIndex);
			}
			final var c = (totalVariances.get(nPoints - 1) - totalVariances.get(nPoints - 2))
					/ (logStrikes.get(nPoints - 1) - logStrikes.get(nPoints - 2));
			final var p = (totalVariances.get(0) - totalVariances.get(1))
					/ (logStrikes.get(1) - logStrikes.get(0));

			final var b = 0.5 * (c + p);
			final var rho = 1 - 2 * p / (c + p);
			final var beta = rho - dwdk / b;
			final var alpha = Math.signum(beta) * Math.sqrt(1 / beta / beta - 1);
			final var m = (w - wMin) / b / (Math.signum(alpha) * Math.sqrt(1 + alpha * alpha)
					- alpha * Math.sqrt(1 - rho * rho) - rho);
			final var sigma = alpha * m;
			final var a = wMin - b * sigma * Math.sqrt(1 - rho * rho);
			return new double[] {a, b, rho, m, sigma};
		}
	}

	public void validate()
	{
		assert getB() >= 0.0;
		assert Math.abs(getRho()) < 1.0;
		assert getSigma() > 0.0;
		assert getA() + getB() * getSigma() * Math.sqrt(1.0 - getRho() * getRho()) >= 0.0;

	}

	public double getTotalVariance(double logStrike)
	{
		return sviTotalVariance(logStrike, getA(), getB(), getRho(), getM(), getSigma());
	}

	public double getTotalVariance(double strike, double forward)
	{
		return sviTotalVariance(Math.log(strike/forward), getA(), getB(), getRho(), getM(), getSigma());
	}

	public double getVolatility(double strike, double forward, double timeToExpiry)
	{
		return sviVolatility(Math.log(strike/forward), getA(), getB(), getRho(), getM(), getSigma(), timeToExpiry);
	}

	public LocalDate getSmileDate() {
		return smileDate;
	}

	public double getSigma() {
		return sigma;
	}

	public double getM() {
		return m;
	}

	public double getRho() {
		return rho;
	}

	public double getB() {
		return b;
	}

	public double getA() {
		return a;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy