mirror of
https://bitbucket.org/librepilot/librepilot.git
synced 2025-02-01 17:52:12 +01:00
fc1e3f574c
Also implement some ordering (quite ugly still) in the module init and task creation order so we can decide which module to start/init first and which module to start/init last. This will be replaced/adapter with the uavobject list later (once it's implemented). reserving some space for module init and task create parameters to customize module/task creation (this will be usefull once we get the list and customization from customer). Changes have been made for OP and CC. Tested comped with CC,OP, sim_posix. Only ran on bench with CC for couple of minutes (code increase expected but no dropping of stack which is good). This gives task creation at the time wherethe all heap is available.
287 lines
8.3 KiB
C
287 lines
8.3 KiB
C
/**
|
|
******************************************************************************
|
|
* @addtogroup OpenPilotModules OpenPilot Modules
|
|
* @{
|
|
* @addtogroup FlightPlan Flight Plan Module
|
|
* @brief Executes flight plan scripts in Python
|
|
* @{
|
|
*
|
|
* @file flightplan.c
|
|
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
|
|
* @brief Executes flight plan scripts in Python
|
|
*
|
|
* @see The GNU Public License (GPL) Version 3
|
|
*
|
|
*****************************************************************************/
|
|
/*
|
|
* 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, write to the Free Software Foundation, Inc.,
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "openpilot.h"
|
|
#include "pm.h"
|
|
#include "flightplan.h"
|
|
#include "flightplanstatus.h"
|
|
#include "flightplancontrol.h"
|
|
#include "flightplansettings.h"
|
|
|
|
// Private constants
|
|
#define STACK_SIZE_BYTES 1500
|
|
#define TASK_PRIORITY (tskIDLE_PRIORITY+1)
|
|
#define MAX_QUEUE_SIZE 2
|
|
|
|
// Private types
|
|
|
|
// Private variables
|
|
static xTaskHandle taskHandle;
|
|
static xQueueHandle queue;
|
|
|
|
// Private functions
|
|
static void flightPlanTask(void *parameters);
|
|
static void objectUpdatedCb(UAVObjEvent * ev);
|
|
|
|
// External variables (temporary, TODO: this will be loaded from the SD card)
|
|
extern unsigned char usrlib_img[];
|
|
|
|
/**
|
|
* Module initialization
|
|
*/
|
|
int32_t FlightPlanStart()
|
|
{
|
|
taskHandle = NULL;
|
|
|
|
// Start VM thread
|
|
xTaskCreate(flightPlanTask, (signed char *)"FlightPlan", STACK_SIZE_BYTES/4, NULL, TASK_PRIORITY, &taskHandle);
|
|
TaskMonitorAdd(TASKINFO_RUNNING_FLIGHTPLAN, taskHandle);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Module initialization
|
|
*/
|
|
int32_t FlightPlanInitialize()
|
|
{
|
|
taskHandle = NULL;
|
|
|
|
// Listen for object updates
|
|
FlightPlanControlConnectCallback(&objectUpdatedCb);
|
|
|
|
// Create object queue
|
|
queue = xQueueCreate(MAX_QUEUE_SIZE, sizeof(UAVObjEvent));
|
|
|
|
// Listen for FlightPlanControl updates
|
|
FlightPlanControlConnectQueue(queue);
|
|
|
|
return 0;
|
|
}
|
|
module_initcall(FlightPlanInitialize, 0, FlightPlanStart, 0, MODULE_EXEC_NOORDER_FLAG);
|
|
/**
|
|
* Module task
|
|
*/
|
|
static void flightPlanTask(void *parameters)
|
|
{
|
|
UAVObjEvent ev;
|
|
PmReturn_t retval;
|
|
FlightPlanStatusData status;
|
|
FlightPlanControlData control;
|
|
|
|
// Setup status object
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_STOPPED;
|
|
status.ErrorFileID = 0;
|
|
status.ErrorLineNum = 0;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_NONE;
|
|
status.Debug[0] = 0.0;
|
|
status.Debug[1] = 0.0;
|
|
FlightPlanStatusSet(&status);
|
|
|
|
// Main thread loop
|
|
while (1)
|
|
{
|
|
// Wait for FlightPlanControl updates
|
|
while (xQueueReceive(queue, &ev, portMAX_DELAY) != pdTRUE) ;
|
|
|
|
// Get object and check if a start command was sent
|
|
FlightPlanControlGet(&control);
|
|
if ( control.Command == FLIGHTPLANCONTROL_COMMAND_START )
|
|
{
|
|
// Init PyMite
|
|
retval = pm_init(MEMSPACE_PROG, usrlib_img);
|
|
if (retval == PM_RET_OK)
|
|
{
|
|
// Update status
|
|
FlightPlanStatusGet(&status);
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_RUNNING;
|
|
FlightPlanStatusSet(&status);
|
|
// Run the test script (TODO: load from SD card)
|
|
retval = pm_run((uint8_t *)"test");
|
|
// Check if an error or exception was thrown
|
|
if (retval == PM_RET_OK || retval == PM_RET_EX_EXIT)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_STOPPED;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_NONE;
|
|
}
|
|
else if (retval == PM_RET_EX)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_EXCEPTION;
|
|
}
|
|
else if (retval == PM_RET_EX_IO)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_IOERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_ZDIV)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_DIVBYZERO;
|
|
}
|
|
else if (retval == PM_RET_EX_ASSRT)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_ASSERTERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_ATTR)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_ATTRIBUTEERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_IMPRT)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_IMPORTERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_INDX)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_INDEXERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_KEY)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_KEYERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_MEM)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_MEMORYERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_NAME)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_NAMEERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_SYNTAX)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_SYNTAXERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_SYS)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_SYSTEMERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_TYPE)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_TYPEERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_VAL)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_VALUEERROR;
|
|
}
|
|
else if (retval == PM_RET_EX_STOP)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_STOPITERATION;
|
|
}
|
|
else if (retval == PM_RET_EX_WARN)
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_WARNING;
|
|
}
|
|
else
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_UNKNOWNERROR;
|
|
}
|
|
// Get file ID and line number of error (if one)
|
|
status.ErrorFileID = gVmGlobal.errFileId;
|
|
status.ErrorLineNum = gVmGlobal.errLineNum;
|
|
}
|
|
else
|
|
{
|
|
status.Status = FLIGHTPLANSTATUS_STATUS_ERROR;
|
|
status.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_VMINITERROR;
|
|
}
|
|
|
|
// Update status object
|
|
FlightPlanStatusSet(&status);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Function called in response to object updates.
|
|
* Used to force kill the VM thread.
|
|
*/
|
|
static void objectUpdatedCb(UAVObjEvent * ev)
|
|
{
|
|
FlightPlanControlData controlData;
|
|
FlightPlanStatusData statusData;
|
|
|
|
// If the object updated was the FlightPlanControl execute requested action
|
|
if ( ev->obj == FlightPlanControlHandle() )
|
|
{
|
|
// Get data
|
|
FlightPlanControlGet(&controlData);
|
|
// Execute command
|
|
if ( controlData.Command == FLIGHTPLANCONTROL_COMMAND_START )
|
|
{
|
|
// Start VM task if not running already
|
|
if ( taskHandle == NULL )
|
|
{
|
|
xTaskCreate(flightPlanTask, (signed char *)"FlightPlan", STACK_SIZE_BYTES/4, NULL, TASK_PRIORITY, &taskHandle);
|
|
TaskMonitorAdd(TASKINFO_RUNNING_FLIGHTPLAN, taskHandle);
|
|
}
|
|
}
|
|
else if ( controlData.Command == FLIGHTPLANCONTROL_COMMAND_KILL )
|
|
{
|
|
// Force kill VM task if it is already running
|
|
// (NOTE: the STOP command is preferred as it allows the script to terminate without killing the VM)
|
|
if ( taskHandle != NULL )
|
|
{
|
|
// Kill VM
|
|
TaskMonitorRemove(TASKINFO_RUNNING_FLIGHTPLAN);
|
|
vTaskDelete(taskHandle);
|
|
taskHandle = NULL;
|
|
// Update status object
|
|
statusData.Status = FLIGHTPLANSTATUS_STATUS_STOPPED;
|
|
statusData.ErrorFileID = 0;
|
|
statusData.ErrorLineNum = 0;
|
|
statusData.ErrorType = FLIGHTPLANSTATUS_ERRORTYPE_NONE;
|
|
statusData.Debug[0] = 0.0;
|
|
statusData.Debug[1] = 0.0;
|
|
FlightPlanStatusSet(&statusData);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @}
|
|
* @}
|
|
*/
|