/**************************************************************************** * * Copyright (c) 2016-2017 PX4 Development Team. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name PX4 nor the names of its contributors may be * used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * ****************************************************************************/ /** * @file input_mavlink.cpp * @author Leon Müller (thedevleon) * @author Beat Küng * */ #include "input_mavlink.h" #include #include #include #include #include #include #include #include #include #include namespace vmount { InputMavlinkROI::~InputMavlinkROI() { if (_vehicle_roi_sub >= 0) { orb_unsubscribe(_vehicle_roi_sub); } if (_position_setpoint_triplet_sub >= 0) { orb_unsubscribe(_position_setpoint_triplet_sub); } } int InputMavlinkROI::initialize() { _vehicle_roi_sub = orb_subscribe(ORB_ID(vehicle_roi)); if (_vehicle_roi_sub < 0) { return -errno; } _position_setpoint_triplet_sub = orb_subscribe(ORB_ID(position_setpoint_triplet)); if (_position_setpoint_triplet_sub < 0) { return -errno; } return 0; } int InputMavlinkROI::update_impl(unsigned int timeout_ms, ControlData **control_data, bool already_active) { // already_active is unused, we don't care what happened previously. // Default to no change, set if we receive anything. *control_data = nullptr; const int num_poll = 2; px4_pollfd_struct_t polls[num_poll]; polls[0].fd = _vehicle_roi_sub; polls[0].events = POLLIN; polls[1].fd = _position_setpoint_triplet_sub; polls[1].events = POLLIN; int ret = px4_poll(polls, num_poll, timeout_ms); if (ret < 0) { return -errno; } if (ret == 0) { // Timeout, _control_data is already null } else { if (polls[0].revents & POLLIN) { vehicle_roi_s vehicle_roi; orb_copy(ORB_ID(vehicle_roi), _vehicle_roi_sub, &vehicle_roi); _control_data.gimbal_shutter_retract = false; if (vehicle_roi.mode == vehicle_roi_s::ROI_NONE) { _control_data.type = ControlData::Type::Neutral; *control_data = &_control_data; } else if (vehicle_roi.mode == vehicle_roi_s::ROI_WPNEXT) { _control_data.type = ControlData::Type::LonLat; _read_control_data_from_position_setpoint_sub(); _control_data.type_data.lonlat.pitch_fixed_angle = -10.f; _control_data.type_data.lonlat.roll_angle = vehicle_roi.roll_offset; _control_data.type_data.lonlat.pitch_angle_offset = vehicle_roi.pitch_offset; _control_data.type_data.lonlat.yaw_angle_offset = vehicle_roi.yaw_offset; *control_data = &_control_data; } else if (vehicle_roi.mode == vehicle_roi_s::ROI_LOCATION) { control_data_set_lon_lat(vehicle_roi.lon, vehicle_roi.lat, vehicle_roi.alt); *control_data = &_control_data; } else if (vehicle_roi.mode == vehicle_roi_s::ROI_TARGET) { //TODO is this even suported? } _cur_roi_mode = vehicle_roi.mode; //set all other control data fields to defaults for (int i = 0; i < 3; ++i) { _control_data.stabilize_axis[i] = false; } } // check whether the position setpoint got updated if (polls[1].revents & POLLIN) { if (_cur_roi_mode == vehicle_roi_s::ROI_WPNEXT) { _read_control_data_from_position_setpoint_sub(); *control_data = &_control_data; } else { // must do an orb_copy() in *every* case position_setpoint_triplet_s position_setpoint_triplet; orb_copy(ORB_ID(position_setpoint_triplet), _position_setpoint_triplet_sub, &position_setpoint_triplet); } } } return 0; } void InputMavlinkROI::_read_control_data_from_position_setpoint_sub() { position_setpoint_triplet_s position_setpoint_triplet; orb_copy(ORB_ID(position_setpoint_triplet), _position_setpoint_triplet_sub, &position_setpoint_triplet); _control_data.type_data.lonlat.lon = position_setpoint_triplet.current.lon; _control_data.type_data.lonlat.lat = position_setpoint_triplet.current.lat; _control_data.type_data.lonlat.altitude = position_setpoint_triplet.current.alt; } void InputMavlinkROI::print_status() { PX4_INFO("Input: Mavlink (ROI)"); } InputMavlinkCmdMount::InputMavlinkCmdMount(bool stabilize) : _stabilize {stabilize, stabilize, stabilize} { param_t handle = param_find("MAV_SYS_ID"); if (handle != PARAM_INVALID) { param_get(handle, &_mav_sys_id); } handle = param_find("MAV_COMP_ID"); if (handle != PARAM_INVALID) { param_get(handle, &_mav_comp_id); } } InputMavlinkCmdMount::~InputMavlinkCmdMount() { if (_vehicle_command_sub >= 0) { orb_unsubscribe(_vehicle_command_sub); } } int InputMavlinkCmdMount::initialize() { if ((_vehicle_command_sub = orb_subscribe(ORB_ID(vehicle_command))) < 0) { return -errno; } // rate-limit inputs to 100Hz. If we don't do this and the output is configured to mavlink mode, // it will publish vehicle_command's as well, causing the input poll() in here to return // immediately, which in turn will cause an output update and thus a busy loop. orb_set_interval(_vehicle_command_sub, 10); return 0; } int InputMavlinkCmdMount::update_impl(unsigned int timeout_ms, ControlData **control_data, bool already_active) { // Default to notify that there was no change. *control_data = nullptr; const int num_poll = 1; px4_pollfd_struct_t polls[num_poll]; polls[0].fd = _vehicle_command_sub; polls[0].events = POLLIN; int poll_timeout = (int)timeout_ms; bool exit_loop = false; while (!exit_loop && poll_timeout >= 0) { hrt_abstime poll_start = hrt_absolute_time(); int ret = px4_poll(polls, num_poll, poll_timeout); if (ret < 0) { return -errno; } poll_timeout -= (hrt_absolute_time() - poll_start) / 1000; // if we get a command that we need to handle, we exit the loop, otherwise we poll until we reach the timeout exit_loop = true; if (ret == 0) { // Timeout control_data already null. } else { if (polls[0].revents & POLLIN) { vehicle_command_s vehicle_command; orb_copy(ORB_ID(vehicle_command), _vehicle_command_sub, &vehicle_command); // Process only if the command is for us or for anyone (component id 0). const bool sysid_correct = (vehicle_command.target_system == _mav_sys_id); const bool compid_correct = ((vehicle_command.target_component == _mav_comp_id) || (vehicle_command.target_component == 0)); if (!sysid_correct || !compid_correct) { exit_loop = false; continue; } for (int i = 0; i < 3; ++i) { _control_data.stabilize_axis[i] = _stabilize[i]; } _control_data.gimbal_shutter_retract = false; if (vehicle_command.command == vehicle_command_s::VEHICLE_CMD_DO_MOUNT_CONTROL) { switch ((int)vehicle_command.param7) { case vehicle_command_s::VEHICLE_MOUNT_MODE_RETRACT: _control_data.gimbal_shutter_retract = true; /* FALLTHROUGH */ case vehicle_command_s::VEHICLE_MOUNT_MODE_NEUTRAL: _control_data.type = ControlData::Type::Neutral; *control_data = &_control_data; break; case vehicle_command_s::VEHICLE_MOUNT_MODE_MAVLINK_TARGETING: _control_data.type = ControlData::Type::Angle; _control_data.type_data.angle.frames[0] = ControlData::TypeData::TypeAngle::Frame::AngleBodyFrame; _control_data.type_data.angle.frames[1] = ControlData::TypeData::TypeAngle::Frame::AngleBodyFrame; _control_data.type_data.angle.frames[2] = ControlData::TypeData::TypeAngle::Frame::AngleBodyFrame; // vmount spec has roll on channel 0, MAVLink spec has pitch on channel 0 _control_data.type_data.angle.angles[0] = vehicle_command.param2 * M_DEG_TO_RAD_F; // vmount spec has pitch on channel 1, MAVLink spec has roll on channel 1 _control_data.type_data.angle.angles[1] = vehicle_command.param1 * M_DEG_TO_RAD_F; // both specs have yaw on channel 2 _control_data.type_data.angle.angles[2] = vehicle_command.param3 * M_DEG_TO_RAD_F; // We expect angle of [-pi..+pi]. If the input range is [0..2pi] we can fix that. if (_control_data.type_data.angle.angles[2] > M_PI_F) { _control_data.type_data.angle.angles[2] -= 2 * M_PI_F; } *control_data = &_control_data; break; case vehicle_command_s::VEHICLE_MOUNT_MODE_RC_TARGETING: break; case vehicle_command_s::VEHICLE_MOUNT_MODE_GPS_POINT: control_data_set_lon_lat((double)vehicle_command.param6, (double)vehicle_command.param5, vehicle_command.param4); *control_data = &_control_data; break; } _ack_vehicle_command(&vehicle_command); } else if (vehicle_command.command == vehicle_command_s::VEHICLE_CMD_DO_MOUNT_CONFIGURE) { _stabilize[0] = (int)(vehicle_command.param2 + 0.5f) == 1; _stabilize[1] = (int)(vehicle_command.param3 + 0.5f) == 1; _stabilize[2] = (int)(vehicle_command.param4 + 0.5f) == 1; const int params[] = { (int)((float)vehicle_command.param5 + 0.5f), (int)((float)vehicle_command.param6 + 0.5f), (int)(vehicle_command.param7 + 0.5f) }; for (int i = 0; i < 3; ++i) { if (params[i] == 0) { _control_data.type_data.angle.frames[i] = ControlData::TypeData::TypeAngle::Frame::AngleBodyFrame; } else if (params[i] == 1) { _control_data.type_data.angle.frames[i] = ControlData::TypeData::TypeAngle::Frame::AngularRate; } else if (params[i] == 2) { _control_data.type_data.angle.frames[i] = ControlData::TypeData::TypeAngle::Frame::AngleAbsoluteFrame; } else { // Not supported, fallback to body angle. _control_data.type_data.angle.frames[i] = ControlData::TypeData::TypeAngle::Frame::AngleBodyFrame; } } _control_data.type = ControlData::Type::Neutral; //always switch to neutral position *control_data = &_control_data; _ack_vehicle_command(&vehicle_command); } else { exit_loop = false; } } } } return 0; } void InputMavlinkCmdMount::_ack_vehicle_command(vehicle_command_s *cmd) { vehicle_command_ack_s vehicle_command_ack{}; vehicle_command_ack.timestamp = hrt_absolute_time(); vehicle_command_ack.command = cmd->command; vehicle_command_ack.result = vehicle_command_s::VEHICLE_CMD_RESULT_ACCEPTED; vehicle_command_ack.target_system = cmd->source_system; vehicle_command_ack.target_component = cmd->source_component; uORB::Publication cmd_ack_pub{ORB_ID(vehicle_command_ack)}; cmd_ack_pub.publish(vehicle_command_ack); } void InputMavlinkCmdMount::print_status() { PX4_INFO("Input: Mavlink (CMD_MOUNT)"); } } /* namespace vmount */