mirror of https://github.com/ARMmbed/mbed-os.git
177 lines
6.5 KiB
C
177 lines
6.5 KiB
C
/* mbed Microcontroller Library
|
|
* Copyright (c) 2019-2020, Arm Limited and affiliates.
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* 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.
|
|
*/
|
|
#include "cmsis.h"
|
|
#include "us_ticker_api.h"
|
|
#include "mbed_error.h"
|
|
#include "cy_us_ticker.h"
|
|
#include "cyhal_timer.h"
|
|
#include "cy_tcpwm_counter.h"
|
|
|
|
#define CY_US_TICKER_IRQ_PRIORITY 3
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
static ticker_info_t cy_us_ticker_info;
|
|
static cyhal_timer_t cy_us_timer;
|
|
static bool cy_us_ticker_initialized = false;
|
|
|
|
static cy_en_syspm_status_t cy_us_ticker_pm_callback(cy_stc_syspm_callback_params_t *params, cy_en_syspm_callback_mode_t mode)
|
|
{
|
|
if (mode == CY_SYSPM_AFTER_TRANSITION) {
|
|
Cy_TCPWM_Counter_Enable(cy_us_timer.base, cy_us_timer.resource.channel_num);
|
|
Cy_TCPWM_TriggerStart(cy_us_timer.base, 1u << cy_us_timer.resource.channel_num);
|
|
}
|
|
return CY_SYSPM_SUCCESS;
|
|
}
|
|
|
|
static cy_stc_syspm_callback_params_t cy_us_ticker_pm_params;
|
|
static cy_stc_syspm_callback_t cy_us_ticker_pm_data = {
|
|
.callback = &cy_us_ticker_pm_callback,
|
|
.type = CY_SYSPM_DEEPSLEEP,
|
|
.callbackParams = &cy_us_ticker_pm_params,
|
|
};
|
|
|
|
static void cy_us_ticker_irq_handler(MBED_UNUSED void *arg, MBED_UNUSED cyhal_timer_event_t event)
|
|
{
|
|
us_ticker_irq_handler();
|
|
}
|
|
|
|
void cy_us_ticker_start()
|
|
{
|
|
cyhal_timer_start(&cy_us_timer);
|
|
}
|
|
|
|
void cy_us_ticker_stop()
|
|
{
|
|
cyhal_timer_stop(&cy_us_timer);
|
|
}
|
|
|
|
void us_ticker_init(void)
|
|
{
|
|
if (!cy_us_ticker_initialized) {
|
|
|
|
#ifdef TARGET_TFM
|
|
/* There are two timers, Timer0 and Timer1, available on the PSoC64.
|
|
* Timer0 has 8 channels and Timer1 has 24 channels. TF-M regression
|
|
* tests make use of Timer0 Channel 1 and Timer0 Channel 2. Therefore,
|
|
* reserve the timer channels used by TF-M. This approach can be
|
|
* replaced once we have a way to allocate dedicated timers for TF-M
|
|
* and Mbed OS. */
|
|
cyhal_resource_inst_t res = { CYHAL_RSC_TCPWM, 0, 0 };
|
|
if(CY_RSLT_SUCCESS != cyhal_hwmgr_reserve(&res)) {
|
|
MBED_ERROR(MBED_MAKE_ERROR(MBED_MODULE_DRIVER, MBED_ERROR_CODE_FAILED_OPERATION), "cyhal_timer_init");
|
|
}
|
|
res.channel_num = 1;
|
|
if(CY_RSLT_SUCCESS != cyhal_hwmgr_reserve(&res)) {
|
|
MBED_ERROR(MBED_MAKE_ERROR(MBED_MODULE_DRIVER, MBED_ERROR_CODE_FAILED_OPERATION), "cyhal_timer_init");
|
|
}
|
|
#endif
|
|
if (CY_RSLT_SUCCESS != cyhal_timer_init(&cy_us_timer, NC, NULL)) {
|
|
MBED_ERROR(MBED_MAKE_ERROR(MBED_MODULE_DRIVER, MBED_ERROR_CODE_FAILED_OPERATION), "cyhal_timer_init");
|
|
}
|
|
// Configure clock for about 1 MHz
|
|
MBED_ASSERT(cy_PeriClkFreqHz >= 1000000);
|
|
uint32_t div_value = cy_PeriClkFreqHz / 1000000;
|
|
cy_us_ticker_info.frequency = cy_PeriClkFreqHz / div_value;
|
|
cy_us_ticker_info.bits = CYHAL_TCPWM_DATA[cy_us_timer.resource.block_num].max_count;
|
|
Cy_SysClk_PeriphSetDivider(cy_us_timer.clock.div_type, cy_us_timer.clock.div_num, div_value - 1u);
|
|
const cyhal_timer_cfg_t cfg = {
|
|
.is_continuous = true,
|
|
.direction = CYHAL_TIMER_DIR_UP,
|
|
.is_compare = true,
|
|
.period = (1 << cy_us_ticker_info.bits) - 1,
|
|
.compare_value = (1 << cy_us_ticker_info.bits) - 1,
|
|
.value = 0u
|
|
};
|
|
if (CY_RSLT_SUCCESS != cyhal_timer_configure(&cy_us_timer, &cfg)) {
|
|
MBED_ERROR(MBED_MAKE_ERROR(MBED_MODULE_DRIVER, MBED_ERROR_CODE_FAILED_OPERATION), "cyhal_timer_set_cfg");
|
|
}
|
|
Cy_TCPWM_ClearInterrupt(cy_us_timer.base, cy_us_timer.resource.channel_num, CY_TCPWM_INT_ON_CC_OR_TC);
|
|
cyhal_timer_register_callback(&cy_us_timer, &cy_us_ticker_irq_handler, NULL);
|
|
if (CY_RSLT_SUCCESS != cyhal_timer_start(&cy_us_timer)) {
|
|
MBED_ERROR(MBED_MAKE_ERROR(MBED_MODULE_DRIVER, MBED_ERROR_CODE_FAILED_OPERATION), "cyhal_timer_start");
|
|
}
|
|
if (!Cy_SysPm_RegisterCallback(&cy_us_ticker_pm_data)) {
|
|
MBED_ERROR(MBED_MAKE_ERROR(MBED_MODULE_DRIVER, MBED_ERROR_CODE_FAILED_OPERATION), "Cy_SysPm_RegisterCallback");
|
|
}
|
|
cy_us_ticker_initialized = true;
|
|
}
|
|
us_ticker_disable_interrupt();
|
|
}
|
|
|
|
void us_ticker_free(void)
|
|
{
|
|
if (cy_us_ticker_initialized) {
|
|
cy_us_ticker_initialized = false;
|
|
us_ticker_disable_interrupt();
|
|
Cy_SysPm_UnregisterCallback(&cy_us_ticker_pm_data);
|
|
if (CY_RSLT_SUCCESS != cyhal_timer_stop(&cy_us_timer)) {
|
|
MBED_ERROR(MBED_MAKE_ERROR(MBED_MODULE_DRIVER, MBED_ERROR_CODE_FAILED_OPERATION), "cyhal_timer_stop");
|
|
}
|
|
cyhal_timer_free(&cy_us_timer);
|
|
}
|
|
}
|
|
|
|
uint32_t us_ticker_read(void)
|
|
{
|
|
MBED_ASSERT(cy_us_ticker_initialized);
|
|
return Cy_TCPWM_Counter_GetCounter(cy_us_timer.base, cy_us_timer.resource.channel_num);
|
|
}
|
|
|
|
void us_ticker_set_interrupt(timestamp_t timestamp)
|
|
{
|
|
MBED_ASSERT(cy_us_ticker_initialized);
|
|
Cy_TCPWM_Counter_SetCompare0(cy_us_timer.base, cy_us_timer.resource.channel_num, timestamp);
|
|
if (CY_TCPWM_INT_NONE == Cy_TCPWM_GetInterruptMask(cy_us_timer.base, cy_us_timer.resource.channel_num)) {
|
|
Cy_TCPWM_ClearInterrupt(cy_us_timer.base, cy_us_timer.resource.channel_num, CY_TCPWM_INT_ON_CC_OR_TC);
|
|
cyhal_timer_enable_event(&cy_us_timer, CYHAL_TIMER_IRQ_CAPTURE_COMPARE, CY_US_TICKER_IRQ_PRIORITY, true);
|
|
}
|
|
}
|
|
|
|
void us_ticker_disable_interrupt(void)
|
|
{
|
|
MBED_ASSERT(cy_us_ticker_initialized);
|
|
cyhal_timer_enable_event(&cy_us_timer, CYHAL_TIMER_IRQ_CAPTURE_COMPARE, CY_US_TICKER_IRQ_PRIORITY, false);
|
|
}
|
|
|
|
void us_ticker_clear_interrupt(void)
|
|
{
|
|
// Nothing to do. cyhal_timer functions clear the interrupt.
|
|
}
|
|
|
|
void us_ticker_fire_interrupt(void)
|
|
{
|
|
MBED_ASSERT(cy_us_ticker_initialized);
|
|
// TODO: no HAL function for this. Needs to work even when masked
|
|
IRQn_Type irq = cy_us_timer.resource.block_num == 0
|
|
? tcpwm_0_interrupts_0_IRQn : tcpwm_1_interrupts_0_IRQn;
|
|
irq = (IRQn_Type)(irq + cy_us_timer.resource.channel_num);
|
|
NVIC_SetPendingIRQ(irq);
|
|
}
|
|
|
|
const ticker_info_t *us_ticker_get_info(void)
|
|
{
|
|
MBED_ASSERT(cy_us_ticker_initialized);
|
|
return &cy_us_ticker_info;
|
|
}
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|