Reloj SPI en PIC inestable

8

Estoy tratando de configurar el módulo MSSP de un PIC18F25K22 en modo maestro SPI. Estoy mirando la sincronización y el reloj no permanece estable durante toda la transmisión. Una imagen lo muestra mejor que las palabras. Diagrama de tiempo de SPI

Después de enviar un bit, el reloj se acorta, y no en la misma cantidad cada vez. No he trabajado con SPI antes, pero los diagramas que encontré en Wikipedia y otros recursos nunca muestran esto. También conecté un Arduino y no vi este comportamiento. Mi código es:

    #pragma config FOSC = INTIO67   // Oscillator Selection bits (Internal oscillator block)
#pragma config PLLCFG = OFF     // 4X PLL Enable (Oscillator used directly)
#pragma config BOREN = OFF      // Brown-out Reset Enable bits (Brown-out Reset disabled in hardware and software)
#pragma config WDTEN = OFF      // Watchdog Timer Enable bits (Watch dog timer is always disabled. SWDTEN has no effect.)
#pragma config MCLRE = EXTMCLR  // MCLR Pin Enable bit (MCLR pin enabled, RE3 input pin disabled)
#pragma config LVP = OFF        // Single-Supply ICSP Enable bit (Single-Supply ICSP disabled)
#pragma config XINST = OFF      // Extended Instruction Set Enable bit (Instruction set extension and Indexed Addressing mode disabled (Legacy mode))

void main(void)
{
    OSCCON = 0b11100110;
    spi_setup();
    __delay_ms(10);
    byte temp;
    while (TRUE)
    {
        temp = spi_transfer(0x00);
        temp = spi_transfer(0x01);
        temp = spi_transfer(0x02);
        temp = spi_transfer(0x03);
        temp = spi_transfer(0x04);
        temp = spi_transfer(0x05);
        __delay_us(1);
    }
}

void spi_setup(void)
{
    SSP1STAT = 0b00000000;
    SSP1STATbits.CKE = HIGH; // data transmitted on rising edge
    SSP1CON1 = 0b00000000; // enable Master SPI mode
    SSP1CON1bits.CKP1 = LOW; //clock idle state is low
    //i2c bits, all don't matters for SPI, cleared just in case
    SSP1CON3 = 0;
    // baud rate generation
    SSP1ADD = 0; //FCLOCK = 8Mhz /2 = 2Mhz
    // configure pins for output/input as needed 
    SDI1 = INPUT;
    SDO1 = OUTPUT;
    SCK1 = OUTPUT;
    SS1 = OUTPUT;
    SSP1CON1bits.SSPEN1 = HIGH; // enable pins for serial mode
}

unsigned char spi_transfer(unsigned char data)
{
    SS1_LAT = LOW; // select slave
    PIR1bits.SSPIF = LOW;
    SSP1BUF = data;
    //while (!SSP1STATbits.BF); //wait for receive to complete
    while( !PIR1bits.SSPIF );
    SS1_LAT = HIGH; // deselect slave
    PIR1bits.SSPIF = LOW;   // clear interrupt
    return SSP1BUF; //return data from the slave
}

(también https://gist.github.com/stumpylog/5095250 )

¿Alguien ha encontrado esto o tiene sugerencias sobre la causa?

Lo que hice

Al final, no pude hacer funcionar el módulo MSSP1. Sin embargo, cambiarlo al módulo MSSP2, exactamente el mismo código, no exhibió este comportamiento. No puedo explicarlo, pero esto resolvió el problema.

Trenton Holmes
fuente
¿Puedes mostrar tu código para usar el SPI?
Gustavo Litovsky
1
En general, SPI (y también I2C) funcionarían con un reloj no uniforme. SPI es sincrónico. Al mismo tiempo, parece extraño que un MSSP de hardware genere un reloj no uniforme. Cuando la línea de datos (verde) es baja, sus relojes son uniformes. Cuando la línea de datos es alta, sus relojes son más cortos. Por si acaso, verifique las erratas de su PIC.
Nick Alexeev
@GustavoLitovsky He agregado el código directamente a la pregunta ahora.
Trenton Holmes
@NickAlexeev Gracias, lo eché un vistazo. Nada se menciona sobre el módulo MSSP. Tendré que verificar si mi esclavo puede manejar el tiempo tal como está.
Trenton Holmes
Probablemente no esté relacionado con su problema, pero no veo código para borrar los bits ANSEL para sus puertos. Microchip ha tomado la molesta elección de que los pines sean entradas analógicas por defecto en lugar de digitales.
apalopohapa

Respuestas:

3

Esto es una suposición, pero probablemente esté restableciendo algo que no debería ser cada byte. Cosas como el generador de velocidad de bits y la configuración periférica general solo deben configurarse una vez.

Adicional:

Ahora dice que no pudo hacer que MSSP1 funcionara pero que MSSP2 funcionó. Eso sugiere que tiene un error en otra parte del código que está haciendo escrituras no deseadas. Resulta que está afectando algún estado de MSSP1, por lo que actúa de manera extraña y por qué funciona MSSP2.

No solo dejes ir esto. Puede parecer que el cambio a MSSP2 haya solucionado el problema, pero en el mejor de los casos lo ha solucionado, probablemente temporalmente. La próxima vez que se vincule con cosas en diferentes lugares, se pueden garabatear diferentes memorias. Si no encuentra y realmente soluciona esto, este firmware será siempre inestable. El peor de los casos es cuando no hay síntomas obvios que te hagan el favor de dejar en claro que hay un problema. El problema aparecerá un año después cuando se encuentren los datos correctos, solo los sitios de un cliente, después de que 1000 estén en el campo. ARREGLAR ESTO DE LA MANERA CORRECTA AHORA.

Olin Lathrop
fuente
2

Parece que puede tener problemas de integridad de la señal: en el disparo LA, parece que la línea del reloj falla cuando cae la línea de datos. Intente asegurarse de que los dos estén bien aislados y que la traza o el cableado no sean demasiado largos. También intente reducir la velocidad del reloj o agregar un pequeño filtro RC en las líneas (si las líneas son largas, solo una resistencia en serie de, por ejemplo, 220Ω puede ayudar)

Si tiene un osciloscopio, verifique las líneas con esto para asegurarse de que la integridad de la señal sea buena. Si no es así, pruebe las sugerencias anteriores y ajuste las cosas hasta que tenga señales de buena calidad.

Oli Glaser
fuente