3.1.1.7. OSPI/QSPI NOR/NAND
Introduction
Octal Serial Peripheral Interface (OSPI) and Quad Serial Peripheral Interface (QSPI) are SPI modules that have x8 IO lines and x4 IO lines respectively. These controllers are mainly used to interface with Octal and Quad SPI flashes. OSPI is backward compatible with QSPI. These modules can also work in dual (x2) and single (x1) modes. OSPI and QSPI controllers on TI SoCs support memory mapped IO interface, which provide a direct interface for accessing data from the external SPI flash, thereby simplifying software requirements. These controllers work only in master mode.
SoC Family |
Capability |
Driver |
---|---|---|
J722S |
1x OSPI NOR |
|
1x OSPI NAND |
|
Note
Not all OSPI flashes can be supported. Users are recommended to check whether or not the OSPI flash part chosen for custom board designs meets all the criteria listed at https://e2e.ti.com/support/processors/f/791/t/946418
Driver Features
OSPI controllers supports PHY Calibration in DQS + Double Data Rate (DDR) mode for OSPI/QSPI NOR flashes in Octal configuration wherein data can be read on both edges of the clock, and non-DQS + Single Data Rate (SDR) mode for OSPI/QSPI NAND flashes in Quad and Octal configuration.
Memory mapped read support
Once the controller is configured in memory map mode, the whole flash memory is made available as a memory region at an SoC specific address. This region can be accessed using normal memcpy() (or mem-to-mem dma copy). Controller hardware will internally communicate with SPI flash over SPI bus and get the requested data. This mode provides the best throughput and is the default mode in the SDK.
Supported SPI modes
The cadence_qspi.c
driver supports standard SPI mode 0 only.
DMA support
The driver uses mem-to-mem DMA copy on top of an OSPI/QSPI memory mapped port during flash read operations for maximum throughput and reduced CPU load.
The OSPI Controller does not support interfacing with non-flash SPI slaves.
Driver Configuration
Source Location
OSPI driver is at: drivers/spi/cadence_qspi.c
under U-Boot source tree.
This driver also supports QSPI version of the same IP.
DT Configuration
The following is an example device-tree node for an OSPI NOR device
&ospi0 {
flash@0{
compatible = "jedec,spi-nor";
reg = <0x0>;
spi-tx-bus-width = <8>;
spi-rx-bus-width = <8>;
spi-max-frequency = <25000000>;
cdns,tshsl-ns = <60>;
cdns,tsd2d-ns = <60>;
cdns,tchsh-ns = <60>;
cdns,tslch-ns = <60>;
cdns,read-delay = <4>;
partitions {
compatible = "fixed-partitions";
#address-cells = <1>;
#size-cells = <1>;
bootph-all;
partition@0 {
label = "ospi.tiboot3";
reg = <0x00 0x80000>;
};
partition@80000 {
label = "ospi.tispl";
reg = <0x80000 0x200000>;
};
// other partitions
};
};
};
The following is an example device-tree node for an OSPI NAND device
&ospi0 {
flash@0 {
compatible = "spi-nand";
reg = <0x0>;
spi-tx-bus-width = <8>;
spi-rx-bus-width = <8>;
spi-max-frequency = <25000000>;
cdns,tshsl-ns = <60>;
cdns,tsd2d-ns = <60>;
cdns,tchsh-ns = <60>;
cdns,tslch-ns = <60>;
cdns,read-delay = <2>;
partitions {
compatible = "fixed-partitions";
#address-cells = <1>;
#size-cells = <1>;
partition@0 {
label = "ospi_nand.tiboot3";
reg = <0x0 0x80000>;
};
partition@80000 {
label = "ospi_nand.tispl";
reg = <0x80000 0x200000>;
};
// other partitions
};
};
};
Flash properties:
compatible: specifies the compatible string for the device, the operating system uses this string to identify and the match the driver for the device. Use ‘jedec,spi-nor’ for OSPI/QSPI NOR flashes and ‘spi-nand’ for OSPI/QSPI NAND flashes.
spi-tx-bus-width and spi-rx-bus-width: specifies the bus width in bits for SPI transactions when transmitting (tx) and receiving (rx) data. Set for ‘8’ for OSPI flashes and ‘4’ for QSPI flashes.
spi-max-frequency: defines the maximum frequency in Hertz at which the SPI bus can operate. Set 1/4th or 1/8th of ‘assigned-clocks’ value of ‘ospi0’ node for SDR and DDR mode respectively. If PHY Calibration is enabled, this value is ignored, and the maximum frequency is determined by the value specified in the ‘assigned-clocks’ property of ‘ospi0’ node.
cdns,read-delay: specifies the delay in clock cycles between the fetch of a command and responding to that command by the flash devices. This differs with flashes, try with different read delays starting from 0 and find the minimum read-delay at which the flash driver probes correctly.
Note
The sf command is used to access SPI NOR flash, supporting read/write/erase and a few other functions. For more information on sf command in U-boot please refer to the u-boot documentation: here. And for accessing SPI NAND flash, the mtd command is used, supporting read/write/erase and bad block management.
AM67 is similar to J721S2, only difference being that OSPI0 is muxed externally between a NOR and a NAND flash through a physical switch. OSPI NOR and OSPI NAND can’t be used at the same time, they need to be selected by changing a physical configuration switch on the EVM board before driver probes them. CONFIG SW3.1 should be in OFF state to use OSPI NOR, and in ON STATE for OSPI NAND.
Flashing images to OSPI NOR flash
Following commands can be used to download tiboot3.bin, tispl.bin and u-boot.img over tftp and then flash it to OSPI NOR/NAND at respective addresses.
OSPI NOR:
=> sf probe => tftp ${loadaddr} tiboot3.bin => sf update $loadaddr 0x0 $filesize => tftp ${loadaddr} tispl.bin => sf update $loadaddr 0x80000 $filesize => tftp ${loadaddr} u-boot.img => sf update $loadaddr 0x280000 $filesize
OSPI NAND:
=> mtd list => mtd erase spi-nand0 => tftp $loadaddr tiboot3.bin => mtd write spi-nand0 $loadaddr 0x0 $filesize => tftp $loadaddr tispl.bin => mtd write spi-nand0 $loadaddr 0x80000 $filesize => tftp $loadaddr u-boot.img => mtd write spi-nand0 $loadaddr 0x280000 $filesize
PHY calibration
PHY calibration allows for higher read performance. To enable PHY, the PHY calibration pattern must be flashed to ospi at the start of the last erase sector. For the cypress s28hs512t flash, this lies at the address 0x3fc0000. Note, PHY calibration is currently supported only for OSPI NOR flash.
download the binary file containing the phy pattern from here
.
below commands can be used to flash the phy pattern, with the location of the
pattern depending on which flash is being used:
=> sf probe
=> tftp ${loadaddr} ospi_phy_pattern
=> sf update $loadaddr 0x3fc0000 $filesize
Flash layout for OSPI NOR
0x0 +----------------------------+
| ospi.tiboot3(1m) |
| |
0x80000 +----------------------------+
| ospi.tispl(2m) |
| |
0x280000 +----------------------------+
| ospi.u-boot(4m) |
| |
0x680000 +----------------------------+
| ospi.env(128k) |
| |
0x6A0000 +----------------------------+
| ospi.env.backup(128k) |
| |
0x6C0000 +----------------------------+
| padding (1280k) |
0x800000 +----------------------------+
| ospi.rootfs(ubifs) |
| |
0x3fc0000 +----------------------------+
| ospi.phypattern (256k) |
| |
+----------------------------+
Flash layout for OSPI NAND
0x0 +---------------------------------+
| ospi_nand.tiboot3(1m) |
| |
0x80000 +---------------------------------+
| ospi_nand.tispl(2m) |
| |
0x280000 +---------------------------------+
| ospi_nand.u-boot(4m) |
| |
0x680000 +---------------------------------+
| ospi_nand.env(128k) |
| |
0x6A0000 +---------------------------------+
| ospi_nand.env.backup(128k) |
| |
0x6C0000 +---------------------------------+
| padding (98048K) |
0x2000000 +---------------------------------+
| ospi_nand.rootfs(ubifs) |
| |
0x7fc0000 +---------------------------------+
| ospi_nand.phypattern (256k) |
| |
+---------------------------------+
Writing to OSPI using DFU
Setup: Connect the Type C port (USB0 port) of EVM to ubuntu host PC. Make sure dfu-util tool is installed and USB0 port is in UFP/DRP mode: SW3[3:4] = 01 or 1x.
#sudo apt-get install dfu-util
From u-boot(for OSPI flash):
U-Boot # env default -a
U-Boot # setenv dfu_alt_info ${dfu_alt_info_ospi}; dfu 0 sf "0:0:25000000:0"
From ubuntu PC: Using dfu-util utilities to flash the binares to OSPI/QSPI flash.
# sudo dfu-util -l
Copyright 2005-2009 Weston Schmidt, Harald Welte and OpenMoko Inc.
Copyright 2010-2016 Tormod Volden and Stefan Schmidt
This program is Free Software and has ABSOLUTELY NO WARRANTY
Found DFU: [0451:6168] ver=0224, devnum=44, cfg=1, intf=0, path="1-3", alt=5, name="rootfs", serial="UNKNOWN"
Found DFU: [0451:6168] ver=0224, devnum=44, cfg=1, intf=0, path="1-3", alt=4, name="sysfw.itb", serial="UNKNOWN"
Found DFU: [0451:6168] ver=0224, devnum=44, cfg=1, intf=0, path="1-3", alt=3, name="u-boot-env", serial="UNKNOWN"
Found DFU: [0451:6168] ver=0224, devnum=44, cfg=1, intf=0, path="1-3", alt=2, name="u-boot.img", serial="UNKNOWN"
Found DFU: [0451:6168] ver=0224, devnum=44, cfg=1, intf=0, path="1-3", alt=1, name="tispl.bin", serial="UNKNOWN"
Found DFU: [0451:6168] ver=0224, devnum=44, cfg=1, intf=0, path="1-3", alt=0, name="tiboot3.bin", serial="UNKNOWN"
Flash the binaries to the respective regions using alternate interface number (alt=<x>).
# sudo dfu-util -c 1 -i 0 -a 0 -D tiboot3.bin
# sudo dfu-util -c 1 -i 0 -a 1 -D tispl.bin
# sudo dfu-util -c 1 -i 0 -a 2 -D u-boot.img