【S32K3 RTD MCAL 篇1】 K344 KEY 控制 EMIOS PWM

【S32K3 RTD MCAL 篇1】 K344 KEY 控制 EMIOS PWM

  • 一,文档简介
  • [二, 功能实现](#二, 功能实现)
    • [2.1 软硬件平台](#2.1 软硬件平台)
    • [2.2 软件控制流程](#2.2 软件控制流程)
    • [2.3 资源分配概览](#2.3 资源分配概览)
    • [2.4 EB 配置](#2.4 EB 配置)
      • [2.4.1 Dio module](#2.4.1 Dio module)
      • [2.4.2 Icu module](#2.4.2 Icu module)
      • [2.4.4 Mcu module](#2.4.4 Mcu module)
      • [2.4.5 Platform module](#2.4.5 Platform module)
      • [2.4.6 Port module](#2.4.6 Port module)
      • [2.4.7 Pwm module](#2.4.7 Pwm module)
    • [2.5 main code](#2.5 main code)
  • 三,测试结果

一,文档简介

搞了有一段时间的S32K3了,MCAL的代码写了几个,但是一直没抽出时间好好写文章,MCAL的文章第一篇就从K344 EMIOS+ICU+TRIMUX+LCU的组合体开始,这个可以涉及到PORT, DIO, EMIOS,中断ICU,TRIGMUX,LCU等综合配置,板子平台还是基于NXP 官方S32K344EVB,RTD400,功能为:使用EMIOS0 两个通道,EMIOS1一个通道。EMIOS 0 其中一个通道输出PWM之后,通过TRIGUMX连接LCU生成一组互补的PWM,另外一个通道可以实现通过SW5 PT26按键实现硬件中断控制PWM占空比。EMIOS1一个通道直接连接到PTA29 板载红灯,通过改变PWM占空比实现亮度渐进改变,然后在亮灭直接循环改变实现呼吸灯效果。同时在PWM关闭红灯的时候,通过DIO打开板载绿灯,PWM打开红灯的时候,DIO关闭板载绿灯。

文字描述总是没有图形描述来的更加直观,直接上图:


图 1

二, 功能实现

本篇文章基于之前把MCAL代码porting到S32DS的demo上,然后就在S32DS的平台下,通过EB配置MCAL 相关模块,再通过S32DS去编译并且下载仿真。当然喜欢命令行模式的,还是可以直接去用EB配置好的xmd文件,然后借助VScode去编译,过程也很简单,本文在此就不赘述命令行方式。

2.1 软硬件平台

板子:S32K344EVB,其他K3板子也可以

IDE:S32DS3.5

RTD: K344 RTD 400

MCAL tool: EBtresos Studio 29.0

2.2 软件控制流程

在讲具体的MCAL配置之前,先给出本文功能的软件流程图:

图 2

这里可以看到,默认的情况就是通过MCAL配置的情况,然后在代码里面也有去修改PWM的频率,然后通过按键以及循环延时等方式去修改PWM的占空比。

2.3 资源分配概览

将本文用到的硬件资源以及功能情况罗列如下:

图 3

关于emios相关总线的罗列配置如下:

图4

这里需要注意的是,对于MCL emios 中master bus和bus mode,需要在PWM模块中选择合适的PWM mode以及counter bus,否则要么配置报错,要么无法生成正确的PWM波形。

从官方的S32K3 RM中可以查阅,时钟通道和bus类型的情况:

图 5

比如,PWM0中选择的是EMIOS_CH23,那么这个bus对应的就是bus A,该时钟是可以给所有的通道使用的,所以PWM0是CH12是可以使用Bus A. CH22对应的是bus F,该时钟也是可以给所有通道使用的,所以PWM1 CH4 选择Bus F也没问题,CH0对应的是bus B,该时钟是对应通道0-7,所以对于PWM2是CH2也是可以使用的。在为自己的EMIOS 通道选择counter bus时钟源的时候,一定要考虑到counter bus对于通道覆盖度的问题。

除了counter bus的选择之外,还有就是PWM模式选择,这个就比较好办,对于时钟向上计数的,选择OPWMB,对于时钟向上向下计数的,选择OPWMCB中间对齐PWM。

在实际的使用中,通常是根据自己的PWM需求来决定模式选择,然后再根据RM下表:

图6

可以找到对应模式所能支持的通道类型,然后再根据图5中的通道类型去选择对应的通道,counter bus等。有了这些基础知识,下面直接进入EB 配置。

2.4 EB 配置

这里罗列出所有本文相关的EB tresos里面使用到的模块,并且重点给予需要具体配置的模块讲解。


图 7

2.4.1 Dio module

需要配置DioPort界面,主要目的是配置PTA30, 板载绿灯,选择DioPort Id=1,Dio Channel Id=14. 关于DioPortId, Dio Channel Id的规则如下:

Channel = DioChannelId + DioPortId∗16

For S32K3X4 derivatives

bash 复制代码
-- Port AL=0
-- Port AH=1
-- Port BL=2
-- Port BH=3
-- Port CL=4
-- Port CH=5
-- Port DL=6
-- Port DH=7
-- Port EL=8
-- Port EH=9
-- Port FL=10
-- Port FH=11
-- Port GL=12
-- Port GH=13

PTA30=>30=DioChannelId(14)+DioPortId*16

2.4.2 Icu module

首先配置IcuSiul2,目标是使能板载SW5,PTB26的输入中断。PTB26对应的是EIRQ[13],

则,对应的中断情况如下:

图 8


图 9

(1) Icu->IcuSiul2->IcuSiul2Channels:13

(2) Icu->IcuChannel配置如下:

图 10

选择IcuChannelRef为之前配置的IcuSiul2Channels,另外就是添加中断通知函数:

User_EdgeDetect,注意,这个函数就是代码里面需要添加的用户中断处理函数名。

(3)Icu->IcuHwInterruptConfigList-> ICU Peripheral ISR Name: SIUL2_0_IRQ_CH_13,IcuIsrEnable enable

2.4.3 Mcl module

该模块主要用于配置emios 计数时钟,trgmux,LCU的配置。

(1)Mcl->Trgmux Logic Instance->Hardware Instance: TRGMUX_IP_HW_INST_0

(2) Mcl->Trgmux Logic Group:

图11

主要目的是通过Trigmux把PWM1 Emios0_ch4 连接到LCU0_IN0.

(3) Mcl->LCU Configuration

这里是配置LCU模块,主要功能是配置逻辑输入,以及逻辑输出情况,输入一个IN0,输出两个OUT0,OUT1.

图12


图13


图14

OUTPUT0 值0XAAAA=43690,

OUTPUT1 值0X5555=21845

这么做的目的是让输入的PWM生成一对互补的PWM。

图 15

(4)Mcl->Emios Common

添加两个Emios,分别为EMIOS_0, EMIOS_1,代表用到两个EMIOS。

EMIOS0配置了两个master bus通道:CH_22,CH_0,而且具有不同的模式类型,向上向下计数和向上计数。

EMIOS1配置了一个master bus通道:CH_23,向上计数

和图4对应。

图 16


图 17

注意,这里的通道并不是真正输出PWM的通道,只是需要使用PWM通道的counter bus通道,具有提供时钟能力的通道。

2.4.4 Mcu module

该模块是整个MCU用来配置时钟的基础,使用原始RTD PWM demo默认的情况,就一个点需要注意。

Mcu->McuClockSettingConfig_0->McuClockReferencePoint->McuClockReferencePoint_0->core clock 48MHZ.

这个时钟是EMIOS时钟的来源,有了时钟源,根据设定的周期,就不难计算真正的PWM频率了。比如需要一个1Khz的PWM,则可以配置period=48M/1K=48000

2.4.5 Platform module

Platform->Interrupt Controller->IntCtrlConfig0, 使能SIUL_1_IRQn, 并且添加Hanlder为:

SIUL2_EXT_IRQ_8_15_ISR

注意,这个SIUL2_EXT_IRQ_8_15_ISR不是乱写的,是要和Siul2_Icu_Ip_Irq.c里面的对应,否则会报错。

图18

不同的中断,有不同的中断服务函数,需要查到代码定义的函数名称填写到EB。

EB配置如下:

图19

2.4.6 Port module

配置8个引脚情况如下:

图 20

可以看到,有3路主EMIOS PWM,一路输入中断,一路输出GPIO,两路输出LCU 互补PWM。

2.4.7 Pwm module

Mcl里面配置的是EMIOS的counter 时钟通道,真正要输出的PWM通道都需要在Pwm模块中配置,并且链接到Mcu时钟源以及Mcl里面的emios counter bus源。

(1)Pwm->PwmEmios

添加两组,用来给对应的EMIOS模块,比如本文用到了EMIOS0,EMIOS1,所以需要添加两个:

图 21

对于PwmEmios_1,里面有一个通道,配置情况如下:

图22

PwmEmiosBusRef: /Mcl/Mcl/MclConfig/EmiosCommon_1/EmiosMclMasterBus_0

图 23

可以看到,这里PwmEmios_1的busRef是Mcl里面的Emios_ch_23,即BusA

也就是说EMIOS1_CH12用的bus参考时钟来自EMIOS1_CH23,即Bus A。

PwmEmios_0里面配置了两个通道,配置情况如下:

图24


图25

EMIOS0_CH4用的bus参考时钟来自EMIOS0_CH22,即Bus F。

另外一个EMIOS0通道:

图 26


图27

EMIOS0_CH2用的bus参考时钟来自EMIOS0_CH0,即Bus B,所以选择Bus BCDE。

到这里,是可以清晰的知道,真正的PWM输出通道和内部的MCL的Emios counter bus通道的关系了。

(2)Pwm->PwmEmios

有了前面配置的PWM的具体信息,下面直接配置PWM的通道,一共三个通道:

PWM0, PWM1, PWM2,也正是代码里面需要用到的标志。

图 28

2.5 main code

c 复制代码
#include "Pwm.h"
#include "Mcu.h"
#include "Port.h"
#include "Mcl.h"
#include "Platform.h"
#include "Dio.h"
#include "Icu.h"
//#include "check_example.h"

#define NUM_BLINK_LED     (uint32)10U
#define DELAY_TIMER       (uint32)5000000U
#define MCL_EMIOS_1_CH_23 (uint16)279U
#define MCL_EMIOS_0_CH_22 (uint16)22U
Mcl_LcuSyncOutputValueType PWM_OutputList[2];

volatile uint8 UserCountIrqCH0;
void TestDelay(uint32 delay);
void TestDelay(uint32 delay)
{
    static volatile uint32 DelayTimer = 0;
    while(DelayTimer<delay)
    {
        DelayTimer++;
    }
    DelayTimer=0;
}
void User_EdgeDetect(void)
{
    /* increment IRQ counter */
    UserCountIrqCH0++;
    if(UserCountIrqCH0 % 2 == 0)
    {
        Pwm_SetDutyCycle(PwmChannel_2, 0X6000);
     }
    else
    {
        Pwm_SetDutyCycle(PwmChannel_2, 0X2000);
    }
}
int main(void)
{
    uint8 num_blink = 0U, i = 0;
    uint16 duty_cnt = 0;
    UserCountIrqCH0 = 0U;
    /* Initialize the Mcu driver */
    Mcu_Init(&Mcu_Config_VS_0);

    /* Initialize the clock tree */
    Mcu_InitClock(McuClockSettingConfig_0);

    /* Apply a mode configuration */
    Mcu_SetMode(McuModeSettingConf_0);

    Platform_Init(NULL_PTR);
    /* Initialize all pins using the Port driver */
    Port_Init(&Port_Config_VS_0);

    /* Initialize Mcl driver */
    Mcl_Init(&Mcl_Config_VS_0);
    /* Initialize the Icu driver */
    Icu_Init(NULL_PTR);

    Icu_EnableEdgeDetection(IcuChannel_0);
    Icu_EnableNotification(IcuChannel_0);

    /* Initialize Pwm driver , after that Led on*/
    Pwm_Init(&Pwm_Config_VS_0);
    /* PTA29 duty cycle is 50% */
    Pwm_SetDutyCycle(PwmChannel_0, 0X4000);

   // PTB16,pwm1 , emios0_ch4, use trigmux LCU output 2 Complementarity PWM
    Mcl_LcuSyncOutputValueType lcuEnable[2U];
    lcuEnable[0].LogicOutputId = 0;
    lcuEnable[0].Value = 1U;
    lcuEnable[1].LogicOutputId = 1;
    lcuEnable[1].Value = 1U;
    Mcl_SetLcuSyncOutputEnable(lcuEnable, 2U);

    TestDelay(DELAY_TIMER);

    /* Set new period for all channels used external counter bus */
    Mcl_Emios_SetCounterBusPeriod(MCL_EMIOS_1_CH_23, 4800, FALSE); // pwmchannel_0 10Khz
    Mcl_Emios_SetCounterBusPeriod(MCL_EMIOS_0_CH_22, 1200, FALSE);// for PwmChannel_1, 20Khz
// PWM0: 10kHZ
//PWM1: 20KHZ
//PWM2:1KHZ
    // PTA29 10kHZ
    /* PTA29 duty cycle is 50% */
    Pwm_SetDutyCycle(PwmChannel_0, 0X4000);
      /* Setup new duty cycle to the pin*/
    Pwm_SetDutyCycle(PwmChannel_1, 0x4000);

    for(i=0; i <= 10; i++)
    {
        duty_cnt = i * 0x800;
        Pwm_SetDutyCycle(PwmChannel_0, duty_cnt);
        TestDelay(DELAY_TIMER);
    }

    /* Using duty cycle 0% and 100% to Blink LED */
    while(1)
    {
        /* pwm1 when duty cycle is 75% */
        Pwm_SetDutyCycle(PwmChannel_1, 0X6000);
        //Led off
        Pwm_SetDutyCycle(PwmChannel_0, 0X0000); //red off
        Dio_WriteChannel(DioConf_DioChannel_Digital_ledgreenPTA30, STD_HIGH); //green on
        TestDelay(DELAY_TIMER);

        /* pwm 1 when duty cycle is 25% */
        Pwm_SetDutyCycle(PwmChannel_1, 0X2000);
        //Led ON
        Dio_WriteChannel(DioConf_DioChannel_Digital_ledgreenPTA30, STD_LOW);     //Green OFF
        Pwm_SetDutyCycle(PwmChannel_0, 0X8000); //RED ON
        TestDelay(DELAY_TIMER);
        num_blink++;
    }

    /* De-Initialize Pwm driver */
    Pwm_DeInit();

    //Exit_Example(TRUE);

    return 0U;
}

三,测试结果

上电后,板载红灯闪烁,然后从灭渐进变亮,并且和绿灯交替闪烁。

测试PWM1 PTB16: EMIOS0_CH4,稳定后波形为20Khz,占空比在25%,75%交替改变。PWM2 PTB14:EMIOS0_CH2,稳定后,频率为1KHZ, 随着板载的SW5按下,在25%,75%占空比交替改变。

测试PTD3,PTD2,可以看到是一对互补波形,并且和PTB16是一样的频率,占空比改变规律也一致,由此可知,本文的按键中断,3个主PWM,2个LCU PWM均已经工作。

图29


图30 文章配套代码链接:

https://community.nxp.com/t5/S32K-Knowledge-Base/RTD400-MCAL-K344-KEY-control-EMIOS-PWM/ta-p/1967347

相关推荐
嵌入式大圣1 小时前
单片机结合OpenCV
单片机·嵌入式硬件·opencv
日晨难再3 小时前
嵌入式:STM32的启动(Startup)文件解析
stm32·单片机·嵌入式硬件
yufengxinpian3 小时前
集成了高性能ARM Cortex-M0+处理器的一款SimpleLink 2.4 GHz无线模块-RF-BM-2340B1
单片机·嵌入式硬件·音视频·智能硬件
__基本操作__4 小时前
历遍单片机下的IIC设备[ESP--0]
单片机·嵌入式硬件
网易独家音乐人Mike Zhou10 小时前
【卡尔曼滤波】数据预测Prediction观测器的理论推导及应用 C语言、Python实现(Kalman Filter)
c语言·python·单片机·物联网·算法·嵌入式·iot
lantiandianzi17 小时前
基于单片机的多功能跑步机控制系统
单片机·嵌入式硬件
哔哥哔特商务网18 小时前
高集成的MCU方案已成电机应用趋势?
单片机·嵌入式硬件
跟着杰哥学嵌入式18 小时前
单片机进阶硬件部分_day2_项目实践
单片机·嵌入式硬件
东芝、铠侠总代1361006839319 小时前
浅谈TLP184小型平面光耦
单片机·嵌入式硬件·物联网·平面
lantiandianzi19 小时前
基于单片机中医药柜管理系统的设计
单片机·嵌入式硬件