mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
93 lines
3.2 KiB
C
93 lines
3.2 KiB
C
/*
|
|
* SPDX-FileCopyrightText: 2022 Espressif Systems (Shanghai) CO LTD
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
#include <stdio.h>
|
|
#include <inttypes.h>
|
|
#include "freertos/FreeRTOS.h"
|
|
#include "freertos/task.h"
|
|
#include "freertos/event_groups.h"
|
|
#include "unity.h"
|
|
#include "unity_test_utils.h"
|
|
#include "soc/soc_caps.h"
|
|
#include "esp_private/esp_clk.h"
|
|
#include "driver/mcpwm_cap.h"
|
|
#include "driver/mcpwm_sync.h"
|
|
#include "driver/gpio.h"
|
|
#include "test_mcpwm_utils.h"
|
|
|
|
static bool IRAM_ATTR test_capture_callback_iram_safe(mcpwm_cap_channel_handle_t cap_channel, const mcpwm_capture_event_data_t *edata, void *user_data)
|
|
{
|
|
uint32_t *cap_value = (uint32_t *)user_data;
|
|
if (edata->cap_edge == MCPWM_CAP_EDGE_NEG) {
|
|
cap_value[1] = edata->cap_value;
|
|
} else {
|
|
cap_value[0] = edata->cap_value;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static void IRAM_ATTR test_simulate_input_post_cache_disable(void *args)
|
|
{
|
|
int gpio_sig = (int)args;
|
|
gpio_set_level(gpio_sig, 1);
|
|
esp_rom_delay_us(1000);
|
|
gpio_set_level(gpio_sig, 0);
|
|
}
|
|
|
|
TEST_CASE("mcpwm_capture_iram_safe", "[mcpwm]")
|
|
{
|
|
printf("install mcpwm capture timer\r\n");
|
|
mcpwm_cap_timer_handle_t cap_timer = NULL;
|
|
mcpwm_capture_timer_config_t cap_timer_config = {
|
|
.clk_src = MCPWM_CAPTURE_CLK_SRC_APB,
|
|
.group_id = 0,
|
|
};
|
|
TEST_ESP_OK(mcpwm_new_capture_timer(&cap_timer_config, &cap_timer));
|
|
|
|
const int cap_gpio = 0;
|
|
// put the GPIO into a preset state
|
|
gpio_set_level(cap_gpio, 0);
|
|
|
|
printf("install mcpwm capture channel\r\n");
|
|
mcpwm_cap_channel_handle_t pps_channel;
|
|
mcpwm_capture_channel_config_t cap_chan_config = {
|
|
.gpio_num = cap_gpio,
|
|
.prescale = 1,
|
|
.flags.pos_edge = true,
|
|
.flags.neg_edge = true,
|
|
.flags.io_loop_back = true, // so we can use GPIO functions to simulate the external capture signal
|
|
.flags.pull_up = true,
|
|
};
|
|
TEST_ESP_OK(mcpwm_new_capture_channel(cap_timer, &cap_chan_config, &pps_channel));
|
|
|
|
printf("install callback for capture channel\r\n");
|
|
mcpwm_capture_event_callbacks_t cbs = {
|
|
.on_cap = test_capture_callback_iram_safe,
|
|
};
|
|
uint32_t cap_value[2] = {0};
|
|
TEST_ESP_OK(mcpwm_capture_channel_register_event_callbacks(pps_channel, &cbs, cap_value));
|
|
|
|
printf("enable capture channel\r\n");
|
|
TEST_ESP_OK(mcpwm_capture_channel_enable(pps_channel));
|
|
|
|
printf("enable and start capture timer\r\n");
|
|
TEST_ESP_OK(mcpwm_capture_timer_enable(cap_timer));
|
|
TEST_ESP_OK(mcpwm_capture_timer_start(cap_timer));
|
|
|
|
printf("disable cache, simulate GPIO capture signal\r\n");
|
|
unity_utils_run_cache_disable_stub(test_simulate_input_post_cache_disable, (void *)cap_gpio);
|
|
|
|
printf("capture value: Pos=%"PRIu32", Neg=%"PRIu32"\r\n", cap_value[0], cap_value[1]);
|
|
// Capture timer is clocked from APB by default
|
|
uint32_t clk_src_res = esp_clk_apb_freq();
|
|
TEST_ASSERT_UINT_WITHIN(2000, clk_src_res / 1000, cap_value[1] - cap_value[0]);
|
|
|
|
printf("uninstall capture channel and timer\r\n");
|
|
TEST_ESP_OK(mcpwm_capture_channel_disable(pps_channel));
|
|
TEST_ESP_OK(mcpwm_del_capture_channel(pps_channel));
|
|
TEST_ESP_OK(mcpwm_capture_timer_disable(cap_timer));
|
|
TEST_ESP_OK(mcpwm_del_capture_timer(cap_timer));
|
|
}
|