blob: dda5b9c1db1f533e827867d8c4dbd1ebb6ec703a [file] [log] [blame] [edit]
#include <unistd.h>
#include <cmath>
#include <cstdio>
#include <cstring>
#include "aos/actions/actions.h"
#include "aos/init.h"
#include "aos/logging/logging.h"
#include "aos/network/team_number.h"
#include "aos/util/log_interval.h"
#include "frc971/autonomous/base_autonomous_actor.h"
#include "frc971/control_loops/drivetrain/localizer_generated.h"
#include "frc971/control_loops/profiled_subsystem_generated.h"
#include "frc971/input/action_joystick_input.h"
#include "frc971/input/driver_station_data.h"
#include "frc971/input/drivetrain_input.h"
#include "frc971/input/joystick_input.h"
#include "frc971/zeroing/wrap.h"
#include "y2022/constants.h"
#include "y2022/control_loops/drivetrain/drivetrain_base.h"
#include "y2022/control_loops/superstructure/superstructure_goal_generated.h"
#include "y2022/control_loops/superstructure/superstructure_status_generated.h"
#include "y2022/setpoint_generated.h"
using frc971::CreateProfileParameters;
using frc971::control_loops::CreateStaticZeroingSingleDOFProfiledSubsystemGoal;
using frc971::control_loops::StaticZeroingSingleDOFProfiledSubsystemGoal;
using frc971::control_loops::catapult::CatapultGoal;
using frc971::input::driver_station::ButtonLocation;
using frc971::input::driver_station::ControlBit;
using frc971::input::driver_station::JoystickAxis;
using frc971::input::driver_station::POVLocation;
namespace y2022::input::joysticks {
namespace superstructure = y2022::control_loops::superstructure;
#if 0
const ButtonLocation kCatapultPos(4, 3);
const ButtonLocation kFire(3, 4);
const ButtonLocation kTurret(4, 15);
const ButtonLocation kAutoAim(4, 2);
const ButtonLocation kIntakeFrontOut(4, 10);
const ButtonLocation kIntakeBackOut(4, 9);
const ButtonLocation kSpitFront(3, 3);
const ButtonLocation kSpitBack(2, 3);
const ButtonLocation kRedLocalizerReset(3, 13);
const ButtonLocation kBlueLocalizerReset(3, 14);
const ButtonLocation kLocalizerReset(3, 8);
#else
const ButtonLocation kCatapultPos(4, 3);
const ButtonLocation kFire(4, 1);
const ButtonLocation kTurret(4, 15);
const ButtonLocation kAutoAim(4, 16);
const ButtonLocation kClimberExtend(1, 2);
const ButtonLocation kClimberIntakes(4, 5);
const ButtonLocation kClimberServo(4, 4);
const ButtonLocation kIntakeFrontOut(4, 10);
const ButtonLocation kIntakeBackOut(4, 9);
const ButtonLocation kSpitFront(4, 8);
const ButtonLocation kSpitBack(4, 7);
const ButtonLocation kSpit(3, 3);
const ButtonLocation kRedLocalizerReset(4, 14);
const ButtonLocation kBlueLocalizerReset(4, 13);
const ButtonLocation kLocalizerReset(3, 8);
#endif
class Reader : public ::frc971::input::ActionJoystickInput {
public:
Reader(::aos::EventLoop *event_loop)
: ::frc971::input::ActionJoystickInput(
event_loop,
::y2022::control_loops::drivetrain::GetDrivetrainConfig(),
::frc971::input::DrivetrainInputReader::InputType::kPistol, {}),
superstructure_goal_sender_(
event_loop->MakeSender<superstructure::Goal>("/superstructure")),
localizer_control_sender_(
event_loop->MakeSender<
::frc971::control_loops::drivetrain::LocalizerControl>(
"/drivetrain")),
superstructure_status_fetcher_(
event_loop->MakeFetcher<superstructure::Status>("/superstructure")),
setpoint_fetcher_(
event_loop->MakeFetcher<Setpoint>("/superstructure")) {}
// Localizer reset positions are with front of robot pressed up against driver
// station in the middle of the field side-to-side.
void BlueResetLocalizer() {
auto builder = localizer_control_sender_.MakeBuilder();
frc971::control_loops::drivetrain::LocalizerControl::Builder
localizer_control_builder = builder.MakeBuilder<
frc971::control_loops::drivetrain::LocalizerControl>();
localizer_control_builder.add_x(-7.9);
localizer_control_builder.add_y(0.0);
localizer_control_builder.add_theta_uncertainty(10.0);
localizer_control_builder.add_theta(M_PI);
localizer_control_builder.add_keep_current_theta(false);
if (builder.Send(localizer_control_builder.Finish()) !=
aos::RawSender::Error::kOk) {
AOS_LOG(ERROR, "Failed to reset blue localizer.\n");
}
}
void RedResetLocalizer() {
auto builder = localizer_control_sender_.MakeBuilder();
frc971::control_loops::drivetrain::LocalizerControl::Builder
localizer_control_builder = builder.MakeBuilder<
frc971::control_loops::drivetrain::LocalizerControl>();
localizer_control_builder.add_x(7.9);
localizer_control_builder.add_y(0.0);
localizer_control_builder.add_theta_uncertainty(10.0);
localizer_control_builder.add_theta(0.0);
localizer_control_builder.add_keep_current_theta(false);
if (builder.Send(localizer_control_builder.Finish()) !=
aos::RawSender::Error::kOk) {
AOS_LOG(ERROR, "Failed to reset red localizer.\n");
}
}
void ResetLocalizer() {
const frc971::control_loops::drivetrain::Status *drivetrain_status =
this->drivetrain_status();
if (drivetrain_status == nullptr) {
return;
}
// Get the current position
// Snap to heading.
auto builder = localizer_control_sender_.MakeBuilder();
// TODO<Henry> Put our starting location here.
frc971::control_loops::drivetrain::LocalizerControl::Builder
localizer_control_builder = builder.MakeBuilder<
frc971::control_loops::drivetrain::LocalizerControl>();
localizer_control_builder.add_x(drivetrain_status->x());
localizer_control_builder.add_y(drivetrain_status->y());
const double new_theta =
frc971::zeroing::Wrap(drivetrain_status->theta(), 0, M_PI);
localizer_control_builder.add_theta(new_theta);
localizer_control_builder.add_theta_uncertainty(10.0);
if (builder.Send(localizer_control_builder.Finish()) !=
aos::RawSender::Error::kOk) {
AOS_LOG(ERROR, "Failed to reset localizer.\n");
}
}
void AutoEnded() override { AOS_LOG(INFO, "Auto ended.\n"); }
void HandleTeleop(
const ::frc971::input::driver_station::Data &data) override {
superstructure_status_fetcher_.Fetch();
if (!superstructure_status_fetcher_.get()) {
AOS_LOG(ERROR, "Got no superstructure status message.\n");
return;
}
setpoint_fetcher_.Fetch();
// Default to the intakes in
constexpr double kIntakeUpPosition = 1.47;
double intake_front_pos = kIntakeUpPosition;
double intake_back_pos = kIntakeUpPosition;
double transfer_roller_speed = 0.0;
std::optional<control_loops::superstructure::RequestedIntake>
requested_intake;
double roller_front_speed = 0.0;
double roller_back_speed = 0.0;
std::optional<double> turret_pos = std::nullopt;
double climber_position = 0.01;
bool climber_servo = false;
double catapult_pos = 0.03;
double catapult_speed = 18.0;
double catapult_return_pos = 0.0;
bool fire = false;
if (data.PosEdge(kLocalizerReset)) {
ResetLocalizer();
}
if (data.PosEdge(kRedLocalizerReset)) {
RedResetLocalizer();
}
if (data.PosEdge(kBlueLocalizerReset)) {
BlueResetLocalizer();
}
if (data.IsPressed(kClimberExtend)) {
climber_position = 0.54;
} else {
climber_position = 0.005;
}
if (data.IsPressed(kClimberServo)) {
climber_servo = true;
}
if (data.IsPressed(kTurret)) {
if (setpoint_fetcher_.get()) {
turret_pos = setpoint_fetcher_->turret();
} else {
turret_pos = -1.5;
}
}
if (setpoint_fetcher_.get()) {
catapult_pos = setpoint_fetcher_->catapult_position();
catapult_speed = setpoint_fetcher_->catapult_velocity();
}
// Keep the catapult return position at the shot one if kCatapultPos is
// pressed
if (data.IsPressed(kCatapultPos)) {
catapult_return_pos = 0.7;
} else {
catapult_return_pos = -0.908;
}
constexpr double kRollerSpeed = 12.0;
constexpr double kTransferRollerSpeed = 12.0;
constexpr double kIntakePosition = -0.14;
constexpr size_t kIntakeCounterIterations = 25;
if (data.PosEdge(kSpit)) {
last_front_intake_has_ball_ =
superstructure_status_fetcher_->front_intake_has_ball();
last_back_intake_has_ball_ =
superstructure_status_fetcher_->back_intake_has_ball();
}
// Extend the intakes and spin the rollers.
// Don't let this happen if there is a ball in the other intake, because
// that would spit this one out.
if (data.IsPressed(kIntakeFrontOut) &&
!superstructure_status_fetcher_->back_intake_has_ball()) {
intake_front_pos = kIntakePosition;
transfer_roller_speed = kTransferRollerSpeed;
intake_front_counter_ = kIntakeCounterIterations;
intake_back_counter_ = 0;
} else if (data.IsPressed(kIntakeBackOut) &&
!superstructure_status_fetcher_->front_intake_has_ball()) {
intake_back_pos = kIntakePosition;
transfer_roller_speed = -kTransferRollerSpeed;
intake_back_counter_ = kIntakeCounterIterations;
intake_front_counter_ = 0;
} else if (data.IsPressed(kSpitFront) ||
(data.IsPressed(kSpit) && last_front_intake_has_ball_)) {
transfer_roller_speed = -kTransferRollerSpeed;
intake_front_counter_ = 0;
} else if (data.IsPressed(kSpitBack) ||
(data.IsPressed(kSpit) && last_back_intake_has_ball_)) {
transfer_roller_speed = kTransferRollerSpeed;
intake_back_counter_ = 0;
}
// Keep spinning the rollers a bit after they let go
if (intake_front_counter_ > 0) {
intake_front_counter_--;
roller_front_speed = kRollerSpeed;
requested_intake = control_loops::superstructure::RequestedIntake::kFront;
}
if (intake_back_counter_ > 0) {
intake_back_counter_--;
roller_back_speed = kRollerSpeed;
requested_intake = control_loops::superstructure::RequestedIntake::kBack;
}
if (data.IsPressed(kFire)) {
fire = true;
// Provide a default turret goal.
}
if (data.IsPressed(kClimberIntakes)) {
intake_back_pos = kIntakeUpPosition;
intake_front_pos = kIntakePosition;
}
{
auto builder = superstructure_goal_sender_.MakeBuilder();
flatbuffers::Offset<StaticZeroingSingleDOFProfiledSubsystemGoal>
intake_front_offset =
CreateStaticZeroingSingleDOFProfiledSubsystemGoal(
*builder.fbb(), intake_front_pos,
CreateProfileParameters(*builder.fbb(), 8.0, 40.0));
flatbuffers::Offset<StaticZeroingSingleDOFProfiledSubsystemGoal>
intake_back_offset =
CreateStaticZeroingSingleDOFProfiledSubsystemGoal(
*builder.fbb(), intake_back_pos,
CreateProfileParameters(*builder.fbb(), 8.0, 40.0));
flatbuffers::Offset<StaticZeroingSingleDOFProfiledSubsystemGoal>
turret_offset;
if (turret_pos.has_value()) {
turret_offset = CreateStaticZeroingSingleDOFProfiledSubsystemGoal(
*builder.fbb(), turret_pos.value(),
CreateProfileParameters(*builder.fbb(), 10.0, 25.0));
}
flatbuffers::Offset<StaticZeroingSingleDOFProfiledSubsystemGoal>
catapult_return_offset =
CreateStaticZeroingSingleDOFProfiledSubsystemGoal(
*builder.fbb(), catapult_return_pos,
frc971::CreateProfileParameters(*builder.fbb(), 9.0, 50.0));
flatbuffers::Offset<StaticZeroingSingleDOFProfiledSubsystemGoal>
climber_offset = CreateStaticZeroingSingleDOFProfiledSubsystemGoal(
*builder.fbb(), climber_position,
frc971::CreateProfileParameters(*builder.fbb(), 1.0, 1.0));
CatapultGoal::Builder catapult_builder =
builder.MakeBuilder<CatapultGoal>();
catapult_builder.add_return_position(catapult_return_offset);
catapult_builder.add_shot_position(catapult_pos);
catapult_builder.add_shot_velocity(catapult_speed);
flatbuffers::Offset<CatapultGoal> catapult_offset =
catapult_builder.Finish();
superstructure::Goal::Builder superstructure_goal_builder =
builder.MakeBuilder<superstructure::Goal>();
superstructure_goal_builder.add_intake_front(intake_front_offset);
superstructure_goal_builder.add_intake_back(intake_back_offset);
if (!turret_offset.IsNull()) {
superstructure_goal_builder.add_turret(turret_offset);
}
superstructure_goal_builder.add_climber(climber_offset);
superstructure_goal_builder.add_catapult(catapult_offset);
superstructure_goal_builder.add_fire(fire);
superstructure_goal_builder.add_roller_speed_front(roller_front_speed);
superstructure_goal_builder.add_roller_speed_back(roller_back_speed);
superstructure_goal_builder.add_transfer_roller_speed(
transfer_roller_speed);
superstructure_goal_builder.add_climber_servo(climber_servo);
superstructure_goal_builder.add_auto_aim(data.IsPressed(kAutoAim));
if (requested_intake.has_value()) {
superstructure_goal_builder.add_turret_intake(requested_intake.value());
}
if (builder.Send(superstructure_goal_builder.Finish()) !=
aos::RawSender::Error::kOk) {
AOS_LOG(ERROR, "Sending superstructure goal failed.\n");
}
}
}
private:
::aos::Sender<superstructure::Goal> superstructure_goal_sender_;
::aos::Sender<frc971::control_loops::drivetrain::LocalizerControl>
localizer_control_sender_;
::aos::Fetcher<superstructure::Status> superstructure_status_fetcher_;
::aos::Fetcher<Setpoint> setpoint_fetcher_;
size_t intake_front_counter_ = 0;
size_t intake_back_counter_ = 0;
bool last_front_intake_has_ball_ = false;
bool last_back_intake_has_ball_ = false;
};
} // namespace y2022::input::joysticks
int main(int argc, char **argv) {
::aos::InitGoogle(&argc, &argv);
aos::FlatbufferDetachedBuffer<aos::Configuration> config =
aos::configuration::ReadConfig("aos_config.json");
::aos::ShmEventLoop event_loop(&config.message());
::y2022::input::joysticks::Reader reader(&event_loop);
event_loop.Run();
return 0;
}