-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsoi2c.c
168 lines (138 loc) · 5.12 KB
/
soi2c.c
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
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
// Copyright 2024 Blues Inc. All rights reserved.
// Use of this source code is governed by licenses granted by the
// copyright holder including that found in the LICENSE file.
#include "soi2c.h"
// Reset the state4 of things by sending a \n to flush anything pending
// on the i2c peripheral from before this host was reset. This ensures that
// our first transaction will be received cleanly.
int soi2cReset(soi2cContext_t *ctx)
{
uint8_t resetReq[25];
resetReq[0] = '\n';
return soi2cTransaction(ctx, SOI2C_IGNORE_RESPONSE, resetReq, sizeof(resetReq));
}
// Get buffer info
uint32_t soi2cBuf(soi2cContext_t *ctx, uint8_t **buf, uint32_t *buflen)
{
if (buf != NULL) {
*buf = ctx->buf;
}
if (buflen != NULL) {
*buflen = ctx->buflen;
}
return ctx->bufused;
}
// Perform a transaction. Note that the transmit buffer will be turned to
// garbage, as it will be used as an I/O buffer for both TX and RX operations.
// The request within the input buf must always be terminated with \n, and the
// buflen on input should be the current full allocated size of that buf.
int soi2cTransaction(soi2cContext_t *ctx, uint32_t flags, uint8_t *buf, uint32_t buflen)
{
// Default i2c address to the notecard
if (ctx->addr == 0) {
ctx->addr = SOI2C_DEFAULT_I2C_ADDR;
}
// Exit if not configured
if (ctx->tx == NULL || ctx->rx == NULL || ctx->delay == NULL || buflen < 5) {
return STATUS_CONFIG;
}
// Exit if request isn't newline-terminated
ctx->buf = buf;
ctx->buflen = buflen;
ctx->bufused = 0;
for (int i=0; i<buflen; i++) {
ctx->bufused++;
if (buf[i] == '\n') {
break;
}
}
if (ctx->bufused == 0) {
return STATUS_TERMINATOR;
}
// Begin by shifting the req in the buf to allow space for the transmit header
if ((ctx->buflen - ctx->bufused) < 1) {
return STATUS_TX_BUFFER_OVERFLOW;
}
memmove(&ctx->buf[1], ctx->buf, ctx->bufused);
// Loop, transmitting at most 250 bytes per chunk every 250 milliseconds
uint32_t left = ctx->bufused;
while (left) {
uint8_t chunklen = 250;
if (left < chunklen) {
chunklen = (uint8_t) left;
}
ctx->buf[0] = chunklen;
if (!ctx->tx(ctx->port, ctx->addr, ctx->buf, 1+chunklen)) {
return STATUS_IO_TRANSMIT;
}
ctx->delay(250);
left -= chunklen;
memmove(&ctx->buf[1], &ctx->buf[1+chunklen], left);
}
// Exit if a "cmd" was sent and no response is expected.
if ((flags & SOI2C_NO_RESPONSE) != 0) {
return STATUS_OK;
}
// Go into a receive loop, using the txbuf as a (potentially-growing) rxbuf.
uint32_t msLeftToWait = 5000;
uint8_t chunklen = 0;
while (true) {
uint8_t hdrlen = 2;
// First, attempt to grow the buffer to ensure we have enough
if (ctx->growFn != NULL) {
if (ctx->bufused + hdrlen + chunklen > ctx->buflen) {
ctx->growFn(&ctx->buf, &ctx->buflen, ctx->bufused + hdrlen + chunklen);
}
}
// Constrain by our buffer size
if (ctx->bufused + hdrlen + chunklen > ctx->buflen) {
chunklen = (ctx->buflen - ctx->bufused) - hdrlen;
}
// Issue special write transaction that is a 'read will come next' transaction
ctx->buf[ctx->bufused+0] = 0;
ctx->buf[ctx->bufused+1] = chunklen;
if (!ctx->tx(ctx->port, ctx->addr, &ctx->buf[ctx->bufused], hdrlen)) {
return STATUS_IO_TRANSMIT;
}
ctx->delay(1);
// Receive the chunk of data
if (!ctx->rx(ctx->port, ctx->addr, &ctx->buf[ctx->bufused], chunklen + hdrlen)) {
return STATUS_IO_TRANSMIT;
}
ctx->delay(5);
// Verify size
uint8_t availableBytes = ctx->buf[ctx->bufused+0];
uint8_t returnedBytes = ctx->buf[ctx->bufused+1];
if (returnedBytes != chunklen) {
return STATUS_IO_BAD_SIZE_RETURNED;
}
// Look at what has just been received for a terminator, and stop if found
bool receivedNewline = (memchr(&ctx->buf[ctx->bufused+2], '\n', chunklen) != NULL);
// Only move bytes into the response buffer if a nonzero length specified,
// else just flush it.
if ((flags & SOI2C_IGNORE_RESPONSE) == 0 && chunklen > 0) {
memmove(&ctx->buf[ctx->bufused], &ctx->buf[ctx->bufused+2], chunklen);
ctx->bufused += chunklen;
}
// Attempt to receive all available bytes in the next chunk
chunklen = availableBytes;
// If more to receive, do it
if (chunklen > 0) {
continue;
}
// If there's nothing available AND we've received a newline, we're done
if (receivedNewline) {
break;
}
// If no time left to process the transaction, give up
uint32_t pollMs = 50;
if (msLeftToWait < pollMs) {
return STATUS_IO_TIMEOUT;
}
// Delay, and subtract from what's left
ctx->delay(pollMs);
msLeftToWait -= pollMs;
}
// Done
return STATUS_OK;
}