Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 1 | #include "frc971/wpilib/dma_edge_counting.h" |
| 2 | |
John Park | 33858a3 | 2018-09-28 23:05:48 -0700 | [diff] [blame] | 3 | #include "aos/logging/logging.h" |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 4 | |
| 5 | namespace frc971 { |
| 6 | namespace wpilib { |
| 7 | |
Brian Silverman | 552350b | 2015-08-02 18:23:34 -0700 | [diff] [blame] | 8 | bool DMAEdgeCounter::ExtractValue(const DMASample &sample) const { |
Brian Silverman | 03a00cf | 2015-08-29 19:26:54 -0700 | [diff] [blame] | 9 | return sample.Get(input_); |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 10 | } |
| 11 | |
| 12 | void DMAEdgeCounter::UpdateFromSample(const DMASample &sample) { |
Brian Silverman | 03a00cf | 2015-08-29 19:26:54 -0700 | [diff] [blame] | 13 | const bool previous_value = |
Austin Schuh | f83da15 | 2017-03-05 22:28:45 -0800 | [diff] [blame] | 14 | have_prev_sample_ ? ExtractValue(prev_sample_) : previous_polled_value_; |
Brian Silverman | 03a00cf | 2015-08-29 19:26:54 -0700 | [diff] [blame] | 15 | have_prev_sample_ = true; |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 16 | prev_sample_ = sample; |
Brian Silverman | 03a00cf | 2015-08-29 19:26:54 -0700 | [diff] [blame] | 17 | |
| 18 | if (!previous_value && ExtractValue(sample)) { |
| 19 | pos_edge_count_++; |
Brian Silverman | 03a00cf | 2015-08-29 19:26:54 -0700 | [diff] [blame] | 20 | pos_last_encoder_ = sample.GetRaw(encoder_); |
| 21 | } else if (previous_value && !ExtractValue(sample)) { |
| 22 | neg_edge_count_++; |
Brian Silverman | 03a00cf | 2015-08-29 19:26:54 -0700 | [diff] [blame] | 23 | neg_last_encoder_ = sample.GetRaw(encoder_); |
| 24 | } |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 25 | } |
| 26 | |
Austin Schuh | 8e5950d | 2018-03-21 20:29:40 -0700 | [diff] [blame] | 27 | void DMAPulseWidthReader::UpdateFromSample(const DMASample &sample) { |
| 28 | if (have_prev_sample_ && prev_sample_.Get(input_) && !sample.Get(input_)) { |
| 29 | last_width_ = sample.GetTimestamp() - prev_sample_.GetTimestamp(); |
| 30 | } |
| 31 | have_prev_sample_ = true; |
| 32 | prev_sample_ = sample; |
| 33 | } |
| 34 | |
Ravago Jones | 128a50e | 2021-09-18 15:19:39 -0700 | [diff] [blame] | 35 | void DMAPulseSeparationReader::UpdateFromSample(const DMASample &sample) { |
Austin Schuh | 6c053ef | 2021-09-26 14:32:16 -0700 | [diff] [blame] | 36 | // save the time of the falling edge of the input one |
| 37 | if (have_prev_sample_ && !sample.Get(input_one_) && |
| 38 | prev_sample_.Get(input_one_)) { |
Ravago Jones | 128a50e | 2021-09-18 15:19:39 -0700 | [diff] [blame] | 39 | input_one_time_ = sample.GetTimestamp(); |
| 40 | } |
| 41 | |
Austin Schuh | 6c053ef | 2021-09-26 14:32:16 -0700 | [diff] [blame] | 42 | // take the difference in time between the falling edge of the input one and |
| 43 | // the falling edge of the input two |
| 44 | if (!sample.Get(input_two_) && input_one_time_.has_value()) { |
Ravago Jones | 128a50e | 2021-09-18 15:19:39 -0700 | [diff] [blame] | 45 | last_width_ = sample.GetTimestamp() - input_one_time_.value(); |
| 46 | pulses_detected_++; |
| 47 | input_one_time_.reset(); |
| 48 | } |
Austin Schuh | 6c053ef | 2021-09-26 14:32:16 -0700 | [diff] [blame] | 49 | |
| 50 | have_prev_sample_ = true; |
| 51 | prev_sample_ = sample; |
Ravago Jones | 128a50e | 2021-09-18 15:19:39 -0700 | [diff] [blame] | 52 | } |
| 53 | |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 54 | void DMASynchronizer::CheckDMA() { |
| 55 | DMASample current_sample; |
| 56 | |
| 57 | size_t remaining = 0; |
| 58 | while (true) { |
| 59 | switch (dma_->Read(¤t_sample, 0, &remaining)) { |
| 60 | case DMA::STATUS_OK: |
| 61 | for (auto &c : handlers_) { |
| 62 | c->UpdateFromSample(current_sample); |
| 63 | } |
| 64 | |
| 65 | if (remaining == 0) { |
Austin Schuh | f853bde | 2017-11-23 13:23:27 -0800 | [diff] [blame] | 66 | if (sample_time_ < static_cast<int64_t>(current_sample.GetTime())) { |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 67 | // If the latest DMA sample happened after we started polling, then |
| 68 | // just use the values from it because they're more recent. |
| 69 | for (auto &c : handlers_) { |
| 70 | c->PollFromSample(current_sample); |
| 71 | } |
| 72 | } |
| 73 | return; |
| 74 | } |
Austin Schuh | 8e5950d | 2018-03-21 20:29:40 -0700 | [diff] [blame] | 75 | break; |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 76 | case DMA::STATUS_TIMEOUT: |
| 77 | return; |
| 78 | case DMA::STATUS_ERROR: |
Austin Schuh | f257f3c | 2019-10-27 21:00:43 -0700 | [diff] [blame] | 79 | AOS_LOG(WARNING, "DMA read failed\n"); |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 80 | break; |
| 81 | } |
| 82 | } |
| 83 | } |
| 84 | |
| 85 | } // namespace wpilib |
| 86 | } // namespace frc971 |