mirror of
				https://github.com/espressif/esp-idf.git
				synced 2025-11-04 06:11:06 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			176 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			176 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * SPDX-FileCopyrightText: 2022-2023 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_prelude.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_DEFAULT,
 | 
						|
        .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]);
 | 
						|
    uint32_t clk_src_res;
 | 
						|
    TEST_ESP_OK(mcpwm_capture_timer_get_resolution(cap_timer, &clk_src_res));
 | 
						|
    clk_src_res /= 1000; // convert to kHz
 | 
						|
    TEST_ASSERT_UINT_WITHIN(100, 1000, (cap_value[1] - cap_value[0]) * 1000 / clk_src_res);
 | 
						|
 | 
						|
    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));
 | 
						|
}
 | 
						|
 | 
						|
static bool IRAM_ATTR test_compare_on_reach(mcpwm_cmpr_handle_t cmpr, const mcpwm_compare_event_data_t *ev_data, void *user_data)
 | 
						|
{
 | 
						|
    uint32_t cmp_val = ev_data->compare_ticks;
 | 
						|
    cmp_val += 10;
 | 
						|
    // compare ticks can't exceed the timer's period ticks
 | 
						|
    if (cmp_val >= 50) {
 | 
						|
        cmp_val = 0;
 | 
						|
    }
 | 
						|
    mcpwm_comparator_set_compare_value(cmpr, cmp_val);
 | 
						|
    return false;
 | 
						|
}
 | 
						|
 | 
						|
static void IRAM_ATTR test_delay_post_cache_disable(void *args)
 | 
						|
{
 | 
						|
    esp_rom_delay_us(1000);
 | 
						|
}
 | 
						|
 | 
						|
TEST_CASE("mcpwm_comparator_iram_safe", "[mcpwm]")
 | 
						|
{
 | 
						|
    mcpwm_timer_handle_t timer;
 | 
						|
    mcpwm_oper_handle_t oper;
 | 
						|
    mcpwm_cmpr_handle_t comparator;
 | 
						|
    mcpwm_gen_handle_t gen;
 | 
						|
 | 
						|
    mcpwm_timer_config_t timer_config = {
 | 
						|
        .group_id = 0,
 | 
						|
        .clk_src = MCPWM_TIMER_CLK_SRC_DEFAULT,
 | 
						|
        .resolution_hz = 1 * 1000 * 1000,
 | 
						|
        .period_ticks = 50, // 50us <-> 20KHz
 | 
						|
        .count_mode = MCPWM_TIMER_COUNT_MODE_UP,
 | 
						|
    };
 | 
						|
    mcpwm_operator_config_t operator_config = {
 | 
						|
        .group_id = 0,
 | 
						|
    };
 | 
						|
    mcpwm_comparator_config_t comparator_config = {
 | 
						|
        .flags.update_cmp_on_tep = true,
 | 
						|
        .flags.update_cmp_on_tez = true,
 | 
						|
    };
 | 
						|
    printf("install timer, operator and comparator\r\n");
 | 
						|
    TEST_ESP_OK(mcpwm_new_timer(&timer_config, &timer));
 | 
						|
    TEST_ESP_OK(mcpwm_new_operator(&operator_config, &oper));
 | 
						|
    TEST_ESP_OK(mcpwm_new_comparator(oper, &comparator_config, &comparator));
 | 
						|
 | 
						|
    printf("connect MCPWM timer and operators\r\n");
 | 
						|
    TEST_ESP_OK(mcpwm_operator_connect_timer(oper, timer));
 | 
						|
    TEST_ESP_OK(mcpwm_comparator_set_compare_value(comparator, 10));
 | 
						|
 | 
						|
    printf("install MCPWM generator\r\n");
 | 
						|
    mcpwm_generator_config_t gen_config = {
 | 
						|
        .gen_gpio_num = 0,
 | 
						|
    };
 | 
						|
    TEST_ESP_OK(mcpwm_new_generator(oper, &gen_config, &gen));
 | 
						|
 | 
						|
    printf("set generator actions on timer and compare events\r\n");
 | 
						|
    TEST_ESP_OK(mcpwm_generator_set_action_on_timer_event(gen,
 | 
						|
                MCPWM_GEN_TIMER_EVENT_ACTION(MCPWM_TIMER_DIRECTION_UP, MCPWM_TIMER_EVENT_EMPTY, MCPWM_GEN_ACTION_HIGH)));
 | 
						|
    TEST_ESP_OK(mcpwm_generator_set_action_on_compare_event(gen,
 | 
						|
                MCPWM_GEN_COMPARE_EVENT_ACTION(MCPWM_TIMER_DIRECTION_UP, comparator, MCPWM_GEN_ACTION_LOW)));
 | 
						|
 | 
						|
    printf("register compare event callback\r\n");
 | 
						|
    mcpwm_comparator_event_callbacks_t cbs = {
 | 
						|
        .on_reach = test_compare_on_reach,
 | 
						|
    };
 | 
						|
    TEST_ESP_OK(mcpwm_comparator_register_event_callbacks(comparator, &cbs, NULL));
 | 
						|
 | 
						|
    printf("start timer\r\n");
 | 
						|
    TEST_ESP_OK(mcpwm_timer_enable(timer));
 | 
						|
    TEST_ESP_OK(mcpwm_timer_start_stop(timer, MCPWM_TIMER_START_NO_STOP));
 | 
						|
 | 
						|
    printf("disable flash cache and check the compare events are still in working\r\n");
 | 
						|
    for (int i = 0; i < 50; i++) {
 | 
						|
        unity_utils_run_cache_disable_stub(test_delay_post_cache_disable, NULL);
 | 
						|
    }
 | 
						|
 | 
						|
    printf("uninstall timer, operator and comparator\r\n");
 | 
						|
    TEST_ESP_OK(mcpwm_timer_start_stop(timer, MCPWM_TIMER_STOP_EMPTY));
 | 
						|
    TEST_ESP_OK(mcpwm_timer_disable(timer));
 | 
						|
    TEST_ESP_OK(mcpwm_del_generator(gen));
 | 
						|
    TEST_ESP_OK(mcpwm_del_comparator(comparator));
 | 
						|
    TEST_ESP_OK(mcpwm_del_operator(oper));
 | 
						|
    TEST_ESP_OK(mcpwm_del_timer(timer));
 | 
						|
}
 |