HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
spi.c
Go to the documentation of this file.
1 
17 #include "user_config.h"
18 #include <stdlib.h>
19 #include "spi.h"
20 
22 static uint32_t SPI0_Speed_value = 0;
23 
46 void SPI0_Speed(uint32_t speed)
47 {
48  uint32_t rate;
49 
50 // Computer Prescale rate - truncate.
51 
52 #if 1
53 // We only make changes if the speed actually changes
54  if(speed == SPI0_Speed_value)
55  return;
56 #endif
57 
58  // rate = ((uint32_t) F_CPU) / speed;
59  // Round
60  rate = ((uint32_t) F_CPU + (speed/2)) / speed;
61  printf("spi rate: %ld\n", (unsigned long) rate);
62 
63 
64  if(rate >= 128)
65  {
66 // 128
67  BIT_CLR(SPSR, SPI2X);
68  BIT_SET(SPCR, SPR1);
69  BIT_SET(SPCR, SPR0);
70  }
71  else if(rate >= 64)
72  {
73 // 64
74  BIT_CLR(SPSR, SPI2X);
75  BIT_SET(SPCR, SPR1);
76  BIT_SET(SPCR, SPR0);
77  }
78  else if(rate >= 32)
79  {
80 // 32
81  BIT_SET(SPSR, SPI2X);
82  BIT_SET(SPCR, SPR1);
83  BIT_CLR(SPCR, SPR0);
84  }
85  else if(rate >= 16)
86  {
87 // 16
88  BIT_CLR(SPSR, SPI2X);
89  BIT_CLR(SPCR, SPR1);
90  BIT_SET(SPCR, SPR0);
91  }
92  else if(rate >= 8)
93  {
94 // 8;
95  BIT_SET(SPSR, SPI2X);
96  BIT_CLR(SPCR, SPR1);
97  BIT_SET(SPCR, SPR0);
98  }
99  else if(rate >= 4)
100  {
101 // 4;
102  BIT_CLR(SPSR, SPI2X);
103  BIT_CLR(SPCR, SPR1);
104  BIT_CLR(SPCR, SPR0);
105  }
106  else if(rate >= 2)
107  {
108 // 2;
109  BIT_SET(SPSR, SPI2X);
110  BIT_CLR(SPCR, SPR1);
111  BIT_CLR(SPCR, SPR0);
112  }
113  else
114  {
115 // 2 fastest rate;
116  BIT_SET(SPSR, SPI2X);
117  BIT_CLR(SPCR, SPR1);
118  BIT_CLR(SPCR, SPR0);
119  }
121  SPI0_Speed_value = speed;
122 }
123 
124 
128 uint32_t SPI0_Get_Speed( void )
129 {
130  return(SPI0_Speed_value);
131 }
132 
133 
136 static int SPI0_Mode_value = 0;
137 
150 void SPI0_Mode(int mode)
151 {
152  switch(mode)
153  {
154  case 0:
155  BIT_CLR(SPCR,CPOL);
156  BIT_CLR(SPCR,CPHA);
157  break;
158  case 1:
159  BIT_CLR(SPCR,CPOL);
160  BIT_SET(SPCR,CPHA);
161  break;
162  case 2:
163  BIT_SET(SPCR,CPOL);
164  BIT_CLR(SPCR,CPHA);
165  break;
166  case 3:
167  BIT_SET(SPCR,CPOL);
168  BIT_SET(SPCR,CPHA);
169  break;
170  default:
171  printf("SPI0_Mode: Invalid mode(%d)\n",mode);
172  break;
173  }
174  SPI0_Mode_value = mode;
175 }
176 
177 
180 int SPI0_Get_Mode( void )
181 {
182  return(SPI0_Mode_value);
183 }
184 
185 
187 static int SPI0_Init_state = 0;
188 
192 void SPI0_Init(uint32_t speed)
193 {
194 
195  volatile uint8_t D __attribute__((unused));
196 
197 #ifdef SPI_DEBUG
198  printf("SPI0_Init speed:%ld\n",speed);
199 #endif
200 
201 #ifdef SPI_DEBUG
202  printf("SS HI\n");
203  printf("Port B DDR: 0x%02x\n", (int) GPIO_PORT_DDR_RD(GPIO_B));
204  printf("Port B LATCH: 0x%02x\n", (int) GPIO_PORT_LATCH_RD(GPIO_B));
205  printf("Port B PINS: 0x%02x\n", (int) GPIO_PORT_LATCH_RD(GPIO_B));
206 #endif
207 
208  SPCR = 0; // Clear SPCR in case we are not called after RESET
209 
210  GPIO_PIN_HI(SCK); // SCK Output
211  GPIO_PIN_HI(MOSI); // MOSI Output
212  GPIO_PIN_FLOAT(MISO); // MISO Input, no pull-up
213 
214 // Warning *** MSTR MUST be set BEFORE SPE!!!! *** otherwise SS will NOT behave as an output
215  GPIO_PIN_LOW(SS); // SS Output must be HI prevent slave mode from getting set while initializing
216  BIT_SET(SPCR, MSTR); // Master Mode
217  BIT_SET(SPCR, SPE); // Enable SPI
218 
219 // SPI Clear any pending interrupt flags
220  D = SPSR;
221  D = SPDR;
222 
226  SPI0_Mode(0);
227 
228 // Set SPI clock speed
229  SPI0_Speed(speed);
230 
231  SPI0_TXRX_Byte(0xff); // Send dummy 0xFF
232  SPI0_Init_state = 1;
233 
234 }
235 
236 
245 uint8_t SPI0_TXRX_Byte(uint8_t Data)
246 {
247  SPDR = Data; // Transmit
248 
249  while( !BIT_TST(SPSR,SPIF) ) // Wait for send
250  ;
251 
252  return (SPDR); // Received Data
253 }
254 
255 
256 // =============================================
259 
264 void SPI0_TX(uint8_t *data, int count)
265 {
266 
267  while(count > 0)
268  {
269  SPI0_TXRX_Byte(*data);
270  ++data;
271  --count;
272  }
273 }
274 
275 
280 
281 void SPI0_TXRX(uint8_t *data, int count)
282 {
283 
284  while(count > 0)
285  {
286  *data = SPI0_TXRX_Byte(*data);
287  ++data;
288  --count;
289  }
290 }
291 
292 
297 void SPI0_RX(uint8_t *data, int count)
298 {
299 
300  while(count > 0)
301  {
302  *data = SPI0_TXRX_Byte(0xff);
303  ++data;
304  --count;
305  }
306 }
printf
MEMSPACE int printf(const char *format,...)
spi.h
SPI Driver AVR8.
SPI0_Init_state
static int SPI0_Init_state
SPI0 initialization flag.
Definition: spi.c:187
SPI0_Mode_value
static int SPI0_Mode_value
Saved SPI Mode.
Definition: spi.c:136
SPI0_Speed_value
static uint32_t SPI0_Speed_value
Saved SPI bus speed.
Definition: spi.c:22
SPI0_Init
void SPI0_Init(uint32_t speed)
Initialize SPI0 device. See Atmel App Note AVR151 Set default speed, IO pins and mode.
Definition: spi.c:192
SPI0_TXRX_Byte
uint8_t SPI0_TXRX_Byte(uint8_t Data)
SPI read/Write byte.
Definition: spi.c:245
BIT_SET
#define BIT_SET(x, y)
Note: IF x and y are constants the compiler will fully reduce the expression.
Definition: bits.h:17
BIT_CLR
#define BIT_CLR(x, y)
Definition: bits.h:18
user_config.h
Master Include for FatFs, RTC, Timers AVR8 - Part of HP85 disk emulator.
BIT_TST
#define BIT_TST(x, y)
Definition: bits.h:19
SPI0_RX
void SPI0_RX(uint8_t *data, int count)
HSPI read using FIFO.
Definition: spi.c:297
SPI0_TX
void SPI0_TX(uint8_t *data, int count)
SPI buffered write functions.
Definition: spi.c:264
SPI0_Mode
void SPI0_Mode(int mode)
Set SPI clock mode.
Definition: spi.c:150
SS
#define SS(fs)
Definition: ff.c:266
SPI0_Get_Speed
uint32_t SPI0_Get_Speed(void)
Return previously saved SPI BUS rate in HZ.
Definition: spi.c:128
GPIO_B
#define GPIO_B
Definition: gpio-1284p.h:19
SPI0_Get_Mode
int SPI0_Get_Mode(void)
Return saved SPI mode as set by SPI0_Mode().
Definition: spi.c:180
SPI0_Speed
void SPI0_Speed(uint32_t speed)
Set AVR SPI bus rate in HZ.
Definition: spi.c:46
SPE
#define SPE
Definition: amigo.h:32
SPI0_TXRX
void SPI0_TXRX(uint8_t *data, int count)
HSPI write and read using FIFO.
Definition: spi.c:281