mirror of https://github.com/ARMmbed/mbed-os.git
commit
4dff32ab5a
|
@ -0,0 +1,132 @@
|
||||||
|
|
||||||
|
/* mbed Microcontroller Library
|
||||||
|
* Copyright (c) 2017 ARM Limited
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if !DEVICE_SLEEP
|
||||||
|
#error [NOT_SUPPORTED] Sleep not supported for this target
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "utest/utest.h"
|
||||||
|
#include "unity/unity.h"
|
||||||
|
#include "greentea-client/test_env.h"
|
||||||
|
|
||||||
|
#include "mbed.h"
|
||||||
|
|
||||||
|
using namespace utest::v1;
|
||||||
|
|
||||||
|
void deep_sleep_lock_lock_test()
|
||||||
|
{
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
{
|
||||||
|
// Check basic usage works
|
||||||
|
DeepSleepLock lock;
|
||||||
|
TEST_ASSERT_EQUAL(false, sleep_manager_can_deep_sleep());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
{
|
||||||
|
// Check that unlock and lock change can deep sleep as expected
|
||||||
|
DeepSleepLock lock;
|
||||||
|
TEST_ASSERT_EQUAL(false, sleep_manager_can_deep_sleep());
|
||||||
|
lock.unlock();
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
lock.lock();
|
||||||
|
TEST_ASSERT_EQUAL(false, sleep_manager_can_deep_sleep());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
{
|
||||||
|
// Check that unlock releases sleep based on count
|
||||||
|
DeepSleepLock lock;
|
||||||
|
lock.lock();
|
||||||
|
lock.lock();
|
||||||
|
lock.unlock();
|
||||||
|
TEST_ASSERT_EQUAL(false, sleep_manager_can_deep_sleep());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
{
|
||||||
|
// Check that unbalanced locks do not leave deep sleep locked
|
||||||
|
DeepSleepLock lock;
|
||||||
|
lock.lock();
|
||||||
|
TEST_ASSERT_EQUAL(false, sleep_manager_can_deep_sleep());
|
||||||
|
}
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void timer_lock_test()
|
||||||
|
{
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
{
|
||||||
|
// Just creating a timer object does not lock sleep
|
||||||
|
Timer timer;
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
{
|
||||||
|
// Starting a timer does lock sleep
|
||||||
|
Timer timer;
|
||||||
|
timer.start();
|
||||||
|
TEST_ASSERT_EQUAL(false, sleep_manager_can_deep_sleep());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
{
|
||||||
|
// Stopping a timer after starting it allows sleep
|
||||||
|
Timer timer;
|
||||||
|
timer.start();
|
||||||
|
timer.stop();
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
{
|
||||||
|
// Starting a timer multiple times still lets you sleep
|
||||||
|
Timer timer;
|
||||||
|
timer.start();
|
||||||
|
timer.start();
|
||||||
|
}
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
{
|
||||||
|
// Stopping a timer multiple times still lets you sleep
|
||||||
|
Timer timer;
|
||||||
|
timer.start();
|
||||||
|
timer.stop();
|
||||||
|
timer.stop();
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
}
|
||||||
|
TEST_ASSERT_EQUAL(true, sleep_manager_can_deep_sleep());
|
||||||
|
}
|
||||||
|
|
||||||
|
Case cases[] = {
|
||||||
|
Case("DeepSleepLock lock test", deep_sleep_lock_lock_test),
|
||||||
|
Case("timer lock test", timer_lock_test),
|
||||||
|
};
|
||||||
|
|
||||||
|
utest::v1::status_t greentea_test_setup(const size_t number_of_cases) {
|
||||||
|
GREENTEA_SETUP(20, "default_auto");
|
||||||
|
return greentea_test_setup_handler(number_of_cases);
|
||||||
|
}
|
||||||
|
|
||||||
|
Specification specification(greentea_test_setup, cases, greentea_test_teardown_handler);
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
Harness::run(specification);
|
||||||
|
}
|
|
@ -46,6 +46,11 @@ CAN::CAN(PinName rd, PinName td, int hz) : _can(), _irq() {
|
||||||
|
|
||||||
CAN::~CAN() {
|
CAN::~CAN() {
|
||||||
// No lock needed in destructor
|
// No lock needed in destructor
|
||||||
|
|
||||||
|
// Detaching interrupts releases the sleep lock if it was locked
|
||||||
|
for (int irq = 0; irq < IrqCnt; irq++) {
|
||||||
|
attach(NULL, (IrqType)irq);
|
||||||
|
}
|
||||||
can_irq_free(&_can);
|
can_irq_free(&_can);
|
||||||
can_free(&_can);
|
can_free(&_can);
|
||||||
}
|
}
|
||||||
|
|
|
@ -133,6 +133,16 @@ void SerialBase:: unlock() {
|
||||||
// Stub
|
// Stub
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SerialBase::~SerialBase()
|
||||||
|
{
|
||||||
|
// No lock needed in destructor
|
||||||
|
|
||||||
|
// Detaching interrupts releases the sleep lock if it was locked
|
||||||
|
for (int irq = 0; irq < IrqCnt; irq++) {
|
||||||
|
attach(NULL, (IrqType)irq);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#if DEVICE_SERIAL_FC
|
#if DEVICE_SERIAL_FC
|
||||||
void SerialBase::set_flow_control(Flow type, PinName flow1, PinName flow2) {
|
void SerialBase::set_flow_control(Flow type, PinName flow1, PinName flow2) {
|
||||||
lock();
|
lock();
|
||||||
|
|
|
@ -241,8 +241,7 @@ protected:
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
SerialBase(PinName tx, PinName rx, int baud);
|
SerialBase(PinName tx, PinName rx, int baud);
|
||||||
virtual ~SerialBase() {
|
virtual ~SerialBase();
|
||||||
}
|
|
||||||
|
|
||||||
int _base_getc();
|
int _base_getc();
|
||||||
int _base_putc(int c);
|
int _base_putc(int c);
|
||||||
|
|
|
@ -22,6 +22,7 @@
|
||||||
#include "platform/NonCopyable.h"
|
#include "platform/NonCopyable.h"
|
||||||
#include "platform/mbed_sleep.h"
|
#include "platform/mbed_sleep.h"
|
||||||
#include "hal/lp_ticker_api.h"
|
#include "hal/lp_ticker_api.h"
|
||||||
|
#include "platform/mbed_critical.h"
|
||||||
|
|
||||||
namespace mbed {
|
namespace mbed {
|
||||||
/** \addtogroup drivers */
|
/** \addtogroup drivers */
|
||||||
|
@ -113,12 +114,14 @@ public:
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
void attach_us(Callback<void()> func, us_timestamp_t t) {
|
void attach_us(Callback<void()> func, us_timestamp_t t) {
|
||||||
|
core_util_critical_section_enter();
|
||||||
// lock only for the initial callback setup and this is not low power ticker
|
// lock only for the initial callback setup and this is not low power ticker
|
||||||
if(!_function && _lock_deepsleep) {
|
if(!_function && _lock_deepsleep) {
|
||||||
sleep_manager_lock_deep_sleep();
|
sleep_manager_lock_deep_sleep();
|
||||||
}
|
}
|
||||||
_function = func;
|
_function = func;
|
||||||
setup(t);
|
setup(t);
|
||||||
|
core_util_critical_section_exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Attach a member function to be called by the Ticker, specifying the interval in micro-seconds
|
/** Attach a member function to be called by the Ticker, specifying the interval in micro-seconds
|
||||||
|
|
|
@ -16,7 +16,9 @@
|
||||||
#ifndef MBED_DEEPSLEEPLOCK_H
|
#ifndef MBED_DEEPSLEEPLOCK_H
|
||||||
#define MBED_DEEPSLEEPLOCK_H
|
#define MBED_DEEPSLEEPLOCK_H
|
||||||
|
|
||||||
|
#include <limits.h>
|
||||||
#include "platform/mbed_sleep.h"
|
#include "platform/mbed_sleep.h"
|
||||||
|
#include "platform/mbed_critical.h"
|
||||||
|
|
||||||
namespace mbed {
|
namespace mbed {
|
||||||
|
|
||||||
|
@ -36,29 +38,47 @@ namespace mbed {
|
||||||
* @endcode
|
* @endcode
|
||||||
*/
|
*/
|
||||||
class DeepSleepLock {
|
class DeepSleepLock {
|
||||||
|
private:
|
||||||
|
uint16_t _lock_count;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
DeepSleepLock()
|
DeepSleepLock(): _lock_count(1)
|
||||||
{
|
{
|
||||||
sleep_manager_lock_deep_sleep();
|
sleep_manager_lock_deep_sleep();
|
||||||
}
|
}
|
||||||
|
|
||||||
~DeepSleepLock()
|
~DeepSleepLock()
|
||||||
{
|
{
|
||||||
sleep_manager_unlock_deep_sleep();
|
if (_lock_count) {
|
||||||
|
sleep_manager_unlock_deep_sleep();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Mark the start of a locked deep sleep section
|
/** Mark the start of a locked deep sleep section
|
||||||
*/
|
*/
|
||||||
void lock()
|
void lock()
|
||||||
{
|
{
|
||||||
sleep_manager_lock_deep_sleep();
|
uint16_t count = core_util_atomic_incr_u16(&_lock_count, 1);
|
||||||
|
if (1 == count) {
|
||||||
|
sleep_manager_lock_deep_sleep();
|
||||||
|
}
|
||||||
|
if (0 == count) {
|
||||||
|
error("DeepSleepLock overflow (> USHRT_MAX)");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Mark the end of a locked deep sleep section
|
/** Mark the end of a locked deep sleep section
|
||||||
*/
|
*/
|
||||||
void unlock()
|
void unlock()
|
||||||
{
|
{
|
||||||
sleep_manager_unlock_deep_sleep();
|
uint16_t count = core_util_atomic_decr_u16(&_lock_count, 1);
|
||||||
|
if (count == 0) {
|
||||||
|
sleep_manager_unlock_deep_sleep();
|
||||||
|
}
|
||||||
|
if (count == USHRT_MAX) {
|
||||||
|
core_util_critical_section_exit();
|
||||||
|
error("DeepSleepLock underflow (< 0)");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue