blob: 1c141a83f6fd293c0a462b3dd8923f5ee0fccdc5 [file] [log] [blame]
Brian Silverman17f503e2015-08-02 18:17:18 -07001#include <stdio.h>
2#include <string.h>
3#include <unistd.h>
4#include <inttypes.h>
5
6#include <thread>
7#include <mutex>
8#include <functional>
9
10#include "aos/common/logging/logging.h"
11#include "aos/common/logging/queue_logging.h"
12#include "aos/common/time.h"
13#include "aos/common/util/log_interval.h"
14#include "aos/common/util/phased_loop.h"
15#include "aos/common/util/wrapping_counter.h"
16#include "aos/common/stl_mutex.h"
17#include "aos/linux_code/init.h"
18#include "aos/common/messages/robot_state.q.h"
19
20#include "y2014/constants.h"
21#include "frc971/control_loops/control_loops.q.h"
22#include "y2014/control_loops/drivetrain/drivetrain.q.h"
23#include "frc971/control_loops/fridge/fridge.q.h"
24#include "y2014/control_loops/claw/claw.q.h"
25
26#include "frc971/wpilib/hall_effect.h"
27#include "frc971/wpilib/joystick_sender.h"
28#include "frc971/wpilib/loop_output_handler.h"
29#include "frc971/wpilib/buffered_solenoid.h"
30#include "frc971/wpilib/buffered_pcm.h"
31#include "frc971/wpilib/gyro_sender.h"
32#include "frc971/wpilib/dma_edge_counting.h"
33#include "frc971/wpilib/interrupt_edge_counting.h"
34#include "frc971/wpilib/encoder_and_potentiometer.h"
35#include "frc971/wpilib/logging.q.h"
36
37#include "Encoder.h"
38#include "Talon.h"
39#include "DriverStation.h"
40#include "AnalogInput.h"
41#include "Compressor.h"
42#include "Relay.h"
43#include "RobotBase.h"
44#include "dma.h"
45#include "ControllerPower.h"
46
47#ifndef M_PI
48#define M_PI 3.14159265358979323846
49#endif
50
51using ::aos::util::SimpleLogInterval;
52using ::frc971::control_loops::drivetrain_queue;
53using ::frc971::control_loops::fridge_queue;
54using ::frc971::control_loops::claw_queue;
55
56namespace frc971 {
57namespace wpilib {
58
59double drivetrain_translate(int32_t in) {
60 return static_cast<double>(in) /
61 (256.0 /*cpr*/ * 4.0 /*4x*/) *
62 constants::GetValues().drivetrain_encoder_ratio *
63 (4 /*wheel diameter*/ * 2.54 / 100.0 * M_PI);
64}
65
66double arm_translate(int32_t in) {
67 return -static_cast<double>(in) /
68 (512.0 /*cpr*/ * 4.0 /*4x*/) *
69 constants::GetValues().arm_encoder_ratio *
70 (2 * M_PI /*radians*/);
71}
72
73double arm_potentiometer_translate(double voltage) {
74 return voltage *
75 constants::GetValues().arm_pot_ratio *
76 (5.0 /*turns*/ / 5.0 /*volts*/) *
77 (2 * M_PI /*radians*/);
78}
79
80double elevator_translate(int32_t in) {
81 return static_cast<double>(in) /
82 (512.0 /*cpr*/ * 4.0 /*4x*/) *
83 constants::GetValues().elev_encoder_ratio *
84 (2 * M_PI /*radians*/) *
85 constants::GetValues().elev_distance_per_radian;
86}
87
88double elevator_potentiometer_translate(double voltage) {
89 return -voltage *
90 constants::GetValues().elev_pot_ratio *
91 (2 * M_PI /*radians*/) *
92 constants::GetValues().elev_distance_per_radian *
93 (5.0 /*turns*/ / 5.0 /*volts*/);
94}
95
96double claw_translate(int32_t in) {
97 return static_cast<double>(in) /
98 (512.0 /*cpr*/ * 4.0 /*4x*/) *
99 constants::GetValues().claw_encoder_ratio *
100 (2 * M_PI /*radians*/);
101}
102
103double claw_potentiometer_translate(double voltage) {
104 return -voltage *
105 constants::GetValues().claw_pot_ratio *
106 (5.0 /*turns*/ / 5.0 /*volts*/) *
107 (2 * M_PI /*radians*/);
108}
109
110static const double kMaximumEncoderPulsesPerSecond =
111 19500.0 /* free speed RPM */ * 12.0 / 56.0 /* belt reduction */ /
112 60.0 /* seconds / minute */ * 256.0 /* CPR */ *
113 4.0 /* index pulse = 1/4 cycle */;
114
115class SensorReader {
116 public:
117 SensorReader() {
118 // Set it to filter out anything shorter than 1/4 of the minimum pulse width
119 // we should ever see.
120 filter_.SetPeriodNanoSeconds(
121 static_cast<int>(1 / 4.0 / kMaximumEncoderPulsesPerSecond * 1e9 + 0.5));
122 }
123
124 void set_arm_left_encoder(::std::unique_ptr<Encoder> encoder) {
125 filter_.Add(encoder.get());
126 arm_left_encoder_.set_encoder(::std::move(encoder));
127 }
128
129 void set_arm_left_index(::std::unique_ptr<DigitalSource> index) {
130 filter_.Add(index.get());
131 arm_left_encoder_.set_index(::std::move(index));
132 }
133
134 void set_arm_left_potentiometer(
135 ::std::unique_ptr<AnalogInput> potentiometer) {
136 arm_left_encoder_.set_potentiometer(::std::move(potentiometer));
137 }
138
139 void set_arm_right_encoder(::std::unique_ptr<Encoder> encoder) {
140 filter_.Add(encoder.get());
141 arm_right_encoder_.set_encoder(::std::move(encoder));
142 }
143
144 void set_arm_right_index(::std::unique_ptr<DigitalSource> index) {
145 filter_.Add(index.get());
146 arm_right_encoder_.set_index(::std::move(index));
147 }
148
149 void set_arm_right_potentiometer(
150 ::std::unique_ptr<AnalogInput> potentiometer) {
151 arm_right_encoder_.set_potentiometer(::std::move(potentiometer));
152 }
153
154 void set_elevator_left_encoder(::std::unique_ptr<Encoder> encoder) {
155 filter_.Add(encoder.get());
156 elevator_left_encoder_.set_encoder(::std::move(encoder));
157 }
158
159 void set_elevator_left_index(::std::unique_ptr<DigitalSource> index) {
160 filter_.Add(index.get());
161 elevator_left_encoder_.set_index(::std::move(index));
162 }
163
164 void set_elevator_left_potentiometer(
165 ::std::unique_ptr<AnalogInput> potentiometer) {
166 elevator_left_encoder_.set_potentiometer(::std::move(potentiometer));
167 }
168
169 void set_elevator_right_encoder(::std::unique_ptr<Encoder> encoder) {
170 filter_.Add(encoder.get());
171 elevator_right_encoder_.set_encoder(::std::move(encoder));
172 }
173
174 void set_elevator_right_index(::std::unique_ptr<DigitalSource> index) {
175 filter_.Add(index.get());
176 elevator_right_encoder_.set_index(::std::move(index));
177 }
178
179 void set_elevator_right_potentiometer(
180 ::std::unique_ptr<AnalogInput> potentiometer) {
181 elevator_right_encoder_.set_potentiometer(::std::move(potentiometer));
182 }
183
184 void set_wrist_encoder(::std::unique_ptr<Encoder> encoder) {
185 filter_.Add(encoder.get());
186 wrist_encoder_.set_encoder(::std::move(encoder));
187 }
188
189 void set_wrist_index(::std::unique_ptr<DigitalSource> index) {
190 filter_.Add(index.get());
191 wrist_encoder_.set_index(::std::move(index));
192 }
193
194 void set_wrist_potentiometer(::std::unique_ptr<AnalogInput> potentiometer) {
195 wrist_encoder_.set_potentiometer(::std::move(potentiometer));
196 }
197
198 void set_left_encoder(::std::unique_ptr<Encoder> left_encoder) {
199 left_encoder_ = ::std::move(left_encoder);
200 }
201
202 void set_right_encoder(::std::unique_ptr<Encoder> right_encoder) {
203 right_encoder_ = ::std::move(right_encoder);
204 }
205
206 // All of the DMA-related set_* calls must be made before this, and it doesn't
207 // hurt to do all of them.
208 void set_dma(::std::unique_ptr<DMA> dma) {
209 dma_synchronizer_.reset(new DMASynchronizer(::std::move(dma)));
210 dma_synchronizer_->Add(&arm_left_encoder_);
211 dma_synchronizer_->Add(&elevator_left_encoder_);
212 dma_synchronizer_->Add(&arm_right_encoder_);
213 dma_synchronizer_->Add(&elevator_right_encoder_);
214 }
215
216 void operator()() {
217 LOG(INFO, "In sensor reader thread\n");
218 ::aos::SetCurrentThreadName("SensorReader");
219
220 my_pid_ = getpid();
221 ds_ = DriverStation::GetInstance();
222
223 wrist_encoder_.Start();
224 dma_synchronizer_->Start();
225 LOG(INFO, "Things are now started\n");
226
227 ::aos::SetCurrentThreadRealtimePriority(kPriority);
228 while (run_) {
229 ::aos::time::PhasedLoopXMS(5, 4000);
230 RunIteration();
231 }
232
233 wrist_encoder_.Stop();
234 }
235
236 void RunIteration() {
237 {
238 auto new_state = ::aos::robot_state.MakeMessage();
239
240 new_state->reader_pid = my_pid_;
241 new_state->outputs_enabled = ds_->IsSysActive();
242 new_state->browned_out = ds_->IsSysBrownedOut();
243
244 new_state->is_3v3_active = ControllerPower::GetEnabled3V3();
245 new_state->is_5v_active = ControllerPower::GetEnabled5V();
246 new_state->voltage_3v3 = ControllerPower::GetVoltage3V3();
247 new_state->voltage_5v = ControllerPower::GetVoltage5V();
248
249 new_state->voltage_roborio_in = ControllerPower::GetInputVoltage();
250 new_state->voltage_battery = ds_->GetBatteryVoltage();
251
252 LOG_STRUCT(DEBUG, "robot_state", *new_state);
253
254 new_state.Send();
255 }
256
257 {
258 auto drivetrain_message = drivetrain_queue.position.MakeMessage();
259 drivetrain_message->right_encoder =
260 -drivetrain_translate(right_encoder_->GetRaw());
261 drivetrain_message->left_encoder =
262 drivetrain_translate(left_encoder_->GetRaw());
263
264 drivetrain_message.Send();
265 }
266
267 dma_synchronizer_->RunIteration();
268
269 const auto &values = constants::GetValues();
270
271 {
272 auto fridge_message = fridge_queue.position.MakeMessage();
273 CopyPotAndIndexPosition(arm_left_encoder_, &fridge_message->arm.left,
274 arm_translate, arm_potentiometer_translate, false,
275 values.fridge.left_arm_potentiometer_offset);
276 CopyPotAndIndexPosition(
277 arm_right_encoder_, &fridge_message->arm.right, arm_translate,
278 arm_potentiometer_translate, true,
279 values.fridge.right_arm_potentiometer_offset);
280 CopyPotAndIndexPosition(
281 elevator_left_encoder_, &fridge_message->elevator.left,
282 elevator_translate, elevator_potentiometer_translate, false,
283 values.fridge.left_elevator_potentiometer_offset);
284 CopyPotAndIndexPosition(
285 elevator_right_encoder_, &fridge_message->elevator.right,
286 elevator_translate, elevator_potentiometer_translate, true,
287 values.fridge.right_elevator_potentiometer_offset);
288 fridge_message.Send();
289 }
290
291 {
292 auto claw_message = claw_queue.position.MakeMessage();
293 CopyPotAndIndexPosition(wrist_encoder_, &claw_message->joint,
294 claw_translate, claw_potentiometer_translate,
295 false, values.claw.potentiometer_offset);
296 claw_message.Send();
297 }
298 }
299
300 void Quit() { run_ = false; }
301
302 private:
303 static const int kPriority = 30;
304 static const int kInterruptPriority = 55;
305
306 int32_t my_pid_;
307 DriverStation *ds_;
308
309 void CopyPotAndIndexPosition(
310 const DMAEncoderAndPotentiometer &encoder, PotAndIndexPosition *position,
311 ::std::function<double(int32_t)> encoder_translate,
312 ::std::function<double(double)> potentiometer_translate, bool reverse,
313 double potentiometer_offset) {
314 const double multiplier = reverse ? -1.0 : 1.0;
315 position->encoder =
316 multiplier * encoder_translate(encoder.polled_encoder_value());
317 position->pot = multiplier * potentiometer_translate(
318 encoder.polled_potentiometer_voltage()) +
319 potentiometer_offset;
320 position->latched_encoder =
321 multiplier * encoder_translate(encoder.last_encoder_value());
322 position->latched_pot =
323 multiplier *
324 potentiometer_translate(encoder.last_potentiometer_voltage()) +
325 potentiometer_offset;
326 position->index_pulses = encoder.index_posedge_count();
327 }
328
329 void CopyPotAndIndexPosition(
330 const InterruptEncoderAndPotentiometer &encoder,
331 PotAndIndexPosition *position,
332 ::std::function<double(int32_t)> encoder_translate,
333 ::std::function<double(double)> potentiometer_translate, bool reverse,
334 double potentiometer_offset) {
335 const double multiplier = reverse ? -1.0 : 1.0;
336 position->encoder =
337 multiplier * encoder_translate(encoder.encoder()->GetRaw());
338 position->pot = multiplier * potentiometer_translate(
339 encoder.potentiometer()->GetVoltage()) +
340 potentiometer_offset;
341 position->latched_encoder =
342 multiplier * encoder_translate(encoder.last_encoder_value());
343 position->latched_pot =
344 multiplier *
345 potentiometer_translate(encoder.last_potentiometer_voltage()) +
346 potentiometer_offset;
347 position->index_pulses = encoder.index_posedge_count();
348 }
349
350 ::std::unique_ptr<DMASynchronizer> dma_synchronizer_;
351
352 DMAEncoderAndPotentiometer arm_left_encoder_, arm_right_encoder_,
353 elevator_left_encoder_, elevator_right_encoder_;
354
355 InterruptEncoderAndPotentiometer wrist_encoder_{kInterruptPriority};
356
357 ::std::unique_ptr<Encoder> left_encoder_;
358 ::std::unique_ptr<Encoder> right_encoder_;
359
360 ::std::atomic<bool> run_{true};
361 DigitalGlitchFilter filter_;
362};
363
364class SolenoidWriter {
365 public:
366 SolenoidWriter(const ::std::unique_ptr<BufferedPcm> &pcm)
367 : pcm_(pcm),
368 fridge_(".frc971.control_loops.fridge_queue.output"),
369 claw_(".frc971.control_loops.claw_queue.output") {}
370
371 void set_pressure_switch(::std::unique_ptr<DigitalSource> pressure_switch) {
372 pressure_switch_ = ::std::move(pressure_switch);
373 }
374
375 void set_compressor_relay(::std::unique_ptr<Relay> compressor_relay) {
376 compressor_relay_ = ::std::move(compressor_relay);
377 }
378
379 void set_fridge_grabbers_top_front(::std::unique_ptr<BufferedSolenoid> s) {
380 fridge_grabbers_top_front_ = ::std::move(s);
381 }
382
383 void set_fridge_grabbers_top_back(::std::unique_ptr<BufferedSolenoid> s) {
384 fridge_grabbers_top_back_ = ::std::move(s);
385 }
386
387 void set_fridge_grabbers_bottom_front(
388 ::std::unique_ptr<BufferedSolenoid> s) {
389 fridge_grabbers_bottom_front_ = ::std::move(s);
390 }
391
392 void set_fridge_grabbers_bottom_back(
393 ::std::unique_ptr<BufferedSolenoid> s) {
394 fridge_grabbers_bottom_back_ = ::std::move(s);
395 }
396
397 void set_claw_pinchers(::std::unique_ptr<BufferedSolenoid> s) {
398 claw_pinchers_ = ::std::move(s);
399 }
400
401 void set_grabber_latch_release(::std::unique_ptr<BufferedSolenoid> s) {
402 grabber_latch_release_ = ::std::move(s);
403 }
404
405 void set_grabber_fold_up(::std::unique_ptr<BufferedSolenoid> s) {
406 grabber_fold_up_ = ::std::move(s);
407 }
408
409 void operator()() {
410 ::aos::SetCurrentThreadName("Solenoids");
411 ::aos::SetCurrentThreadRealtimePriority(30);
412
413 while (run_) {
414 ::aos::time::PhasedLoopXMS(20, 1000);
415
416 {
417 fridge_.FetchLatest();
418 if (fridge_.get()) {
419 LOG_STRUCT(DEBUG, "solenoids", *fridge_);
420 fridge_grabbers_top_front_->Set(!fridge_->grabbers.top_front);
421 fridge_grabbers_top_back_->Set(!fridge_->grabbers.top_back);
422 fridge_grabbers_bottom_front_->Set(!fridge_->grabbers.bottom_front);
423 fridge_grabbers_bottom_back_->Set(!fridge_->grabbers.bottom_back);
424 }
425 }
426
427 {
428 claw_.FetchLatest();
429 if (claw_.get()) {
430 LOG_STRUCT(DEBUG, "solenoids", *claw_);
431 claw_pinchers_->Set(claw_->rollers_closed);
432 }
433 }
434
435 ::aos::joystick_state.FetchLatest();
436 grabber_latch_release_->Set(::aos::joystick_state.get() != nullptr &&
437 ::aos::joystick_state->autonomous);
438 grabber_fold_up_->Set(::aos::joystick_state.get() != nullptr &&
439 ::aos::joystick_state->joysticks[1].buttons & 1);
440
441 {
442 PneumaticsToLog to_log;
443 {
444 const bool compressor_on = !pressure_switch_->Get();
445 to_log.compressor_on = compressor_on;
446 if (compressor_on) {
447 compressor_relay_->Set(Relay::kForward);
448 } else {
449 compressor_relay_->Set(Relay::kOff);
450 }
451 }
452
453 pcm_->Flush();
454 to_log.read_solenoids = pcm_->GetAll();
455 LOG_STRUCT(DEBUG, "pneumatics info", to_log);
456 }
457 }
458 }
459
460 void Quit() { run_ = false; }
461
462 private:
463 const ::std::unique_ptr<BufferedPcm> &pcm_;
464 ::std::unique_ptr<BufferedSolenoid> fridge_grabbers_top_front_;
465 ::std::unique_ptr<BufferedSolenoid> fridge_grabbers_top_back_;
466 ::std::unique_ptr<BufferedSolenoid> fridge_grabbers_bottom_front_;
467 ::std::unique_ptr<BufferedSolenoid> fridge_grabbers_bottom_back_;
468 ::std::unique_ptr<BufferedSolenoid> claw_pinchers_;
469 ::std::unique_ptr<BufferedSolenoid> grabber_latch_release_;
470 ::std::unique_ptr<BufferedSolenoid> grabber_fold_up_;
471 ::std::unique_ptr<DigitalSource> pressure_switch_;
472 ::std::unique_ptr<Relay> compressor_relay_;
473
474 ::aos::Queue<::frc971::control_loops::FridgeQueue::Output> fridge_;
475 ::aos::Queue<::frc971::control_loops::ClawQueue::Output> claw_;
476
477 ::std::atomic<bool> run_{true};
478};
479
480class DrivetrainWriter : public LoopOutputHandler {
481 public:
482 void set_left_drivetrain_talon(::std::unique_ptr<Talon> t) {
483 left_drivetrain_talon_ = ::std::move(t);
484 }
485
486 void set_right_drivetrain_talon(::std::unique_ptr<Talon> t) {
487 right_drivetrain_talon_ = ::std::move(t);
488 }
489
490 private:
491 virtual void Read() override {
492 ::frc971::control_loops::drivetrain_queue.output.FetchAnother();
493 }
494
495 virtual void Write() override {
496 auto &queue = ::frc971::control_loops::drivetrain_queue.output;
497 LOG_STRUCT(DEBUG, "will output", *queue);
498 left_drivetrain_talon_->Set(queue->left_voltage / 12.0);
499 right_drivetrain_talon_->Set(-queue->right_voltage / 12.0);
500 }
501
502 virtual void Stop() override {
503 LOG(WARNING, "drivetrain output too old\n");
504 left_drivetrain_talon_->Disable();
505 right_drivetrain_talon_->Disable();
506 }
507
508 ::std::unique_ptr<Talon> left_drivetrain_talon_;
509 ::std::unique_ptr<Talon> right_drivetrain_talon_;
510};
511
512class FridgeWriter : public LoopOutputHandler {
513 public:
514 void set_left_arm_talon(::std::unique_ptr<Talon> t) {
515 left_arm_talon_ = ::std::move(t);
516 }
517
518 void set_right_arm_talon(::std::unique_ptr<Talon> t) {
519 right_arm_talon_ = ::std::move(t);
520 }
521
522 void set_left_elevator_talon(::std::unique_ptr<Talon> t) {
523 left_elevator_talon_ = ::std::move(t);
524 }
525
526 void set_right_elevator_talon(::std::unique_ptr<Talon> t) {
527 right_elevator_talon_ = ::std::move(t);
528 }
529
530 private:
531 virtual void Read() override {
532 ::frc971::control_loops::fridge_queue.output.FetchAnother();
533 }
534
535 virtual void Write() override {
536 auto &queue = ::frc971::control_loops::fridge_queue.output;
537 LOG_STRUCT(DEBUG, "will output", *queue);
538 left_arm_talon_->Set(queue->left_arm / 12.0);
539 right_arm_talon_->Set(-queue->right_arm / 12.0);
540 left_elevator_talon_->Set(queue->left_elevator / 12.0);
541 right_elevator_talon_->Set(-queue->right_elevator / 12.0);
542 }
543
544 virtual void Stop() override {
545 LOG(WARNING, "Fridge output too old.\n");
546 left_arm_talon_->Disable();
547 right_arm_talon_->Disable();
548 left_elevator_talon_->Disable();
549 right_elevator_talon_->Disable();
550 }
551
552 ::std::unique_ptr<Talon> left_arm_talon_;
553 ::std::unique_ptr<Talon> right_arm_talon_;
554 ::std::unique_ptr<Talon> left_elevator_talon_;
555 ::std::unique_ptr<Talon> right_elevator_talon_;
556};
557
558class ClawWriter : public LoopOutputHandler {
559 public:
560 void set_left_intake_talon(::std::unique_ptr<Talon> t) {
561 left_intake_talon_ = ::std::move(t);
562 }
563
564 void set_right_intake_talon(::std::unique_ptr<Talon> t) {
565 right_intake_talon_ = ::std::move(t);
566 }
567
568 void set_wrist_talon(::std::unique_ptr<Talon> t) {
569 wrist_talon_ = ::std::move(t);
570 }
571
572 private:
573 virtual void Read() override {
574 ::frc971::control_loops::claw_queue.output.FetchAnother();
575 }
576
577 virtual void Write() override {
578 auto &queue = ::frc971::control_loops::claw_queue.output;
579 LOG_STRUCT(DEBUG, "will output", *queue);
580 left_intake_talon_->Set(queue->intake_voltage / 12.0);
581 right_intake_talon_->Set(-queue->intake_voltage / 12.0);
582 wrist_talon_->Set(-queue->voltage / 12.0);
583 }
584
585 virtual void Stop() override {
586 LOG(WARNING, "Claw output too old.\n");
587 left_intake_talon_->Disable();
588 right_intake_talon_->Disable();
589 wrist_talon_->Disable();
590 }
591
592 ::std::unique_ptr<Talon> left_intake_talon_;
593 ::std::unique_ptr<Talon> right_intake_talon_;
594 ::std::unique_ptr<Talon> wrist_talon_;
595};
596
597// TODO(brian): Replace this with ::std::make_unique once all our toolchains
598// have support.
599template <class T, class... U>
600std::unique_ptr<T> make_unique(U &&... u) {
601 return std::unique_ptr<T>(new T(std::forward<U>(u)...));
602}
603
604class WPILibRobot : public RobotBase {
605 public:
606 ::std::unique_ptr<Encoder> encoder(int index) {
607 return make_unique<Encoder>(10 + index * 2, 11 + index * 2, false,
608 Encoder::k4X);
609 }
610 virtual void StartCompetition() {
611 ::aos::InitNRT();
612 ::aos::SetCurrentThreadName("StartCompetition");
613
614 JoystickSender joystick_sender;
615 ::std::thread joystick_thread(::std::ref(joystick_sender));
616 // TODO(austin): Compressor needs to use a spike.
617
618 SensorReader reader;
619 LOG(INFO, "Creating the reader\n");
620 reader.set_arm_left_encoder(encoder(1));
621 reader.set_arm_left_index(make_unique<DigitalInput>(1));
622 reader.set_arm_left_potentiometer(make_unique<AnalogInput>(1));
623
624 reader.set_arm_right_encoder(encoder(5));
625 reader.set_arm_right_index(make_unique<DigitalInput>(5));
626 reader.set_arm_right_potentiometer(make_unique<AnalogInput>(5));
627
628 reader.set_elevator_left_encoder(encoder(0));
629 reader.set_elevator_left_index(make_unique<DigitalInput>(0));
630 reader.set_elevator_left_potentiometer(make_unique<AnalogInput>(0));
631
632 reader.set_elevator_right_encoder(encoder(4));
633 reader.set_elevator_right_index(make_unique<DigitalInput>(4));
634 reader.set_elevator_right_potentiometer(make_unique<AnalogInput>(4));
635
636 reader.set_wrist_encoder(encoder(6));
637 reader.set_wrist_index(make_unique<DigitalInput>(6));
638 reader.set_wrist_potentiometer(make_unique<AnalogInput>(6));
639
640 reader.set_left_encoder(encoder(2));
641 reader.set_right_encoder(encoder(3));
642 reader.set_dma(make_unique<DMA>());
643 ::std::thread reader_thread(::std::ref(reader));
644 GyroSender gyro_sender;
645 ::std::thread gyro_thread(::std::ref(gyro_sender));
646
647 DrivetrainWriter drivetrain_writer;
648 drivetrain_writer.set_left_drivetrain_talon(
649 ::std::unique_ptr<Talon>(new Talon(8)));
650 drivetrain_writer.set_right_drivetrain_talon(
651 ::std::unique_ptr<Talon>(new Talon(0)));
652 ::std::thread drivetrain_writer_thread(::std::ref(drivetrain_writer));
653
654 // TODO(sensors): Get real PWM output and relay numbers for the fridge and
655 // claw.
656 FridgeWriter fridge_writer;
657 fridge_writer.set_left_arm_talon(
658 ::std::unique_ptr<Talon>(new Talon(6)));
659 fridge_writer.set_right_arm_talon(
660 ::std::unique_ptr<Talon>(new Talon(2)));
661 fridge_writer.set_left_elevator_talon(
662 ::std::unique_ptr<Talon>(new Talon(7)));
663 fridge_writer.set_right_elevator_talon(
664 ::std::unique_ptr<Talon>(new Talon(1)));
665 ::std::thread fridge_writer_thread(::std::ref(fridge_writer));
666
667 ClawWriter claw_writer;
668 claw_writer.set_left_intake_talon(
669 ::std::unique_ptr<Talon>(new Talon(5)));
670 claw_writer.set_right_intake_talon(
671 ::std::unique_ptr<Talon>(new Talon(3)));
672 claw_writer.set_wrist_talon(
673 ::std::unique_ptr<Talon>(new Talon(4)));
674 ::std::thread claw_writer_thread(::std::ref(claw_writer));
675
676 ::std::unique_ptr<::frc971::wpilib::BufferedPcm> pcm(
677 new ::frc971::wpilib::BufferedPcm());
678 SolenoidWriter solenoid_writer(pcm);
679 solenoid_writer.set_fridge_grabbers_top_front(pcm->MakeSolenoid(0));
680 solenoid_writer.set_fridge_grabbers_top_back(pcm->MakeSolenoid(0));
681 solenoid_writer.set_fridge_grabbers_bottom_front(pcm->MakeSolenoid(2));
682 solenoid_writer.set_fridge_grabbers_bottom_back(pcm->MakeSolenoid(1));
683 solenoid_writer.set_claw_pinchers(pcm->MakeSolenoid(4));
684 solenoid_writer.set_grabber_latch_release(pcm->MakeSolenoid(7));
685 solenoid_writer.set_grabber_fold_up(pcm->MakeSolenoid(5));
686
687 solenoid_writer.set_pressure_switch(make_unique<DigitalInput>(9));
688 solenoid_writer.set_compressor_relay(make_unique<Relay>(0));
689 ::std::thread solenoid_thread(::std::ref(solenoid_writer));
690
691 // Wait forever. Not much else to do...
692 PCHECK(select(0, nullptr, nullptr, nullptr, nullptr));
693
694 LOG(ERROR, "Exiting WPILibRobot\n");
695
696 joystick_sender.Quit();
697 joystick_thread.join();
698 reader.Quit();
699 reader_thread.join();
700 gyro_sender.Quit();
701 gyro_thread.join();
702
703 drivetrain_writer.Quit();
704 drivetrain_writer_thread.join();
705 solenoid_writer.Quit();
706 solenoid_thread.join();
707
708 ::aos::Cleanup();
709 }
710};
711
712} // namespace wpilib
713} // namespace frc971
714
715
716START_ROBOT_CLASS(::frc971::wpilib::WPILibRobot);