SPI Master/Slave communication between two NUCLEO module. You can send objects ( float, text, ..etc) between them. These projects should be used together: SPI_NUCLEO_MASTER SPI_NUCLEO_SLAVE

Nucleo module as slave

Using Nucleo modules as slave was not easy to start with. Many thing could go wrong. I provide two projects that works and can send a float number between them.

Still I get sometimes packet lost, but it is working in general. You might modify it so it will be more secure to communicate between the modules.

I used SPI 16 bits mode to allow sending some kind of checksum or control code at the MSByte .. The LSByte should be always the data. Using this technique, you can make a quite reliable communication between the two module.

Master code is as follow :

include the mbed library with this snippet

SPI device(SPI_MOSI, SPI_MISO, SPI_SCK);

typedef unsigned char byte;

typedef union SPI_Packet_t{
    float sensor;
    byte SPI_Packet[sizeof(float)];
};
unsigned short tem;
int i;
unsigned int paSize;
SPI_Packet_t spipacket;

DigitalOut cs(SPI_CS);


#define SPI_DUMMY     0X0000
#define SPI_CMD       0XADAD
#define SPI_FAULT     0xDD

Serial pc(SERIAL_TX, SERIAL_RX);

short sendToSlave(unsigned short value )
{
    short result=0;
    cs=1;
    wait_us(500);
    cs=0;
    device.write(value);
    wait_us(500);
    cs=1;
    wait_ms(40);
    cs=0;
    wait_us(500);
    result=device.write(0x0000);
    wait_us(500);
    cs=1;
    wait_ms(40);
    return result;
}

int main()
{
    device.format(16,0);
    device.frequency(1000000);
    tem=0;
    while(1) {
        
        tem=sendToSlave(SPI_CMD);       //RETURNED BYT IS DUMMY
        paSize=(tem & 0xFF00)>>8;
        if(paSize>0){
            spipacket.SPI_Packet[0]=(byte)(tem & 0x00FF);
            pc.printf("Packet=%x\n",spipacket.SPI_Packet[0]);
            pc.printf("packet size=%i\n",paSize);
            for (i=1; i<paSize; i++) {
                cs=1;
                wait_us(500);
                cs=0;
                tem=(unsigned short)device.write(SPI_DUMMY);
                wait_us(500);
                cs=1;
                wait_ms(40);
                if( ((tem & 0xFF00)>>8)!=SPI_FAULT){
                    spipacket.SPI_Packet[i]=(byte)(tem & 0x00FF);
                    pc.printf("Packet=%x\n",spipacket.SPI_Packet[i]);
                }
                    else {
                        pc.printf("wrong packet\n");
                                  break;    //go out from for loop.
                          }
            }
        pc.printf("Temp= %04.2f\n",spipacket.sensor);
        pc.printf("------------------------------\n");
     }
    spipacket.sensor=0.0;// Reset the variable.
    wait(1);
    }
}

And the Slave side is as follow :

include the mbed library with this snippet


Serial pc(USBTX,USBRX);



#define SPI_DUMMY     0X00
#define SPI_CMD       0XADAD     //Anything as a CMD code.
#define SPI_FAULT     0xDD       //Anything as a part of 16 bit as control byte

#ifdef _DEBUG
extern Serial pc;
#define DEBUG_MESSAGE(...) pc.printf(__VA_ARGS__)
#else
#define DEBUG_MESSAGE(...)
#endif

typedef unsigned char byte;

SPISlave slave(SPI_MOSI, SPI_MISO, SPI_SCK ,PA_4);
int res=0;
float d,c,e;

/*This will be the float number we wish to transfer. 
You can make a struct of multiple variable and send it in the same way*/

typedef union SPI_Packet_t{
float sensor;
byte SPI_Packet[sizeof(float)];
};

SPI_Packet_t packet;
int main()
{
    pc.printf("Testing is started from slave side\n");
    slave.format(16,0);
    slave.frequency(1000000);
    unsigned short vs=0;
    slave.reply(SPI_DUMMY);   // Prime reply
    unsigned short tempVal=0;
    packet.sensor=1000.02;    //Value of the float number we wish to communicate
    while(1) {
        if(slave.receive()) {
            tempVal=(unsigned short) slave.read();        //Real command
            printf("----------------\n%x\n",tempVal);
            /*
                The switch is not necessary if you have only one command between
                 master-slave. This is useful if you wish to send different 
                 object depending on the command.
            */
            switch(tempVal){
            case SPI_CMD: {    /*Use this if you have different commands 
                                for sending different objects to the master.*/
                            tempVal=(sizeof(float))<<8| packet.SPI_Packet[0];
                            pc.printf("Packet=%x\n",tempVal);
                            slave.reply(tempVal);
                            for(unsigned int i=1; i<sizeof(float); i++){
                                while(!(slave.receive()))
                                    wait_ms(1);
                            vs=slave.read();        //Real command
                            if (vs==SPI_DUMMY){
                                //Continue sending bytes
                                {
                                  tempVal=0x00FF & packet.SPI_Packet[i];
                                  pc.printf("Packet=%x\n",tempVal);
                                  slave.reply(tempVal);
                                }
                            }
                            else {
                                slave.reply(SPI_FAULT); //ERROR CODE
                                slave.reply(SPI_DUMMY);     //dummy ..skiped value
                            }
                        }
                    }

                    break;
            }

         }

    }
}


Note:

Slave communication is shifted by one byte. To respond for ex for a START byte, you need to put the ACK byte prepared on the bus even before having the START byte received by the slave. As the Master write the SPI bus the START Command, the prepared byte on the slave SPI bus will be transferred to the Master immediately. So, you need to prepare the Slave device to shift the bytes by one byte as in the following example.


All wikipages