/* * © 2020-2022 Harald Barth * * This file is part of CommandStation-EX * * This is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * It is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with CommandStation. If not, see . */ // ATTENTION: this file only compiles on an ESP8266 and ESP32 // On ESP32 we do not even use the functions but they are here for completeness sake // Please refer to DCCTimer.h for general comments about how this class works // This is to avoid repetition and duplication. #ifdef ARDUINO_ARCH_ESP8266 #include "DCCTimer.h" INTERRUPT_CALLBACK interruptHandler=0; void DCCTimer::begin(INTERRUPT_CALLBACK callback) { interruptHandler=callback; timer1_disable(); // There seem to be differnt ways to attach interrupt handler // ETS_FRC_TIMER1_INTR_ATTACH(NULL, NULL); // ETS_FRC_TIMER1_NMI_INTR_ATTACH(interruptHandler); // Let us choose the one from the API timer1_attachInterrupt(interruptHandler); // not exactly sure of order: timer1_enable(TIM_DIV1, TIM_EDGE, TIM_LOOP); timer1_write(CLOCK_CYCLES); } // We do not support to use PWM to make the Waveform on ESP bool IRAM_ATTR DCCTimer::isPWMPin(byte pin) { return false; } void IRAM_ATTR DCCTimer::setPWM(byte pin, bool high) { } void IRAM_ATTR DCCTimer::clearPWM() { } // Fake this as it should not be used void DCCTimer::getSimulatedMacAddress(byte mac[6]) { mac[0] = 0xFE; mac[1] = 0xBE; mac[2] = 0xEF; mac[3] = 0xC0; mac[4] = 0xFF; mac[5] = 0xEE; } volatile int DCCTimer::minimum_free_memory=__INT_MAX__; // Return low memory value... int DCCTimer::getMinimumFreeMemory() { noInterrupts(); // Disable interrupts to get volatile value int retval = minimum_free_memory; interrupts(); return retval; } int DCCTimer::freeMemory() { return ESP.getFreeHeap(); } #endif //////////////////////////////////////////////////////////////////////// #ifdef ARDUINO_ARCH_ESP32 #include "DCCTimer.h" INTERRUPT_CALLBACK interruptHandler=0; // https://www.visualmicro.com/page/Timer-Interrupts-Explained.aspx portMUX_TYPE timerMux = portMUX_INITIALIZER_UNLOCKED; void DCCTimer::begin(INTERRUPT_CALLBACK callback) { interruptHandler = callback; hw_timer_t *timer = NULL; timer = timerBegin(0, 2, true); // prescaler can be 2 to 65536 so choose 2 timerAttachInterrupt(timer, interruptHandler, true); timerAlarmWrite(timer, CLOCK_CYCLES / 6, true); // divide by prescaler*3 (Clockbase is 80Mhz and not F_CPU 240Mhz) timerAlarmEnable(timer); } // We do not support to use PWM to make the Waveform on ESP bool IRAM_ATTR DCCTimer::isPWMPin(byte pin) { return false; } void IRAM_ATTR DCCTimer::setPWM(byte pin, bool high) { } // Fake this as it should not be used void DCCTimer::getSimulatedMacAddress(byte mac[6]) { mac[0] = 0xFE; mac[1] = 0xBE; mac[2] = 0xEF; mac[3] = 0xC0; mac[4] = 0xFF; mac[5] = 0xEE; } volatile int DCCTimer::minimum_free_memory=__INT_MAX__; // Return low memory value... int DCCTimer::getMinimumFreeMemory() { noInterrupts(); // Disable interrupts to get volatile value int retval = minimum_free_memory; interrupts(); return retval; } int DCCTimer::freeMemory() { return ESP.getFreeHeap(); } void DCCTimmer:: reset() { ESP.restart(); } #endif