From e77805bdbe51c7449e7d315dcc1ccee2b9753e00 Mon Sep 17 00:00:00 2001 From: simaki Date: Wed, 23 Mar 2022 17:56:56 +0900 Subject: [PATCH] ENH: Analytical BS American binary formula (#437) (#554) --- pfhedge/nn/functional.py | 38 ++++++++++++++++++++++++-------------- 1 file changed, 24 insertions(+), 14 deletions(-) diff --git a/pfhedge/nn/functional.py b/pfhedge/nn/functional.py index aa5008da..aecbd46a 100644 --- a/pfhedge/nn/functional.py +++ b/pfhedge/nn/functional.py @@ -731,7 +731,8 @@ def bs_european_price( """ s, t, v = broadcast_all(log_moneyness, time_to_maturity, volatility) - price = strike * (s.exp() * ncdf(d1(s, t, v)) - ncdf(d2(s, t, v))) + spot = s.exp() * strike + price = spot * ncdf(d1(s, t, v)) - strike * ncdf(d2(s, t, v)) price = price + strike * (1 - s.exp()) if not call else price # put-call parity return price @@ -930,7 +931,7 @@ def bs_american_binary_price( # By this substitution we get N([log(S(0) / K) + ...] / sigma T) --> 1. s, t, v = broadcast_all(log_moneyness, time_to_maturity, volatility) - p = ncdf(d2(s, t, v)) + s.exp() * (1 - ncdf(d2(-s, t, v))) + p = ncdf(d2(s, t, v)) + s.exp() * ncdf(d1(s, t, v)) return p.where(max_log_moneyness < 0, torch.ones_like(p)) @@ -948,13 +949,17 @@ def bs_american_binary_delta( """ s, t, v = broadcast_all(log_moneyness, time_to_maturity, volatility) spot = s.exp() * strike + + d1_tensor = d1(s, t, v) + d2_tensor = d2(s, t, v) + w = v * t.sqrt() + # ToDo: fix 0/0 issue p = ( - npdf(d2(s, t, v)) / (spot * v * t.sqrt()) - - (1 - ncdf(d2(-s, t, v))) / strike - + npdf(d2(-s, t, v)) / (strike * v * t.sqrt()) + npdf(d2_tensor).div(spot * w) + + ncdf(d1_tensor).div(strike) + + npdf(d1_tensor).div(strike * w) ) - return p.where(max_log_moneyness < 0, torch.zeros_like(p)) @@ -969,15 +974,20 @@ def bs_american_binary_gamma( See :func:`pfhedge.nn.BSAmericanBinaryOption.gamma` for details. """ - # TODO(simaki): Compute analytically - return autogreek.gamma_from_delta( - bs_american_binary_price, - log_moneyness=log_moneyness, - max_log_moneyness=max_log_moneyness, - time_to_maturity=time_to_maturity, - volatility=volatility, - strike=strike, + s, t, v = broadcast_all(log_moneyness, time_to_maturity, volatility) + spot = s.exp() * strike + + d1_tensor = d1(s, t, v) + d2_tensor = d2(s, t, v) + w = v * t.sqrt() + + p = ( + -npdf(d2_tensor).div(spot.square() * w) + - d2_tensor * npdf(d2_tensor).div(spot.square() * w.square()) + + npdf(d1_tensor).div(spot * strike * w) + - d1_tensor * npdf(d1_tensor).div(spot * strike * w.square()) ) + return p.where(max_log_moneyness < 0, torch.zeros_like(p)) def bs_american_binary_vega(