/* (c) Magnus Auvinen. See licence.txt in the root of the distribution for more information. */ /* If you are missing that file, acquire a complete release at teeworlds.com. */ #include #include class CKernel : public IKernel { enum { MAX_INTERFACES = 32, }; class CInterfaceInfo { public: CInterfaceInfo() { m_aName[0] = 0; m_pInterface = 0x0; m_AutoDestroy = false; } char m_aName[64]; IInterface *m_pInterface; bool m_AutoDestroy; }; CInterfaceInfo m_aInterfaces[MAX_INTERFACES]; int m_NumInterfaces; CInterfaceInfo *FindInterfaceInfo(const char *pName) { for(int i = 0; i < m_NumInterfaces; i++) { if(str_comp(pName, m_aInterfaces[i].m_aName) == 0) return &m_aInterfaces[i]; } return 0x0; } public: CKernel() { m_NumInterfaces = 0; } virtual ~CKernel() { // delete interfaces in reverse order just the way it would happen to objects on the stack for(int i = m_NumInterfaces - 1; i >= 0; i--) { if(m_aInterfaces[i].m_AutoDestroy) { delete m_aInterfaces[i].m_pInterface; m_aInterfaces[i].m_pInterface = 0; } } } bool RegisterInterfaceImpl(const char *pName, IInterface *pInterface, bool Destroy) override { // TODO: More error checks here if(!pInterface) { dbg_msg("kernel", "ERROR: couldn't register interface %s. null pointer given", pName); return false; } if(m_NumInterfaces == MAX_INTERFACES) { dbg_msg("kernel", "ERROR: couldn't register interface '%s'. maximum of interfaces reached", pName); return false; } if(FindInterfaceInfo(pName) != 0) { dbg_msg("kernel", "ERROR: couldn't register interface '%s'. interface already exists", pName); return false; } pInterface->m_pKernel = this; m_aInterfaces[m_NumInterfaces].m_pInterface = pInterface; str_copy(m_aInterfaces[m_NumInterfaces].m_aName, pName); m_aInterfaces[m_NumInterfaces].m_AutoDestroy = Destroy; m_NumInterfaces++; return true; } bool ReregisterInterfaceImpl(const char *pName, IInterface *pInterface) override { if(FindInterfaceInfo(pName) == 0) { dbg_msg("kernel", "ERROR: couldn't reregister interface '%s'. interface doesn't exist", pName); return false; } pInterface->m_pKernel = this; return true; } IInterface *RequestInterfaceImpl(const char *pName) override { CInterfaceInfo *pInfo = FindInterfaceInfo(pName); if(!pInfo) { dbg_msg("kernel", "failed to find interface with the name '%s'", pName); return 0; } return pInfo->m_pInterface; } }; IKernel *IKernel::Create() { return new CKernel; }