diff options
| author | Bdale Garbee <bdale@gag.com> | 2012-08-28 23:39:53 -0600 | 
|---|---|---|
| committer | Bdale Garbee <bdale@gag.com> | 2012-08-28 23:39:53 -0600 | 
| commit | 5ed88fb72c3e3ecf3333c700d838667db71cfbdc (patch) | |
| tree | 3b371f563c0f7607f2fe53242673adb352b48514 /src/cc1111/ao_spi.c | |
| parent | adbe64c5a9402b7c5075a444a12629131b663877 (diff) | |
| parent | 621d0930244f25165d2ac5da596dcc87e253b965 (diff) | |
Merge branch 'master' of ssh://git.gag.com/scm/git/fw/altos
Conflicts:
	debian/control
Diffstat (limited to 'src/cc1111/ao_spi.c')
| -rw-r--r-- | src/cc1111/ao_spi.c | 172 | 
1 files changed, 141 insertions, 31 deletions
| diff --git a/src/cc1111/ao_spi.c b/src/cc1111/ao_spi.c index 1bf5e155..2b4fd186 100644 --- a/src/cc1111/ao_spi.c +++ b/src/cc1111/ao_spi.c @@ -17,6 +17,109 @@  #include "ao.h" +/* Default pin usage for existing Altus Metrum devices */ +#if !HAS_SPI_0 && !HAS_SPI_1 +#define HAS_SPI_0	1 +#define SPI_0_ALT_2	1 +#endif + +#ifndef SPI_CONST +#define SPI_CONST	0xff +#endif + +/* + * USART0 SPI config alt 1 + *  + *	MO	P0_3 + *	MI	P0_2 + *	CLK	P0_5 + *	SS	P0_4 + * + * USART0 SPI config alt 2 + * + * 	MO	P1_5 + * 	MI	P1_4 + * 	CLK	P1_3 + *	CSS	P1_2 + * + * USART1 SPI config alt 1 + * + *	MO	P0_4 + *	MI	P0_5 + *	CLK	P0_3 + *	SS	P0_2 + * + * USART1 SPI config alt 2 + * + *	MO	P1_6 + *	MI	P1_7 + *	CLK	P1_5 + *	SS	P1_4 + * + * + * Chip select is the responsibility of the caller in master mode + */ + +#if HAS_SPI_0 +#define SPI_CSR		U0CSR +#define SPI_BUF		U0DBUFXADDR +#define SPI_BAUD	U0BAUD +#define SPI_GCR		U0GCR +#define SPI_CFG_MASK	PERCFG_U0CFG_ALT_MASK +#define SPI_DMA_TX	DMA_CFG0_TRIGGER_UTX0 +#define SPI_DMA_RX	DMA_CFG0_TRIGGER_URX0 + +#if SPI_0_ALT_1 +#define SPI_CFG		PERCFG_U0CFG_ALT_1 +#define SPI_SEL		P0SEL +#define SPI_BITS	(1 << 3) | (1 << 2) | (1 << 5) +#define SPI_CSS_BIT	(1 << 4) +#endif + +#if SPI_0_ALT_2 +#define SPI_CFG		PERCFG_U0CFG_ALT_2 +#define SPI_SEL		P1SEL +#define SPI_PRI		P2SEL_PRI3P1_USART0 +#define SPI_BITS	(1 << 5) | (1 << 4) | (1 << 3) +#define SPI_CSS_BIT	(1 << 2) +#endif + +#endif + +#if HAS_SPI_1 +#define SPI_CSR		U1CSR +#define SPI_BUF		U1DBUFXADDR +#define SPI_BAUD	U1BAUD +#define SPI_GCR		U1GCR +#define SPI_CFG_MASK	PERCFG_U1CFG_ALT_MASK +#define SPI_DMA_TX	DMA_CFG0_TRIGGER_UTX1 +#define SPI_DMA_RX	DMA_CFG0_TRIGGER_URX1 + +#if SPI_1_ALT_1 +#define SPI_CFG		PERCFG_U1CFG_ALT_1 +#define SPI_SEL		P0SEL +#define SPI_BITS	(1 << 4) | (1 << 5) | (1 << 3) +#define SPI_CSS_BIT	(1 << 2) +#endif + +#if SPI_1_ALT_2 +#define SPI_CFG		PERCFG_U1CFG_ALT_2 +#define SPI_SEL		P1SEL +#define SPI_PRI		P2SEL_PRI3P1_USART1 +#define SPI_BITS	(1 << 6) | (1 << 7) | (1 << 5) +#define SPI_CSS_BIT	(1 << 4) +#endif + +#endif + +#if AO_SPI_SLAVE +#define CSS		SPI_CSS_BIT +#define UxCSR_DIRECTION	UxCSR_SLAVE +#else +#define CSS		0 +#define UxCSR_DIRECTION	UxCSR_MASTER +#endif +  /* Shared mutex to protect SPI bus, must cover the entire   * operation, from CS low to CS high. This means that any SPI   * user must protect the SPI bus with this mutex @@ -40,35 +143,45 @@ static __xdata uint8_t ao_spi_const;  void  ao_spi_send_bus(void __xdata *block, uint16_t len) __reentrant  { +#if !AO_SPI_SLAVE  	ao_dma_set_transfer(ao_spi_dma_in_id, -			    &U0DBUFXADDR, +			    &SPI_BUF,  			    &ao_spi_const,  			    len,  			    DMA_CFG0_WORDSIZE_8 |  			    DMA_CFG0_TMODE_SINGLE | -			    DMA_CFG0_TRIGGER_URX0, +			    SPI_DMA_RX,  			    DMA_CFG1_SRCINC_0 |  			    DMA_CFG1_DESTINC_0 |  			    DMA_CFG1_PRIORITY_NORMAL); - +#endif  	ao_dma_set_transfer(ao_spi_dma_out_id,  			    block, -			    &U0DBUFXADDR, +			    &SPI_BUF,  			    len,  			    DMA_CFG0_WORDSIZE_8 |  			    DMA_CFG0_TMODE_SINGLE | -			    DMA_CFG0_TRIGGER_UTX0, +			    SPI_DMA_TX,  			    DMA_CFG1_SRCINC_1 |  			    DMA_CFG1_DESTINC_0 |  			    DMA_CFG1_PRIORITY_NORMAL); +#if !AO_SPI_SLAVE  	ao_dma_start(ao_spi_dma_in_id); +#endif  	ao_dma_start(ao_spi_dma_out_id);  	ao_dma_trigger(ao_spi_dma_out_id); +#if AO_SPI_SLAVE +	__critical while (!ao_spi_dma_out_done) +			   ao_sleep(&ao_spi_dma_out_done); +#else  	__critical while (!ao_spi_dma_in_done)  		ao_sleep(&ao_spi_dma_in_done); +#endif  } + +  /* Receive bytes over SPI.   *   * This sets up tow DMA engines, one reading the data and another @@ -79,59 +192,56 @@ void  ao_spi_recv_bus(void __xdata *block, uint16_t len) __reentrant  {  	ao_dma_set_transfer(ao_spi_dma_in_id, -			    &U0DBUFXADDR, +			    &SPI_BUF,  			    block,  			    len,  			    DMA_CFG0_WORDSIZE_8 |  			    DMA_CFG0_TMODE_SINGLE | -			    DMA_CFG0_TRIGGER_URX0, +			    SPI_DMA_RX,  			    DMA_CFG1_SRCINC_0 |  			    DMA_CFG1_DESTINC_1 |  			    DMA_CFG1_PRIORITY_NORMAL); -	ao_spi_const = 0xff; +	ao_spi_const = SPI_CONST; +#if !AO_SPI_SLAVE  	ao_dma_set_transfer(ao_spi_dma_out_id,  			    &ao_spi_const, -			    &U0DBUFXADDR, +			    &SPI_BUF,  			    len,  			    DMA_CFG0_WORDSIZE_8 |  			    DMA_CFG0_TMODE_SINGLE | -			    DMA_CFG0_TRIGGER_UTX0, +			    SPI_DMA_TX,  			    DMA_CFG1_SRCINC_0 |  			    DMA_CFG1_DESTINC_0 |  			    DMA_CFG1_PRIORITY_NORMAL); +#endif  	ao_dma_start(ao_spi_dma_in_id); +#if !AO_SPI_SLAVE  	ao_dma_start(ao_spi_dma_out_id);  	ao_dma_trigger(ao_spi_dma_out_id); +#endif  	__critical while (!ao_spi_dma_in_done)  		ao_sleep(&ao_spi_dma_in_done);  } -/* - * Initialize USART0 for SPI using config alt 2 - * - * 	MO	P1_5 - * 	MI	P1_4 - * 	CLK	P1_3 - * - * Chip select is the responsibility of the caller - */  void  ao_spi_init(void)  {  	/* Set up the USART pin assignment */ -	PERCFG = (PERCFG & ~PERCFG_U0CFG_ALT_MASK) | PERCFG_U0CFG_ALT_2; +	PERCFG = (PERCFG & ~SPI_CFG_MASK) | SPI_CFG; -	/* Ensure that USART0 takes precidence over USART1 for pins that -	 * they share +	/* Ensure that SPI USART takes precidence over the other USART +	 * for pins that they share  	 */ -	P2SEL = (P2SEL & ~P2SEL_PRI3P1_MASK) | P2SEL_PRI3P1_USART0; +#ifdef SPI_PRI +	P2SEL = (P2SEL & ~P2SEL_PRI3P1_MASK) | SPI_PRI; +#endif  	/* Make the SPI pins be controlled by the USART peripheral */ -	P1SEL |= ((1 << 5) | (1 << 4) | (1 << 3)); +	SPI_SEL |= SPI_BITS | CSS;  	/* Set up OUT DMA */  	ao_spi_dma_out_id = ao_dma_alloc(&ao_spi_dma_out_done); @@ -141,9 +251,9 @@ ao_spi_init(void)  	/* Set up the USART.  	 * -	 * SPI master mode +	 * SPI master/slave mode  	 */ -	U0CSR = (UxCSR_MODE_SPI | UxCSR_RE | UxCSR_MASTER); +	SPI_CSR = (UxCSR_MODE_SPI | UxCSR_RE | UxCSR_DIRECTION);  	/* Set the baud rate and signal parameters  	 * @@ -151,9 +261,9 @@ ao_spi_init(void)  	 * Every peripheral I've ever seen goes faster than that,  	 * so set the clock to 3MHz (BAUD_E 17, BAUD_M 0)  	 */ -	U0BAUD = 0; -	U0GCR = (UxGCR_CPOL_NEGATIVE | -		 UxGCR_CPHA_FIRST_EDGE | -		 UxGCR_ORDER_MSB | -		 (17 << UxGCR_BAUD_E_SHIFT)); +	SPI_BAUD = 0; +	SPI_GCR = (UxGCR_CPOL_NEGATIVE | +		   UxGCR_CPHA_FIRST_EDGE | +		   UxGCR_ORDER_MSB | +		   (17 << UxGCR_BAUD_E_SHIFT));  } | 
