[应用方案] NUC451RG6AE实现的PID!

[复制链接]
1769|21
 楼主| doit888 发表于 2018-3-8 09:05 | 显示全部楼层 |阅读模式
NUC451RG6AE实现的PID!
增量的方式。开贴探讨一下啊!
xinpian101 发表于 2018-3-8 09:08 | 显示全部楼层
这个自带DSP啊,提供PID库
xinpian101 发表于 2018-3-8 09:09 | 显示全部楼层
程序数学计算太复杂,微控制器来不及运算吗?讯号处理、文件压缩时间太久该怎么办?新唐M4微控制器内含浮点运算单元和DSP,加速数**算解决难题!

新唐NuMicro® M451微控制器系列采ARM® Cortex®-M4内核,内含DSP和浮点运算单元,其强大的运算能力,可帮助程序加速运算。本次内容将介绍如何使用M451 实现DSP程序加速运算,其ARM®的CMSIS DSP Software Library提供了许多优化过后的数学程序运算,内容包含:基本数学加减乘除运算、三角函数运算、复数运算、矩阵运算、统计运算、滤波器应用、转移函数、线性插值与PID控制器…等,用户可以依需求自行组合多任务指令,迅速实现方程式,亦可直接调用函数进行运算。使用这些函数的优点即可缩短十多倍的计算时间,以硬件空间换取运算时间,让用户在处理大量计算或讯号处理之应用下,可更顺利的进行。

NuMicro® M451系列32位微控制器最高可运行至72 MHz,内建256K/128/72/40K字节FLASH存贮器、32/16K字节SRAM和4K字节用于在线程序设计的引导存贮器,具有丰富外设,其中内建高精度16位PWM且速度最高可达144MHz,由于内建DSP和浮点运算单元,非常适合数字讯号产品应用开发之使用、例如:变声装置,影像音源压缩,解析讯号和图片音源后制处理…等应用。
 楼主| doit888 发表于 2018-3-8 09:11 | 显示全部楼层
/**************************************************************************//**
* @file     main.c
* @version  V3.00
* $Revision: 8 $
* $Date: 15/09/02 10:04a $
* @brief    Implement timer counting in periodic mode.
* @note
* Copyright (C) 2013~2015 Nuvoton Technology Corp. All rights reserved.
******************************************************************************/
#include <stdio.h>
#include "string.h"

#include "M451Series.h"

#include "PID.h"

#define PLL_CLOCK           72000000
 楼主| doit888 发表于 2018-3-8 09:12 | 显示全部楼层
我这个 没有用 DSP的库……用的是一个 自己写的 增量式的PID
 楼主| doit888 发表于 2018-3-8 09:12 | 显示全部楼层
/*---------------------------------------------------------------------------------------------------------*/
/* Global Interface Variables Declarations                                                                 */
/*---------------------------------------------------------------------------------------------------------*/
volatile uint32_t g_au32TMRINTCount0 =0;
volatile uint32_t T0_FLAG =0;

volatile uint32_t TDt_FLAG =0;
volatile uint32_t TDt_Count =0;

volatile uint32_t g_u32AdcIntFlag, g_u32COVNUMFlag = 0;

uint8_t  u8Option, u32SAMPLECount = 0;
int32_t  i32ConversionData[8] = {0};
uint8_t RETUEN=0;
 楼主| doit888 发表于 2018-3-8 09:13 | 显示全部楼层
volatile uint8_t USART0_RCVBUF[255];
volatile uint8_t USART0_SNDBUF[255];
volatile uint8_t USART0_State;
volatile uint8_t USART0_FLAG;
volatile uint8_t USART0_CNT;
 楼主| doit888 发表于 2018-3-8 09:13 | 显示全部楼层
void UART0_Init(void)
{
    SYS_ResetModule(UART0_RST);
    UART_Open(UART0, 9600);
    UART_EnableInt(UART0, UART_INTEN_RDAIEN_Msk);
}
 楼主| doit888 发表于 2018-3-8 09:17 | 显示全部楼层
void UART0_IRQHandler(void)
{
    volatile uint32_t u32IntSts = UART0->INTSTS;;

    /* Rx Ready or Time-out INT*/
    if(UART_GET_INT_FLAG(UART0, UART_INTSTS_RDAINT_Msk))
    {
        USART0_RCVBUF[USART0_CNT]=UART_READ(UART0);

        if(USART0_CNT==0&&USART0_RCVBUF[0]!='<') return;      //第0号数据不是帧头

        USART0_CNT++;

        if(USART0_CNT==8)             //接收到11个数据
        {
            USART0_CNT=0;               //重新赋值,准备下一帧数据的接收

            USART0_FLAG=1;

        }

    }
}
 楼主| doit888 发表于 2018-3-8 09:18 | 显示全部楼层
//串口0解析数据
void UART0TEST(void)
{

    if(USART0_FLAG==1)
    {
        USART0_FLAG=0;


        switch(USART0_RCVBUF[1])
        {

        case 'C'://
            switch(USART0_RCVBUF[2])
            {

            case '1'://p
                Kp=USART0_RCVBUF[3];
                PID_1.Proportion=Kp;
                printf("Kp@:%d;\r\n",PID_1.Proportion);
                break;

            case '2'://i
                Ki=USART0_RCVBUF[3];
                PID_1.Integral=Ki;
                printf("Ki@:%d;\r\n",PID_1.Integral);
                break;

            case '3'://d
                Kd=USART0_RCVBUF[3];
                PID_1.Derivative=Kd;
                printf("Kd@:%d;\r\n",PID_1.Derivative);
                break;

            case '4'://dt
                Dt=USART0_RCVBUF[3];
                printf("Dt@:%d;\r\n",Dt);
                break;

            case '5'://sv
                Sv=USART0_RCVBUF[3];
                printf("Sv@:%d;\r\n",Sv);
                break;

            case '6'://d
                RETUEN=USART0_RCVBUF[3];
                printf("RETUEN@:%d;\r\n",RETUEN);
                break;

            }
            break;

        }
    }
}
 楼主| doit888 发表于 2018-3-8 09:18 | 显示全部楼层
/**
* @brief       Timer0 IRQ
*
* @param       None
*
* @return      None
*
* @Details     The Timer0 default IRQ, declared in startup_M451Series.s.
*/
void TMR0_IRQHandler(void)
{
    if(TIMER_GetIntFlag(TIMER0) == 1)
    {
        /* Clear Timer0 time-out interrupt flag */
        TIMER_ClearIntFlag(TIMER0);

        if(++g_au32TMRINTCount0>99)
        {
            g_au32TMRINTCount0=0;
            T0_FLAG=1;
        }
        if(++TDt_Count>Dt)
        {
            TDt_Count=0;
            TDt_FLAG=1;
        }
    }
}
 楼主| doit888 发表于 2018-3-8 09:18 | 显示全部楼层
void SYS_Init(void)
{
    /*---------------------------------------------------------------------------------------------------------*/
    /* Init System Clock                                                                                       */
    /*---------------------------------------------------------------------------------------------------------*/
    /* Enable HIRC clock */
    CLK_EnableXtalRC(CLK_PWRCTL_HIRCEN_Msk);

    /* Waiting for HIRC clock ready */
    CLK_WaitClockReady(CLK_STATUS_HIRCSTB_Msk);

    /* Switch HCLK clock source to HIRC */
    CLK_SetHCLK(CLK_CLKSEL0_HCLKSEL_HIRC, CLK_CLKDIV0_HCLK(1));

    /* Enable HXT */
    CLK_EnableXtalRC(CLK_PWRCTL_HXTEN_Msk);

    /* Waiting for clock ready */
    CLK_WaitClockReady(CLK_STATUS_HXTSTB_Msk);

    /* Set core clock as PLL_CLOCK from PLL and SysTick source to HCLK/2*/
    CLK_SetCoreClock(PLL_CLOCK);
    CLK_SetSysTickClockSrc(CLK_CLKSEL0_STCLKSEL_HCLK_DIV2);

    /* Enable peripheral clock */
    CLK_EnableModuleClock(UART0_MODULE);
    CLK_EnableModuleClock(TMR0_MODULE);
    CLK_EnableModuleClock(EADC_MODULE);


    /* Peripheral clock source */
    CLK_SetModuleClock(UART0_MODULE, CLK_CLKSEL1_UARTSEL_PLL, CLK_CLKDIV0_UART(1));
    CLK_SetModuleClock(TMR0_MODULE, CLK_CLKSEL1_TMR0SEL_PCLK0, 0);
    CLK_SetModuleClock(EADC_MODULE, 0, CLK_CLKDIV0_EADC(8));

    //SYS->GPA_MFPH = 0x00000000;
    //SYS->GPA_MFPL = 0x00002200;
    //SYS->GPB_MFPH = 0x00000000;
    //SYS->GPB_MFPL = 0x00000001;
    //SYS->GPC_MFPH = 00000000;
    //SYS->GPC_MFPL = 0x00000000;
    //SYS->GPD_MFPH = 0x00000000;
    //SYS->GPD_MFPL = 0x00000000;
    //SYS->GPE_MFPH = 0x00000000;
    //SYS->GPE_MFPL = 0x00000000;
    //SYS->GPF_MFPL = 0x01100000;

    //If the defines do not exist in your project, please refer to the related sys.h in the sys_h folder appended to the tool package.
    SYS->GPA_MFPH = 0x00000000;
    SYS->GPA_MFPL = SYS_GPA_MFPL_PA3MFP_UART0_RXD | SYS_GPA_MFPL_PA2MFP_UART0_TXD;
    SYS->GPB_MFPH = 0x00000000;
    SYS->GPB_MFPL = SYS_GPB_MFPL_PB0MFP_EADC_CH0;
    SYS->GPC_MFPH = 0x00000000;
    SYS->GPC_MFPL = 0x00000000;
    SYS->GPD_MFPH = 0x00000000;
    SYS->GPD_MFPL = 0x00000000;
    SYS->GPE_MFPH = 0x00000000;
    SYS->GPE_MFPL = 0x00000000;
    SYS->GPF_MFPL = SYS_GPF_MFPL_PF6MFP_ICE_DAT | SYS_GPF_MFPL_PF5MFP_ICE_CLK;

}
 楼主| doit888 发表于 2018-3-8 09:19 | 显示全部楼层
void SYS_Init(void)

这个函数 试用 新唐的 寄存器 配置軟體开发

主要是 GPIO的配置 以及CLOCK的时钟的配置
xinpian101 发表于 2018-3-8 09:23 | 显示全部楼层
  1. /****************************************************************************
  2. * [url=home.php?mod=space&uid=288409]@file[/url]     main.c
  3. * [url=home.php?mod=space&uid=895143]@version[/url]  V1.0
  4. * $Date: 15/09/02 10:04a $
  5. * @brief
  6. *         Display how to use M4 DSP PID Controller
  7. *                      and compare with calcultion without DSP
  8. * @note
  9. * Copyright (C) 2016 Nuvoton Technology Corp. All rights reserved.
  10. *
  11. ******************************************************************************/
  12. #include <stdio.h>
  13. #include "M451Series.h"
  14. #include "arm_math.h"
  15. #define USE_DSP
  16. int i,j,CalTime;
  17. arm_pid_instance_f32 PIDS;
  18. float32_t output[100],ee;
  19. float A0,A1,A2,state[3],Kp=0.4,Ki=0.4,Kd=0,target,ival;
  20. float PID(float in)
  21. {
  22.     float out;
  23.                 A0=Kp+Ki;
  24.                 A1 = -Kp-(2* Kd);
  25.                 A2 = Kd;
  26.     /* y[n] = y[n-1] + A0 * x[n] + A1 * x[n-1] + A2 * x[n-2]  */
  27.     out = (A0 * in) + (A1 * state[0]) + (A2 * state[1]) + (state[2]);

  28.     /* Update state */
  29.     state[1] = state[0];
  30.     state[0] = in;
  31.     state[2] = out;

  32.     /* return to application */
  33.     return (out);
  34. }
  35. void SYS_Init(void)
  36. {
  37.     /*---------------------------------------------------------------------------------------------------------*/
  38.     /* Init System Clock                                                                                       */
  39.     /*---------------------------------------------------------------------------------------------------------*/
  40.     /* Enable HIRC clock */
  41.     CLK_EnableXtalRC(CLK_PWRCTL_HIRCEN_Msk);

  42.     /* Waiting for HIRC clock ready */
  43.     CLK_WaitClockReady(CLK_STATUS_HIRCSTB_Msk);

  44.     /* Switch HCLK clock source to HIRC */
  45.     CLK_SetHCLK(CLK_CLKSEL0_HCLKSEL_HIRC, CLK_CLKDIV0_HCLK(1));

  46.     /* Enable HXT */
  47.     CLK_EnableXtalRC(CLK_PWRCTL_HXTEN_Msk);

  48.     /* Waiting for clock ready */
  49.     CLK_WaitClockReady(CLK_STATUS_HXTSTB_Msk);

  50.     /* Set core clock as PLL_CLOCK from PLL and SysTick source to HCLK/2*/
  51.     CLK_SetCoreClock(72000000);
  52.     CLK_SetSysTickClockSrc(CLK_CLKSEL0_STCLKSEL_HCLK_DIV2);

  53.     /* Enable peripheral clock */
  54.     CLK_EnableModuleClock(UART0_MODULE);
  55.     CLK_EnableModuleClock(TMR0_MODULE);

  56.     /* Peripheral clock source */
  57.     CLK_SetModuleClock(UART0_MODULE, CLK_CLKSEL1_TMR0SEL_HXT, CLK_CLKDIV0_UART(1));
  58.     CLK_SetModuleClock(TMR0_MODULE, CLK_CLKSEL1_TMR0SEL_HXT, 0);

  59. }
  60. void UART_Init(void)
  61. {
  62.     /* Set PD multi-function pins for UART0 RXD, TXD */
  63.     SYS->GPD_MFPL = SYS_GPD_MFPL_PD0MFP_UART0_RXD | SYS_GPD_MFPL_PD1MFP_UART0_TXD;
  64.     /* Reset UART module */
  65.     SYS_ResetModule(UART0_RST);

  66.     /* Configure UART0 and set UART0 Baudrate */
  67.     UART_Open(UART0, 115200);
  68. }

  69. /*---------------------------------------------------------------------------------------------------------*/
  70. /*  Main Function                                                                                          */
  71. /*---------------------------------------------------------------------------------------------------------*/
  72. int32_t main(void)
  73. {               
  74.           uint32_t i;
  75.                 PIDS.Kp=0.4;                               
  76.                 PIDS.Ki=0.4;
  77.                 PIDS.Kd=0;
  78.                 /* Target value*/
  79.                 target=500;
  80.                 /* Inital value */
  81.                 ival=0;
  82.                 /* Initial value and target value error */
  83.                 ee=target-ival;
  84.        
  85.                 /* Unlock protected registers */
  86.     SYS_UnlockReg();

  87.     /* Init System, peripheral clock and multi-function I/O */
  88.     SYS_Init();

  89.     /* Lock protected registers */
  90.     SYS_LockReg();

  91.     /* Init UART for printf */
  92.     UART_Init();
  93.                
  94.                 TIMER_Open(TIMER0, TIMER_CONTINUOUS_MODE, 1);
  95.                 TIMER_Start(TIMER0);
  96. #ifdef USE_DSP
  97.                 /* Initial DSP PID controller function*/
  98.                 arm_pid_init_f32(&PIDS,0);
  99.                 /* Calculate PID controller function 100 times*/
  100.                 for(i=1;i<100;i++)
  101.                 {
  102.                         output[i]=arm_pid_f32(&PIDS,ee);
  103.                         //printf("%0.2f\n",output[i]);
  104.                         /* Update error */
  105.                         ee=target-output[i-1];
  106.                 }
  107. #else
  108.                
  109.                 for(i=1;i<100;i++)
  110.                 {
  111.                         output[i]=PID(ee);
  112.                         ee=target-output[i-1];
  113.                 }
  114.                
  115. #endif
  116.           TIMER_Close(TIMER0);
  117.                 CalTime=TIMER_GetCounter(TIMER0);       
  118.                 printf("time is %d \n",CalTime);

  119.     while(1);                             /* main function does not return */
  120. }
dongnanxibei 发表于 2018-3-8 14:42 | 显示全部楼层
楼主你这种PID好用不啊,如果有内核的DSP不用,耗费ARM CPU跑PID,不是浪费吗
 楼主| doit888 发表于 2018-3-9 13:22 | 显示全部楼层
我感觉 我这种才是精简的啊!
 楼主| doit888 发表于 2018-3-9 13:23 | 显示全部楼层
//PID初始化
void PIDInit (PID *pp)
{
    memset( pp,0,sizeof(PID));
}
 楼主| doit888 发表于 2018-3-9 13:23 | 显示全部楼层
//PID计算
int PIDCalc( struct PID *pp, unsigned int NextPoint )
{
    int dError,Error,pError;
    //增量法计算公式:
    //Pdt=Kp*[E(t)-E(t-1)]+Ki*E(t)+Kd*[E(t)-2*E(t-1)+E(t-2)]
    Error = Sv - NextPoint;       // 偏差E(t)
    pError=Error-pp->LastError;         //E(t)-E(t-1)
    dError=Error-2*pp->LastError+pp->PrevError; //E(t)-2*E(t-1)+E(t-2)
    pp->PrevError = pp->LastError;
    pp->LastError = Error;
    return (
               pp->Proportion * pError        //比例
               + pp->Integral *Error  //积分项
               + pp->Derivative * dError          // 微分项
           );
}
 楼主| doit888 发表于 2018-3-9 13:23 | 显示全部楼层
typedef struct PID
{

    unsigned char Proportion; // 比例常数 Proportional Const
    unsigned char Integral; // 积分常数 Integral Const
    unsigned char Derivative; // 微分常数 Derivative Const
    unsigned int LastError; // Error[-1]
    unsigned int PrevError; // Error[-2]

} PID;
 楼主| doit888 发表于 2018-3-9 13:24 | 显示全部楼层
extern volatile PID PID_1;


extern int rOut;
extern  unsigned int rIn;
extern unsigned char Kp;
extern unsigned char Ki;
extern unsigned char Kd;
extern unsigned char Dt;
extern unsigned char Sv;
您需要登录后才可以回帖 登录 | 注册

本版积分规则

10

主题

173

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部