mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-12 10:58:30 -04:00
169b5a30a5
for triggering every N meters of GPS distance travelled
240 lines
8.2 KiB
C++
240 lines
8.2 KiB
C++
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
#include <AP_Camera.h>
|
|
#include <AP_Relay.h>
|
|
#include <AP_Math.h>
|
|
#include <RC_Channel.h>
|
|
#include <AP_HAL.h>
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
extern int32_t wp_distance; // Note: unfortunately this variable is in meter for ArduPlane and cm for ArduCopter
|
|
|
|
// ------------------------------
|
|
#define CAM_DEBUG DISABLED
|
|
|
|
const AP_Param::GroupInfo AP_Camera::var_info[] PROGMEM = {
|
|
// @Param: TRIGG_TYPE
|
|
// @DisplayName: Camera shutter (trigger) type
|
|
// @Description: how to trigger the camera to take a picture
|
|
// @Values: 0:Servo,1:Relay,2:Servo and turn off throttle,3:Servo when 3m from waypoint,4:transistor
|
|
// @User: Standard
|
|
AP_GROUPINFO("TRIGG_TYPE", 0, AP_Camera, _trigger_type, AP_CAMERA_TRIGGER_DEFAULT_TRIGGER_TYPE),
|
|
|
|
// @Param: DURATION
|
|
// @DisplayName: Duration that shutter is held open
|
|
// @Description: How long the shutter will be held open in 10ths of a second (i.e. enter 10 for 1second, 50 for 5seconds)
|
|
// @Range: 0 50
|
|
// @User: Standard
|
|
AP_GROUPINFO("DURATION", 1, AP_Camera, _trigger_duration, AP_CAMERA_TRIGGER_DEFAULT_DURATION),
|
|
|
|
// @Param: SERVO_ON
|
|
// @DisplayName: Servo ON PWM value
|
|
// @Description: PWM value to move servo to when shutter is activated
|
|
// @Range: 1000 2000
|
|
// @User: Standard
|
|
AP_GROUPINFO("SERVO_ON", 2, AP_Camera, _servo_on_pwm, AP_CAMERA_SERVO_ON_PWM),
|
|
|
|
// @Param: SERVO_OFF
|
|
// @DisplayName: Servo OFF PWM value
|
|
// @Description: PWM value to move servo to when shutter is deactivated
|
|
// @Range: 1000 2000
|
|
// @User: Standard
|
|
AP_GROUPINFO("SERVO_OFF", 3, AP_Camera, _servo_off_pwm, AP_CAMERA_SERVO_OFF_PWM),
|
|
|
|
// @Param: TRIGG_DIST
|
|
// @DisplayName: Camera trigger distance
|
|
// @Description: Distance in meters between camera triggers. If this value is non-zero then the camera will trigger whenever the GPS position changes by this number of meters regardless of what mode the APM is in
|
|
// @User: Standard
|
|
// @Range: 0 1000
|
|
AP_GROUPINFO("TRIGG_DIST", 4, AP_Camera, _trigg_dist, 0),
|
|
|
|
AP_GROUPEND
|
|
};
|
|
|
|
|
|
/// Servo operated camera
|
|
void
|
|
AP_Camera::servo_pic()
|
|
{
|
|
RC_Channel_aux::set_radio(RC_Channel_aux::k_cam_trigger, _servo_on_pwm);
|
|
|
|
// leave a message that it should be active for this many loops (assumes 50hz loops)
|
|
_trigger_counter = constrain_int16(_trigger_duration*5,0,255);
|
|
}
|
|
|
|
/// basic relay activation
|
|
void
|
|
AP_Camera::relay_pic()
|
|
{
|
|
_apm_relay->on();
|
|
|
|
// leave a message that it should be active for this many loops (assumes 50hz loops)
|
|
_trigger_counter = constrain_int16(_trigger_duration*5,0,255);
|
|
}
|
|
|
|
/// pictures blurry? use this trigger. Turns off the throttle until for # of cycles of medium loop then takes the picture and re-enables the throttle.
|
|
void
|
|
AP_Camera::throttle_pic()
|
|
{
|
|
// TODO find a way to do this without using the global parameter g
|
|
// g.channel_throttle.radio_out = g.throttle_min;
|
|
if (_thr_pic_counter == 10) {
|
|
servo_pic(); // triggering method
|
|
_thr_pic_counter = 0;
|
|
// g.channel_throttle.radio_out = g.throttle_cruise;
|
|
}
|
|
_thr_pic_counter++;
|
|
}
|
|
|
|
/// distance_pic - triggers picture when within 3m of waypoint
|
|
void
|
|
AP_Camera::distance_pic()
|
|
{
|
|
if (wp_distance < AP_CAMERA_WP_DISTANCE) {
|
|
servo_pic(); // triggering method
|
|
}
|
|
}
|
|
|
|
/// hacked the circuit to run a transistor? use this trigger to send output.
|
|
void
|
|
AP_Camera::transistor_pic()
|
|
{
|
|
// TODO: Assign pin spare pin for output
|
|
hal.gpio->write(AP_CAMERA_TRANSISTOR_PIN,1);
|
|
|
|
// leave a message that it should be active for two event loop cycles
|
|
_trigger_counter = 1;
|
|
}
|
|
|
|
/// single entry point to take pictures
|
|
void
|
|
AP_Camera::trigger_pic()
|
|
{
|
|
switch (_trigger_type)
|
|
{
|
|
case AP_CAMERA_TRIGGER_TYPE_SERVO:
|
|
servo_pic(); // Servo operated camera
|
|
break;
|
|
case AP_CAMERA_TRIGGER_TYPE_RELAY:
|
|
relay_pic(); // basic relay activation
|
|
break;
|
|
case AP_CAMERA_TRIGGER_TYPE_THROTTLE_OFF_TIME:
|
|
throttle_pic(); // pictures blurry? use this trigger. Turns off the throttle until for # of cycles of medium loop then takes the picture and re-enables the throttle.
|
|
break;
|
|
case AP_CAMERA_TRIGGER_TYPE_WP_DISTANCE:
|
|
distance_pic(); // pictures blurry? use this trigger. Turns off the throttle until closer to waypoint then takes the picture and re-enables the throttle.
|
|
break;
|
|
case AP_CAMERA_TRIGGER_TYPE_TRANSISTOR:
|
|
transistor_pic(); // hacked the circuit to run a transistor? use this trigger to send output.
|
|
break;
|
|
}
|
|
}
|
|
|
|
/// de-activate the trigger after some delay, but without using a delay() function
|
|
/// should be called at 50hz
|
|
void
|
|
AP_Camera::trigger_pic_cleanup()
|
|
{
|
|
if (_trigger_counter) {
|
|
_trigger_counter--;
|
|
} else {
|
|
switch (_trigger_type) {
|
|
case AP_CAMERA_TRIGGER_TYPE_SERVO:
|
|
case AP_CAMERA_TRIGGER_TYPE_THROTTLE_OFF_TIME:
|
|
case AP_CAMERA_TRIGGER_TYPE_WP_DISTANCE:
|
|
RC_Channel_aux::set_radio(RC_Channel_aux::k_cam_trigger, _servo_off_pwm);
|
|
break;
|
|
case AP_CAMERA_TRIGGER_TYPE_RELAY:
|
|
_apm_relay->off();
|
|
break;
|
|
case AP_CAMERA_TRIGGER_TYPE_TRANSISTOR:
|
|
hal.gpio->write(AP_CAMERA_TRANSISTOR_PIN, 0);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// decode MavLink that configures camera
|
|
void
|
|
AP_Camera::configure_msg(mavlink_message_t* msg)
|
|
{
|
|
__mavlink_digicam_configure_t packet;
|
|
mavlink_msg_digicam_configure_decode(msg, &packet);
|
|
if (mavlink_check_target(packet.target_system, packet.target_component)) {
|
|
// not for us
|
|
return;
|
|
}
|
|
// This values may or not be used by APM
|
|
// They are bypassed as "echo" to a external specialized board
|
|
/*
|
|
* packet.aperture
|
|
* packet.command_id
|
|
* packet.engine_cut_off
|
|
* packet.exposure_type
|
|
* packet.extra_param
|
|
* packet.extra_value
|
|
* packet.iso
|
|
* packet.mode
|
|
* packet.shutter_speed
|
|
*/
|
|
// echo the message to the ArduCam OSD camera control board
|
|
// for more info see: http://code.google.com/p/arducam-osd/
|
|
// TODO is it connected to MAVLINK_COMM_3 ?
|
|
mavlink_msg_digicam_configure_send(MAVLINK_COMM_3, packet.target_system, packet.target_component, packet.mode, packet.shutter_speed, packet.aperture, packet.iso, packet.exposure_type, packet.command_id, packet.engine_cut_off, packet.extra_param, packet.extra_value);
|
|
}
|
|
|
|
/// decode MavLink that controls camera
|
|
void
|
|
AP_Camera::control_msg(mavlink_message_t* msg)
|
|
{
|
|
__mavlink_digicam_control_t packet;
|
|
mavlink_msg_digicam_control_decode(msg, &packet);
|
|
if (mavlink_check_target(packet.target_system, packet.target_component)) {
|
|
// not for us
|
|
return;
|
|
}
|
|
// This values may or not be used by APM (the shot is)
|
|
// They are bypassed as "echo" to a external specialized board
|
|
/*
|
|
* packet.command_id
|
|
* packet.extra_param
|
|
* packet.extra_value
|
|
* packet.focus_lock
|
|
* packet.session
|
|
* packet.shot
|
|
* packet.zoom_pos
|
|
* packet.zoom_step
|
|
*/
|
|
if (packet.shot)
|
|
{
|
|
trigger_pic();
|
|
}
|
|
// echo the message to the ArduCam OSD camera control board
|
|
// for more info see: http://code.google.com/p/arducam-osd/
|
|
// TODO is it connected to MAVLINK_COMM_3 ?
|
|
mavlink_msg_digicam_control_send(MAVLINK_COMM_3, packet.target_system, packet.target_component, packet.session, packet.zoom_pos, packet.zoom_step, packet.focus_lock, packet.shot, packet.command_id, packet.extra_param, packet.extra_value);
|
|
}
|
|
|
|
|
|
// update location, for triggering by GPS distance moved
|
|
void AP_Camera::update_location(const struct Location &loc)
|
|
{
|
|
if (_trigg_dist == 0.0f) {
|
|
return;
|
|
}
|
|
if (_last_location.lat == 0 && _last_location.lng == 0) {
|
|
_last_location = loc;
|
|
return;
|
|
}
|
|
if (_last_location.lat == loc.lat && _last_location.lng == loc.lng) {
|
|
// we haven't moved - this can happen as update_location() may
|
|
// be called without a new GPS fix
|
|
return;
|
|
}
|
|
if (get_distance(&loc, &_last_location) < _trigg_dist) {
|
|
return;
|
|
}
|
|
_last_location = loc;
|
|
trigger_pic();
|
|
}
|