/* ======== SourceHook ======== * Copyright (C) 2004-2005 Metamod:Source Development Team * No warranties of any kind * * License: zlib/libpng * * Author(s): Pavol "PM OnoTo" Marko * Contributors: Scott "Damaged Soul" Ehlert * ============================ */ /** * @file sourcehook.cpp * @brief Contains the implementation of the SourceHook API */ #if defined __GNUC__ #include #endif #include "sourcehook_impl.h" #include "sh_memory.h" namespace SourceHook { template<> int SourceHook::HashFunction(const int & k) { return k; } template<> int SourceHook::Compare(const int & k1, const int & k2) { if (k1 == k2) return 0; if (k1 > k2) return 1; if (k1 < k2) return -1; return 0; } CSourceHookImpl::CSourceHookImpl() { } CSourceHookImpl::~CSourceHookImpl() { } int CSourceHookImpl::GetIfaceVersion() { return SH_IFACE_VERSION; } int CSourceHookImpl::GetImplVersion() { return SH_IMPL_VERSION; } bool CSourceHookImpl::IsPluginInUse(Plugin plug) { // Iterate through all hook managers which are in this plugin // Iterate through their vfnptrs, ifaces, hooks // If a hook from an other plugin is found, return true // Return false otherwise #define TMP_CHECK_LIST(name) \ for (hook_iter = iface_iter->name.m_List.begin(); hook_iter != iface_iter->name.m_List.end(); ++hook_iter) \ if (hook_iter->plug == plug) \ return true; for (HookManInfoList::iterator hmil_iter = m_HookMans.begin(); hmil_iter != m_HookMans.end(); ++hmil_iter) { if (hmil_iter->m_Plug != plug) continue; for (CHookManagerInfo::VfnPtrListIter vfnptr_iter = hmil_iter->m_VfnPtrs.begin(); vfnptr_iter != hmil_iter->m_VfnPtrs.end(); ++vfnptr_iter) { for (CVfnPtr::IfaceListIter iface_iter = vfnptr_iter->m_Ifaces.begin(); iface_iter != vfnptr_iter->m_Ifaces.end(); ++iface_iter) { List::iterator hook_iter; TMP_CHECK_LIST(m_PreHooks); TMP_CHECK_LIST(m_PostHooks); } } } #undef TMP_CHECK_LIST return false; } void CSourceHookImpl::UnloadPlugin(Plugin plug) { // 1) Manually remove all hooks by this plugin List hookstoremove; HookManInfoList::iterator hmil_iter; #define TMP_CHECK_LIST(name, ispost) \ for (hook_iter = iface_iter->name.m_List.begin(); hook_iter != iface_iter->name.m_List.end(); ++hook_iter) \ if (hook_iter->plug == plug) \ hookstoremove.push_back(RemoveHookInfo(hook_iter->plug, iface_iter->m_Ptr, \ hook_iter->thisptr_offs, hmil_iter->m_Func, hook_iter->handler, ispost)) for (hmil_iter = m_HookMans.begin(); hmil_iter != m_HookMans.end(); ++hmil_iter) { for (CHookManagerInfo::VfnPtrListIter vfnptr_iter = hmil_iter->m_VfnPtrs.begin(); vfnptr_iter != hmil_iter->m_VfnPtrs.end(); ++vfnptr_iter) { for (CVfnPtr::IfaceListIter iface_iter = vfnptr_iter->m_Ifaces.begin(); iface_iter != vfnptr_iter->m_Ifaces.end(); ++iface_iter) { List::iterator hook_iter; TMP_CHECK_LIST(m_PreHooks, false); TMP_CHECK_LIST(m_PostHooks, true); } } } #undef TMP_CHECK_LIST for (List::iterator rmiter = hookstoremove.begin(); rmiter != hookstoremove.end(); ++rmiter) RemoveHook(*rmiter); // 2) Other plugins may use hook managers in this plugin. // Get a list of hook managers that are in this plugin and are used by other plugins // Delete all hook managers that are in this plugin HookManInfoList tmphookmans; bool erase = false; for (hmil_iter = m_HookMans.begin(); hmil_iter != m_HookMans.end(); erase ? hmil_iter=m_HookMans.erase(hmil_iter) : ++hmil_iter) { if (hmil_iter->m_Plug == plug) { if (!hmil_iter->m_VfnPtrs.empty()) { // All hooks by this plugin are already removed // So if there is a vfnptr, it has to be used by an other plugin tmphookmans.push_back(*hmil_iter); } erase = true; } else erase = false; } // For each hook manager that is used in an other plugin: for (hmil_iter = tmphookmans.begin(); hmil_iter != tmphookmans.end(); ++hmil_iter) { // Find a suitable hook manager in an other plugin HookManInfoList::iterator newHookMan = FindHookMan(m_HookMans.begin(), m_HookMans.end(), hmil_iter->m_Proto, hmil_iter->m_VtblOffs, hmil_iter->m_VtblIdx); // This should _never_ happen. // If there is a hook from an other plugin, the plugin must have provided a hook manager as well. SH_ASSERT(newHookMan != m_HookMans.end(), ("Could not find a suitable hook manager in an other plugin!")); // AddHook should make sure that every plugin only has _one_ hook manager for _one_ proto/vi/vo SH_ASSERT(newHookMan->m_Plug != plug, ("New hook manager from same plugin!")); // The first hook manager should be always used - so the new hook manager has to be empty SH_ASSERT(newHookMan->m_VfnPtrs.empty(), ("New hook manager not empty!")); // Move the vfnptrs from the old hook manager to the new one newHookMan->m_VfnPtrs = hmil_iter->m_VfnPtrs; // Unregister the old one, register the new one hmil_iter->m_Func(HA_Unregister, NULL); newHookMan->m_Func(HA_Register, &(*newHookMan)); // zOMG BAIL, here is part of what you wanted: // Go through all vfnptrs in this hookman and patch them to point to the new manager's handler! // or whatever for (CHookManagerInfo::VfnPtrListIter vfnptr_iter = newHookMan->m_VfnPtrs.begin(); vfnptr_iter != newHookMan->m_VfnPtrs.end(); ++vfnptr_iter) { // And DEREFERENCE newHookMan->m_HookfuncVfnptr! // otherwise it will be executing the vtable... had to find out the hard way *reinterpret_cast(vfnptr_iter->m_Ptr) = *reinterpret_cast(newHookMan->m_HookfuncVfnptr); } // That should fix it, bail! } } void CSourceHookImpl::CompleteShutdown() { List hookstoremove; #define TMP_CHECK_LIST(name, ispost) \ for (hook_iter = iface_iter->name.m_List.begin(); hook_iter != iface_iter->name.m_List.end(); ++hook_iter) \ hookstoremove.push_back(RemoveHookInfo(hook_iter->plug, iface_iter->m_Ptr, \ hook_iter->thisptr_offs, hmil_iter->m_Func, hook_iter->handler, ispost)) for (HookManInfoList::iterator hmil_iter = m_HookMans.begin(); hmil_iter != m_HookMans.end(); ++hmil_iter) { for (CHookManagerInfo::VfnPtrListIter vfnptr_iter = hmil_iter->m_VfnPtrs.begin(); vfnptr_iter != hmil_iter->m_VfnPtrs.end(); ++vfnptr_iter) { for (CVfnPtr::IfaceListIter iface_iter = vfnptr_iter->m_Ifaces.begin(); iface_iter != vfnptr_iter->m_Ifaces.end(); ++iface_iter) { List::iterator hook_iter; TMP_CHECK_LIST(m_PreHooks, false); TMP_CHECK_LIST(m_PostHooks, true); } } } #undef TMP_CHECK_LIST for (List::iterator rmiter = hookstoremove.begin(); rmiter != hookstoremove.end(); ++rmiter) RemoveHook(*rmiter); m_HookMans.clear(); } bool CSourceHookImpl::AddHook(Plugin plug, void *iface, int thisptr_offs, HookManagerPubFunc myHookMan, ISHDelegate *handler, bool post) { void *adjustediface = reinterpret_cast(reinterpret_cast(iface) + thisptr_offs); // 1) Get info about the hook manager CHookManagerInfo tmp; if (myHookMan(HA_GetInfo, &tmp) != 0) return false; // Add the proposed hook manager to the _end_ of the list if the plugin doesn't have a hook manager // with this proto/vo/vi registered HookManInfoList::iterator hkmi_iter; for (hkmi_iter = m_HookMans.begin(); hkmi_iter != m_HookMans.end(); ++hkmi_iter) { if (hkmi_iter->m_Plug == plug && strcmp(hkmi_iter->m_Proto, tmp.m_Proto) == 0 && hkmi_iter->m_VtblOffs == tmp.m_VtblOffs && hkmi_iter->m_VtblIdx == tmp.m_VtblIdx) break; } if (hkmi_iter == m_HookMans.end()) { // No such hook manager from this plugin yet, add it! tmp.m_Func = myHookMan; tmp.m_Plug = plug; m_HookMans.push_back(tmp); } // Then, search for a suitable hook manager (from the beginning) HookManInfoList::iterator hookman = FindHookMan(m_HookMans.begin(), m_HookMans.end(), tmp.m_Proto, tmp.m_VtblOffs, tmp.m_VtblIdx); SH_ASSERT(hookman != m_HookMans.end(), ("No hookman found - but if there was none, we've just added one!")); // Tell it to store the pointer if it's not already active if (hookman->m_VfnPtrs.empty()) hookman->m_Func(HA_Register, &(*hookman)); void **cur_vtptr = *reinterpret_cast( reinterpret_cast(adjustediface) + tmp.m_VtblOffs); void *cur_vfnptr = reinterpret_cast(cur_vtptr + tmp.m_VtblIdx); CHookManagerInfo::VfnPtrListIter vfnptr_iter = hookman->m_VfnPtrs.find(cur_vfnptr); if (vfnptr_iter == hookman->m_VfnPtrs.end()) { // Add a new one CVfnPtr vfp(cur_vfnptr); // Alter vtable entry if (!SetMemAccess(cur_vtptr, sizeof(void*) * (tmp.m_VtblIdx + 1), SH_MEM_READ | SH_MEM_WRITE)) return false; *reinterpret_cast(cur_vfnptr) = *reinterpret_cast(hookman->m_HookfuncVfnptr); hookman->m_VfnPtrs.push_back(vfp); // Make vfnptr_iter point to the new element vfnptr_iter = hookman->m_VfnPtrs.end(); --vfnptr_iter; // Now that it is done, check whether we have to update any callclasses ApplyCallClassPatches(adjustediface, tmp.m_VtblOffs, tmp.m_VtblIdx, vfp.m_OrigEntry); } CVfnPtr::IfaceListIter iface_iter = vfnptr_iter->m_Ifaces.find(adjustediface); if (iface_iter == vfnptr_iter->m_Ifaces.end()) { // Add a new one vfnptr_iter->m_Ifaces.push_back(CIface(adjustediface)); // Make iface_iter point to the new element iface_iter = vfnptr_iter->m_Ifaces.end(); --iface_iter; } // Add the hook HookInfo hookinfo; hookinfo.handler = handler; hookinfo.plug = plug; hookinfo.paused = false; hookinfo.thisptr_offs = thisptr_offs; if (post) iface_iter->m_PostHooks.m_List.push_back(hookinfo); else iface_iter->m_PreHooks.m_List.push_back(hookinfo); return true; } bool CSourceHookImpl::RemoveHook(RemoveHookInfo info) { return RemoveHook(info.plug, info.iface, info.thisptr_offs, info.hookman, info.handler, info.post); } bool CSourceHookImpl::RemoveHook(Plugin plug, void *iface, int thisptr_offs, HookManagerPubFunc myHookMan, ISHDelegate *handler, bool post) { void *adjustediface = reinterpret_cast(reinterpret_cast(iface)+thisptr_offs); CHookManagerInfo tmp; if (myHookMan(HA_GetInfo, &tmp) != 0) return false; // Find the hook manager and the hook HookManInfoList::iterator hookman = FindHookMan(m_HookMans.begin(), m_HookMans.end(), tmp.m_Proto, tmp.m_VtblOffs, tmp.m_VtblIdx); if (hookman == m_HookMans.end()) return false; if (!ModuleInMemory(reinterpret_cast(adjustediface) + tmp.m_VtblOffs, sizeof(void*) * (tmp.m_VtblIdx + 1))) { // The module the vtable was in is already unloaded. hookman->m_VfnPtrs.clear(); hookman->m_Func(HA_Unregister, NULL); return true; } void **cur_vtptr = *reinterpret_cast( reinterpret_cast(adjustediface) + tmp.m_VtblOffs); void *cur_vfnptr = reinterpret_cast(cur_vtptr + tmp.m_VtblIdx); CHookManagerInfo::VfnPtrListIter vfnptr_iter = hookman->m_VfnPtrs.find(cur_vfnptr); if (vfnptr_iter == hookman->m_VfnPtrs.end()) return false; for (CVfnPtr::IfaceListIter iface_iter = vfnptr_iter->m_Ifaces.begin(); iface_iter != vfnptr_iter->m_Ifaces.end();) { List &hooks = post ? iface_iter->m_PostHooks.m_List : iface_iter->m_PreHooks.m_List; bool erase; for (List::iterator hookiter = hooks.begin(); hookiter != hooks.end(); ) { erase = hookiter->plug == plug && hookiter->handler->IsEqual(handler) && hookiter->thisptr_offs == thisptr_offs; if (erase) { hookiter->handler->DeleteThis(); // Make the _plugin_ delete the handler object // Move all iterators pointing at this List::iterator oldhookiter = hookiter; hookiter = hooks.erase(hookiter); List::iterator newhookiter = hookiter; --newhookiter; // The hook loop will ++ it then CHookList::CIter *pItIter; for (pItIter = iface_iter->m_PreHooks.m_UsedIters; pItIter; pItIter = pItIter->m_pNext) if (pItIter->m_Iter == oldhookiter) pItIter->m_Iter = newhookiter; } else ++hookiter; } if (iface_iter->m_PostHooks.m_List.empty() && iface_iter->m_PreHooks.m_List.empty()) { // There are no hooks on this iface anymore... if (m_CurIface == static_cast(&(*iface_iter))) m_HLS = HLS_Stop; iface_iter = vfnptr_iter->m_Ifaces.erase(iface_iter); if (vfnptr_iter->m_Ifaces.empty()) { // No ifaces at all -> Deactivate the hook *reinterpret_cast(vfnptr_iter->m_Ptr) = vfnptr_iter->m_OrigEntry; hookman->m_VfnPtrs.erase(vfnptr_iter); // Remove callclass patch for (Impl_CallClassList::iterator cciter = m_CallClasses.begin(); cciter != m_CallClasses.end(); ++cciter) if (cciter->m_Ptr == adjustediface) cciter->RemoveCallClassPatch(tmp.m_VtblOffs, tmp.m_VtblIdx); if (hookman->m_VfnPtrs.empty()) { // Unregister the hook manager hookman->m_Func(HA_Unregister, NULL); } // Don't try to continue looping through ifaces // - the list is already invalid return true; } } else ++iface_iter; } return true; } GenericCallClass *CSourceHookImpl::GetCallClass(void *iface, size_t size) { for (Impl_CallClassList::iterator cciter = m_CallClasses.begin(); cciter != m_CallClasses.end(); ++cciter) { if (cciter->m_Ptr == iface && cciter->m_ObjSize == size) { ++cciter->m_RefCounter; return &(*cciter); } } // Make a new one CCallClassImpl tmp(iface, size); ApplyCallClassPatches(tmp); m_CallClasses.push_back(tmp); return &m_CallClasses.back(); } void CSourceHookImpl::ReleaseCallClass(GenericCallClass *ptr) { Impl_CallClassList::iterator iter = m_CallClasses.find(ptr); if (iter == m_CallClasses.end()) return; --iter->m_RefCounter; if (iter->m_RefCounter < 1) m_CallClasses.erase(iter); } void CSourceHookImpl::ApplyCallClassPatches(CCallClassImpl &cc) { for (HookManInfoList::iterator hookman = m_HookMans.begin(); hookman != m_HookMans.end(); ++hookman) { for (CHookManagerInfo::VfnPtrListIter vfnptr_iter = hookman->m_VfnPtrs.begin(); vfnptr_iter != hookman->m_VfnPtrs.end(); ++vfnptr_iter) { for (CVfnPtr::IfaceListIter iface_iter = vfnptr_iter->m_Ifaces.begin(); iface_iter != vfnptr_iter->m_Ifaces.end(); ++iface_iter) { if (iface_iter->m_Ptr >= cc.m_Ptr && iface_iter->m_Ptr < (reinterpret_cast(cc.m_Ptr) + cc.m_ObjSize)) { cc.ApplyCallClassPatch(static_cast(reinterpret_cast(iface_iter->m_Ptr) - reinterpret_cast(cc.m_Ptr)) + hookman->m_VtblOffs, hookman->m_VtblIdx, vfnptr_iter->m_OrigEntry); } } } } } void CSourceHookImpl::ApplyCallClassPatches(void *ifaceptr, int vtbl_offs, int vtbl_idx, void *orig_entry) { for (Impl_CallClassList::iterator cc_iter = m_CallClasses.begin(); cc_iter != m_CallClasses.end(); ++cc_iter) { if (ifaceptr >= cc_iter->m_Ptr && ifaceptr < (reinterpret_cast(cc_iter->m_Ptr) + cc_iter->m_ObjSize)) { cc_iter->ApplyCallClassPatch(static_cast(reinterpret_cast(ifaceptr) - reinterpret_cast(cc_iter->m_Ptr)) + vtbl_offs, vtbl_idx, orig_entry); } } } void CSourceHookImpl::RemoveCallClassPatches(void *ifaceptr, int vtbl_offs, int vtbl_idx) { for (Impl_CallClassList::iterator cc_iter = m_CallClasses.begin(); cc_iter != m_CallClasses.end(); ++cc_iter) { if (ifaceptr >= cc_iter->m_Ptr && ifaceptr < (reinterpret_cast(cc_iter->m_Ptr) + cc_iter->m_ObjSize)) { cc_iter->RemoveCallClassPatch(vtbl_offs, vtbl_idx); } } } CSourceHookImpl::HookManInfoList::iterator CSourceHookImpl::FindHookMan(HookManInfoList::iterator begin, HookManInfoList::iterator end, const char *proto, int vtblofs, int vtblidx) { HookManInfoList::iterator hookmaniter; for (hookmaniter = m_HookMans.begin(); hookmaniter != m_HookMans.end(); ++hookmaniter) { if (strcmp(hookmaniter->m_Proto, proto) == 0 && hookmaniter->m_VtblOffs == vtblofs && hookmaniter->m_VtblIdx == vtblidx) break; } return hookmaniter; } void CSourceHookImpl::SetPluginPaused(Plugin plug, bool paused) { // Go through all hook managers, all interfaces, and set the status of all hooks of this plugin to paused for (HookManInfoList::iterator hookmaniter = m_HookMans.begin(); hookmaniter != m_HookMans.end(); ++hookmaniter) for (CHookManagerInfo::VfnPtrListIter vfnptr_iter = hookmaniter->m_VfnPtrs.begin(); vfnptr_iter != hookmaniter->m_VfnPtrs.end(); ++vfnptr_iter) for (CVfnPtr::IfaceListIter ifaceiter = vfnptr_iter->m_Ifaces.begin(); ifaceiter != vfnptr_iter->m_Ifaces.end(); ++ifaceiter) { for (List::iterator hookiter = ifaceiter->m_PreHooks.m_List.begin(); hookiter != ifaceiter->m_PreHooks.m_List.end(); ++hookiter) if (plug == hookiter->plug) hookiter->paused = paused; for (List::iterator hookiter = ifaceiter->m_PostHooks.m_List.begin(); hookiter != ifaceiter->m_PostHooks.m_List.end(); ++hookiter) if (plug == hookiter->plug) hookiter->paused = paused; } } void CSourceHookImpl::PausePlugin(Plugin plug) { SetPluginPaused(plug, true); } void CSourceHookImpl::UnpausePlugin(Plugin plug) { SetPluginPaused(plug, false); } void CSourceHookImpl::SetRes(META_RES res) { m_CurRes = res; } META_RES CSourceHookImpl::GetPrevRes() { return m_PrevRes; } META_RES CSourceHookImpl::GetStatus() { return m_Status; } const void *CSourceHookImpl::GetOrigRet() { return m_OrigRet; } const void *CSourceHookImpl::GetOverrideRet() { return m_OverrideRet; } META_RES &CSourceHookImpl::GetCurResRef() { return m_CurRes; } META_RES &CSourceHookImpl::GetPrevResRef() { return m_PrevRes; } META_RES &CSourceHookImpl::GetStatusRef() { return m_Status; } void * &CSourceHookImpl::GetIfacePtrRef() { return m_IfacePtr; } void CSourceHookImpl::SetOrigRet(const void *ptr) { m_OrigRet = ptr; } void CSourceHookImpl::SetOverrideRet(const void *ptr) { m_OverrideRet = ptr; } void *CSourceHookImpl::GetIfacePtr() { return m_IfacePtr; } HookLoopStatus &CSourceHookImpl::GetStatusVarRef(IIface *pIface) { m_CurIface = pIface; return m_HLS; } void CSourceHookImpl::HookLoopDone() { m_CurIface = NULL; } //////////////////////////// // CCallClassImpl //////////////////////////// CSourceHookImpl::CCallClassImpl::CCallClassImpl(void *ptr, size_t size) : m_Ptr(ptr), m_ObjSize(size), m_RefCounter(1) { } CSourceHookImpl::CCallClassImpl::~CCallClassImpl() { } void *CSourceHookImpl::CCallClassImpl::GetThisPtr() { return m_Ptr; } void *CSourceHookImpl::CCallClassImpl::GetOrigFunc(int vtbloffs, int vtblidx) { OrigVTables::iterator iter = m_VT.find(vtbloffs); if (iter != m_VT.end() && iter->val.size() > (size_t)vtblidx) { return iter->val[vtblidx]; } return NULL; } void CSourceHookImpl::CCallClassImpl::ApplyCallClassPatch(int vtbl_offs, int vtbl_idx, void *orig_entry) { OrigFuncs &tmpvec = m_VT[vtbl_offs]; if (tmpvec.size() <= (size_t)vtbl_idx) tmpvec.resize(vtbl_idx+1); tmpvec[vtbl_idx] = orig_entry; } void CSourceHookImpl::CCallClassImpl::RemoveCallClassPatch(int vtbl_offs, int vtbl_idx) { OrigVTables::iterator iter = m_VT.find(vtbl_offs); if (iter != m_VT.end()) { if (iter->val.size() > (size_t)vtbl_idx) { iter->val[vtbl_idx] = 0; OrigFuncs &of = iter->val; OrigFuncs::iterator lastused = of.end(); for (OrigFuncs::iterator viter = of.begin(); viter != of.end(); ++viter) { if (*viter) lastused = viter; } if (lastused == of.end()) { // No used element => Remove the whole vector m_VT.erase(iter); return; } of.resize(lastused - of.begin() + 1); } } } //////////////////////////// // CHookManagerInfo //////////////////////////// CSourceHookImpl::CHookManagerInfo::~CHookManagerInfo() { } IVfnPtr *CSourceHookImpl::CHookManagerInfo::FindVfnPtr(void *vfnptr) { VfnPtrListIter iter = m_VfnPtrs.find(vfnptr); return iter == m_VfnPtrs.end() ? NULL : &(*iter); } void CSourceHookImpl::CHookManagerInfo::SetInfo(int vtbl_offs, int vtbl_idx, const char *proto) { m_VtblOffs = vtbl_offs; m_VtblIdx = vtbl_idx; m_Proto = proto; } void CSourceHookImpl::CHookManagerInfo::SetHookfuncVfnptr(void *hookfunc_vfnptr) { m_HookfuncVfnptr = hookfunc_vfnptr; } //////////////////////////// // CVfnPtr //////////////////////////// CSourceHookImpl::CVfnPtr::CVfnPtr(void *ptr) : m_Ptr(ptr), m_OrigEntry(*reinterpret_cast(ptr)) { } CSourceHookImpl::CVfnPtr::~CVfnPtr() { } void *CSourceHookImpl::CVfnPtr::GetVfnPtr() { return m_Ptr; } void *CSourceHookImpl::CVfnPtr::GetOrigEntry() { return m_OrigEntry; } IIface *CSourceHookImpl::CVfnPtr::FindIface(void *ptr) { IfaceListIter iter = m_Ifaces.find(ptr); return iter == m_Ifaces.end() ? NULL : &(*iter); } //////////////////////////// // CIface //////////////////////////// CSourceHookImpl::CIface::CIface(void *ptr) : m_Ptr(ptr) { } CSourceHookImpl::CIface::~CIface() { } void *CSourceHookImpl::CIface::GetPtr() { return m_Ptr; } IHookList *CSourceHookImpl::CIface::GetPreHooks() { return &m_PreHooks; } IHookList *CSourceHookImpl::CIface::GetPostHooks() { return &m_PostHooks; } //////////////////////////// // CHookList //////////////////////////// CSourceHookImpl::CHookList::CHookList() : m_FreeIters(NULL), m_UsedIters(NULL) { } CSourceHookImpl::CHookList::CHookList(const CHookList &other) : m_List(other.m_List), m_FreeIters(NULL), m_UsedIters(NULL) { } CSourceHookImpl::CHookList::~CHookList() { while (m_FreeIters) { CIter *iter = m_FreeIters->m_pNext; delete m_FreeIters; m_FreeIters = iter; } } IHookList::IIter *CSourceHookImpl::CHookList::GetIter() { CIter *ret; if (m_FreeIters) { ret = m_FreeIters; m_FreeIters = ret->m_pNext; ret->GoToBegin(); } else { ret = new CIter(this); } ret->m_pNext = m_UsedIters; ret->m_pPrev = NULL; if (m_UsedIters) m_UsedIters->m_pPrev = ret; m_UsedIters = ret; return ret; } void CSourceHookImpl::CHookList::ReleaseIter(IIter *pIter) { CIter *pIter2 = static_cast(pIter); // Unlink from m_UsedIters if (pIter2->m_pNext) pIter2->m_pNext->m_pPrev = pIter2->m_pPrev; if (pIter2->m_pPrev) pIter2->m_pPrev->m_pNext = pIter2->m_pNext; if (pIter2 == m_UsedIters) m_UsedIters = NULL; // Link to m_FreeIters pIter2->m_pNext = m_FreeIters; m_FreeIters = pIter2; } CSourceHookImpl::CHookList::CIter::CIter(CHookList *pList) : m_pList(pList), m_pNext(NULL) { GoToBegin(); } CSourceHookImpl::CHookList::CIter::~CIter() { } void CSourceHookImpl::CHookList::CIter::GoToBegin() { m_Iter = m_pList->m_List.begin(); SkipPaused(); } bool CSourceHookImpl::CHookList::CIter::End() { if (!m_pList) return false; return m_Iter == m_pList->m_List.end(); } void CSourceHookImpl::CHookList::CIter::Next() { if (!m_pList) return; ++m_Iter; SkipPaused(); } void CSourceHookImpl::CHookList::CIter::Clear() { m_pList = NULL; } void CSourceHookImpl::CHookList::CIter::SkipPaused() { while (m_Iter != m_pList->m_List.end() && m_Iter->paused) ++m_Iter; } ISHDelegate *CSourceHookImpl::CHookList::CIter::Handler() { return m_Iter->handler; } int CSourceHookImpl::CHookList::CIter::ThisPtrOffs() { return m_Iter->thisptr_offs; } }