2022-05-28 17:03:05 +08:00
/*
2023-01-06 14:17:04 +08:00
* SPDX - FileCopyrightText : 2022 - 2023 Espressif Systems ( Shanghai ) CO LTD
2022-05-28 17:03:05 +08:00
*
* SPDX - License - Identifier : Apache - 2.0
*/
# include <stdint.h>
# include <sys/lock.h>
# include "sdkconfig.h"
# if CONFIG_MCPWM_ENABLE_DEBUG_LOG
// The local log level must be defined before including esp_log.h
// Set the maximum log level for this source file
# define LOG_LOCAL_LEVEL ESP_LOG_DEBUG
# endif
# include "esp_log.h"
# include "esp_check.h"
2023-04-23 15:49:59 +08:00
# include "esp_clk_tree.h"
2022-05-28 17:03:05 +08:00
# include "esp_private/periph_ctrl.h"
# include "soc/mcpwm_periph.h"
# include "hal/mcpwm_ll.h"
# include "mcpwm_private.h"
2023-09-13 19:13:01 +08:00
# if SOC_PERIPH_CLK_CTRL_SHARED
# define MCPWM_CLOCK_SRC_ATOMIC() PERIPH_RCC_ATOMIC()
# else
# define MCPWM_CLOCK_SRC_ATOMIC()
# endif
# if !SOC_RCC_IS_INDEPENDENT
# define MCPWM_RCC_ATOMIC() PERIPH_RCC_ATOMIC()
# else
# define MCPWM_RCC_ATOMIC()
# endif
2022-05-28 17:03:05 +08:00
static const char * TAG = " mcpwm " ;
typedef struct {
_lock_t mutex ; // platform level mutex lock
mcpwm_group_t * groups [ SOC_MCPWM_GROUPS ] ; // array of MCPWM group instances
int group_ref_counts [ SOC_MCPWM_GROUPS ] ; // reference count used to protect group install/uninstall
} mcpwm_platform_t ;
static mcpwm_platform_t s_platform ; // singleton platform
mcpwm_group_t * mcpwm_acquire_group_handle ( int group_id )
{
bool new_group = false ;
mcpwm_group_t * group = NULL ;
// prevent install mcpwm group concurrently
_lock_acquire ( & s_platform . mutex ) ;
if ( ! s_platform . groups [ group_id ] ) {
group = heap_caps_calloc ( 1 , sizeof ( mcpwm_group_t ) , MCPWM_MEM_ALLOC_CAPS ) ;
if ( group ) {
new_group = true ;
s_platform . groups [ group_id ] = group ;
group - > group_id = group_id ;
2023-08-16 10:51:30 +08:00
group - > intr_priority = - 1 ;
2022-05-28 17:03:05 +08:00
group - > spinlock = ( portMUX_TYPE ) portMUX_INITIALIZER_UNLOCKED ;
// enable APB to access MCPWM registers
2023-09-13 19:13:01 +08:00
MCPWM_RCC_ATOMIC ( ) {
mcpwm_ll_enable_bus_clock ( group_id , true ) ;
mcpwm_ll_reset_register ( group_id ) ;
}
2022-05-28 17:03:05 +08:00
// initialize HAL context
mcpwm_hal_init_config_t hal_config = {
. group_id = group_id
} ;
mcpwm_hal_context_t * hal = & group - > hal ;
mcpwm_hal_init ( hal , & hal_config ) ;
// disable all interrupts and clear pending status
mcpwm_ll_intr_enable ( hal - > dev , UINT32_MAX , false ) ;
mcpwm_ll_intr_clear_status ( hal - > dev , UINT32_MAX ) ;
2023-09-13 19:13:01 +08:00
// enable function clock
MCPWM_CLOCK_SRC_ATOMIC ( ) {
mcpwm_ll_group_enable_clock ( group - > hal . dev , true ) ;
}
2022-05-28 17:03:05 +08:00
}
} else { // group already install
group = s_platform . groups [ group_id ] ;
}
if ( group ) {
// someone acquired the group handle means we have a new object that refer to this group
s_platform . group_ref_counts [ group_id ] + + ;
}
_lock_release ( & s_platform . mutex ) ;
if ( new_group ) {
ESP_LOGD ( TAG , " new group(%d) at %p " , group_id , group ) ;
}
return group ;
}
void mcpwm_release_group_handle ( mcpwm_group_t * group )
{
int group_id = group - > group_id ;
bool do_deinitialize = false ;
_lock_acquire ( & s_platform . mutex ) ;
s_platform . group_ref_counts [ group_id ] - - ;
if ( s_platform . group_ref_counts [ group_id ] = = 0 ) {
do_deinitialize = true ;
s_platform . groups [ group_id ] = NULL ; // deregister from platfrom
2023-09-13 19:13:01 +08:00
MCPWM_CLOCK_SRC_ATOMIC ( ) {
mcpwm_ll_group_enable_clock ( group - > hal . dev , false ) ;
}
2022-05-28 17:03:05 +08:00
// hal layer deinitialize
mcpwm_hal_deinit ( & group - > hal ) ;
2023-09-13 19:13:01 +08:00
MCPWM_RCC_ATOMIC ( ) {
mcpwm_ll_enable_bus_clock ( group_id , false ) ;
}
2022-05-28 17:03:05 +08:00
free ( group ) ;
}
_lock_release ( & s_platform . mutex ) ;
if ( do_deinitialize ) {
ESP_LOGD ( TAG , " del group(%d) " , group_id ) ;
}
}
2023-08-16 10:51:30 +08:00
esp_err_t mcpwm_check_intr_priority ( mcpwm_group_t * group , int intr_priority )
{
esp_err_t ret = ESP_OK ;
bool intr_priority_conflict = false ;
portENTER_CRITICAL ( & group - > spinlock ) ;
if ( group - > intr_priority = = - 1 ) {
group - > intr_priority = intr_priority ;
} else if ( intr_priority ! = 0 ) {
intr_priority_conflict = ( group - > intr_priority ! = intr_priority ) ;
}
portEXIT_CRITICAL ( & group - > spinlock ) ;
ESP_RETURN_ON_FALSE ( ! intr_priority_conflict , ESP_ERR_INVALID_STATE , TAG , " intr_priority conflict, already is %d but attempt to %d " , group - > intr_priority , intr_priority ) ;
return ret ;
}
int mcpwm_get_intr_priority_flag ( mcpwm_group_t * group )
{
int isr_flags = 0 ;
if ( group - > intr_priority ) {
isr_flags | = 1 < < ( group - > intr_priority ) ;
} else {
isr_flags | = MCPWM_ALLOW_INTR_PRIORITY_MASK ;
}
return isr_flags ;
}
2022-09-05 14:40:58 +08:00
esp_err_t mcpwm_select_periph_clock ( mcpwm_group_t * group , soc_module_clk_t clk_src )
2022-05-28 17:03:05 +08:00
{
esp_err_t ret = ESP_OK ;
bool clock_selection_conflict = false ;
bool do_clock_init = false ;
2023-08-22 15:43:30 +08:00
// check if we need to update the group clock source, group clock source is shared by all mcpwm modules
2022-05-28 17:03:05 +08:00
portENTER_CRITICAL ( & group - > spinlock ) ;
if ( group - > clk_src = = 0 ) {
group - > clk_src = clk_src ;
do_clock_init = true ;
} else {
clock_selection_conflict = ( group - > clk_src ! = clk_src ) ;
}
portEXIT_CRITICAL ( & group - > spinlock ) ;
ESP_RETURN_ON_FALSE ( ! clock_selection_conflict , ESP_ERR_INVALID_STATE , TAG ,
" group clock conflict, already is %d but attempt to %d " , group - > clk_src , clk_src ) ;
if ( do_clock_init ) {
2023-01-06 14:17:04 +08:00
2022-05-28 17:03:05 +08:00
# if CONFIG_PM_ENABLE
2023-01-06 14:17:04 +08:00
sprintf ( group - > pm_lock_name , " mcpwm_%d " , group - > group_id ) ; // e.g. mcpwm_0
ret = esp_pm_lock_create ( ESP_PM_NO_LIGHT_SLEEP , 0 , group - > pm_lock_name , & group - > pm_lock ) ;
ESP_RETURN_ON_ERROR ( ret , TAG , " create pm lock failed " ) ;
ESP_LOGD ( TAG , " install NO_LIGHT_SLEEP lock for MCPWM group(%d) " , group - > group_id ) ;
2022-05-28 17:03:05 +08:00
# endif // CONFIG_PM_ENABLE
2022-09-05 14:40:58 +08:00
2023-09-13 19:13:01 +08:00
MCPWM_CLOCK_SRC_ATOMIC ( ) {
mcpwm_ll_group_set_clock_source ( group - > hal . dev , clk_src ) ;
}
2022-05-28 17:03:05 +08:00
}
return ret ;
}
2023-08-22 15:43:30 +08:00
2023-09-13 19:13:01 +08:00
esp_err_t mcpwm_set_prescale ( mcpwm_group_t * group , uint32_t expect_module_resolution_hz , uint32_t module_prescale_max , uint32_t * ret_module_prescale )
2023-08-22 15:43:30 +08:00
{
ESP_RETURN_ON_FALSE ( group & & expect_module_resolution_hz & & module_prescale_max , ESP_ERR_INVALID_ARG , TAG , " invalid argument " ) ;
uint32_t periph_src_clk_hz = 0 ;
int group_id = group - > group_id ;
uint32_t group_resolution_hz = 0 ;
uint32_t group_prescale = group - > prescale > 0 ? group - > prescale : MCPWM_GROUP_CLOCK_DEFAULT_PRESCALE ; // range: 1~256, 0 means not calculated
uint32_t module_prescale = 0 ; // range: 1~256 (for timer) or 1~16 (for carrier) or 1 (for capture)
ESP_RETURN_ON_ERROR ( esp_clk_tree_src_get_freq_hz ( group - > clk_src , ESP_CLK_TREE_SRC_FREQ_PRECISION_CACHED , & periph_src_clk_hz ) , TAG , " get clock source freq failed " ) ;
// calc the group prescale
group_resolution_hz = periph_src_clk_hz / group_prescale ;
module_prescale = group_resolution_hz / expect_module_resolution_hz ;
// default prescale cannot match
// try to ensure accurate division. If none of the division factors can be guaranteed to be integers, then allocate the clock frequency to the highest divisor
uint32_t fit_module_prescale = 0 ;
uint32_t fit_group_prescale = 0 ;
if ( ! ( module_prescale > = 1 & & module_prescale < = module_prescale_max ) ) {
group_prescale = 0 ;
while ( + + group_prescale < = MCPWM_LL_MAX_GROUP_PRESCALE ) {
group_resolution_hz = periph_src_clk_hz / group_prescale ;
module_prescale = group_resolution_hz / expect_module_resolution_hz ;
if ( module_prescale > = 1 & & module_prescale < = module_prescale_max ) {
// maintain the first value found during the search that satisfies the division requirement (highest frequency), applicable for cases where integer division is not possible."
fit_module_prescale = fit_module_prescale ? fit_module_prescale : module_prescale ;
fit_group_prescale = fit_group_prescale ? fit_group_prescale : group_prescale ;
// find accurate division
if ( group_resolution_hz = = expect_module_resolution_hz * module_prescale ) {
fit_module_prescale = module_prescale ;
fit_group_prescale = group_prescale ;
break ;
}
}
}
module_prescale = fit_module_prescale ;
group_prescale = fit_group_prescale ;
group_resolution_hz = periph_src_clk_hz / group_prescale ;
}
ESP_LOGD ( TAG , " group (%d) calc prescale:% " PRIu32 " , module calc prescale:% " PRIu32 " " , group_id , group_prescale , module_prescale ) ;
ESP_RETURN_ON_FALSE ( group_prescale > 0 & & group_prescale < = MCPWM_LL_MAX_GROUP_PRESCALE , ESP_ERR_INVALID_STATE , TAG ,
" set group prescale failed, group clock cannot match the resolution " ) ;
// check if we need to update the group prescale, group prescale is shared by all mcpwm modules
bool prescale_conflict = false ;
portENTER_CRITICAL ( & group - > spinlock ) ;
if ( group - > prescale = = 0 ) {
group - > prescale = group_prescale ;
group - > resolution_hz = group_resolution_hz ;
2023-09-13 19:13:01 +08:00
MCPWM_CLOCK_SRC_ATOMIC ( ) {
mcpwm_ll_group_set_clock_prescale ( group - > hal . dev , group_prescale ) ;
}
2023-08-22 15:43:30 +08:00
} else {
prescale_conflict = ( group - > prescale ! = group_prescale ) ;
}
portEXIT_CRITICAL ( & group - > spinlock ) ;
ESP_RETURN_ON_FALSE ( ! prescale_conflict , ESP_ERR_INVALID_STATE , TAG ,
" group prescale conflict, already is % " PRIu32 " but attempt to % " PRIu32 " " , group - > prescale , group_prescale ) ;
ESP_LOGD ( TAG , " group (%d) clock resolution:% " PRIu32 " Hz " , group_id , group - > resolution_hz ) ;
// set module resolution
if ( ret_module_prescale ) {
* ret_module_prescale = module_prescale ;
}
return ESP_OK ;
}