blob: b6d432f4fb0adaa68fe0b87a8e1cb2efce8c5b6e [file] [log] [blame]
Sabina Davisabeae332019-02-01 21:12:57 -08001#include <inttypes.h>
2#include <stdio.h>
3#include <string.h>
4#include <unistd.h>
5
6#include <array>
7#include <chrono>
8#include <cmath>
9#include <functional>
10#include <mutex>
11#include <thread>
12
13#include "frc971/wpilib/ahal/AnalogInput.h"
14#include "frc971/wpilib/ahal/Counter.h"
15#include "frc971/wpilib/ahal/DigitalGlitchFilter.h"
16#include "frc971/wpilib/ahal/DriverStation.h"
17#include "frc971/wpilib/ahal/Encoder.h"
18#include "frc971/wpilib/ahal/VictorSP.h"
Sabina Davisc6329342019-03-01 20:44:42 -080019#include "ctre/phoenix/CANifier.h"
Sabina Davisabeae332019-02-01 21:12:57 -080020#undef ERROR
21
22#include "aos/commonmath.h"
Alex Perrycb7da4b2019-08-28 19:35:56 -070023#include "aos/events/event_loop.h"
24#include "aos/events/shm_event_loop.h"
Sabina Davisabeae332019-02-01 21:12:57 -080025#include "aos/init.h"
26#include "aos/logging/logging.h"
Sabina Davisabeae332019-02-01 21:12:57 -080027#include "aos/make_unique.h"
Alex Perrycb7da4b2019-08-28 19:35:56 -070028#include "aos/realtime.h"
29#include "aos/robot_state/robot_state_generated.h"
Sabina Davisabeae332019-02-01 21:12:57 -080030#include "aos/time/time.h"
Sabina Davisabeae332019-02-01 21:12:57 -080031#include "aos/util/log_interval.h"
32#include "aos/util/phased_loop.h"
33#include "aos/util/wrapping_counter.h"
Brian Silvermanc41fb862019-03-02 21:14:46 -080034#include "ctre/phoenix/motorcontrol/can/TalonSRX.h"
Austin Schuhed5b26d2019-12-05 20:51:59 -080035#include "frc971/autonomous/auto_mode_generated.h"
Alex Perrycb7da4b2019-08-28 19:35:56 -070036#include "frc971/control_loops/drivetrain/drivetrain_position_generated.h"
Sabina Davisabeae332019-02-01 21:12:57 -080037#include "frc971/wpilib/ADIS16448.h"
Austin Schuhc1d6f832019-02-15 23:22:17 -080038#include "frc971/wpilib/buffered_pcm.h"
39#include "frc971/wpilib/buffered_solenoid.h"
Sabina Davisabeae332019-02-01 21:12:57 -080040#include "frc971/wpilib/dma.h"
Sabina Davisd004fd62019-02-02 23:51:46 -080041#include "frc971/wpilib/drivetrain_writer.h"
Sabina Davisabeae332019-02-01 21:12:57 -080042#include "frc971/wpilib/encoder_and_potentiometer.h"
Sabina Davisabeae332019-02-01 21:12:57 -080043#include "frc971/wpilib/joystick_sender.h"
Alex Perrycb7da4b2019-08-28 19:35:56 -070044#include "frc971/wpilib/logging_generated.h"
Sabina Davisabeae332019-02-01 21:12:57 -080045#include "frc971/wpilib/loop_output_handler.h"
46#include "frc971/wpilib/pdp_fetcher.h"
Sabina Davisadc58542019-02-01 22:23:00 -080047#include "frc971/wpilib/sensor_reader.h"
Sabina Davisabeae332019-02-01 21:12:57 -080048#include "frc971/wpilib/wpilib_robot_base.h"
Austin Schuhed5b26d2019-12-05 20:51:59 -080049#include "y2019/camera_log_generated.h"
Sabina Davis7be49f32019-02-02 00:30:19 -080050#include "y2019/constants.h"
Alex Perrycb7da4b2019-08-28 19:35:56 -070051#include "y2019/control_loops/drivetrain/camera_generated.h"
52#include "y2019/control_loops/superstructure/superstructure_output_generated.h"
53#include "y2019/control_loops/superstructure/superstructure_position_generated.h"
Brian Silvermanf8b75252019-02-24 16:13:58 -080054#include "y2019/jevois/spi.h"
Alex Perrycb7da4b2019-08-28 19:35:56 -070055#include "y2019/status_light_generated.h"
Sabina Davisabeae332019-02-01 21:12:57 -080056
57#ifndef M_PI
58#define M_PI 3.14159265358979323846
59#endif
60
Sabina Davis7be49f32019-02-02 00:30:19 -080061using ::y2019::constants::Values;
Sabina Davisabeae332019-02-01 21:12:57 -080062using ::aos::monotonic_clock;
Alex Perrycb7da4b2019-08-28 19:35:56 -070063namespace superstructure = ::y2019::control_loops::superstructure;
Sabina Davisabeae332019-02-01 21:12:57 -080064namespace chrono = ::std::chrono;
65using aos::make_unique;
66
67namespace y2019 {
68namespace wpilib {
69namespace {
70
71constexpr double kMaxBringupPower = 12.0;
72
73// TODO(Brian): Fix the interpretation of the result of GetRaw here and in the
74// DMA stuff and then removing the * 2.0 in *_translate.
75// The low bit is direction.
76
77// TODO(brian): Use ::std::max instead once we have C++14 so that can be
78// constexpr.
79template <typename T>
80constexpr T max(T a, T b) {
81 return (a > b) ? a : b;
82}
83
84template <typename T, typename... Rest>
85constexpr T max(T a, T b, T c, Rest... rest) {
86 return max(max(a, b), c, rest...);
87}
88
89double drivetrain_translate(int32_t in) {
Sabina Davis7be49f32019-02-02 00:30:19 -080090 return ((static_cast<double>(in) /
91 Values::kDrivetrainEncoderCountsPerRevolution()) *
Sabina Davisabeae332019-02-01 21:12:57 -080092 (2.0 * M_PI)) *
93 Values::kDrivetrainEncoderRatio() *
Sabina Davis7be49f32019-02-02 00:30:19 -080094 control_loops::drivetrain::kWheelRadius;
Sabina Davisabeae332019-02-01 21:12:57 -080095}
96
97double drivetrain_velocity_translate(double in) {
Sabina Davis7be49f32019-02-02 00:30:19 -080098 return (((1.0 / in) / Values::kDrivetrainCyclesPerRevolution()) *
Sabina Davisabeae332019-02-01 21:12:57 -080099 (2.0 * M_PI)) *
100 Values::kDrivetrainEncoderRatio() *
Sabina Davis7be49f32019-02-02 00:30:19 -0800101 control_loops::drivetrain::kWheelRadius;
Sabina Davisabeae332019-02-01 21:12:57 -0800102}
103
Alex Perry5fb5ff22019-02-09 21:53:17 -0800104double elevator_pot_translate(double voltage) {
105 return voltage * Values::kElevatorPotRatio() *
Austin Schuhed7f8632019-02-15 23:12:20 -0800106 (10.0 /*turns*/ / 5.0 /*volts*/) * (2 * M_PI /*radians*/);
Alex Perry5fb5ff22019-02-09 21:53:17 -0800107}
108
109double wrist_pot_translate(double voltage) {
Austin Schuhed7f8632019-02-15 23:12:20 -0800110 return voltage * Values::kWristPotRatio() * (5.0 /*turns*/ / 5.0 /*volts*/) *
Alex Perry5fb5ff22019-02-09 21:53:17 -0800111 (2 * M_PI /*radians*/);
112}
113
114double stilts_pot_translate(double voltage) {
115 return voltage * Values::kStiltsPotRatio() *
116 (10.0 /*turns*/ / 5.0 /*volts*/) * (2 * M_PI /*radians*/);
117}
118
Sabina Davisabeae332019-02-01 21:12:57 -0800119constexpr double kMaxFastEncoderPulsesPerSecond =
Alex Perry5fb5ff22019-02-09 21:53:17 -0800120 max(Values::kMaxDrivetrainEncoderPulsesPerSecond(),
121 Values::kMaxIntakeEncoderPulsesPerSecond());
Sabina Davisabeae332019-02-01 21:12:57 -0800122static_assert(kMaxFastEncoderPulsesPerSecond <= 1300000,
123 "fast encoders are too fast");
Sabina Davisabeae332019-02-01 21:12:57 -0800124constexpr double kMaxMediumEncoderPulsesPerSecond =
Alex Perry5fb5ff22019-02-09 21:53:17 -0800125 max(Values::kMaxElevatorEncoderPulsesPerSecond(),
126 Values::kMaxWristEncoderPulsesPerSecond());
Theo Bafrali00e42272019-02-12 01:07:46 -0800127
Sabina Davisabeae332019-02-01 21:12:57 -0800128static_assert(kMaxMediumEncoderPulsesPerSecond <= 400000,
129 "medium encoders are too fast");
130
131// Class to send position messages with sensor readings to our loops.
Sabina Davisadc58542019-02-01 22:23:00 -0800132class SensorReader : public ::frc971::wpilib::SensorReader {
Sabina Davisabeae332019-02-01 21:12:57 -0800133 public:
Austin Schuh217a9782019-12-21 23:02:50 -0800134 SensorReader(::aos::ShmEventLoop *event_loop)
Austin Schuha250b2d2019-05-27 16:14:02 -0700135 : ::frc971::wpilib::SensorReader(event_loop),
136 auto_mode_sender_(
137 event_loop->MakeSender<::frc971::autonomous::AutonomousMode>(
Austin Schuhed5b26d2019-12-05 20:51:59 -0800138 "/autonomous")),
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700139 superstructure_position_sender_(
Alex Perrycb7da4b2019-08-28 19:35:56 -0700140 event_loop->MakeSender<superstructure::Position>(
141 "/superstructure")),
Austin Schuhbd0a40f2019-06-30 14:56:31 -0700142 drivetrain_position_sender_(
Alex Perrycb7da4b2019-08-28 19:35:56 -0700143 event_loop
144 ->MakeSender<::frc971::control_loops::drivetrain::Position>(
145 "/drivetrain")) {
Sabina Davisabeae332019-02-01 21:12:57 -0800146 // Set to filter out anything shorter than 1/4 of the minimum pulse width
147 // we should ever see.
Austin Schuh45a549f2019-02-02 15:43:56 -0800148 UpdateFastEncoderFilterHz(kMaxFastEncoderPulsesPerSecond);
149 UpdateMediumEncoderFilterHz(kMaxMediumEncoderPulsesPerSecond);
Sabina Davisabeae332019-02-01 21:12:57 -0800150 }
151
Alex Perry5fb5ff22019-02-09 21:53:17 -0800152 // Elevator
153
154 void set_elevator_encoder(::std::unique_ptr<frc::Encoder> encoder) {
155 medium_encoder_filter_.Add(encoder.get());
156 elevator_encoder_.set_encoder(::std::move(encoder));
157 }
158
159 void set_elevator_absolute_pwm(
160 ::std::unique_ptr<frc::DigitalInput> absolute_pwm) {
161 elevator_encoder_.set_absolute_pwm(::std::move(absolute_pwm));
162 }
163
164 void set_elevator_potentiometer(
165 ::std::unique_ptr<frc::AnalogInput> potentiometer) {
166 elevator_encoder_.set_potentiometer(::std::move(potentiometer));
167 }
168
169 // Intake
170
171 void set_intake_encoder(::std::unique_ptr<frc::Encoder> encoder) {
172 medium_encoder_filter_.Add(encoder.get());
173 intake_encoder_.set_encoder(::std::move(encoder));
174 }
175
176 void set_intake_absolute_pwm(
177 ::std::unique_ptr<frc::DigitalInput> absolute_pwm) {
178 intake_encoder_.set_absolute_pwm(::std::move(absolute_pwm));
179 }
180
181 // Wrist
182
183 void set_wrist_encoder(::std::unique_ptr<frc::Encoder> encoder) {
184 medium_encoder_filter_.Add(encoder.get());
185 wrist_encoder_.set_encoder(::std::move(encoder));
186 }
187
188 void set_wrist_absolute_pwm(
189 ::std::unique_ptr<frc::DigitalInput> absolute_pwm) {
190 wrist_encoder_.set_absolute_pwm(::std::move(absolute_pwm));
191 }
192
193 void set_wrist_potentiometer(
194 ::std::unique_ptr<frc::AnalogInput> potentiometer) {
195 wrist_encoder_.set_potentiometer(::std::move(potentiometer));
196 }
197
198 // Stilts
199
200 void set_stilts_encoder(::std::unique_ptr<frc::Encoder> encoder) {
201 medium_encoder_filter_.Add(encoder.get());
202 stilts_encoder_.set_encoder(::std::move(encoder));
203 }
204
205 void set_stilts_absolute_pwm(
206 ::std::unique_ptr<frc::DigitalInput> absolute_pwm) {
207 stilts_encoder_.set_absolute_pwm(::std::move(absolute_pwm));
208 }
209
210 void set_stilts_potentiometer(
211 ::std::unique_ptr<frc::AnalogInput> potentiometer) {
212 stilts_encoder_.set_potentiometer(::std::move(potentiometer));
213 }
214
Austin Schuhe2f22482019-04-13 23:05:43 -0700215 void set_platform_left_detect(
216 ::std::unique_ptr<frc::DigitalInput> platform_left_detect) {
217 platform_left_detect_ = ::std::move(platform_left_detect);
218 }
219
220 void set_platform_right_detect(
221 ::std::unique_ptr<frc::DigitalInput> platform_right_detect) {
222 platform_right_detect_ = ::std::move(platform_right_detect);
223 }
224
Austin Schuh461e1182019-02-17 14:56:44 -0800225 // Vacuum pressure sensor
226 void set_vacuum_sensor(int port) {
227 vacuum_sensor_ = make_unique<frc::AnalogInput>(port);
228 }
229
Austin Schuha9644062019-03-28 14:31:52 -0700230 // Auto mode switches.
231 void set_autonomous_mode(int i, ::std::unique_ptr<frc::DigitalInput> sensor) {
232 autonomous_modes_.at(i) = ::std::move(sensor);
233 }
234
Sabina Davis399dbd82019-02-01 23:06:08 -0800235 void RunIteration() override {
Sabina Davisabeae332019-02-01 21:12:57 -0800236 {
Alex Perrycb7da4b2019-08-28 19:35:56 -0700237 auto builder = drivetrain_position_sender_.MakeBuilder();
238 frc971::control_loops::drivetrain::Position::Builder drivetrain_builder =
239 builder.MakeBuilder<frc971::control_loops::drivetrain::Position>();
240 drivetrain_builder.add_left_encoder(
241 drivetrain_translate(drivetrain_left_encoder_->GetRaw()));
242 drivetrain_builder.add_left_speed(
243 drivetrain_velocity_translate(drivetrain_left_encoder_->GetPeriod()));
Sabina Davisabeae332019-02-01 21:12:57 -0800244
Alex Perrycb7da4b2019-08-28 19:35:56 -0700245 drivetrain_builder.add_right_encoder(
246 -drivetrain_translate(drivetrain_right_encoder_->GetRaw()));
247 drivetrain_builder.add_right_speed(-drivetrain_velocity_translate(
248 drivetrain_right_encoder_->GetPeriod()));
Sabina Davisabeae332019-02-01 21:12:57 -0800249
Alex Perrycb7da4b2019-08-28 19:35:56 -0700250 builder.Send(drivetrain_builder.Finish());
Sabina Davisabeae332019-02-01 21:12:57 -0800251 }
Alex Perry5fb5ff22019-02-09 21:53:17 -0800252 const auto values = constants::GetValues();
253
254 {
Alex Perrycb7da4b2019-08-28 19:35:56 -0700255 auto builder = superstructure_position_sender_.MakeBuilder();
Alex Perry5fb5ff22019-02-09 21:53:17 -0800256
257 // Elevator
Alex Perrycb7da4b2019-08-28 19:35:56 -0700258 frc971::PotAndAbsolutePositionT elevator;
259 CopyPosition(elevator_encoder_, &elevator,
Alex Perry5fb5ff22019-02-09 21:53:17 -0800260 Values::kElevatorEncoderCountsPerRevolution(),
261 Values::kElevatorEncoderRatio(), elevator_pot_translate,
262 false, values.elevator.potentiometer_offset);
Alex Perrycb7da4b2019-08-28 19:35:56 -0700263 flatbuffers::Offset<frc971::PotAndAbsolutePosition> elevator_offset =
264 frc971::PotAndAbsolutePosition::Pack(*builder.fbb(), &elevator);
265
Alex Perry5fb5ff22019-02-09 21:53:17 -0800266 // Intake
Alex Perrycb7da4b2019-08-28 19:35:56 -0700267 frc971::AbsolutePositionT intake_joint;
268 CopyPosition(intake_encoder_, &intake_joint,
Alex Perry5fb5ff22019-02-09 21:53:17 -0800269 Values::kIntakeEncoderCountsPerRevolution(),
270 Values::kIntakeEncoderRatio(), false);
Alex Perrycb7da4b2019-08-28 19:35:56 -0700271 flatbuffers::Offset<frc971::AbsolutePosition> intake_joint_offset =
272 frc971::AbsolutePosition::Pack(*builder.fbb(), &intake_joint);
Alex Perry5fb5ff22019-02-09 21:53:17 -0800273
274 // Wrist
Alex Perrycb7da4b2019-08-28 19:35:56 -0700275 frc971::PotAndAbsolutePositionT wrist;
276 CopyPosition(wrist_encoder_, &wrist,
Alex Perry5fb5ff22019-02-09 21:53:17 -0800277 Values::kWristEncoderCountsPerRevolution(),
278 Values::kWristEncoderRatio(), wrist_pot_translate, false,
279 values.wrist.potentiometer_offset);
Alex Perrycb7da4b2019-08-28 19:35:56 -0700280 flatbuffers::Offset<frc971::PotAndAbsolutePosition> wrist_offset =
281 frc971::PotAndAbsolutePosition::Pack(*builder.fbb(), &wrist);
Alex Perry5fb5ff22019-02-09 21:53:17 -0800282
283 // Stilts
Alex Perrycb7da4b2019-08-28 19:35:56 -0700284 frc971::PotAndAbsolutePositionT stilts;
285 CopyPosition(stilts_encoder_, &stilts,
Alex Perry5fb5ff22019-02-09 21:53:17 -0800286 Values::kStiltsEncoderCountsPerRevolution(),
287 Values::kStiltsEncoderRatio(), stilts_pot_translate, false,
288 values.stilts.potentiometer_offset);
Alex Perrycb7da4b2019-08-28 19:35:56 -0700289 flatbuffers::Offset<frc971::PotAndAbsolutePosition> stilts_offset =
290 frc971::PotAndAbsolutePosition::Pack(*builder.fbb(), &stilts);
291
292 superstructure::Position::Builder position_builder =
293 builder.MakeBuilder<superstructure::Position>();
294
295 position_builder.add_elevator(elevator_offset);
296 position_builder.add_intake_joint(intake_joint_offset);
297 position_builder.add_wrist(wrist_offset);
298 position_builder.add_stilts(stilts_offset);
Alex Perry5fb5ff22019-02-09 21:53:17 -0800299
Austin Schuh461e1182019-02-17 14:56:44 -0800300 // Suction
301 constexpr float kMinVoltage = 0.5;
302 constexpr float kMaxVoltage = 2.1;
Alex Perrycb7da4b2019-08-28 19:35:56 -0700303 position_builder.add_suction_pressure(
Austin Schuh461e1182019-02-17 14:56:44 -0800304 (vacuum_sensor_->GetVoltage() - kMinVoltage) /
Alex Perrycb7da4b2019-08-28 19:35:56 -0700305 (kMaxVoltage - kMinVoltage));
Austin Schuh461e1182019-02-17 14:56:44 -0800306
Alex Perrycb7da4b2019-08-28 19:35:56 -0700307 position_builder.add_platform_left_detect(!platform_left_detect_->Get());
308 position_builder.add_platform_right_detect(
309 !platform_right_detect_->Get());
Austin Schuhe2f22482019-04-13 23:05:43 -0700310
Alex Perrycb7da4b2019-08-28 19:35:56 -0700311 builder.Send(position_builder.Finish());
Alex Perry5fb5ff22019-02-09 21:53:17 -0800312 }
Austin Schuha9644062019-03-28 14:31:52 -0700313
314 {
Alex Perrycb7da4b2019-08-28 19:35:56 -0700315 auto builder = auto_mode_sender_.MakeBuilder();
316
317 uint32_t mode = 0;
Austin Schuha9644062019-03-28 14:31:52 -0700318 for (size_t i = 0; i < autonomous_modes_.size(); ++i) {
319 if (autonomous_modes_[i] && autonomous_modes_[i]->Get()) {
Alex Perrycb7da4b2019-08-28 19:35:56 -0700320 mode |= 1 << i;
Austin Schuha9644062019-03-28 14:31:52 -0700321 }
322 }
Alex Perrycb7da4b2019-08-28 19:35:56 -0700323
324 auto auto_mode_builder =
325 builder.MakeBuilder<frc971::autonomous::AutonomousMode>();
326
327 auto_mode_builder.add_mode(mode);
328
329 builder.Send(auto_mode_builder.Finish());
Austin Schuha9644062019-03-28 14:31:52 -0700330 }
Alex Perry5fb5ff22019-02-09 21:53:17 -0800331 }
332
333 private:
Austin Schuha250b2d2019-05-27 16:14:02 -0700334 ::aos::Sender<::frc971::autonomous::AutonomousMode> auto_mode_sender_;
Alex Perrycb7da4b2019-08-28 19:35:56 -0700335 ::aos::Sender<superstructure::Position> superstructure_position_sender_;
336 ::aos::Sender<::frc971::control_loops::drivetrain::Position>
Austin Schuhbd0a40f2019-06-30 14:56:31 -0700337 drivetrain_position_sender_;
Austin Schuha250b2d2019-05-27 16:14:02 -0700338
Alex Perry5fb5ff22019-02-09 21:53:17 -0800339 ::frc971::wpilib::AbsoluteEncoderAndPotentiometer elevator_encoder_,
340 wrist_encoder_, stilts_encoder_;
341
Austin Schuhe2f22482019-04-13 23:05:43 -0700342 ::std::unique_ptr<frc::DigitalInput> platform_left_detect_;
343 ::std::unique_ptr<frc::DigitalInput> platform_right_detect_;
344
Austin Schuh461e1182019-02-17 14:56:44 -0800345 ::std::unique_ptr<frc::AnalogInput> vacuum_sensor_;
346
Austin Schuha9644062019-03-28 14:31:52 -0700347 ::std::array<::std::unique_ptr<frc::DigitalInput>, 2> autonomous_modes_;
348
Alex Perry5fb5ff22019-02-09 21:53:17 -0800349 ::frc971::wpilib::AbsoluteEncoder intake_encoder_;
Alex Perry5fb5ff22019-02-09 21:53:17 -0800350};
351
Brian Silvermanf8b75252019-02-24 16:13:58 -0800352class CameraReader {
353 public:
Austin Schuh8a633d52019-05-12 15:04:01 -0700354 CameraReader(::aos::EventLoop *event_loop)
355 : camera_frame_sender_(
356 event_loop
357 ->MakeSender<::y2019::control_loops::drivetrain::CameraFrame>(
Austin Schuhed5b26d2019-12-05 20:51:59 -0800358 "/camera")),
Austin Schuh5671a8c2019-05-19 17:01:04 -0700359 camera_log_fetcher_(
Austin Schuhed5b26d2019-12-05 20:51:59 -0800360 event_loop->MakeFetcher<::y2019::CameraLog>("/camera")) {}
Austin Schuh8a633d52019-05-12 15:04:01 -0700361
Brian Silvermanf8b75252019-02-24 16:13:58 -0800362 CameraReader(const CameraReader &) = delete;
363 CameraReader &operator=(const CameraReader &) = delete;
364
365 void set_spi(frc::SPI *spi) {
366 spi_ = spi;
367 spi_->SetClockRate(1e6);
368 spi_->SetChipSelectActiveHigh();
369 spi_->SetClockActiveLow();
370 spi_->SetSampleDataOnFalling();
371 // It ignores you if you try changing this...
372 spi_->SetMSBFirst();
373 }
374
Brian Silverman7ecf0672019-03-02 15:30:03 -0800375 void set_activate_usb(std::unique_ptr<frc::DigitalInput> activate_usb) {
376 activate_usb_ = std::move(activate_usb);
377 }
378
379 void set_activate_passthrough(
380 std::unique_ptr<frc::DigitalInput> activate_passthrough) {
381 activate_passthrough_ = std::move(activate_passthrough);
382 }
383
Brian Silvermanf8b75252019-02-24 16:13:58 -0800384 void DoSpiTransaction() {
385 using namespace frc971::jevois;
386 RoborioToTeensy to_teensy{};
387 to_teensy.realtime_now = aos::realtime_clock::now();
Austin Schuh5671a8c2019-05-19 17:01:04 -0700388 camera_log_fetcher_.Fetch();
Brian Silverman7ecf0672019-03-02 15:30:03 -0800389 if (activate_usb_ && !activate_usb_->Get()) {
390 to_teensy.camera_command = CameraCommand::kUsb;
391 } else if (activate_passthrough_ && !activate_passthrough_->Get()) {
392 to_teensy.camera_command = CameraCommand::kCameraPassthrough;
Alex Perrycb7da4b2019-08-28 19:35:56 -0700393 } else if (camera_log_fetcher_.get() && camera_log_fetcher_->log()) {
Austin Schuh4e2629d2019-03-28 14:44:37 -0700394 to_teensy.camera_command = CameraCommand::kLog;
Brian Silverman7ecf0672019-03-02 15:30:03 -0800395 } else {
396 to_teensy.camera_command = CameraCommand::kNormal;
397 }
Brian Silvermanf8b75252019-02-24 16:13:58 -0800398
399 std::array<char, spi_transfer_size() + 1> to_send{};
400 {
401 const auto to_send_data =
402 gsl::make_span(to_send).last<spi_transfer_size()>();
403 const auto encoded = SpiPackToTeensy(to_teensy);
404 std::copy(encoded.begin(), encoded.end(), to_send_data.begin());
405 }
406 rx_clearer_.ClearRxFifo();
407 // First, send recieve a dummy byte because the Teensy can't control what it
408 // sends for the first byte.
409 std::array<char, spi_transfer_size() + 1> to_receive;
410 DoTransaction(to_send, to_receive);
411 const auto unpacked = SpiUnpackToRoborio(
412 gsl::make_span(to_receive).last(spi_transfer_size()));
413 if (!unpacked) {
Austin Schuhf257f3c2019-10-27 21:00:43 -0700414 AOS_LOG(INFO, "Decoding SPI data failed\n");
Brian Silvermanf8b75252019-02-24 16:13:58 -0800415 return;
416 }
417
Alex Perrycb7da4b2019-08-28 19:35:56 -0700418 const aos::monotonic_clock::time_point now = aos::monotonic_clock::now();
Brian Silvermanc41fb862019-03-02 21:14:46 -0800419 for (const auto &received : unpacked->frames) {
Alex Perrycb7da4b2019-08-28 19:35:56 -0700420 auto builder = camera_frame_sender_.MakeBuilder();
421
422 std::array<
423 flatbuffers::Offset<y2019::control_loops::drivetrain::CameraTarget>,
424 3>
425 targets;
426
427 for (size_t i = 0; i < received.targets.size(); ++i) {
428 y2019::control_loops::drivetrain::CameraTarget::Builder
429 camera_target_builder = builder.MakeBuilder<
430 y2019::control_loops::drivetrain::CameraTarget>();
431
432 camera_target_builder.add_distance(received.targets[i].distance);
433 camera_target_builder.add_height(received.targets[i].height);
434 camera_target_builder.add_heading(received.targets[i].heading);
435 camera_target_builder.add_skew(received.targets[i].skew);
436
437 targets[i] = camera_target_builder.Finish();
438 }
439
440 flatbuffers::Offset<flatbuffers::Vector<
441 flatbuffers::Offset<y2019::control_loops::drivetrain::CameraTarget>>>
442 targets_offset = builder.fbb()->CreateVector(targets.begin(),
443 received.targets.size());
444
445 y2019::control_loops::drivetrain::CameraFrame::Builder
446 camera_frame_builder =
447 builder
448 .MakeBuilder<y2019::control_loops::drivetrain::CameraFrame>();
449
450 camera_frame_builder.add_targets(targets_offset);
451
James Kuszmaule08f04e2019-05-01 21:46:50 -0500452 // Add an extra 10ms delay to account for unmodeled delays that Austin
453 // thinks exists.
Alex Perrycb7da4b2019-08-28 19:35:56 -0700454 camera_frame_builder.add_timestamp(
James Kuszmaule08f04e2019-05-01 21:46:50 -0500455 std::chrono::nanoseconds(
456 (now - received.age - ::std::chrono::milliseconds(10))
Alex Perrycb7da4b2019-08-28 19:35:56 -0700457 .time_since_epoch())
458 .count());
459 camera_frame_builder.add_camera(received.camera_index);
460 builder.Send(camera_frame_builder.Finish());
Brian Silvermanc41fb862019-03-02 21:14:46 -0800461 }
Brian Silvermanf8b75252019-02-24 16:13:58 -0800462
463 if (dummy_spi_) {
464 uint8_t dummy_send, dummy_receive;
465 dummy_spi_->Transaction(&dummy_send, &dummy_receive, 1);
466 }
467 }
468
469 void DoTransaction(gsl::span<char> to_send, gsl::span<char> to_receive) {
Austin Schuhf257f3c2019-10-27 21:00:43 -0700470 AOS_CHECK_EQ(to_send.size(), to_receive.size());
Brian Silvermanf8b75252019-02-24 16:13:58 -0800471 const auto result = spi_->Transaction(
472 reinterpret_cast<uint8_t *>(to_send.data()),
473 reinterpret_cast<uint8_t *>(to_receive.data()), to_send.size());
474 if (result == to_send.size()) {
475 return;
476 }
477 if (result == -1) {
Austin Schuhf257f3c2019-10-27 21:00:43 -0700478 AOS_LOG(INFO, "SPI::Transaction of %zd bytes failed\n", to_send.size());
Brian Silvermanf8b75252019-02-24 16:13:58 -0800479 return;
480 }
Austin Schuhf257f3c2019-10-27 21:00:43 -0700481 AOS_LOG(FATAL, "SPI::Transaction returned something weird\n");
Brian Silvermanf8b75252019-02-24 16:13:58 -0800482 }
483
484 void SetDummySPI(frc::SPI::Port port) {
485 dummy_spi_.reset(new frc::SPI(port));
486 // Pick the same settings here in case the roboRIO decides to try something
487 // stupid when switching.
488 if (dummy_spi_) {
489 dummy_spi_->SetClockRate(1e5);
490 dummy_spi_->SetChipSelectActiveLow();
491 dummy_spi_->SetClockActiveLow();
492 dummy_spi_->SetSampleDataOnFalling();
493 dummy_spi_->SetMSBFirst();
494 }
495 }
496
497 private:
Austin Schuh8a633d52019-05-12 15:04:01 -0700498 ::aos::Sender<::y2019::control_loops::drivetrain::CameraFrame>
499 camera_frame_sender_;
Austin Schuh5671a8c2019-05-19 17:01:04 -0700500 ::aos::Fetcher<::y2019::CameraLog> camera_log_fetcher_;
Austin Schuh8a633d52019-05-12 15:04:01 -0700501
Brian Silvermanf8b75252019-02-24 16:13:58 -0800502 frc::SPI *spi_ = nullptr;
503 ::std::unique_ptr<frc::SPI> dummy_spi_;
504
Brian Silverman7ecf0672019-03-02 15:30:03 -0800505 std::unique_ptr<frc::DigitalInput> activate_usb_;
506 std::unique_ptr<frc::DigitalInput> activate_passthrough_;
507
Brian Silvermanf8b75252019-02-24 16:13:58 -0800508 frc971::wpilib::SpiRxClearer rx_clearer_;
509};
510
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700511class SuperstructureWriter
Alex Perrycb7da4b2019-08-28 19:35:56 -0700512 : public ::frc971::wpilib::LoopOutputHandler<superstructure::Output> {
Alex Perry5fb5ff22019-02-09 21:53:17 -0800513 public:
Austin Schuhdf6cbb12019-02-02 13:46:52 -0800514 SuperstructureWriter(::aos::EventLoop *event_loop)
Alex Perrycb7da4b2019-08-28 19:35:56 -0700515 : ::frc971::wpilib::LoopOutputHandler<superstructure::Output>(
516 event_loop, "/superstructure"),
Austin Schuhdf6cbb12019-02-02 13:46:52 -0800517 robot_state_fetcher_(
Alex Perrycb7da4b2019-08-28 19:35:56 -0700518 event_loop->MakeFetcher<::aos::RobotState>("/aos")) {}
Austin Schuhdf6cbb12019-02-02 13:46:52 -0800519
Alex Perry5fb5ff22019-02-09 21:53:17 -0800520 void set_elevator_victor(::std::unique_ptr<::frc::VictorSP> t) {
521 elevator_victor_ = ::std::move(t);
522 }
523
Austin Schuh461e1182019-02-17 14:56:44 -0800524 void set_suction_victor(::std::unique_ptr<::frc::VictorSP> t) {
525 suction_victor_ = ::std::move(t);
526 }
527
Alex Perry5fb5ff22019-02-09 21:53:17 -0800528 void set_intake_victor(::std::unique_ptr<::frc::VictorSP> t) {
529 intake_victor_ = ::std::move(t);
530 }
Alex Perry5fb5ff22019-02-09 21:53:17 -0800531
532 void set_wrist_victor(::std::unique_ptr<::frc::VictorSP> t) {
533 wrist_victor_ = ::std::move(t);
534 }
535
536 void set_stilts_victor(::std::unique_ptr<::frc::VictorSP> t) {
537 stilts_victor_ = ::std::move(t);
538 }
539
540 private:
Alex Perrycb7da4b2019-08-28 19:35:56 -0700541 void Write(const superstructure::Output &output) override {
542 elevator_victor_->SetSpeed(::aos::Clip(output.elevator_voltage(),
Alex Perry5fb5ff22019-02-09 21:53:17 -0800543 -kMaxBringupPower,
544 kMaxBringupPower) /
545 12.0);
546
Alex Perrycb7da4b2019-08-28 19:35:56 -0700547 intake_victor_->SetSpeed(::aos::Clip(output.intake_joint_voltage(),
Alex Perry5fb5ff22019-02-09 21:53:17 -0800548 -kMaxBringupPower, kMaxBringupPower) /
549 12.0);
550
Alex Perrycb7da4b2019-08-28 19:35:56 -0700551 wrist_victor_->SetSpeed(::aos::Clip(-output.wrist_voltage(),
Alex Perry5fb5ff22019-02-09 21:53:17 -0800552 -kMaxBringupPower, kMaxBringupPower) /
553 12.0);
554
Alex Perrycb7da4b2019-08-28 19:35:56 -0700555 stilts_victor_->SetSpeed(::aos::Clip(output.stilts_voltage(),
Alex Perry5fb5ff22019-02-09 21:53:17 -0800556 -kMaxBringupPower, kMaxBringupPower) /
557 12.0);
Austin Schuh461e1182019-02-17 14:56:44 -0800558
Austin Schuhdf6cbb12019-02-02 13:46:52 -0800559 robot_state_fetcher_.Fetch();
560 const double battery_voltage = robot_state_fetcher_.get()
Alex Perrycb7da4b2019-08-28 19:35:56 -0700561 ? robot_state_fetcher_->voltage_battery()
Austin Schuhdf6cbb12019-02-02 13:46:52 -0800562 : 12.0;
Austin Schuhc2ee66b2019-02-19 13:37:46 -0800563
564 // Throw a fast low pass filter on the battery voltage so we don't respond
565 // too fast to noise.
566 filtered_battery_voltage_ =
567 0.5 * filtered_battery_voltage_ + 0.5 * battery_voltage;
568
569 suction_victor_->SetSpeed(::aos::Clip(
Alex Perrycb7da4b2019-08-28 19:35:56 -0700570 output.pump_voltage() / filtered_battery_voltage_, -1.0, 1.0));
Alex Perry5fb5ff22019-02-09 21:53:17 -0800571 }
572
Austin Schuh461e1182019-02-17 14:56:44 -0800573 void Stop() override {
Austin Schuhf257f3c2019-10-27 21:00:43 -0700574 AOS_LOG(WARNING, "Superstructure output too old.\n");
Alex Perry5fb5ff22019-02-09 21:53:17 -0800575
576 elevator_victor_->SetDisabled();
577 intake_victor_->SetDisabled();
Alex Perry5fb5ff22019-02-09 21:53:17 -0800578 wrist_victor_->SetDisabled();
579 stilts_victor_->SetDisabled();
Austin Schuh461e1182019-02-17 14:56:44 -0800580 suction_victor_->SetDisabled();
Alex Perry5fb5ff22019-02-09 21:53:17 -0800581 }
582
Austin Schuhdf6cbb12019-02-02 13:46:52 -0800583 ::aos::Fetcher<::aos::RobotState> robot_state_fetcher_;
584
Alex Perry5fb5ff22019-02-09 21:53:17 -0800585 ::std::unique_ptr<::frc::VictorSP> elevator_victor_, intake_victor_,
Austin Schuh461e1182019-02-17 14:56:44 -0800586 wrist_victor_, stilts_victor_, suction_victor_;
Austin Schuhc2ee66b2019-02-19 13:37:46 -0800587
588 double filtered_battery_voltage_ = 12.0;
Sabina Davisabeae332019-02-01 21:12:57 -0800589};
590
Austin Schuhc1d6f832019-02-15 23:22:17 -0800591class SolenoidWriter {
592 public:
Austin Schuhff973552019-05-19 16:49:28 -0700593 SolenoidWriter(::aos::EventLoop *event_loop)
594 : event_loop_(event_loop),
Alex Perrycb7da4b2019-08-28 19:35:56 -0700595 superstructure_fetcher_(
596 event_loop->MakeFetcher<superstructure::Output>("/superstructure")),
597 status_light_fetcher_(
598 event_loop->MakeFetcher<::y2019::StatusLight>("/superstructure")),
599 pneumatics_to_log_sender_(
600 event_loop->MakeSender<::frc971::wpilib::PneumaticsToLog>("/aos")) {
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700601 ::aos::SetCurrentThreadName("Solenoids");
602 ::aos::SetCurrentThreadRealtimePriority(27);
603
604 event_loop_->AddPhasedLoop([this](int iterations) { Loop(iterations); },
605 ::std::chrono::milliseconds(20),
606 ::std::chrono::milliseconds(1));
607 }
Austin Schuhc1d6f832019-02-15 23:22:17 -0800608
Austin Schuh461e1182019-02-17 14:56:44 -0800609 void set_big_suction_cup(int index0, int index1) {
610 big_suction_cup0_ = pcm_.MakeSolenoid(index0);
611 big_suction_cup1_ = pcm_.MakeSolenoid(index1);
Austin Schuhc1d6f832019-02-15 23:22:17 -0800612 }
Austin Schuh461e1182019-02-17 14:56:44 -0800613 void set_small_suction_cup(int index0, int index1) {
614 small_suction_cup0_ = pcm_.MakeSolenoid(index0);
615 small_suction_cup1_ = pcm_.MakeSolenoid(index1);
Austin Schuhc1d6f832019-02-15 23:22:17 -0800616 }
617
618 void set_intake_roller_talon(
619 ::std::unique_ptr<::ctre::phoenix::motorcontrol::can::TalonSRX> t) {
620 intake_rollers_talon_ = ::std::move(t);
Austin Schuh23a51632019-02-19 16:50:36 -0800621 intake_rollers_talon_->ConfigContinuousCurrentLimit(10.0, 0);
Austin Schuhc1d6f832019-02-15 23:22:17 -0800622 intake_rollers_talon_->EnableCurrentLimit(true);
623 }
624
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700625 void Loop(const int iterations) {
626 if (iterations != 1) {
Austin Schuhf257f3c2019-10-27 21:00:43 -0700627 AOS_LOG(DEBUG, "Solenoids skipped %d iterations\n", iterations - 1);
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700628 }
Austin Schuhc1d6f832019-02-15 23:22:17 -0800629
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700630 {
631 superstructure_fetcher_.Fetch();
632 if (superstructure_fetcher_.get()) {
Alex Perrycb7da4b2019-08-28 19:35:56 -0700633 big_suction_cup0_->Set(
634 !superstructure_fetcher_->intake_suction_bottom());
635 big_suction_cup1_->Set(
636 !superstructure_fetcher_->intake_suction_bottom());
637 small_suction_cup0_->Set(superstructure_fetcher_->intake_suction_top());
638 small_suction_cup1_->Set(superstructure_fetcher_->intake_suction_top());
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700639
640 intake_rollers_talon_->Set(
641 ctre::phoenix::motorcontrol::ControlMode::PercentOutput,
Alex Perrycb7da4b2019-08-28 19:35:56 -0700642 ::aos::Clip(superstructure_fetcher_->intake_roller_voltage(),
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700643 -kMaxBringupPower, kMaxBringupPower) /
644 12.0);
645 }
646 }
647
648 {
Alex Perrycb7da4b2019-08-28 19:35:56 -0700649 auto builder = pneumatics_to_log_sender_.MakeBuilder();
650
651 ::frc971::wpilib::PneumaticsToLog::Builder to_log_builder =
652 builder.MakeBuilder<frc971::wpilib::PneumaticsToLog>();
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700653
654 pcm_.Flush();
Alex Perrycb7da4b2019-08-28 19:35:56 -0700655 to_log_builder.add_read_solenoids(pcm_.GetAll());
656 builder.Send(to_log_builder.Finish());
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700657 }
658
659 status_light_fetcher_.Fetch();
660 // If we don't have a light request (or it's an old one), we are borked.
661 // Flash the red light slowly.
Alex Perrycb7da4b2019-08-28 19:35:56 -0700662 StatusLightT color;
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700663 if (!status_light_fetcher_.get() ||
Austin Schuhad154822019-12-27 15:45:13 -0800664 status_light_fetcher_.context().monotonic_event_time +
Alex Perrycb7da4b2019-08-28 19:35:56 -0700665 chrono::milliseconds(100) <
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700666 event_loop_->monotonic_now()) {
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700667 color.red = 0.0;
668 color.green = 0.0;
669 color.blue = 0.0;
670
671 ++light_flash_;
672 if (light_flash_ > 10) {
673 color.red = 0.5;
Austin Schuhc1d6f832019-02-15 23:22:17 -0800674 }
675
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700676 if (light_flash_ > 20) {
677 light_flash_ = 0;
Austin Schuhc1d6f832019-02-15 23:22:17 -0800678 }
679
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700680 } else {
Alex Perrycb7da4b2019-08-28 19:35:56 -0700681 status_light_fetcher_->UnPackTo(&color);
Sabina Davisc6329342019-03-01 20:44:42 -0800682 }
Alex Perrycb7da4b2019-08-28 19:35:56 -0700683 SetColor(color);
Sabina Davisc6329342019-03-01 20:44:42 -0800684 }
685
Alex Perrycb7da4b2019-08-28 19:35:56 -0700686 void SetColor(const StatusLightT status_light) {
Sabina Davisc6329342019-03-01 20:44:42 -0800687 // Save CAN bandwidth and CPU at the cost of RT. Only change the light when
688 // it actually changes. This is pretty low priority anyways.
689 static int time_since_last_send = 0;
690 ++time_since_last_send;
691 if (time_since_last_send > 10) {
692 time_since_last_send = 0;
693 }
694 if (status_light.green != last_green_ || time_since_last_send == 0) {
Sabina Davis77a11cf2019-03-09 18:20:26 -0800695 canifier_.SetLEDOutput(status_light.green,
696 ::ctre::phoenix::CANifier::LEDChannelA);
Sabina Davisc6329342019-03-01 20:44:42 -0800697 last_green_ = status_light.green;
698 }
699
700 if (status_light.blue != last_blue_ || time_since_last_send == 0) {
Sabina Davis77a11cf2019-03-09 18:20:26 -0800701 canifier_.SetLEDOutput(status_light.blue,
702 ::ctre::phoenix::CANifier::LEDChannelC);
Sabina Davisc6329342019-03-01 20:44:42 -0800703 last_blue_ = status_light.blue;
704 }
705
706 if (status_light.red != last_red_ || time_since_last_send == 0) {
Sabina Davis77a11cf2019-03-09 18:20:26 -0800707 canifier_.SetLEDOutput(status_light.red,
708 ::ctre::phoenix::CANifier::LEDChannelB);
Sabina Davisc6329342019-03-01 20:44:42 -0800709 last_red_ = status_light.red;
Austin Schuhc1d6f832019-02-15 23:22:17 -0800710 }
711 }
712
Austin Schuhc1d6f832019-02-15 23:22:17 -0800713 private:
Austin Schuhff973552019-05-19 16:49:28 -0700714 ::aos::EventLoop *event_loop_;
715
Austin Schuhc1d6f832019-02-15 23:22:17 -0800716 ::frc971::wpilib::BufferedPcm pcm_;
717
Austin Schuh461e1182019-02-17 14:56:44 -0800718 ::std::unique_ptr<::frc971::wpilib::BufferedSolenoid> big_suction_cup0_,
719 big_suction_cup1_, small_suction_cup0_, small_suction_cup1_;
Austin Schuhc1d6f832019-02-15 23:22:17 -0800720
721 ::std::unique_ptr<::ctre::phoenix::motorcontrol::can::TalonSRX>
722 intake_rollers_talon_;
723
Alex Perrycb7da4b2019-08-28 19:35:56 -0700724 ::aos::Fetcher<::y2019::control_loops::superstructure::Output>
Austin Schuhff973552019-05-19 16:49:28 -0700725 superstructure_fetcher_;
726 ::aos::Fetcher<::y2019::StatusLight> status_light_fetcher_;
Austin Schuhc1d6f832019-02-15 23:22:17 -0800727
Alex Perrycb7da4b2019-08-28 19:35:56 -0700728 aos::Sender<::frc971::wpilib::PneumaticsToLog> pneumatics_to_log_sender_;
729
Sabina Davisc6329342019-03-01 20:44:42 -0800730 ::ctre::phoenix::CANifier canifier_{0};
731
Sabina Davisc6329342019-03-01 20:44:42 -0800732 double last_red_ = -1.0;
733 double last_green_ = -1.0;
734 double last_blue_ = -1.0;
735
736 int light_flash_ = 0;
Austin Schuhc1d6f832019-02-15 23:22:17 -0800737};
738
Sabina Davisabeae332019-02-01 21:12:57 -0800739class WPILibRobot : public ::frc971::wpilib::WPILibRobotBase {
740 public:
741 ::std::unique_ptr<frc::Encoder> make_encoder(int index) {
742 return make_unique<frc::Encoder>(10 + index * 2, 11 + index * 2, false,
743 frc::Encoder::k4X);
744 }
745
746 void Run() override {
Alex Perrycb7da4b2019-08-28 19:35:56 -0700747 aos::FlatbufferDetachedBuffer<aos::Configuration> config =
748 aos::configuration::ReadConfig("config.json");
749
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700750 // Thread 1.
Alex Perrycb7da4b2019-08-28 19:35:56 -0700751 ::aos::ShmEventLoop joystick_sender_event_loop(&config.message());
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700752 ::frc971::wpilib::JoystickSender joystick_sender(
753 &joystick_sender_event_loop);
754 AddLoop(&joystick_sender_event_loop);
Sabina Davisabeae332019-02-01 21:12:57 -0800755
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700756 // Thread 2.
Alex Perrycb7da4b2019-08-28 19:35:56 -0700757 ::aos::ShmEventLoop pdp_fetcher_event_loop(&config.message());
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700758 ::frc971::wpilib::PDPFetcher pdp_fetcher(&pdp_fetcher_event_loop);
759 AddLoop(&pdp_fetcher_event_loop);
Austin Schuhdf6cbb12019-02-02 13:46:52 -0800760
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700761 // Thread 3.
Alex Perrycb7da4b2019-08-28 19:35:56 -0700762 ::aos::ShmEventLoop sensor_reader_event_loop(&config.message());
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700763 SensorReader sensor_reader(&sensor_reader_event_loop);
764 sensor_reader.set_drivetrain_left_encoder(make_encoder(0));
765 sensor_reader.set_drivetrain_right_encoder(make_encoder(1));
Sabina Davisabeae332019-02-01 21:12:57 -0800766
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700767 sensor_reader.set_elevator_encoder(make_encoder(4));
768 sensor_reader.set_elevator_absolute_pwm(make_unique<frc::DigitalInput>(4));
769 sensor_reader.set_elevator_potentiometer(make_unique<frc::AnalogInput>(4));
Sabina Davisabeae332019-02-01 21:12:57 -0800770
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700771 sensor_reader.set_wrist_encoder(make_encoder(5));
772 sensor_reader.set_wrist_absolute_pwm(make_unique<frc::DigitalInput>(5));
773 sensor_reader.set_wrist_potentiometer(make_unique<frc::AnalogInput>(5));
Sabina Davisabeae332019-02-01 21:12:57 -0800774
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700775 sensor_reader.set_intake_encoder(make_encoder(2));
776 sensor_reader.set_intake_absolute_pwm(make_unique<frc::DigitalInput>(2));
Alex Perry5fb5ff22019-02-09 21:53:17 -0800777
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700778 sensor_reader.set_stilts_encoder(make_encoder(3));
779 sensor_reader.set_stilts_absolute_pwm(make_unique<frc::DigitalInput>(3));
780 sensor_reader.set_stilts_potentiometer(make_unique<frc::AnalogInput>(3));
Alex Perry5fb5ff22019-02-09 21:53:17 -0800781
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700782 sensor_reader.set_pwm_trigger(true);
783 sensor_reader.set_vacuum_sensor(7);
Alex Perry5fb5ff22019-02-09 21:53:17 -0800784
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700785 sensor_reader.set_platform_right_detect(make_unique<frc::DigitalInput>(6));
786 sensor_reader.set_platform_left_detect(make_unique<frc::DigitalInput>(7));
Alex Perry5fb5ff22019-02-09 21:53:17 -0800787
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700788 sensor_reader.set_autonomous_mode(0, make_unique<frc::DigitalInput>(22));
789 sensor_reader.set_autonomous_mode(0, make_unique<frc::DigitalInput>(23));
790 AddLoop(&sensor_reader_event_loop);
Sabina Davisabeae332019-02-01 21:12:57 -0800791
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700792 // Thread 4.
Alex Perrycb7da4b2019-08-28 19:35:56 -0700793 ::aos::ShmEventLoop imu_event_loop(&config.message());
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700794 CameraReader camera_reader(&imu_event_loop);
Brian Silvermanf8b75252019-02-24 16:13:58 -0800795 frc::SPI camera_spi(frc::SPI::Port::kOnboardCS3);
796 camera_reader.set_spi(&camera_spi);
797 camera_reader.SetDummySPI(frc::SPI::Port::kOnboardCS2);
Brian Silverman7ecf0672019-03-02 15:30:03 -0800798 // Austin says 8, 9, 24, and 25 are good options to choose from for these.
799 camera_reader.set_activate_usb(make_unique<frc::DigitalInput>(24));
800 camera_reader.set_activate_passthrough(make_unique<frc::DigitalInput>(25));
Brian Silvermanf8b75252019-02-24 16:13:58 -0800801
Austin Schuh3e3d4ba2019-02-15 23:14:52 -0800802 auto imu_trigger = make_unique<frc::DigitalInput>(0);
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700803 ::frc971::wpilib::ADIS16448 imu(&imu_event_loop, frc::SPI::Port::kOnboardCS1,
Sabina Davisabeae332019-02-01 21:12:57 -0800804 imu_trigger.get());
Brian Silvermanf8b75252019-02-24 16:13:58 -0800805 imu.set_spi_idle_callback(
806 [&camera_reader]() { camera_reader.DoSpiTransaction(); });
Austin Schuh3e3d4ba2019-02-15 23:14:52 -0800807 auto imu_reset = make_unique<frc::DigitalOutput>(1);
Sabina Davisabeae332019-02-01 21:12:57 -0800808 imu.set_reset(imu_reset.get());
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700809 AddLoop(&imu_event_loop);
Sabina Davisabeae332019-02-01 21:12:57 -0800810
811 // While as of 2/9/18 the drivetrain Victors are SPX, it appears as though
812 // they are identical, as far as DrivetrainWriter is concerned, to the SP
813 // variety so all the Victors are written as SPs.
814
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700815 // Thread 5.
Alex Perrycb7da4b2019-08-28 19:35:56 -0700816 ::aos::ShmEventLoop output_event_loop(&config.message());
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700817 ::frc971::wpilib::DrivetrainWriter drivetrain_writer(&output_event_loop);
Sabina Davisd004fd62019-02-02 23:51:46 -0800818 drivetrain_writer.set_left_controller0(
Sabina Davis1b84afa2019-02-09 01:20:21 -0800819 ::std::unique_ptr<::frc::VictorSP>(new ::frc::VictorSP(0)), true);
Sabina Davisd004fd62019-02-02 23:51:46 -0800820 drivetrain_writer.set_right_controller0(
Sabina Davis1b84afa2019-02-09 01:20:21 -0800821 ::std::unique_ptr<::frc::VictorSP>(new ::frc::VictorSP(1)), false);
Sabina Davisabeae332019-02-01 21:12:57 -0800822
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700823 SuperstructureWriter superstructure_writer(&output_event_loop);
Alex Perry5fb5ff22019-02-09 21:53:17 -0800824 superstructure_writer.set_elevator_victor(
Alex Perry5fb5ff22019-02-09 21:53:17 -0800825 ::std::unique_ptr<::frc::VictorSP>(new ::frc::VictorSP(4)));
Austin Schuh3e3d4ba2019-02-15 23:14:52 -0800826 // TODO(austin): Do the vacuum
Austin Schuh461e1182019-02-17 14:56:44 -0800827 superstructure_writer.set_suction_victor(
828 ::std::unique_ptr<::frc::VictorSP>(new ::frc::VictorSP(6)));
Austin Schuh3e3d4ba2019-02-15 23:14:52 -0800829 superstructure_writer.set_intake_victor(
830 ::std::unique_ptr<::frc::VictorSP>(new ::frc::VictorSP(2)));
Alex Perry5fb5ff22019-02-09 21:53:17 -0800831 superstructure_writer.set_wrist_victor(
832 ::std::unique_ptr<::frc::VictorSP>(new ::frc::VictorSP(5)));
833 superstructure_writer.set_stilts_victor(
Austin Schuh3e3d4ba2019-02-15 23:14:52 -0800834 ::std::unique_ptr<::frc::VictorSP>(new ::frc::VictorSP(3)));
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700835 AddLoop(&output_event_loop);
Alex Perry5fb5ff22019-02-09 21:53:17 -0800836
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700837 // Thread 6.
Alex Perrycb7da4b2019-08-28 19:35:56 -0700838 ::aos::ShmEventLoop solenoid_writer_event_loop(&config.message());
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700839 SolenoidWriter solenoid_writer(&solenoid_writer_event_loop);
Austin Schuhc1d6f832019-02-15 23:22:17 -0800840 solenoid_writer.set_intake_roller_talon(
841 make_unique<::ctre::phoenix::motorcontrol::can::TalonSRX>(10));
Austin Schuh461e1182019-02-17 14:56:44 -0800842 solenoid_writer.set_big_suction_cup(0, 1);
843 solenoid_writer.set_small_suction_cup(2, 3);
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700844 AddLoop(&solenoid_writer_event_loop);
Austin Schuhc1d6f832019-02-15 23:22:17 -0800845
Austin Schuhbd1fe9c2019-06-29 16:35:48 -0700846 RunLoops();
Sabina Davisabeae332019-02-01 21:12:57 -0800847 }
848};
849
850} // namespace
851} // namespace wpilib
852} // namespace y2019
853
854AOS_ROBOT_CLASS(::y2019::wpilib::WPILibRobot);