From 592ebca5289aa9d40dc6ed8ea04206702229a06c Mon Sep 17 00:00:00 2001 From: s1lentq Date: Wed, 7 Aug 2024 17:55:49 +0700 Subject: [PATCH] Add new return type for amxx forwards --- .../extra/amxmodx/scripting/include/reapi.inc | 5 ++- reapi/src/hook_callback.h | 36 +++++++++++-------- 2 files changed, 26 insertions(+), 15 deletions(-) diff --git a/reapi/extra/amxmodx/scripting/include/reapi.inc b/reapi/extra/amxmodx/scripting/include/reapi.inc index 1616029..b3cbfe2 100644 --- a/reapi/extra/amxmodx/scripting/include/reapi.inc +++ b/reapi/extra/amxmodx/scripting/include/reapi.inc @@ -120,8 +120,11 @@ enum { HC_CONTINUE = 0, // Plugin didn't take any action HC_SUPERCEDE, // Skip real function, use my return value - HC_BREAK // Skip all forwards and real function, use my return value + HC_BREAK, // Skip all forwards and real function, use my return value // @note Warning: Be very careful, using this type of return will skip calls for all following AMXX plugins + + HC_BYPASS // Skip calls for all following AMXX plugins, but call the original function + // @note Warning: In PRE skips all forwards including POST forwards }; /** diff --git a/reapi/src/hook_callback.h b/reapi/src/hook_callback.h index 84356d3..3e3733a 100644 --- a/reapi/src/hook_callback.h +++ b/reapi/src/hook_callback.h @@ -5,10 +5,13 @@ // hookchain return type enum HookChainState { - HC_CONTINUE = 0, // plugin didn't take any action - HC_SUPERCEDE, // skip real function, use my return value - HC_BREAK // skip all forwards and real function, use my return value - // @note Warning: Be very careful using this type of return will skip calls for all following AMXX the plugins. + HC_CONTINUE = 0, // Plugin didn't take any action + HC_SUPERCEDE, // Skip real function, use my return value + HC_BREAK, // Skip all forwards and real function, use my return value + // @note Warning: Be very careful using this type of return will skip calls for all following AMXX the plugins + + HC_BYPASS // Skip calls for all following AMXX plugins, but call the original function + // @note Warning: In PRE skips all forwards including POST forwards }; // api types @@ -168,15 +171,17 @@ NOINLINE void DLLEXPORT _callVoidForward(hook_t* hook, original_t original, f_ar if (likely(fwd->GetState() == FSTATE_ENABLED)) { hookCtx->SetId(fwd->GetIndex()); // set current handler hook - auto ret = g_amxxapi.ExecuteForward(fwd->GetFwdIndex(), std::forward(args)...); + int ret = g_amxxapi.ExecuteForward(fwd->GetFwdIndex(), std::forward(args)...); hookCtx->ResetId(); - if (unlikely(ret == HC_BREAK)) { + if (unlikely(ret == HC_BREAK)) return; - } if (unlikely(ret > hc_state)) hc_state = ret; + + if (unlikely(ret == HC_BYPASS)) + break; } } @@ -187,16 +192,19 @@ NOINLINE void DLLEXPORT _callVoidForward(hook_t* hook, original_t original, f_ar hook->wasCalled = true; } - for (auto fwd : hook->post) + if (hc_state != HC_BYPASS) { - if (likely(fwd->GetState() == FSTATE_ENABLED)) + for (auto fwd : hook->post) { - hookCtx->SetId(fwd->GetIndex()); // set current handler hook - auto ret = g_amxxapi.ExecuteForward(fwd->GetFwdIndex(), std::forward(args)...); - hookCtx->ResetId(); + if (likely(fwd->GetState() == FSTATE_ENABLED)) + { + hookCtx->SetId(fwd->GetIndex()); // set current handler hook + int ret = g_amxxapi.ExecuteForward(fwd->GetFwdIndex(), std::forward(args)...); + hookCtx->ResetId(); - if (unlikely(ret == HC_BREAK)) - break; + if (unlikely(ret == HC_BREAK || ret == HC_BYPASS)) + break; + } } }