Package com.opengamma.analytics.financial.model.volatility.smile.function

Source Code of com.opengamma.analytics.financial.model.volatility.smile.function.SABRPaulotVolatilityFunction

/**
* Copyright (C) 2009 - present by OpenGamma Inc. and the OpenGamma group of companies
*
* Please see distribution for license.
*/
package com.opengamma.analytics.financial.model.volatility.smile.function;

import static com.opengamma.analytics.math.FunctionUtils.square;

import org.apache.commons.lang.NotImplementedException;
import org.apache.commons.lang.Validate;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.opengamma.analytics.financial.model.option.pricing.analytic.formula.EuropeanVanillaOption;
import com.opengamma.analytics.math.function.Function1D;
import com.opengamma.util.CompareUtils;

/**
* Expansion from Paulot, Louis, Asymptotic Implied Volatility at the Second Order With Applications to the SABR Model (2009)
* <b>DO NOT USE This formulating gives very odd (i.e. wrong) smiles for certain parameters. It is not clear whether this is a problem with the actual paper or the
* Implementation.  </b>
*/
public class SABRPaulotVolatilityFunction extends VolatilityFunctionProvider<SABRFormulaData> {
  private static final VolatilityFunctionProvider<SABRFormulaData> HAGAN = new SABRHaganVolatilityFunction();

  private static final Logger s_logger = LoggerFactory.getLogger(SABRPaulotVolatilityFunction.class);

  private static final double CUTOFF_MONEYNESS = 1e-6;
  private static final double EPS = 1e-15;

  @Override
  public Function1D<SABRFormulaData, Double> getVolatilityFunction(final EuropeanVanillaOption option, final double forward) {
    Validate.notNull(option, "option");
    final double strike = option.getStrike();

    final double cutoff = forward * CUTOFF_MONEYNESS;
    final double k;
    if (strike < cutoff) {
      s_logger.info("Given strike of " + strike + " is less than cutoff at " + cutoff + ", therefore the strike is taken as " + cutoff);
      k = cutoff;
    } else {
      k = strike;
    }

    final double t = option.getTimeToExpiry();
    return new Function1D<SABRFormulaData, Double>() {

      @SuppressWarnings("synthetic-access")
      @Override
      public final Double evaluate(final SABRFormulaData data) {
        Validate.notNull(data, "data");
        final double alpha = data.getAlpha();
        final double beta = data.getBeta();
        final double rho = data.getRho();
        final double nu = data.getNu();

        double sigma0, sigma1;

        final double beta1 = 1 - beta;

        final double x = Math.log(k / forward);
        if (CompareUtils.closeEquals(nu, 0, EPS)) {
          if (CompareUtils.closeEquals(beta, 1.0, EPS)) {
            return alpha; // this is just log-normal
          }
          throw new NotImplementedException("Have not implemented the case where nu = 0, beta != 0");
        }

        // the formula behaves very badly close to ATM
        if (CompareUtils.closeEquals(x, 0.0, 1e-3)) {
          final double delta = 1.01e-3;
          final double a0 = (HAGAN.getVolatilityFunction(option, forward)).evaluate(data);
          double kPlus, kMinus;
          kPlus = forward * Math.exp(delta);
          kMinus = forward * Math.exp(-delta);
          EuropeanVanillaOption other = new EuropeanVanillaOption(kPlus, option.getTimeToExpiry(), option.isCall());
          final double yPlus = getVolatilityFunction(other, forward).evaluate(data);
          other = new EuropeanVanillaOption(kMinus, option.getTimeToExpiry(), option.isCall());
          final double yMinus = getVolatilityFunction(other, forward).evaluate(data);
          final double a2 = (yPlus + yMinus - 2 * a0) / 2 / delta / delta;
          final double a1 = (yPlus - yMinus) / 2 / delta;
          return a2 * x * x + a1 * x + a0;
        }
        final double tScale = nu * nu * t;
        final double alphaScale = alpha / nu;

        double q;
        if (CompareUtils.closeEquals(beta, 1.0, EPS)) {
          q = x;
        } else {
          q = (Math.pow(k, beta1) - Math.pow(forward, beta1)) / beta1;
        }

        final double vMin = Math.sqrt(alphaScale * alphaScale + 2 * rho * alphaScale * q + q * q);
        final double logTerm = Math.log((vMin + rho * alphaScale + q) / (1 + rho) / alphaScale);
        sigma0 = x / logTerm;

        final double cTilde = getCTilde(forward, k, alphaScale, beta, rho, q);
        sigma1 = -(cTilde + Math.log(sigma0 * Math.sqrt(k * forward))) / square(logTerm);
        return nu * sigma0 * (1 + sigma1 * tScale);
      }
    };
  }

  private double getCTilde(final double f, final double k, final double alpha, final double beta, final double rho, final double q) {
    final double rhoStar = Math.sqrt(1 - rho * rho);
    final double beta1 = 1 - beta;
    final double vMin = Math.sqrt(alpha * alpha + 2 * rho * alpha * q + q * q);
    double res = -0.5 * Math.log(alpha * vMin * Math.pow(f * k, beta));
    if (CompareUtils.closeEquals(beta, 1.0, EPS)) {
      res += rho / 2 / rhoStar / rhoStar * (rho * Math.log(k / f) - vMin + alpha);
    } else {
      final double a = Math.pow(f, beta1);
      final double b = beta1 * rhoStar;
      final double c = beta1 * rho;
      final double x1 = -rho * alpha / rhoStar;
      final double x2 = (q - rho * vMin) / rhoStar;
      final double y1 = alpha;
      final double y2 = vMin;
      final double xCap = (x2 * x2 - x1 * x1 + y2 * y2 - y1 * y1) / 2 / (x2 - x1);
      final double rCap = Math.sqrt(y1 * y1 + square(x1 - xCap));
      final double t1 = Math.sqrt((rCap - x1 + xCap) / (rCap + x1 - xCap));
      final double t2 = Math.sqrt((rCap - x2 + xCap) / (rCap + x2 - xCap));
      res -= rho * beta / beta1 / rhoStar * (getG(a, b, c, xCap, rCap, beta, t2) - getG(a, b, c, xCap, rCap, beta, t1));
    }
    return res;
  }

  private double getG(final double a, final double b, final double c, final double xCap, final double rCap, final double beta, final double t) {
    final double beta1 = 1 - beta;
    double res = Math.atan(t);
    final double y = square(a + b * xCap) - square((beta1 * rCap));
    if (y > 0) {
      final double temp = Math.sqrt(y);
      res -= (a + b * xCap) / temp * Math.atan((c * rCap + t * (a + b * (xCap - rCap))) / temp);
    } else if (y < 0) {
      final double temp = Math.sqrt(-y);
      res += (a + b * xCap) / temp * modAtanh((c * rCap + t * (a + b * (xCap - rCap))) / temp);
    } else {
      res += (a + b * xCap) / (c * rCap + t * (a + b * (xCap - rCap)));
    }
    return res;
  }

  private double modAtanh(final double z) {
    return 0.5 * Math.log(Math.abs((1 + z) / (1 - z)));
  }

  @Override
  public int hashCode() {
    return toString().hashCode();
  }

  @Override
  public boolean equals(final Object obj) {
    if (obj == null) {
      return false;
    }
    if (this == obj) {
      return true;
    }
    if (getClass() != obj.getClass()) {
      return false;
    }
    return true;
  }

  @Override
  public String toString() {
    return "SABR (Paulot)";
  }
}
TOP

Related Classes of com.opengamma.analytics.financial.model.volatility.smile.function.SABRPaulotVolatilityFunction

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.