/* ---------------------------- Original copyright ----------------------------- * * Encoder Library, for measuring quadrature encoded signals * http://www.pjrc.com/teensy/td_libs_Encoder.html * Copyright (c) 2011,2013 PJRC.COM, LLC - Paul Stoffregen <paul@pjrc.com> * * Version 1.2 - fix -2 bug in C-only code * Version 1.1 - expand to support boards with up to 60 interrupts * Version 1.0 - initial release * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. * * ----------------------------------------------------------------------------- * * Encoder.h, updated for ESP8266 use in ESPurna. Other hardware is not supported. * * - Added ESP-specific attributes to ISR handlers to place them in IRAM. * - Reduced per-encoder structure sizes - only 5 Encoders can be used on ESP8266, * and we can directly reference pin number instead of storing both register and bitmask * */ #pragma once // _______ _______ // Pin1 ______| |_______| |______ Pin1 // negative <--- _______ _______ __ --> positive // Pin2 __| |_______| |_______| Pin2 // new new old old // pin2 pin1 pin2 pin1 Result // ---- ---- ---- ---- ------ // 0 0 0 0 no movement // 0 0 0 1 +1 // 0 0 1 0 -1 // 0 0 1 1 +2 (assume pin1 edges only) // 0 1 0 0 -1 // 0 1 0 1 no movement // 0 1 1 0 -2 (assume pin1 edges only) // 0 1 1 1 +1 // 1 0 0 0 +1 // 1 0 0 1 -2 (assume pin1 edges only) // 1 0 1 0 no movement // 1 0 1 1 -1 // 1 1 0 0 +2 (assume pin1 edges only) // 1 1 0 1 -1 // 1 1 1 0 +1 // 1 1 1 1 no movement namespace EncoderLibrary { typedef struct { uint8_t pin1; uint8_t pin2; uint8_t state; int32_t position; } encoder_values_t; constexpr const unsigned char ENCODERS_MAXIMUM {5u}; encoder_values_t * EncoderValues[ENCODERS_MAXIMUM] = {nullptr}; uint8_t _encoderFindStorage() { for (uint8_t i = 0; i < ENCODERS_MAXIMUM; i++) { if (EncoderValues[i] == nullptr) { return i; } } return ENCODERS_MAXIMUM; } void _encoderCleanStorage(uint8_t pin1, uint8_t pin2) { for (uint8_t i = 0; i < ENCODERS_MAXIMUM; i++) { if (EncoderValues[i] == nullptr) continue; if (((EncoderValues[i])->pin1 == pin1) && ((EncoderValues[i])->pin2 == pin2)) { EncoderValues[i] = nullptr; break; } } } // update() is not meant to be called from outside Encoder, // but it is public to allow static interrupt routines. void ICACHE_RAM_ATTR update(encoder_values_t *target) { uint8_t p1val = GPIP(target->pin1); uint8_t p2val = GPIP(target->pin2); uint8_t state = target->state & 3; if (p1val) state |= 4; if (p2val) state |= 8; target->state = (state >> 2); switch (state) { case 1: case 7: case 8: case 14: target->position++; return; case 2: case 4: case 11: case 13: target->position--; return; case 3: case 12: target->position += 2; return; case 6: case 9: target->position -= 2; return; } } // 2 pins per encoder, 1 isr per encoder void ICACHE_RAM_ATTR isr0() { update(EncoderValues[0]); } void ICACHE_RAM_ATTR isr1() { update(EncoderValues[1]); } void ICACHE_RAM_ATTR isr2() { update(EncoderValues[2]); } void ICACHE_RAM_ATTR isr3() { update(EncoderValues[3]); } void ICACHE_RAM_ATTR isr4() { update(EncoderValues[4]); } constexpr void (*_isr_funcs[5])() = { isr0, isr1, isr2, isr3, isr4 }; class Encoder { private: encoder_values_t values; public: Encoder(uint8_t pin1, uint8_t pin2) { values.pin1 = pin1; values.pin2 = pin2; pinMode(values.pin1, INPUT_PULLUP); pinMode(values.pin2, INPUT_PULLUP); values.position = 0; // allow time for a passive R-C filter to charge // through the pullup resistors, before reading // the initial state delayMicroseconds(2000); uint8_t current = 0; if (GPIP(values.pin1)) { current |= 1; } if (GPIP(values.pin2)) { current |= 2; } values.state = current; attach(); } ~Encoder() { detach(); } uint8_t pin1() { return values.pin1; } uint8_t pin2() { return values.pin2; } int32_t read() { noInterrupts(); update(&values); int32_t ret = values.position; interrupts(); return ret; } void write(int32_t position) { noInterrupts(); values.position = position; interrupts(); } bool attach() { uint8_t index = _encoderFindStorage(); if (index >= ENCODERS_MAXIMUM) return false; EncoderValues[index] = &values; attachInterrupt(values.pin1, _isr_funcs[index], CHANGE); attachInterrupt(values.pin2, _isr_funcs[index], CHANGE); return true; } void detach() { noInterrupts(); _encoderCleanStorage(values.pin1, values.pin2); detachInterrupt(values.pin1); detachInterrupt(values.pin2); interrupts(); } }; } using EncoderLibrary::Encoder;