2017-10-08 21:47:38 +00:00
/*************************************************************************/
/* pluginscript_script.cpp */
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
2018-01-04 23:50:27 +00:00
/* https://godotengine.org */
2017-10-08 21:47:38 +00:00
/*************************************************************************/
2021-01-01 19:13:46 +00:00
/* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md). */
2017-10-08 21:47:38 +00:00
/* */
/* Permission is hereby granted, free of charge, to any person obtaining */
/* a copy of this software and associated documentation files (the */
/* "Software"), to deal in the Software without restriction, including */
/* without limitation the rights to use, copy, modify, merge, publish, */
/* distribute, sublicense, and/or sell copies of the Software, and to */
/* permit persons to whom the Software is furnished to do so, subject to */
/* the following conditions: */
/* */
/* The above copyright notice and this permission notice shall be */
/* included in all copies or substantial portions of the Software. */
/* */
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
/*************************************************************************/
// Godot imports
2021-06-11 12:51:48 +00:00
# include "core/io/file_access.h"
2017-10-08 21:47:38 +00:00
// PluginScript imports
# include "pluginscript_instance.h"
# include "pluginscript_script.h"
2020-02-12 10:51:50 +00:00
# include <stdint.h>
2019-04-05 21:20:20 +00:00
# ifdef DEBUG_ENABLED
2019-08-09 04:49:33 +00:00
# define __ASSERT_SCRIPT_REASON "Cannot retrieve PluginScript class for this script, is your code correct?"
# define ASSERT_SCRIPT_VALID() \
{ \
ERR_FAIL_COND_MSG ( ! can_instance ( ) , __ASSERT_SCRIPT_REASON ) ; \
2017-10-08 21:47:38 +00:00
}
2019-08-09 04:49:33 +00:00
# define ASSERT_SCRIPT_VALID_V(ret) \
{ \
ERR_FAIL_COND_V_MSG ( ! can_instance ( ) , ret , __ASSERT_SCRIPT_REASON ) ; \
2017-10-08 21:47:38 +00:00
}
# else
# define ASSERT_SCRIPT_VALID()
# define ASSERT_SCRIPT_VALID_V(ret)
# endif
void PluginScript : : _bind_methods ( ) {
2019-08-26 16:36:51 +00:00
ClassDB : : bind_vararg_method ( METHOD_FLAGS_DEFAULT , " new " , & PluginScript : : _new , MethodInfo ( " new " ) ) ;
2018-11-09 11:26:07 +00:00
}
2020-02-19 19:27:19 +00:00
PluginScriptInstance * PluginScript : : _create_instance ( const Variant * * p_args , int p_argcount , Object * p_owner , Callable : : CallError & r_error ) {
r_error . error = Callable : : CallError : : CALL_OK ;
2018-11-09 11:26:07 +00:00
// Create instance
PluginScriptInstance * instance = memnew ( PluginScriptInstance ( ) ) ;
if ( instance - > init ( this , p_owner ) ) {
_language - > lock ( ) ;
_instances . insert ( instance - > get_owner ( ) ) ;
_language - > unlock ( ) ;
} else {
2020-02-19 19:27:19 +00:00
r_error . error = Callable : : CallError : : CALL_ERROR_INSTANCE_IS_NULL ;
2018-11-09 11:26:07 +00:00
memdelete ( instance ) ;
2020-04-01 23:20:12 +00:00
ERR_FAIL_V ( nullptr ) ;
2018-11-09 11:26:07 +00:00
}
// Construct
// TODO: Support arguments in the constructor?
// There is currently no way to get the constructor function name of the script.
// instance->call("__init__", p_args, p_argcount, r_error);
if ( p_argcount > 0 ) {
2019-06-11 12:49:34 +00:00
WARN_PRINT ( " PluginScript doesn't support arguments in the constructor " ) ;
2018-11-09 11:26:07 +00:00
}
return instance ;
}
2020-02-19 19:27:19 +00:00
Variant PluginScript : : _new ( const Variant * * p_args , int p_argcount , Callable : : CallError & r_error ) {
r_error . error = Callable : : CallError : : CALL_OK ;
2018-11-09 11:26:07 +00:00
if ( ! _valid ) {
2020-02-19 19:27:19 +00:00
r_error . error = Callable : : CallError : : CALL_ERROR_INVALID_METHOD ;
2018-11-09 11:26:07 +00:00
return Variant ( ) ;
}
REF ref ;
2020-04-01 23:20:12 +00:00
Object * owner = nullptr ;
2018-11-09 11:26:07 +00:00
if ( get_instance_base_type ( ) = = " " ) {
2021-06-04 16:03:15 +00:00
owner = memnew ( RefCounted ) ;
2018-11-09 11:26:07 +00:00
} else {
owner = ClassDB : : instance ( get_instance_base_type ( ) ) ;
}
if ( ! owner ) {
2020-02-19 19:27:19 +00:00
r_error . error = Callable : : CallError : : CALL_ERROR_INSTANCE_IS_NULL ;
2018-11-09 11:26:07 +00:00
return Variant ( ) ;
}
2021-06-04 16:03:15 +00:00
RefCounted * r = Object : : cast_to < RefCounted > ( owner ) ;
2018-11-09 11:26:07 +00:00
if ( r ) {
ref = REF ( r ) ;
}
PluginScriptInstance * instance = _create_instance ( p_args , p_argcount , owner , r_error ) ;
if ( ! instance ) {
if ( ref . is_null ( ) ) {
memdelete ( owner ) ; //no owner, sorry
}
return Variant ( ) ;
}
if ( ref . is_valid ( ) ) {
return ref ;
} else {
return owner ;
}
2017-10-08 21:47:38 +00:00
}
# ifdef TOOLS_ENABLED
void PluginScript : : _placeholder_erased ( PlaceHolderScriptInstance * p_placeholder ) {
placeholders . erase ( p_placeholder ) ;
}
# endif
bool PluginScript : : can_instance ( ) const {
bool can = _valid | | ( ! _tool & & ! ScriptServer : : is_scripting_enabled ( ) ) ;
return can ;
}
2020-04-20 22:06:00 +00:00
bool PluginScript : : inherits_script ( const Ref < Script > & p_script ) const {
# ifndef _MSC_VER
# warning inheritance needs to be implemented in PluginScript
# endif
return false ;
}
2017-10-08 21:47:38 +00:00
Ref < Script > PluginScript : : get_base_script ( ) const {
if ( _ref_base_parent . is_valid ( ) ) {
return Ref < PluginScript > ( _ref_base_parent ) ;
} else {
return Ref < Script > ( ) ;
}
}
StringName PluginScript : : get_instance_base_type ( ) const {
2020-05-14 14:41:43 +00:00
if ( _native_parent ) {
2017-10-08 21:47:38 +00:00
return _native_parent ;
2020-05-14 14:41:43 +00:00
}
if ( _ref_base_parent . is_valid ( ) ) {
2017-10-08 21:47:38 +00:00
return _ref_base_parent - > get_instance_base_type ( ) ;
2020-05-14 14:41:43 +00:00
}
2017-10-08 21:47:38 +00:00
return StringName ( ) ;
}
void PluginScript : : update_exports ( ) {
# ifdef TOOLS_ENABLED
ASSERT_SCRIPT_VALID ( ) ;
2018-06-15 19:16:01 +00:00
if ( placeholders . size ( ) ) {
2017-10-08 21:47:38 +00:00
//update placeholders if any
Map < StringName , Variant > propdefvalues ;
List < PropertyInfo > propinfos ;
2018-06-15 19:16:01 +00:00
get_script_property_list ( & propinfos ) ;
2017-10-08 21:47:38 +00:00
for ( Set < PlaceHolderScriptInstance * > : : Element * E = placeholders . front ( ) ; E ; E = E - > next ( ) ) {
2018-06-15 19:16:01 +00:00
E - > get ( ) - > update ( propinfos , _properties_default_values ) ;
2017-10-08 21:47:38 +00:00
}
}
# endif
}
// TODO: rename p_this "p_owner" ?
ScriptInstance * PluginScript : : instance_create ( Object * p_this ) {
2020-04-01 23:20:12 +00:00
ASSERT_SCRIPT_VALID_V ( nullptr ) ;
2017-10-08 21:47:38 +00:00
// TODO check script validity ?
if ( ! _tool & & ! ScriptServer : : is_scripting_enabled ( ) ) {
# ifdef TOOLS_ENABLED
// Instance a fake script for editing the values
PlaceHolderScriptInstance * si = memnew ( PlaceHolderScriptInstance ( get_language ( ) , Ref < Script > ( this ) , p_this ) ) ;
placeholders . insert ( si ) ;
update_exports ( ) ;
return si ;
# else
2020-04-01 23:20:12 +00:00
return nullptr ;
2017-10-08 21:47:38 +00:00
# endif
}
2018-01-15 22:20:02 +00:00
StringName base_type = get_instance_base_type ( ) ;
if ( base_type ) {
if ( ! ClassDB : : is_parent_class ( p_this - > get_class_name ( ) , base_type ) ) {
String msg = " Script inherits from native type ' " + String ( base_type ) + " ', so it can't be instanced in object of type: ' " + p_this - > get_class ( ) + " ' " ;
2017-10-08 21:47:38 +00:00
// TODO: implement PluginscriptLanguage::debug_break_parse
2020-02-27 02:30:20 +00:00
// if (EngineDebugger::is_active()) {
2017-10-08 21:47:38 +00:00
// _language->debug_break_parse(get_path(), 0, msg);
// }
2020-04-01 23:20:12 +00:00
ERR_FAIL_V_MSG ( nullptr , msg ) ;
2017-10-08 21:47:38 +00:00
}
}
2020-02-19 19:27:19 +00:00
Callable : : CallError unchecked_error ;
2020-04-01 23:20:12 +00:00
return _create_instance ( nullptr , 0 , p_this , unchecked_error ) ;
2017-10-08 21:47:38 +00:00
}
bool PluginScript : : instance_has ( const Object * p_this ) const {
2021-05-29 16:12:26 +00:00
ERR_FAIL_COND_V ( ! _language , false ) ;
2017-10-08 21:47:38 +00:00
_language - > lock ( ) ;
bool hasit = _instances . has ( ( Object * ) p_this ) ;
_language - > unlock ( ) ;
return hasit ;
}
bool PluginScript : : has_source_code ( ) const {
bool has = _source ! = " " ;
return has ;
}
String PluginScript : : get_source_code ( ) const {
return _source ;
}
void PluginScript : : set_source_code ( const String & p_code ) {
2020-05-14 14:41:43 +00:00
if ( _source = = p_code ) {
2017-10-08 21:47:38 +00:00
return ;
2020-05-14 14:41:43 +00:00
}
2017-10-08 21:47:38 +00:00
_source = p_code ;
}
Error PluginScript : : reload ( bool p_keep_state ) {
2019-06-29 12:51:33 +00:00
ERR_FAIL_COND_V ( ! _language , ERR_UNCONFIGURED ) ;
2017-10-08 21:47:38 +00:00
_language - > lock ( ) ;
ERR_FAIL_COND_V ( ! p_keep_state & & _instances . size ( ) , ERR_ALREADY_IN_USE ) ;
_language - > unlock ( ) ;
_valid = false ;
String basedir = _path ;
2020-05-14 14:41:43 +00:00
if ( basedir = = " " ) {
2017-10-08 21:47:38 +00:00
basedir = get_path ( ) ;
2020-05-14 14:41:43 +00:00
}
2017-10-08 21:47:38 +00:00
2020-05-14 14:41:43 +00:00
if ( basedir ! = " " ) {
2017-10-08 21:47:38 +00:00
basedir = basedir . get_base_dir ( ) ;
2020-05-14 14:41:43 +00:00
}
2017-10-08 21:47:38 +00:00
if ( _data ) {
_desc - > finish ( _data ) ;
}
Error err ;
godot_pluginscript_script_manifest manifest = _desc - > init (
_language - > _data ,
( godot_string * ) & _path ,
( godot_string * ) & _source ,
( godot_error * ) & err ) ;
2019-12-01 11:10:18 +00:00
// Manifest's attributes must be explicitly freed
# define FREE_SCRIPT_MANIFEST(manifest) \
{ \
godot_string_name_destroy ( & manifest . name ) ; \
godot_string_name_destroy ( & manifest . base ) ; \
godot_dictionary_destroy ( & manifest . member_lines ) ; \
godot_array_destroy ( & manifest . methods ) ; \
godot_array_destroy ( & manifest . signals ) ; \
godot_array_destroy ( & manifest . properties ) ; \
}
2017-10-08 21:47:38 +00:00
if ( err ) {
2019-12-01 11:10:18 +00:00
FREE_SCRIPT_MANIFEST ( manifest ) ;
2017-10-08 21:47:38 +00:00
// TODO: GDscript uses `ScriptDebugger` here to jump into the parsing error
return err ;
}
2018-01-15 22:20:02 +00:00
// Script's parent is passed as base_name which can make reference to a
2018-01-18 20:37:17 +00:00
// ClassDB name (i.e. `Node2D`) or a resource path (i.e. `res://foo/bar.gd`)
2018-01-15 22:20:02 +00:00
StringName * base_name = ( StringName * ) & manifest . base ;
if ( * base_name ) {
if ( ClassDB : : class_exists ( * base_name ) ) {
_native_parent = * base_name ;
} else {
Ref < Script > res = ResourceLoader : : load ( * base_name ) ;
if ( res . is_valid ( ) ) {
_ref_base_parent = res ;
} else {
String name = * ( StringName * ) & manifest . name ;
2019-12-01 11:10:18 +00:00
FREE_SCRIPT_MANIFEST ( manifest ) ;
2019-08-09 04:49:33 +00:00
ERR_FAIL_V_MSG ( ERR_PARSE_ERROR , _path + " : Script ' " + name + " ' has an invalid parent ' " + * base_name + " '. " ) ;
2018-01-15 22:20:02 +00:00
}
}
}
2017-10-08 21:47:38 +00:00
_valid = true ;
// Use the manifest to configure this script object
_data = manifest . data ;
_name = * ( StringName * ) & manifest . name ;
_tool = manifest . is_tool ;
2020-12-07 14:08:59 +00:00
_icon_path = * ( String * ) & manifest . icon_path ;
2017-10-08 21:47:38 +00:00
Dictionary * members = ( Dictionary * ) & manifest . member_lines ;
2020-04-01 23:20:12 +00:00
for ( const Variant * key = members - > next ( ) ; key ! = nullptr ; key = members - > next ( key ) ) {
2019-06-29 10:06:29 +00:00
_member_lines [ * key ] = ( * members ) [ * key ] ;
2017-10-08 21:47:38 +00:00
}
Array * methods = ( Array * ) & manifest . methods ;
2020-02-12 10:51:50 +00:00
_rpc_methods . clear ( ) ;
if ( _ref_base_parent . is_valid ( ) ) {
2021-05-26 12:07:57 +00:00
/// XXX TODO Should this be _rpc_methods.append_array(...)
2020-02-12 10:51:50 +00:00
_rpc_methods = _ref_base_parent - > get_rpc_methods ( ) ;
}
2017-10-08 21:47:38 +00:00
for ( int i = 0 ; i < methods - > size ( ) ; + + i ) {
Dictionary v = ( * methods ) [ i ] ;
MethodInfo mi = MethodInfo : : from_dict ( v ) ;
_methods_info [ mi . name ] = mi ;
// rpc_mode is passed as an optional field and is not part of MethodInfo
Variant var = v [ " rpc_mode " ] ;
2020-02-12 10:51:50 +00:00
if ( var ! = Variant ( ) ) {
2021-05-26 12:07:57 +00:00
MultiplayerAPI : : RPCConfig nd ;
2020-02-12 10:51:50 +00:00
nd . name = mi . name ;
2021-05-26 12:07:57 +00:00
nd . rpc_mode = MultiplayerAPI : : RPCMode ( int ( var ) ) ;
// TODO Transfer Channel
2020-02-12 10:51:50 +00:00
if ( _rpc_methods . find ( nd ) = = - 1 ) {
_rpc_methods . push_back ( nd ) ;
}
2017-10-08 21:47:38 +00:00
}
}
2020-02-12 10:51:50 +00:00
// Sort so we are 100% that they are always the same.
2021-05-26 12:07:57 +00:00
_rpc_methods . sort_custom < MultiplayerAPI : : SortRPCConfig > ( ) ;
2020-02-12 10:51:50 +00:00
2017-10-08 21:47:38 +00:00
Array * signals = ( Array * ) & manifest . signals ;
for ( int i = 0 ; i < signals - > size ( ) ; + + i ) {
Variant v = ( * signals ) [ i ] ;
MethodInfo mi = MethodInfo : : from_dict ( v ) ;
_signals_info [ mi . name ] = mi ;
}
Array * properties = ( Array * ) & manifest . properties ;
for ( int i = 0 ; i < properties - > size ( ) ; + + i ) {
Dictionary v = ( * properties ) [ i ] ;
PropertyInfo pi = PropertyInfo : : from_dict ( v ) ;
_properties_info [ pi . name ] = pi ;
_properties_default_values [ pi . name ] = v [ " default_value " ] ;
}
# ifdef TOOLS_ENABLED
/*for (Set<PlaceHolderScriptInstance*>::Element *E=placeholders.front();E;E=E->next()) {
_update_placeholder ( E - > get ( ) ) ;
} */
# endif
2019-12-01 11:10:18 +00:00
FREE_SCRIPT_MANIFEST ( manifest ) ;
2017-10-08 21:47:38 +00:00
return OK ;
2019-12-01 11:10:18 +00:00
# undef FREE_SCRIPT_MANIFEST
2017-10-08 21:47:38 +00:00
}
void PluginScript : : get_script_method_list ( List < MethodInfo > * r_methods ) const {
ASSERT_SCRIPT_VALID ( ) ;
2020-04-01 23:20:12 +00:00
for ( Map < StringName , MethodInfo > : : Element * e = _methods_info . front ( ) ; e ! = nullptr ; e = e - > next ( ) ) {
2017-10-08 21:47:38 +00:00
r_methods - > push_back ( e - > get ( ) ) ;
}
}
void PluginScript : : get_script_property_list ( List < PropertyInfo > * r_properties ) const {
ASSERT_SCRIPT_VALID ( ) ;
2020-04-01 23:20:12 +00:00
for ( Map < StringName , PropertyInfo > : : Element * e = _properties_info . front ( ) ; e ! = nullptr ; e = e - > next ( ) ) {
2017-10-08 21:47:38 +00:00
r_properties - > push_back ( e - > get ( ) ) ;
}
}
bool PluginScript : : has_method ( const StringName & p_method ) const {
ASSERT_SCRIPT_VALID_V ( false ) ;
return _methods_info . has ( p_method ) ;
}
MethodInfo PluginScript : : get_method_info ( const StringName & p_method ) const {
ASSERT_SCRIPT_VALID_V ( MethodInfo ( ) ) ;
const Map < StringName , MethodInfo > : : Element * e = _methods_info . find ( p_method ) ;
2020-04-01 23:20:12 +00:00
if ( e ! = nullptr ) {
2017-10-08 21:47:38 +00:00
return e - > get ( ) ;
} else {
return MethodInfo ( ) ;
}
}
bool PluginScript : : has_property ( const StringName & p_method ) const {
ASSERT_SCRIPT_VALID_V ( false ) ;
return _properties_info . has ( p_method ) ;
}
PropertyInfo PluginScript : : get_property_info ( const StringName & p_property ) const {
ASSERT_SCRIPT_VALID_V ( PropertyInfo ( ) ) ;
const Map < StringName , PropertyInfo > : : Element * e = _properties_info . find ( p_property ) ;
2020-04-01 23:20:12 +00:00
if ( e ! = nullptr ) {
2017-10-08 21:47:38 +00:00
return e - > get ( ) ;
} else {
return PropertyInfo ( ) ;
}
}
bool PluginScript : : get_property_default_value ( const StringName & p_property , Variant & r_value ) const {
ASSERT_SCRIPT_VALID_V ( false ) ;
# ifdef TOOLS_ENABLED
const Map < StringName , Variant > : : Element * e = _properties_default_values . find ( p_property ) ;
2020-04-01 23:20:12 +00:00
if ( e ! = nullptr ) {
2017-10-08 21:47:38 +00:00
r_value = e - > get ( ) ;
return true ;
} else {
return false ;
}
# endif
return false ;
}
ScriptLanguage * PluginScript : : get_language ( ) const {
return _language ;
}
Error PluginScript : : load_source_code ( const String & p_path ) {
2020-02-17 21:06:54 +00:00
Vector < uint8_t > sourcef ;
2017-10-08 21:47:38 +00:00
Error err ;
FileAccess * f = FileAccess : : open ( p_path , FileAccess : : READ , & err ) ;
2019-09-25 08:28:50 +00:00
ERR_FAIL_COND_V_MSG ( err , err , " Cannot open file ' " + p_path + " '. " ) ;
2017-10-08 21:47:38 +00:00
2021-05-25 06:58:49 +00:00
uint64_t len = f - > get_length ( ) ;
2017-10-08 21:47:38 +00:00
sourcef . resize ( len + 1 ) ;
2020-02-17 21:06:54 +00:00
uint8_t * w = sourcef . ptrw ( ) ;
2019-03-26 17:51:13 +00:00
uint64_t r = f - > get_buffer ( w , len ) ;
2017-10-08 21:47:38 +00:00
f - > close ( ) ;
memdelete ( f ) ;
ERR_FAIL_COND_V ( r ! = len , ERR_CANT_OPEN ) ;
w [ len ] = 0 ;
String s ;
2020-02-17 21:06:54 +00:00
if ( s . parse_utf8 ( ( const char * ) w ) ) {
2019-08-09 04:49:33 +00:00
ERR_FAIL_V_MSG ( ERR_INVALID_DATA , " Script ' " + p_path + " ' contains invalid unicode (UTF-8), so it was not loaded. Please ensure that scripts are saved in valid UTF-8 unicode. " ) ;
2017-10-08 21:47:38 +00:00
}
_source = s ;
# ifdef TOOLS_ENABLED
// source_changed_cache=true;
# endif
_path = p_path ;
return OK ;
}
bool PluginScript : : has_script_signal ( const StringName & p_signal ) const {
ASSERT_SCRIPT_VALID_V ( false ) ;
return _signals_info . has ( p_signal ) ;
}
void PluginScript : : get_script_signal_list ( List < MethodInfo > * r_signals ) const {
ASSERT_SCRIPT_VALID ( ) ;
2020-04-01 23:20:12 +00:00
for ( Map < StringName , MethodInfo > : : Element * e = _signals_info . front ( ) ; e ! = nullptr ; e = e - > next ( ) ) {
2017-10-08 21:47:38 +00:00
r_signals - > push_back ( e - > get ( ) ) ;
}
}
int PluginScript : : get_member_line ( const StringName & p_member ) const {
# ifdef TOOLS_ENABLED
2020-05-14 14:41:43 +00:00
if ( _member_lines . has ( p_member ) ) {
2017-10-08 21:47:38 +00:00
return _member_lines [ p_member ] ;
2020-05-14 14:41:43 +00:00
}
2017-10-08 21:47:38 +00:00
# endif
2020-05-14 14:41:43 +00:00
return - 1 ;
2017-10-08 21:47:38 +00:00
}
2021-05-26 12:07:57 +00:00
const Vector < MultiplayerAPI : : RPCConfig > PluginScript : : get_rpc_methods ( ) const {
2020-02-12 10:51:50 +00:00
return _rpc_methods ;
}
2017-12-06 20:36:34 +00:00
PluginScript : : PluginScript ( ) :
_script_list ( this ) {
2017-10-08 21:47:38 +00:00
}
void PluginScript : : init ( PluginScriptLanguage * language ) {
_desc = & language - > _desc . script_desc ;
_language = language ;
# ifdef DEBUG_ENABLED
_language - > lock ( ) ;
_language - > _script_list . add ( & _script_list ) ;
_language - > unlock ( ) ;
# endif
}
PluginScript : : ~ PluginScript ( ) {
2019-06-29 12:51:33 +00:00
if ( _desc & & _data ) {
_desc - > finish ( _data ) ;
}
2017-10-08 21:47:38 +00:00
# ifdef DEBUG_ENABLED
2019-06-29 12:51:33 +00:00
if ( _language ) {
_language - > lock ( ) ;
_language - > _script_list . remove ( & _script_list ) ;
_language - > unlock ( ) ;
}
2017-10-08 21:47:38 +00:00
# endif
}