mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
106 lines
3.6 KiB
C
106 lines
3.6 KiB
C
/*
|
|
* SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <stdarg.h>
|
|
#include <sys/cdefs.h>
|
|
#include "sdkconfig.h"
|
|
#include "freertos/FreeRTOS.h"
|
|
#include "esp_attr.h"
|
|
#include "esp_check.h"
|
|
#include "esp_err.h"
|
|
#include "esp_log.h"
|
|
#include "soc/soc_caps.h"
|
|
#include "soc/mcpwm_periph.h"
|
|
#include "hal/mcpwm_ll.h"
|
|
#include "driver/mcpwm_etm.h"
|
|
#include "mcpwm_private.h"
|
|
#include "esp_private/etm_interface.h"
|
|
|
|
static const char *TAG = "mcpwm-etm";
|
|
|
|
typedef struct {
|
|
esp_etm_event_t base;
|
|
mcpwm_cmpr_handle_t cmpr;
|
|
} mcpwm_comparator_etm_event_t;
|
|
|
|
static esp_err_t mcpwm_del_etm_event(esp_etm_event_t *event)
|
|
{
|
|
mcpwm_comparator_etm_event_t *etm_event = __containerof(event, mcpwm_comparator_etm_event_t, base);
|
|
mcpwm_cmpr_handle_t cmpr = etm_event->cmpr;
|
|
mcpwm_oper_t *oper = cmpr->oper;
|
|
mcpwm_group_t *group = oper->group;
|
|
mcpwm_hal_context_t *hal = &group->hal;
|
|
|
|
switch (cmpr->type) {
|
|
case MCPWM_OPERATOR_COMPARATOR:
|
|
portENTER_CRITICAL(&group->spinlock);
|
|
mcpwm_ll_etm_enable_comparator_event(hal->dev, oper->oper_id, cmpr->cmpr_id, false);
|
|
portEXIT_CRITICAL(&group->spinlock);
|
|
break;
|
|
#if SOC_MCPWM_SUPPORT_EVENT_COMPARATOR
|
|
case MCPWM_EVENT_COMPARATOR:
|
|
portENTER_CRITICAL(&group->spinlock);
|
|
mcpwm_ll_etm_enable_evt_comparator_event(hal->dev, oper->oper_id, cmpr->cmpr_id, false);
|
|
portEXIT_CRITICAL(&group->spinlock);
|
|
break;
|
|
#endif // SOC_MCPWM_SUPPORT_EVENT_COMPARATOR
|
|
}
|
|
free(etm_event);
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t mcpwm_comparator_new_etm_event(mcpwm_cmpr_handle_t cmpr, const mcpwm_cmpr_etm_event_config_t *config, esp_etm_event_handle_t *out_event)
|
|
{
|
|
esp_err_t ret = ESP_OK;
|
|
mcpwm_comparator_etm_event_t *event = NULL;
|
|
ESP_RETURN_ON_FALSE(cmpr && config && out_event, ESP_ERR_INVALID_ARG, TAG, "invalid argument");
|
|
event = heap_caps_calloc(1, sizeof(mcpwm_comparator_etm_event_t), MCPWM_MEM_ALLOC_CAPS);
|
|
ESP_RETURN_ON_FALSE(event, ESP_ERR_NO_MEM, TAG, "no memory for ETM event");
|
|
|
|
mcpwm_oper_t *oper = cmpr->oper;
|
|
mcpwm_group_t *group = oper->group;
|
|
mcpwm_hal_context_t *hal = &group->hal;
|
|
int group_id = group->group_id;
|
|
int oper_id = oper->oper_id;
|
|
int cmpr_id = cmpr->cmpr_id;
|
|
uint32_t event_id = 0;
|
|
|
|
switch (cmpr->type) {
|
|
case MCPWM_OPERATOR_COMPARATOR:
|
|
portENTER_CRITICAL(&group->spinlock);
|
|
mcpwm_ll_etm_enable_comparator_event(hal->dev, oper_id, cmpr_id, true);
|
|
portEXIT_CRITICAL(&group->spinlock);
|
|
event_id = MCPWM_LL_ETM_COMPARATOR_EVENT_TABLE(group_id, oper_id, cmpr_id, config->event_type);
|
|
break;
|
|
#if SOC_MCPWM_SUPPORT_EVENT_COMPARATOR
|
|
case MCPWM_EVENT_COMPARATOR:
|
|
portENTER_CRITICAL(&group->spinlock);
|
|
mcpwm_ll_etm_enable_evt_comparator_event(hal->dev, oper->oper_id, cmpr->cmpr_id, true);
|
|
portEXIT_CRITICAL(&group->spinlock);
|
|
event_id = MCPWM_LL_ETM_EVENT_COMPARATOR_EVENT_TABLE(group_id, oper_id, cmpr_id, config->event_type);
|
|
break;
|
|
#endif // SOC_MCPWM_SUPPORT_EVENT_COMPARATOR
|
|
}
|
|
ESP_GOTO_ON_FALSE(event_id != 0, ESP_ERR_NOT_SUPPORTED, err, TAG, "not supported event type");
|
|
ESP_LOGD(TAG, "MCPWM (%d) oper (%d) cmpr(%d) event_id (%"PRId32")", group_id, oper_id, cmpr_id, event_id);
|
|
|
|
// fill the ETM event object
|
|
event->cmpr = cmpr;
|
|
event->base.event_id = event_id;
|
|
event->base.trig_periph = ETM_TRIG_PERIPH_MCPWM;
|
|
event->base.del = mcpwm_del_etm_event;
|
|
|
|
*out_event = &event->base;
|
|
return ESP_OK;
|
|
|
|
err:
|
|
if (event) {
|
|
mcpwm_del_etm_event(&event->base);
|
|
}
|
|
return ret;
|
|
}
|