

# THIS SPEC IS OBSOLETE

**Spec No**: 001-98566

**Spec Title**: AN98566 - LINUX CONFIGURATION FOR

**CYPRESS SPI** 

Replaced by: NONE





## **Linux Configuration for Cypress SPI**

**Author: Cypress** 

AN98566 compares the software driver stacks used for parallel NOR and SPI flash devices in Linux and discusses the requirements needed in order to get Cypress SPI devices up and running.

### 1 Introduction

Many customers have been employing parallel NOR flash for years and are now moving towards serial (SPI) flash. This requires not only to replace the underlying hardware but also to apply software changes addressing e.g. the different command set used by SPI flash devices. This document compares the software driver stacks used for parallel NOR and SPI flash devices in Linux. It shows their differences and discusses the requirements needed in order to get Cypress SPI devices up and running.

### 2 Driver Stacks for Parallel and Serial Cypress Flash Devices

The driver stack for parallel NOR flash under Linux is pretty simple. Basically, it includes three layers: a mapping driver that is responsible to make the flash device accessible, a chip driver that implements the vendor specific command set, and finally a flash file system such as JFFS2 on top of the stack that offers the standard file system API to user space applications. Figure 1 shows a block diagram of the driver infrastructure.

File System
(JFFS2)

Chip Driver
(cfi\_cmdset\_0002.c)

Mapping Driver

Memory
Controller

Parallel bus

Flash Device

Devices

Figure 1. Driver Infrastructure for Parallel Flash Devices

For new system designs, only the mapping driver has to be changed. The Linux kernel sources already include a generic mapping driver (physmap.c) that needs just three configuration parameters (physical flash base address, size, and bus width) before it can be used. In any case, mapping drivers for parallel NOR flash are very simple and lightweight. Most of them just map the physical I/O memory of the entire device into the (virtual) kernel address space at initialization time. Afterwards, they forward all read and write word requests directly to the mapped memory window. If multiple devices are attached to the same bus or controller they all get their own independent



driver stack. Proper synchronization of the bus accesses is guaranteed by the memory controller, i.e. it is fully implemented in hardware.

For SPI devices things are more complicated, see Figure 2. File system and chip driver look similar as in the parallel case but the hardware specific portion of the stack is much different. A SPI memory controller typically comprises small buffers (FIFOs) to queue incoming and outgoing data from the bus. These buffers need to be loaded and monitored by a software driver. For example, when the read buffer overflows the controller might generate an interrupt indicating to the software driver that it should retrieve the data from its buffer before moving on.



For every SPI memory controller in the system there must be a dedicated controller driver. That controller driver is not directly connected to the chip driver. Instead, it is connected to a central SPI core framework that manages and routes all SPI traffic in the system. There might be multiple SPI buses in a system each with its own controller driver but there is always just one SPI core framework. All controller and chip drivers register with this framework. Furthermore, board drivers provide detailed information about the system configuration, e.g. the kind of SPI devices available in the system, their maximum supported frequency, the bus and chipselect numbers they are attached to as well as the names of the chip driver that are needed for them. Based on this data, the SPI framework binds every SPI device to the appropriate chip driver at initialization time.

## 3 Building Blocks

The following chapters discuss the individual building blocks and drivers for a SPI solution in more detail.



#### 3.1 Flash File Systems

Due to the standardized MTD interface provided by the chip driver to upper layers, the flash file system drivers usually remain unchanged. The FFS drivers do not need to know many details about the underlying hardware and driver infrastructure. All relevant parameters such as sector size and geometry can be dynamically retrieved via the standardized MTD interface.

When configuring the kernel make sure that the preferred FFS is enabled, i.e. that the kernel option

CONFIG JFFS2 FS

or similar is set.

### 3.2 Chip Driver

Support for Cypress SPI flash devices comes with the MTD chip driver m25p80.c. This driver implements the device specific command set, i.e. the protocol or language needed to talk to the physical device. It includes support for a whole series of compatible SPI flash devices from various vendors. Its code base has been growing continuously in the past with an increasing number of features as well as number of supported devices. Direct support for Cypress SPI devices was first added with kernel version 2.6.27. Since then, the number of directly supported Cypress OPNs (mainly FL-A and FL-P) has increased to nine for kernel version 2.6.34.

If needed, new OPNs can easily be added. For example, the following device entries describe the supported Cypress FL129P devices (256 kB and 64 kB sector versions):

```
{ "s25fl129p0", INFO(0x012018, 0x4d00, 256 * 1024, 64, 0) },
{ "s25fl129p1", INFO(0x012018, 0x4d01, 64 * 1024, 256, 0) },
```

The structure of device entries is pretty simple. Data sets start with a unique string or name followed by the standard and extended device IDs, sector size in bytes, number of sectors and finally optional flags.

Note that m25p80.c currently supports uniform sector layouts only. Furthermore, it supports single I/O only as well as no suspend/resume. In order to enable this driver, the kernel configuration macros

```
CONFIG_MTD_M25P80
CONFIG_MTD
CONFIG_MTD_CHAR
CONFIG_MTD_BLOCK
```

need to be set at least. Basic MTD support is needed so that the chip driver can register its MTD objects with the kernel offering an API to upper driver layers as well as the MTD user utilities (e.g. to format the flash device for a file system or to program an image). The MTD user utilities can be used without changes together with this chip driver.

#### 3.3 SPI Core Framework

The SPI core framework serves as a kind of switchboard for all SPI traffic in the system. Drivers register with the framework and use a message based protocol to issue and process SPI transactions. Its API is architecture independent and is defined in include/linux/spi/spi.h.

One of the most important tasks of the SPI framework is to bind SPI chip drivers to the corresponding controller drivers and physical devices. Usually this happens via a spi\_board\_info struct that is initialized and provided by a board driver during system initialization. This struct provides the name of the appropriate chip driver, the maximum supported clock speed of the device, the bus and chipset numbers the device is attached to as well as some other parameters. The following example describes a SPI flash device that is attached to bus number 1, uses chip select 0 and should be managed by the m25p80 chip driver:

```
static struct spi_board_info spsn_pci_board_info = {
   .modalias = "m25p80",
   .max_speed_hz = 40000000,
   .bus_num = 1,
   .chip_select = 0
};
```



Usually, the struct is registered by the board driver via spi\_register\_board\_info() together with other platform resources. Afterwards, the SPI core framework calls the probing function of the chip driver, i.e. m25p80.c, to check for the presence of a physical device.

To include the SPI core framework in the linux kernel make sure that the configuration option

CONFIG SPI

is enabled and that a spi board info struct is provided by the board driver.

#### 3.4 Controller Driver

The lowest but in many cases also the most complex driver in the stack is the controller driver, i.e. the driver that operates the SPI controller hardware. Unlike with parallel NOR flash, for SPI flash there is a non-trivial software driver needed to manage the memory controller. The controller driver receives and normally also queues incoming SPI read/write requests (often spi\_bitbang.c is used for the queuing purpose). Once all preceding transactions have been completed the controller driver sets the SPI bus clock to the correct value and then starts the transaction. Usually, it has to manage both the Rx and Tx buffers of the hardware controller, i.e. it keeps feeding the Tx side and at the same time drains the Rx side. In most cases this is done via interrupts where the controller issues an interrupt if either the Tx buffer is close to empty (needs to be refilled) and/or the Rx buffer is close to full (needs to be drained). If available, DMA hardware can be used at this stage to run more of these tasks in hardware, thus achieving an improved throughput and freeing some CPU cycles. Finally, once all data has been transferred, the controller driver calls the provided callback function that signals completion of the transaction to the SPI framework.

Figure 3 shows the control flow of a synchronous SPI transaction for a typical SPI controller driver and hardware.



Figure 3. Synchronous SPI Transaction

The SPI core framework provides both synchronous and asynchronous I/O functions. In the asynchronous case, control immediately returns to the chip driver. Once the SPI transaction has been completed the framework informs the chip driver via a callback. Note that the SPI flash chip driver m25p80.c uses synchronous read/write functions only.

When configuring the kernel make sure that

CONFIG\_SPI\_MASTER

[CONFIG YOUR CONTROLLER]



is set where [CONFIG\_YOUR\_CONTROLLER] denotes the SPI memory controller of your system. If there is no suitable controller driver included in the kernel sources you will have to develop your own. There are several controller drivers in drivers/spi that might be used as an example or starting point.

Furthermore, even if there is a controller driver included in your kernel sources it might have been developed for other, more basic SPI devices. For example, some controller drivers support fixed transaction lengths only — mainly because many SPI devices require just a few bytes to be read/written (e.g. a touch screen device). SPI flash is much more demanding in this regard. Basically, flash device transactions on the SPI bus can have any length — if supported by the controller. The m25p80.c chip driver does not split up requests by default so that e.g. read requests can have any length. If there is a hardware limit it needs to be taken care of. Basically, this means splitting up large transfers into smaller chunks.

### 4 Locking and Synchronization

If multiple applications and drivers compete for the same piece of hardware then the code has to include some sort of synchronization. In the SPI flash case there are synchronization points at two levels.

The higher level point is inside the chip driver. For every registered flash device m25p80.c allocates a mutex to synchronize concurrent requests from higher levels. The mutex is locked for the entire duration of any MTD operation (read/write/erase). If other drivers or applications want to access the flash device at the same time they will have to wait until the operation has completed and the mutex is released. This locking scheme implements a coarse granularity compared to the locking scheme used by the parallel NOR flash drivers. Parallel NOR flash drivers can suspend an ongoing operation for incoming high priority requests, thus temporarily unlocking the device. For example the AMD/Cypress driver cfi\_cmdset\_0002.c suspends an ongoing erase if it gets a read request. After the erase has been suspended it serves the read request and then resumes the erase. This is currently not supported by the SPI chip driver.

Next to the protocol level locking there is a low level synchronization point in the controller driver to synchronize concurrent requests to multiple SPI devices on the same bus. If two chip drivers issue requests at the same time to two devices on the same SPI bus then the controller driver needs to guarantee that one request waits until the other has finished. This kind of synchronization is usually implemented by the queuing mechanism within the controller driver. The common queue represents a unique resource that is protected via some sort of locking. Only the driver that owns the lock can add new requests to the queue, others will have to wait until the lock is released. Once all requests have been queued the controller driver does not need to apply further synchronization means. It can simply take the next request from the queue head, serve it and continue this way until all requests are served, i.e. until the queue is empty.

# 5 Example

Let's consider a simple example showing how to add SPI support for a FL128P under Linux 2.6.31, based on a Cypress PCI card. Cypress PCI cards are universal development and debugging tools that are available from Cypress Hardware Development Tools.

In this case, our board driver is a simple Linux PCI driver. Its initialization or probing function is automatically called by the kernel when the given PCI vendor and device IDs are detected during the PCI bus scan at system start-up. The PCI probing function serves as a good location to initialize the SPI infrastructure as well, i.e. to provide the required spi\_board\_info struct. The following piece of code shows how this initialization is done in detail.

```
#include #include inux/spi/spi.h>
...
static struct spi_board_info spsn_pci_board_info = {
    .modalias = "m25p80",
    .max_speed_hz = 40000000,
    .bus_num = 0,
    .chip_select = 0
};
static int spsn_pci_probe(struct pci_dev *dev, const struct pci_device_id *id)
{
    ...
}
```



```
// Register SPI master controller driver and device
board.smaster = spi_alloc_master( &dev->dev, 0);
if (board.smaster == NULL)
   { printk( "ERROR: spi alloc master failed!\n"); return -ENODEV; }
board.smaster->bus num = 0;
board.smaster->num chipselect = 1;
board.smaster->setup
                      = spsn_pci_spi_setup;
board.smaster->transfer = spsn pci spi transfer;
board.smaster->cleanup = spsn pci spi cleanup;
if (spi register master( board.smaster))
   { printk ("ERROR spi register master() failed!\n");
      spi_master_put( board.smaster); return -1; }
if (spi new device (board.smaster, &spsn pci board info) == NULL)
   { printk ("ERROR spi new device() failed!\n");
       spi master put( board.smaster); return -1; }
printk( "spi master controller and device registered\n");
```

Next to the registration of the spi\_board\_info struct the code also allocates, initializes, and registers a master controller driver. The exact implementation of this part is beyond the scope of this document.

Nevertheless, the piece of code shows how the different components nicely play together. After the SPI device has been registered via spi\_new\_device(), the SPI core framework loads the m25p80 kernel module, binds the chip driver to the controller driver and finally calls the probe() function of m25p80.c. If a physical device is present, it will be detected by the chip driver and a MTD object will be registered.

Of course, the kernel configuration options mentioned above have to be set as well. The m25p80.c driver can remain unchanged in this case since the FL128P is already supported by default.

# 6 Questions and Support

If you have further questions or support requests please direct them to the Cypress Engineering Solutions portal at http://www.spansion.com/Support/SES/Pages/ask\_spansion.html.



# **Document History Page**

| Document Title: AN98566 - Linux Configuration for Cypress SPI Document Number: 001-98566 |         |                    |                    |                                                                      |
|------------------------------------------------------------------------------------------|---------|--------------------|--------------------|----------------------------------------------------------------------|
| Rev.                                                                                     | ECN No. | Orig. of<br>Change | Submission<br>Date | Description of Change                                                |
| **                                                                                       | -       | -                  | 08/09/2010         | Initial version                                                      |
| *A                                                                                       | 4928144 | MSWI               | 09/21/2015         | Updated in Cypress template                                          |
| *B                                                                                       | 5844295 | AESATMP8           | 08/04/2017         | Updated logo and Copyright.                                          |
| *C                                                                                       | 6035383 | GEHO               | 01/19/2018         | Obsolete the document, as the contents are old and no longer needed. |





## **Worldwide Sales and Design Support**

Cypress maintains a worldwide network of offices, solution centers, manufacturer's representatives, and distributors. To find the office closest to you, visit us at Cypress Locations.

#### **Products**

Interface

Arm® Cortex® Microcontrollers

Automotive

Clocks & Buffers

Internet of Things

Memory

Microcontrollers

MICIOCOLITIONEIS

PSoC

Power Management ICs

Touch Sensing

USB Controllers
Wireless Connectivity

cypress.com/arm

cypress.com/automotive

cypress.com/clocks

cypress.com/interface

cypress.com/iot

cypress.com/memory

cypress.com/mcu

cypress.com/psoc

cypress.com/psoc

cypress.com/touch

cypress.com/usb

cypress.com/wireless

#### PSoC® Solutions

PSoC 1 | PSoC 3 | PSoC 4 | PSoC 5LP | PSoC 6

#### **Cypress Developer Community**

Forums | WICED IOT Forums | Projects | Video | Blogs | Training | Components

#### **Technical Support**

cypress.com/support

All other trademarks or registered trademarks referenced herein are the property of their respective owners.



Cypress Semiconductor 198 Champion Court San Jose, CA 95134-1709

© Cypress Semiconductor Corporation, 2010-2018. This document is the property of Cypress Semiconductor Corporation and its subsidiaries, including Spansion LLC ("Cypress"). This document, including any software or firmware included or referenced in this document ("Software"), is owned by Cypress under the intellectual property laws and treaties of the United States and other countries worldwide. Cypress reserves all rights under such laws and treaties and does not, except as specifically stated in this paragraph, grant any license under its patents, copyrights, trademarks, or other intellectual property rights. If the Software is not accompanied by a license agreement and you do not otherwise have a written agreement with Cypress governing the use of the Software, then Cypress hereby grants you a personal, non-exclusive, nontransferable license (without the right to sublicense) (1s) under its copyright rights in the Software (a) for Software provided in source code form, to modify and reproduce the Software solely for use with Cypress hardware products, only internally within your organization, and (b) to distribute the Software in binary code form externally to end users (either directly or indirectly through resellers and distributors), solely for use on Cypress hardware product units, and (2) under those claims of Cypress's patents that are infringed by the Software (as provided by Cypress, unmodified) to make, use, distribute, and import the Software solely for use with Cypress hardware products. Any other use, reproduction, modification, translation, or compilation of the Software is prohibited.

TO THE EXTENT PERMITTED BY APPLICABLE LAW, CYPRESS MAKES NO WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, WITH REGARD TO THIS DOCUMENT OR ANY SOFTWARE OR ACCOMPANYING HARDWARE, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. To the extent permitted by applicable law, Cypress reserves the right to make changes to this document without further notice. Cypress does not assume any liability arising out of the application or use of any product or circuit described in this document. Any information provided in this document, including any sample design information or programming code, is provided only for reference purposes. It is the responsibility of the user of this document to properly design, program, and test the functionality and safety of any application made of this information and any resulting product. Cypress products are not designed, intended, or authorized for use as critical components in systems designed or intended for the operation of weapons, weapons systems, nuclear installations, life-support devices or systems, other medical devices or systems (including resuscitation equipment and surgical implants), pollution control or hazardous substances management, or other uses where the failure of the device or system could cause personal injury, death, or property damage ("Unintended Uses"). A critical component is any component of a device or system whose failure to perform can be reasonably expected to cause the failure of the device or system, or to affect its safety or effectiveness. Cypress is not liable, in whole or in part, and you shall and hereby do release Cypress from any claim, damage, or other liability arising from or related to all Unintended Uses of Cypress products. You shall indemnify and hold Cypress harmless from and against all claims, costs, damages, and other liabilities, including claims for personal injury or death, arising from or related to any Unintended Uses of Cypress products.

Cypress, the Cypress logo, Spansion, the Spansion logo, and combinations thereof, WICED, PSoC, CapSense, EZ-USB, F-RAM, and Traveo are trademarks or registered trademarks of Cypress in the United States and other countries. For a more complete list of Cypress trademarks, visit cypress.com. Other names and brands may be claimed as property of their respective owners.