/** * vim: set ts=4 sw=4 tw=99 noet : * ============================================================================= * SourcePawn * Copyright (C) 2004-2009 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 . */ #ifndef _INCLUDE_SOURCEPAWN_VM_API_H_ #define _INCLUDE_SOURCEPAWN_VM_API_H_ /** * @file sp_vm_api.h * @brief Contains all of the object structures used in the SourcePawn API. */ #include #include "sp_vm_types.h" /** SourcePawn Engine API Version */ #define SOURCEPAWN_ENGINE_API_VERSION 4 #define SOURCEPAWN_ENGINE2_API_VERSION 6 #if !defined SOURCEMOD_BUILD #define SOURCEMOD_BUILD #endif #if defined SOURCEMOD_BUILD namespace SourceMod { struct IdentityToken_t; }; #endif struct sp_context_s; typedef struct sp_context_s sp_context_t; namespace SourcePawn { class IVirtualMachine; class IPluginRuntime; /* Parameter flags */ #define SM_PARAM_COPYBACK (1<<0) /**< Copy an array/reference back after call */ /* String parameter flags (separate from parameter flags) */ #define SM_PARAM_STRING_UTF8 (1<<0) /**< String should be UTF-8 handled */ #define SM_PARAM_STRING_COPY (1<<1) /**< String should be copied into the plugin */ #define SM_PARAM_STRING_BINARY (1<<2) /**< String should be handled as binary data */ #if defined SOURCEMOD_BUILD /** * @brief Pseudo-NULL reference types. */ enum SP_NULL_TYPE { SP_NULL_VECTOR = 0, /**< Float[3] reference */ SP_NULL_STRING = 1, /**< const String[1] reference */ }; #endif /** * @brief Represents what a function needs to implement in order to be callable. */ class ICallable { public: /** * @brief Pushes a cell onto the current call. * * @param cell Parameter value to push. * @return Error code, if any. */ virtual int PushCell(cell_t cell) =0; /** * @brief Pushes a cell by reference onto the current call. * NOTE: On Execute, the pointer passed will be modified if copyback is enabled. * NOTE: By reference parameters are cached and thus are not read until execution. * This means you cannot push a pointer, change it, and push it again and expect * two different values to come out. * * @param cell Address containing parameter value to push. * @param flags Copy-back flags. * @return Error code, if any. */ virtual int PushCellByRef(cell_t *cell, int flags=SM_PARAM_COPYBACK) =0; /** * @brief Pushes a float onto the current call. * * @param number Parameter value to push. * @return Error code, if any. */ virtual int PushFloat(float number) =0; /** * @brief Pushes a float onto the current call by reference. * NOTE: On Execute, the pointer passed will be modified if copyback is enabled. * NOTE: By reference parameters are cached and thus are not read until execution. * This means you cannot push a pointer, change it, and push it again and expect * two different values to come out. * * @param number Parameter value to push. & @param flags Copy-back flags. * @return Error code, if any. */ virtual int PushFloatByRef(float *number, int flags=SM_PARAM_COPYBACK) =0; /** * @brief Pushes an array of cells onto the current call. * * On Execute, the pointer passed will be modified if non-NULL and copy-back * is enabled. * * By reference parameters are cached and thus are not read until execution. * This means you cannot push a pointer, change it, and push it again and expect * two different values to come out. * * @param inarray Array to copy, NULL if no initial array should be copied. * @param cells Number of cells to allocate and optionally read from the input array. * @param flags Whether or not changes should be copied back to the input array. * @return Error code, if any. */ virtual int PushArray(cell_t *inarray, unsigned int cells, int flags=0) =0; /** * @brief Pushes a string onto the current call. * * @param string String to push. * @return Error code, if any. */ virtual int PushString(const char *string) =0; /** * @brief Pushes a string or string buffer. * * NOTE: On Execute, the pointer passed will be modified if copy-back is enabled. * * @param buffer Pointer to string buffer. * @param length Length of buffer. * @param sz_flags String flags. In copy mode, the string will be copied * according to the handling (ascii, utf-8, binary, etc). * @param cp_flags Copy-back flags. * @return Error code, if any. */ virtual int PushStringEx(char *buffer, size_t length, int sz_flags, int cp_flags) =0; /** * @brief Cancels a function call that is being pushed but not yet executed. * This can be used be reset for CallFunction() use. */ virtual void Cancel() =0; }; /** * @brief Encapsulates a function call in a plugin. * * NOTE: Function calls must be atomic to one execution context. * NOTE: This object should not be deleted. It lives for the lifetime of the plugin. */ class IPluginFunction : public ICallable { public: /** * @brief Executes the function, resets the pushed parameter list, and * performs any copybacks. * * @param result Pointer to store return value in. * @return Error code, if any. */ virtual int Execute(cell_t *result) =0; /** * @brief Executes the function with the given parameter array. * Parameters are read in forward order (i.e. index 0 is parameter #1) * NOTE: You will get an error if you attempt to use CallFunction() with * previously pushed parameters. * * @param params Array of cell parameters. * @param num_params Number of parameters to push. * @param result Pointer to store result of function on return. * @return SourcePawn error code (if any). */ virtual int CallFunction(const cell_t *params, unsigned int num_params, cell_t *result) =0; /** * @brief Deprecated, do not use. * * @return GetDefaultContext() of parent runtime. */ virtual IPluginContext *GetParentContext() =0; /** * @brief Returns whether the parent plugin is paused. * * @return True if runnable, false otherwise. */ virtual bool IsRunnable() =0; /** * @brief Returns the function ID of this function. * * Note: This was added in API version 4. * * @return Function id. */ virtual funcid_t GetFunctionID() =0; /** * @brief Executes the forward, resets the pushed parameter list, and * performs any copybacks. * * Note: A function can only be executed given a runtime it was created in. * * @param ctx Context to execute the function in. * @param result Pointer to store return value in. * @return Error code, if any. */ virtual int Execute2(IPluginContext *ctx, cell_t *result) =0; /** * @brief Executes the function with the given parameter array. * Parameters are read in forward order (i.e. index 0 is parameter #1) * NOTE: You will get an error if you attempt to use CallFunction() with * previously pushed parameters. * * Note: A function can only be executed given a runtime it was created in. * * @param ctx Context to execute the function in. * @param params Array of cell parameters. * @param num_params Number of parameters to push. * @param result Pointer to store result of function on return. * @return SourcePawn error code (if any). */ virtual int CallFunction2(IPluginContext *ctx, const cell_t *params, unsigned int num_params, cell_t *result) =0; /** * @brief Returns parent plugin's runtime * * @return IPluginRuntime pointer. */ virtual IPluginRuntime *GetParentRuntime() =0; }; /** * @brief Interface to managing a debug context at runtime. */ class IPluginDebugInfo { public: /** * @brief Given a code pointer, finds the file it is associated with. * * @param addr Code address offset. * @param filename Pointer to store filename pointer in. */ virtual int LookupFile(ucell_t addr, const char **filename) =0; /** * @brief Given a code pointer, finds the function it is associated with. * * @param addr Code address offset. * @param name Pointer to store function name pointer in. */ virtual int LookupFunction(ucell_t addr, const char **name) =0; /** * @brief Given a code pointer, finds the line it is associated with. * * @param addr Code address offset. * @param line Pointer to store line number in. */ virtual int LookupLine(ucell_t addr, uint32_t *line) =0; }; /** * @brief Represents a JIT compilation or plugin loading options. */ class ICompilation { public: /** * @brief Sets a compilation option. * * @param key Option name. * @param val Option value. * @return True on success, false on failure. */ virtual bool SetOption(const char *key, const char *val) =0; /** * @brief Aborts the compilation and destroys this object. */ virtual void Abort() =0; }; /** * @brief Interface to managing a runtime plugin. */ class IPluginRuntime { public: /** * @brief Virtual destructor (you may call delete). */ virtual ~IPluginRuntime() { } /** * @brief Returns debug info. * * @return IPluginDebugInfo, or NULL if no debug info found. */ virtual IPluginDebugInfo *GetDebugInfo() =0; /** * @brief Finds a native by name. * * @param name Name of native. * @param index Optionally filled with native index number. */ virtual int FindNativeByName(const char *name, uint32_t *index) =0; /** * @brief Gets native info by index. * * @param index Index number of native. * @param native Optionally filled with pointer to native structure. */ virtual int GetNativeByIndex(uint32_t index, sp_native_t **native) =0; /** * @brief Gets the number of natives. * * @return Filled with the number of natives. */ virtual uint32_t GetNativesNum() =0; /** * @brief Finds a public function by name. * * @param name Name of public * @param index Optionally filled with public index number. */ virtual int FindPublicByName(const char *name, uint32_t *index) =0; /** * @brief Gets public function info by index. * * @param index Public function index number. * @param publicptr Optionally filled with pointer to public structure. */ virtual int GetPublicByIndex(uint32_t index, sp_public_t **publicptr) =0; /** * @brief Gets the number of public functions. * * @return Filled with the number of public functions. */ virtual uint32_t GetPublicsNum() =0; /** * @brief Gets public variable info by index. * * @param index Public variable index number. * @param pubvar Optionally filled with pointer to pubvar structure. */ virtual int GetPubvarByIndex(uint32_t index, sp_pubvar_t **pubvar) =0; /** * @brief Finds a public variable by name. * * @param name Name of pubvar * @param index Optionally filled with pubvar index number. */ virtual int FindPubvarByName(const char *name, uint32_t *index) =0; /** * @brief Gets the addresses of a public variable. * * @param index Index of public variable. * @param local_addr Address to store local address in. * @param phys_addr Address to store physically relocated in. */ virtual int GetPubvarAddrs(uint32_t index, cell_t *local_addr, cell_t **phys_addr) =0; /** * @brief Returns the number of public variables. * * @return Number of public variables. */ virtual uint32_t GetPubVarsNum() =0; /** * @brief Returns a function by name. * * @param public_name Name of the function. * @return A new IPluginFunction pointer, NULL if not found. */ virtual IPluginFunction *GetFunctionByName(const char *public_name) =0; /** * @brief Returns a function by its id. * * @param func_id Function ID. * @return A new IPluginFunction pointer, NULL if not found. */ virtual IPluginFunction *GetFunctionById(funcid_t func_id) =0; /** * @brief Returns the default context. The default context * should not be destroyed. * * @return Default context pointer. */ virtual IPluginContext *GetDefaultContext() =0; /** * @brief Returns true if the plugin is in debug mode. * * @return True if in debug mode, false otherwise. */ virtual bool IsDebugging() =0; /** * @brief Applies new compilation/runtime settings to the runtime code. * * The compilation object is destroyed once this function completes. * * @return Error code (SP_ERROR_NONE on success). */ virtual int ApplyCompilationOptions(ICompilation *co) =0; /** * @brief Sets whether or not the plugin is paused (cannot be run). * * @param pause Pause state. */ virtual void SetPauseState(bool paused) =0; /** * @brief Returns whether or not the plugin is paused (runnable). * * @return Pause state (true = paused, false = not). */ virtual bool IsPaused() =0; /** * @brief Returns the estimated memory usage of this plugin. * * @return Memory usage, in bytes. */ virtual size_t GetMemUsage() =0; /** * @brief Returns the MD5 hash of the plugin's P-Code. * * @return 16-byte buffer with MD5 hash of the plugin's P-Code. */ virtual unsigned char *GetCodeHash() =0; /** * @brief Returns the MD5 hash of the plugin's Data. * * @return 16-byte buffer with MD5 hash of the plugin's Data. */ virtual unsigned char *GetDataHash() =0; }; /** * @brief Interface to managing a context at runtime. */ class IPluginContext { public: /** Virtual destructor */ virtual ~IPluginContext() { }; public: /** * @brief Deprecated, does nothing. * * @return NULL. */ virtual IVirtualMachine *GetVirtualMachine() =0; /** * @brief Deprecated, do not use. * * Returns the pointer of this object, casted to an opaque structure. * * @return Returns this. */ virtual sp_context_t *GetContext() =0; /** * @brief Returns true if the plugin is in debug mode. * * @return True if in debug mode, false otherwise. */ virtual bool IsDebugging() =0; /** * @brief Deprecated, does nothing. * * @param newpfn Unused. * @param oldpfn Unused. */ virtual int SetDebugBreak(void *newpfn, void *oldpfn) =0; /** * @brief Deprecated, do not use. * * @return NULL. */ virtual IPluginDebugInfo *GetDebugInfo() =0; /** * @brief Allocates memory on the secondary stack of a plugin. * Note that although called a heap, it is in fact a stack. * * @param cells Number of cells to allocate. * @param local_addr Will be filled with data offset to heap. * @param phys_addr Physical address to heap memory. */ virtual int HeapAlloc(unsigned int cells, cell_t *local_addr, cell_t **phys_addr) =0; /** * @brief Pops a heap address off the heap stack. Use this to free memory allocated with * SP_HeapAlloc(). * Note that in SourcePawn, the heap is in fact a bottom-up stack. Deallocations * with this native should be performed in precisely the REVERSE order. * * @param local_addr Local address to free. */ virtual int HeapPop(cell_t local_addr) =0; /** * @brief Releases a heap address using a different method than SP_HeapPop(). * This allows you to release in any order. However, if you allocate N * objects, release only some of them, then begin allocating again, * you cannot go back and starting freeing the originals. * In other words, for each chain of allocations, if you start deallocating, * then allocating more in a chain, you must only deallocate from the current * allocation chain. This is basically HeapPop() except on a larger scale. * * @param local_addr Local address to free. */ virtual int HeapRelease(cell_t local_addr) =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @param name Name of native. * @param index Optionally filled with native index number. */ virtual int FindNativeByName(const char *name, uint32_t *index) =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @param index Index number of native. * @param native Optionally filled with pointer to native structure. */ virtual int GetNativeByIndex(uint32_t index, sp_native_t **native) =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @return Filled with the number of natives. */ virtual uint32_t GetNativesNum() =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @param name Name of public * @param index Optionally filled with public index number. */ virtual int FindPublicByName(const char *name, uint32_t *index) =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @param index Public function index number. * @param publicptr Optionally filled with pointer to public structure. */ virtual int GetPublicByIndex(uint32_t index, sp_public_t **publicptr) =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @return Filled with the number of public functions. */ virtual uint32_t GetPublicsNum() =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @param index Public variable index number. * @param pubvar Optionally filled with pointer to pubvar structure. */ virtual int GetPubvarByIndex(uint32_t index, sp_pubvar_t **pubvar) =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @param name Name of pubvar * @param index Optionally filled with pubvar index number. */ virtual int FindPubvarByName(const char *name, uint32_t *index) =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @param index Index of public variable. * @param local_addr Address to store local address in. * @param phys_addr Address to store physically relocated in. */ virtual int GetPubvarAddrs(uint32_t index, cell_t *local_addr, cell_t **phys_addr) =0; /** * @brief Deprecated, use IPluginRuntime instead. * * @return Number of public variables. */ virtual uint32_t GetPubVarsNum() =0; /** * @brief Converts a plugin reference to a physical address * * @param local_addr Local address in plugin. * @param phys_addr Optionally filled with relocated physical address. */ virtual int LocalToPhysAddr(cell_t local_addr, cell_t **phys_addr) =0; /** * @brief Converts a local address to a physical string. * * @param local_addr Local address in plugin. * @param addr Destination output pointer. */ virtual int LocalToString(cell_t local_addr, char **addr) =0; /** * @brief Converts a physical string to a local address. * * @param local_addr Local address in plugin. * @param bytes Number of chars to write, including NULL terminator. * @param source Source string to copy. */ virtual int StringToLocal(cell_t local_addr, size_t bytes, const char *source) =0; /** * @brief Converts a physical UTF-8 string to a local address. * This function is the same as the ANSI version, except it will copy the maximum number * of characters possible without accidentally chopping a multi-byte character. * * @param local_addr Local address in plugin. * @param maxbytes Number of bytes to write, including NULL terminator. * @param source Source string to copy. * @param wrtnbytes Optionally set to the number of actual bytes written. */ virtual int StringToLocalUTF8(cell_t local_addr, size_t maxbytes, const char *source, size_t *wrtnbytes) =0; /** * @brief Deprecated, does nothing. * * @param value Unused. */ virtual int PushCell(cell_t value) =0; /** * @brief Deprecated, does nothing. * * @param local_addr Unused. * @param phys_addr Unused. * @param array Unused. * @param numcells Unused. */ virtual int PushCellArray(cell_t *local_addr, cell_t **phys_addr, cell_t array[], unsigned int numcells) =0; /** * @brief Deprecated, does nothing. * * @param local_addr Unused. * @param phys_addr Unused. * @param string Unused. */ virtual int PushString(cell_t *local_addr, char **phys_addr, const char *string) =0; /** * @brief Deprecated, does nothing. * * @param array Unused. * @param numcells Unused. */ virtual int PushCellsFromArray(cell_t array[], unsigned int numcells) =0; /** * @brief Deprecated, does nothing. * * @param natives Deprecated; do not use. * @param num Deprecated; do not use. * @param overwrite Deprecated; do not use. */ virtual int BindNatives(const sp_nativeinfo_t *natives, unsigned int num, int overwrite) =0; /** * @brief Deprecated, does nothing. * * @param native Deprecated; do not use. */ virtual int BindNative(const sp_nativeinfo_t *native) =0; /** * @brief Deprecated, does nothing. * * @param native Unused. */ virtual int BindNativeToAny(SPVM_NATIVE_FUNC native) =0; /** * @brief Deprecated, does nothing. * * @param code_addr Unused. * @param result Unused. * @return SP_ERROR_ABORTED. */ virtual int Execute(uint32_t code_addr, cell_t *result) =0; /** * @brief Throws a error and halts any current execution. * * @param error The error number to set. * @param msg Custom error message format. NULL to use default. * @param ... Message format arguments, if any. * @return 0 for convenience. */ virtual cell_t ThrowNativeErrorEx(int error, const char *msg, ...) =0; /** * @brief Throws a generic native error and halts any current execution. * * @param msg Custom error message format. NULL to set no message. * @param ... Message format arguments, if any. * @return 0 for convenience. */ virtual cell_t ThrowNativeError(const char *msg, ...) =0; /** * @brief Returns a function by name. * * @param public_name Name of the function. * @return A new IPluginFunction pointer, NULL if not found. */ virtual IPluginFunction *GetFunctionByName(const char *public_name) =0; /** * @brief Returns a function by its id. * * @param func_id Function ID. * @return A new IPluginFunction pointer, NULL if not found. */ virtual IPluginFunction *GetFunctionById(funcid_t func_id) =0; /** * @brief Returns the identity token for this context. * * Note: This is a compatibility shim and is the same as GetKey(1). * * @return Identity token. */ virtual SourceMod::IdentityToken_t *GetIdentity() =0; /** * @brief Returns a NULL reference based on one of the available NULL * reference types. * * @param type NULL reference type. * @return cell_t address to compare to. */ virtual cell_t *GetNullRef(SP_NULL_TYPE type) =0; /** * @brief Converts a local address to a physical string, and allows * for NULL_STRING to be set. * * @param local_addr Local address in plugin. * @param addr Destination output pointer. */ virtual int LocalToStringNULL(cell_t local_addr, char **addr) =0; /** * @brief Deprecated; do not use. * * @param index Deprecated; do not use. * @param native Deprecated; do not use. */ virtual int BindNativeToIndex(uint32_t index, SPVM_NATIVE_FUNC native) =0; /** * @brief Returns if there is currently an execution in progress. * * @return True if in exec, false otherwise. */ virtual bool IsInExec() =0; /** * @brief Returns the parent runtime of a context. * * @return Parent runtime. */ virtual IPluginRuntime *GetRuntime() =0; /** * @brief Executes a function in the context. The function must be * a member of the context's parent runtime. * * @param function Function. * @param params Parameters. * @param num_params Number of parameters in the parameter array. * @param result Optional pointer to store the result on success. * @return Error code. */ virtual int Execute2(IPluginFunction *function, const cell_t *params, unsigned int num_params, cell_t *result) =0; /** * @brief Returns whether a context is in an error state. * * This should only be used inside natives to determine whether * a prior call failed. */ virtual int GetLastNativeError() =0; /** * @brief Returns the local parameter stack, starting from the * cell that contains the number of parameters passed. * * Local parameters are the parameters passed to the function * from which a native was called (and thus this can only be * called inside a native). * * @return Parameter stack. */ virtual cell_t *GetLocalParams() =0; /** * @brief Sets a local "key" that can be used for fast lookup. * * Only the "owner" of the context should be setting this. * * @param key Key number (values allowed: 1 through 4). * @param value Pointer value. */ virtual void SetKey(int k, void *value) =0; /** * @brief Retrieves a previously set "key." * * @param key Key number (values allowed: 1 through 4). * @param value Pointer to store value. * @return True on success, false on failure. */ virtual bool GetKey(int k, void **value) =0; /** * @brief Clears the last native error. */ virtual void ClearLastNativeError() =0; }; /** * @brief Information about a position in a call stack. */ struct CallStackInfo { const char *filename; /**< NULL if not found */ unsigned int line; /**< 0 if not found */ const char *function; /**< NULL if not found */ }; /** * @brief Retrieves error information from a debug hook. */ class IContextTrace { public: /** * @brief Returns the integer error code. * * @return Integer error code. */ virtual int GetErrorCode() =0; /** * @brief Returns a string describing the error. * * @return Error string. */ virtual const char *GetErrorString() =0; /** * @brief Returns whether debug info is available. * * @return True if debug info is available, false otherwise. */ virtual bool DebugInfoAvailable() =0; /** * @brief Returns a custom error message. * * @return A pointer to a custom error message, or NULL otherwise. */ virtual const char *GetCustomErrorString() =0; /** * @brief Returns trace info for a specific point in the backtrace, if any. * The next subsequent call to GetTraceInfo() will return the next item in the call stack. * Calls are retrieved in descending order (i.e. the first item is at the top of the stack/call sequence). * * @param trace An ErrorTraceInfo buffer to store information (NULL to ignore). * @return True if successful, false if there are no more traces. */ virtual bool GetTraceInfo(CallStackInfo *trace) =0; /** * @brief Resets the trace to its original position (the call on the top of the stack). */ virtual void ResetTrace() =0; /** * @brief Retrieves the name of the last native called. * Returns NULL if there was no native that caused the error. * * @param index Optional pointer to store index. * @return Native name, or NULL if none. */ virtual const char *GetLastNative(uint32_t *index) =0; }; /** * @brief Provides callbacks for debug information. */ class IDebugListener { public: /** * @brief Invoked on a context execution error. * * @param ctx Context. * @param error Object holding error information and a backtrace. */ virtual void OnContextExecuteError(IPluginContext *ctx, IContextTrace *error) =0; /** * @brief Called on debug spew. * * @param msg Message text. * @param fmt Message formatting arguments (printf-style). */ virtual void OnDebugSpew(const char *msg, ...) =0; }; /** * @brief Represents a code profiler for plugins. */ class IProfiler { public: /** * @brief Invoked by the JIT to notify that a native is being started. * * @param pContext Plugin context. * @param native Native information. */ virtual void OnNativeBegin(IPluginContext *pContext, sp_native_t *native) =0; /** * @brief Invoked by the JIT to notify that the last native on the stack * is no longer being executed. */ virtual void OnNativeEnd() =0; /** * @brief Invoked by the JIT to notify that a function call is starting. * * @param pContext Plugin context. * @param name Function name, or NULL if not known. * @param code_addr P-Code address. */ virtual void OnFunctionBegin(IPluginContext *pContext, const char *name) =0; /** * @brief Invoked by the JIT to notify that the last function call has * concluded. In the case of an error inside a function, this will not * be called. Instead, the VM will call OnCallbackEnd() and the profiler * stack must be unwound. */ virtual void OnFunctionEnd() =0; /** * @brief Invoked by the VM to notify that a forward/callback is starting. * * @param pContext Plugin context. * @param pubfunc Public function information. * @return Unique number to pass to OnFunctionEnd(). */ virtual int OnCallbackBegin(IPluginContext *pContext, sp_public_t *pubfunc) =0; /** * @brief Invoked by the JIT to notify that a callback has ended. * * As noted in OnFunctionEnd(), this my be called with a misaligned * profiler stack. To correct this, the stack should be unwound * (discarding data as appropriate) to a matching serial number. * * @param serial Unique number from OnCallbackBegin(). */ virtual void OnCallbackEnd(int serial) =0; }; struct sp_plugin_s; typedef struct sp_plugin_s sp_plugin_t; /** * @brief Contains helper functions used by VMs and the host app */ class ISourcePawnEngine { public: /** * @brief Deprecated, do not use. * * @param fp Unused. * @param err Unused. * @return NULL. */ virtual sp_plugin_t *LoadFromFilePointer(FILE *fp, int *err) =0; /** * @brief Deprecated, do not use, * * @param base Unused. * @param plugin Unused. * @param err Unused. * @return NULL. */ virtual sp_plugin_t *LoadFromMemory(void *base, sp_plugin_t *plugin, int *err) =0; /** * @brief Deprecated, do not use. * * @param plugin Unused. * @return SP_ERROR_ABORTED. */ virtual int FreeFromMemory(sp_plugin_t *plugin) =0; /** * @brief Allocates large blocks of temporary memory. * * @param size Size of memory to allocate. * @return Pointer to memory, NULL if allocation failed. */ virtual void *BaseAlloc(size_t size) =0; /** * @brief Frees memory allocated with BaseAlloc. * * @param memory Memory address to free. */ virtual void BaseFree(void *memory) =0; /** * @brief Allocates executable memory. * @deprecated Use AllocPageMemory() * * @param size Size of memory to allocate. * @return Pointer to memory, NULL if allocation failed. */ virtual void *ExecAlloc(size_t size) =0; /** * @brief Frees executable memory. * @deprecated Use FreePageMemory() * * @param address Address to free. */ virtual void ExecFree(void *address) =0; /** * @brief Sets the debug listener. This should only be called once. * If called successively (using manual chaining), only the last function should * attempt to call back into the same plugin. Otherwise, globally cached states * can be accidentally overwritten. * * @param listener Pointer to an IDebugListener. * @return Old IDebugListener, or NULL if none. */ virtual IDebugListener *SetDebugListener(IDebugListener *listener) =0; /** * @brief Deprecated, do not use. * * @return 0. */ virtual unsigned int GetContextCallCount() =0; /** * @brief Returns the engine API version. * * @return Engine API version. */ virtual unsigned int GetEngineAPIVersion() =0; /** * @brief Allocates executable memory. * * @param size Size of memory to allocate. * @return Pointer to memory, NULL if allocation failed. */ virtual void *AllocatePageMemory(size_t size) =0; /** * @brief Sets the input memory permissions to read+write. * * @param ptr Memory block. */ virtual void SetReadWrite(void *ptr) =0; /** * @brief Sets the input memory permissions to read+execute. * * @param ptr Memory block. */ virtual void SetReadExecute(void *ptr) =0; /** * @brief Frees executable memory. * * @param ptr Address to free. */ virtual void FreePageMemory(void *ptr) =0; }; /** * @brief Outlines the interface a Virtual Machine (JIT) must expose */ class ISourcePawnEngine2 { public: /** * @brief Returns the second engine API version. * * @return API version. */ virtual unsigned int GetAPIVersion() =0; /** * @brief Returns the string name of a VM implementation. */ virtual const char *GetEngineName() =0; /** * @brief Returns a version string. * * @return Versioning string. */ virtual const char *GetVersionString() =0; /** * @brief Creates a new compilation options object. * * @return Compilation options object. */ virtual ICompilation *StartCompilation() =0; /** * @brief Loads a plugin from disk. * * If a compilation object is supplied, it is destroyed upon * the function's return. * * @param co Compilation options, or NULL for defaults. * @param file Path to the file to compile. * @param err Error code (filled on failure); required. * @return New runtime pointer, or NULL on failure. */ virtual IPluginRuntime *LoadPlugin(ICompilation *co, const char *file, int *err) =0; /** * @brief Creates a fake native and binds it to a general callback function. * * @param callback Callback function to bind the native to. * @param pData Private data to pass to the callback when the native is invoked. * @return A new fake native function as a wrapper around the callback. */ virtual SPVM_NATIVE_FUNC CreateFakeNative(SPVM_FAKENATIVE_FUNC callback, void *pData) =0; /** * @brief Destroys a fake native function wrapper. * * @param func Pointer to the fake native created by CreateFakeNative. */ virtual void DestroyFakeNative(SPVM_NATIVE_FUNC func) =0; /** * @brief Sets the debug listener. This should only be called once. * If called successively (using manual chaining), only the last function should * attempt to call back into the same plugin. Otherwise, globally cached states * can be accidentally overwritten. * * @param listener Pointer to an IDebugListener. * @return Old IDebugListener, or NULL if none. */ virtual IDebugListener *SetDebugListener(IDebugListener *listener) =0; /** * @brief Sets the global profiler. * * @param profiler Profiler pointer. */ virtual void SetProfiler(IProfiler *profiler) =0; /** * @brief Returns the string representation of an error message. * * @param err Error code. * @return Error string, or NULL if not found. */ virtual const char *GetErrorString(int err) =0; /** * @brief Initializes the SourcePawn engine. * * @return True on success, false if failed. */ virtual bool Initialize() =0; /** * @brief Shuts down the SourcePawn engine. Only needs to be called if * Initialize() succeeded. */ virtual void Shutdown() =0; /** * @brief Creates an empty plugin with a blob of memory. * * @param name Name, for debugging (NULL for anonymous). * @param bytes Number of bytes of memory (hea+stk). * @return New runtime, or NULL if not enough memory. */ virtual IPluginRuntime *CreateEmptyRuntime(const char *name, uint32_t memory) =0; /** * @brief Initiates the watchdog timer with the specified timeout * length. This cannot be called more than once. * * @param timeout Timeout, in ms. * @return True on success, false on failure. */ virtual bool InstallWatchdogTimer(size_t timeout_ms) =0; /** * @brief Sets whether the JIT is enabled or disabled. * * @param enabled True or false to enable or disable. * @return True if successful, false otherwise. */ virtual bool SetJitEnabled(bool enabled) =0; /** * @brief Returns whether the JIT is enabled. * * @return True if the JIT is enabled, false otherwise. */ virtual bool IsJitEnabled() =0; }; }; #endif //_INCLUDE_SOURCEPAWN_VM_API_H_