你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32f030f4p6 DMA驱动2811 为什么PWM频率达不到800KHZ

[复制链接]
l441867854 提问时间:2019-7-12 11:35 /
本帖最后由 l441867854 于 2019-7-13 13:34 编辑

#define TIM_PERIOD                59                                        // Number of CPU cycles that will constitute 1 period
#define PWM_HIGH_WIDTH                45                                // Duty cycle of pwm signal for a logical 1 to be read by the ws2812 chip.
                                                                                                                                                //Duty cycle = PWM_HIGH_WIDTH/TIM_PERIOD*100
#define PWM_LOW_WIDTH                10        
uint8_t ledBuff[69]={0};        

void TIM3_PWM_Init(void)
{
GPIO_InitTypeDef  GPIO_InitStructure;
  TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_OCInitTypeDef       TIM_OCInitStructure;
  DMA_InitTypeDef DMA_InitStructure;          /* 配置相应引脚PB1*/
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOB|RCC_AHBPeriph_DMA1, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_PuPd =  GPIO_PuPd_UP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_PinAFConfig(GPIOB, GPIO_PinSource1, GPIO_AF_1);
GPIO_Init(GPIOB, &GPIO_InitStructure);

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //使能定时器3时钟
TIM_TimeBaseStructure.TIM_Period        = TIM_PERIOD;// 设置自动重装周期值
TIM_TimeBaseStructure.TIM_Prescaler     = 0;//设置预分频值
TIM_TimeBaseStructure.TIM_ClockDivision = 0;//设置时钟分割
TIM_TimeBaseStructure.TIM_CounterMode   = TIM_CounterMode_Up;//向上计数
TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);    //初始化定时器3
TIM_OCInitStructure.TIM_OCMode      = TIM_OCMode_PWM1;// PWM2模式
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;//比较输出使能
TIM_OCInitStructure.TIM_Pulse = 0;    TIM_OCInitStructure.TIM_OCPolarity  = TIM_OCPolarity_High;//输出高
TIM_OC4Init(TIM3, &TIM_OCInitStructure);    TIM_OC4PreloadConfig(TIM3, TIM_OCPreload_Enable);//使能预装载寄存器

DMA_DeInit(DMA1_Channel3);
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&(TIM3->CCR4);        // physical address of Timer 3 CCR4        DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ledBuff;                // this is the buffer memory
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;                                                // data shifted from memory to peripheral
DMA_InitStructure.DMA_BufferSize = sizeof(ledBuff);         
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;                                        // automatically increase buffer index
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;                                                        // stop DMA feed after buffer size is reached
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel3, &DMA_InitStructure);        
TIM_DMACmd(TIM3, TIM_DMA_CC4, ENABLE);
}        

void Ws2812_reflash(void)
{
DMA_SetCurrDataCounter(DMA1_Channel3, sizeof(ledBuff));
DMA_Cmd(DMA1_Channel3,ENABLE);
TIM_Cmd(TIM3, ENABLE);
while(!DMA_GetFlagStatus(DMA1_FLAG_TC3));
TIM_Cmd(TIM3, DISABLE);
DMA_Cmd(DMA1_Channel3,DISABLE);        DMA_ClearFlag(DMA1_FLAG_TC3);        
}        

void Set2812RGB(uint8_t redcolor,uint8_t greencolor,uint8_t bluecolor)
{
        uint8_t i=0;
        uint32_t rgb_value=redcolor<<16|greencolor<<8|bluecolor;
        
        
        for(i=0;i<69;i++)
        {
                if((rgb_value<<i)&0x800000)
                {
                        ledBuff=PWM_HIGH_WIDTH;
                }
                else
                {
                        ledBuff=PWM_LOW_WIDTH;
                }
                        
        }


int main(void)
{
        //delay_init();
        TIM3_PWM_Init();
        while(1)
        {
        
        Set2812RGB(0,57,0);
        Ws2812_reflash();

        }               

}

[img]file:///C:\Users\Administrator\AppData\Roaming\Tencent\Users\441867853\QQ\WinTemp\RichOle\WEUORX)~(R[ZUVX259~FMXM.png[/img]
QQ图片20190712113329.png

测试出来的PWM波形频率只有8K   为什么!!!

收藏 评论6 发布时间:2019-7-12 11:35

举报

6个回答
七哥 回答时间:2019-7-12 19:27:35
本帖最后由 toofree 于 2019-7-12 19:35 编辑

程序问题较多
uint8_t ledBuff[69]={0}; -- 定义是8位数组;
ledBuff=PWM_HIGH_WIDTH;  --这样赋值为哪般?ledBuff是指针,不是得给数组的元素赋值吗?赋完值后,指针也要增加的。或者改成ledBuff[i++] = PWM_HIGH_WIDTH;
if((rgb_value<<i)&0x800000)  ---这里的 rgb_value<<i 是个什么东西?i的取值可是0到68呀!

既然你的ledBuff是8位的数组,那么DMA传数也得是8位的。
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;

是不是应该改成字节呢?定时器是16位的,初始化高8位为0即可,DMA只传输低8位。
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;

另外,你对WS2812灯的理解是不是正确呢?这个灯是RGB888,正确点一颗灯,需要发送24个脉冲。依此类推,点2颗灯需要48个脉冲,3灯72个脉冲。而你用的ledBuff[69],69个元素,与谁都不搭边啊。


评分

参与人数 1蝴蝶豆 +3 收起 理由
STMCU + 3

查看全部评分

l441867854 回答时间:2019-7-12 11:45:20
哪位高手帮忙看下!!
l441867854 回答时间:2019-7-12 11:51:18
时钟使用的是默认库函数设置,没有更改。单独输出PWM输出的是正确的,但是DMA加上以后频率就变了。
l441867854 回答时间:2019-7-12 13:46:59
高手高手 高高手 帮忙看下
l441867854 回答时间:2019-7-12 16:10:09
Any one can give me a solution,help!!
l441867854 回答时间:2019-7-13 13:26:14
toofree 发表于 2019-7-12 19:27
程序问题较多
uint8_t ledBuff[69]={0}; -- 定义是8位数组;
ledBuff=PWM_HIGH_WIDTH;  --这样赋值为哪般? ...

1,是前面的数组位数定义错了,改成16位重新测试了一下,正常了。非常感谢
2,i在for里面有增加。本来是24,因为频率出不来,就改了。
3 这个程序只是初次来测试电平时间用的,只点一颗灯, 只是为了在后面加一个低电平复位时间。 非常感谢

所属标签

相似问题

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32Cube扩展软件包
意法半导体边缘AI套件
ST - 理想汽车豪华SUV案例
ST意法半导体智能家居案例
STM32 ARM Cortex 32位微控制器
关注我们
st-img 微信公众号
st-img 手机版