219 lines
6.6 KiB
C
219 lines
6.6 KiB
C
|
/*
|
||
|
* Copyright (c) 2006-2022, RT-Thread Development Team
|
||
|
*
|
||
|
* SPDX-License-Identifier: Apache-2.0
|
||
|
*
|
||
|
* Change Logs:
|
||
|
* Date Author Notes
|
||
|
* 2022-07-30 Emuzit first version
|
||
|
*/
|
||
|
#ifndef __CH56X_SPI_H__
|
||
|
#define __CH56X_SPI_H__
|
||
|
|
||
|
#include "soc.h"
|
||
|
#include "ch56x_gpio.h"
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
|
||
|
#define SPI0_BUS_NAME "spi0"
|
||
|
#define SPI1_BUS_NAME "spi1"
|
||
|
|
||
|
#ifndef SPI0_SCS_PIN
|
||
|
#define SPI0_SCS_PIN GET_PIN(A, 12)
|
||
|
#endif
|
||
|
#define SPI0_SCK_PIN GET_PIN(A, 13)
|
||
|
#define SPI0_MOSI_PIN GET_PIN(A, 14)
|
||
|
#define SPI0_MISO_PIN GET_PIN(A, 15)
|
||
|
|
||
|
#ifdef SOC_SERIES_CH569
|
||
|
#ifndef SPI1_SCS_PIN
|
||
|
#define SPI1_SCS_PIN GET_PIN(B, 11)
|
||
|
#endif
|
||
|
#define SPI1_SCK_PIN GET_PIN(B, 12)
|
||
|
#define SPI1_MOSI_PIN GET_PIN(B, 13)
|
||
|
#define SPI1_MISO_PIN GET_PIN(B, 14)
|
||
|
|
||
|
#else
|
||
|
#define SPI1_SCK_PIN GET_PIN(A, 0)
|
||
|
#define SPI1_MOSI_PIN GET_PIN(A, 1)
|
||
|
#define SPI1_MISO_PIN GET_PIN(A, 2)
|
||
|
#endif
|
||
|
|
||
|
#define SPI_FIFO_SIZE 8
|
||
|
|
||
|
union _spi_ctrl_mod
|
||
|
{
|
||
|
uint8_t reg;
|
||
|
struct
|
||
|
{
|
||
|
uint8_t mode_slave : 1; // RW, SPI master/slave (0/1) mode select
|
||
|
uint8_t all_clear : 1; // RW, clear FIFO/count/int-flag
|
||
|
uint8_t two_wire : 1; // RW, 2/3-wire mode (0/1), SPI slave
|
||
|
uint8_t mst_sck_mod : 1; // RW, mode0/mode3 (0/1) for SCK idle L/H
|
||
|
uint8_t fifo_dir : 1; // RW, FIFO direction is output/input (0/1)
|
||
|
uint8_t sck_oe : 1; // RW, SCK pin output enable
|
||
|
uint8_t mosi_oe : 1; // RW, MOSI pin output enable
|
||
|
uint8_t miso_oe : 1; // RW, MISO pin output enable
|
||
|
};
|
||
|
struct
|
||
|
{
|
||
|
uint8_t stuff_0 : 3;
|
||
|
uint8_t slv_cmd_mod : 1; // RW, 1st byte is data/cmd (0/1), SPI slave
|
||
|
uint8_t stuff_4 : 4;
|
||
|
};
|
||
|
};
|
||
|
#define RB_SPI_MODE_SLAVE 0x01
|
||
|
#define RB_SPI_ALL_CLEAR 0x02
|
||
|
#define RB_SPI_2WIRE_MOD 0x04
|
||
|
#define RB_SPI_MST_SCK_MOD 0x08
|
||
|
#define RB_SPI_SLV_CMD_MOD 0x08
|
||
|
#define RB_SPI_FIFO_DIR 0x10
|
||
|
#define RB_SPI_SCK_OE 0x20
|
||
|
#define RB_SPI_MOSI_OE 0x40
|
||
|
#define RB_SPI_MISO_OE 0x80
|
||
|
|
||
|
#define MST_SCK_MOD_0 0
|
||
|
#define MST_SCK_MOD_3 1
|
||
|
|
||
|
#define SPI_FIFO_DIR_OUTPUT 0
|
||
|
#define SPI_FIFO_DIR_INPUT 1
|
||
|
|
||
|
union _spi_ctrl_cfg
|
||
|
{
|
||
|
uint8_t reg;
|
||
|
struct
|
||
|
{
|
||
|
uint8_t dma_enable : 1; // RW, enable DMA function
|
||
|
uint8_t resv_1 : 1;
|
||
|
uint8_t dma_loop : 1; // RW, enable DMA loop mode (0 => single)
|
||
|
uint8_t resv_3 : 1;
|
||
|
uint8_t auto_if : 1; // RW, enable auto clear RB_SPI_IF_BYTE_END
|
||
|
uint8_t bit_order : 1; // RW, data bit ordering, LSB/MSB first (0/1)
|
||
|
uint8_t resv_6 : 2;
|
||
|
};
|
||
|
};
|
||
|
#define RB_SPI_DMA_ENABLE 0x01
|
||
|
#define RB_SPI_DMA_LOOP 0x04
|
||
|
#define RB_SPI_AUTO_IF 0x10
|
||
|
#define RB_SPI_BIT_ORDER 0x20
|
||
|
|
||
|
#define SPI_BIT_ORDER_MSB 0
|
||
|
#define SPI_BIT_ORDER_LSB 1
|
||
|
|
||
|
union _spi_inter_en
|
||
|
{
|
||
|
uint8_t reg;
|
||
|
struct
|
||
|
{
|
||
|
uint8_t cnt_end : 1; // RW, IE for all bytes transfered
|
||
|
uint8_t byte_end : 1; // RW, IE for single byte transfered
|
||
|
uint8_t fifo_hf : 1; // RW, IE for FIFO half full
|
||
|
uint8_t dma_end : 1; // RW, IE for end of DMA
|
||
|
uint8_t fifo_ov : 1; // RW, IE for FIFO full or empty
|
||
|
uint8_t resv_5 : 2;
|
||
|
uint8_t fst_byte : 1; // RW, IE for 1st byte received, SPI slave
|
||
|
};
|
||
|
};
|
||
|
#define RB_SPI_IE_CNT_END 0x01
|
||
|
#define RB_SPI_IE_BYTE_END 0x02
|
||
|
#define RB_SPI_IE_FIFO_HF 0x04
|
||
|
#define RB_SPI_IE_DMA_END 0x08
|
||
|
#define RB_SPI_IE_FIFO_OV 0x10
|
||
|
#define RB_SPI_IE_FST_BYTE 0x80
|
||
|
|
||
|
union _spi_run_flag
|
||
|
{
|
||
|
uint8_t reg;
|
||
|
struct
|
||
|
{
|
||
|
uint8_t resv_0 : 4;
|
||
|
uint8_t slv_cmd_act : 1; // RO, SPI slave cmd received
|
||
|
uint8_t fifo_ready : 1; // RO, SPI FIFO ready to transfer
|
||
|
uint8_t slv_cs_load : 1; // RO, SPI slave is loading R8_SPIx_SLAVE_PRE
|
||
|
uint8_t slv_select : 1; // RO, SPI slave CS active (selected)
|
||
|
};
|
||
|
};
|
||
|
#define RB_SPI_SLV_CMD_ACT 0x10
|
||
|
#define RB_SPI_FIFO_READY 0x20
|
||
|
#define RB_SPI_SLV_CS_LOAD 0x40
|
||
|
#define RB_SPI_SLV_SELECT 0x80
|
||
|
|
||
|
union _spi_int_flag
|
||
|
{
|
||
|
uint8_t reg;
|
||
|
struct
|
||
|
{
|
||
|
uint8_t cnt_end : 1; // RW1, IF for all bytes transfered
|
||
|
uint8_t byte_end : 1; // RW1, IF for single byte transfered
|
||
|
uint8_t fifo_hf : 1; // RW1, IF for FIFO half full
|
||
|
uint8_t dma_end : 1; // RW1, IF for end of DMA
|
||
|
uint8_t fifo_ov : 1; // RW1, IF for FIFO full or empty
|
||
|
uint8_t resv_5 : 1;
|
||
|
uint8_t free : 1; // RO, current SPI state is free
|
||
|
uint8_t fst_byte : 1; // RW1, IF for 1st byte received, SPI slave
|
||
|
};
|
||
|
};
|
||
|
#define RB_SPI_IF_CNT_END 0x01
|
||
|
#define RB_SPI_IF_BYTE_END 0x02
|
||
|
#define RB_SPI_IF_FIFO_HF 0x04
|
||
|
#define RB_SPI_IF_DMA_END 0x08
|
||
|
#define RB_SPI_IF_FIFO_OV 0x10
|
||
|
#define RB_SPI_FREE 0x40
|
||
|
#define RB_SPI_IF_FST_BYTE 0x80
|
||
|
|
||
|
/*
|
||
|
* 0x00 R8_SPIx_CTRL_MOD: SPI mode setting register
|
||
|
* 0x01 R8_SPIx_CTRL_CFG: SPI configuration register
|
||
|
* 0x02 R8_SPIx_INTER_EN: SPI interrupt enable register
|
||
|
* 0x03 R8_SPIx_CLOCK_DIV: SPI master clock divisor, minimum 2
|
||
|
* 0x03 R8_SPIx_SLAVE_PRE: SPI slave preset data (reset as 10h)
|
||
|
* 0x04 R8_SPIx_BUFFER: SPI data buffer
|
||
|
* 0x05 R8_SPIx_RUN_FLAG: SPI working state register
|
||
|
* 0x06 R8_SPIx_INT_FLAG: SPI interrupt flags register
|
||
|
* 0x07 R8_SPIx_FIFO_COUNT: SPI FIFO data count
|
||
|
* 0x0c R16_SPIx_TOTAL_CNT: SPI total data length to transfer
|
||
|
* 0x10 R8_SPIx_FIFO: SPI FIFO
|
||
|
* 0x13 R8_SPIx_FIFO_COUNT1: SPI FIFO data count
|
||
|
* 0x14 R32_SPIx_DMA_NOW: SPI DMA current address
|
||
|
* 0x18 R32_SPIx_DMA_BEG: SPI DMA start address
|
||
|
* 0x1c R32_SPIx_DMA_END: SPI DMA end address
|
||
|
*
|
||
|
* CAVEAT: gcc (as of 8.2.0) tends to read 32-bit word for bit field test.
|
||
|
* Be careful for those with side effect for read (e.g. RBR, IIR).
|
||
|
*/
|
||
|
struct spi_registers
|
||
|
{
|
||
|
union _spi_ctrl_mod CTRL_MOD;
|
||
|
union _spi_ctrl_cfg CTRL_CFG;
|
||
|
union _spi_inter_en INTER_EN;
|
||
|
union
|
||
|
{
|
||
|
uint8_t CLOCK_DIV;
|
||
|
uint8_t SLAVE_PRE;
|
||
|
};
|
||
|
uint8_t BUFFER;
|
||
|
union _spi_run_flag RUN_FLAG;
|
||
|
union _spi_int_flag INT_FLAG;
|
||
|
uint8_t FIFO_COUNT;
|
||
|
uint32_t resv_8;
|
||
|
uint16_t TOTAL_COUNT;
|
||
|
uint16_t resv_0e;
|
||
|
uint8_t FIFO;
|
||
|
uint8_t resv_11[2];
|
||
|
uint8_t FIFO_COUNT1;
|
||
|
uint32_t DMA_NOW;
|
||
|
uint32_t DMA_BIG;
|
||
|
uint32_t DMA_END;
|
||
|
};
|
||
|
CHECK_STRUCT_SIZE(struct spi_registers, 0x20);
|
||
|
|
||
|
rt_err_t spi_cs_pin_assign(int spi_n, rt_base_t cs_pin);
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#endif
|