Skip to content

Commit

Permalink
Allow forward price as additional result and in implied vol calculation
Browse files Browse the repository at this point in the history
  • Loading branch information
lballabio committed Jul 17, 2024
1 parent 351feff commit c1241da
Show file tree
Hide file tree
Showing 3 changed files with 32 additions and 13 deletions.
13 changes: 9 additions & 4 deletions ql/instruments/swaption.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -180,20 +180,25 @@ namespace QuantLib {
}

Volatility Swaption::impliedVolatility(Real targetValue,
const Handle<YieldTermStructure>& d,
const Handle<YieldTermStructure>& discountCurve,
Volatility guess,
Real accuracy,
Natural maxEvaluations,
Volatility minVol,
Volatility maxVol,
VolatilityType type,
Real displacement) const {
Real displacement,
PriceType priceType) const {

QL_REQUIRE(!isExpired(), "instrument expired");
QL_REQUIRE(exercise_->type() == Exercise::European, "not a European option");

ImpliedSwaptionVolHelper f(*this, d, targetValue, displacement, type);
//Brent solver;
if (priceType == Forward) {
// convert to spot
targetValue *= discountCurve->discount(exercise_->date(0));
}

ImpliedSwaptionVolHelper f(*this, discountCurve, targetValue, displacement, type);
NewtonSafe solver;
solver.setMaxEvaluations(maxEvaluations);
return solver.solve(f, accuracy, guess, minVol, maxVol);
Expand Down
4 changes: 3 additions & 1 deletion ql/instruments/swaption.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,7 @@ namespace QuantLib {
*/
class Swaption : public Option {
public:
enum PriceType { Spot, Forward };
class arguments;
class engine;
Swaption(ext::shared_ptr<FixedVsFloatingSwap> swap,
Expand Down Expand Up @@ -131,7 +132,8 @@ namespace QuantLib {
Volatility minVol = 1.0e-7,
Volatility maxVol = 4.0,
VolatilityType type = ShiftedLognormal,
Real displacement = 0.0) const;
Real displacement = 0.0,
PriceType priceType = Spot) const;
private:
// arguments
ext::shared_ptr<FixedVsFloatingSwap> swap_;
Expand Down
28 changes: 20 additions & 8 deletions test-suite/swaption.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -831,6 +831,7 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatility, *precondition(if_speed(Faster))) {

Settlement::Type types[] = { Settlement::Physical, Settlement::Cash };
Settlement::Method methods[] = { Settlement::PhysicalOTC, Settlement::ParYieldCurve };
Swaption::PriceType priceTypes[] = { Swaption::Spot, Swaption::Forward };
// test data
Rate strikes[] = { 0.02, 0.03, 0.04, 0.05, 0.06, 0.07 };
Volatility vols[] = { 0.01, 0.05, 0.10, 0.20, 0.30, 0.70, 0.90 };
Expand All @@ -851,12 +852,13 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatility, *precondition(if_speed(Faster))) {
.withFloatingLegSpread(0.0)
.withType(k);
for (Size h=0; h<LENGTH(types); h++) {
for (auto priceType : priceTypes) {
for (Real vol : vols) {
ext::shared_ptr<Swaption> swaption =
vars.makeSwaption(swap, exerciseDate, vol, types[h], methods[h],
BlackSwaptionEngine::DiscountCurve);
// Black price
Real value = swaption->NPV();
Real value = priceType == Swaption::Spot? swaption->NPV() : swaption->result<Real>("forwardPrice");
Volatility implVol = 0.0;
try {
implVol =
Expand All @@ -868,11 +870,12 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatility, *precondition(if_speed(Faster))) {
1.0e-7,
4.0,
ShiftedLognormal,
0.0);
0.0,
priceType);
} catch (std::exception& e) {
// couldn't bracket?
swaption->setPricingEngine(vars.makeEngine(0.0, BlackSwaptionEngine::DiscountCurve));
Real value2 = swaption->NPV();
Real value2 = priceType == Swaption::Spot? swaption->NPV() : swaption->result<Real>("forwardPrice");
if (std::fabs(value-value2) < tolerance) {
// ok, just skip:
continue;
Expand All @@ -885,12 +888,13 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatility, *precondition(if_speed(Faster))) {
<< "\natm level: " << io::rate(swap->fairRate())
<< "\nvol: " << io::volatility(vol)
<< "\nprice: " << value << "\n"
<< "\ntype: " << (priceType == Swaption::Spot? "spot" : "forward") << "\n"
<< e.what());
}
if (std::fabs(implVol - vol) > tolerance) {
// the difference might not matter
swaption->setPricingEngine(vars.makeEngine(implVol, BlackSwaptionEngine::DiscountCurve));
Real value2 = swaption->NPV();
Real value2 = priceType == Swaption::Spot? swaption->NPV() : swaption->result<Real>("forwardPrice");
if (std::fabs(value-value2) > tolerance) {
BOOST_ERROR("implied vol failure: "
<< exercise << "x" << length << " " << k
Expand All @@ -899,11 +903,13 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatility, *precondition(if_speed(Faster))) {
<< "\natm level: " << io::rate(swap->fairRate())
<< "\nvol: " << io::volatility(vol)
<< "\nprice: " << value
<< "\ntype: " << (priceType == Swaption::Spot? "spot" : "forward") << "\n"
<< "\nimplied vol: " << io::volatility(implVol)
<< "\nimplied price: " << value2);
}
}
}
}
}
}
}
Expand All @@ -923,6 +929,7 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatilityOis, *precondition(if_speed(Fast))) {

Settlement::Type types[] = { Settlement::Physical, Settlement::Cash };
Settlement::Method methods[] = { Settlement::PhysicalOTC, Settlement::ParYieldCurve };
Swaption::PriceType priceTypes[] = { Swaption::Spot, Swaption::Forward };
// test data
Rate strikes[] = { 0.02, 0.03, 0.04, 0.05, 0.06, 0.07 };
Volatility vols[] = { 0.01, 0.05, 0.10, 0.20, 0.30, 0.70, 0.90 };
Expand All @@ -942,12 +949,13 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatilityOis, *precondition(if_speed(Fast))) {
.withFixedLegDayCount(vars.fixedDayCount)
.withType(k);
for (Size h=0; h<LENGTH(types); h++) {
for (auto priceType : priceTypes) {
for (Real vol : vols) {
ext::shared_ptr<Swaption> swaption =
vars.makeOISwaption(swap, exerciseDate, vol, types[h], methods[h],
BlackSwaptionEngine::DiscountCurve);
// Black price
Real value = swaption->NPV();
Real value = priceType == Swaption::Spot? swaption->NPV() : swaption->result<Real>("forwardPrice");
Volatility implVol = 0.0;
try {
implVol =
Expand All @@ -959,11 +967,12 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatilityOis, *precondition(if_speed(Fast))) {
1.0e-7,
4.0,
ShiftedLognormal,
0.0);
0.0,
priceType);
} catch (std::exception& e) {
// couldn't bracket?
swaption->setPricingEngine(vars.makeEngine(0.0, BlackSwaptionEngine::DiscountCurve));
Real value2 = swaption->NPV();
Real value2 = priceType == Swaption::Spot? swaption->NPV() : swaption->result<Real>("forwardPrice");
if (std::fabs(value-value2) < tolerance) {
// ok, just skip:
continue;
Expand All @@ -976,12 +985,13 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatilityOis, *precondition(if_speed(Fast))) {
<< "\natm level: " << io::rate(swap->fairRate())
<< "\nvol: " << io::volatility(vol)
<< "\nprice: " << value << "\n"
<< "\ntype: " << (priceType == Swaption::Spot? "spot" : "forward") << "\n"
<< e.what());
}
if (std::fabs(implVol - vol) > tolerance) {
// the difference might not matter
swaption->setPricingEngine(vars.makeEngine(implVol, BlackSwaptionEngine::DiscountCurve));
Real value2 = swaption->NPV();
Real value2 = priceType == Swaption::Spot? swaption->NPV() : swaption->result<Real>("forwardPrice");
if (std::fabs(value-value2) > tolerance) {
BOOST_ERROR("implied vol failure: "
<< exercise << "x" << length << " " << k
Expand All @@ -990,11 +1000,13 @@ BOOST_AUTO_TEST_CASE(testImpliedVolatilityOis, *precondition(if_speed(Fast))) {
<< "\natm level: " << io::rate(swap->fairRate())
<< "\nvol: " << io::volatility(vol)
<< "\nprice: " << value
<< "\ntype: " << (priceType == Swaption::Spot? "spot" : "forward") << "\n"
<< "\nimplied vol: " << io::volatility(implVol)
<< "\nimplied price: " << value2);
}
}
}
}
}
}
}
Expand Down

0 comments on commit c1241da

Please sign in to comment.