blob: 852b94e8bee45e9a7adfea91dbcdda403a74b4b2 [file] [log] [blame]
#include "aos/input/drivetrain_input.h"
#include <math.h>
#include <stdio.h>
#include <string.h>
#include <cmath>
#include "aos/common/commonmath.h"
#include "aos/common/input/driver_station_data.h"
#include "aos/common/logging/logging.h"
#include "frc971/control_loops/drivetrain/drivetrain.q.h"
using ::frc971::control_loops::drivetrain_queue;
using ::aos::input::driver_station::ButtonLocation;
using ::aos::input::driver_station::ControlBit;
using ::aos::input::driver_station::JoystickAxis;
using ::aos::input::driver_station::POVLocation;
namespace aos {
namespace input {
const ButtonLocation kShiftHigh(2, 3), kShiftHigh2(2, 2), kShiftLow(2, 1);
void DrivetrainInputReader::HandleDrivetrain(
const ::aos::input::driver_station::Data &data) {
bool is_control_loop_driving = false;
const auto wheel_and_throttle = GetWheelAndThrottle(data);
const double wheel = wheel_and_throttle.wheel;
const double wheel_velocity = wheel_and_throttle.wheel_velocity;
const double wheel_torque = wheel_and_throttle.wheel_torque;
const double throttle = wheel_and_throttle.throttle;
const double throttle_velocity = wheel_and_throttle.throttle_velocity;
const double throttle_torque = wheel_and_throttle.throttle_torque;
const bool high_gear = wheel_and_throttle.high_gear;
drivetrain_queue.status.FetchLatest();
if (drivetrain_queue.status.get()) {
robot_velocity_ = drivetrain_queue.status->robot_speed;
}
if (data.PosEdge(turn1_) || data.PosEdge(turn2_)) {
if (drivetrain_queue.status.get()) {
left_goal_ = drivetrain_queue.status->estimated_left_position;
right_goal_ = drivetrain_queue.status->estimated_right_position;
}
}
const double current_left_goal =
left_goal_ - wheel * wheel_multiplier_ + throttle * 0.3;
const double current_right_goal =
right_goal_ + wheel * wheel_multiplier_ + throttle * 0.3;
if (data.IsPressed(turn1_) || data.IsPressed(turn2_)) {
is_control_loop_driving = true;
}
auto new_drivetrain_goal = drivetrain_queue.goal.MakeMessage();
new_drivetrain_goal->wheel = wheel;
new_drivetrain_goal->wheel_velocity = wheel_velocity;
new_drivetrain_goal->wheel_torque = wheel_torque;
new_drivetrain_goal->throttle = throttle;
new_drivetrain_goal->throttle_velocity = throttle_velocity;
new_drivetrain_goal->throttle_torque = throttle_torque;
new_drivetrain_goal->highgear = high_gear;
new_drivetrain_goal->quickturn = data.IsPressed(quick_turn_);
new_drivetrain_goal->control_loop_driving = is_control_loop_driving;
new_drivetrain_goal->left_goal = current_left_goal;
new_drivetrain_goal->right_goal = current_right_goal;
new_drivetrain_goal->left_velocity_goal = 0;
new_drivetrain_goal->right_velocity_goal = 0;
new_drivetrain_goal->linear.max_velocity = 3.0;
new_drivetrain_goal->linear.max_acceleration = 20.0;
if (!new_drivetrain_goal.Send()) {
LOG(WARNING, "sending stick values failed\n");
}
}
DrivetrainInputReader::WheelAndThrottle
SteeringWheelDrivetrainInputReader::GetWheelAndThrottle(
const ::aos::input::driver_station::Data &data) {
const double wheel = -data.GetAxis(wheel_);
const double throttle = -data.GetAxis(throttle_);
if (!data.GetControlBit(ControlBit::kEnabled)) {
high_gear_ = default_high_gear_;
}
if (data.PosEdge(kShiftLow)) {
high_gear_ = false;
}
if (data.PosEdge(kShiftHigh) || data.PosEdge(kShiftHigh2)) {
high_gear_ = true;
}
return DrivetrainInputReader::WheelAndThrottle{
wheel, 0.0, 0.0, throttle, 0.0, 0.0, high_gear_};
}
double UnwrappedAxis(const ::aos::input::driver_station::Data &data,
const JoystickAxis &high_bits,
const JoystickAxis &low_bits) {
const float high_bits_data = data.GetAxis(high_bits);
const float low_bits_data = data.GetAxis(low_bits);
const int16_t high_bits_data_int =
(high_bits_data < 0.0f ? high_bits_data * 128.0f
: high_bits_data * 127.0f);
const int16_t low_bits_data_int =
(low_bits_data < 0.0f ? low_bits_data * 128.0f : low_bits_data * 127.0f);
const uint16_t high_bits_data_uint =
((static_cast<uint16_t>(high_bits_data_int) & 0xff) + 0x80) & 0xff;
const uint16_t low_bits_data_uint =
((static_cast<uint16_t>(low_bits_data_int) & 0xff) + 0x80) & 0xff;
const uint16_t data_uint = (high_bits_data_uint << 8) | low_bits_data_uint;
const int32_t data_int = static_cast<int32_t>(data_uint) - 0x8000;
if (data_int < 0) {
return static_cast<double>(data_int) / static_cast<double>(0x8000);
} else {
return static_cast<double>(data_int) / static_cast<double>(0x7fff);
}
}
DrivetrainInputReader::WheelAndThrottle
PistolDrivetrainInputReader::GetWheelAndThrottle(
const ::aos::input::driver_station::Data &data) {
const double wheel =
-UnwrappedAxis(data, wheel_, wheel_low_);
const double wheel_velocity =
-UnwrappedAxis(data, wheel_velocity_high_, wheel_velocity_low_) * 50.0;
const double wheel_torque =
-UnwrappedAxis(data, wheel_torque_high_, wheel_torque_low_) / 2.0;
double throttle =
UnwrappedAxis(data, throttle_, throttle_low_);
const double throttle_velocity =
UnwrappedAxis(data, throttle_velocity_high_, throttle_velocity_low_) * 50.0;
const double throttle_torque =
UnwrappedAxis(data, throttle_torque_high_, throttle_torque_low_) / 2.0;
// TODO(austin): Deal with haptics here.
if (throttle < 0) {
throttle = ::std::max(-1.0, throttle / 0.7);
}
if (!data.GetControlBit(ControlBit::kEnabled)) {
high_gear_ = default_high_gear_;
}
if (data.PosEdge(shift_low_)) {
high_gear_ = false;
}
if (data.PosEdge(shift_high_)) {
high_gear_ = true;
}
return DrivetrainInputReader::WheelAndThrottle{
wheel, wheel_velocity, wheel_torque,
throttle, throttle_velocity, throttle_torque,
high_gear_};
}
DrivetrainInputReader::WheelAndThrottle
XboxDrivetrainInputReader::GetWheelAndThrottle(
const ::aos::input::driver_station::Data &data) {
// xbox
constexpr double kWheelDeadband = 0.05;
constexpr double kThrottleDeadband = 0.05;
const double wheel =
aos::Deadband(-data.GetAxis(wheel_), kWheelDeadband, 1.0);
const double unmodified_throttle =
aos::Deadband(-data.GetAxis(throttle_), kThrottleDeadband, 1.0);
// Apply a sin function that's scaled to make it feel better.
constexpr double throttle_range = M_PI_2 * 0.9;
double throttle = ::std::sin(throttle_range * unmodified_throttle) /
::std::sin(throttle_range);
throttle = ::std::sin(throttle_range * throttle) / ::std::sin(throttle_range);
throttle = 2.0 * unmodified_throttle - throttle;
return DrivetrainInputReader::WheelAndThrottle{wheel, 0.0, 0.0, throttle,
0.0, 0.0, true};
}
std::unique_ptr<SteeringWheelDrivetrainInputReader>
SteeringWheelDrivetrainInputReader::Make(bool default_high_gear) {
const JoystickAxis kSteeringWheel(1, 1), kDriveThrottle(2, 2);
const ButtonLocation kQuickTurn(1, 5);
const ButtonLocation kTurn1(1, 7);
const ButtonLocation kTurn2(1, 11);
std::unique_ptr<SteeringWheelDrivetrainInputReader> result(
new SteeringWheelDrivetrainInputReader(kSteeringWheel, kDriveThrottle,
kQuickTurn, kTurn1, kTurn2));
result.get()->set_default_high_gear(default_high_gear);
return result;
}
std::unique_ptr<PistolDrivetrainInputReader> PistolDrivetrainInputReader::Make(
bool default_high_gear) {
// Pistol Grip controller
const JoystickAxis kTriggerHigh(1, 1), kTriggerLow(1, 4),
kTriggerVelocityHigh(1, 2), kTriggerVelocityLow(1, 5),
kTriggerTorqueHigh(1, 3), kTriggerTorqueLow(1, 6);
const JoystickAxis kWheelHigh(2, 1), kWheelLow(2, 4),
kWheelVelocityHigh(2, 2), kWheelVelocityLow(2, 5), kWheelTorqueHigh(2, 3),
kWheelTorqueLow(2, 6);
const ButtonLocation kQuickTurn(1, 3);
const ButtonLocation kShiftHigh(1, 1);
const ButtonLocation kShiftLow(1, 2);
// Nop
const ButtonLocation kTurn1(1, 9);
const ButtonLocation kTurn2(1, 10);
std::unique_ptr<PistolDrivetrainInputReader> result(
new PistolDrivetrainInputReader(
kWheelHigh, kWheelLow, kTriggerVelocityHigh, kTriggerVelocityLow,
kTriggerTorqueHigh, kTriggerTorqueLow, kTriggerHigh, kTriggerLow,
kWheelVelocityHigh, kWheelVelocityLow, kWheelTorqueHigh,
kWheelTorqueLow, kQuickTurn, kShiftHigh, kShiftLow, kTurn1, kTurn2));
result->set_default_high_gear(default_high_gear);
return result;
}
std::unique_ptr<XboxDrivetrainInputReader> XboxDrivetrainInputReader::Make() {
// xbox
const JoystickAxis kSteeringWheel(1, 5), kDriveThrottle(1, 2);
const ButtonLocation kQuickTurn(1, 5);
// Nop
const ButtonLocation kTurn1(1, 1);
const ButtonLocation kTurn2(1, 2);
std::unique_ptr<XboxDrivetrainInputReader> result(
new XboxDrivetrainInputReader(kSteeringWheel, kDriveThrottle, kQuickTurn,
kTurn1, kTurn2));
return result;
}
::std::unique_ptr<DrivetrainInputReader> DrivetrainInputReader::Make(
InputType type,
const ::frc971::control_loops::drivetrain::DrivetrainConfig<double>
&dt_config) {
std::unique_ptr<DrivetrainInputReader> drivetrain_input_reader;
using InputType = DrivetrainInputReader::InputType;
switch (type) {
case InputType::kSteeringWheel:
drivetrain_input_reader =
SteeringWheelDrivetrainInputReader::Make(dt_config.default_high_gear);
break;
case InputType::kPistol:
drivetrain_input_reader =
PistolDrivetrainInputReader::Make(dt_config.default_high_gear);
break;
case InputType::kXbox:
drivetrain_input_reader = XboxDrivetrainInputReader::Make();
break;
}
return drivetrain_input_reader;
}
} // namespace input
} // namespace aos