discretizedconvertible.cpp
来自「有很多的函数库」· C++ 代码 · 共 204 行
CPP
204 行
/* -*- mode: c++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
Copyright (C) 2005, 2006 Theo Boafo
Copyright (C) 2006 StatPro Italia srl
This file is part of QuantLib, a free-software/open-source library
for financial quantitative analysts and developers - http://quantlib.org/
QuantLib is free software: you can redistribute it and/or modify it
under the terms of the QuantLib license. You should have received a
copy of the license along with this program; if not, please email
<quantlib-dev@lists.sf.net>. The license is also available online at
<http://quantlib.org/license.shtml>.
This program is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
FOR A PARTICULAR PURPOSE. See the license for more details.
*/
#include <ql/pricingengines/hybrid/discretizedconvertible.hpp>
#include <ql/processes/blackscholesprocess.hpp>
#include <ql/math/comparison.hpp>
namespace QuantLib {
DiscretizedConvertible::DiscretizedConvertible(
const ConvertibleBond::option::arguments& args)
: arguments_(args) {
dividendValues_ = Array(arguments_.dividends.size(), 0.0);
boost::shared_ptr<GeneralizedBlackScholesProcess> process =
boost::dynamic_pointer_cast<GeneralizedBlackScholesProcess>(
arguments_.stochasticProcess);
QL_REQUIRE(process, "Black-Scholes process required");
Date settlementDate = process->riskFreeRate()->referenceDate();
for (Size i=0; i<arguments_.dividends.size(); i++) {
if (arguments_.dividends[i]->date() >= settlementDate) {
dividendValues_[i] =
arguments_.dividends[i]->amount() *
process->riskFreeRate()->discount(
arguments_.dividends[i]->date());
}
}
}
void DiscretizedConvertible::reset(Size size) {
// Set to bond redemption values
values_ = Array(size, arguments_.redemption);
// coupon amounts should be added when adjusting
// values_ = Array(size, arguments_.cashFlows.back()->amount());
conversionProbability_ = Array(size, 0.0);
spreadAdjustedRate_ = Array(size, 0.0);
boost::shared_ptr<GeneralizedBlackScholesProcess> process =
boost::dynamic_pointer_cast<GeneralizedBlackScholesProcess>(
arguments_.stochasticProcess);
QL_REQUIRE(process, "Black-Scholes process required");
DayCounter rfdc = process->riskFreeRate()->dayCounter();
// this takes care of convertibility and conversion probabilities
adjustValues();
Real creditSpread = arguments_.creditSpread->value();
Date exercise = arguments_.exercise->lastDate();
Rate riskFreeRate =
process->riskFreeRate()->zeroRate(exercise, rfdc,
Continuous, NoFrequency);
// Calculate blended discount rate to be used on roll back.
for (Size j=0; j<values_.size(); j++) {
spreadAdjustedRate_[j] =
conversionProbability_[j] * riskFreeRate +
(1-conversionProbability_[j])*(riskFreeRate + creditSpread);
}
}
void DiscretizedConvertible::postAdjustValuesImpl() {
Size i;
bool convertible = false;
switch (arguments_.exercise->type()) {
case Exercise::American:
if (time() <= arguments_.stoppingTimes[1] &&
time() >= arguments_.stoppingTimes[0])
convertible = true;
break;
case Exercise::European:
if (isOnTime(arguments_.stoppingTimes[0]))
convertible = true;
break;
case Exercise::Bermudan:
for (i = 0; i<arguments_.stoppingTimes.size(); i++) {
if (isOnTime(arguments_.stoppingTimes[i]))
convertible = true;
}
break;
default:
QL_FAIL("invalid option type");
}
for (i=0; i<arguments_.callabilityTimes.size(); i++) {
if (isOnTime(arguments_.callabilityTimes[i]))
applyCallability(i,convertible);
}
for (i=0; i<arguments_.couponTimes.size(); i++) {
if (isOnTime(arguments_.couponTimes[i]))
addCoupon(i);
}
if (convertible)
applyConvertibility();
}
void DiscretizedConvertible::applyConvertibility() {
Array grid = adjustedGrid();
for (Size j=0; j<values_.size(); j++) {
Real payoff = arguments_.conversionRatio*grid[j];
if (values_[j] <= payoff) {
values_[j] = payoff;
conversionProbability_[j] = 1.0;
}
}
}
void DiscretizedConvertible::applyCallability(Size i, bool convertible) {
Size j;
Array grid = adjustedGrid();
switch (arguments_.callabilityTypes[i]) {
case Callability::Call:
if (arguments_.callabilityTriggers[i] != Null<Real>()) {
Real conversionValue =
arguments_.redemption/arguments_.conversionRatio;
Real trigger =
conversionValue*arguments_.callabilityTriggers[i];
for (j=0; j<values_.size(); j++) {
// the callability is conditioned by the trigger...
if (grid[j] >= trigger) {
// ...and might trigger conversion
values_[j] =
std::min(std::max(
arguments_.callabilityPrices[i],
arguments_.conversionRatio*grid[j]),
values_[j]);
}
}
} else if (convertible) {
for (j=0; j<values_.size(); j++) {
// exercising the callability might trigger conversion
values_[j] =
std::min(std::max(arguments_.callabilityPrices[i],
arguments_.conversionRatio*grid[j]),
values_[j]);
}
} else {
for (j=0; j<values_.size(); j++) {
values_[j] = std::min(arguments_.callabilityPrices[i],
values_[j]);
}
}
break;
case Callability::Put:
for (j=0; j<values_.size(); j++) {
values_[j] = std::max(values_[j],
arguments_.callabilityPrices[i]);
}
break;
default:
QL_FAIL("unknown callability type");
}
}
void DiscretizedConvertible::addCoupon(Size i) {
values_ += arguments_.couponAmounts[i];
}
Disposable<Array> DiscretizedConvertible::adjustedGrid() const {
Time t = time();
Array grid = method()->grid(t);
// add back all dividend amounts in the future
for (Size i=0; i<arguments_.dividends.size(); i++) {
Time dividendTime = arguments_.dividendTimes[i];
if (dividendTime >= t || close(dividendTime,t)) {
const boost::shared_ptr<Dividend>& d = arguments_.dividends[i];
for (Size j=0; j<grid.size(); j++)
grid[j] += d->amount(grid[j]);
}
}
return grid;
}
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?