mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-25 09:07:23 +08:00
5debfdd84d
增加c28x芯片的pwm驱动 已经在TMS320F28379中通过测试 pwm设备框架增加如下方法: #define PWM_CMD_SET_DEAD_TIME (RT_DEVICE_CTRL_BASE(PWM) + 8) #define PWM_CMD_SET_PHASE (RT_DEVICE_CTRL_BASE(PWM) + 9) #define PWM_CMD_ENABLE_IRQ (RT_DEVICE_CTRL_BASE(PWM) + 10) #define PWM_CMD_DISABLE_IRQ (RT_DEVICE_CTRL_BASE(PWM) + 11)
479 lines
13 KiB
C
479 lines
13 KiB
C
/*
|
|
* Copyright (c) 2006-2022, RT-Thread Development Team
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Change Logs:
|
|
* Date Author Notes
|
|
* 2022-09-24 qiyu first version
|
|
*/
|
|
|
|
#include "rtdbg.h"
|
|
#include "drv_pwm.h"
|
|
#include "F2837xD_device.h"
|
|
#include "F28x_Project.h" /* Device Headerfile and Examples Include File */
|
|
#include "drv_config.h"
|
|
#include "F2837xD_epwm.h"
|
|
/*
|
|
* for now, cpu rate is a fixed value, waiting to be modified to an auto-ajustable variable.
|
|
*/
|
|
|
|
rt_err_t rt_device_pwm_register(struct rt_device_pwm *device, const char *name, const struct rt_pwm_ops *ops, const void *user_data);
|
|
|
|
#define CPU_FREQUENCY 200e6
|
|
/*
|
|
* TODO unknown issue, according to the configuration,
|
|
* this division should be 2,
|
|
* while 2 is inconsistent with the measured result
|
|
*/
|
|
#define PWM_DIVISION 2
|
|
#define CHANNEL_A 1
|
|
#define CHANNEL_B 2
|
|
#define UPDOWN 1
|
|
|
|
enum
|
|
{
|
|
#ifdef BSP_USING_PWM1
|
|
PWM1_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM2
|
|
PWM2_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM3
|
|
PWM3_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM4
|
|
PWM4_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM5
|
|
PWM5_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM6
|
|
PWM6_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM7
|
|
PWM7_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM8
|
|
PWM8_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM9
|
|
PWM9_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM10
|
|
PWM10_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM11
|
|
PWM11_INDEX,
|
|
#endif
|
|
#ifdef BSP_USING_PWM12
|
|
PWM12_INDEX,
|
|
#endif
|
|
};
|
|
|
|
static rt_err_t drv_pwm_control(struct rt_device_pwm *device, int cmd, void *arg);
|
|
|
|
static struct rt_pwm_ops rt_pwm_ops =
|
|
{
|
|
drv_pwm_control
|
|
};
|
|
|
|
static struct c28x_pwm c28x_pwm_obj[] =
|
|
{
|
|
#ifdef BSP_USING_PWM1
|
|
PWM1_CONFIG,
|
|
#endif
|
|
|
|
#ifdef BSP_USING_PWM2
|
|
PWM2_CONFIG,
|
|
#endif
|
|
|
|
#ifdef BSP_USING_PWM3
|
|
PWM3_CONFIG,
|
|
#endif
|
|
|
|
#ifdef BSP_USING_PWM4
|
|
PWM4_CONFIG,
|
|
#endif
|
|
|
|
#ifdef BSP_USING_PWM5
|
|
PWM5_CONFIG,
|
|
#endif
|
|
|
|
#ifdef BSP_USING_PWM6
|
|
PWM6_CONFIG,
|
|
#endif
|
|
|
|
#ifdef BSP_USING_PWM7
|
|
PWM7_CONFIG,
|
|
#endif
|
|
|
|
#ifdef BSP_USING_PWM8
|
|
PWM8_CONFIG,
|
|
#endif
|
|
|
|
};
|
|
|
|
static rt_err_t drv_pwm_set(volatile struct EPWM_REGS *epwm,struct rt_pwm_configuration *configuration)
|
|
{
|
|
if(epwm == RT_NULL)
|
|
{
|
|
return -RT_ERROR;
|
|
}
|
|
|
|
|
|
/*
|
|
* TODO Unknown problem
|
|
* the clock division configuration of PWM module is 1
|
|
* however, the experiment result shows the division is 2
|
|
*/
|
|
|
|
/* Set the configuration of PWM according to the parameter*/
|
|
rt_uint32_t prd = configuration->period/(1e9/(CPU_FREQUENCY/PWM_DIVISION))/2;
|
|
rt_uint32_t comp = prd*configuration->pulse/configuration->period;
|
|
rt_uint32_t dead_time = configuration->dead_time/(1e9/(CPU_FREQUENCY/PWM_DIVISION));
|
|
rt_uint32_t phase = configuration->phase;
|
|
|
|
epwm->TBPRD = prd; /* Set timer period*/
|
|
epwm->TBCTR = 0x0000; /* Clear counter*/
|
|
epwm->TBCTL.bit.CTRMODE = RT_CTRMODE; /* Count up*/
|
|
epwm->TBCTL.bit.HSPCLKDIV = TB_DIV1; /* Clock ratio to SYSCLKOUT*/
|
|
epwm->TBCTL.bit.CLKDIV = TB_DIV1;
|
|
epwm->CMPCTL.bit.SHDWAMODE = RT_SHADOW_MODE; /* Load registers every ZERO*/
|
|
epwm->CMPCTL.bit.SHDWBMODE = RT_SHADOW_MODE;
|
|
epwm->CMPCTL.bit.LOADAMODE = RT_LOAD_TIME;
|
|
epwm->CMPCTL.bit.LOADBMODE = RT_LOAD_TIME;
|
|
/* Setup compare */
|
|
if(configuration->channel == CHANNEL_A)
|
|
{
|
|
epwm->CMPA.bit.CMPA = comp;
|
|
}else
|
|
{
|
|
epwm->CMPB.bit.CMPB = comp;
|
|
}
|
|
|
|
/* Set actions */
|
|
epwm->AQCTLA.bit.CAU = AQ_CLEAR; /* Set PWMA on Zero*/
|
|
epwm->AQCTLA.bit.CAD = AQ_SET;
|
|
epwm->AQCTLB.bit.CBU = AQ_CLEAR; /* Set PWMB on Zero*/
|
|
epwm->AQCTLB.bit.CBD = AQ_SET;
|
|
|
|
/* Active Low PWMs - Setup Deadband */
|
|
/* TODO finish complementary setting */
|
|
epwm->DBCTL.bit.POLSEL = DB_ACTV_HIC;
|
|
epwm->DBRED.bit.DBRED = dead_time;
|
|
epwm->DBFED.bit.DBFED = dead_time;
|
|
epwm->DBCTL.bit.OUT_MODE = DB_FULL_ENABLE;
|
|
/*
|
|
if(configuration->complementary)
|
|
{
|
|
}
|
|
else
|
|
{
|
|
epwm->DBRED.bit.DBRED = 0;
|
|
epwm->DBFED.bit.DBFED = 0;
|
|
epwm->DBCTL.bit.POLSEL = DB_ACTV_HI;
|
|
epwm->DBCTL.bit.OUT_MODE = DB_DISABLE;
|
|
}
|
|
*/
|
|
|
|
epwm->DBCTL.bit.IN_MODE = DBA_ALL;
|
|
/* if disable dead time, set dead_time to 0 */
|
|
|
|
epwm->ETSEL.bit.INTSEL = ET_CTR_ZERO; /* Select INT on Zero event */
|
|
epwm->ETPS.bit.INTPRD = ET_1ST; /* Generate INT on 1st event */
|
|
|
|
if(phase<180)
|
|
{
|
|
epwm->TBPHS.bit.TBPHS = prd * phase/180;
|
|
epwm->TBCTL.bit.PHSDIR = 0; /* count up */
|
|
}else
|
|
{
|
|
epwm->TBPHS.bit.TBPHS = prd-prd * (phase-180)/180;
|
|
epwm->TBCTL.bit.PHSDIR = 1; /* count up*/
|
|
}
|
|
if(epwm == &EPwm1Regs)
|
|
{
|
|
epwm->TBCTL.bit.PHSEN = TB_DISABLE; /* Disable phase loading */
|
|
epwm->TBCTL.bit.SYNCOSEL = TB_CTR_ZERO;
|
|
}else
|
|
{
|
|
epwm->TBCTL.bit.PHSEN = TB_ENABLE; /* Disable phase loading */
|
|
epwm->TBCTL.bit.SYNCOSEL = TB_SYNC_IN;
|
|
}
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t drv_pwm_get(struct EPWM_REGS *epwm,struct rt_pwm_configuration *configuration)
|
|
{
|
|
/* Retrieve the pwm configuration */
|
|
if(epwm == RT_NULL)
|
|
{
|
|
return -RT_ERROR;
|
|
}
|
|
rt_uint32_t prd = epwm->TBPRD;
|
|
rt_uint32_t comp = epwm->CMPA.bit.CMPA;
|
|
if(UPDOWN)
|
|
{
|
|
/* if in updown mode, period in configuration has to be doubled */
|
|
configuration->period = prd*(1e9/(CPU_FREQUENCY/PWM_DIVISION))*2;
|
|
}
|
|
else
|
|
{
|
|
configuration->period = prd*(1e9/(CPU_FREQUENCY/PWM_DIVISION));
|
|
}
|
|
configuration->pulse = comp*configuration->period/prd;
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t drv_pwm_set_period(struct EPWM_REGS *epwm, rt_uint32_t period)
|
|
{
|
|
if(epwm == RT_NULL)
|
|
{
|
|
return -RT_ERROR;
|
|
}
|
|
rt_uint32_t prd = period/(1e9/(CPU_FREQUENCY/PWM_DIVISION))/2;
|
|
epwm->TBPRD = prd; /* Set timer period */
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t drv_pwm_set_pulse(struct EPWM_REGS *epwm, int channel, rt_uint32_t pulse)
|
|
{
|
|
if(epwm == RT_NULL)
|
|
{
|
|
return -RT_ERROR;
|
|
}
|
|
rt_uint32_t comp = pulse/(1e9/(CPU_FREQUENCY/PWM_DIVISION));
|
|
if(channel == CHANNEL_A)
|
|
{
|
|
epwm->CMPA.bit.CMPA = comp; /* set comparator value */
|
|
}else
|
|
{
|
|
epwm->CMPB.bit.CMPB = comp; /* set comparator value */
|
|
}
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t drv_pwm_set_dead_time(struct EPWM_REGS *epwm, rt_uint32_t dead_time)
|
|
{
|
|
if(epwm == RT_NULL)
|
|
{
|
|
return -RT_ERROR;
|
|
}
|
|
rt_uint32_t _dead_time = dead_time/(1e9/(CPU_FREQUENCY/PWM_DIVISION));
|
|
epwm->DBRED.bit.DBRED = _dead_time; /* rising dead time */
|
|
epwm->DBFED.bit.DBFED = _dead_time; /* falling dead time */
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t drv_pwm_set_phase(struct EPWM_REGS *epwm, rt_uint32_t phase)
|
|
{
|
|
if(epwm == RT_NULL)
|
|
{
|
|
return -RT_ERROR;
|
|
}
|
|
if(phase<180)
|
|
{
|
|
epwm->TBPHS.bit.TBPHS = epwm->TBPRD * phase/180;
|
|
epwm->TBCTL.bit.PHSDIR = 0;/* count up */
|
|
}else
|
|
{
|
|
epwm->TBPHS.bit.TBPHS = epwm->TBPRD-epwm->TBPRD * (phase-180)/180;
|
|
epwm->TBCTL.bit.PHSDIR = 1;/* count up */
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t drv_pwm_enable_irq(volatile struct EPWM_REGS *epwm,rt_bool_t enable)
|
|
{
|
|
if(epwm == RT_NULL)
|
|
{
|
|
return -RT_ERROR;
|
|
}
|
|
if(enable == RT_TRUE)
|
|
{
|
|
/* Interrupt setting */
|
|
epwm->ETSEL.bit.INTEN = 1; /* Enable INT */
|
|
}else{
|
|
epwm->ETSEL.bit.INTEN = 0; /* Enable INT */
|
|
}
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t drv_pwm_enable(volatile struct EPWM_REGS *epwm,rt_bool_t enable)
|
|
{
|
|
/*
|
|
* TODO
|
|
* Still not sure about how to stop PWM in C2000
|
|
*/
|
|
if(epwm == RT_NULL)
|
|
{
|
|
return -RT_ERROR;
|
|
}
|
|
if(enable == RT_TRUE)
|
|
{
|
|
/* clear trip zone flag */
|
|
EALLOW;
|
|
epwm->TZCLR.bit.OST = 1;
|
|
EDIS;
|
|
}
|
|
else
|
|
{
|
|
/* set trip zone flag */
|
|
EALLOW;
|
|
epwm->TZFRC.bit.OST = 1;
|
|
EDIS;
|
|
}
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t drv_pwm_control(struct rt_device_pwm *device, int cmd, void *arg)
|
|
{
|
|
struct rt_pwm_configuration *configuration = (struct rt_pwm_configuration *)arg;
|
|
struct c28x_pwm *pwm = (struct c28x_pwm *)device->parent.user_data;
|
|
|
|
switch (cmd)
|
|
{
|
|
case PWM_CMD_ENABLE:
|
|
return drv_pwm_enable((struct EPWM_REGS *)(pwm->pwm_regs), RT_TRUE);
|
|
case PWM_CMD_DISABLE:
|
|
return drv_pwm_enable((struct EPWM_REGS *)(pwm->pwm_regs), RT_FALSE);
|
|
case PWM_CMD_SET:
|
|
return drv_pwm_set((struct EPWM_REGS *)(pwm->pwm_regs), configuration);
|
|
case PWM_CMD_GET:
|
|
return drv_pwm_get((struct EPWM_REGS *)(pwm->pwm_regs), configuration);
|
|
case PWM_CMD_SET_PERIOD:
|
|
return drv_pwm_set_period((struct EPWM_REGS *)(pwm->pwm_regs), configuration->period);
|
|
case PWM_CMD_SET_PULSE:
|
|
return drv_pwm_set_pulse((struct EPWM_REGS *)(pwm->pwm_regs), configuration->channel,configuration->pulse);
|
|
case PWM_CMD_SET_DEAD_TIME:
|
|
return drv_pwm_set_dead_time((struct EPWM_REGS *)(pwm->pwm_regs), configuration->dead_time);
|
|
case PWM_CMD_SET_PHASE:
|
|
return drv_pwm_set_phase((struct EPWM_REGS *)(pwm->pwm_regs), configuration->phase);
|
|
case PWM_CMD_ENABLE_IRQ:
|
|
return drv_pwm_enable_irq((struct EPWM_REGS *)(pwm->pwm_regs), RT_TRUE);
|
|
case PWM_CMD_DISABLE_IRQ:
|
|
return drv_pwm_enable_irq((struct EPWM_REGS *)(pwm->pwm_regs), RT_FALSE);
|
|
default:
|
|
return RT_EINVAL;
|
|
}
|
|
}
|
|
|
|
static void pwm_isr(struct rt_device_pwm *rt_pwm)
|
|
{
|
|
struct c28x_pwm *pwm;
|
|
pwm = (struct c28x_pwm *)rt_pwm->parent.user_data;
|
|
PieCtrlRegs.PIEACK.all = PIEACK_GROUP3;
|
|
pwm->pwm_regs->ETCLR.bit.INT = 1;
|
|
}
|
|
|
|
#define EPWM_ISR_DEFINE(i) void EPWM##i##_Isr(){\
|
|
rt_interrupt_enter(); \
|
|
pwm_isr(&(c28x_pwm_obj[PWM##i##_INDEX].pwm_device)); \
|
|
rt_interrupt_leave(); \
|
|
}
|
|
|
|
#ifdef BSP_USING_PWM1
|
|
EPWM_ISR_DEFINE(1)
|
|
#endif
|
|
|
|
void EPWM1_Isr();
|
|
|
|
static int c28x_hw_pwm_init(struct c28x_pwm *device)
|
|
{
|
|
EALLOW;
|
|
/* Assigning ISR to PIE */
|
|
PieVectTable.EPWM1_INT = &EPWM1_Isr;
|
|
/* ENABLE Interrupt */
|
|
EDIS;
|
|
IER |= M_INT3;
|
|
rt_err_t result = 0;
|
|
|
|
EALLOW;
|
|
#ifdef BSP_USING_PWM1
|
|
GpioCtrlRegs.GPAPUD.all |= 5<<(1-1)*4; /* Disable pull-up on GPIO0 (EPWM1A) */
|
|
GpioCtrlRegs.GPAMUX1.all|= 5<<(1-1)*4; /* Configure GPIO0 as EPWM1A */
|
|
EPwm1Regs.TZCTL.bit.TZA = TZ_OFF; /* diable A when trip zone */
|
|
EPwm1Regs.TZCTL.bit.TZB = TZ_OFF; /* diable B when trip zone */
|
|
#endif
|
|
#ifdef BSP_USING_PWM2
|
|
GpioCtrlRegs.GPAPUD.all |= 5<<(2-1)*4; /* Disable pull-up on GPIO0 (EPWM1A) */
|
|
GpioCtrlRegs.GPAMUX1.all|= 5<<(2-1)*4; /* Configure GPIO0 as EPWM1A */
|
|
EPwm2Regs.TZCTL.bit.TZA = TZ_OFF; /* diable A when trip zone */
|
|
EPwm2Regs.TZCTL.bit.TZB = TZ_OFF; /* diable B when trip zone */
|
|
#endif
|
|
#ifdef BSP_USING_PWM3
|
|
GpioCtrlRegs.GPAPUD.all |= 5<<(3-1)*4; /* Disable pull-up on GPIO0 (EPWM1A) */
|
|
GpioCtrlRegs.GPAMUX1.all|= 5<<(3-1)*4; /* Configure GPIO0 as EPWM1A */
|
|
EPwm3Regs.TZCTL.bit.TZA = TZ_OFF; /* diable A when trip zone */
|
|
EPwm3Regs.TZCTL.bit.TZB = TZ_OFF; /* diable B when trip zone */
|
|
#endif
|
|
#ifdef BSP_USING_PWM4
|
|
GpioCtrlRegs.GPAPUD.all |= 5<<(4-1)*4; /* Disable pull-up on GPIO0 (EPWM1A) */
|
|
GpioCtrlRegs.GPAMUX1.all|= 5<<(4-1)*4; /* Configure GPIO0 as EPWM1A */
|
|
EPwm4Regs.TZCTL.bit.TZA = TZ_OFF; /* diable A when trip zone */
|
|
EPwm4Regs.TZCTL.bit.TZB = TZ_OFF; /* diable B when trip zone */
|
|
#endif
|
|
#ifdef BSP_USING_PWM5
|
|
GpioCtrlRegs.GPAPUD.all |= 5<<(5-1)*4; /* Disable pull-up on GPIO0 (EPWM1A) */
|
|
GpioCtrlRegs.GPAMUX1.all|= 5<<(5-1)*4; /* Configure GPIO0 as EPWM1A */
|
|
EPwm5Regs.TZCTL.bit.TZA = TZ_OFF; /* diable A when trip zone */
|
|
EPwm5Regs.TZCTL.bit.TZB = TZ_OFF; /* diable B when trip zone */
|
|
#endif
|
|
EDIS;
|
|
|
|
return result;
|
|
}
|
|
|
|
int c28x_pwm_init(void)
|
|
{
|
|
int i = 0;
|
|
int result = RT_EOK;
|
|
|
|
for (i = 0; i < sizeof(c28x_pwm_obj) / sizeof(c28x_pwm_obj[0]); i++)
|
|
{
|
|
/* pwm init */
|
|
if (c28x_hw_pwm_init(&c28x_pwm_obj[i]) != RT_EOK)
|
|
{
|
|
LOG_E("%s init failed", c28x_pwm_obj[i].name);
|
|
result = -RT_ERROR;
|
|
return result;
|
|
}
|
|
else
|
|
{
|
|
LOG_D("%s init success", c28x_pwm_obj[i].name);
|
|
|
|
/* register pwm device */
|
|
if (rt_device_pwm_register(&c28x_pwm_obj[i].pwm_device, c28x_pwm_obj[i].name, &rt_pwm_ops, &c28x_pwm_obj[i]) == RT_EOK)
|
|
{
|
|
LOG_D("%s register success", c28x_pwm_obj[i].name);
|
|
}
|
|
else
|
|
{
|
|
LOG_E("%s register failed", c28x_pwm_obj[i].name);
|
|
result = -RT_ERROR;
|
|
}
|
|
}
|
|
}
|
|
struct rt_pwm_configuration config_tmp1 =
|
|
{
|
|
.channel = CHANNEL_A,
|
|
.period = 10000,
|
|
.pulse = 5000,
|
|
.dead_time = 100,
|
|
.phase = 0,
|
|
.complementary = RT_TRUE
|
|
};
|
|
drv_pwm_set(c28x_pwm_obj[0].pwm_regs,&config_tmp1);
|
|
config_tmp1.phase = 180;
|
|
drv_pwm_set(c28x_pwm_obj[1].pwm_regs,&config_tmp1);
|
|
config_tmp1.phase = 90;
|
|
drv_pwm_set(c28x_pwm_obj[2].pwm_regs,&config_tmp1);
|
|
config_tmp1.phase = 270;
|
|
drv_pwm_set(c28x_pwm_obj[3].pwm_regs,&config_tmp1);
|
|
return result;
|
|
|
|
}
|
|
INIT_DEVICE_EXPORT(c28x_pwm_init);
|