-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathsupport.c
265 lines (218 loc) · 6.88 KB
/
support.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
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
/*
* Copyright 2015 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*
* This file was copied from https://github.com/devttys0/libmpsse.git (sha1
* f1a6744b), and modified to suite the Chromium OS project.
*
* Internal functions used by libmpsse.
*
* Craig Heffner
* 27 December 2011
*/
#include <string.h>
#include "support.h"
/* Write data to the FTDI chip */
int raw_write(struct mpsse_context* mpsse, uint8_t* buf, int size) {
int retval = MPSSE_FAIL;
if (mpsse->mode) {
if (ftdi_write_data(&mpsse->ftdi, buf, size) == size) {
retval = MPSSE_OK;
}
}
return retval;
}
/* Read data from the FTDI chip */
int raw_read(struct mpsse_context* mpsse, uint8_t* buf, int size) {
int n = 0, r = 0;
if (mpsse->mode) {
while (n < size) {
r = ftdi_read_data(&mpsse->ftdi, buf, size);
if (r < 0)
break;
n += r;
}
if (mpsse->flush_after_read) {
/*
* Make sure the buffers are cleared after a read or subsequent reads may
*fail.
*
* Is this needed anymore? It slows down repetitive read operations by
*~8%.
*/
ftdi_usb_purge_rx_buffer(&mpsse->ftdi);
}
}
return n;
}
/* Sets the read and write timeout periods for bulk usb data transfers. */
void set_timeouts(struct mpsse_context* mpsse, int timeout) {
if (mpsse->mode) {
mpsse->ftdi.usb_read_timeout = timeout;
mpsse->ftdi.usb_write_timeout = timeout;
}
return;
}
/* Convert a frequency to a clock divisor */
uint16_t freq2div(uint32_t system_clock, uint32_t freq) {
return (((system_clock / freq) / 2) - 1);
}
/* Convert a clock divisor to a frequency */
uint32_t div2freq(uint32_t system_clock, uint16_t div) {
return (system_clock / ((1 + div) * 2));
}
/* Builds a buffer of commands + data blocks */
uint8_t* build_block_buffer(struct mpsse_context* mpsse,
uint8_t cmd,
const uint8_t* data,
int size,
int* buf_size) {
uint8_t* buf = NULL;
int i = 0, j = 0, k = 0, dsize = 0, num_blocks = 0, total_size = 0,
xfer_size = 0;
uint16_t rsize = 0;
*buf_size = 0;
/* Data block size is 1 in I2C, or when in bitmode */
if (mpsse->mode == I2C || (cmd & MPSSE_BITMODE)) {
xfer_size = 1;
} else {
xfer_size = mpsse->xsize;
}
num_blocks = (size / xfer_size);
if (size % xfer_size) {
num_blocks++;
}
/* The total size of the data will be the data size + the write command */
total_size = size + (CMD_SIZE * num_blocks);
/* In I2C we have to add 3 additional commands per data block */
if (mpsse->mode == I2C) {
total_size += (CMD_SIZE * 3 * num_blocks);
}
buf = malloc(total_size);
if (buf) {
memset(buf, 0, total_size);
for (j = 0; j < num_blocks; j++) {
dsize = size - k;
if (dsize > xfer_size) {
dsize = xfer_size;
}
/* The reported size of this block is block size - 1 */
rsize = dsize - 1;
/* For I2C we need to ensure that the clock pin is set low prior to
* clocking out data */
if (mpsse->mode == I2C) {
buf[i++] = SET_BITS_LOW;
buf[i++] = mpsse->pstart & ~SK;
/* On receive, we need to ensure that the data out line is set as an
* input to avoid contention on the bus */
if (cmd == mpsse->rx) {
buf[i++] = mpsse->tris & ~DO;
} else {
buf[i++] = mpsse->tris;
}
}
/* Copy in the command for this block */
buf[i++] = cmd;
buf[i++] = (rsize & 0xFF);
if (!(cmd & MPSSE_BITMODE)) {
buf[i++] = ((rsize >> 8) & 0xFF);
}
/* On a write, copy the data to transmit after the command */
if (cmd == mpsse->tx || cmd == mpsse->txrx) {
memcpy(buf + i, data + k, dsize);
/* i == offset into buf */
i += dsize;
/* k == offset into data */
k += dsize;
}
/* In I2C mode we need to clock one ACK bit after each byte */
if (mpsse->mode == I2C) {
/* If we are receiving data, then we need to clock out an ACK for each
* byte */
if (cmd == mpsse->rx) {
buf[i++] = SET_BITS_LOW;
buf[i++] = mpsse->pstart & ~SK;
buf[i++] = mpsse->tris;
buf[i++] = mpsse->tx | MPSSE_BITMODE;
buf[i++] = 0;
buf[i++] = mpsse->tack;
}
/* If we are sending data, then we need to clock in an ACK for each byte
*/
else if (cmd == mpsse->tx) {
/* Need to make data out an input to avoid contention on the bus when
* the slave sends an ACK */
buf[i++] = SET_BITS_LOW;
buf[i++] = mpsse->pstart & ~SK;
buf[i++] = mpsse->tris & ~DO;
buf[i++] = mpsse->rx | MPSSE_BITMODE;
buf[i++] = 0;
buf[i++] = SEND_IMMEDIATE;
}
}
}
*buf_size = i;
}
return buf;
}
/* Set the low bit pins high/low */
int set_bits_low(struct mpsse_context* mpsse, int port) {
char buf[CMD_SIZE] = {0};
buf[0] = SET_BITS_LOW;
buf[1] = port;
buf[2] = mpsse->tris;
return raw_write(mpsse, (uint8_t*)&buf, sizeof(buf));
}
/* Set the high bit pins high/low */
int set_bits_high(struct mpsse_context* mpsse, int port) {
char buf[CMD_SIZE] = {0};
buf[0] = SET_BITS_HIGH;
buf[1] = port;
buf[2] = mpsse->trish;
return raw_write(mpsse, (uint8_t*)&buf, sizeof(buf));
}
/* Set the GPIO pins high/low */
int gpio_write(struct mpsse_context* mpsse, int pin, int direction) {
int retval = MPSSE_FAIL;
if (mpsse->mode == BITBANG) {
if (direction == HIGH) {
mpsse->bitbang |= (1 << pin);
} else {
mpsse->bitbang &= ~(1 << pin);
}
if (set_bits_high(mpsse, mpsse->bitbang) == MPSSE_OK) {
retval = raw_write(mpsse, (uint8_t*)&mpsse->bitbang, 1);
}
} else {
/* The first four pins can't be changed unless we are in a stopped status */
if (pin < NUM_GPIOL_PINS && mpsse->status == STOPPED) {
/* Convert pin number (0-3) to the corresponding pin bit */
pin = (GPIO0 << pin);
if (direction == HIGH) {
mpsse->pstart |= pin;
mpsse->pidle |= pin;
mpsse->pstop |= pin;
} else {
mpsse->pstart &= ~pin;
mpsse->pidle &= ~pin;
mpsse->pstop &= ~pin;
}
retval = set_bits_low(mpsse, mpsse->pstop);
} else if (pin >= NUM_GPIOL_PINS && pin < NUM_GPIO_PINS) {
/* Convert pin number (4 - 11) to the corresponding pin bit */
pin -= NUM_GPIOL_PINS;
if (direction == HIGH) {
mpsse->gpioh |= (1 << pin);
} else {
mpsse->gpioh &= ~(1 << pin);
}
retval = set_bits_high(mpsse, mpsse->gpioh);
}
}
return retval;
}
/* Checks if a given MPSSE context is valid. */
int is_valid_context(struct mpsse_context* mpsse) {
return mpsse != NULL;
}