2004-09-24 02:45:00 +00:00
//This file should be easily portable.
//The biggest strength of this plugin system is that ALL interactions are performed via
//named functions, this makes it *really* easy to port plugins from one engine to annother.
# include "quakedef.h"
2004-09-30 22:51:15 +00:00
# ifdef PLUGINS
2005-12-15 19:15:39 +00:00
//#define GNUTLS
# ifdef GNUTLS
# if defined(_WIN32) && !defined(MINGW)
//lets rip stuff out of the header and supply a seperate dll.
//gnutls is huge.
//also this helps get around the whole msvc/mingw thing.
struct DSTRUCT ;
typedef struct DSTRUCT * gnutls_certificate_credentials ;
typedef gnutls_certificate_credentials gnutls_certificate_client_credentials ;
typedef struct DSTRUCT * gnutls_anon_client_credentials ;
struct gnutls_session_int ;
typedef struct gnutls_session_int * gnutls_session ;
typedef void * gnutls_transport_ptr ;
2011-05-19 13:34:07 +00:00
typedef enum gnutls_kx_algorithm { GNUTLS_KX_RSA = 1 , GNUTLS_KX_DHE_DSS ,
2005-12-15 19:15:39 +00:00
GNUTLS_KX_DHE_RSA , GNUTLS_KX_ANON_DH , GNUTLS_KX_SRP ,
GNUTLS_KX_RSA_EXPORT , GNUTLS_KX_SRP_RSA , GNUTLS_KX_SRP_DSS
} gnutls_kx_algorithm ;
2011-05-19 13:34:07 +00:00
typedef enum gnutls_certificate_type { GNUTLS_CRT_X509 = 1 , GNUTLS_CRT_OPENPGP
2005-12-15 19:15:39 +00:00
} gnutls_certificate_type ;
typedef enum gnutls_connection_end { GNUTLS_SERVER = 1 , GNUTLS_CLIENT } gnutls_connection_end ;
typedef enum gnutls_credentials_type { GNUTLS_CRD_CERTIFICATE = 1 , GNUTLS_CRD_ANON , GNUTLS_CRD_SRP } gnutls_credentials_type ;
typedef enum gnutls_close_request { GNUTLS_SHUT_RDWR = 0 , GNUTLS_SHUT_WR = 1 } gnutls_close_request ;
# define GNUTLS_E_AGAIN -28
# define GNUTLS_E_INTERRUPTED -52
int ( VARGS * gnutls_bye ) ( gnutls_session session , gnutls_close_request how ) ;
void ( VARGS * gnutls_perror ) ( int error ) ;
int ( VARGS * gnutls_handshake ) ( gnutls_session session ) ;
void ( VARGS * gnutls_transport_set_ptr ) ( gnutls_session session , gnutls_transport_ptr ptr ) ;
int ( VARGS * gnutls_certificate_type_set_priority ) ( gnutls_session session , const int * ) ;
int ( VARGS * gnutls_credentials_set ) ( gnutls_session , gnutls_credentials_type type , void * cred ) ;
int ( VARGS * gnutls_kx_set_priority ) ( gnutls_session session , const int * ) ;
int ( VARGS * gnutls_init ) ( gnutls_session * session , gnutls_connection_end con_end ) ;
int ( VARGS * gnutls_set_default_priority ) ( gnutls_session session ) ;
int ( VARGS * gnutls_certificate_allocate_credentials ) ( gnutls_certificate_credentials * sc ) ;
int ( VARGS * gnutls_anon_allocate_client_credentials ) ( gnutls_anon_client_credentials * sc ) ;
int ( VARGS * gnutls_global_init ) ( void ) ;
int ( VARGS * gnutls_record_send ) ( gnutls_session session , const void * data , size_t sizeofdata ) ;
int ( VARGS * gnutls_record_recv ) ( gnutls_session session , void * data , size_t sizeofdata ) ;
qboolean Init_GNUTLS ( void )
{
HMODULE hmod ;
hmod = LoadLibrary ( " gnutls.dll " ) ;
if ( ! hmod )
return false ;
gnutls_bye = ( void * ) GetProcAddress ( hmod , " gnutls_bye " ) ;
gnutls_perror = ( void * ) GetProcAddress ( hmod , " gnutls_perror " ) ;
gnutls_handshake = ( void * ) GetProcAddress ( hmod , " gnutls_handshake " ) ;
gnutls_transport_set_ptr = ( void * ) GetProcAddress ( hmod , " gnutls_transport_set_ptr " ) ;
gnutls_certificate_type_set_priority = ( void * ) GetProcAddress ( hmod , " gnutls_certificate_type_set_priority " ) ;
gnutls_credentials_set = ( void * ) GetProcAddress ( hmod , " gnutls_credentials_set " ) ;
gnutls_kx_set_priority = ( void * ) GetProcAddress ( hmod , " gnutls_kx_set_priority " ) ;
gnutls_init = ( void * ) GetProcAddress ( hmod , " gnutls_init " ) ;
gnutls_set_default_priority = ( void * ) GetProcAddress ( hmod , " gnutls_set_default_priority " ) ;
gnutls_certificate_allocate_credentials = ( void * ) GetProcAddress ( hmod , " gnutls_certificate_allocate_credentials " ) ;
gnutls_anon_allocate_client_credentials = ( void * ) GetProcAddress ( hmod , " gnutls_anon_allocate_client_credentials " ) ;
gnutls_global_init = ( void * ) GetProcAddress ( hmod , " gnutls_global_init " ) ;
gnutls_record_send = ( void * ) GetProcAddress ( hmod , " gnutls_record_send " ) ;
gnutls_record_recv = ( void * ) GetProcAddress ( hmod , " gnutls_record_recv " ) ;
if ( ! gnutls_bye | | ! gnutls_perror | | ! gnutls_handshake | | ! gnutls_transport_set_ptr
| | ! gnutls_certificate_type_set_priority | | ! gnutls_credentials_set
| | ! gnutls_kx_set_priority | | ! gnutls_init | | ! gnutls_set_default_priority
| | ! gnutls_certificate_allocate_credentials | | ! gnutls_anon_allocate_client_credentials
| | ! gnutls_global_init | | ! gnutls_record_send | | ! gnutls_record_recv )
{
Con_Printf ( " gnutls.dll doesn't contain all required exports \n " ) ;
FreeLibrary ( hmod ) ;
return false ;
}
return true ;
}
# else
# include <gnutls/gnutls.h>
qboolean Init_GNUTLS ( void ) { return true ; }
# endif
# endif
2004-09-24 02:45:00 +00:00
2006-02-11 02:09:43 +00:00
cvar_t plug_sbar = SCVAR ( " plug_sbar " , " 1 " ) ;
cvar_t plug_loaddefault = SCVAR ( " plug_loaddefault " , " 1 " ) ;
2005-01-15 17:46:40 +00:00
2009-11-04 21:16:50 +00:00
# ifdef GLQUAKE
2004-10-10 06:32:29 +00:00
# include "glquake.h"
2005-08-01 14:17:27 +00:00
# endif
2004-10-10 06:32:29 +00:00
2005-12-15 19:45:04 +00:00
//custom plugin builtins.
2009-07-18 20:21:02 +00:00
typedef qintptr_t ( EXPORT_FN * Plug_Builtin_t ) ( void * offset , quintptr_t mask , const qintptr_t * arg ) ;
2005-12-21 03:07:33 +00:00
void Plug_RegisterBuiltin ( char * name , Plug_Builtin_t bi , int flags ) ;
2005-12-15 19:45:04 +00:00
# define PLUG_BIF_DLLONLY 1
# define PLUG_BIF_QVMONLY 2
2006-01-02 22:55:56 +00:00
# define PLUG_BIF_NEEDSRENDERER 4
2005-12-15 19:45:04 +00:00
2005-11-29 13:32:15 +00:00
# include "netinc.h"
2004-09-24 02:45:00 +00:00
typedef struct plugin_s {
char * name ;
vm_t * vm ;
2005-12-15 19:15:39 +00:00
2006-01-29 03:45:00 +00:00
int blockcloses ;
2009-04-06 00:34:32 +00:00
void * inputptr ;
unsigned int inputbytes ;
2004-09-24 02:45:00 +00:00
int tick ;
int executestring ;
2005-12-15 19:15:39 +00:00
# ifndef SERVERONLY
2005-01-13 16:50:37 +00:00
int conexecutecommand ;
2004-09-26 00:30:42 +00:00
int menufunction ;
2004-10-10 06:32:29 +00:00
int sbarlevel [ 3 ] ; //0 - main sbar, 1 - supplementry sbar sections (make sure these can be switched off), 2 - overlays (scoreboard). menus kill all.
int reschange ;
2004-09-24 02:45:00 +00:00
2005-08-03 23:14:59 +00:00
//protocol-in-a-plugin
int connectionlessclientpacket ;
2009-04-06 00:34:32 +00:00
//called to discolour console input text if they spelt it wrongly
int spellcheckmaskedtext ;
2005-12-15 19:15:39 +00:00
# endif
2006-01-01 09:01:15 +00:00
int svmsgfunction ;
int chatmsgfunction ;
int centerprintfunction ;
2012-10-08 04:36:10 +00:00
int shutdown ;
2005-11-30 05:01:43 +00:00
2004-09-24 02:45:00 +00:00
struct plugin_s * next ;
} plugin_t ;
2005-03-10 03:55:18 +00:00
void Plug_SubConsoleCommand ( console_t * con , char * line ) ;
2004-09-24 02:45:00 +00:00
plugin_t * currentplug ;
2005-12-15 19:15:39 +00:00
# ifndef SERVERONLY
# include "cl_plugin.inc"
# else
void Plug_Client_Init ( void ) { }
void Plug_Client_Close ( plugin_t * plug ) { }
# endif
2004-09-24 02:45:00 +00:00
void Plug_Init ( void ) ;
2005-08-03 23:14:59 +00:00
void Plug_Close ( plugin_t * plug ) ;
2004-09-24 02:45:00 +00:00
void Plug_Tick ( void ) ;
qboolean Plugin_ExecuteString ( void ) ;
void Plug_Shutdown ( void ) ;
static plugin_t * plugs ;
typedef struct {
char * name ;
Plug_Builtin_t func ;
2005-11-30 01:20:53 +00:00
int flags ;
2004-09-24 02:45:00 +00:00
} Plug_Plugins_t ;
Plug_Plugins_t * plugbuiltins ;
int numplugbuiltins ;
void Plug_RegisterBuiltin ( char * name , Plug_Builtin_t bi , int flags )
{
//randomize the order a little.
int newnum ;
2005-12-15 19:15:39 +00:00
2005-12-06 15:39:57 +00:00
newnum = ( rand ( ) % 128 ) + 1 ;
2004-09-24 02:45:00 +00:00
while ( newnum < numplugbuiltins & & plugbuiltins [ newnum ] . func )
newnum + = 128 ;
2004-12-04 16:26:22 +00:00
if ( newnum > = numplugbuiltins )
2004-09-24 02:45:00 +00:00
{
2008-05-09 14:22:37 +00:00
int newbuiltins = newnum + 128 ;
plugbuiltins = BZ_Realloc ( plugbuiltins , sizeof ( Plug_Plugins_t ) * newbuiltins ) ;
memset ( plugbuiltins + numplugbuiltins , 0 , sizeof ( Plug_Plugins_t ) * ( newbuiltins - numplugbuiltins ) ) ;
numplugbuiltins = newbuiltins ;
2004-09-24 02:45:00 +00:00
}
//got an empty number.
2005-12-15 19:15:39 +00:00
Con_DPrintf ( " %s: %i \n " , name , newnum ) ;
2004-09-24 02:45:00 +00:00
plugbuiltins [ newnum ] . name = name ;
plugbuiltins [ newnum ] . func = bi ;
2005-11-30 01:20:53 +00:00
plugbuiltins [ newnum ] . flags = flags ;
2004-09-24 02:45:00 +00:00
}
/*
static void Plug_RegisterBuiltinIndex ( char * name , Plug_Builtin_t bi , int flags , int index ) //I d
{
//randomize the order a little.
int newnum ;
2005-12-15 19:15:39 +00:00
2004-09-24 02:45:00 +00:00
newnum = rand ( ) % 128 ;
while ( newnum + 1 < numplugbuiltins & & plugbuiltins [ newnum + 1 ] . func )
newnum + = 128 ;
newnum + + ;
2004-12-04 16:26:22 +00:00
if ( newnum > = numplugbuiltins )
2004-09-24 02:45:00 +00:00
{
numplugbuiltins = newnum + 128 ;
plugbuiltins = BZ_Realloc ( plugbuiltins , sizeof ( Plug_Plugins_t ) * numplugbuiltins ) ;
}
//got an empty number.
plugbuiltins [ newnum ] . name = name ;
plugbuiltins [ newnum ] . func = bi ;
}
*/
2011-01-29 21:01:40 +00:00
static qintptr_t Plug_FindBuiltin ( qboolean native , char * p )
2004-09-24 02:45:00 +00:00
{
int i ;
for ( i = 0 ; i < numplugbuiltins ; i + + )
if ( plugbuiltins [ i ] . name )
2007-10-11 15:40:28 +00:00
if ( p & & ! strcmp ( plugbuiltins [ i ] . name , p ) )
2005-11-30 01:20:53 +00:00
{
2011-01-29 21:01:40 +00:00
if ( ! native & & plugbuiltins [ i ] . flags & PLUG_BIF_DLLONLY )
2005-11-30 01:20:53 +00:00
return 0 ; //block it, if not native
2011-01-29 21:01:40 +00:00
if ( native & & plugbuiltins [ i ] . flags & PLUG_BIF_QVMONLY )
2005-11-30 01:20:53 +00:00
return 0 ; //block it, if not native
2004-09-24 02:45:00 +00:00
return - i ;
2005-11-30 01:20:53 +00:00
}
2004-09-24 02:45:00 +00:00
return 0 ;
}
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_GetBuiltin ( void * offset , quintptr_t mask , const qintptr_t * args )
2011-01-29 21:01:40 +00:00
{
char * p = ( char * ) VM_POINTER ( args [ 0 ] ) ;
return Plug_FindBuiltin ( ! offset , p ) ;
}
2004-09-24 02:45:00 +00:00
2012-10-08 05:10:14 +00:00
static int Plug_SystemCallsVM ( void * offset , quintptr_t mask , int fn , const int * arg )
2004-09-24 02:45:00 +00:00
{
2009-07-18 20:21:02 +00:00
# if FTE_WORDSIZE == 32
# define args arg
# else
qintptr_t args [ 9 ] ;
args [ 0 ] = arg [ 0 ] ;
args [ 1 ] = arg [ 1 ] ;
args [ 2 ] = arg [ 2 ] ;
args [ 3 ] = arg [ 3 ] ;
args [ 4 ] = arg [ 4 ] ;
args [ 5 ] = arg [ 5 ] ;
args [ 6 ] = arg [ 6 ] ;
args [ 7 ] = arg [ 7 ] ;
args [ 8 ] = arg [ 8 ] ;
# endif
2004-09-24 02:45:00 +00:00
fn = fn + 1 ;
if ( fn > = 0 & & fn < numplugbuiltins & & plugbuiltins [ fn ] . func ! = NULL )
2011-06-03 03:10:39 +00:00
return plugbuiltins [ fn ] . func ( offset , mask , ( qintptr_t * ) args ) ;
2009-07-18 20:21:02 +00:00
# undef args
2004-09-24 02:45:00 +00:00
Sys_Error ( " QVM Plugin tried calling invalid builtin %i " , fn ) ;
return 0 ;
}
//I'm not keen on this.
//but dlls call it without saying what sort of vm it comes from, so I've got to have them as specifics
2010-11-02 23:17:25 +00:00
static qintptr_t EXPORT_FN Plug_SystemCallsNative ( qintptr_t arg , . . . )
2004-09-24 02:45:00 +00:00
{
2009-07-18 20:21:02 +00:00
qintptr_t args [ 9 ] ;
2004-09-24 02:45:00 +00:00
va_list argptr ;
va_start ( argptr , arg ) ;
2009-07-18 20:21:02 +00:00
args [ 0 ] = va_arg ( argptr , qintptr_t ) ;
args [ 1 ] = va_arg ( argptr , qintptr_t ) ;
args [ 2 ] = va_arg ( argptr , qintptr_t ) ;
args [ 3 ] = va_arg ( argptr , qintptr_t ) ;
args [ 4 ] = va_arg ( argptr , qintptr_t ) ;
args [ 5 ] = va_arg ( argptr , qintptr_t ) ;
args [ 6 ] = va_arg ( argptr , qintptr_t ) ;
args [ 7 ] = va_arg ( argptr , qintptr_t ) ;
args [ 8 ] = va_arg ( argptr , qintptr_t ) ;
2004-09-24 02:45:00 +00:00
va_end ( argptr ) ;
arg = - arg ;
if ( arg > = 0 & & arg < numplugbuiltins & & plugbuiltins [ arg ] . func )
return plugbuiltins [ arg ] . func ( NULL , ~ 0 , args ) ;
2009-10-06 00:32:28 +00:00
Sys_Error ( " DLL Plugin tried calling invalid builtin %i " , ( int ) arg ) ;
2004-09-24 02:45:00 +00:00
return 0 ;
}
plugin_t * Plug_Load ( char * file )
{
plugin_t * newplug ;
for ( newplug = plugs ; newplug ; newplug = newplug - > next )
{
if ( ! stricmp ( newplug - > name , file ) )
2005-08-03 23:14:59 +00:00
return newplug ;
2004-09-24 02:45:00 +00:00
}
newplug = Z_Malloc ( sizeof ( plugin_t ) + strlen ( file ) + 1 ) ;
newplug - > name = ( char * ) ( newplug + 1 ) ;
strcpy ( newplug - > name , file ) ;
2005-12-15 19:15:39 +00:00
2010-11-02 23:17:25 +00:00
newplug - > vm = VM_Create ( NULL , file , Plug_SystemCallsNative , Plug_SystemCallsVM ) ;
2004-09-24 02:45:00 +00:00
currentplug = newplug ;
if ( newplug - > vm )
{
2005-08-03 23:14:59 +00:00
Con_Printf ( " Created plugin %s \n " , file ) ;
2004-09-24 02:45:00 +00:00
newplug - > next = plugs ;
plugs = newplug ;
2011-01-29 21:01:40 +00:00
if ( ! VM_Call ( newplug - > vm , 0 , Plug_FindBuiltin ( true , " Plug_GetEngineFunction " ) ) )
2005-08-03 23:14:59 +00:00
{
Plug_Close ( newplug ) ;
return NULL ;
}
2004-10-10 06:32:29 +00:00
2005-12-15 19:15:39 +00:00
# ifndef SERVERONLY
2004-10-10 06:32:29 +00:00
if ( newplug - > reschange )
VM_Call ( newplug - > vm , newplug - > reschange , vid . width , vid . height ) ;
2005-12-15 19:15:39 +00:00
# endif
2004-09-24 02:45:00 +00:00
}
else
{
Z_Free ( newplug ) ;
newplug = NULL ;
}
currentplug = NULL ;
return newplug ;
}
2012-10-08 05:10:14 +00:00
static int Plug_Emumerated ( const char * name , int size , void * param )
2004-09-24 02:45:00 +00:00
{
char vmname [ MAX_QPATH ] ;
2009-04-01 22:03:56 +00:00
Q_strncpyz ( vmname , name , sizeof ( vmname ) ) ;
2004-09-24 02:45:00 +00:00
vmname [ strlen ( vmname ) - strlen ( param ) ] = ' \0 ' ;
2005-12-15 19:15:39 +00:00
if ( ! Plug_Load ( vmname ) )
Con_Printf ( " Couldn't load plugin %s \n " , vmname ) ;
2004-09-24 02:45:00 +00:00
return true ;
}
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Con_Print ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-09-24 02:45:00 +00:00
{
2010-02-06 01:25:04 +00:00
// if (qrenderer == QR_NONE)
2007-03-28 13:27:35 +00:00
// return false;
2005-12-15 19:15:39 +00:00
Con_Printf ( " %s " , ( char * ) VM_POINTER ( arg [ 0 ] ) ) ;
2004-09-24 02:45:00 +00:00
return 0 ;
}
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Sys_Error ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-09-24 02:45:00 +00:00
{
Sys_Error ( " %s " , ( char * ) offset + arg [ 0 ] ) ;
return 0 ;
}
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Sys_Milliseconds ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-03-01 15:36:23 +00:00
{
return Sys_DoubleTime ( ) * 1000 ;
}
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_ExportToEngine ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-09-24 02:45:00 +00:00
{
2004-09-26 00:30:42 +00:00
char * name = ( char * ) VM_POINTER ( arg [ 0 ] ) ;
2008-05-25 01:08:54 +00:00
unsigned int functionid = VM_LONG ( arg [ 1 ] ) ;
2004-09-24 02:45:00 +00:00
if ( ! strcmp ( name , " Tick " ) )
2008-05-25 01:08:54 +00:00
currentplug - > tick = functionid ;
2004-09-24 02:45:00 +00:00
else if ( ! strcmp ( name , " ExecuteCommand " ) )
2008-05-25 01:08:54 +00:00
currentplug - > executestring = functionid ;
2012-10-08 04:36:10 +00:00
else if ( ! strcmp ( name , " Shutdown " ) )
currentplug - > shutdown = functionid ;
2005-12-15 19:15:39 +00:00
# ifndef SERVERONLY
2005-01-13 16:50:37 +00:00
else if ( ! strcmp ( name , " ConExecuteCommand " ) )
2008-05-25 01:08:54 +00:00
currentplug - > conexecutecommand = functionid ;
2004-09-26 00:30:42 +00:00
else if ( ! strcmp ( name , " MenuEvent " ) )
2008-05-25 01:08:54 +00:00
currentplug - > menufunction = functionid ;
2004-10-10 06:32:29 +00:00
else if ( ! strcmp ( name , " UpdateVideo " ) )
2008-05-25 01:08:54 +00:00
currentplug - > reschange = functionid ;
2004-10-13 07:24:59 +00:00
else if ( ! strcmp ( name , " SbarBase " ) ) //basic SBAR.
2008-05-25 01:08:54 +00:00
currentplug - > sbarlevel [ 0 ] = functionid ;
2004-10-13 07:24:59 +00:00
else if ( ! strcmp ( name , " SbarSupplement " ) ) //supplementry stuff - teamplay
2008-05-25 01:08:54 +00:00
currentplug - > sbarlevel [ 1 ] = functionid ;
2004-10-13 07:24:59 +00:00
else if ( ! strcmp ( name , " SbarOverlay " ) ) //overlay - scoreboard type stuff.
2008-05-25 01:08:54 +00:00
currentplug - > sbarlevel [ 2 ] = functionid ;
2005-08-03 23:14:59 +00:00
else if ( ! strcmp ( name , " ConnectionlessClientPacket " ) )
2008-05-25 01:08:54 +00:00
currentplug - > connectionlessclientpacket = functionid ;
2006-01-01 09:01:15 +00:00
else if ( ! strcmp ( name , " ServerMessageEvent " ) )
2008-05-25 01:08:54 +00:00
currentplug - > svmsgfunction = functionid ;
2006-01-01 09:01:15 +00:00
else if ( ! strcmp ( name , " ChatMessageEvent " ) )
2008-05-25 01:08:54 +00:00
currentplug - > chatmsgfunction = functionid ;
2006-01-01 09:01:15 +00:00
else if ( ! strcmp ( name , " CenterPrintMessage " ) )
2008-05-25 01:08:54 +00:00
currentplug - > centerprintfunction = functionid ;
2009-04-06 00:34:32 +00:00
else if ( ! strcmp ( name , " SpellCheckMaskedText " ) )
currentplug - > spellcheckmaskedtext = functionid ;
2005-12-15 19:15:39 +00:00
# endif
2004-09-24 02:45:00 +00:00
else
return 0 ;
return 1 ;
}
2004-10-10 06:32:29 +00:00
2005-12-15 19:15:39 +00:00
//retrieve a plugin's name
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_GetPluginName ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-12-15 19:15:39 +00:00
{
int plugnum = VM_LONG ( arg [ 0 ] ) ;
plugin_t * plug ;
//int plugnum (0 for current), char *buffer, int bufferlen
if ( VM_OOB ( arg [ 1 ] , arg [ 2 ] ) )
return false ;
if ( plugnum < = 0 )
{
Q_strncpyz ( VM_POINTER ( arg [ 1 ] ) , currentplug - > name , VM_LONG ( arg [ 2 ] ) ) ;
return true ;
}
for ( plug = plugs ; plug ; plug = plug - > next )
{
if ( - - plugnum = = 0 )
{
Q_strncpyz ( VM_POINTER ( arg [ 1 ] ) , plug - > name , VM_LONG ( arg [ 2 ] ) ) ;
return true ;
}
}
return false ;
}
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_ExportNative ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-11-30 01:20:53 +00:00
{
2012-04-24 07:59:11 +00:00
void * func ;
2005-11-30 01:20:53 +00:00
char * name = ( char * ) VM_POINTER ( arg [ 0 ] ) ;
arg + + ;
2012-04-24 07:59:11 +00:00
func = ( ( void * * ) arg ) [ 0 ] ;
2005-11-30 01:20:53 +00:00
2006-01-29 03:45:00 +00:00
if ( ! strcmp ( name , " UnsafeClose " ) )
{
//not used by the engine, but stops the user from being able to unload the plugin.
//this is useful for certain things, like if the plugin uses some external networking or direct disk access or whatever.
currentplug - > blockcloses + + ;
}
/*
else if ( ! strncmp ( name , " FS_LoadModule " ) ) //module as in pak/pk3
{
FS_RegisterModuleDriver ( name + 13 , func ) ;
currentplug - > blockcloses + + ;
}
*/
/*
else if ( ! strncmp ( name , " S_OutputDriver " ) ) //a sound driver (takes higher priority over the built-in ones)
{
S_RegisterOutputDriver ( name + 13 , func ) ;
currentplug - > blockcloses + + ;
}
*/
/*
else if ( ! strncmp ( name , " VID_DisplayDriver " ) ) //a video driver, loaded by name as given by vid_renderer
{
FS_RegisterModuleDriver ( , func ) ;
currentplug - > blockcloses + + ;
}
*/
2012-04-24 07:59:11 +00:00
# if defined(PLUGINS) && !defined(NOMEDIA) && !defined(SERVERONLY)
else if ( ! strcmp ( name , " Media_VideoDecoder " ) )
{
Media_RegisterDecoder ( currentplug , func ) ;
2012-11-27 03:23:19 +00:00
// currentplug->blockcloses++;
}
else if ( ! strcmp ( name , " Media_VideoEncoder " ) )
{
Media_RegisterEncoder ( currentplug , func ) ;
2012-10-10 22:58:51 +00:00
// currentplug->blockcloses++;
2012-04-24 07:59:11 +00:00
}
# endif
2006-01-29 03:45:00 +00:00
2005-12-15 19:15:39 +00:00
# ifndef SERVERONLY
2006-01-29 03:45:00 +00:00
else if ( ! strcmp ( name , " S_LoadSound " ) ) //a hook for loading extra types of sound (wav, mp3, ogg, midi, whatever you choose to support)
{
2006-02-11 14:51:36 +00:00
S_RegisterSoundInputPlugin ( ( void * ) func ) ;
2006-01-29 03:45:00 +00:00
currentplug - > blockcloses + + ;
}
2005-12-15 19:15:39 +00:00
# endif
2006-01-29 03:45:00 +00:00
else
2005-11-30 01:20:53 +00:00
return 0 ;
return 1 ;
}
2004-10-10 06:32:29 +00:00
typedef struct {
//Make SURE that the engine has resolved all cvar pointers into globals before this happens.
plugin_t * plugin ;
cvar_t * var ;
} plugincvararray_t ;
int plugincvararraylen ;
plugincvararray_t * plugincvararray ;
//qhandle_t Cvar_Register (char *name, char *defaultval, int flags, char *grouphint);
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cvar_Register ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-10-10 06:32:29 +00:00
{
char * name = VM_POINTER ( arg [ 0 ] ) ;
char * defaultvalue = VM_POINTER ( arg [ 1 ] ) ;
unsigned int flags = VM_LONG ( arg [ 2 ] ) ;
char * groupname = VM_POINTER ( arg [ 3 ] ) ;
cvar_t * var ;
int i ;
var = Cvar_Get ( name , defaultvalue , flags & 1 , groupname ) ;
for ( i = 0 ; i < plugincvararraylen ; i + + )
{
if ( ! plugincvararray [ i ] . var )
{ //hmm... a gap...
plugincvararray [ i ] . plugin = currentplug ;
plugincvararray [ i ] . var = var ;
return i ;
}
}
2007-09-17 20:35:39 +00:00
i = plugincvararraylen ;
2004-10-10 06:32:29 +00:00
plugincvararraylen + + ;
2007-09-17 20:35:39 +00:00
plugincvararray = BZ_Realloc ( plugincvararray , ( plugincvararraylen ) * sizeof ( plugincvararray_t ) ) ;
plugincvararray [ i ] . plugin = currentplug ;
plugincvararray [ i ] . var = var ;
return i ;
2004-10-10 06:32:29 +00:00
}
//int Cvar_Update, (qhandle_t handle, int modificationcount, char *stringv, float *floatv)); //stringv is 256 chars long, don't expect this function to do anything if modification count is unchanged.
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cvar_Update ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-10-10 06:32:29 +00:00
{
int handle ;
int modcount ;
char * stringv ; //255 bytes long.
float * floatv ;
cvar_t * var ;
handle = VM_LONG ( arg [ 0 ] ) ;
if ( handle < 0 | | handle > = plugincvararraylen )
return 0 ;
if ( plugincvararray [ handle ] . plugin ! = currentplug )
return 0 ; //I'm not letting you know what annother plugin has registered.
if ( VM_OOB ( arg [ 2 ] , 256 ) | | VM_OOB ( arg [ 3 ] , 4 ) ) //Oi, plugin - you screwed up
return 0 ;
modcount = VM_LONG ( arg [ 1 ] ) ;
stringv = VM_POINTER ( arg [ 2 ] ) ;
floatv = VM_POINTER ( arg [ 3 ] ) ;
var = plugincvararray [ handle ] . var ;
strcpy ( stringv , var - > string ) ;
* floatv = var - > value ;
return var - > modified ;
}
//void Cmd_Args(char *buffer, int buffersize)
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cmd_Args ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-09-24 02:45:00 +00:00
{
2004-09-26 00:30:42 +00:00
char * buffer = ( char * ) VM_POINTER ( arg [ 0 ] ) ;
2004-09-24 02:45:00 +00:00
char * args ;
args = Cmd_Args ( ) ;
if ( strlen ( args ) + 1 > arg [ 1 ] )
return 0 ;
strcpy ( buffer , args ) ;
return 1 ;
}
2004-10-10 06:32:29 +00:00
//void Cmd_Argv(int num, char *buffer, int buffersize)
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cmd_Argv ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-09-24 02:45:00 +00:00
{
2004-09-26 00:30:42 +00:00
char * buffer = ( char * ) VM_POINTER ( arg [ 1 ] ) ;
2004-09-24 02:45:00 +00:00
char * args ;
args = Cmd_Argv ( arg [ 0 ] ) ;
if ( strlen ( args ) + 1 > arg [ 2 ] )
return 0 ;
strcpy ( buffer , args ) ;
return 1 ;
}
2004-10-10 06:32:29 +00:00
//int Cmd_Argc(void)
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cmd_Argc ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-09-24 02:45:00 +00:00
{
return Cmd_Argc ( ) ;
}
2004-09-26 00:30:42 +00:00
2004-10-10 06:32:29 +00:00
//void Cvar_SetString (char *name, char *value);
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cvar_SetString ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-10-10 06:32:29 +00:00
{
char * name = VM_POINTER ( arg [ 0 ] ) ,
* value = VM_POINTER ( arg [ 1 ] ) ;
cvar_t * var = Cvar_Get ( name , value , 0 , " Plugin vars " ) ;
if ( var )
{
Cvar_Set ( var , value ) ;
return 1 ;
}
return 0 ;
}
//void Cvar_SetFloat (char *name, float value);
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cvar_SetFloat ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-10-10 06:32:29 +00:00
{
char * name = VM_POINTER ( arg [ 0 ] ) ;
float value = VM_FLOAT ( arg [ 1 ] ) ;
cvar_t * var = Cvar_Get ( name , " " , 0 , " Plugin vars " ) ; //"" because I'm lazy
if ( var )
{
Cvar_SetValue ( var , value ) ;
return 1 ;
}
return 0 ;
}
//void Cvar_GetFloat (char *name);
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cvar_GetFloat ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-10-10 06:32:29 +00:00
{
char * name = VM_POINTER ( arg [ 0 ] ) ;
int ret ;
cvar_t * var = Cvar_Get ( name , " " , 0 , " Plugin vars " ) ;
if ( var )
{
VM_FLOAT ( ret ) = var - > value ;
}
else
VM_FLOAT ( ret ) = 0 ;
return ret ;
}
//qboolean Cvar_GetString (char *name, char *retstring, int sizeofretstring);
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cvar_GetString ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-10-10 06:32:29 +00:00
{
char * name , * ret ;
int retsize ;
cvar_t * var ;
if ( VM_OOB ( arg [ 1 ] , arg [ 2 ] ) )
{
return false ;
}
name = VM_POINTER ( arg [ 0 ] ) ;
ret = VM_POINTER ( arg [ 1 ] ) ;
retsize = VM_LONG ( arg [ 2 ] ) ;
var = Cvar_Get ( name , " " , 0 , " Plugin vars " ) ;
if ( strlen ( var - > name ) + 1 > retsize )
return false ;
strcpy ( ret , var - > string ) ;
return true ;
}
//void Cmd_AddText (char *text, qboolean insert); //abort the entire engine.
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cmd_AddText ( void * offset , quintptr_t mask , const qintptr_t * arg )
2004-10-10 06:32:29 +00:00
{
if ( VM_LONG ( arg [ 1 ] ) )
2006-02-06 01:06:17 +00:00
Cbuf_InsertText ( VM_POINTER ( arg [ 0 ] ) , RESTRICT_LOCAL , false ) ;
2004-10-10 06:32:29 +00:00
else
Cbuf_AddText ( VM_POINTER ( arg [ 0 ] ) , RESTRICT_LOCAL ) ;
return 1 ;
2004-09-26 00:30:42 +00:00
}
2005-01-15 17:46:40 +00:00
int plugincommandarraylen ;
typedef struct {
plugin_t * plugin ;
char command [ 64 ] ;
} plugincommand_t ;
plugincommand_t * plugincommandarray ;
void Plug_Command_f ( void )
{
int i ;
char * cmd = Cmd_Argv ( 0 ) ;
plugin_t * oldplug = currentplug ;
for ( i = 0 ; i < plugincommandarraylen ; i + + )
{
if ( ! plugincommandarray [ i ] . plugin )
continue ; //don't check commands who's owners died.
if ( stricmp ( plugincommandarray [ i ] . command , cmd ) ) //not the right command
continue ;
currentplug = plugincommandarray [ i ] . plugin ;
if ( currentplug - > executestring )
VM_Call ( currentplug - > vm , currentplug - > executestring , 0 ) ;
break ;
}
currentplug = oldplug ;
}
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Cmd_AddCommand ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-01-15 17:46:40 +00:00
{
int i ;
char * name = VM_POINTER ( arg [ 0 ] ) ;
for ( i = 0 ; i < plugincommandarraylen ; i + + )
{
if ( ! plugincommandarray [ i ] . plugin )
break ;
if ( plugincommandarray [ i ] . plugin = = currentplug )
{
if ( ! strcmp ( name , plugincommandarray [ i ] . command ) )
break ;
}
}
if ( i = = plugincommandarraylen )
{
plugincommandarraylen + + ;
plugincommandarray = BZ_Realloc ( plugincommandarray , plugincommandarraylen * sizeof ( plugincommand_t ) ) ;
}
Q_strncpyz ( plugincommandarray [ i ] . command , name , sizeof ( plugincommandarray [ i ] . command ) ) ;
2012-05-09 15:30:53 +00:00
if ( ! Cmd_AddCommand ( plugincommandarray [ i ] . command , Plug_Command_f ) )
2005-01-15 17:46:40 +00:00
return false ;
plugincommandarray [ i ] . plugin = currentplug ; //worked
return true ;
}
2012-10-08 05:10:14 +00:00
static void VARGS Plug_FreeConCommands ( plugin_t * plug )
2005-01-15 17:46:40 +00:00
{
int i ;
for ( i = 0 ; i < plugincommandarraylen ; i + + )
{
if ( plugincommandarray [ i ] . plugin = = plug )
{
plugincommandarray [ i ] . plugin = NULL ;
Cmd_RemoveCommand ( plugincommandarray [ i ] . command ) ;
}
}
}
2005-03-01 15:36:23 +00:00
typedef enum {
STREAM_NONE ,
STREAM_SOCKET ,
2005-12-15 19:15:39 +00:00
STREAM_TLS ,
2005-11-30 01:20:53 +00:00
STREAM_OSFILE ,
2005-03-01 15:36:23 +00:00
STREAM_FILE
} plugstream_e ;
typedef struct {
plugin_t * plugin ;
plugstream_e type ;
int socket ;
2005-11-30 01:20:53 +00:00
struct {
char filename [ MAX_QPATH ] ;
qbyte * buffer ;
int buflen ;
int curlen ;
int curpos ;
} file ;
2005-12-15 19:15:39 +00:00
# ifdef GNUTLS
gnutls_session session ;
# endif
2005-03-01 15:36:23 +00:00
} pluginstream_t ;
pluginstream_t * pluginstreamarray ;
int pluginstreamarraylen ;
2012-10-08 05:10:14 +00:00
static int Plug_NewStreamHandle ( plugstream_e type )
2005-03-01 15:36:23 +00:00
{
int i ;
2012-04-24 07:59:11 +00:00
for ( i = 1 ; i < pluginstreamarraylen ; i + + )
2005-03-01 15:36:23 +00:00
{
if ( ! pluginstreamarray [ i ] . plugin )
break ;
}
2012-04-24 07:59:11 +00:00
if ( i > = pluginstreamarraylen )
2005-03-01 15:36:23 +00:00
{
2012-04-24 07:59:11 +00:00
pluginstreamarraylen = i + 16 ;
2005-03-01 15:36:23 +00:00
pluginstreamarray = BZ_Realloc ( pluginstreamarray , pluginstreamarraylen * sizeof ( pluginstream_t ) ) ;
}
memset ( & pluginstreamarray [ i ] , 0 , sizeof ( pluginstream_t ) ) ;
pluginstreamarray [ i ] . plugin = currentplug ;
pluginstreamarray [ i ] . type = type ;
pluginstreamarray [ i ] . socket = - 1 ;
2005-11-30 01:20:53 +00:00
pluginstreamarray [ i ] . file . buffer = NULL ;
* pluginstreamarray [ i ] . file . filename = ' \0 ' ;
2005-03-01 15:36:23 +00:00
return i ;
}
2012-04-09 19:12:12 +00:00
# ifndef NACL
2005-03-01 15:36:23 +00:00
//EBUILTIN(int, NET_TCPListen, (char *ip, int port, int maxcount));
//returns a new socket with listen enabled.
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Net_TCPListen ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-03-01 15:36:23 +00:00
{
int handle ;
int sock ;
struct sockaddr_qstorage address ;
int _true = 1 ;
2012-04-24 07:59:11 +00:00
int alen ;
2005-12-15 19:15:39 +00:00
2005-03-01 15:36:23 +00:00
char * localip = VM_POINTER ( arg [ 0 ] ) ;
unsigned short localport = VM_LONG ( arg [ 1 ] ) ;
int maxcount = VM_LONG ( arg [ 2 ] ) ;
netadr_t a ;
if ( localip )
{
if ( ! NET_StringToAdr ( localip , & a ) )
return - 1 ;
NetadrToSockadr ( & a , & address ) ;
}
else
{
memset ( & address , 0 , sizeof ( address ) ) ;
( ( struct sockaddr_in * ) & address ) - > sin_family = AF_INET ;
}
if ( ( ( struct sockaddr_in * ) & address ) - > sin_family = = AF_INET & & ! ( ( struct sockaddr_in * ) & address ) - > sin_port )
( ( struct sockaddr_in * ) & address ) - > sin_port = htons ( localport ) ;
# ifdef IPPROTO_IPV6
2005-03-20 02:57:11 +00:00
else if ( ( ( struct sockaddr_in6 * ) & address ) - > sin6_family = = AF_INET6 & & ! ( ( struct sockaddr_in6 * ) & address ) - > sin6_port )
2005-03-01 15:36:23 +00:00
( ( struct sockaddr_in6 * ) & address ) - > sin6_port = htons ( localport ) ;
# endif
2012-04-24 07:59:11 +00:00
switch ( ( ( struct sockaddr * ) & address ) - > sa_family )
{
case AF_INET :
alen = sizeof ( struct sockaddr_in ) ;
break ;
2012-05-10 12:53:36 +00:00
# ifdef IPPROTO_IPV6
2012-04-24 07:59:11 +00:00
case AF_INET6 :
alen = sizeof ( struct sockaddr_in6 ) ;
break ;
2012-05-10 12:53:36 +00:00
# endif
2012-04-24 07:59:11 +00:00
default :
return - 2 ;
}
2005-03-01 15:36:23 +00:00
if ( ( sock = socket ( ( ( struct sockaddr * ) & address ) - > sa_family , SOCK_STREAM , 0 ) ) = = - 1 )
{
Con_Printf ( " Failed to create socket \n " ) ;
return - 2 ;
}
2011-05-19 13:34:07 +00:00
if ( ioctlsocket ( sock , FIONBIO , ( u_long * ) & _true ) = = - 1 )
2005-03-01 15:36:23 +00:00
{
closesocket ( sock ) ;
return - 2 ;
}
2012-04-24 07:59:11 +00:00
setsockopt ( sock , SOL_SOCKET , SO_REUSEADDR , ( const char * ) & _true , sizeof ( _true ) ) ;
2005-03-01 15:36:23 +00:00
2012-04-24 07:59:11 +00:00
if ( bind ( sock , ( void * ) & address , alen ) = = - 1 )
2005-03-01 15:36:23 +00:00
{
closesocket ( sock ) ;
return - 2 ;
}
2012-04-24 07:59:11 +00:00
if ( listen ( sock , maxcount ) = = - 1 )
2005-03-01 15:36:23 +00:00
{
closesocket ( sock ) ;
return - 2 ;
}
handle = Plug_NewStreamHandle ( STREAM_SOCKET ) ;
pluginstreamarray [ handle ] . socket = sock ;
return handle ;
}
2012-10-08 05:10:14 +00:00
static qintptr_t VARGS Plug_Net_Accept ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-03-01 15:36:23 +00:00
{
int handle = VM_LONG ( arg [ 0 ] ) ;
struct sockaddr_in address ;
int addrlen ;
int sock ;
int _true = 1 ;
2008-06-08 14:37:57 +00:00
char adr [ MAX_ADR_SIZE ] ;
2005-03-01 15:36:23 +00:00
if ( handle < 0 | | handle > = pluginstreamarraylen | | pluginstreamarray [ handle ] . plugin ! = currentplug | | pluginstreamarray [ handle ] . type ! = STREAM_SOCKET )
return - 2 ;
sock = pluginstreamarray [ handle ] . socket ;
2012-04-24 07:59:11 +00:00
if ( sock < 0 )
return - 1 ;
2005-03-01 15:36:23 +00:00
addrlen = sizeof ( address ) ;
sock = accept ( sock , ( struct sockaddr * ) & address , & addrlen ) ;
if ( sock < 0 )
return - 1 ;
2011-05-19 13:34:07 +00:00
if ( ioctlsocket ( sock , FIONBIO , ( u_long * ) & _true ) = = - 1 ) //now make it non blocking.
2005-03-01 15:36:23 +00:00
{
closesocket ( sock ) ;
return - 1 ;
}
if ( arg [ 2 ] & & ! VM_OOB ( arg [ 1 ] , arg [ 2 ] ) )
{
netadr_t a ;
char * s ;
SockadrToNetadr ( ( struct sockaddr_qstorage * ) & address , & a ) ;
2008-06-08 14:37:57 +00:00
s = NET_AdrToString ( adr , sizeof ( adr ) , a ) ;
2005-03-01 15:36:23 +00:00
Q_strncpyz ( VM_POINTER ( arg [ 1 ] ) , s , addrlen ) ;
}
handle = Plug_NewStreamHandle ( STREAM_SOCKET ) ;
pluginstreamarray [ handle ] . socket = sock ;
return handle ;
}
//EBUILTIN(int, NET_TCPConnect, (char *ip, int port));
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_Net_TCPConnect ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-03-01 15:36:23 +00:00
{
char * localip = VM_POINTER ( arg [ 0 ] ) ;
unsigned short localport = VM_LONG ( arg [ 1 ] ) ;
int handle ;
struct sockaddr_qstorage to , from ;
int sock ;
int _true = 1 ;
netadr_t a ;
2005-06-18 23:54:34 +00:00
if ( ! NET_StringToAdr ( localip , & a ) )
return - 1 ;
2005-03-01 15:36:23 +00:00
NetadrToSockadr ( & a , & to ) ;
if ( ( ( struct sockaddr_in * ) & to ) - > sin_family = = AF_INET & & ! ( ( struct sockaddr_in * ) & to ) - > sin_port )
( ( struct sockaddr_in * ) & to ) - > sin_port = htons ( localport ) ;
# ifdef IPPROTO_IPV6
2005-03-20 03:23:46 +00:00
else if ( ( ( struct sockaddr_in6 * ) & to ) - > sin6_family = = AF_INET6 & & ! ( ( struct sockaddr_in6 * ) & to ) - > sin6_port )
2005-03-01 15:36:23 +00:00
( ( struct sockaddr_in6 * ) & to ) - > sin6_port = htons ( localport ) ;
# endif
if ( ( sock = socket ( PF_INET , SOCK_STREAM , IPPROTO_TCP ) ) = = - 1 )
{
return - 2 ;
}
memset ( & from , 0 , sizeof ( from ) ) ;
( ( struct sockaddr * ) & from ) - > sa_family = ( ( struct sockaddr * ) & to ) - > sa_family ;
if ( bind ( sock , ( struct sockaddr * ) & from , sizeof ( from ) ) = = - 1 )
{
return - 2 ;
}
//not yet blocking. So no frequent attempts please...
//non blocking prevents connect from returning worthwhile sensible value.
if ( connect ( sock , ( struct sockaddr * ) & to , sizeof ( to ) ) = = - 1 )
{
closesocket ( sock ) ;
return - 2 ;
}
2005-12-15 19:15:39 +00:00
2011-05-19 13:34:07 +00:00
if ( ioctlsocket ( sock , FIONBIO , ( u_long * ) & _true ) = = - 1 ) //now make it non blocking.
2005-03-01 15:36:23 +00:00
{
return - 1 ;
}
handle = Plug_NewStreamHandle ( STREAM_SOCKET ) ;
pluginstreamarray [ handle ] . socket = sock ;
return handle ;
}
2005-12-15 19:15:39 +00:00
# ifdef GNUTLS
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_Net_SetTLSClient ( void * offset , unsigned int mask , const qintptr_t * arg )
2005-12-15 19:15:39 +00:00
{
static gnutls_anon_client_credentials anoncred ;
static gnutls_certificate_credentials xcred ;
int ret ;
long _false = false ;
long _true = true ;
/* Need to enable anonymous KX specifically. */
const int kx_prio [ ] = { GNUTLS_KX_ANON_DH , 0 } ;
const int cert_type_priority [ 3 ] = { GNUTLS_CRT_X509 , GNUTLS_CRT_OPENPGP , 0 } ;
pluginstream_t * stream ;
int handle = VM_LONG ( arg [ 0 ] ) ;
2007-02-26 03:00:25 +00:00
qboolean anon = false ;
2005-12-15 19:15:39 +00:00
if ( handle < 0 | | handle > = pluginstreamarraylen | | pluginstreamarray [ handle ] . plugin ! = currentplug )
{
Con_Printf ( " Plug_Net_SetTLSClient: socket does not belong to you (or is invalid) \n " ) ;
return - 2 ;
}
stream = & pluginstreamarray [ handle ] ;
if ( stream - > type ! = STREAM_SOCKET )
{ //not a socket - invalid
Con_Printf ( " Plug_Net_SetTLSClient: Not a socket handle \n " ) ;
return - 2 ;
}
2007-02-23 00:21:33 +00:00
ioctlsocket ( stream - > socket , FIONBIO , & _false ) ;
2005-12-15 19:15:39 +00:00
{
static qboolean needinit = true ;
if ( needinit )
{
gnutls_global_init ( ) ;
gnutls_anon_allocate_client_credentials ( & anoncred ) ;
gnutls_certificate_allocate_credentials ( & xcred ) ;
// gnutls_certificate_set_x509_trust_file (xcred, "ca.pem", GNUTLS_X509_FMT_PEM);
needinit = false ;
}
}
stream - > type = STREAM_TLS ;
// Initialize TLS session
gnutls_init ( & stream - > session , GNUTLS_CLIENT ) ;
// Use default priorities
gnutls_set_default_priority ( stream - > session ) ;
if ( anon )
{
gnutls_kx_set_priority ( stream - > session , kx_prio ) ;
gnutls_credentials_set ( stream - > session , GNUTLS_CRD_ANON , anoncred ) ;
}
else
{
gnutls_certificate_type_set_priority ( stream - > session , cert_type_priority ) ;
gnutls_credentials_set ( stream - > session , GNUTLS_CRD_CERTIFICATE , xcred ) ;
}
// connect to the peer
gnutls_transport_set_ptr ( stream - > session , ( gnutls_transport_ptr ) stream - > socket ) ;
// Perform the TLS handshake
ret = GNUTLS_E_AGAIN ;
while ( ( ret = = GNUTLS_E_AGAIN ) | | ( ret = = GNUTLS_E_INTERRUPTED ) )
{
ret = gnutls_handshake ( stream - > session ) ;
}
if ( ret < 0 )
{
2007-09-23 15:28:06 +00:00
Con_Printf ( CON_ERROR " *** TLS handshake failed (%i) \n " , ret ) ;
2005-12-15 19:15:39 +00:00
gnutls_perror ( ret ) ;
stream - > type = STREAM_SOCKET ; //go back to regular socket
gnutls_bye ( pluginstreamarray [ handle ] . session , GNUTLS_SHUT_RDWR ) ;
return - 2 ;
}
ioctlsocket ( stream - > socket , FIONBIO , & _true ) ;
return 0 ;
}
# endif
2012-04-09 19:12:12 +00:00
# endif
2005-12-15 19:15:39 +00:00
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_FS_Open ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-11-30 01:20:53 +00:00
{
//modes:
//1: read
//2: write
//char *name, int *handle, int mode
//return value is length of the file.
int handle ;
int * ret ;
char * data ;
if ( VM_OOB ( arg [ 1 ] , sizeof ( int ) ) )
return - 2 ;
ret = VM_POINTER ( arg [ 1 ] ) ;
2005-12-15 19:15:39 +00:00
2005-11-30 01:20:53 +00:00
if ( arg [ 2 ] = = 1 )
{
2009-05-24 10:11:17 +00:00
FS_LoadFile ( VM_POINTER ( arg [ 0 ] ) , ( void * * ) & data ) ;
2005-11-30 01:20:53 +00:00
if ( ! data )
return - 1 ;
handle = Plug_NewStreamHandle ( STREAM_FILE ) ;
pluginstreamarray [ handle ] . file . buffer = data ;
pluginstreamarray [ handle ] . file . curpos = 0 ;
pluginstreamarray [ handle ] . file . curlen = com_filesize ;
pluginstreamarray [ handle ] . file . buflen = com_filesize ;
* ret = handle ;
2005-12-15 19:15:39 +00:00
return com_filesize ;
2005-11-30 01:20:53 +00:00
}
else if ( arg [ 2 ] = = 2 )
{
data = BZ_Malloc ( 8192 ) ;
if ( ! data )
return - 1 ;
handle = Plug_NewStreamHandle ( STREAM_FILE ) ;
Q_strncpyz ( pluginstreamarray [ handle ] . file . filename , VM_POINTER ( arg [ 0 ] ) , MAX_QPATH ) ;
pluginstreamarray [ handle ] . file . buffer = data ;
pluginstreamarray [ handle ] . file . curpos = 0 ;
pluginstreamarray [ handle ] . file . curlen = 0 ;
pluginstreamarray [ handle ] . file . buflen = 8192 ;
* ret = handle ;
2005-12-15 19:15:39 +00:00
return com_filesize ;
2005-11-30 01:20:53 +00:00
}
else
return - 2 ;
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_memset ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-12-01 01:21:08 +00:00
{
2007-10-11 15:40:28 +00:00
void * p = VM_POINTER ( arg [ 0 ] ) ;
2005-12-01 01:21:08 +00:00
if ( VM_OOB ( arg [ 0 ] , arg [ 2 ] ) )
return false ;
2007-10-11 15:40:28 +00:00
if ( p )
memset ( p , VM_LONG ( arg [ 1 ] ) , VM_LONG ( arg [ 2 ] ) ) ;
2005-12-01 01:21:08 +00:00
return arg [ 0 ] ;
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_memcpy ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-12-01 01:21:08 +00:00
{
2007-10-11 15:40:28 +00:00
void * p1 = VM_POINTER ( arg [ 0 ] ) ;
void * p2 = VM_POINTER ( arg [ 1 ] ) ;
2005-12-01 01:21:08 +00:00
if ( VM_OOB ( arg [ 0 ] , arg [ 2 ] ) )
return false ;
2007-10-11 15:40:28 +00:00
2005-12-01 01:21:08 +00:00
if ( VM_OOB ( arg [ 1 ] , arg [ 2 ] ) )
return false ;
2007-10-11 15:40:28 +00:00
if ( p1 & & p2 )
memcpy ( p1 , p2 , VM_LONG ( arg [ 2 ] ) ) ;
2005-12-01 01:21:08 +00:00
return arg [ 0 ] ;
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_memmove ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-12-01 01:21:08 +00:00
{
2007-10-11 15:40:28 +00:00
void * p1 = VM_POINTER ( arg [ 0 ] ) ;
void * p2 = VM_POINTER ( arg [ 1 ] ) ;
2005-12-01 01:21:08 +00:00
if ( VM_OOB ( arg [ 0 ] , arg [ 2 ] ) )
return false ;
2007-10-11 15:40:28 +00:00
2005-12-01 01:21:08 +00:00
if ( VM_OOB ( arg [ 1 ] , arg [ 2 ] ) )
return false ;
2007-10-11 15:40:28 +00:00
if ( p1 & & p2 )
memmove ( p1 , p2 , VM_LONG ( arg [ 2 ] ) ) ;
2005-12-01 01:21:08 +00:00
return arg [ 0 ] ;
}
2005-11-30 01:20:53 +00:00
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_sqrt ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-12-06 15:39:57 +00:00
{
int ret ;
VM_FLOAT ( ret ) = sqrt ( VM_FLOAT ( arg [ 0 ] ) ) ;
return ret ;
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_sin ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-12-10 00:21:45 +00:00
{
int ret ;
VM_FLOAT ( ret ) = sin ( VM_FLOAT ( arg [ 0 ] ) ) ;
return ret ;
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_cos ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-12-10 00:21:45 +00:00
{
int ret ;
VM_FLOAT ( ret ) = cos ( VM_FLOAT ( arg [ 0 ] ) ) ;
return ret ;
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_atan2 ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-12-10 00:21:45 +00:00
{
int ret ;
VM_FLOAT ( ret ) = atan2 ( VM_FLOAT ( arg [ 0 ] ) , VM_FLOAT ( arg [ 1 ] ) ) ;
return ret ;
}
2005-12-06 15:39:57 +00:00
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_Net_Recv ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-03-01 15:36:23 +00:00
{
int read ;
int handle = VM_LONG ( arg [ 0 ] ) ;
void * dest = VM_POINTER ( arg [ 1 ] ) ;
int destlen = VM_LONG ( arg [ 2 ] ) ;
if ( VM_OOB ( arg [ 1 ] , arg [ 2 ] ) )
return - 2 ;
if ( handle < 0 | | handle > = pluginstreamarraylen | | pluginstreamarray [ handle ] . plugin ! = currentplug )
return - 2 ;
switch ( pluginstreamarray [ handle ] . type )
{
2012-04-09 19:12:12 +00:00
# ifndef NACL
2005-03-01 15:36:23 +00:00
case STREAM_SOCKET :
read = recv ( pluginstreamarray [ handle ] . socket , dest , destlen , 0 ) ;
if ( read < 0 )
{
if ( qerrno = = EWOULDBLOCK )
return - 1 ;
else
return - 2 ;
}
else if ( read = = 0 )
return - 2 ; //closed by remote connection.
return read ;
2012-04-09 19:12:12 +00:00
# endif
2005-12-15 19:15:39 +00:00
# ifdef GNUTLS
case STREAM_TLS :
read = gnutls_record_recv ( pluginstreamarray [ handle ] . session , dest , destlen ) ;
if ( read < 0 )
{
if ( read = = GNUTLS_E_AGAIN | | read = = - 9 )
return - 1 ;
else
{
Con_Printf ( " TLS Read Error %i (bufsize %i) \n " , read , destlen ) ;
return - 2 ;
}
}
else if ( read = = 0 )
return - 2 ; //closed by remote connection.
return read ;
# endif
2005-11-30 01:20:53 +00:00
case STREAM_FILE :
if ( pluginstreamarray [ handle ] . file . curlen - pluginstreamarray [ handle ] . file . curpos < destlen )
{
destlen = pluginstreamarray [ handle ] . file . curlen - pluginstreamarray [ handle ] . file . curpos ;
if ( destlen < 0 )
return - 2 ;
}
memcpy ( dest , pluginstreamarray [ handle ] . file . buffer + pluginstreamarray [ handle ] . file . curpos , destlen ) ;
pluginstreamarray [ handle ] . file . curpos + = destlen ;
return destlen ;
2005-03-01 15:36:23 +00:00
default :
return - 2 ;
}
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_Net_Send ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-03-01 15:36:23 +00:00
{
int written ;
int handle = VM_LONG ( arg [ 0 ] ) ;
void * src = VM_POINTER ( arg [ 1 ] ) ;
int srclen = VM_LONG ( arg [ 2 ] ) ;
if ( handle < 0 | | handle > = pluginstreamarraylen | | pluginstreamarray [ handle ] . plugin ! = currentplug )
return - 2 ;
switch ( pluginstreamarray [ handle ] . type )
{
2012-04-09 19:12:12 +00:00
# ifndef NACL
2005-03-01 15:36:23 +00:00
case STREAM_SOCKET :
written = send ( pluginstreamarray [ handle ] . socket , src , srclen , 0 ) ;
if ( written < 0 )
{
if ( qerrno = = EWOULDBLOCK )
return - 1 ;
else
return - 2 ;
}
else if ( written = = 0 )
return - 2 ; //closed by remote connection.
return written ;
2012-04-09 19:12:12 +00:00
# endif
2005-12-15 19:15:39 +00:00
# ifdef GNUTLS
case STREAM_TLS :
written = gnutls_record_send ( pluginstreamarray [ handle ] . session , src , srclen ) ;
if ( written < 0 )
{
if ( written = = GNUTLS_E_AGAIN | | written = = GNUTLS_E_INTERRUPTED )
return - 1 ;
else
{
Con_Printf ( " TLS Send Error %i (%i bytes) \n " , written , srclen ) ;
return - 2 ;
}
}
else if ( written = = 0 )
return - 2 ; //closed by remote connection.
return written ;
# endif
2005-11-30 01:20:53 +00:00
case STREAM_FILE :
if ( pluginstreamarray [ handle ] . file . buflen < pluginstreamarray [ handle ] . file . curpos + srclen )
{
pluginstreamarray [ handle ] . file . buflen = pluginstreamarray [ handle ] . file . curpos + srclen + 8192 ;
2005-12-15 19:15:39 +00:00
pluginstreamarray [ handle ] . file . buffer =
2005-11-30 01:20:53 +00:00
BZ_Realloc ( pluginstreamarray [ handle ] . file . buffer , pluginstreamarray [ handle ] . file . buflen ) ;
}
memcpy ( pluginstreamarray [ handle ] . file . buffer + pluginstreamarray [ handle ] . file . curpos , src , srclen ) ;
pluginstreamarray [ handle ] . file . curpos + = srclen ;
if ( pluginstreamarray [ handle ] . file . curpos > pluginstreamarray [ handle ] . file . curlen )
pluginstreamarray [ handle ] . file . curlen = pluginstreamarray [ handle ] . file . curpos ;
return - 2 ;
2005-03-01 15:36:23 +00:00
default :
return - 2 ;
}
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_Net_SendTo ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-08-03 23:14:59 +00:00
{
int written ;
int handle = VM_LONG ( arg [ 0 ] ) ;
void * src = VM_POINTER ( arg [ 1 ] ) ;
int srclen = VM_LONG ( arg [ 2 ] ) ;
netadr_t * address = VM_POINTER ( arg [ 3 ] ) ;
struct sockaddr_qstorage sockaddr ;
if ( handle = = - 1 )
{
NET_SendPacket ( NS_CLIENT , srclen , src , * address ) ;
return srclen ;
}
NetadrToSockadr ( address , & sockaddr ) ;
if ( handle < 0 | | handle > = pluginstreamarraylen | | pluginstreamarray [ handle ] . plugin ! = currentplug )
return - 2 ;
switch ( pluginstreamarray [ handle ] . type )
{
2012-04-09 19:12:12 +00:00
# ifndef NACL
2005-08-03 23:14:59 +00:00
case STREAM_SOCKET :
written = sendto ( pluginstreamarray [ handle ] . socket , src , srclen , 0 , ( struct sockaddr * ) & sockaddr , sizeof ( sockaddr ) ) ;
if ( written < 0 )
{
if ( qerrno = = EWOULDBLOCK )
return - 1 ;
else
return - 2 ;
}
else if ( written = = 0 )
return - 2 ; //closed by remote connection.
return written ;
2012-04-09 19:12:12 +00:00
# endif
2005-08-03 23:14:59 +00:00
default :
return - 2 ;
}
}
2005-03-01 15:36:23 +00:00
2005-12-15 19:15:39 +00:00
void Plug_Net_Close_Internal ( int handle )
{
2005-11-30 01:20:53 +00:00
switch ( pluginstreamarray [ handle ] . type )
{
2007-08-07 19:16:32 +00:00
case STREAM_FILE :
if ( * pluginstreamarray [ handle ] . file . filename )
2009-05-24 10:11:17 +00:00
{
2007-08-07 19:16:32 +00:00
COM_WriteFile ( pluginstreamarray [ handle ] . file . filename , pluginstreamarray [ handle ] . file . buffer , pluginstreamarray [ handle ] . file . curlen ) ;
2009-05-24 10:11:17 +00:00
BZ_Free ( pluginstreamarray [ handle ] . file . buffer ) ;
}
else
FS_FreeFile ( pluginstreamarray [ handle ] . file . buffer ) ;
2007-08-07 19:16:32 +00:00
break ;
case STREAM_NONE :
break ;
case STREAM_OSFILE :
break ;
case STREAM_SOCKET :
2012-04-09 19:12:12 +00:00
# ifndef NACL
2007-08-07 19:16:32 +00:00
closesocket ( pluginstreamarray [ handle ] . socket ) ;
2012-04-09 19:12:12 +00:00
# endif
2007-08-07 19:16:32 +00:00
break ;
2005-12-15 19:15:39 +00:00
case STREAM_TLS :
2007-08-07 19:16:32 +00:00
# ifdef GNUTLS
2005-12-15 19:15:39 +00:00
gnutls_bye ( pluginstreamarray [ handle ] . session , GNUTLS_SHUT_RDWR ) ;
pluginstreamarray [ handle ] . type = STREAM_SOCKET ;
Plug_Net_Close_Internal ( handle ) ;
return ;
# endif
2005-11-30 01:20:53 +00:00
break ;
}
2005-03-01 15:36:23 +00:00
pluginstreamarray [ handle ] . plugin = NULL ;
2005-12-15 19:15:39 +00:00
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_Net_Close ( void * offset , quintptr_t mask , const qintptr_t * arg )
2005-12-15 19:15:39 +00:00
{
int handle = VM_LONG ( arg [ 0 ] ) ;
if ( handle < 0 | | handle > = pluginstreamarraylen | | pluginstreamarray [ handle ] . plugin ! = currentplug )
return - 2 ;
2005-03-01 15:36:23 +00:00
2005-12-15 19:15:39 +00:00
Plug_Net_Close_Internal ( handle ) ;
2005-03-01 15:36:23 +00:00
return 0 ;
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_ReadInputBuffer ( void * offset , quintptr_t mask , const qintptr_t * arg )
2009-04-06 00:34:32 +00:00
{
void * buffer = VM_POINTER ( arg [ 0 ] ) ;
int bufferlen = VM_LONG ( arg [ 1 ] ) ;
if ( bufferlen > currentplug - > inputbytes )
bufferlen = currentplug - > inputbytes ;
memcpy ( buffer , currentplug - > inputptr , currentplug - > inputbytes ) ;
return bufferlen ;
}
2009-07-18 20:21:02 +00:00
qintptr_t VARGS Plug_UpdateInputBuffer ( void * offset , quintptr_t mask , const qintptr_t * arg )
2009-04-06 00:34:32 +00:00
{
void * buffer = VM_POINTER ( arg [ 0 ] ) ;
int bufferlen = VM_LONG ( arg [ 1 ] ) ;
if ( bufferlen > currentplug - > inputbytes )
bufferlen = currentplug - > inputbytes ;
memcpy ( currentplug - > inputptr , buffer , currentplug - > inputbytes ) ;
return bufferlen ;
}
2005-01-15 17:46:40 +00:00
void Plug_CloseAll_f ( void ) ;
2005-03-01 15:36:23 +00:00
void Plug_List_f ( void ) ;
2005-01-15 20:50:45 +00:00
void Plug_Close_f ( void ) ;
2005-01-15 17:46:40 +00:00
void Plug_Load_f ( void )
{
2005-06-18 23:54:34 +00:00
char * plugin ;
plugin = Cmd_Argv ( 1 ) ;
if ( ! * plugin )
{
Con_Printf ( " Loads a plugin \n " ) ;
Con_Printf ( " plug_load [pluginpath] \n " ) ;
2005-08-03 23:14:59 +00:00
Con_Printf ( " example pluginpath: plugins/blah \n " ) ;
Con_Printf ( " will load blahx86.dll or blah.so \n " ) ;
2005-06-18 23:54:34 +00:00
return ;
}
if ( ! Plug_Load ( plugin ) )
{
if ( ! Plug_Load ( va ( " plugins/%s " , plugin ) ) )
Con_Printf ( " Couldn't load plugin %s \n " , Cmd_Argv ( 1 ) ) ;
}
2005-01-15 17:46:40 +00:00
}
2005-08-26 22:56:51 +00:00
/*
2009-07-18 20:21:02 +00:00
static qintptr_t Test_SysCalls_Ex ( void * offset , quintptr_t mask , int fn , qintptr_t * arg )
2005-08-26 22:56:51 +00:00
{
switch ( fn )
{
case 1 :
Con_Printf ( " %s " , VM_POINTER ( arg [ 0 ] ) ) ;
break ;
default :
Con_Printf ( " Can't handle %i \n " , fn ) ;
}
return 0 ;
}
static int EXPORT_FN Test_SysCalls ( int arg , . . . )
{
return 0 ;
}
void VM_Test_f ( void )
{
vm_t * vm ;
vm = VM_Create ( NULL , " vm/test " , Test_SysCalls , Test_SysCalls_Ex ) ;
if ( vm )
{
VM_Call ( vm , 0 , " " ) ;
VM_Destroy ( vm ) ;
}
} */
2005-01-15 17:46:40 +00:00
2004-09-24 02:45:00 +00:00
void Plug_Init ( void )
{
2005-08-26 22:56:51 +00:00
// Cmd_AddCommand("testvm", VM_Test_f);
2005-01-15 17:46:40 +00:00
Cvar_Register ( & plug_sbar , " plugins " ) ;
Cvar_Register ( & plug_loaddefault , " plugins " ) ;
Cmd_AddCommand ( " plug_closeall " , Plug_CloseAll_f ) ;
2005-01-15 20:50:45 +00:00
Cmd_AddCommand ( " plug_close " , Plug_Close_f ) ;
2005-01-15 17:46:40 +00:00
Cmd_AddCommand ( " plug_load " , Plug_Load_f ) ;
2005-03-01 15:36:23 +00:00
Cmd_AddCommand ( " plug_list " , Plug_List_f ) ;
2005-01-15 17:46:40 +00:00
2011-01-29 21:01:40 +00:00
Plug_RegisterBuiltin ( " Plug_GetEngineFunction " , Plug_GetBuiltin , 0 ) ; //plugin wishes to find a builtin number.
2004-10-10 06:32:29 +00:00
Plug_RegisterBuiltin ( " Plug_ExportToEngine " , Plug_ExportToEngine , 0 ) ; //plugin has a call back that we might be interested in.
2005-11-30 01:20:53 +00:00
Plug_RegisterBuiltin ( " Plug_ExportNative " , Plug_ExportNative , PLUG_BIF_DLLONLY ) ;
------------------------------------------------------------------------
r4169 | acceptthis | 2013-01-17 08:55:12 +0000 (Thu, 17 Jan 2013) | 31 lines
removed MAX_VISEDICTS limit.
PEXT2_REPLACEMENTDELTAS tweaked, now has 4 million entity limit. still not enabled by default.
TE_BEAM now maps to a separate TEQW_BEAM to avoid conflicts with QW.
added android multitouch emulation for windows/rawinput (in_simulatemultitouch).
split topcolor/bottomcolor from scoreboard, for dp's colormap|1024 feature.
now using utf-8 for windows consoles.
qcc warnings/errors now give clickable console links for quick+easy editing.
disabled menutint when the currently active item changes contrast or gamma (for OneManClan).
Added support for drawfont/drawfontscale.
tweaked the qcvm a little to reduce the number of pointers.
.doll file loading. still experimental and will likely crash. requires csqc active, even if its a dummy progs. this will be fixed in time. Still other things that need cleaning up.
windows: gl_font "?" shows the standard windows font-selection dialog, and can be used to select windows fonts. not all work. and you probably don't want to use windings.
fixed splitscreen support when playing mvds. added mini-scoreboards to splitscreen.
editor/debugger now shows asm if there's no linenumber info. also, pressing f1 for help shows the shortcuts.
Added support for .framegroups files for psk(psa) and iqm formats.
True support for ezquake's colour codes. Mutually exclusive with background colours.
path command output slightly more readable.
added support for digest_hex (MD4, SHA1, CRC16).
skingroups now colourmap correctly.
Fix terrain colour hints, and litdata from the wrong bsp.
fix ftp dual-homed issue. support epsv command, and enable ipv6 (eprt still not supported).
remove d3d11 compilation from the makefile. the required headers are not provided by mingw, and are not available to the build bot, so don't bother.
fix v *= v.x and similar opcodes.
fteqcc: fixed support for áéÃóú type chars in names. utf-8 files now properly supported (even with the utf-8 bom/identifier). utf-16 also supported.
fteqcc: fixed '#if 1 == 3 && 4' parsing.
fteqcc: -Werror acts on the warning, rather than as a separate error. Line numbers are thus more readable.
fteqcc: copyright message now includes compile date instead.
fteqccgui: the treeview control is now coloured depending on whether there were warnings/errors in the last compile.
fteqccgui: the output window is now focused and scrolls down as compilation progresses.
pr_dumpplatform command dumps out some pragmas to convert more serious warnings to errors. This is to avoid the infamous 'fteqcc sucks cos my code sucks' issue.
rewrote prespawn/modelist/soundlist code. server tracks progress now.
------------------------------------------------------------------------
git-svn-id: https://svn.code.sf.net/p/fteqw/code/trunk@4167 fc73d0e0-1445-4013-8a0c-d673dee63da5
2013-03-12 22:29:40 +00:00
Plug_RegisterBuiltin ( " Plug_GetPluginName " , Plug_GetPluginName , 0 ) ;
2004-10-10 06:32:29 +00:00
Plug_RegisterBuiltin ( " Con_Print " , Plug_Con_Print , 0 ) ; //printf is not possible - qvm floats are never doubles, vararg floats in a cdecl call are always converted to doubles.
Plug_RegisterBuiltin ( " Sys_Error " , Plug_Sys_Error , 0 ) ;
2005-03-01 15:36:23 +00:00
Plug_RegisterBuiltin ( " Sys_Milliseconds " , Plug_Sys_Milliseconds , 0 ) ;
2004-10-10 06:32:29 +00:00
Plug_RegisterBuiltin ( " Com_Error " , Plug_Sys_Error , 0 ) ; //make zquake programmers happy.
2005-01-15 17:46:40 +00:00
Plug_RegisterBuiltin ( " Cmd_AddCommand " , Plug_Cmd_AddCommand , 0 ) ;
2004-10-10 06:32:29 +00:00
Plug_RegisterBuiltin ( " Cmd_Args " , Plug_Cmd_Args , 0 ) ;
Plug_RegisterBuiltin ( " Cmd_Argc " , Plug_Cmd_Argc , 0 ) ;
Plug_RegisterBuiltin ( " Cmd_Argv " , Plug_Cmd_Argv , 0 ) ;
Plug_RegisterBuiltin ( " Cmd_AddText " , Plug_Cmd_AddText , 0 ) ;
2004-09-24 02:45:00 +00:00
2004-10-10 06:32:29 +00:00
Plug_RegisterBuiltin ( " Cvar_Register " , Plug_Cvar_Register , 0 ) ;
2005-08-01 12:35:16 +00:00
Plug_RegisterBuiltin ( " Cvar_Update " , Plug_Cvar_Update , 0 ) ;
2004-10-10 06:32:29 +00:00
Plug_RegisterBuiltin ( " Cvar_SetString " , Plug_Cvar_SetString , 0 ) ;
Plug_RegisterBuiltin ( " Cvar_SetFloat " , Plug_Cvar_SetFloat , 0 ) ;
Plug_RegisterBuiltin ( " Cvar_GetString " , Plug_Cvar_GetString , 0 ) ;
Plug_RegisterBuiltin ( " Cvar_GetFloat " , Plug_Cvar_GetFloat , 0 ) ;
2012-04-09 19:12:12 +00:00
# ifndef NACL
2005-03-01 15:36:23 +00:00
Plug_RegisterBuiltin ( " Net_TCPListen " , Plug_Net_TCPListen , 0 ) ;
Plug_RegisterBuiltin ( " Net_Accept " , Plug_Net_Accept , 0 ) ;
Plug_RegisterBuiltin ( " Net_TCPConnect " , Plug_Net_TCPConnect , 0 ) ;
2005-12-15 19:15:39 +00:00
# ifdef GNUTLS
if ( Init_GNUTLS ( ) )
Plug_RegisterBuiltin ( " Net_SetTLSClient " , Plug_Net_SetTLSClient , 0 ) ;
# endif
2005-03-01 15:36:23 +00:00
Plug_RegisterBuiltin ( " Net_Recv " , Plug_Net_Recv , 0 ) ;
Plug_RegisterBuiltin ( " Net_Send " , Plug_Net_Send , 0 ) ;
2005-08-03 23:14:59 +00:00
Plug_RegisterBuiltin ( " Net_SendTo " , Plug_Net_SendTo , 0 ) ;
2005-03-01 15:36:23 +00:00
Plug_RegisterBuiltin ( " Net_Close " , Plug_Net_Close , 0 ) ;
2012-04-09 19:12:12 +00:00
# endif
2005-03-01 15:36:23 +00:00
2005-11-30 01:20:53 +00:00
Plug_RegisterBuiltin ( " FS_Open " , Plug_FS_Open , 0 ) ;
Plug_RegisterBuiltin ( " FS_Read " , Plug_Net_Recv , 0 ) ;
Plug_RegisterBuiltin ( " FS_Write " , Plug_Net_Send , 0 ) ;
Plug_RegisterBuiltin ( " FS_Close " , Plug_Net_Close , 0 ) ;
2005-12-01 01:21:08 +00:00
Plug_RegisterBuiltin ( " memset " , Plug_memset , 0 ) ;
Plug_RegisterBuiltin ( " memcpy " , Plug_memcpy , 0 ) ;
Plug_RegisterBuiltin ( " memmove " , Plug_memmove , 0 ) ;
2005-12-06 15:39:57 +00:00
Plug_RegisterBuiltin ( " sqrt " , Plug_sqrt , 0 ) ;
2005-12-10 00:21:45 +00:00
Plug_RegisterBuiltin ( " sin " , Plug_sin , 0 ) ;
Plug_RegisterBuiltin ( " cos " , Plug_cos , 0 ) ;
Plug_RegisterBuiltin ( " atan2 " , Plug_atan2 , 0 ) ;
2005-12-01 01:21:08 +00:00
2009-04-06 00:34:32 +00:00
Plug_RegisterBuiltin ( " ReadInputBuffer " , Plug_ReadInputBuffer , 0 ) ;
Plug_RegisterBuiltin ( " UpdateInputBuffer " , Plug_UpdateInputBuffer , 0 ) ;
2005-12-01 01:21:08 +00:00
2005-12-15 19:15:39 +00:00
Plug_Client_Init ( ) ;
2005-03-01 15:36:23 +00:00
2005-08-03 23:14:59 +00:00
if ( plug_loaddefault . value )
{
2004-09-24 02:45:00 +00:00
# ifdef _WIN32
2005-08-03 23:14:59 +00:00
COM_EnumerateFiles ( " plugins/*x86.dll " , Plug_Emumerated , " x86.dll " ) ;
2004-09-24 02:45:00 +00:00
# elif defined(__linux__)
2005-08-03 23:14:59 +00:00
COM_EnumerateFiles ( " plugins/*x86.so " , Plug_Emumerated , " x86.so " ) ;
2004-09-24 02:45:00 +00:00
# endif
2005-08-03 23:14:59 +00:00
COM_EnumerateFiles ( " plugins/*.qvm " , Plug_Emumerated , " .qvm " ) ;
}
2004-09-24 02:45:00 +00:00
}
void Plug_Tick ( void )
{
2005-01-15 17:46:40 +00:00
plugin_t * oldplug = currentplug ;
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
2004-09-24 02:45:00 +00:00
{
2005-01-15 17:46:40 +00:00
if ( currentplug - > tick )
2004-09-24 02:45:00 +00:00
{
2005-01-15 17:46:40 +00:00
VM_Call ( currentplug - > vm , currentplug - > tick , ( int ) ( realtime * 1000 ) ) ;
2004-09-24 02:45:00 +00:00
}
}
2005-01-15 17:46:40 +00:00
currentplug = oldplug ;
2004-09-24 02:45:00 +00:00
}
2005-12-15 19:15:39 +00:00
# ifndef SERVERONLY
2004-10-10 06:32:29 +00:00
void Plug_ResChanged ( void )
{
plugin_t * oldplug = currentplug ;
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
{
if ( currentplug - > reschange )
VM_Call ( currentplug - > vm , currentplug - > reschange , vid . width , vid . height ) ;
}
currentplug = oldplug ;
}
2005-12-15 19:15:39 +00:00
# endif
2004-10-10 06:32:29 +00:00
2004-09-24 02:45:00 +00:00
qboolean Plugin_ExecuteString ( void )
{
2005-01-15 17:46:40 +00:00
plugin_t * oldplug = currentplug ;
2004-09-24 02:45:00 +00:00
if ( Cmd_Argc ( ) > 0 )
{
2005-01-15 17:46:40 +00:00
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
2004-09-24 02:45:00 +00:00
{
2005-01-15 17:46:40 +00:00
if ( currentplug - > executestring )
2004-09-24 02:45:00 +00:00
{
2005-01-15 17:46:40 +00:00
if ( VM_Call ( currentplug - > vm , currentplug - > executestring , 0 ) )
2005-01-15 20:50:45 +00:00
{
currentplug = oldplug ;
2004-09-24 02:45:00 +00:00
return true ;
2005-01-15 20:50:45 +00:00
}
2004-09-24 02:45:00 +00:00
}
}
}
2005-01-15 17:46:40 +00:00
currentplug = oldplug ;
2004-09-24 02:45:00 +00:00
return false ;
}
2005-12-15 19:15:39 +00:00
# ifndef SERVERONLY
2005-03-10 03:55:18 +00:00
void Plug_SubConsoleCommand ( console_t * con , char * line )
{
char buffer [ 2048 ] ;
plugin_t * oldplug = currentplug ; //shouldn't really be needed, but oh well
currentplug = con - > userdata ;
Q_strncpyz ( buffer , va ( " %s %s " , con - > name , line ) , sizeof ( buffer ) ) ;
Cmd_TokenizeString ( buffer , false , false ) ;
VM_Call ( currentplug - > vm , currentplug - > conexecutecommand , 0 ) ;
currentplug = oldplug ;
}
2009-04-06 00:34:32 +00:00
void Plug_SpellCheckMaskedText ( unsigned int * maskedstring , int maskedchars , int x , int y , int cs , int firstc , int charlimit )
{
plugin_t * oldplug = currentplug ;
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
{
if ( currentplug - > spellcheckmaskedtext )
{
currentplug - > inputptr = maskedstring ;
currentplug - > inputbytes = sizeof ( * maskedstring ) * maskedchars ;
VM_Call ( currentplug - > vm , currentplug - > spellcheckmaskedtext , x , y , cs , firstc , charlimit ) ;
currentplug - > inputptr = NULL ;
currentplug - > inputbytes = 0 ;
}
}
currentplug = oldplug ;
}
2005-12-15 19:15:39 +00:00
# endif
2005-03-10 03:55:18 +00:00
2005-12-15 19:15:39 +00:00
# ifndef SERVERONLY
2004-09-26 00:30:42 +00:00
qboolean Plug_Menu_Event ( int eventtype , int param ) //eventtype = draw/keydown/keyup, param = time/key
{
2005-01-15 20:50:45 +00:00
plugin_t * oc = currentplug ;
qboolean ret ;
2004-10-13 07:24:59 +00:00
2004-09-26 00:30:42 +00:00
if ( ! menuplug )
return false ;
2005-01-15 20:50:45 +00:00
currentplug = menuplug ;
------------------------------------------------------------------------
r4169 | acceptthis | 2013-01-17 08:55:12 +0000 (Thu, 17 Jan 2013) | 31 lines
removed MAX_VISEDICTS limit.
PEXT2_REPLACEMENTDELTAS tweaked, now has 4 million entity limit. still not enabled by default.
TE_BEAM now maps to a separate TEQW_BEAM to avoid conflicts with QW.
added android multitouch emulation for windows/rawinput (in_simulatemultitouch).
split topcolor/bottomcolor from scoreboard, for dp's colormap|1024 feature.
now using utf-8 for windows consoles.
qcc warnings/errors now give clickable console links for quick+easy editing.
disabled menutint when the currently active item changes contrast or gamma (for OneManClan).
Added support for drawfont/drawfontscale.
tweaked the qcvm a little to reduce the number of pointers.
.doll file loading. still experimental and will likely crash. requires csqc active, even if its a dummy progs. this will be fixed in time. Still other things that need cleaning up.
windows: gl_font "?" shows the standard windows font-selection dialog, and can be used to select windows fonts. not all work. and you probably don't want to use windings.
fixed splitscreen support when playing mvds. added mini-scoreboards to splitscreen.
editor/debugger now shows asm if there's no linenumber info. also, pressing f1 for help shows the shortcuts.
Added support for .framegroups files for psk(psa) and iqm formats.
True support for ezquake's colour codes. Mutually exclusive with background colours.
path command output slightly more readable.
added support for digest_hex (MD4, SHA1, CRC16).
skingroups now colourmap correctly.
Fix terrain colour hints, and litdata from the wrong bsp.
fix ftp dual-homed issue. support epsv command, and enable ipv6 (eprt still not supported).
remove d3d11 compilation from the makefile. the required headers are not provided by mingw, and are not available to the build bot, so don't bother.
fix v *= v.x and similar opcodes.
fteqcc: fixed support for áéÃóú type chars in names. utf-8 files now properly supported (even with the utf-8 bom/identifier). utf-16 also supported.
fteqcc: fixed '#if 1 == 3 && 4' parsing.
fteqcc: -Werror acts on the warning, rather than as a separate error. Line numbers are thus more readable.
fteqcc: copyright message now includes compile date instead.
fteqccgui: the treeview control is now coloured depending on whether there were warnings/errors in the last compile.
fteqccgui: the output window is now focused and scrolls down as compilation progresses.
pr_dumpplatform command dumps out some pragmas to convert more serious warnings to errors. This is to avoid the infamous 'fteqcc sucks cos my code sucks' issue.
rewrote prespawn/modelist/soundlist code. server tracks progress now.
------------------------------------------------------------------------
git-svn-id: https://svn.code.sf.net/p/fteqw/code/trunk@4167 fc73d0e0-1445-4013-8a0c-d673dee63da5
2013-03-12 22:29:40 +00:00
ret = VM_Call ( menuplug - > vm , menuplug - > menufunction , eventtype , param , ( int ) mousecursor_x , ( int ) mousecursor_y ) ;
2005-01-15 20:50:45 +00:00
currentplug = oc ;
return ret ;
2004-10-13 07:24:59 +00:00
}
2005-12-15 19:15:39 +00:00
# endif
# ifndef SERVERONLY
2005-08-03 23:14:59 +00:00
int Plug_ConnectionlessClientPacket ( char * buffer , int size )
{
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
{
if ( currentplug - > connectionlessclientpacket )
{
switch ( VM_Call ( currentplug - > vm , currentplug - > connectionlessclientpacket , buffer , size , & net_from ) )
{
case 0 :
continue ; //wasn't handled
case 1 :
currentplug = NULL ; //was handled with no apparent result
return true ;
case 2 :
# ifndef SERVERONLY
cls . protocol = CP_PLUGIN ; //woo, the plugin wants to connect to them!
protocolclientplugin = currentplug ;
# endif
currentplug = NULL ;
return true ;
}
}
}
return false ;
}
2005-12-15 19:15:39 +00:00
# endif
# ifndef SERVERONLY
2004-10-13 07:24:59 +00:00
void Plug_SBar ( void )
{
2006-01-21 00:06:49 +00:00
extern qboolean sb_showscores , sb_showteamscores ;
2004-10-13 07:24:59 +00:00
plugin_t * oc = currentplug ;
2006-08-20 01:35:56 +00:00
int cp , ret ;
2004-10-13 07:24:59 +00:00
vrect_t rect ;
2007-02-26 03:00:25 +00:00
if ( ! Sbar_ShouldDraw ( ) )
return ;
2006-08-20 01:35:56 +00:00
ret = 0 ;
2009-11-04 21:16:50 +00:00
if ( ! plug_sbar . ival | | cl . splitclients > 1 )
2005-01-15 17:46:40 +00:00
currentplug = NULL ;
else
2004-10-13 07:24:59 +00:00
{
2005-01-15 17:46:40 +00:00
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
2004-10-13 07:24:59 +00:00
{
2005-01-15 17:46:40 +00:00
if ( currentplug - > sbarlevel [ 0 ] )
{
for ( cp = 0 ; cp < cl . splitclients ; cp + + )
{ //if you don't use splitscreen, use a full videosize rect.
SCR_VRectForPlayer ( & rect , cp ) ;
2011-03-31 01:14:01 +00:00
R2D_ImageColours ( 1 , 1 , 1 , 1 ) ; // ensure menu colors are reset
2006-08-20 01:35:56 +00:00
ret | = VM_Call ( currentplug - > vm , currentplug - > sbarlevel [ 0 ] , cp , rect . x , rect . y , rect . width , rect . height , sb_showscores + sb_showteamscores * 2 ) ;
2005-01-15 17:46:40 +00:00
}
break ;
2004-10-13 07:24:59 +00:00
}
}
}
2007-02-26 03:00:25 +00:00
if ( ! ( ret & 1 ) )
2004-10-13 07:24:59 +00:00
{
Sbar_Draw ( ) ;
}
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
{
if ( currentplug - > sbarlevel [ 1 ] )
{
for ( cp = 0 ; cp < cl . splitclients ; cp + + )
{ //if you don't use splitscreen, use a full videosize rect.
SCR_VRectForPlayer ( & rect , cp ) ;
2011-03-31 01:14:01 +00:00
R2D_ImageColours ( 1 , 1 , 1 , 1 ) ; // ensure menu colors are reset
2007-02-26 03:00:25 +00:00
ret | = VM_Call ( currentplug - > vm , currentplug - > sbarlevel [ 1 ] , cp , rect . x , rect . y , rect . width , rect . height , sb_showscores + sb_showteamscores * 2 ) ;
2004-10-13 07:24:59 +00:00
}
}
}
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
{
if ( currentplug - > sbarlevel [ 2 ] )
{
for ( cp = 0 ; cp < cl . splitclients ; cp + + )
{ //if you don't use splitscreen, use a full videosize rect.
SCR_VRectForPlayer ( & rect , cp ) ;
2011-03-31 01:14:01 +00:00
R2D_ImageColours ( 1 , 1 , 1 , 1 ) ; // ensure menu colors are reset
2007-02-26 03:00:25 +00:00
ret | = VM_Call ( currentplug - > vm , currentplug - > sbarlevel [ 2 ] , cp , rect . x , rect . y , rect . width , rect . height , sb_showscores + sb_showteamscores * 2 ) ;
2004-10-13 07:24:59 +00:00
}
}
}
2007-02-26 03:00:25 +00:00
if ( ! ( ret & 2 ) )
{
Sbar_DrawScoreboard ( ) ;
}
2004-10-13 07:24:59 +00:00
currentplug = oc ;
2004-09-26 00:30:42 +00:00
}
2005-12-15 19:15:39 +00:00
# endif
2004-09-26 00:30:42 +00:00
2006-01-01 09:01:15 +00:00
qboolean Plug_ServerMessage ( char * buffer , int messagelevel )
2005-11-30 05:01:43 +00:00
{
2006-01-01 09:01:15 +00:00
qboolean ret = true ;
2007-02-23 00:21:33 +00:00
Cmd_TokenizeString ( buffer , false , false ) ;
Cmd_Args_Set ( buffer ) ;
2006-01-01 09:01:15 +00:00
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
{
if ( currentplug - > svmsgfunction )
{
2007-02-23 00:21:33 +00:00
ret & = VM_Call ( currentplug - > vm , currentplug - > svmsgfunction , messagelevel ) ;
2006-01-01 09:01:15 +00:00
}
}
2007-02-23 00:21:33 +00:00
Cmd_Args_Set ( NULL ) ;
2006-01-01 09:01:15 +00:00
return ret ; // true to display message, false to supress
}
qboolean Plug_ChatMessage ( char * buffer , int talkernum , int tpflags )
{
qboolean ret = true ;
2007-02-23 00:21:33 +00:00
Cmd_TokenizeString ( buffer , false , false ) ;
Cmd_Args_Set ( buffer ) ;
2006-01-01 09:01:15 +00:00
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
{
if ( currentplug - > chatmsgfunction )
{
2007-02-23 00:21:33 +00:00
ret & = VM_Call ( currentplug - > vm , currentplug - > chatmsgfunction , talkernum , tpflags ) ;
2006-01-01 09:01:15 +00:00
}
}
2007-02-23 00:21:33 +00:00
Cmd_Args_Set ( NULL ) ;
2006-01-01 09:01:15 +00:00
return ret ; // true to display message, false to supress
}
qboolean Plug_CenterPrintMessage ( char * buffer , int clientnum )
{
qboolean ret = true ;
2007-02-23 00:21:33 +00:00
Cmd_TokenizeString ( buffer , false , false ) ;
Cmd_Args_Set ( buffer ) ;
2005-11-30 05:01:43 +00:00
for ( currentplug = plugs ; currentplug ; currentplug = currentplug - > next )
{
2006-01-01 09:01:15 +00:00
if ( currentplug - > centerprintfunction )
2005-11-30 05:01:43 +00:00
{
2007-02-23 00:21:33 +00:00
ret & = VM_Call ( currentplug - > vm , currentplug - > centerprintfunction , clientnum ) ;
2005-11-30 05:01:43 +00:00
}
}
2007-02-23 00:21:33 +00:00
Cmd_Args_Set ( NULL ) ;
2006-01-01 09:01:15 +00:00
return ret ; // true to display message, false to supress
2005-11-30 05:01:43 +00:00
}
2004-09-24 02:45:00 +00:00
void Plug_Close ( plugin_t * plug )
{
2006-01-29 03:45:00 +00:00
if ( plug - > blockcloses )
{
Con_Printf ( " Plugin %s provides driver features, and cannot safely be unloaded \n " , plug - > name ) ;
return ;
}
2004-09-24 02:45:00 +00:00
if ( plugs = = plug )
plugs = plug - > next ;
else
{
plugin_t * prev ;
for ( prev = plugs ; prev ; prev = prev - > next )
{
if ( prev - > next = = plug )
break ;
}
if ( ! prev )
Sys_Error ( " Plug_Close: not linked \n " ) ;
prev - > next = plug - > next ;
}
2005-01-15 20:50:45 +00:00
Con_Printf ( " Closing plugin %s \n " , plug - > name ) ;
2012-10-08 04:36:10 +00:00
# if defined(PLUGINS) && !defined(NOMEDIA) && !defined(SERVERONLY)
Media_UnregisterDecoder ( plug , NULL ) ;
2012-11-27 03:23:19 +00:00
Media_UnregisterEncoder ( plug , NULL ) ;
2012-10-08 04:36:10 +00:00
# endif
if ( plug - > shutdown )
VM_Call ( plug - > vm , plug - > shutdown ) ;
2004-09-24 02:45:00 +00:00
VM_Destroy ( plug - > vm ) ;
2005-01-15 17:46:40 +00:00
Plug_FreeConCommands ( plug ) ;
2005-12-15 19:15:39 +00:00
Plug_Client_Close ( plug ) ;
2012-07-05 19:42:36 +00:00
Z_Free ( plug ) ;
2005-12-15 19:15:39 +00:00
2005-01-15 17:46:40 +00:00
if ( currentplug = = plug )
currentplug = NULL ;
2005-01-15 20:50:45 +00:00
}
void Plug_Close_f ( void )
{
plugin_t * plug ;
char * name = Cmd_Argv ( 1 ) ;
if ( Cmd_Argc ( ) < 2 )
{
Con_Printf ( " Close which plugin? \n " ) ;
return ;
}
2007-02-23 00:21:33 +00:00
if ( currentplug )
Sys_Error ( " Plug_CloseAll_f called inside a plugin! \n " ) ;
2005-01-15 20:50:45 +00:00
for ( plug = plugs ; plug ; plug = plug - > next )
{
2005-01-24 23:47:32 +00:00
if ( ! strcmp ( plug - > name , name ) )
2005-01-15 20:50:45 +00:00
{
Plug_Close ( plug ) ;
return ;
}
}
2005-06-18 23:54:34 +00:00
name = va ( " plugins/%s " , name ) ;
for ( plug = plugs ; plug ; plug = plug - > next )
{
if ( ! strcmp ( plug - > name , name ) )
{
Plug_Close ( plug ) ;
return ;
}
}
Con_Printf ( " Plugin %s does not appear to be loaded \n " , Cmd_Argv ( 1 ) ) ;
2004-09-24 02:45:00 +00:00
}
2005-01-15 17:46:40 +00:00
void Plug_CloseAll_f ( void )
2004-09-24 02:45:00 +00:00
{
2006-02-02 01:13:52 +00:00
plugin_t * p ;
2005-01-15 17:46:40 +00:00
if ( currentplug )
Sys_Error ( " Plug_CloseAll_f called inside a plugin! \n " ) ;
while ( plugs )
{
2006-02-02 01:13:52 +00:00
p = plugs ;
while ( p - > blockcloses )
{
p = p - > next ;
if ( ! p )
return ;
}
Plug_Close ( p ) ;
2005-01-15 17:46:40 +00:00
}
}
2004-09-24 02:45:00 +00:00
2005-03-01 15:36:23 +00:00
void Plug_List_f ( void )
{
plugin_t * plug ;
for ( plug = plugs ; plug ; plug = plug - > next )
{
2005-11-30 05:01:43 +00:00
Con_Printf ( " %s - \n " , plug - > name ) ;
VM_PrintInfo ( plug - > vm ) ;
2005-03-01 15:36:23 +00:00
}
}
2005-01-15 17:46:40 +00:00
void Plug_Shutdown ( void )
{
while ( plugs )
{
2012-07-05 19:42:36 +00:00
plugs - > blockcloses = 0 ;
2005-01-15 17:46:40 +00:00
Plug_Close ( plugs ) ;
}
2012-07-05 19:42:36 +00:00
numplugbuiltins = 0 ;
BZ_Free ( plugbuiltins ) ;
plugbuiltins = NULL ;
2012-11-27 03:23:19 +00:00
plugincvararraylen = 0 ;
BZ_Free ( plugincvararray ) ;
plugincvararray = NULL ;
2012-07-05 19:42:36 +00:00
plugincommandarraylen = 0 ;
BZ_Free ( plugincommandarray ) ;
plugincommandarray = NULL ;
2004-09-30 22:51:15 +00:00
}
# endif