NXP iMX8MM Cortex-M4 核心 GPT Capture 测试

news2025/1/21 2:58:53

By Toradex秦海

1). 简介

NXP i.MX8 系列处理器均为异构多核架构 SoC,除了可以运行 Linux 等复杂操作系统的 Cortax-A 核心,还包含了可以运行实时操作系统比如 FreeRTOS 的 Cortex-M 核心,本文就演示通过 NXP i.MX8MM 处理器集成的 Cortex-M4 核心来运行 GPT (General Purpose Timer) 输入采集功能模块的测试。

I.MX8M Mini 处理器 GPT 模块硬件比较简单,如下框图,可以实现 Capture 捕获输入功能和 Compare 定时输出功能。

NXP iMX8MM Cortex-M4 核心 GPT Capture 测试_web333.png

本文所演示的ARM平台来自于Toradex 基于NXP i.MX8M Mini ARM处理器的Verdin iMX8MM ARM嵌入式平台。

2. 准备

a). Verdin i.MX8MM ARM核心版配合Verdin Development Board,连接调试串口(载板X66)到开发主机方便调试,X66 连接了4个串口,其中第三个是 Cortex-M4 核心的默认调试串口,第四个是 Cortex-A53 核心的默认调试串口。

b). 为了测试 GPT 输入捕获, 相应的需要一个PWM 波发生设备,这里使用Toradex 基于NXP i.MX8M Plus ARM处理器的Verdin i.MX8MP 核心板配合 Dahlia Board 作为PWM output使用。同样连接调试串口(载板X18)到开发主机方便调试。

c). Verdin i.MX8MP Cortex-A53 核心系统使用Toradex Yocto Linux BSP6, 更多说明请参考这里。

d). 参考如下将 Verdin i.MX8MP PWM1 连接到 Verdin i.MX8MM GPT1 Capture 管脚,同时为了阻断载板其他电路干扰,将 Verdin Development Board X6 Pin_24 的跳线帽去掉。

Dahlia Board X20 Pin_9 -> Verdin Development Board X5Pin_24 SODIMM_252

3). Verdin i.MX8MM M4核心FreeRTOS基本资料

a). Verdin i.MX8MM HMP(Heterogeneous Multi-core Processing) 架构基本说明请参考如下:

Cortex-M and Memory Areas on Toradex SoMs | Toradex Developer Center

b). 参考如下说明下载配置 NXP 用于开发 Cortex-M 核心的 MCUXpresso SDK

Setting Up MCUXpresso SDK and Toolchain for Cortex-M development | Toradex Developer Center

c). Verdin i.MX8MM 编译运行 M4 firmware 操作流程请参考如下文章

Running FreeRTOS on the Cortex-M4 of a Verdin iMX8M Mini | Toradex Developer Center

d). MCUXpresso SDK 包含的 sample 示例应用可以参考如下 SDK 源位置

-----------------------------

$cd <SDK_root>/boards/evkmimx8mm/

$ tree -L 2

.

├── cmsis_driver_examples

│   ├── ecspi

│   ├── enet

│   ├── i2c

│   └── uart

├── demo_apps

│   ├── hello_world

│   └── sai_low_power_audio

├── driver_examples

│   ├── ecspi

│   ├── enet

│   ├── gpio

│   ├── gpt

│   ├── i2c

│   ├── pdm

│   ├── pwm

│   ├── rdc

│   ├── sai

│   ├── sdma

│   ├── sema4

│   ├── tmu

│   ├── uart

│   └── wdog

├── evkmimx8mm.png

├── freertos_examples

│   ├── freertos_event

│   ├── freertos_generic

│   ├── freertos_hello

│   ├── freertos_mutex

│   ├── freertos_queue

│   ├── freertos_sem

│   ├── freertos_swtimer

│   └── freertos_tickless

├── multicore_examples

│   ├── rpmsg_lite_pingpong_rtos

│   └── rpmsg_lite_str_echo_rtos

└── project_template

    ├── board.c

    ├── board.h

    ├── BOARD_Project_Template_evkmimx8mm.cmake

    ├── clock_config.c

    ├── clock_config.h

    ├── peripherals.c

    ├── peripherals.h

    ├── pin_mux.c

    └── pin_mux.h

-----------------------------

4). Verdin i.MX8MM Cortex-M4核心FreeRTOS GPT Capture示例驱动开发

a). Verdin i.MX8MM MCUXpresso SDK 已经包含一个简单的 GPT Capture sample驱动,本文基于此 sample 进行修改测试。

-----------------------------

$cd <SDK_root>/boards/evkmimx8mm/driver_examples/gpt/capture

$ tree -L 1

.

├── armgcc

├── board.c

├── board.h

├── clock_config.c

├── clock_config.h

├── empty_rsc_table.c

├── fsl_iomuxc.h

├── gpt_capture.c

├── gpt_capture_v3_14.xml

├── pin_mux.c

├── pin_mux.h

└── readme.md

-----------------------------

b). 首先先确认 pin_mux 定义以及其他 i.MX8MM 初始化基本配置,如果需要可以进行修改

./ pin_mux.h/pin_mux.c 用于确定项目中使用的管脚定义,本文中使用的正好就是示例默认的 GPT1 Capture1 管脚,因此无需修改。如果用到其他管脚,就需要进行修改,支持的所有管脚定义可以参考 fsl_iomuxc.h 文件。

-----------------------------

/* FUNCTION ************************************************************************************************************

 *

 * Function Name : BOARD_InitPins

 * Description   : Configures pin routing and optionally pin electrical features.

 *

 * END ****************************************************************************************************************/

void BOARD_InitPins(void) {                                /*!< Function assigned for the core: Cortex-M4[m4] */

    IOMUXC_SetPinMux(IOMUXC_SAI3_RXFS_GPT1_CAPTURE1, 0U);

    IOMUXC_SetPinMux(IOMUXC_UART4_RXD_UART4_RX, 0U);

...

-----------------------------

./ board.h/board.c 用于 i.MX8MM M4 核心基本初始化配置,本文不做修改。

./ clock_config.h/clock_config.c 用于 i.MX8MM M4 核心基本时钟配置,本文不做修改。

c). GPT Capture 功能实现

./ 本文 GPT Capture 功能定义

GPT1 capture1 管脚输入一个给定频率(如 1k Hz )和占空比(如  50% ) 的PWM 信号,通过捕获输入上升/下降沿中断,分别获得相邻两次中断的 GPT Counter 计数器的计数,并以此来计算输入 PWM 信号的半波周期。

./ GPT Capture 功能基本都是通过 gpt_capture.c 文件代码来实现,默认 sample 是捕获上升沿中断后,打印中断当时的 GPT Counter 计数数值。

./ 为了实现本文定义的捕获功能,首先增加如下全局变量定义

-----------------------------

/*******************************************************************************

 * Variables

 ******************************************************************************/

volatile bool gptIsrFlag_Start = false;

volatile bool gptIsrFlag_Finish = false;

volatile bool gptIsrFlag_Overflow = false;

volatile uint32_t captureVal = 0;

volatile uint32_t captureVal_Last = 0;

-----------------------------

// gptIsrFlag_Start 定义为第一次捕获中断开始标志

// gptIsrFlag_Finish 定义为第二次捕获中断结束标志

// gptIsrFlag_Overflow 定义为 GPT Counter 溢出标志

// captureVal 定义为第二次中断 GPT Counter 数值

// captureVal_Last 定义为第一次中断 GPT Counter 数值

./ GPT Interrupt 函数修改如下:

首先处理计数器溢出情况,如果中断发生时候已经发生溢出,则声明 gptIsrFlag_Overflow 溢出标志位;然后通过 gptIsrFlag_Start / gptIsrFlag_Finish 标志位来分别处理第一次和第二次中断,获取第一次和第二次中断时候的 GPT Counter 数值,同时分别翻转 GPT Capture Interrupt 模式。另外,当溢出发生在第一次中断计数和第二次中断计数之间的时候,需要将第二次中断计数数值增加溢出值 0xffffffff。

-----------------------------

void EXAMPLE_GPT_CAPTURE_IRQHandler(void)

{

    /* GPT Counter Overflow processing */

    if (GPT_GetStatusFlags(DEMO_GPT_BASE, kGPT_RollOverFlag) != false)

    {

        if (gptIsrFlag_Start != true)

        {

           gptIsrFlag_Overflow = true;

        }

        GPT_ClearStatusFlags(DEMO_GPT_BASE, kGPT_RollOverFlag);

    }

    if (GPT_GetStatusFlags(DEMO_GPT_BASE, kGPT_InputCapture1Flag) != false)

    {

        if(gptIsrFlag_Finish != true)

        {

           /* First time IRQ */

           if (gptIsrFlag_Start == false)

           {

               captureVal_Last = GPT_GetInputCaptureValue(DEMO_GPT_BASE, BOARD_GPT_INPUT_CAPTURE_CHANNEL);

               /* Switch Interrupt mode to falling edge */

               GPT_SetInputOperationMode(DEMO_GPT_BASE, BOARD_GPT_INPUT_CAPTURE_CHANNEL, kGPT_InputOperation_FallEdge);

               gptIsrFlag_Start = true;

           }

           /* Second time IRQ */

   /* GPT counter overflow */

           else if (gptIsrFlag_Overflow == true)

           {

               captureVal = GPT_GetInputCaptureValue(DEMO_GPT_BASE, BOARD_GPT_INPUT_CAPTURE_CHANNEL) + 0xffffffff;

               /* Switch Interrupt mode to rising edge */

               GPT_SetInputOperationMode(DEMO_GPT_BASE, BOARD_GPT_INPUT_CAPTURE_CHANNEL, kGPT_InputOperation_RiseEdge);

               gptIsrFlag_Start = false;

               gptIsrFlag_Finish = true;

           }

   else

   {

               captureVal = GPT_GetInputCaptureValue(DEMO_GPT_BASE, BOARD_GPT_INPUT_CAPTURE_CHANNEL);

               /* Switch Interrupt mode to rising edge */

               GPT_SetInputOperationMode(DEMO_GPT_BASE, BOARD_GPT_INPUT_CAPTURE_CHANNEL, kGPT_InputOperation_RiseEdge);

               gptIsrFlag_Start = false;

               gptIsrFlag_Finish = true;

   }

      }

      GPT_ClearStatusFlags(DEMO_GPT_BASE, BOARD_GPT_CHANNEL_FLAG);

   }

   SDK_ISR_EXIT_BARRIER;

}

-----------------------------

./ Main 主函数修改如下:

-----------------------------

int main(void)

{

    gpt_config_t gptConfig;

...

GPT_GetDefaultConfig(&gptConfig);

    /* Initialize GPT module */

    GPT_Init(DEMO_GPT_BASE, &gptConfig);

...

/* Setup input capture on a gpt channel */

    GPT_SetInputOperationMode(DEMO_GPT_BASE, BOARD_GPT_INPUT_CAPTURE_CHANNEL, kGPT_InputOperation_RiseEdge);

...

    /* Enable GPT Overflow interrupt */

    GPT_EnableInterrupts(DEMO_GPT_BASE, kGPT_RollOverFlagInterruptEnable);

...

while (true)

    {

        /* Check whether occur 2nd interupt */

        if (true == gptIsrFlag_Finish)

        {

            PRINTF("\r\n Input Capture Half Period Value =%u us\r\n", (captureVal - captureVal_Last)/24);

            gptIsrFlag_Finish = false;

        }

        else

        {

            __WFI();

        }

    }

}

-----------------------------

// 通过 GPT_GetDefaultConfig 函数获取默认的 GPT Capture 配置,参考 docs 目录下的 MCUXpresso SDK API Reference Manual_MIMX8MM6.pdf 文档,可以查到默认配置如下,如果需要也可以修改这个配置

-----------------------------

config->clockSource = kGPT_ClockSource_Periph;

config->divider = 1U;

config->enableRunInStop = true;

config->enableRunInWait = true;

config->enableRunInDoze = false;

config->enableRunInDbg = false;

config->enableFreeRun = false;

config->enableMode = true;

-----------------------------

// 通过 GPT_SetInputOperationMode 函数将 GPT Capture 模式初始配置为上升沿触发

// 为了处理 GPT Counter Overflow,使能对应中断

// while 函数循环执行当  gptIsrFlag_Finish 第二次中断采集结束标志位声明后,打印捕获的输入 PWM 波的半波周期。这里需要说明下,由于 NXP iMX8MM SoC 也受到如下 Errata 影响,因此 GPT Clock Source 只能使用内部 24M Hz 时钟源,所以这里 printf 函数直接使用 24M 来算出半波周期是多少 us 。

https://www.nxp.com.cn/docs/en/errata/IMX8X_C0_0N99Z_ER.pdf

5). Verdin i.MX8MM Cortex-M4核心FreeRTOS GPT Capture示例部署测试

a). 将上述修改后的项目参考章节 3 的相关资料编译后,复制 gpt_capture.bin 可执行文件到 Verdin i.MX8MM 核心板 Linux /home/root 目录下保存。

b). 对Verdin i.MX8MM 模块进入 U-boot 命令行,通过如下命令配置 Cortex-M4 核心 Firmware 下载和运行

-----------------------------

# setenv load_cmd "ext4load mmc 0:2"

# setenv m4image "/home/root/gpt_capture.bin"

> setenv m4image_size 17000

> setenv loadm4image "${load_cmd} ${loadaddr} ${m4image}"

> setenv m4boot "${loadm4image}; cp.b ${loadaddr} 0x7e0000 ${m4image_size}; dcache flush; bootaux 0x7e0000"

> saveenv

> run m4boot

-----------------------------

c). Verdin i.MX8MM Cortex-M4 核心运行后其调试串口打印信息

-----------------------------

GPT input capture example

Once the input signal is received the input capture half peroid is printed

-----------------------------

d). 此时在 Verdin i.MX8MP 平台通过如下脚本使能 1kHz 50% 占空比 PWM 输出 10s 时间

-----------------------------

#!/bin/sh

cd /sys/class/pwm/pwmchip0/

echo 0 > export

echo 1000000 > pwm0/period

echo 500000 > pwm0/duty_cycle

echo "normal" > pwm0/polarity

echo 1 > pwm0/enable

sleep 10

echo 0 > pwm0/enable

-----------------------------

e). 这时 Verdin i.MX8MM Cortex-M4 调试串口就会打印出对应的半波周期

-----------------------------

...

 Input Capture Half Period Value =500 us

 Input Capture Half Period Value =499 us

 Input Capture Half Period Value =499 us

 Input Capture Half Period Value =499 us

...

-----------------------------

f). 尝试将 Verdin i.MX8MP PWM 修改为 10kHz 80%/20% 占空比

-----------------------------

...

echo 100000 > pwm0/period

echo 80000 > pwm0/duty_cycle

...

-----------------------------

g). Verdin i.MX8MM Cortex-M4 输出周期会对应变化

-----------------------------

 Input Capture Half Period Value =80 us

 Input Capture Half Period Value =80 us

 Input Capture Half Period Value =80 us

 Input Capture Half Period Value =80 us

-----------------------------

h). 最后,由于 Verdin i.MX8MM GPT1 CAPTURE1 管脚在 Cortex-A53 核心 Linux 下默认是用于 WAKEUP GPIO 使用,如果需要同时运行 Verdin i.MX8MM Cortex-A53 核心和 Cortex-M4 核心,就需要在 Linux device-tree 文件中将 WAKEUP gpio-key 功能替换为其他 GPIO 管脚资源。

imx8mm-verdin.dtsi « freescale « dts « boot « arm64 « arch - linux-toradex.git - Linux kernel for Apalis and Colibri modules

6). 总结

本文简单示例了基于i.MX8MM Cortex-M4 核心 GPT Capture 功能供参考。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1495873.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于Springboot+Layui餐厅点餐系统

一、项目背景 在互联网经济飞速发展的时代&#xff0c;网络化企业管理也在其带领下快速兴起&#xff0c;开发一款自主点餐系统会受到众多商家的青睐。现如今市场上的人力资源价格是非常高昂的&#xff0c;一款自主点餐系统可以减少餐厅的人力开销&#xff0c;将服务员从繁忙的…

建立自己的富足金字塔

本文的理论参考&#xff1a; 1、《金字塔原理》&#xff0c;可以简述为&#xff1a;上下层互为支撑的结构化表达&#xff0c;通常为1-3-9结构&#xff1b;完全穷举、互不交叉&#xff0c;或不重不漏MECE的分门分类&#xff1b;真实有效&#xff0c;真知灼见。 2、《金线》在金字…

贷齐乐错误的waf引起的SQL注入漏洞复现

君衍. 一、环境介绍1、第一道WAF2、第二道WAF 二、环境部署1、模拟源码2、连接数据库源码3、数据库创建4、测试 三、源码分析1、模拟WAF2、注入思路3、PHP下划线特性4、完成假设 四、联合查询注入1、测试回显字段2、爆出库名3、爆出表名4、爆出表下的列名4、爆出flag 一、环境介…

AI大模型与小模型之间的“脱胎”与“反哺”(第五篇)

一、背景 AI大模型与小模型之间存在一种“脱胎”与“反哺”的关系&#xff0c;这种关系在AI技术的发展中起到了重要的作用。 首先&#xff0c;我们来理解一下“脱胎”的概念。在AI领域&#xff0c;大模型通常具有更大的参数量、更强的表达能力和更高的计算需求。这些大模型通…

第一个 Angular 项目 - 添加路由

第一个 Angular 项目 - 添加路由 前置项目是 第一个 Angular 项目 - 添加服务&#xff0c;之前的切换页面使用的是 ngIf 对渲染的组件进行判断&#xff0c;从而完成渲染。这一步的打算是添加路由&#xff0c;同时添加 edit recipe 的功能(同样通过路由实现) 用到的内容为&…

RabbitMQ 交换器

RabbitMQ 交换器 官方例子 http://www.rabbitmq.com/getstarted.html direct 如上图所示&#xff0c;两个队列绑定到了direct交换器上&#xff0c;第一个队列绑定的 binding key 为 orange &#xff0c;第二个队列有两个绑定&#xff0c;分别是 black 和 green 。 如上图所示…

vue+element模仿实现云码自动验证码识别平台官网

一、项目介绍 项目使用传统vue项目结构实现&#xff0c;前端采用element实现。 element官网&#xff1a;Element - The worlds most popular Vue UI framework 云码官网地址&#xff1a;云码-自动验证码识别平台_验证码识别API接口_免费验证码软件 项目截图&#xff0c;支持…

一键安装conda-batch脚本

conda的安装可以更简化些&#xff0c;即使用batch文件&#xff0c;具体步骤如下&#xff1a; 新建文本文档&#xff0c;并改后缀名为.bat 使用文本编辑器编辑&#xff0c;并输入内容如下 echo offSET MINICONDA_INSTALLER_PATHminiconda3_installer.exe SET MINICONDA_UR…

实现一个网页版的简易猜数字游戏

实现一个网页版的简易猜数字游戏 效果 代码截图 相关代码 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><t…

基于Matlab实现免疫荧光图像中的区域定位算法

基于Matlab实现免疫荧光图像中的区域定位算法 免疫荧光法。以荧光染料为标记物,试纸条为载体,发生抗原抗体特异性反应,根据免疫复合物被激发的荧光强度对待测物进行定量分析[。该方法具有环境要求不高、操作简单快速、无污染且荧光染料丰富等优点。 常用于食品安全检测过程中…

Python实现选择排序算法

Python实现选择排序算法 以下是使用Python实现选择排序算法的示例代码&#xff1a; def selection_sort(arr):n len(arr)for i in range(n):min_index i# 找到未排序部分的最小元素的索引for j in range(i 1, n):if arr[j] < arr[min_index]:min_index j# 将最小元素与…

【BUG】cmd运行wmic提示‘wmic‘ 不是内部或外部命令

cmd运行wmic提示‘wmic‘ 不是内部或外部命令 解决办法 将C:\Windows\System32\wbem添加到系统环境变量

《系统架构设计师教程(第2版)》第6章-据库设计基础知识-01-数据库基本概念

文章目录 1. 概述1.1 基本概念1&#xff09;信息 (Information)2&#xff09;数据 (Data)3&#xff09;数据库 (DB)4&#xff09;数据库系统(DBS)5&#xff09;数据库管理系统&#xff08;DBMS&#xff09; 1.2 数据库技术的发展1.2.1 人工管理阶段1.2.2 文件系统阶段1&#xf…

SQL中如何添加数据

SQL中如何添加数据 一、SQL中如何添加数据&#xff08;方法汇总&#xff09;二、SQL中如何添加数据&#xff08;方法详细解说&#xff09;1. 使用SQL脚本&#xff08;推荐&#xff09;1.1 在表中插入1.1.1 **第一种形式**1.1.2 **第二种形式**SQL INSERT INTO 语法示例SQL INSE…

代码学习记录10

随想录日记part10 t i m e &#xff1a; time&#xff1a; time&#xff1a; 2024.03.03 主要内容&#xff1a;今天的主要内容是深入了解数据结构中栈和队列&#xff0c;并通过三个 l e e t c o d e leetcode leetcode 题目深化认识。 20. 有效的括号1047. 删除字符串中的所有…

day13_微服务监控Nginx(微服务集成SBA)

文章目录 1 微服务系统监控1.1 监控系统的意义1.2 SBA监控方案1.3 SBA实战1.3.1 创建SBA服务端1.3.2 微服务集成SBA 1.4 微服务集成logback1.5 配置邮件告警 2 Nginx2.1 Nginx简介2.2 下载和安装2.2.1 方式1&#xff1a;window本地安装2.2.1.1 下载2.2.1.2 安装2.2.1.3 目录结构…

如何使用 CSS object-fit 进行图片的缩放和裁剪

简介 在处理图片时&#xff0c;你可能会遇到需要保持原始宽高比的情况。保持宽高比可以防止图片被拉伸或压缩而出现失真。解决这个问题的常见方法是使用 background-image CSS 属性。更现代的方法是使用 object-fit CSS 属性。 在本文中&#xff0c;你将探索 object-fit CSS …

【系统安全加固】Centos 设置禁用密码并打开密钥登录

文章目录 一&#xff0c;概述二&#xff0c;操作步骤1. 服务器端生成密钥2. 在服务器上安装公钥3.下载私钥到本地&#xff08;重要&#xff0c;否则后面无法登录&#xff09;4. 修改配置文件&#xff0c;禁用密码并打开密钥登录5. 重启sshd服务6. 配置xshell使用密钥登录 一&am…

【异常处理】sbt构建Chisel库时出现extracting structure failed:build status:error的解决办法

文章目录 报错背景&#xff1a;解决思路&#xff1a;①IDEA中配置本地的SBT进行下载②更改下载源为华为的镜像站1. 修改sbtconfig.txt2. 增加repositories文件 ③查看报错信息 总结整理的Scala-Chisel-Chiseltest版本信息对应表 报错背景&#xff1a; 最近在写Chisel时&#x…

14、电源管理入门之Watchdog看门狗

目录 1. 软硬件watchdog的区别 2. 软件看门狗 2.1 kernel watchdog 2.1.1 soft lockup 2.1.1 hard lockup 2.2 用户态watchdog 2.2.1 softdog 2.2.1 hardware watchdog 3. 硬件看门狗 3.1 硬件寄存器介绍 3.2 喂狗操作 3.3 watchdog硬件驱动编写 参考: 看门狗,又…