/** * vim: set ts=4 : * ============================================================================= * SourceMod * Copyright (C) 2004-2007 AlliedModders LLC. All rights reserved. * ============================================================================= * * This program is free software; you can redistribute it and/or modify it under * the terms of the GNU General Public License, version 3.0, as published by the * Free Software Foundation. * * 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 GNU General Public License for more * details. * * You should have received a copy of the GNU General Public License along with * this program. If not, see . * * As a special exception, AlliedModders LLC gives you permission to link the * code of this program (as well as its derivative works) to "Half-Life 2," the * "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software * by the Valve Corporation. You must obey the GNU General Public License in * all respects for all other code used. Additionally, AlliedModders LLC grants * this exception to all derivative works. AlliedModders LLC defines further * exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007), * or . * * Version: $Id$ */ #ifndef _INCLUDE_SOURCEMOD_PLUGINSYSTEM_H_ #define _INCLUDE_SOURCEMOD_PLUGINSYSTEM_H_ #include #include #include #include #include #include #include #include #include #include #include "sm_globals.h" #include "vm/sp_vm_basecontext.h" #include "PluginInfoDatabase.h" #include "sm_trie.h" #include "sourcemod.h" #include #if defined ORANGEBOX_BUILD #include "convar_sm_ob.h" #else #include "convar_sm.h" #endif using namespace SourceHook; /** * NOTES: * * Currently this system needs a lot of work but it's good skeletally. Plugin creation * is done without actually compiling anything. This is done by Load functions in the * manager. This will need a rewrite when we add context switching. * * The plugin object itself has a few things to note. The most important is that it stores * a table of function objects. The manager marshals allocation and freeing of these objects. * The plugin object can be in erroneous states, they are: * Plugin_Error --> Some error occurred any time during or after compilation. * This error can be cleared since the plugin itself is valid. * However, the state itself being set prevents any runtime action. * Plugin_BadLoad --> The plugin failed to load entirely and nothing can be done to save it. * * If a plugin fails to load externally, it is never added to the internal tracker. However, * plugins that failed to load from the internal loading mechanism are always tracked. This * allows users to see which automatically loaded plugins failed, and makes the interface a bit * more flexible. * * Once a plugin is compiled, it sets its own state to Plugin_Created. This state is still invalid * for execution. SourceMod is a two pass system, and even though the second pass is not implemented * yet, it is structured so Plugin_Created must be switched to Plugin_Running in the second pass. When * implemented, a Created plugin will be switched to Error in the second pass if it not loadable. * * The two pass loading mechanism is described below. Modules/natives are not implemented yet. * PASS ONE: All loadable plugins are found and have the following steps performed: * 1. Loading and compilation is attempted. * 2. If successful, all natives from Core are added. * 3. OnPluginLoad() is called. * 4. If failed, any user natives are scrapped and the process halts here. * 5. If successful, the plugin is ready for Pass 2. * INTERMEDIATE: * 1. All forced modules are loaded. * PASS TWO: All loaded plugins are found and have these steps performed: * 1. Any modules referenced in the plugin that are not already loaded, are loaded. * 2. If any module fails to load and the plugin requires it, load fails and jump to step 6. * 3. If any natives are unresolved, check if they are found in the user-natives pool. * 4. If yes, load succeeds. If not, natives are passed through a native acceptance filter. * 5. If the filter fails, the plugin is marked as failed. * 6. If the plugin has failed to load at this point, any dynamic natives it has added are scrapped. * Furthermore, any plugin that referenced these natives must now have pass 2 re-ran. * PASS THREE (not a real pass): * 7. Once all plugins are deemed to be loaded, OnPluginStart() is called */ #define SM_CONTEXTVAR_MYSELF 0 struct ContextPair { ContextPair() : base(NULL), ctx(NULL), co(NULL) { }; BaseContext *base; sp_context_t *ctx; ICompilation *co; IVirtualMachine *vm; }; struct FakeNative { IPluginContext *ctx; IPluginFunction *call; String name; SPVM_NATIVE_FUNC func; }; enum LoadRes { LoadRes_Successful, LoadRes_AlreadyLoaded, LoadRes_Failure, LoadRes_NeverLoad }; struct AutoConfig { String autocfg; String folder; bool create; }; class CPlugin; struct WeakNative { WeakNative(CPlugin *plugin, uint32_t index) { pl = plugin; idx = index; } CPlugin *pl; uint32_t idx; }; class CPlugin : public IPlugin { friend class CPluginManager; friend class CFunction; public: CPlugin(const char *file); ~CPlugin(); public: PluginType GetType(); SourcePawn::IPluginContext *GetBaseContext(); sp_context_t *GetContext(); const sm_plugininfo_t *GetPublicInfo(); const char *GetFilename(); bool IsDebugging(); PluginStatus GetStatus(); bool SetPauseState(bool paused); unsigned int GetSerial(); const sp_plugin_t *GetPluginStructure(); IdentityToken_t *GetIdentity(); unsigned int CalcMemUsage(); bool SetProperty(const char *prop, void *ptr); bool GetProperty(const char *prop, void **ptr, bool remove=false); public: /** * Creates a plugin object with default values. * If an error buffer is specified, and an error occurs, the error will be copied to the buffer * and NULL will be returned. * If an error buffer is not specified, the error will be copied to an internal buffer and * a valid (but error-stated) CPlugin will be returned. */ static CPlugin *CreatePlugin(const char *file, char *error, size_t maxlength); public: /** * Starts the initial compilation of a plugin. * Returns false if another compilation exists or there is a current context set. */ ICompilation *StartMyCompile(IVirtualMachine *vm); /** * Finalizes a compilation. If error buffer is NULL, the error is saved locally. */ bool FinishMyCompile(char *error, size_t maxlength); void CancelMyCompile(); /** * Sets an error state on the plugin */ void SetErrorState(PluginStatus status, const char *error_fmt, ...); /** * Initializes the plugin's identity information */ void InitIdentity(); /** * Calls the OnPluginLoad function, and sets any failed states if necessary. * NOTE: Valid pre-states are: Plugin_Created * NOTE: If validated, plugin state is changed to Plugin_Loaded * * If the error buffer is NULL, the error message is cached locally. */ bool Call_AskPluginLoad(char *error, size_t maxlength); /** * Calls the OnPluginStart function. * NOTE: Valid pre-states are: Plugin_Created * NOTE: Post-state will be Plugin_Running */ void Call_OnPluginStart(); /** * Calls the OnPluginEnd function. */ void Call_OnPluginEnd(); /** * Calls the OnAllPluginsLoaded function. */ void Call_OnAllPluginsLoaded(); /** * Toggles debug mode in the plugin */ bool ToggleDebugMode(bool debug, char *error, size_t maxlength); /** * Returns true if a plugin is usable. */ bool IsRunnable(); /** * Adds a language file index to the plugin's list. */ void AddLangFile(unsigned int index); /** * Get language file count for this plugin. */ size_t GetLangFileCount(); /** * Get language file index based on the vector index. */ unsigned int GetLangFileByIndex(unsigned int index); public: /** * Returns the modification time during last plugin load. */ time_t GetTimeStamp(); /** * Returns the current modification time of the plugin file. */ time_t GetFileTimeStamp(); /** * Returns true if the plugin was running, but is now invalid. */ bool WasRunning(); Handle_t GetMyHandle(); void AddConfig(bool autoCreate, const char *cfg, const char *folder); unsigned int GetConfigCount(); AutoConfig *GetConfig(unsigned int i); inline void AddLibrary(const char *name) { m_Libraries.push_back(name); } void LibraryActions(bool dropping); protected: void UpdateInfo(); void SetTimeStamp(time_t t); void DependencyDropped(CPlugin *pOwner); private: ContextPair m_ctx; PluginType m_type; char m_filename[PLATFORM_MAX_PATH]; PluginStatus m_status; unsigned int m_serial; sm_plugininfo_t m_info; sp_plugin_t *m_plugin; char m_errormsg[256]; time_t m_LastAccess; IdentityToken_t *m_ident; Handle_t m_handle; bool m_WasRunning; CVector m_PhraseFiles; List m_dependents; List m_dependsOn; List m_fakeNatives; List m_WeakNatives; List m_RequiredLibs; List m_Libraries; Trie *m_pProps; bool m_FakeNativesMissing; bool m_LibraryMissing; CVector m_configs; bool m_bGotAllLoaded; }; class CPluginManager : public IPluginManager, public SMGlobalClass, public IHandleTypeDispatch, public IRootConsoleCommand { friend class CPlugin; public: CPluginManager(); ~CPluginManager(); public: /* Implements iterator class */ class CPluginIterator : public IPluginIterator { public: CPluginIterator(List *mylist); virtual ~CPluginIterator(); virtual bool MorePlugins(); virtual IPlugin *GetPlugin(); virtual void NextPlugin(); void Release(); public: void Reset(); private: List *mylist; List::iterator current; }; friend class CPluginManager::CPluginIterator; public: //IPluginManager IPlugin *LoadPlugin(const char *path, bool debug, PluginType type, char error[], size_t maxlength, bool *wasloaded); bool UnloadPlugin(IPlugin *plugin); IPlugin *FindPluginByContext(const sp_context_t *ctx); unsigned int GetPluginCount(); IPluginIterator *GetPluginIterator(); void AddPluginsListener(IPluginsListener *listener); void RemovePluginsListener(IPluginsListener *listener); public: //SMGlobalClass void OnSourceModAllInitialized(); void OnSourceModShutdown(); public: //IHandleTypeDispatch void OnHandleDestroy(HandleType_t type, void *object); bool GetHandleApproxSize(HandleType_t type, void *object, unsigned int *pSize); public: //IRootConsoleCommand void OnRootConsoleCommand(const char *cmdname, const CCommand &command); public: /** * Loads all plugins not yet loaded */ void LoadAll_FirstPass(const char *config, const char *basedir); /** * Runs the second loading pass for all plugins */ void LoadAll_SecondPass(); /** * Tests a plugin file mask against a local folder. * The alias is searched backwards from localdir - i.e., given this input: * csdm/ban csdm/ban * ban csdm/ban * csdm/ban optional/csdm/ban * All of these will return true for an alias match. * Wildcards are allowed in the filename. */ bool TestAliasMatch(const char *alias, const char *localdir); /** * Returns whether anything loaded will be a late load. */ bool IsLateLoadTime() const; /** * Adds natives from core into the native pool. */ void RegisterNativesFromCore(sp_nativeinfo_t *natives); /** * Converts a Handle to an IPlugin if possible. */ IPlugin *PluginFromHandle(Handle_t handle, HandleError *err); /** * Finds a plugin based on its index. (starts on index 1) */ CPlugin *GetPluginByOrder(int num); int GetOrderOfPlugin(IPlugin *pl); /** * Internal version of FindPluginByContext() */ inline CPlugin *GetPluginByCtx(const sp_context_t *ctx) { return reinterpret_cast(ctx->user[SM_CONTEXTVAR_MYSELF]); } /** * Gets status text for a status code */ const char *GetStatusText(PluginStatus status); /** * Reload or update plugins on level shutdown. */ void ReloadOrUnloadPlugins(); /** * Add public functions from all running or paused * plugins to the specified forward if the names match. */ void AddFunctionsToForward(const char *name, IChangeableForward *pForward); /** * Iterates through plugins to call OnAllPluginsLoaded. */ void AllPluginsLoaded(); CPlugin *GetPluginFromIdentity(IdentityToken_t *pToken); void Shutdown(); void OnLibraryAction(const char *lib, bool is_a_plugin, bool drop); bool LibraryExists(const char *lib); bool ReloadPlugin(CPlugin *pl); void UnloadAll(); CPlugin *FindPluginByConsoleArg(const char *arg); private: LoadRes _LoadPlugin(CPlugin **pPlugin, const char *path, bool debug, PluginType type, char error[], size_t maxlength); void LoadAutoPlugin(const char *plugin); /** * Recursively loads all plugins in the given directory. */ void LoadPluginsFromDir(const char *basedir, const char *localdir); /** * Adds a plugin object. This is wrapped by LoadPlugin functions. */ void AddPlugin(CPlugin *pPlugin); /** * Runs the second loading pass on a plugin. */ bool RunSecondPass(CPlugin *pPlugin, char *error, size_t maxlength); /** * Adds any globally registered natives to a plugin */ void AddCoreNativesToPlugin(CPlugin *pPlugin); /** * Runs an extension pass on a plugin. */ bool LoadOrRequireExtensions(CPlugin *pPlugin, unsigned int pass, char *error, size_t maxlength); /** * Manages required natives. */ bool FindOrRequirePluginDeps(CPlugin *pPlugin, char *error, size_t maxlength); void _SetPauseState(CPlugin *pPlugin, bool pause); protected: /** * Caching internal objects */ void ReleaseIterator(CPluginIterator *iter); public: inline IdentityToken_t *GetIdentity() { return m_MyIdent; } public: bool AddFakeNative(IPluginFunction *pFunction, const char *name, SPVM_FAKENATIVE_FUNC func); SPVM_NATIVE_FUNC FindCoreNative(const char *name); private: void AddFakeNativesToPlugin(CPlugin *pPlugin); void TryRefreshDependencies(CPlugin *pOther); private: List m_listeners; List m_plugins; CStack m_iters; CPluginInfoDatabase m_PluginInfo; Trie *m_LoadLookup; bool m_AllPluginsLoaded; IdentityToken_t *m_MyIdent; Trie *m_pCoreNatives; /* Dynamic native stuff */ List m_Natives; Trie *m_pNativeLookup; bool m_LoadingLocked; }; extern CPluginManager g_PluginSys; #endif //_INCLUDE_SOURCEMOD_PLUGINSYSTEM_H_