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 | |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 35 | void DMASynchronizer::CheckDMA() { |
| 36 | DMASample current_sample; |
| 37 | |
| 38 | size_t remaining = 0; |
| 39 | while (true) { |
| 40 | switch (dma_->Read(¤t_sample, 0, &remaining)) { |
| 41 | case DMA::STATUS_OK: |
| 42 | for (auto &c : handlers_) { |
| 43 | c->UpdateFromSample(current_sample); |
| 44 | } |
| 45 | |
| 46 | if (remaining == 0) { |
Austin Schuh | f853bde | 2017-11-23 13:23:27 -0800 | [diff] [blame] | 47 | if (sample_time_ < static_cast<int64_t>(current_sample.GetTime())) { |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 48 | // If the latest DMA sample happened after we started polling, then |
| 49 | // just use the values from it because they're more recent. |
| 50 | for (auto &c : handlers_) { |
| 51 | c->PollFromSample(current_sample); |
| 52 | } |
| 53 | } |
| 54 | return; |
| 55 | } |
Austin Schuh | 8e5950d | 2018-03-21 20:29:40 -0700 | [diff] [blame] | 56 | break; |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 57 | case DMA::STATUS_TIMEOUT: |
| 58 | return; |
| 59 | case DMA::STATUS_ERROR: |
Austin Schuh | f257f3c | 2019-10-27 21:00:43 -0700 | [diff] [blame] | 60 | AOS_LOG(WARNING, "DMA read failed\n"); |
Brian Silverman | ff7b347 | 2015-01-26 17:53:04 -0500 | [diff] [blame] | 61 | break; |
| 62 | } |
| 63 | } |
| 64 | } |
| 65 | |
| 66 | } // namespace wpilib |
| 67 | } // namespace frc971 |