2011-06-15 18:18:29 +02:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
* @addtogroup OpenPilotModules OpenPilot Modules
|
|
|
|
* @{
|
|
|
|
* @addtogroup CameraStab Camera Stabilization Module
|
|
|
|
* @brief Camera stabilization module
|
|
|
|
* Updates accessory outputs with values appropriate for camera stabilization
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file camerastab.c
|
|
|
|
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
|
|
|
|
* @brief Stabilize camera against the roll pitch and yaw of aircraft
|
|
|
|
*
|
|
|
|
* @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
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Output object: Accessory
|
|
|
|
*
|
|
|
|
* This module will periodically calculate the output values for stabilizing the camera
|
|
|
|
*
|
|
|
|
* UAVObjects are automatically generated by the UAVObjectGenerator from
|
|
|
|
* the object definition XML file.
|
|
|
|
*
|
|
|
|
* Modules have no API, all communication to other modules is done through UAVObjects.
|
|
|
|
* However modules may use the API exposed by shared libraries.
|
|
|
|
* See the OpenPilot wiki for more details.
|
|
|
|
* http://www.openpilot.org/OpenPilot_Application_Architecture
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "openpilot.h"
|
|
|
|
|
|
|
|
#include "accessorydesired.h"
|
|
|
|
#include "attitudeactual.h"
|
2011-06-23 07:46:41 +02:00
|
|
|
#include "camerastabsettings.h"
|
2011-08-10 03:43:48 +02:00
|
|
|
#include "cameradesired.h"
|
2011-11-11 11:39:57 +01:00
|
|
|
#include "hwsettings.h"
|
2011-06-15 18:18:29 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Configuration
|
|
|
|
//
|
2011-06-23 07:46:41 +02:00
|
|
|
#define SAMPLE_PERIOD_MS 10
|
2011-06-15 18:18:29 +02:00
|
|
|
|
|
|
|
// Private types
|
|
|
|
|
|
|
|
// Private variables
|
|
|
|
|
2011-11-11 11:39:57 +01:00
|
|
|
static uint8_t camerastabEnabled = 0;
|
|
|
|
|
2011-06-15 18:18:29 +02:00
|
|
|
// Private functions
|
|
|
|
static void attitudeUpdated(UAVObjEvent* ev);
|
2011-08-11 21:34:38 +02:00
|
|
|
static float bound(float val);
|
2011-06-15 18:18:29 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialise the module, called on startup
|
|
|
|
* \returns 0 on success or -1 if initialisation failed
|
|
|
|
*/
|
|
|
|
int32_t CameraStabInitialize(void)
|
|
|
|
{
|
|
|
|
static UAVObjEvent ev;
|
|
|
|
|
2011-11-11 11:39:57 +01:00
|
|
|
HwSettingsInitialize();
|
|
|
|
uint8_t optionalModules[HWSETTINGS_OPTIONALMODULES_NUMELEM];
|
|
|
|
HwSettingsOptionalModulesGet(optionalModules);
|
2011-11-13 04:25:17 +01:00
|
|
|
if (optionalModules[HWSETTINGS_OPTIONALMODULES_CAMERASTAB] == HWSETTINGS_OPTIONALMODULES_ENABLED) {
|
2011-11-11 11:39:57 +01:00
|
|
|
camerastabEnabled=1;
|
|
|
|
} else {
|
|
|
|
camerastabEnabled=0;
|
|
|
|
}
|
2011-06-23 07:46:41 +02:00
|
|
|
|
2011-11-11 11:39:57 +01:00
|
|
|
if (camerastabEnabled) {
|
2011-06-15 18:18:29 +02:00
|
|
|
|
2011-11-11 11:39:57 +01:00
|
|
|
AttitudeActualInitialize();
|
|
|
|
|
|
|
|
ev.obj = AttitudeActualHandle();
|
|
|
|
ev.instId = 0;
|
|
|
|
ev.event = 0;
|
|
|
|
|
|
|
|
CameraStabSettingsInitialize();
|
|
|
|
CameraDesiredInitialize();
|
|
|
|
|
|
|
|
EventPeriodicCallbackCreate(&ev, attitudeUpdated, SAMPLE_PERIOD_MS / portTICK_RATE_MS);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
2011-06-15 18:18:29 +02:00
|
|
|
}
|
|
|
|
|
2011-08-20 01:24:06 +02:00
|
|
|
/* stub: module has no module thread */
|
|
|
|
int32_t CameraStabStart(void)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-11-11 11:39:57 +01:00
|
|
|
MODULE_INITCALL(CameraStabInitialize, CameraStabStart)
|
|
|
|
|
2011-06-15 18:18:29 +02:00
|
|
|
static void attitudeUpdated(UAVObjEvent* ev)
|
|
|
|
{
|
|
|
|
if (ev->obj != AttitudeActualHandle())
|
|
|
|
return;
|
|
|
|
|
|
|
|
float attitude;
|
2011-08-10 03:43:48 +02:00
|
|
|
float output;
|
2011-06-15 18:18:29 +02:00
|
|
|
AccessoryDesiredData accessory;
|
2011-06-23 07:46:41 +02:00
|
|
|
|
|
|
|
CameraStabSettingsData cameraStab;
|
|
|
|
CameraStabSettingsGet(&cameraStab);
|
|
|
|
|
|
|
|
// Read any input channels
|
|
|
|
float inputs[3] = {0,0,0};
|
|
|
|
if(cameraStab.Inputs[CAMERASTABSETTINGS_INPUTS_ROLL] != CAMERASTABSETTINGS_INPUTS_NONE) {
|
2011-06-23 10:01:37 +02:00
|
|
|
if(AccessoryDesiredInstGet(cameraStab.Inputs[CAMERASTABSETTINGS_INPUTS_ROLL] - CAMERASTABSETTINGS_INPUTS_ACCESSORY0, &accessory) == 0)
|
2011-06-23 07:46:41 +02:00
|
|
|
inputs[0] = accessory.AccessoryVal * cameraStab.InputRange[CAMERASTABSETTINGS_INPUTRANGE_ROLL];
|
|
|
|
}
|
|
|
|
if(cameraStab.Inputs[CAMERASTABSETTINGS_INPUTS_PITCH] != CAMERASTABSETTINGS_INPUTS_NONE) {
|
2011-06-23 10:01:37 +02:00
|
|
|
if(AccessoryDesiredInstGet(cameraStab.Inputs[CAMERASTABSETTINGS_INPUTS_PITCH] - CAMERASTABSETTINGS_INPUTS_ACCESSORY0, &accessory) == 0)
|
2011-06-23 07:46:41 +02:00
|
|
|
inputs[1] = accessory.AccessoryVal * cameraStab.InputRange[CAMERASTABSETTINGS_INPUTRANGE_PITCH];
|
|
|
|
}
|
|
|
|
if(cameraStab.Inputs[CAMERASTABSETTINGS_INPUTS_YAW] != CAMERASTABSETTINGS_INPUTS_NONE) {
|
2011-06-23 10:01:37 +02:00
|
|
|
if(AccessoryDesiredInstGet(cameraStab.Inputs[CAMERASTABSETTINGS_INPUTS_YAW] - CAMERASTABSETTINGS_INPUTS_ACCESSORY0, &accessory) == 0)
|
2011-06-23 07:46:41 +02:00
|
|
|
inputs[2] = accessory.AccessoryVal * cameraStab.InputRange[CAMERASTABSETTINGS_INPUTRANGE_YAW];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set output channels
|
2011-08-10 03:43:48 +02:00
|
|
|
AttitudeActualRollGet(&attitude);
|
2011-08-11 21:34:38 +02:00
|
|
|
output = bound((attitude + inputs[0]) / cameraStab.OutputRange[CAMERASTABSETTINGS_OUTPUTRANGE_ROLL]);
|
2011-08-10 03:43:48 +02:00
|
|
|
CameraDesiredRollSet(&output);
|
2011-06-23 07:46:41 +02:00
|
|
|
|
2011-08-10 03:43:48 +02:00
|
|
|
AttitudeActualPitchGet(&attitude);
|
2011-08-11 21:34:38 +02:00
|
|
|
output = bound((attitude + inputs[1]) / cameraStab.OutputRange[CAMERASTABSETTINGS_OUTPUTRANGE_PITCH]);
|
2011-08-10 03:43:48 +02:00
|
|
|
CameraDesiredPitchSet(&output);
|
|
|
|
|
|
|
|
AttitudeActualYawGet(&attitude);
|
2011-08-11 21:34:38 +02:00
|
|
|
output = bound((attitude + inputs[2]) / cameraStab.OutputRange[CAMERASTABSETTINGS_OUTPUTRANGE_YAW]);
|
2011-08-10 03:43:48 +02:00
|
|
|
CameraDesiredYawSet(&output);
|
2011-06-23 07:46:41 +02:00
|
|
|
|
2011-06-15 18:18:29 +02:00
|
|
|
}
|
|
|
|
|
2011-08-11 21:34:38 +02:00
|
|
|
float bound(float val)
|
|
|
|
{
|
|
|
|
return (val > 1) ? 1 :
|
|
|
|
(val < -1) ? -1 :
|
|
|
|
val;
|
|
|
|
}
|
2011-06-15 18:18:29 +02:00
|
|
|
/**
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @}
|
|
|
|
*/
|