[DSP编程] DSP编程 做实验的思路?大家帮忙看看对不对?

[复制链接]
1875|8
 楼主| 潇湘夜雨涩 发表于 2016-11-16 16:30 | 显示全部楼层 |阅读模式
刚开始学习dsp  28335 , 在学习了epwm 和adc 模块之后,自己想做个实验,但是不知道思路对不对?麻烦大家指导下。  我想通过Epwmx A 的soc 触发ADC 中断, 在中断里边进行adc 采样 和PI 调节占空比,然后通过示波器观察输出的pwm 波的占空比是否有变化, 不知道这样可行吗? ( 另外,我把采样的电压作为反馈电压和参考值对比,得到误差Err . )   因为没有实际的系统,所以我的pi 参数随便给了个值, 做这个实验目的只是为了把epwm  , ADC ,PI   调节纳合在一块,验证下自己的想法和掌握的情况。。 不知道自己编写程序的时序不对还是缺少什么东西, 在示波器上看不见PWM 波。    我把自己写的程序贴上来,大家帮我看看,谢谢了。。。
  1. /*
  2. 使用epwm6A 产生pwm波,并且触发ADC 采样 ADCA0(就是板子上外扩的那个点位器的电压)
  3. */
  4. #include "DSP2833x_Device.h"     // DSP2833x Headerfile Include File
  5. #include "DSP2833x_Examples.h"   // DSP2833x Examples Include File


  6. typedef struct
  7. {
  8.   float Ref ;
  9.   float Fdb ;

  10.   float Kp;
  11.   float Ki ;
  12.   float Kd;

  13.   float T ;  //  此值待修改

  14.   float a0 ;
  15.   float a1 ;
  16.   float a2 ;

  17.   float Err ;
  18.   float Err_1 ;
  19.   float ERR_2 ;

  20.   float Out ;
  21.   float Out_1 ;
  22.   float OutMax ;
  23.   float OutMin ;
  24. }PID;


  25. PID epwm_info ;

  26. float adc[10] ;
  27. float adcresultsa0[] ;
  28. float SampleVoltage ;

  29. void ADC_Convert(void) ;
  30. void InitEPwm6Example(void);
  31. void pid_cal(PID*);
  32. void sequence(float a[],int i) ;
  33. interrupt void adc_isr(void);


  34. #define EPWM6_TIMER_TBPRD  750 // Period register value
  35. #define ADC_MODCLK 0x3 // HSPCLK = SYSCLKOUT/2*ADC_MODCLK2 = 150/(2*3)   = 25.0 MHz
  36. #define ADC_CKPS   0x1   // ADC module clock = HSPCLK/2*ADC_CKPS   = 25MHz/(2*1)   = 12.5MHz
  37. #define ADC_SHCLK  0xf   // S/H width in ADC module periods        = 16 ADC clocks
  38. void main(void)
  39. {
  40. // Step 1. Initialize System Control:
  41.          InitSysCtrl();

  42.          EALLOW;
  43.          SysCtrlRegs.HISPCP.all = ADC_MODCLK;        // HSPCLK = SYSCLKOUT/(2*ADC_MODCLK)=25MHZ
  44.          EDIS;

  45. // Step 2. Initialize GPIO:
  46.          InitEPwm6Gpio();

  47. //Step 3. Clear all interrupts and initialize PIE vector table:
  48.          // Disable CPU interrupts
  49.          DINT;
  50.          // Initialize the PIE control registers to their default state.
  51.          // The default state is all PIE interrupts disabled and flags
  52.          // are cleared.
  53.          InitPieCtrl();
  54.          // Disable CPU interrupts and clear all CPU interrupt flags:
  55.          IER = 0x0000;
  56.          IFR = 0x0000;

  57.          InitPieVectTable();
  58.          // Interrupts that are used in this example are re-mapped to
  59.          // ISR functions found within this file.
  60.          EALLOW;
  61.          PieVectTable.ADCINT=&adc_isr;
  62.          EDIS;
  63. // Step 4. Initialize all the Device Peripherals:
  64.           InitAdc();
  65.           // Specific ADC setup for this example:
  66.          AdcRegs.ADCTRL3.bit.ADCCLKPS = ADC_CKPS;//ADC内核时钟分频:HSPCLK/2=12.5MHZ
  67.          AdcRegs.ADCTRL1.bit.ACQ_PS = ADC_SHCLK;//设置采样窗口时间:(15+1)*ADCCLK
  68.          AdcRegs.ADCTRL1.bit.SEQ_CASC = 1;        // 1选择级联模式
  69.          AdcRegs.ADCTRL1.bit.CONT_RUN = 1;       // 设置为连续运行
  70.          AdcRegs.ADCMAXCONV.all = 0x0000;       // Setup 1 conv's on SEQ1
  71.          AdcRegs.ADCCHSELSEQ1.bit.CONV00 = 0x0; // Setup ADCINA0 as 1st SEQ1 conv.
  72.          AdcRegs.ADCTRL2.bit.EPWM_SOCA_SEQ1 = 1;// Enable SOCA from ePWM to start SEQ1
  73.          AdcRegs.ADCTRL2.bit.INT_ENA_SEQ1 = 1;  // Enable SEQ1 interrupt (every EOS)

  74.      //  configure  EPwm module
  75.          EALLOW;
  76.          SysCtrlRegs.PCLKCR0.bit.TBCLKSYNC = 0;
  77.          EDIS;

  78.          InitEPwm6Example();

  79.          EALLOW;
  80.          SysCtrlRegs.PCLKCR0.bit.TBCLKSYNC = 1;
  81.          EDIS;

  82. // Step 5. User specific code, enable interrupts:
  83.          IER |= M_INT1;
  84.          PieCtrlRegs.PIEIER1.bit.INTx6 = 1;

  85.          EINT;   // Enable Global interrupt INTM
  86.          ERTM;   // Enable Global realtime interrupt DBGM



  87. }

  88. interrupt void adc_isr()
  89. {
  90.          ADC_Convert();
  91.          epwm_info.Ref=1.5 ;
  92.          epwm_info.Fdb=SampleVoltage ;
  93.          epwm_info.Err=epwm_info.Ref-epwm_info.Fdb ;
  94.          epwm_info.Kp=0.02 ;
  95.          epwm_info.Ki=0.001 ;
  96.          epwm_info.Kd=0 ;

  97.          epwm_info.OutMax=0.5 ;
  98.          epwm_info.OutMin=0 ;

  99.          pid_cal(&epwm_info) ;

  100.          AdcRegs.ADCST.bit.INT_SEQ1_CLR = 1;       // Clear INT SEQ1 bit
  101.          PieCtrlRegs.PIEACK.all = PIEACK_GROUP1;
  102. }

  103. void InitEPwm6Example(void)
  104. {
  105.         // Setup TBCLK
  106.         EPwm6Regs.TBCTL.bit.CTRMODE=TB_COUNT_UP; // Count up;
  107.         EPwm6Regs.TBPRD= EPWM6_TIMER_TBPRD ;
  108.         EPwm6Regs.TBCTL.bit.PHSEN=TB_DISABLE ;
  109.         EPwm6Regs.TBPHS.half.TBPHS=0x0000 ;
  110.         EPwm6Regs.TBCTR=0x0000;
  111.         EPwm6Regs.TBCTL.bit.HSPCLKDIV=TB_DIV2 ;
  112.         EPwm6Regs.TBCTL.bit.CLKDIV=TB_DIV1 ;

  113.         // Setup shadow register load on ZERO
  114.         EPwm6Regs.CMPCTL.bit.SHDWAMODE= CC_SHADOW;
  115.         EPwm6Regs.CMPCTL.bit.LOADAMODE= CC_CTR_ZERO;

  116.         // Set Compare values
  117.         EPwm6Regs.CMPA.half.CMPA= (epwm_info.Out)*EPWM6_TIMER_TBPRD;   ////此值待修改

  118.         // Set actions
  119.         EPwm6Regs.AQCTLA.bit.ZRO=AQ_SET;
  120.         EPwm6Regs.AQCTLA.bit.CAU=AQ_CLEAR;

  121.         EPwm6Regs.ETSEL.bit.SOCAEN = 1;        // Enable SOC on A group
  122.         EPwm6Regs.ETSEL.bit.SOCASEL = 4;       // Select SOC from from CPMA on upcount
  123.         EPwm6Regs.ETPS.bit.SOCAPRD = 1;        // Generate pulse on 1st event

  124. }


  125. void ADC_Convert(void)
  126. {
  127.         AdcRegs.ADCTRL2.bit.RST_SEQ1=1 ;

  128.             SampleVoltage=((AdcRegs.ADCRESULT0)>>4)*3/4096.0 ;



  129.         }
  130. void pid_cal(PID *epwm_info)
  131. {
  132.         float a0 ,a1 ,a2 ;
  133.         a0=epwm_info->Kp+epwm_info->Ki*epwm_info->T+epwm_info->Kd/epwm_info->T ;
  134.         a1=epwm_info->Kp+2*epwm_info->Kd/epwm_info->T ;
  135.         a2=epwm_info->Kd/epwm_info->T ;

  136.         epwm_info->Out=epwm_info->Out_1+a0*epwm_info->Err-a1*epwm_info->Err_1+a2*epwm_info->ERR_2 ;

  137.         if(epwm_info->Out>epwm_info->OutMax)
  138.                 epwm_info->Out=epwm_info->OutMax ;
  139.         if(epwm_info->Out<epwm_info->OutMin)
  140.                 epwm_info->Out=epwm_info->OutMin ;

  141.         epwm_info->Out_1=epwm_info->Out ;
  142.         epwm_info->ERR_2=epwm_info->Err_1 ;
  143.         epwm_info->Err_1=epwm_info->Err ;

  144.         }



 楼主| 潇湘夜雨涩 发表于 2016-11-16 16:31 | 显示全部楼层
求大家帮忙看看。。。
 楼主| 潇湘夜雨涩 发表于 2016-11-16 21:00 | 显示全部楼层
各位大神,帮我看看 。。。。
泽曦儿 发表于 2016-11-16 21:37 | 显示全部楼层
楼主,main函数中怎么么有循环体,这不是初始完也就执行完程序了么
泽曦儿 发表于 2016-11-16 21:43 | 显示全部楼层
泽曦儿 发表于 2016-11-16 21:37
楼主,main函数中怎么么有循环体,这不是初始完也就执行完程序了么

还有,pid空中pwm的比较值,不是在初始化函数中做的,而是应该根据pid的运算结果,不断来给比较寄存器赋值,这样才能得到一个占空比变化的pwm。其他的设置没有看手册,没有发现错误。
zhangmangui 发表于 2016-11-16 21:58 | 显示全部楼层
你是通过PWM控制等效电压吗?
zhangmangui 发表于 2016-11-16 21:58 | 显示全部楼层
速度环等直接是采集速度啊    ADC用来干啥
泽曦儿 发表于 2016-11-17 00:15 来自手机 | 显示全部楼层
adc可以做位移环和电流检测等
海中水 发表于 2016-11-17 09:17 | 显示全部楼层
问题解决了没有呢?
您需要登录后才可以回帖 登录 | 注册

本版积分规则

1

主题

5

帖子

0

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