This thread has been locked.

If you have a related question, please click the "Ask a related question" button in the top right corner. The newly created question will be automatically linked to this question.

CCS/LAUNCHXL-F28069M: ADC: The signal measurements oscillate if the frequency changes

Part Number: LAUNCHXL-F28069M

Tool/software: Code Composer Studio

Hi Everyone.

This is Mike, I want to thank you all for your support.

I take again the project I left out. So the problem I'm having right now is about the mesurement of the signal's voltage.

I coded a True RMS voltmeter so It has an error of 40 mV, I think this is not pretty bad

My waveform of input is a pretty sine of 60 Hz, and my measure, as an example, is 3V in the oscilloscope. In the same way, the true rms measures the same voltage

Then If change the frequency, just to say, 58.9 Hz, my measure starts to oscillate  2.78 V to 3.02V

When I graph the signal, It is like if the waveform is clipped. 

I think it is something aabout the Period and Compare of the PWM but I don't know how to fix it.

Regards

Mike

  • Is Anybody out there?
  • Mike,
    When you say that the graph is clipped, do you mean the output of the ADC appears saturated? If so is it physically saturated at 0 and 4095 or an induced saturation at a non min/max code that clips the peak of the input voltage waveform?

    Best,
    Matt
  • Hi Matt
    I think I could not explain as well as I wanted, I'm not a native english XD
    But not, it is not saturated, If it was that, It could be solved just limiting the amplitude's input, but it is not the problem.
    When I say "it clipped" I just wanna say something about a waveform that is distorted, as if the frequency change alters the compare and the signal begins to distort.
  • Is anybody out there?
  • Mike,
    Would it be possible to provide some more background on how you sampling the incoming AC voltage? Since you see some distortion when moving frequencies I think you may be controlling the ADC sampling based on some estimation of the AC voltage period? If there is enough BW we could increase the sample rate of the ADC to not "miss" the peaks/valleys of the wave.

    If there is some sample code you're comfortable sharing please feel free to attach it.

    Best,
    Matthew
  • Hi Matthew

    Here it is the code.

    #include "F2806x_Device.h"
    #include "F2806x_Examples.h"
    #include "DSP28x_Project.h"
    #include <stdlib.h>
    #include <stdio.h>
    #include <string.h>
    #include <math.h>
    #include "IQmathLib.h"
    
    extern Uint16 RamfuncsLoadStart;
    extern Uint16 RamfuncsLoadEnd;
    extern Uint16 RamfuncsRunStart;
    __interrupt void adc_isr(void);
    
    
    /* Declaración de variables Globales*/
    
    Uint16 ADC_Buffer[154];
    Uint32 ADC_Buffer2[154];
    _iq ADC_Buffer3;
    _iq ADC_Buf_Sum;
    _iq ADC_Buffer4;
    Uint32 ADC_BufferVDC;
    _iq ADC_Buff_VDC2;
    _iq RMS;
    _iq DC;
    Uint16 conv;
    Uint16 cont;
    Uint16 i;
    
    
    void main(void)
    {
    
    
        memcpy(&RamfuncsRunStart, &RamfuncsLoadStart, (size_t)&RamfuncsLoadEnd); // Manipulación de bloque de memoria: El contenido de RamfuncsLoadStart
                                                                                 // se copia en RamfuncsRunStart, la cantidad de bytes copiados lo define
                                                                                 // el tamaño de RamfuncsLoadEnd. Estas acciones las realiza el enlazador (F28069M.cmd)
        InitFlash();   // Inicializa las funciones para copiar el programa a la memoria Flash
        InitSysCtrl(); // Inicializa el Sistema de Control: PLL, WD, Peripheral Clocks.
        DINT;          // Deshabilita las interrupciones del CPU
        InitPieCtrl(); // Inicializa los registros de control del Puerto de Interrupciones Expandido a sus estado predeterminado
        IER = 0X0000;  // Deshabiita las interrupciones del CPU
        IFR = 0X0000;  // Limpia las banderas de interrupción del CPU
        InitPieVectTable(); //Inicializa la tabla de arreglos de interrupción, las interrupciones utilizadas son remapeadas en esta función
        EALLOW;             // Habilita la emulación de registros
        PieVectTable.ADCINT1 = &adc_isr; // El contenido del subprograma de interrupción "adc_isr" es copiado por el subprograma de interrupción "ADCINT1" contenido en la tabla.
        EDIS;                            // Deshabilita la emulación de registros
        InitAdc();                       // Inicializa el módulo del ADC
        AdcOffsetSelfCal();              // Ajuste de offset del ADC debido a su construcción
        PieCtrlRegs.PIEIER1.bit.INTx1 = 1;// Habilita la Interrupción 1.1 en el Puerto de expansión de interrupciones
        IER |= M_INT1;                   // Habilita la interrupción 1 del CPU
        EINT;                            // Habilita las interrupciones globales
        ERTM;                            // Habilita las interrupciones en tiempo real
        conv=0;                          // Variable auxiliar que define, temporalmente, la longitud del Buffer
        EALLOW;                          // Habilita la emulación de registros
        AdcRegs.ADCCTL2.bit.ADCNONOVERLAP = 1; // Habilita el modo de sobrelapamiento de muestras
        AdcRegs.ADCCTL1.bit.INTPULSEPOS = 1;   // Control de pulso generador de interrupción -> 0 = Inicio de conversión, 1 = Un ciclo por resultado
        AdcRegs.INTSEL1N2.bit.INT1E = 1;       // Habilita ADCINT1
        AdcRegs.INTSEL1N2.bit.INT1CONT = 0;    // Modo de interrupciones continuas deshabilitado -> 0= Pulso generado si la bandera es limpiada por SW, 1=Pulso generado cada EOC
        AdcRegs.INTSEL1N2.bit.INT1SEL = 1;     // Final de conversion 1 (EOC 1) es el disparo (trigger) de ADCINT1
        AdcRegs.ADCSOC1CTL.bit.CHSEL = 1;      // Inicio de conversión 0 (SOC) para el canal ADCIN1 (pin 29)
        AdcRegs.ADCSOC1CTL.bit.TRIGSEL = 5;    // ePWM 1 es el disparo para SOC1
        AdcRegs.ADCSOC1CTL.bit.ACQPS = 6;      // Ventana de Sample/Holding a 7 ciclos del reloj del ADC
        EPwm1Regs.ETSEL.bit.SOCAEN = 1;        // Selección de registros para Eventos de Disparo -> Disparos en SOC A habilitados
        EPwm1Regs.ETSEL.bit.SOCASEL = 4;       // Selección de SOC en la comparación del contador ascente de A
        EPwm1Regs.ETPS.bit.SOCAPRD = 1;        // Generación de SOC al primer evento
        EPwm1Regs.CMPA.half.CMPA = 0xE647;     // Valor decimal por default: 29220. Para un Duty Cycle en modo de conteo ascendente del 99% -> CMPA = (100% - Duty Cycle) * (TBPRD + 1) - 1
        EPwm1Regs.TBPRD = 0xFFDE;              // Valor decimal por default: 29220. Periodo de conteo en modo ascendente
        EPwm1Regs.TBCTL.bit.CTRMODE = 0;       // Mode conteo ascendente
        GpioCtrlRegs.GPBMUX1.bit.GPIO39 = 0;   // MUX -> GPIO
        GpioDataRegs.GPBCLEAR.bit.GPIO39 = 1;  // Salida en estado bajo antes de configurar el pin como Salida, Modo Drenador (Sink). Consultar spruh18g.pdf (pág 120)
        GpioCtrlRegs.GPBDIR.bit.GPIO39 = 1;    // GPIO39 como salida
        EDIS;                                  // Deshabilita la emulación de registros
        for(i=0;i<154;i++)                     // Ciclo for para inicializar ADC_Buffer con un arreglo de 0's
        {
            ADC_Buffer[i]=0;                   // Ciclo for para inicializar ADC_Buffer con un arreglo de 0's
        }
        for(i=0;i<154;i++)                     // Ciclo for para inicializar ADC_Buffer2 con un arreglo de 0's
          {
              ADC_Buffer2[i]=0;
          }
        for(;;)
       { // Inicio de For
            EALLOW;                                // Habilita la emulación de registros
            GpioDataRegs.GPBSET.bit.GPIO39 = 1;    // Pin Modo Fuente (Source) ó ALTO
            DELAY_US(100000);                      // Intervalo de tiempo de 0.1 s
            GpioDataRegs.GPBCLEAR.bit.GPIO39 = 1;  // Pin Modo Sumidero (Sink) ó LOW
            DELAY_US(100000);                      // Intervalo de tiempo de 0.1 s
            EDIS;                                  // Deshabilita la emulación de registros
       } // Fin de for
    
    } // Fin de programa principal
    
    __interrupt void
    adc_isr(void) // Inicio del programa de interrupción
    {
    
        ADC_Buffer[conv] = AdcResult.ADCRESULT1;     // Cada espacio o localidad del arreglo es llenado con el resultado del registro AdcResult 1
        ADC_Buffer2[conv] = pow(ADC_Buffer[conv],2); /* VRMS Se eleva al cuadrado cada dato ingresado en el buffer */
        ADC_BufferVDC += ADC_Buffer[conv];           /* VDC  Se hace una suma de todos los datos ingresados en el buffer*/
        ADC_Buf_Sum += ADC_Buffer2[conv];            /* VRMS Se hace una suma de todos los datos elevados al cuadrado que ingresaron al buffer*/
    
    
        if (conv == 153)                             // Si la variable conv si es igual a la longitud del buffer menos uno realiza los algoritmos de VDC y VRMS y despues se reinicia
        {
            ADC_Buffer3 = ADC_Buf_Sum/conv+1;     /* VRMS La suma de los datos elevados al cuadro del ADC es divida entre la longitud del buffer */
            ADC_Buff_VDC2 = ADC_BufferVDC/conv+1; /* VDC  La suma de los datos del ADC es divida entre la longitud del buffer */
            DC=(3.3*ADC_Buff_VDC2)/4096;          /* VDC  Se multiplica el promedio de los datos por un factor de conversión y se tiene el valor DC de la señal muestreada */
            ADC_Buffer4 = sqrt(ADC_Buffer3);      /* VRMS Se obtiene la raíz cuadrada de la división (Sumatoria/Longitud_Buffer) */
            RMS=ADC_Buffer4*(3.3/4096);           /* VRMS Se multiplica el resultado de la raíz cuadrada por el factor de conversión y se obtiene el valor RMS de la señal muestreada */
            conv = 0;                             /* El contador auxiliar del buffer se reinicia */
            ADC_BufferVDC=0;                      /* El contenido de la suma de todos los datos del buffer se iguala a cero, con el fin de evitar sobrelapamiento de datos */
            ADC_Buf_Sum =0;                       /* El contenido de la suma del cuadrado de todos los datos del buffer se iguala a cero, con el fin de evitar sobrelapamiento de datos */
        }
    
        else                                      // Si la variable conv no es igual a la longitud del buffer menos uno entonces incrementa en uno
        {
            conv++;
        }
        AdcRegs.ADCINTFLGCLR.bit.ADCINT1 = 1;     // Limpia la bandera de interrupción del ADC
        PieCtrlRegs.PIEACK.all = PIEACK_GROUP1;   // El bloque de expandido de interrupción de perifericos reconoce una interrupción en el grupo 1
        return;
    
    } 

  • Anybody ??

  • Mike,
    I missed that you had responded, I'll take a look and reply this afternoon.

    Matt
  • Mike,
    Thanks for the code, assuming you are running the F28069 at max speed, I believe you have set up the PWM to trigger an ADC sample every 728us against a 60Hz input, so we have around 23x oversampling. You then are taking 153 of those samples to derive your RMS voltage.

    With regard to the error, there could be some things to try to see if we can the result to be more stable:

    1)Increase the sampling speed of the ADC such that more points along the sine wave are taken
    2)Increase the number of samples used to generate the RMS to make sure we are detecting the full magnitude no matter the frequency
    3)Floating Point or Fixed Point math; by this I want to verify if you are using the floating point unit on this device(with the -f compile option) vs fixed point or IQmath. It could be that the physical math is causing errors at some input frequencies.

    Matt
  • Hi Matt, sorry for answering too late. I've been working around other projects.

    1) Increase the sampling speed of the ADC such that more points along the sine wave are taken

    How to achieve this? I think is taking more than 153 points. Or are you talking about the sample window? I have the smallest window in the register  AdcRegs.ADCSOC1CTL.bit.ACQPS = 6;

    2)Increase the number of samples used to generate the RMS to make sure we are detecting the full magnitude no matter the frequency

    I put 153 samples taking it of the Nyquist's Theorem. So I think if I put more samples, the detection o sampling will be slow. I will put 1000 sampling points. What do you think about it?

    3)Floating Point or Fixed Point math; by this I want to verify if you are using the floating point unit on this device(with the -f compile option) vs fixed point or IQmath. It could be that the physical math is causing errors at some input frequencies.

    I'm using the floating point Unit.

     

    Best Regards 

    Mike.

     

  • Mike,
    I realize that my first point is not clear, while you are using the smallest sample window for the ADC, you could increase the sample rate by lowering the compare/match value for CMPA which triggers the ADC.

    We can also try to re-cast the ADC results from Uint32 to floats, prior their usage in the code.

    Matt