You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
202 lines
5.4 KiB
202 lines
5.4 KiB
4 years ago
|
/**
|
||
|
* \file
|
||
|
*
|
||
|
* \brief RTC Driver (Calendar Mode)
|
||
|
*
|
||
|
* Copyright (C) 2014-2017 Atmel Corporation. All rights reserved.
|
||
|
*
|
||
|
* \asf_license_start
|
||
|
*
|
||
|
* \page License
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions are met:
|
||
|
*
|
||
|
* 1. Redistributions of source code must retain the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer.
|
||
|
*
|
||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer in the documentation
|
||
|
* and/or other materials provided with the distribution.
|
||
|
*
|
||
|
* 3. The name of Atmel may not be used to endorse or promote products derived
|
||
|
* from this software without specific prior written permission.
|
||
|
*
|
||
|
* 4. This software may only be redistributed and used in connection with an
|
||
|
* Atmel microcontroller product.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR IMPLIED
|
||
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||
|
* EXPRESSLY AND SPECIFICALLY DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR
|
||
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
||
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
||
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
||
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||
|
* POSSIBILITY OF SUCH DAMAGE.
|
||
|
*
|
||
|
* \asf_license_stop
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include "hpl_calendar.h"
|
||
|
#include "hpl_timer.h"
|
||
|
#include <hpl_rtc_config.h>
|
||
|
#include <utils_assert.h>
|
||
|
|
||
|
/*!< Pointer to hpl device */
|
||
|
|
||
|
static struct _timer_device *_rtc_dev = NULL;
|
||
|
|
||
|
/**
|
||
|
* \brief Initialize Timer
|
||
|
*/
|
||
|
int32_t _timer_init(struct _timer_device *const dev, void *const hw)
|
||
|
{
|
||
|
ASSERT(dev);
|
||
|
|
||
|
uint16_t register_value;
|
||
|
dev->hw = hw;
|
||
|
|
||
|
hri_rtcmode0_write_CTRL_reg(dev->hw, RTC_MODE0_CTRL_SWRST);
|
||
|
hri_rtcmode0_wait_for_sync(dev->hw);
|
||
|
|
||
|
/* Set mode 0 */
|
||
|
register_value = RTC_MODE0_CTRL_MODE(0);
|
||
|
|
||
|
/* Set prescaler */
|
||
|
register_value |= RTC_MODE0_CTRL_PRESCALER(CONF_RTC_PRESCALER);
|
||
|
|
||
|
/* clear counter on compare/timer match */
|
||
|
register_value |= RTC_MODE0_CTRL_MATCHCLR;
|
||
|
|
||
|
hri_rtcmode0_write_CTRL_reg(dev->hw, register_value);
|
||
|
|
||
|
hri_rtcmode0_write_COMP_COMP_bf(dev->hw, 0, CONF_RTC_COMP_VAL);
|
||
|
hri_rtcmode0_set_INTEN_CMP0_bit(dev->hw);
|
||
|
|
||
|
/* set event control */
|
||
|
#if CONF_RTC_EVENT_CONTROL_ENABLE == 1
|
||
|
hri_rtcmode0_write_EVCTRL_reg(dev->hw,
|
||
|
(CONF_RTC_PEREO0 << RTC_MODE0_EVCTRL_PEREO0_Pos)
|
||
|
| (CONF_RTC_PEREO1 << RTC_MODE0_EVCTRL_PEREO1_Pos)
|
||
|
| (CONF_RTC_PEREO2 << RTC_MODE0_EVCTRL_PEREO2_Pos)
|
||
|
| (CONF_RTC_PEREO3 << RTC_MODE0_EVCTRL_PEREO3_Pos)
|
||
|
| (CONF_RTC_PEREO4 << RTC_MODE0_EVCTRL_PEREO4_Pos)
|
||
|
| (CONF_RTC_PEREO5 << RTC_MODE0_EVCTRL_PEREO5_Pos)
|
||
|
| (CONF_RTC_PEREO6 << RTC_MODE0_EVCTRL_PEREO6_Pos)
|
||
|
| (CONF_RTC_PEREO7 << RTC_MODE0_EVCTRL_PEREO7_Pos)
|
||
|
| (CONF_RTC_COMPE0 << RTC_MODE0_EVCTRL_CMPEO_Pos)
|
||
|
| (CONF_RTC_OVFEO << RTC_MODE0_EVCTRL_OVFEO_Pos));
|
||
|
#endif
|
||
|
|
||
|
_rtc_dev = dev;
|
||
|
|
||
|
return ERR_NONE;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief De-initialize Timer
|
||
|
*/
|
||
|
void _timer_deinit(struct _timer_device *const dev)
|
||
|
{
|
||
|
ASSERT(dev && dev->hw);
|
||
|
|
||
|
NVIC_DisableIRQ(RTC_IRQn);
|
||
|
|
||
|
hri_rtcmode0_write_CTRL_reg(dev->hw, RTC_MODE0_CTRL_SWRST);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief Start hardware timer
|
||
|
*/
|
||
|
void _timer_start(struct _timer_device *const dev)
|
||
|
{
|
||
|
ASSERT(dev && dev->hw);
|
||
|
|
||
|
NVIC_EnableIRQ(RTC_IRQn);
|
||
|
hri_rtcmode0_write_COUNT_COUNT_bf(dev->hw, 0);
|
||
|
hri_rtcmode0_wait_for_sync(dev->hw);
|
||
|
hri_rtcmode0_set_CTRL_ENABLE_bit(dev->hw);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief Stop hardware timer
|
||
|
*/
|
||
|
void _timer_stop(struct _timer_device *const dev)
|
||
|
{
|
||
|
ASSERT(dev && dev->hw);
|
||
|
|
||
|
hri_rtcmode0_clear_CTRL_ENABLE_bit(dev->hw);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief Set timer period
|
||
|
*/
|
||
|
void _timer_set_period(struct _timer_device *const dev, const uint32_t clock_cycles)
|
||
|
{
|
||
|
hri_rtcmode0_write_COMP_COMP_bf(dev->hw, 0, clock_cycles);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief Retrieve timer period
|
||
|
*/
|
||
|
uint32_t _timer_get_period(const struct _timer_device *const dev)
|
||
|
{
|
||
|
return hri_rtcmode0_read_COMP_COMP_bf(dev->hw, 0);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief Check if timer is running
|
||
|
*/
|
||
|
bool _timer_is_started(const struct _timer_device *const dev)
|
||
|
{
|
||
|
return hri_rtcmode0_get_CTRL_ENABLE_bit(dev->hw);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief Set timer IRQ
|
||
|
*/
|
||
|
void _timer_set_irq(struct _timer_device *const dev)
|
||
|
{
|
||
|
(void)dev;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief RTC Timer interrupt handler
|
||
|
*
|
||
|
* \param[in] p The pointer to calendar device struct
|
||
|
*/
|
||
|
static void _rtc_timer_interrupt_handler(struct _timer_device *dev)
|
||
|
{
|
||
|
/* Read and mask interrupt flag register */
|
||
|
uint16_t flag = hri_rtcmode0_read_INTFLAG_reg(dev->hw);
|
||
|
|
||
|
if (flag & RTC_MODE0_INTFLAG_CMP0) {
|
||
|
if (dev->timer_cb.period_expired) {
|
||
|
dev->timer_cb.period_expired(dev);
|
||
|
}
|
||
|
|
||
|
/* Clear interrupt flag */
|
||
|
hri_rtcmode0_clear_interrupt_CMP0_bit(dev->hw);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief Retrieve timer helper functions
|
||
|
*/
|
||
|
struct _timer_hpl_interface *_rtc_get_timer(void)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief Rtc interrupt handler
|
||
|
*/
|
||
|
void RTC_Handler(void)
|
||
|
{
|
||
|
_rtc_timer_interrupt_handler(_rtc_dev);
|
||
|
}
|