mirror of
https://github.com/esp8266/Arduino.git
synced 2025-04-19 23:22:16 +03:00
PolledTimeout Class for wrapping millis() loops (WIP) (#5198)
* PolledTimeout Class for wrapping millis() loops * Add yield policies, improve reset, add host tests * Fix copyright, comments * adjust host tests for better time precision * add fuzzyness to timing tests for CI jitter * add blink example with polledTimeout * improve namespace and type naming, add copyright, comments * fix astyle
This commit is contained in:
parent
cd05bae0e8
commit
a501d3ca3b
126
cores/esp8266/PolledTimeout.h
Normal file
126
cores/esp8266/PolledTimeout.h
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
#ifndef __POLLEDTIMING_H__
|
||||||
|
#define __POLLEDTIMING_H__
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
PolledTimeout.h - Encapsulation of a polled Timeout
|
||||||
|
|
||||||
|
Copyright (c) 2018 Daniel Salazar. All rights reserved.
|
||||||
|
This file is part of the esp8266 core for Arduino environment.
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2.1 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This library 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
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with this library; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
namespace esp8266
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
namespace polledTimeout
|
||||||
|
{
|
||||||
|
|
||||||
|
namespace YieldPolicy
|
||||||
|
{
|
||||||
|
|
||||||
|
struct DoNothing
|
||||||
|
{
|
||||||
|
static void execute() {}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct YieldOrSkip
|
||||||
|
{
|
||||||
|
static void execute() {delay(0);}
|
||||||
|
};
|
||||||
|
|
||||||
|
} //YieldPolicy
|
||||||
|
|
||||||
|
|
||||||
|
template <bool PeriodicT, typename YieldPolicyT = YieldPolicy::DoNothing>
|
||||||
|
class timeoutTemplate
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
using timeType = decltype(millis());
|
||||||
|
|
||||||
|
timeoutTemplate(timeType timeout)
|
||||||
|
: _timeout(timeout), _start(millis())
|
||||||
|
{}
|
||||||
|
|
||||||
|
bool expired()
|
||||||
|
{
|
||||||
|
YieldPolicyT::execute(); //in case of DoNothing: gets optimized away
|
||||||
|
if(PeriodicT) //in case of false: gets optimized away
|
||||||
|
return expiredRetrigger();
|
||||||
|
return expiredOneShot();
|
||||||
|
}
|
||||||
|
|
||||||
|
operator bool()
|
||||||
|
{
|
||||||
|
return expired();
|
||||||
|
}
|
||||||
|
|
||||||
|
void reset(timeType newTimeout)
|
||||||
|
{
|
||||||
|
_timeout = newTimeout;
|
||||||
|
reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
void reset()
|
||||||
|
{
|
||||||
|
_start = millis();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
bool checkExpired(timeType t) const
|
||||||
|
{
|
||||||
|
return (t - _start) >= _timeout;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool expiredRetrigger()
|
||||||
|
{
|
||||||
|
timeType current = millis();
|
||||||
|
if(checkExpired(current))
|
||||||
|
{
|
||||||
|
unsigned long n = (current - _start) / _timeout; //how many _timeouts periods have elapsed, will usually be 1 (current - _start >= _timeout)
|
||||||
|
_start += n * _timeout;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool expiredOneShot() const
|
||||||
|
{
|
||||||
|
return checkExpired(millis());
|
||||||
|
}
|
||||||
|
|
||||||
|
timeType _timeout;
|
||||||
|
timeType _start;
|
||||||
|
};
|
||||||
|
|
||||||
|
using oneShot = polledTimeout::timeoutTemplate<false>;
|
||||||
|
using periodic = polledTimeout::timeoutTemplate<true>;
|
||||||
|
|
||||||
|
} //polledTimeout
|
||||||
|
|
||||||
|
|
||||||
|
/* A 1-shot timeout that auto-yields when in CONT can be built as follows:
|
||||||
|
* using oneShotYield = esp8266::polledTimeout::timeoutTemplate<false, esp8266::polledTimeout::YieldPolicy::YieldOrSkip>;
|
||||||
|
*
|
||||||
|
* Other policies can be implemented by the user, e.g.: simple yield that panics in SYS, and the polledTimeout types built as needed as shown above, without modifying this file.
|
||||||
|
*/
|
||||||
|
|
||||||
|
}//esp8266
|
||||||
|
|
||||||
|
#endif
|
@ -0,0 +1,78 @@
|
|||||||
|
/*
|
||||||
|
ESP8266 Blink with polledTimeout by Daniel Salazar
|
||||||
|
|
||||||
|
Copyright (c) 2018 Daniel Salazar. All rights reserved.
|
||||||
|
This file is part of the esp8266 core for Arduino environment.
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2.1 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This library 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
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with this library; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
|
||||||
|
|
||||||
|
Note that this sketch uses LED_BUILTIN to find the pin with the internal LED
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include <PolledTimeout.h>
|
||||||
|
|
||||||
|
void ledOn() {
|
||||||
|
digitalWrite(LED_BUILTIN, LOW); // Turn the LED on (Note that LOW is the voltage level
|
||||||
|
}
|
||||||
|
|
||||||
|
void ledOff() {
|
||||||
|
digitalWrite(LED_BUILTIN, HIGH); // Turn the LED off by making the voltage HIGH
|
||||||
|
}
|
||||||
|
|
||||||
|
void ledToggle() {
|
||||||
|
digitalWrite(LED_BUILTIN, !digitalRead(LED_BUILTIN)); // Change the state of the LED
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
esp8266::polledTimeout::periodic halfPeriod(500); //use fully qualified type and avoid importing all ::esp8266 namespace to the global namespace
|
||||||
|
|
||||||
|
// the setup function runs only once at start
|
||||||
|
void setup() {
|
||||||
|
pinMode(LED_BUILTIN, OUTPUT); // Initialize the LED_BUILTIN pin as an output
|
||||||
|
|
||||||
|
using esp8266::polledTimeout::oneShot; //import the type to the local namespace
|
||||||
|
|
||||||
|
//STEP1; turn the led ON
|
||||||
|
ledOn();
|
||||||
|
|
||||||
|
//STEP2: wait for ON timeout
|
||||||
|
oneShot timeoutOn(2000);
|
||||||
|
while (!timeoutOn) {
|
||||||
|
yield();
|
||||||
|
}
|
||||||
|
|
||||||
|
//STEP3: turn the led OFF
|
||||||
|
ledOff();
|
||||||
|
|
||||||
|
//STEP4: wait for OFF timeout to assure the led is kept off for this time before exiting setup
|
||||||
|
oneShot timeoutOff(2000);
|
||||||
|
while (!timeoutOff) {
|
||||||
|
yield();
|
||||||
|
}
|
||||||
|
|
||||||
|
//Done with STEPs, do other stuff
|
||||||
|
halfPeriod.reset(); //halfPeriod is global, so it gets inited on sketch start. Clear it here to make it ready for loop, where it's actually used.
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// the loop function runs over and over again forever
|
||||||
|
void loop() {
|
||||||
|
if (halfPeriod) {
|
||||||
|
ledToggle();
|
||||||
|
}
|
||||||
|
}
|
@ -107,7 +107,8 @@ TEST_CPP_FILES := \
|
|||||||
fs/test_fs.cpp \
|
fs/test_fs.cpp \
|
||||||
core/test_pgmspace.cpp \
|
core/test_pgmspace.cpp \
|
||||||
core/test_md5builder.cpp \
|
core/test_md5builder.cpp \
|
||||||
core/test_string.cpp
|
core/test_string.cpp \
|
||||||
|
core/test_PolledTimeout.cpp
|
||||||
|
|
||||||
PREINCLUDES := \
|
PREINCLUDES := \
|
||||||
-include common/mock.h \
|
-include common/mock.h \
|
||||||
|
175
tests/host/core/test_PolledTimeout.cpp
Normal file
175
tests/host/core/test_PolledTimeout.cpp
Normal file
@ -0,0 +1,175 @@
|
|||||||
|
#include <catch.hpp>
|
||||||
|
#include "PolledTimeout.h"
|
||||||
|
|
||||||
|
//This won't work for
|
||||||
|
template<typename argT>
|
||||||
|
inline bool
|
||||||
|
fuzzycomp(argT a, argT b)
|
||||||
|
{
|
||||||
|
const argT epsilon = 10;
|
||||||
|
return (std::max(a,b) - std::min(a,b) <= epsilon);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("OneShot Timeout 3000ms", "[polledTimeout]")
|
||||||
|
{
|
||||||
|
using esp8266::polledTimeout::oneShot;
|
||||||
|
using timeType = oneShot::timeType;
|
||||||
|
timeType before, after, delta;
|
||||||
|
|
||||||
|
Serial.println("OneShot Timeout 3000ms");
|
||||||
|
|
||||||
|
oneShot timeout(3000);
|
||||||
|
before = millis();
|
||||||
|
while(!timeout.expired())
|
||||||
|
yield();
|
||||||
|
after = millis();
|
||||||
|
|
||||||
|
delta = after - before;
|
||||||
|
Serial.printf("delta = %lu\n", delta);
|
||||||
|
|
||||||
|
REQUIRE(fuzzycomp(delta, (timeType)3000));
|
||||||
|
|
||||||
|
|
||||||
|
Serial.print("reset\n");
|
||||||
|
|
||||||
|
timeout.reset();
|
||||||
|
before = millis();
|
||||||
|
while(!timeout)
|
||||||
|
yield();
|
||||||
|
after = millis();
|
||||||
|
|
||||||
|
delta = after - before;
|
||||||
|
Serial.printf("delta = %lu\n", delta);
|
||||||
|
|
||||||
|
REQUIRE(fuzzycomp(delta, (timeType)3000));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("OneShot Timeout 3000ms reset to 1000ms", "[polledTimeout]")
|
||||||
|
{
|
||||||
|
using esp8266::polledTimeout::oneShot;
|
||||||
|
using timeType = oneShot::timeType;
|
||||||
|
timeType before, after, delta;
|
||||||
|
|
||||||
|
Serial.println("OneShot Timeout 3000ms");
|
||||||
|
|
||||||
|
oneShot timeout(3000);
|
||||||
|
before = millis();
|
||||||
|
while(!timeout.expired())
|
||||||
|
yield();
|
||||||
|
after = millis();
|
||||||
|
|
||||||
|
delta = after - before;
|
||||||
|
Serial.printf("delta = %lu\n", delta);
|
||||||
|
|
||||||
|
REQUIRE(fuzzycomp(delta, (timeType)3000));
|
||||||
|
|
||||||
|
|
||||||
|
Serial.print("reset\n");
|
||||||
|
|
||||||
|
timeout.reset(1000);
|
||||||
|
before = millis();
|
||||||
|
while(!timeout)
|
||||||
|
yield();
|
||||||
|
after = millis();
|
||||||
|
|
||||||
|
delta = after - before;
|
||||||
|
Serial.printf("delta = %lu\n", delta);
|
||||||
|
|
||||||
|
REQUIRE(fuzzycomp(delta, (timeType)1000));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Periodic Timeout 1T 3000ms", "[polledTimeout]")
|
||||||
|
{
|
||||||
|
using esp8266::polledTimeout::periodic;
|
||||||
|
using timeType = periodic::timeType;
|
||||||
|
timeType before, after, delta;
|
||||||
|
|
||||||
|
Serial.println("Periodic Timeout 1T 3000ms");
|
||||||
|
|
||||||
|
periodic timeout(3000);
|
||||||
|
before = millis();
|
||||||
|
while(!timeout)
|
||||||
|
yield();
|
||||||
|
after = millis();
|
||||||
|
|
||||||
|
delta = after - before;
|
||||||
|
Serial.printf("delta = %lu\n", delta);
|
||||||
|
|
||||||
|
REQUIRE(fuzzycomp(delta, (timeType)3000));
|
||||||
|
|
||||||
|
Serial.print("no reset needed\n");
|
||||||
|
|
||||||
|
before = millis();
|
||||||
|
while(!timeout)
|
||||||
|
yield();
|
||||||
|
after = millis();
|
||||||
|
|
||||||
|
delta = after - before;
|
||||||
|
Serial.printf("delta = %lu\n", delta);
|
||||||
|
|
||||||
|
REQUIRE(fuzzycomp(delta, (timeType)3000));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Periodic Timeout 10T 1000ms", "[polledTimeout]")
|
||||||
|
{
|
||||||
|
using esp8266::polledTimeout::periodic;
|
||||||
|
using timeType = periodic::timeType;
|
||||||
|
timeType before, after, delta;
|
||||||
|
|
||||||
|
Serial.println("Periodic 10T Timeout 1000ms");
|
||||||
|
|
||||||
|
int counter = 10;
|
||||||
|
|
||||||
|
periodic timeout(1000);
|
||||||
|
before = millis();
|
||||||
|
while(1)
|
||||||
|
{
|
||||||
|
if(timeout)
|
||||||
|
{
|
||||||
|
Serial.print("*");
|
||||||
|
if(!--counter)
|
||||||
|
break;
|
||||||
|
yield();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
after = millis();
|
||||||
|
|
||||||
|
delta = after - before;
|
||||||
|
Serial.printf("\ndelta = %lu\n", delta);
|
||||||
|
REQUIRE(fuzzycomp(delta, (timeType)10000));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("OneShot Timeout 3000ms reset to 1000ms custom yield", "[polledTimeout]")
|
||||||
|
{
|
||||||
|
using YieldOrSkipPolicy = esp8266::polledTimeout::YieldPolicy::YieldOrSkip;
|
||||||
|
using oneShotYield = esp8266::polledTimeout::timeoutTemplate<false, YieldOrSkipPolicy>;
|
||||||
|
using timeType = oneShotYield::timeType;
|
||||||
|
timeType before, after, delta;
|
||||||
|
|
||||||
|
Serial.println("OneShot Timeout 3000ms");
|
||||||
|
|
||||||
|
|
||||||
|
oneShotYield timeout(3000);
|
||||||
|
before = millis();
|
||||||
|
while(!timeout.expired());
|
||||||
|
after = millis();
|
||||||
|
|
||||||
|
delta = after - before;
|
||||||
|
Serial.printf("delta = %lu\n", delta);
|
||||||
|
|
||||||
|
REQUIRE(fuzzycomp(delta, (timeType)3000));
|
||||||
|
|
||||||
|
|
||||||
|
Serial.print("reset\n");
|
||||||
|
|
||||||
|
timeout.reset(1000);
|
||||||
|
before = millis();
|
||||||
|
while(!timeout);
|
||||||
|
after = millis();
|
||||||
|
|
||||||
|
delta = after - before;
|
||||||
|
Serial.printf("delta = %lu\n", delta);
|
||||||
|
|
||||||
|
REQUIRE(fuzzycomp(delta, (timeType)1000));
|
||||||
|
}
|
||||||
|
|
Loading…
x
Reference in New Issue
Block a user