/**
* Copyright (C) 2009 - present by OpenGamma Inc. and the OpenGamma group of companies
*
* Please see distribution for license.
*/
package com.opengamma.analytics.financial.model.option.pricing.analytic;
import org.apache.commons.lang.Validate;
import com.opengamma.analytics.financial.model.option.definition.AssetOrNothingOptionDefinition;
import com.opengamma.analytics.financial.model.option.definition.StandardOptionDataBundle;
import com.opengamma.analytics.math.function.Function1D;
import com.opengamma.analytics.math.statistics.distribution.NormalDistribution;
import com.opengamma.analytics.math.statistics.distribution.ProbabilityDistribution;
/**
* Class for pricing asset-or-nothing options (see {@link com.opengamma.analytics.financial.model.option.definition.AssetOrNothingOptionDefinition}).
* The price is calculated using the Reiner-Rubenstein formula:
* $$
* \begin{align*}
* c &= S e^{(b-r)T}N(d)\\
* p &= S e^{(b-r)T}N(-d)
* \end{align*}
* $$
* where
* $$
* \begin{align*}
* d = \frac{\ln{\frac{S}{K}} + (b + \frac{\sigma^2}{2})T}{\sigma\sqrt{T}}
* \end{align*}
* $$
*
*/
public class AssetOrNothingOptionModel extends AnalyticOptionModel<AssetOrNothingOptionDefinition, StandardOptionDataBundle> {
private static final ProbabilityDistribution<Double> NORMAL = new NormalDistribution(0, 1);
/**
* {@inheritDoc}
*/
@Override
public Function1D<StandardOptionDataBundle, Double> getPricingFunction(final AssetOrNothingOptionDefinition definition) {
Validate.notNull(definition, "definition");
return new Function1D<StandardOptionDataBundle, Double>() {
@SuppressWarnings("synthetic-access")
@Override
public Double evaluate(final StandardOptionDataBundle data) {
Validate.notNull(data, "data");
final double s = data.getSpot();
final double k = definition.getStrike();
final double t = definition.getTimeToExpiry(data.getDate());
final double r = data.getInterestRate(t);
final double sigma = data.getVolatility(t, k);
final double b = data.getCostOfCarry();
final double d = getD1(s, k, t, sigma, b);
return s * Math.exp(t * (b - r)) * NORMAL.getCDF(definition.isCall() ? d : -d);
}
};
}
}