blob: 884985d51cd5ab58c260d1f43e02b7a9bcb0c916 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
|
/**
*
* Author: Dylan Muller
* Copyright (c) 2025
* All rights reserved.
*
* - Commercial/IP use prohibited.
* - Attribution required.
* See License.txt
*
*/
#include "setup.h"
#include "peripheral/spi.h"
#include <stdio.h>
#include <avr/io.h>
#include <avr/interrupt.h>
#include <util/delay.h>
#if SPI_MODE == 0
#define CPOL_VALUE 0
#define CPHA_VALUE 0
#elif SPI_MODE == 1
#define CPOL_VALUE 0
#define CPHA_VALUE 1
#elif SPI_MODE == 2
#define CPOL_VALUE 1
#define CPHA_VALUE 0
#elif SPI_MODE == 3
#define CPOL_VALUE 1
#define CPHA_VALUE 1
#else
#error "No valid SPI_MODE defined!"
#endif
#if SPI_DORD == 0
#define DORD_VALUE 0
#elif SPI_DORD == 1
#define DORD_VALUE 1
#else
#error "No valid SPI_DORD defined!"
#endif
static volatile uint8_t *spi_out;
static volatile uint8_t *spi_in;
static volatile size_t spi_len;
static volatile uint8_t *spi_port;
static volatile uint8_t spi_pin;
void spi_init(void)
{
DDRB |= (1 << DDB2) | (1 << DDB3) | (1 << DDB5);
SPSR |= (SPI2X_VALUE << SPI2X);
SPCR |= (1 << SPIE) | (1 << SPE) | (DORD_VALUE << DORD) | (1 << MSTR)
| (CPOL_VALUE << CPOL) | (CPHA_VALUE << CPHA)
| (SPR1_VALUE << SPR1) | (SPR0_VALUE << SPR0);
}
bool spi_busy(void)
{
return spi_len;
}
void spi_flush(void)
{
while(spi_len);
}
void spi_start(
uint8_t *out,
uint8_t *in,
size_t len,
uint8_t *port,
uint8_t pin
)
{
spi_flush();
spi_out = out;
spi_in = in;
spi_len = len;
spi_port = port;
spi_pin = pin;
if(spi_port)
*spi_port &= ~(1 << spi_pin);
SPDR = *spi_out++;
}
ISR(SPI_STC_vect)
{
if(spi_in)
{
*spi_in++ = SPDR;
}
if(--spi_len)
SPDR = *spi_out++;
else
if(spi_port)
*spi_port |= (1 << spi_pin);
}
|