2018-09-27 19:30:55 -03:00
/*
This program is free software : you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation , either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include <AP_Scripting/AP_Scripting.h>
# include <AP_HAL/AP_HAL.h>
2018-09-27 20:04:37 -03:00
# include <GCS_MAVLink/GCS.h>
2018-10-31 19:43:23 -03:00
# include "lua_scripts.h"
2018-10-03 17:37:43 -03:00
2018-09-27 20:04:37 -03:00
// ensure that we have a set of stack sizes, and enforce constraints around it
// except for the minimum size, these are allowed to be defined by the build system
# undef SCRIPTING_STACK_MIN_SIZE
2018-10-31 19:43:23 -03:00
# define SCRIPTING_STACK_MIN_SIZE (8 * 1024)
2018-09-27 20:04:37 -03:00
# if !defined(SCRIPTING_STACK_SIZE)
2019-07-17 01:27:25 -03:00
# define SCRIPTING_STACK_SIZE (17 * 1024) // Linux experiences stack corruption at ~16.25KB when handed bad scripts
2018-09-27 20:04:37 -03:00
# endif // !defined(SCRIPTING_STACK_SIZE)
# if !defined(SCRIPTING_STACK_MAX_SIZE)
2018-10-31 19:43:23 -03:00
# define SCRIPTING_STACK_MAX_SIZE (64 * 1024)
2018-09-27 20:04:37 -03:00
# endif // !defined(SCRIPTING_STACK_MAX_SIZE)
2020-01-08 18:10:18 -04:00
# if !defined(SCRIPTING_HEAP_SIZE)
2022-04-10 23:59:49 -03:00
# if CONFIG_HAL_BOARD == HAL_BOARD_SITL || CONFIG_HAL_BOARD == HAL_BOARD_LINUX || HAL_MEM_CLASS >= HAL_MEM_CLASS_500
# define SCRIPTING_HEAP_SIZE (100 * 1024)
2020-01-08 18:10:18 -04:00
# else
# define SCRIPTING_HEAP_SIZE (43 * 1024)
# endif
# endif // !defined(SCRIPTING_HEAP_SIZE)
2018-09-27 20:04:37 -03:00
static_assert ( SCRIPTING_STACK_SIZE > = SCRIPTING_STACK_MIN_SIZE , " Scripting requires a larger minimum stack size " ) ;
static_assert ( SCRIPTING_STACK_SIZE < = SCRIPTING_STACK_MAX_SIZE , " Scripting requires a smaller stack size " ) ;
2021-06-14 13:01:09 -03:00
# ifndef SCRIPTING_ENABLE_DEFAULT
# define SCRIPTING_ENABLE_DEFAULT 0
# endif
2018-09-27 20:04:37 -03:00
extern const AP_HAL : : HAL & hal ;
const AP_Param : : GroupInfo AP_Scripting : : var_info [ ] = {
// @Param: ENABLE
// @DisplayName: Enable Scripting
// @Description: Controls if scripting is enabled
// @Values: 0:None,1:Lua Scripts
// @RebootRequired: True
// @User: Advanced
2021-06-14 13:01:09 -03:00
AP_GROUPINFO_FLAGS ( " ENABLE " , 1 , AP_Scripting , _enable , SCRIPTING_ENABLE_DEFAULT , AP_PARAM_FLAG_ENABLE ) ,
2018-09-27 20:04:37 -03:00
2018-10-30 23:06:32 -03:00
// @Param: VM_I_COUNT
// @DisplayName: Scripting Virtual Machine Instruction Count
// @Description: The number virtual machine instructions that can be run before considering a script to have taken an excessive amount of time
// @Range: 1000 1000000
// @Increment: 10000
// @User: Advanced
AP_GROUPINFO ( " VM_I_COUNT " , 2 , AP_Scripting , _script_vm_exec_count , 10000 ) ,
2018-12-08 22:35:08 -04:00
// @Param: HEAP_SIZE
// @DisplayName: Scripting Heap Size
// @Description: Amount of memory available for scripting
// @Range: 1024 1048576
// @Increment: 1024
// @User: Advanced
// @RebootRequired: True
2020-01-08 18:10:18 -04:00
AP_GROUPINFO ( " HEAP_SIZE " , 3 , AP_Scripting , _script_heap_size , SCRIPTING_HEAP_SIZE ) ,
2018-12-08 22:35:08 -04:00
2021-11-13 13:09:09 -04:00
// @Param: DEBUG_OPTS
2020-02-18 21:10:24 -04:00
// @DisplayName: Scripting Debug Level
2021-11-13 13:09:09 -04:00
// @Description: Debugging options
2022-08-28 16:31:20 -03:00
// @Bitmask: 0:No Scripts to run message if all scripts have stopped, 1:Runtime messages for memory usage and execution time, 2:Suppress logging scripts to dataflash, 3:log runtime memory usage and execution time, 4:Disable pre-arm check
2020-02-17 22:39:17 -04:00
// @User: Advanced
2021-11-13 13:09:09 -04:00
AP_GROUPINFO ( " DEBUG_OPTS " , 4 , AP_Scripting , _debug_options , 0 ) ,
2019-03-22 12:19:59 -03:00
2020-07-09 09:36:45 -03:00
// @Param: USER1
// @DisplayName: Scripting User Parameter1
// @Description: General purpose user variable input for scripts
// @User: Standard
AP_GROUPINFO ( " USER1 " , 5 , AP_Scripting , _user [ 0 ] , 0.0 ) ,
// @Param: USER2
// @DisplayName: Scripting User Parameter2
// @Description: General purpose user variable input for scripts
// @User: Standard
AP_GROUPINFO ( " USER2 " , 6 , AP_Scripting , _user [ 1 ] , 0.0 ) ,
// @Param: USER3
// @DisplayName: Scripting User Parameter3
// @Description: General purpose user variable input for scripts
// @User: Standard
AP_GROUPINFO ( " USER3 " , 7 , AP_Scripting , _user [ 2 ] , 0.0 ) ,
// @Param: USER4
// @DisplayName: Scripting User Parameter4
// @Description: General purpose user variable input for scripts
// @User: Standard
AP_GROUPINFO ( " USER4 " , 8 , AP_Scripting , _user [ 3 ] , 0.0 ) ,
2021-11-30 02:09:20 -04:00
// @Param: USER5
// @DisplayName: Scripting User Parameter5
// @Description: General purpose user variable input for scripts
// @User: Standard
AP_GROUPINFO ( " USER5 " , 10 , AP_Scripting , _user [ 4 ] , 0.0 ) ,
// @Param: USER6
// @DisplayName: Scripting User Parameter6
// @Description: General purpose user variable input for scripts
// @User: Standard
AP_GROUPINFO ( " USER6 " , 11 , AP_Scripting , _user [ 5 ] , 0.0 ) ,
2020-10-26 22:08:14 -03:00
// @Param: DIR_DISABLE
// @DisplayName: Directory disable
// @Description: This will stop scripts being loaded from the given locations
// @Bitmask: 0:ROMFS, 1:APM/scripts
// @RebootRequired: True
// @User: Advanced
AP_GROUPINFO ( " DIR_DISABLE " , 9 , AP_Scripting , _dir_disable , 0 ) ,
2018-09-27 20:04:37 -03:00
AP_GROUPEND
} ;
2018-09-27 19:30:55 -03:00
2018-10-31 19:43:23 -03:00
AP_Scripting : : AP_Scripting ( ) {
2018-09-27 20:04:37 -03:00
AP_Param : : setup_object_defaults ( this , var_info ) ;
2018-09-27 19:30:55 -03:00
# if CONFIG_HAL_BOARD == HAL_BOARD_SITL
if ( _singleton ! = nullptr ) {
AP_HAL : : panic ( " Scripting must be a singleton " ) ;
}
# endif // CONFIG_HAL_BOARD == HAL_BOARD_SITL
_singleton = this ;
}
2019-11-26 01:39:17 -04:00
void AP_Scripting : : init ( void ) {
2018-09-27 20:04:37 -03:00
if ( ! _enable ) {
2019-11-26 01:39:17 -04:00
return ;
2018-09-27 20:04:37 -03:00
}
2018-09-27 19:30:55 -03:00
2020-04-17 15:46:27 -03:00
const char * dir_name = SCRIPTING_DIRECTORY ;
2020-06-20 09:35:38 -03:00
if ( AP : : FS ( ) . mkdir ( dir_name ) ) {
if ( errno ! = EEXIST ) {
2022-08-28 16:25:55 -03:00
gcs ( ) . send_text ( MAV_SEVERITY_INFO , " Scripting: failed to create (%s) " , dir_name ) ;
2020-06-20 09:35:38 -03:00
}
2020-04-17 15:46:27 -03:00
}
2018-09-27 20:04:37 -03:00
if ( ! hal . scheduler - > thread_create ( FUNCTOR_BIND_MEMBER ( & AP_Scripting : : thread , void ) ,
" Scripting " , SCRIPTING_STACK_SIZE , AP_HAL : : Scheduler : : PRIORITY_SCRIPTING , 0 ) ) {
2022-08-28 16:24:21 -03:00
gcs ( ) . send_text ( MAV_SEVERITY_ERROR , " Scripting: %s " , " failed to start " ) ;
_thread_failed = true ;
2018-09-27 20:04:37 -03:00
}
}
2018-09-27 19:30:55 -03:00
2020-02-05 18:58:33 -04:00
MAV_RESULT AP_Scripting : : handle_command_int_packet ( const mavlink_command_int_t & packet ) {
switch ( ( SCRIPTING_CMD ) packet . param1 ) {
case SCRIPTING_CMD_REPL_START :
return repl_start ( ) ? MAV_RESULT_ACCEPTED : MAV_RESULT_FAILED ;
case SCRIPTING_CMD_REPL_STOP :
repl_stop ( ) ;
return MAV_RESULT_ACCEPTED ;
2021-08-04 13:49:17 -03:00
case SCRIPTING_CMD_STOP :
_restart = false ;
_stop = true ;
return MAV_RESULT_ACCEPTED ;
case SCRIPTING_CMD_STOP_AND_RESTART :
_restart = true ;
_stop = true ;
return MAV_RESULT_ACCEPTED ;
2020-02-05 18:58:33 -04:00
case SCRIPTING_CMD_ENUM_END : // cope with MAVLink generator appending to our enum
break ;
}
return MAV_RESULT_UNSUPPORTED ;
}
bool AP_Scripting : : repl_start ( void ) {
if ( terminal . session ) { // it's already running, this is fine
return true ;
}
// nuke the old folder and all contents
struct stat st ;
if ( ( AP : : FS ( ) . stat ( REPL_DIRECTORY , & st ) = = - 1 ) & &
( AP : : FS ( ) . unlink ( REPL_DIRECTORY ) = = - 1 ) & &
( errno ! = EEXIST ) ) {
2022-08-28 16:25:55 -03:00
gcs ( ) . send_text ( MAV_SEVERITY_INFO , " Scripting: Unable to delete old REPL %s " , strerror ( errno ) ) ;
2020-02-05 18:58:33 -04:00
}
// create a new folder
AP : : FS ( ) . mkdir ( REPL_DIRECTORY ) ;
// delete old files in case we couldn't
AP : : FS ( ) . unlink ( REPL_DIRECTORY " /in " ) ;
AP : : FS ( ) . unlink ( REPL_DIRECTORY " /out " ) ;
// make the output pointer
terminal . output_fd = AP : : FS ( ) . open ( REPL_OUT , O_WRONLY | O_CREAT | O_TRUNC ) ;
if ( terminal . output_fd = = - 1 ) {
2022-08-28 16:25:55 -03:00
gcs ( ) . send_text ( MAV_SEVERITY_INFO , " Scripting: %s " , " Unable to make new REPL " ) ;
2020-02-05 18:58:33 -04:00
return false ;
}
terminal . session = true ;
return true ;
}
void AP_Scripting : : repl_stop ( void ) {
terminal . session = false ;
// can't do any more cleanup here, closing the open FD's is the REPL's responsibility
}
2018-09-27 20:04:37 -03:00
void AP_Scripting : : thread ( void ) {
2021-08-04 13:49:17 -03:00
while ( true ) {
// reset flags
_stop = false ;
_restart = false ;
2022-08-28 16:24:21 -03:00
_init_failed = false ;
2021-08-04 13:49:17 -03:00
2021-11-13 13:09:09 -04:00
lua_scripts * lua = new lua_scripts ( _script_vm_exec_count , _script_heap_size , _debug_options , terminal ) ;
2021-08-04 13:49:17 -03:00
if ( lua = = nullptr | | ! lua - > heap_allocated ( ) ) {
2022-08-28 16:25:55 -03:00
gcs ( ) . send_text ( MAV_SEVERITY_CRITICAL , " Scripting: %s " , " Unable to allocate memory " ) ;
2021-08-04 13:49:17 -03:00
_init_failed = true ;
} else {
// run won't return while scripting is still active
lua - > run ( ) ;
// only reachable if the lua backend has died for any reason
2022-08-28 16:25:55 -03:00
gcs ( ) . send_text ( MAV_SEVERITY_CRITICAL , " Scripting: %s " , " stopped " ) ;
2021-08-04 13:49:17 -03:00
}
2020-03-28 00:11:09 -03:00
delete lua ;
2022-10-12 15:33:54 -03:00
lua = nullptr ;
2018-10-29 22:05:36 -03:00
2022-09-10 13:22:08 -03:00
// clear allocated i2c devices
for ( uint8_t i = 0 ; i < SCRIPTING_MAX_NUM_I2C_DEVICE ; i + + ) {
delete _i2c_dev [ i ] ;
_i2c_dev [ i ] = nullptr ;
}
num_i2c_devices = 0 ;
2021-08-04 13:49:17 -03:00
bool cleared = false ;
while ( true ) {
// 1hz check if we should restart
hal . scheduler - > delay ( 1000 ) ;
if ( ! enabled ( ) ) {
// enable must be put to 0 and back to 1 to restart from params
cleared = true ;
continue ;
}
// must be enabled to get this far
if ( cleared | | _restart ) {
2022-08-28 16:25:55 -03:00
gcs ( ) . send_text ( MAV_SEVERITY_CRITICAL , " Scripting: %s " , " restarted " ) ;
2021-08-04 13:49:17 -03:00
break ;
}
2021-11-13 13:09:09 -04:00
if ( ( _debug_options . get ( ) & uint8_t ( lua_scripts : : DebugLevel : : NO_SCRIPTS_TO_RUN ) ) ! = 0 ) {
2022-08-28 16:25:55 -03:00
gcs ( ) . send_text ( MAV_SEVERITY_DEBUG , " Scripting: %s " , " stopped " ) ;
2021-08-04 13:49:17 -03:00
}
}
}
2018-09-27 19:30:55 -03:00
}
2021-02-25 21:09:17 -04:00
void AP_Scripting : : handle_mission_command ( const AP_Mission : : Mission_Command & cmd_in )
{
if ( ! _enable ) {
return ;
}
if ( mission_data = = nullptr ) {
// load buffer
mission_data = new ObjectBuffer < struct AP_Scripting : : scripting_mission_cmd > ( mission_cmd_queue_size ) ;
2023-01-01 17:33:22 -04:00
if ( mission_data ! = nullptr & & mission_data - > get_size ( ) = = 0 ) {
delete mission_data ;
mission_data = nullptr ;
}
2021-02-25 21:09:17 -04:00
if ( mission_data = = nullptr ) {
2022-08-28 16:25:55 -03:00
gcs ( ) . send_text ( MAV_SEVERITY_INFO , " Scripting: %s " , " unable to receive mission command " ) ;
2021-02-25 21:09:17 -04:00
return ;
}
}
struct scripting_mission_cmd cmd { cmd_in . p1 ,
cmd_in . content . scripting . p1 ,
cmd_in . content . scripting . p2 ,
cmd_in . content . scripting . p3 ,
AP_HAL : : millis ( ) } ;
mission_data - > push ( cmd ) ;
}
2022-08-28 16:24:21 -03:00
bool AP_Scripting : : arming_checks ( size_t buflen , char * buffer ) const
{
2022-08-28 16:31:20 -03:00
if ( ! enabled ( ) | | ( ( _debug_options . get ( ) & uint8_t ( lua_scripts : : DebugLevel : : DISABLE_PRE_ARM ) ) ! = 0 ) ) {
2022-08-28 16:24:21 -03:00
return true ;
}
if ( _thread_failed ) {
hal . util - > snprintf ( buffer , buflen , " Scripting: %s " , " failed to start " ) ;
return false ;
}
if ( _init_failed ) {
hal . util - > snprintf ( buffer , buflen , " Scripting: %s " , " out of memory " ) ;
return false ;
}
2022-08-30 17:47:36 -03:00
lua_scripts : : get_last_error_semaphore ( ) - > take_blocking ( ) ;
2022-08-28 16:24:21 -03:00
const char * error_buf = lua_scripts : : get_last_error_message ( ) ;
if ( error_buf ! = nullptr ) {
hal . util - > snprintf ( buffer , buflen , " Scripting: %s " , error_buf ) ;
2022-08-30 17:47:36 -03:00
lua_scripts : : get_last_error_semaphore ( ) - > give ( ) ;
2022-08-28 16:24:21 -03:00
return false ;
}
2022-08-30 17:47:36 -03:00
lua_scripts : : get_last_error_semaphore ( ) - > give ( ) ;
2022-08-28 16:24:21 -03:00
return true ;
}
2018-09-27 20:04:37 -03:00
AP_Scripting * AP_Scripting : : _singleton = nullptr ;
2018-09-27 19:30:55 -03:00
2018-09-27 20:04:37 -03:00
namespace AP {
AP_Scripting * scripting ( ) {
return AP_Scripting : : get_singleton ( ) ;
}
}