SPI on ZED-F9R

Hi all, I spent some time digging through the forum and tried solutions that seemed to work for others. No luck. My issue is I need to implement SPI on my nav board for an autonomous RC car application. I have a custom PCB with ESP32 devkitc v4 installed on it. I am trying to use the HSPI pins. I confirmed they work as expected by hooking up an sd card reader to those pins. I blobbed the SPI jumper on the nav board and confirmed continuity with ground.

The example code for SPI does not work for me even with a redboard (though I noticed that redboard is kicking out 3.9V logic!), so it may be I’m not changing the right settings in u-center. But I went to CFG-PRT and set to SPI, 0+1+5 for protocol in/out. Set I2C and Uart1 to none for protocol in/out. Enabled SPI in CFG-MSG for NAV-PVT. And of course saved current config in CFG-CFG. I tried to do some low-level SPI comms with the nav board to no avail. The code I adapted from the example is below, not sure how to make it all go into the code box.

I should mention I’ve used this board hundreds of times with I2C, though not since I started trying to make SPI work.

#include <SPI.h> //Needed for SPI to GNSS
#include <SparkFun_u-blox_GNSS_v3.h> //http://librarymanager/All#SparkFun_u-blox_GNSS_v3

SFE_UBLOX_GNSS_SPI myGNSS; // SFE_UBLOX_GNSS_SPI uses SPI

#define myCS 15 // Define the GPIO pin number for the SPI Chip Select
#define GNSS_MOSI_PIN 13
#define GNSS_MISO_PIN 12
#define GNSS_SCK_PIN 14

#define mySPI SPI // Define which SPI port to use
#define myClockSpeed 4000000 // Define what SPI clock speed to use

// Variables for timing and statistics
unsigned long GPSlastTime = 0;
unsigned long GPS_Interval = 100; // 100ms = 10Hz rate
unsigned long Readtime = 0;
unsigned long Max = 0;
unsigned long entrymillis = 0;
long counter = 0;

void setup()
{
Serial.begin(115200);

// Initialize SPI bus
SPI.begin(GNSS_SCK_PIN, GNSS_MISO_PIN, GNSS_MOSI_PIN, myCS);
delay(50);

// Do a fake transaction to initialize the SPI pins
SPI.beginTransaction(SPISettings(myClockSpeed, MSBFIRST, SPI_MODE0));
SPI.transfer(0xFF);
SPI.endTransaction();
delay(50);

// Try to connect with timeout
Serial.println(“Connecting to ZED-F9R over SPI…”);
while (myGNSS.begin(mySPI, myCS, myClockSpeed, 20) == false)
{
Serial.println(“Waiting for ZED-F9R to connect over SPI”);
delay(50);
}

Serial.println(“GNSS connected!”);

// Configure SPI port to only output UBX (disable NMEA)
myGNSS.setSPIOutput(COM_TYPE_UBX);

// THIS IS THE CRITICAL CHANGE - Set Auto PVT mode
myGNSS.setAutoPVT(true);

// Save the communications port settings to flash and BBR
myGNSS.saveConfigSelective(VAL_CFG_SUBSEC_IOPORT);

Serial.println(“Setup complete!”);
}

void loop()
{
// Only read at the specified interval
if ((millis() - GPSlastTime) >= GPS_Interval)
{
GPSlastTime = millis(); // Update the timer

// Try to get the latest PVT data
if (myGNSS.getPVT())
{
  entrymillis = millis();
  
  // Get position data
  int32_t latitude = myGNSS.getLatitude();
  int32_t longitude = myGNSS.getLongitude();
  int32_t altitude = myGNSS.getAltitudeMSL();
  
  // Calculate read time for performance monitoring
  Readtime = millis() - entrymillis;
  
  // Track maximum read time
  if (Readtime > Max)
    Max = Readtime;
  
  counter++;
  
  // Display position and timing information
  Serial.print("Lat: ");
  Serial.print(latitude);
  Serial.print(" Long: ");
  Serial.print(longitude);
  Serial.print(" Alt: ");
  Serial.print(altitude);
  Serial.print(" (mm) | ReadTime: ");
  Serial.print(Readtime);
  Serial.print("ms Max: ");
  Serial.print(Max);
  Serial.print("ms Count: ");
  Serial.println(counter);
}
else
{
  Serial.println("No new PVT data");
}

}

Hi Brent (@brenton.dickinson ),

Welcome! A few pointers / suggestions:

Have you checked your connections?

Edit: Ah! Hang on… I’m seeing some confusion on the POCI / PICO pin labels on the ZED -F9R breakout silk and schematic. ZED Peripheral Out Controller In (POCI) is Pin 42 SPI_SDO / TXD1. ZED Peripheral In Controller Out (PICO) is Pin 43 SPI_SDI / RXD1. Your connections should be:

ESP32 <=> ZED-F9R
13 = MOSI => RX1 (Pin 43)
12 = MISO <= TX1 (Pin 42)
14 = SCK => SCL / CLK
15 = CS => SDA / !CS
GND <=> GND

If the Redboard SPI signals are 3.9V, it is probably a 5V board and the ZED I/O clamp diodes are limiting the voltage. Do be careful there.

What do you see in the console? Is the myGNSS.begin failing?

What do you see if you add myGNSS.enableDebugging(); before the myGNSS.begin?

The 20ms timeout looks very short. What do you see with while (myGNSS.begin(mySPI, myCS, myClockSpeed) == false)?

Try adding:

  pinMode(myCS, OUTPUT);
  digitalWrite(myCS, HIGH);

before the SPI.begin.

I hope this helps,
Paul

Important life lesson: maybe post here before spending an entire week smacking my head against the wall. Thank you so much!!! Success after switching my PICO/POCI pins and adding the CS pin initialization (did both at the same time).

Hi Brent (@brenton.dickinson ),

Thanks for the update. Glad you got it working.

It looks like we introduced some badness when we migrated from the old MISO / MOSI labelling to the new PICO / POCI labelling on the ZED-F9R boards.

In summary:

ZED pin 42 was previously correctly labelled TX / MISO
ZED pin 43 was previously correctly labelled RX / MOSI

The new labelling should be:

ZED pin 42 : TX / POCI (a.k.a. SPI_SDO / TXO (TX Out) / TX1 (TX UART1))
ZED pin 43 : RX / PICO (a.k.a. SPI_SDI / RXI (RX In) / RX1 (RX UART1))

But I see that the updated ZED-F9R silk and schematic use: TXO / PICO, and RXI / POCI. The TXO and RXI are correct. But the PICO and POCI are reversed…

Sorry about that.

I’ll pass the message on.

Best wishes,
Paul