## SPI Driver

Version 1.0.0

Generated by Doxygen 1.8.13

# **Contents**

## **Chapter 1**

# **Portable SPI driver**

Written as per the standards of Beningo's book. Marks the final driver I'll probably be making for the hal. TODO for all communication drivers: implement systick for timeouts

2 Portable SPI driver

# Chapter 2

# **Data Structure Index**

## 2.1 Data Structures

| Here are the data structures w | vith brief | descriptions |
|--------------------------------|------------|--------------|
|--------------------------------|------------|--------------|

| spi_config_t . |  |  | <br> |  |  |  |  |  |  |  | <br> |  |  |  |  |  | <br> |  |  |  | 7 | ?? |
|----------------|--|--|------|--|--|--|--|--|--|--|------|--|--|--|--|--|------|--|--|--|---|----|
| spi transfer t |  |  | <br> |  |  |  |  |  |  |  | <br> |  |  |  |  |  | <br> |  |  |  | - | ?? |

Data Structure Index

# **Chapter 3**

# File Index

## 3.1 File List

Here is a list of all documented files with brief descriptions:

| /home/marko/Documents/embedded_workspace/spi_driver/spi_interface.h                               |    |
|---------------------------------------------------------------------------------------------------|----|
| General interface covering user accesses to the spi communication bus                             | ?? |
| /home/marko/Documents/embedded_workspace/spi_driver/spi_stm32f411.c                               |    |
| Chip specific implementation for spi communication                                                | ?? |
| /home/marko/Documents/embedded_workspace/spi_driver/spi_stm32f411_config.c                        |    |
| Contains the configuration information for each spi channel                                       | ?? |
| /home/marko/Documents/embedded_workspace/spi_driver/spi_stm32f411_config.h                        |    |
| Contains the definitions and structures required to configure the spi peripherals on an stm32f411 | ?? |
|                                                                                                   |    |

6 File Index

## **Chapter 4**

## **Data Structure Documentation**

## 4.1 spi\_config\_t Struct Reference

```
#include <spi_stm32f411_config.h>
```

#### **Data Fields**

- spi\_enable\_t spi\_enable
- spi\_master\_slave\_t master\_slave
- spi\_slave\_mgmt\_t slave\_management
- spi\_bidir\_t bidirectional\_mode
- spi\_baud\_rate\_t baud\_rate

## 4.1.1 Detailed Description

An initialisation config struct which configures the global parameters for a spi device

#### 4.1.2 Field Documentation

```
4.1.2.1 baud_rate
```

spi\_baud\_rate\_t baud\_rate

Communication rate of the spi

4.1.2.2 bidirectional\_mode

```
spi_bidir_t bidirectional_mode
```

Configured based upon physical topology of the spi

#### 4.1.2.3 master\_slave

```
spi_master_slave_t master_slave
```

Decides whether the spi is in master or slave mode

## 4.1.2.4 slave\_management

```
spi_slave_mgmt_t slave_management
```

Determines method of (self) slave management

## 4.1.2.5 spi\_enable

```
spi_enable_t spi_enable
```

Decides whether the spi will be used at all

The documentation for this struct was generated from the following file:

• /home/marko/Documents/embedded\_workspace/spi\_driver/spi\_stm32f411\_config.h

## 4.2 spi\_transfer\_t Struct Reference

```
#include <spi_interface.h>
```

## **Data Fields**

- spi\_channel\_t channel
- gpio\_pin\_t slave\_pin
- spi\_ss\_polarity\_t ss\_polarity
- uint16\_t \* tx\_buffer
- uint32\_t tx\_length
- uint16\_t \* rx\_buffer
- uint32\_t rx\_length
- spi\_data\_format\_t data\_format
- spi\_bit\_format\_t bit\_format
- spi\_clock\_polarity\_t clock\_polarity
- spi\_clock\_phase\_t clock\_phase
- spi\_bidir\_dir\_t bidir\_direction

## 4.2.1 Detailed Description

Struct containing implementation agnostic transfer information.

Pointer to the data buffer for reception

Generated by Doxygen

## 4.2.2 Field Documentation

```
4.2.2.1 bidir_direction
spi_bidir_dir_t bidir_direction
Direction of the single data line transfer
4.2.2.2 bit_format
spi_bit_format_t bit_format
MSB or LSB first
4.2.2.3 channel
spi_channel_t channel
The on-chip spi device to manage the transfer
4.2.2.4 clock_phase
spi_clock_phase_t clock_phase
Edge sensitivity on sampling and shifts
4.2.2.5 clock_polarity
spi_clock_polarity_t clock_polarity
Selection of the clock's active and idle states
4.2.2.6 data_format
spi_data_format_t data_format
Data size of the transfer elements
4.2.2.7 rx_buffer
uint16_t* rx_buffer
```

```
4.2.2.8 rx_length

uint32_t rx_length

Length of the reception buffer

4.2.2.9 slave_pin

gpio_pin_t slave_pin

The slave's ss pin
```

spi\_ss\_polarity\_t ss\_polarity

The polarity of slave\_pin

4.2.2.10 ss\_polarity

4.2.2.11 tx\_buffer

uint16\_t\* tx\_buffer

Pointer to the data buffer for transfers

4.2.2.12 tx\_length

uint32\_t tx\_length

Length of the transfer buffer

The documentation for this struct was generated from the following file:

• /home/marko/Documents/embedded\_workspace/spi\_driver/spi\_interface.h

## **Chapter 5**

## **File Documentation**

5.1 /home/marko/Documents/embedded\_workspace/spi\_driver/spi\_interface.h File Reference

General interface covering user accesses to the spi communication bus.

```
#include "spi_stm32f411_config.h"
#include "gpio_interface.h"
#include <stdint.h>
Include dependency graph for spi_interface.h:
```



This graph shows which files directly or indirectly include this file:



## **Data Structures**

· struct spi transfer t

#### **Enumerations**

- enum spi\_ss\_polarity\_t { SS\_ACTIVE\_LOW, SS\_ACTIVE\_HIGH }
- enum spi\_clock\_polarity\_t { ACTIVE\_HIGH, ACTIVE\_LOW }
- enum spi\_clock\_phase\_t { FIRST\_EDGE, SECOND\_EDGE }
- enum spi\_bit\_format\_t { MSB\_FIRST, LSB\_FIRST }
- enum spi data format t { SPI DATA 8BIT, SPI DATA 16BIT }
- enum spi\_bidir\_dir\_t { BIDIR\_RECEIVE, BIDIR\_TRANSMIT }

#### **Functions**

- void spi init (spi config t \*config table)
- void spi\_transfer (spi\_transfer\_t \*transfer)
- void spi\_transfer\_it (spi\_transfer\_t \*transfer)
- · void spi irq handler (spi channel t channel)
- void spi\_register\_write (uint32\_t spi\_register, uint16\_t value)
- uint16\_t spi\_register\_read (uint32\_t spi\_register)

## 5.1.1 Detailed Description

General interface covering user accesses to the spi communication bus.

## 5.1.2 Enumeration Type Documentation

```
5.1.2.1 spi_bidir_dir_t
enum spi_bidir_dir_t
```

Contains the direction selections when using single data wire (Bidrectional) spi

## Enumerator

| BIDIR_RECEIVE  | The data line is being used for data reception    |
|----------------|---------------------------------------------------|
|                | The data line is being used for data transmission |
| BIDIR_TRANSMIT |                                                   |

## 5.1.2.2 spi\_bit\_format\_t

enum spi\_bit\_format\_t

Contanins the options for the bit order of transfers

### Enumerator

| MSB_FIRST | The most significant bit is sent first  |
|-----------|-----------------------------------------|
| LSB_FIRST | The least significant bit is sent first |

## 5.1.2.3 spi\_clock\_phase\_t

enum spi\_clock\_phase\_t

Contains the options for the clock's phase

### Enumerator

| FIRST_EDGE  | Data is clocked out/sampled on the first edge  |
|-------------|------------------------------------------------|
| SECOND_EDGE | Data is clocked out/sampled on the second edge |

## 5.1.2.4 spi\_clock\_polarity\_t

enum spi\_clock\_polarity\_t

Contains the options for the clock's polarity

## Enumerator

| Ī | ACTIVE_HIGH | The clock's idle state is low and ticks upwards    |
|---|-------------|----------------------------------------------------|
|   | ACTIVE_LOW  | The clock's idle state is high and ticks downwards |

```
5.1.2.5 spi_data_format_t
```

```
enum spi_data_format_t
```

Contains the options for payload size

#### Enumerator

| SPI_DATA_8BIT  | The data is in 8 bit frames  |
|----------------|------------------------------|
|                | The data is in 16 bit frames |
| SPI_DATA_16BIT |                              |

```
5.1.2.6 spi_ss_polarity_t
```

```
enum spi_ss_polarity_t
```

Contains the options for slave select

#### Enumerator

| SS_ACTIVE_LOW  | A slave is selected by pulling its select pin low  |
|----------------|----------------------------------------------------|
| SS_ACTIVE_HIGH | A slave is selected by pulling its select pin high |

## 5.1.3 Function Documentation

```
5.1.3.1 spi_init()
```

## **Description:**

Carries out the initialisation of the spi channels as per the information in the config table

PRE-CONDITION: The config table has been obtained and is non-null PRE-CONDITION: The required GPIO pins for the spi combination have been configured correctly with gpio\_init PRE-CONDITION: The appropriate peripheral clocks have been activated

POST-CONDITION: The selected spi channels have been activated and ready to be used

Returns

void

## Example:

```
const spi_config_t *config_table = spi_config_get();
spi_init(config_table);
```

See also

spi\_config\_get

- CHANGE HISTORY -

| Date Software Version | Initials | Description |
|-----------------------|----------|-------------|
|-----------------------|----------|-------------|

## 5.1.3.2 spi\_irq\_handler()

## **Description:**

Calls the appropriate callback function (registered during spi\_transfer\_it) and feeds it a safe copy of the desired transfer (made during spi\_transfer\_it).

PRE-CONDITION: spi\_transfer\_it has been called and set up on the desired channel POST-CONDITION: The callback has been called and has handled a single reception/transfer/end of transfer

Returns

void

**Example:** Called from withi the hardware defined irgs defined in the vector table

```
SPI3_IRQHandler()
{
     spi_irq_handler(SPI_3);
}
```

## See also

spi\_transfer\_it

- CHANGE HISTORY -

| Date Software Version | Initials | Description |  |
|-----------------------|----------|-------------|--|
|-----------------------|----------|-------------|--|

```
5.1.3.3 spi_register_read()
```

### **Description:**

Reads the current value from the register in spi address space

PRE-CONDITION: the spi\_register is within spi address space POST-CONDITION: Returns the value within the register

#### Returns

uint16\_t

## Example:

```
uin16_t current_value = spi_register_write(SPI1_BASE + 0x20UL);
```

## See also

spi\_register\_write

- CHANGE HISTORY -

| Date Software Version | Initials | Description |
|-----------------------|----------|-------------|
|-----------------------|----------|-------------|

## 5.1.3.4 spi\_register\_write()

## Description:

Write the desired value into the register in spi address space

PRE-CONDITION: the spi\_register is within spi address space POST-CONDITION: the spi\_register contains the desired value

Returns

void

## Example:

```
uint16_t new_value = 0xFF2A;
spi_register_write(SPI1_BASE + 0x20UL, new_value);
```

#### See also

```
spi_register_read
```

- CHANGE HISTORY -

| Date | Software Version | Initials | Description |
|------|------------------|----------|-------------|
|------|------------------|----------|-------------|

## 5.1.3.5 spi\_transfer()

## **Description:**

```
Carries out a blocking spi transfer according to the specifications of the transfer parameter.
```

PRE-CONDITION: spi\_init() has been successfully carried out for the required spi channel PRE-CONDITION 
: gpio\_init() has been called for the slave select pin to configure it as an output/input, depending on desired direction 
PRE-CONDITION: The proper data buffers and lengths are non-null/non-zero

POST-CONDITION: The desired transfer has been successfully carried out

Returns

void

## Example:

```
spi_transfer_t flash_transfer;
flash_transfer.channel = SPI_3;
flash_transfer.slave_pin = GPIO_C_3;
flash_transfer.ss_polarity = ACTIVE_LOW;
flash_transfer.tx_buffer = &data_out;
flash_transfer.tx_length = sizeof(data_out);
flash_transfer.rx_buffer = &dummy_data;
flash_transfer.rx_length = sizeof(data_out);
flash_transfer.rx_length = sizeof(data_out);
flash_transfer.data_format = SPI_DATA_8BIT;
flash_transfer.bit_format = MSB_FIRST;
flash_transfer.clock_polarity = ACTIVE_HIGH;
flash_transfer.clock_phase = SECOND_EDGE;
spi_transfer(&flash_transfer);
```

#### See also

```
spi_init
spi_transfer_it
```

- CHANGE HISTORY -

| Date Software Version | Initials | Description |
|-----------------------|----------|-------------|
|-----------------------|----------|-------------|

```
5.1.3.6 spi_transfer_it()
```

## **Description:**

```
Sets up an interrupt based spi transfer according to the specifications of the transfer parameter.
```

PRE-CONDITION: spi\_init() has been successfully carried out for the required spi channel PRE-CONDITION 

: gpio\_init() has been called for the slave select pin to configure it as an output/input, depending on desired direction 
PRE-CONDITION: The proper data buffers and lengths are non-null/non-zero

POST-CONDITION: The irq handler will now handle the rest of the transfer

#### Returns

void

## Example:

```
spi_transfer_t flash_transfer;
flash_transfer.channel = SPI_3;
flash_transfer.slave_pin = GPIO_C_3;
flash_transfer.ss_polarity = ACTIVE_LOW;
flash_transfer.tx_buffer = &data_out;
flash_transfer.tx_length = sizeof(data_out);
flash_transfer.rx_length = sizeof(data_out);
flash_transfer.rx_length = sizeof(data_out);
flash_transfer.data_format = SPI_DATA_8BIT;
flash_transfer.bit_format = MSB_FIRST;
flash_transfer.clock_polarity = ACTIVE_HIGH;
flash_transfer.clock_phase = SECOND_EDGE;
spi_transfer_it(&flash_transfer);
```

See also

```
spi_init
spi_transfer
spi_irq_handler
- CHANGE HISTORY -
```

| Date   Software Version | Initials | Description |
|-------------------------|----------|-------------|
|-------------------------|----------|-------------|

# 5.2 /home/marko/Documents/embedded\_workspace/spi\_driver/spi\_stm32f411.c File Reference

Chip specific implementation for spi communication.

```
#include "spi_interface.h"
#include "stm32f411xe.h"
#include <assert.h>
Include dependency graph for spi_stm32f411.c:
```



### **Macros**

• #define **NULL** (void\*) 0

## **Typedefs**

typedef void(\* spi\_interrupt\_callback\_t) (spi\_transfer\_t \*)

#### **Functions**

- static void **spi select slave** (spi transfer t \*transfer)
- static void **spi\_release\_slave** (**spi\_transfer\_t** \*transfer)
- static void **spi configure clock** (**spi transfer t** \*transfer)
- static void **spi\_configure\_data\_frame** (**spi\_transfer** t \*transfer)
- static void **spi transfer bidir** (spi transfer t \*transfer)
- static void spi transfer bidir transmit (spi transfer t \*transfer)
- static void **spi\_transfer\_bidir\_receive** (**spi\_transfer** t \*transfer)
- static void spi\_transfer\_full\_duplex\_rxonly (spi\_transfer\_t \*transfer)
- static void **spi\_transfer\_full\_duplex** (**spi\_transfer\_t** \*transfer)
- static void **spi\_transfer\_full\_duplex\_master** (**spi\_transfer\_t** \*transfer)
- static void spi\_transfer\_full\_duplex\_slave (spi\_transfer\_t \*transfer)
- static void **spi\_transfer\_it\_bidir** (**spi\_transfer\_t** \*transfer)
- static void spi\_transfer\_it\_full\_duplex\_rxonly (spi\_transfer\_t \*transfer)
- static void spi transfer it full duplex (spi transfer t \*transfer)
- void spi\_init (spi\_config\_t \*config\_table)
- void spi\_transfer (spi\_transfer\_t \*transfer)
- void spi transfer it (spi transfer t \*transfer)
- void spi\_irq\_handler (spi\_channel\_t channel)
- void spi\_register\_write (uint32\_t spi\_register, uint16\_t value)
- uint16 t spi register read (uint32 t spi register)
- static void spi transfer it bidir transmit callback (spi transfer t \*transfer)
- static void spi\_transfer\_it\_bidir\_receive\_callback (spi\_transfer\_t \*transfer)
- static void spi\_transfer\_it\_full\_duplex\_rxonly\_callback (spi\_transfer\_t \*transfer)
- static void spi\_transfer\_it\_full\_duplex\_callback (spi\_transfer\_t \*transfer)

## **Variables**

- static volatile uint16 t \*const SPI CR1 [NUM SPI]
- static volatile uint16 t \*const SPI CR2 [NUM SPI]
- static volatile uint16\_t \*const SPI\_SR [NUM\_SPI]
- static volatile uint16\_t \*const SPI\_DR [NUM\_SPI]
- static volatile uint16\_t \*const SPI\_CRCPR [NUM\_SPI]
- static volatile uint16\_t \*const SPI\_RXCRCR [NUM\_SPI]
- static volatile uint16\_t \*const SPI\_TXCRCR [NUM\_SPI]
- static spi transfer t spi interrupt transfers [NUM SPI]
- static spi\_interrupt\_callback\_t spi\_interrupt\_callbacks [NUM\_SPI]

## 5.2.1 Detailed Description

Chip specific implementation for spi communication.

## 5.2.2 Typedef Documentation

```
5.2.2.1 spi_interrupt_callback_t
```

```
typedef void(* spi_interrupt_callback_t) (spi_transfer_t *)
```

Callback typedef for interrupt callbacks

## 5.2.3 Function Documentation

```
5.2.3.1 spi_init()
```

#### **Description:**

Carries out the initialisation of the spi channels as per the information in the config table  $% \left( 1\right) =\left( 1\right) +\left( 1$ 

PRE-CONDITION: The config table has been obtained and is non-null PRE-CONDITION: The required GPIO pins for the spi combination have been configured correctly with gpio\_init PRE-CONDITION: The appropriate peripheral clocks have been activated

POST-CONDITION: The selected spi channels have been activated and ready to be used

#### Returns

void

## Example:

```
const spi_config_t *config_table = spi_config_get();
spi_init(config_table);
```

#### See also

## spi\_config\_get

- CHANGE HISTORY -

| Date   Software Version   Initials   Description |
|--------------------------------------------------|
|--------------------------------------------------|

#### 5.2.3.2 spi\_irq\_handler()

#### **Description:**

Calls the appropriate callback function (registered during spi\_transfer\_it) and feeds it a safe copy of the desired transfer (made during spi\_transfer\_it).

PRE-CONDITION: spi\_transfer\_it has been called and set up on the desired channel POST-CONDITION: The callback has been called and has handled a single reception/transfer/end of transfer

#### Returns

void

**Example:** Called from withi the hardware defined irgs defined in the vector table

```
SPI3_IRQHandler()
{
    spi_irq_handler(SPI_3);
}
```

#### See also

```
spi_transfer_it
```

- CHANGE HISTORY -

| Date | Software Version | Initials | Description |  |
|------|------------------|----------|-------------|--|
|------|------------------|----------|-------------|--|

#### 5.2.3.3 spi\_register\_read()

## **Description:**

Reads the current value from the register in spi address space

PRE-CONDITION: the spi\_register is within spi address space POST-CONDITION: Returns the value within the register

Returns

uint16\_t

## Example:

```
uin16_t current_value = spi_register_write(SPI1_BASE + 0x20UL);
```

#### See also

spi\_register\_write

- CHANGE HISTORY -

| Date | Software Version | Initials | Description |
|------|------------------|----------|-------------|
|------|------------------|----------|-------------|

## 5.2.3.4 spi\_register\_write()

## **Description:**

Write the desired value into the register in spi address space

PRE-CONDITION: the spi\_register is within spi address space POST-CONDITION: the spi\_register contains the desired value

Returns

void

## Example:

```
uint16_t new_value = 0xFF2A;
spi_register_write(SPI1_BASE + 0x20UL, new_value);
```

## See also

spi\_register\_read

- CHANGE HISTORY -

| Date | Software Version | Initials | Description |
|------|------------------|----------|-------------|
|------|------------------|----------|-------------|

#### 5.2.3.5 spi\_transfer()

#### **Description:**

Carries out a blocking spi transfer according to the specifications of the transfer parameter.

PRE-CONDITION: spi\_init() has been successfully carried out for the required spi channel PRE-CONDITION 

: gpio\_init() has been called for the slave select pin to configure it as an output/input, depending on desired direction 
PRE-CONDITION: The proper data buffers and lengths are non-null/non-zero

POST-CONDITION: The desired transfer has been successfully carried out

#### Returns

void

#### Example:

```
spi_transfer_t flash_transfer;
flash_transfer.channel = SPI_3;
flash_transfer.slave_pin = GPIO_C_3;
flash_transfer.ss_polarity = ACTIVE_LOW;
flash_transfer.tx_buffer = &data_out;
flash_transfer.tx_length = sizeof(data_out);
flash_transfer.rx_buffer = &dummy_data;
flash_transfer.rx_length = sizeof(data_out);
flash_transfer.data_format = SPI_DATA_8BIT;
flash_transfer.bit_format = MSB_FIRST;
flash_transfer.clock_polarity = ACTIVE_HIGH;
flash_transfer.clock_phase = SECOND_EDGE;
spi_transfer(&flash_transfer);
```

#### See also

```
spi_init
spi_transfer_it
```

- CHANGE HISTORY -

| Date Software Version | Initials | Description |
|-----------------------|----------|-------------|
|-----------------------|----------|-------------|

```
5.2.3.6 spi_transfer_it()
```

#### **Description:**

```
Sets up an interrupt based spi transfer according to the specifications of the transfer parameter.
```

PRE-CONDITION: spi\_init() has been successfully carried out for the required spi channel PRE-CONDITION 

: gpio\_init() has been called for the slave select pin to configure it as an output/input, depending on desired direction 
PRE-CONDITION: The proper data buffers and lengths are non-null/non-zero

POST-CONDITION: The irq handler will now handle the rest of the transfer

#### Returns

void

#### Example:

```
spi_transfer_t flash_transfer;
flash_transfer.channel = SPI_3;
flash_transfer.slave_pin = GPIO_C_3;
flash_transfer.ss_polarity = ACTIVE_LOW;
flash_transfer.tx_buffer = &data_out;
flash_transfer.tx_length = sizeof(data_out);
flash_transfer.rx_length = sizeof(data_out);
flash_transfer.rx_length = sizeof(data_out);
flash_transfer.data_format = SPI_DATA_8BIT;
flash_transfer.bit_format = MSB_FIRST;
flash_transfer.clock_polarity = ACTIVE_HIGH;
flash_transfer.clock_phase = SECOND_EDGE;
spi_transfer_it(&flash_transfer);
```

#### See also

```
spi_init
spi_transfer
spi_irq_handler
```

- CHANGE HISTORY -

| Date | Software Version | Initials | Description |
|------|------------------|----------|-------------|

#### 5.2.4 Variable Documentation

```
5.2.4.1 SPI_CR1
```

```
volatile uint16_t* const SPI_CR1[NUM_SPI] [static]
```

#### Initial value:

```
{
  (uint16_t *)SPI1_BASE, (uint16_t *)SPI2_BASE,
  (uint16_t *)SPI3_BASE, (uint16_t *)SPI4_BASE,
  (uint16_t *)SPI5_BASE
```

Array of pointers to Control Register 1 registers

```
5.2.4.2 SPI_CR2
```

```
volatile uint16_t* const SPI_CR2[NUM_SPI] [static]
```

#### Initial value:

```
= {
    (uint16_t *)SPI1_BASE + 0x04UL, (uint16_t *)SPI2_BASE + 0x04UL,
    (uint16_t *)SPI3_BASE + 0x04UL, (uint16_t *)SPI4_BASE + 0x04UL,
    (uint16_t *)SPI5_BASE + 0x04UL
```

Array of pointers to Control Register 2 registers

#### 5.2.4.3 SPI\_CRCPR

```
volatile uint16_t* const SPI_CRCPR[NUM_SPI] [static]
```

#### Initial value:

```
(uint16_t *)SPI1_BASE + 0x10UL, (uint16_t *)SPI2_BASE + 0x10UL,
  (uint16_t *)SPI3_BASE + 0x10UL, (uint16_t *)SPI4_BASE + 0x10UL,
  (uint16_t *)SPI5_BASE + 0x10UL
```

Array of pointers to CRC Polynomial registers

```
5.2.4.4 SPI_DR
```

```
volatile uint16_t* const SPI_DR[NUM_SPI] [static]
```

## Initial value:

```
(uint16_t *)SPI1_BASE + 0x0CUL, (uint16_t *)SPI2_BASE + 0x0CUL,
(uint16_t *)SPI3_BASE + 0x0CUL, (uint16_t *)SPI4_BASE + 0x0CUL,
(uint16_t *)SPI5_BASE + 0x0CUL
```

Array of pointers to Data registers

```
5.2.4.5 spi_interrupt_callbacks
```

```
spi_interrupt_callback_t spi_interrupt_callbacks[NUM_SPI] [static]
```

Static array of callback functions mapped to each spi device

#### 5.2.4.6 spi\_interrupt\_transfers

```
spi_transfer_t spi_interrupt_transfers[NUM_SPI] [static]
```

Static array which holds safe copies of transfers for interrupt routines, mapped to spi devices

#### 5.2.4.7 SPI RXCRCR

```
volatile uint16_t* const SPI_RXCRCR[NUM_SPI] [static]
```

#### Initial value:

```
{
    (uint16_t *)SPI1_BASE + 0x14UL, (uint16_t *)SPI2_BASE + 0x14UL,
     (uint16_t *)SPI3_BASE + 0x14UL, (uint16_t *)SPI4_BASE + 0x14UL,
     (uint16_t *)SPI5_BASE + 0x14UL
}
```

Array of pointers to reception CRC registers

## 5.2.4.8 SPI\_SR

```
volatile uint16_t* const SPI_SR[NUM_SPI] [static]
```

#### Initial value:

```
{
    (uint16_t *)SPI1_BASE + 0x08UL, (uint16_t *)SPI2_BASE + 0x08UL,
    (uint16_t *)SPI3_BASE + 0x08UL, (uint16_t *)SPI4_BASE + 0x08UL,
    (uint16_t *)SPI5_BASE + 0x08UL
}
```

Array of pointers to Status registers

## 5.2.4.9 SPI\_TXCRCR

```
volatile uint16_t* const SPI_TXCRCR[NUM_SPI] [static]
```

#### Initial value:

```
=
{
    (uint16_t *)SPI1_BASE + 0x18UL, (uint16_t *)SPI2_BASE + 0x18UL,
    (uint16_t *)SPI3_BASE + 0x18UL, (uint16_t *)SPI4_BASE + 0x18UL,
    (uint16_t *)SPI5_BASE + 0x18UL
```

Array of pointers to transmission CRC registers

# 5.3 /home/marko/Documents/embedded\_workspace/spi\_driver/spi\_stm32f411\_config.c File Reference

Contains the configuration information for each spi channel.

#include "spi\_stm32f411\_config.h"
Include dependency graph for spi\_stm32f411\_config.c:



## **Functions**

const spi\_config\_t \* spi\_config\_get (void)

## **Variables**

• static const spi\_config\_t config\_table [NUM\_SPI]

## 5.3.1 Detailed Description

Contains the configuration information for each spi channel.

## 5.3.2 Function Documentation

```
5.3.2.1 spi_config_get()
{\tt const} \ {\tt spi\_config\_t*} \ {\tt spi\_config\_get} \ (
                 void )
```

## **Description:**

Returns a pointer to the base of the configuration table for spi peripherals

PRE-CONDITION: The config table has been filled out and is non-null

Returns

```
*spi_config_t
```

## Example:

```
const spi_config_t *config_table = spi_config_get();
spi_init(config_table);
```

See also

spi init

- CHANGE HISTORY -

| Date Software Version | Initials | Description |
|-----------------------|----------|-------------|
|-----------------------|----------|-------------|

## 5.3.3 Variable Documentation

```
5.3.3.1 config_table
```

```
const spi_config_t config_table[NUM_SPI] [static]
```

## Initial value:

# 5.4 /home/marko/Documents/embedded\_workspace/spi\_driver/spi\_stm32f411\_config.h File Reference

Contains the definitions and structures required to configure the spi peripherals on an stm32f411.

This graph shows which files directly or indirectly include this file:



## **Data Structures**

• struct spi\_config\_t

## **Enumerations**

```
enum spi channel t {
 SPI 1 = 0x00UL, SPI 2 = 0x01UL, SPI 3 = 0x02UL, SPI 4 = 0x03UL,
 SPI_5 = 0x04UL, NUM_SPI = 0x05UL
• enum spi master slave t { SPI SLAVE, SPI MASTER }
enum spi baud rate t {
 PCLK DIV 2, PCLK DIV 4, PCLK DIV 8, PCLK DIV 16,
 PCLK_DIV_32, PCLK_DIV_64, PCLK_DIV_128, PCLK_DIV_256 }

    enum spi enable t { SPI_DISABLE, SPI_ENABLE }

    enum spi slave mgmt t { HARDWARE SMM, SOFTWARE SMM }

enum spi_bidir_t { UNIDIR_FULL_DUPLEX, UNIDIR_RXONLY, BIDIR_MODE }
• enum spi_crc_en_t { CRC_DISABLE, CRC_ENABLE }
enum spi_rx_dma_t { RX_DMA_REQ_DISABLE, RX_DMA_REQ_ENABLE }
• enum spi tx dma t { TX DMA REQ DISABLE, TX DMA REQ ENABLE }

    enum spi ssoe t { SS OUTPUT DISABLE, SS OUTPUT ENABLE }

enum spi_frame_format_t { SPI_MOTOROLA, SPI_TI }
enum spi_interrupt_t { ERROR_INTERRUPT, RXNE_INTERRUPT, TXE_INTERRUPT }
```

## **Functions**

const spi\_config\_t \* spi\_config\_get (void)

## 5.4.1 Detailed Description

Contains the definitions and structures required to configure the spi peripherals on an stm32f411.

## 5.4.2 Enumeration Type Documentation

```
5.4.2.1 spi_baud_rate_t
enum spi_baud_rate_t
```

Contains all of the prescaler options for the master clock generation

```
5.4.2.2 spi_bidir_t
enum spi_bidir_t
```

Contains options determining the topology of the bus system

## Enumerator

| UNIDIR_FULL_DUPLEX | Two data wire spi, with simultaneous data reception and transmission |
|--------------------|----------------------------------------------------------------------|
| UNIDIR_RXONLY      | Two data wire spi, but with reception only                           |
| BIDIR_MODE         | Single data wire spi, must select transfer direction upon call       |

```
5.4.2.3 spi_channel_t
```

enum spi\_channel\_t

Contains all of the spi devices found on chip

```
5.4.2.4 spi_crc_en_t
enum spi_crc_en_t
```

Contains options for enabling the hardware CRC calculation of received data

```
5.4.2.5 spi_enable_t
```

```
enum spi_enable_t
```

Spi devices which are disabled are ignored during init

```
5.4.2.6 spi_frame_format_t
```

```
enum spi_frame_format_t
```

Options for motorola's vs Tl's spi format. Motorola is the defualt and should suffice

```
5.4.2.7 spi_interrupt_t
```

```
enum spi_interrupt_t
```

Contains all the possible interrupts a spi device can handle

#### Enumerator

| ERROR_INTERRUPT | Interrupt generated upon a communication error             |
|-----------------|------------------------------------------------------------|
| RXNE_INTERRUPT  | Interrupt generated upon having non-read received data     |
| TXE_INTERRUPT   | Interrupt generated when no data is in the transfer buffer |

```
5.4.2.8 spi_master_slave_t
```

```
enum spi_master_slave_t
```

Contains the options for the spi device's functioning mode

## Enumerator

| SPI_SLAVE  | The spi is functioning as a slave and will await selection and a clock         |
|------------|--------------------------------------------------------------------------------|
| SPI_MASTER | The spi is functioning as a master and will select slaves and generate a clock |

5.4.2.9 spi\_rx\_dma\_t

enum spi\_rx\_dma\_t

Contains options to enable DMA requests upon data reception

```
5.4.2.10 spi_slave_mgmt_t
enum spi_slave_mgmt_t
```

Contains options to enable the self selection of the spi (as a slave) through software

```
5.4.2.11 spi_ssoe_t
enum spi_ssoe_t
```

Contains options for slave select output control. Allows work in multi-master mode when disabled

```
5.4.2.12 spi_tx_dma_t
enum spi_tx_dma_t
```

Contains options to enable DMA requests upon data transmission

## 5.4.3 Function Documentation

#### **Description:**

Returns a pointer to the base of the configuration table for spi peripherals

PRE-CONDITION: The config table has been filled out and is non-null

Returns

```
*spi_config_t
```

## Example:

```
const spi_config_t *config_table = spi_config_get();
spi_init(config_table);
```

See also

```
spi init
```

- CHANGE HISTORY -