forked from Klipper3d/klipper
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtmcuart.c
249 lines (234 loc) · 7.49 KB
/
tmcuart.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
// Commands for sending messages to a TMC2208 via its single wire UART
//
// Copyright (C) 2018 Kevin O'Connor <[email protected]>
//
// This file may be distributed under the terms of the GNU GPLv3 license.
#include <string.h> // memcpy
#include "board/gpio.h" // gpio_out_write
#include "board/irq.h" // irq_disable
#include "board/misc.h" // timer_read_time
#include "basecmd.h" // oid_alloc
#include "command.h" // DECL_COMMAND
#include "sched.h" // DECL_SHUTDOWN
struct tmcuart_s {
struct timer timer;
struct gpio_out tx_pin;
struct gpio_in rx_pin;
uint8_t flags;
uint8_t pos, read_count, write_count;
uint32_t cfg_bit_time, bit_time;
uint8_t data[10];
};
enum {
TU_LINE_HIGH = 1<<0, TU_ACTIVE = 1<<1, TU_READ_SYNC = 1<<2,
TU_REPORT = 1<<3, TU_PULLUP = 1<<4, TU_SINGLE_WIRE = 1<<5
};
static struct task_wake tmcuart_wake;
// Restore uart line to normal "idle" mode
static void
tmcuart_reset_line(struct tmcuart_s *t)
{
if (t->flags & TU_SINGLE_WIRE)
gpio_out_reset(t->tx_pin, 1);
else
gpio_out_write(t->tx_pin, 1);
t->flags = (t->flags & (TU_PULLUP | TU_SINGLE_WIRE)) | TU_LINE_HIGH;
}
// Helper function to end a transmission and schedule a response
static uint_fast8_t
tmcuart_finalize(struct tmcuart_s *t)
{
tmcuart_reset_line(t);
t->flags |= TU_REPORT;
sched_wake_task(&tmcuart_wake);
return SF_DONE;
}
// Event handler for reading uart bits
static uint_fast8_t
tmcuart_read_event(struct timer *timer)
{
struct tmcuart_s *t = container_of(timer, struct tmcuart_s, timer);
uint8_t v = gpio_in_read(t->rx_pin);
// Read and store bit
uint8_t pos = t->pos, mask = 1 << (pos & 0x07), data = t->data[pos >> 3];
if (v)
data |= mask;
else
data &= ~mask;
t->data[pos >> 3] = data;
pos++;
if (pos >= t->read_count)
return tmcuart_finalize(t);
t->pos = pos;
t->timer.waketime += t->bit_time;
return SF_RESCHEDULE;
}
// Event handler for detecting start of data reception
static uint_fast8_t
tmcuart_read_sync_event(struct timer *timer)
{
struct tmcuart_s *t = container_of(timer, struct tmcuart_s, timer);
uint8_t v = gpio_in_read(t->rx_pin);
if (v) {
t->flags |= TU_READ_SYNC;
} else if (t->flags & TU_READ_SYNC) {
// Now synchronized - begin reading
t->pos = 0;
t->timer.func = tmcuart_read_event;
return tmcuart_read_event(timer);
}
if (t->pos++ >= 64) {
// Timeout
t->read_count = 0;
return tmcuart_finalize(t);
}
t->timer.waketime += t->bit_time;
return SF_RESCHEDULE;
}
// Event handler called at end of uart writing
static uint_fast8_t
tmcuart_send_finish_event(struct timer *timer)
{
struct tmcuart_s *t = container_of(timer, struct tmcuart_s, timer);
if (!t->read_count)
// This is a tx only operation - success
return tmcuart_finalize(t);
// Prepare for message rx
if (t->flags & TU_SINGLE_WIRE)
gpio_in_reset(t->rx_pin, t->flags & TU_PULLUP);
t->pos = 0;
t->timer.func = tmcuart_read_sync_event;
t->timer.waketime += t->bit_time * 4;
return SF_RESCHEDULE;
}
// Event handler for sending uart bits
static uint_fast8_t
tmcuart_send_event(struct timer *timer)
{
struct tmcuart_s *t = container_of(timer, struct tmcuart_s, timer);
// Toggle uart output
gpio_out_toggle_noirq(t->tx_pin);
t->flags ^= TU_LINE_HIGH;
// Determine next toggle time
uint8_t line_state = t->flags & TU_LINE_HIGH;
uint32_t bit_time = t->bit_time, next = bit_time;
uint8_t pos = t->pos;
for (;;) {
pos++;
if (pos >= t->write_count) {
// No more toggles necessary - schedule finish event
t->timer.func = tmcuart_send_finish_event;
t->timer.waketime += next;
return SF_RESCHEDULE;
}
uint8_t data = t->data[pos >> 3], bit = (data >> (pos & 0x07)) & 0x01;
if (bit != line_state)
break;
next += bit_time;
}
t->pos = pos;
t->timer.waketime += next;
return SF_RESCHEDULE;
}
// Event handler for sending sync nibble with enhanced baud detection
static uint_fast8_t
tmcuart_send_sync_event(struct timer *timer)
{
struct tmcuart_s *t = container_of(timer, struct tmcuart_s, timer);
// Toggle uart output and note toggle time
gpio_out_toggle_noirq(t->tx_pin);
uint32_t cur = timer_read_time();
t->flags ^= TU_LINE_HIGH;
// Determine next wakeup time
t->pos++;
if (t->pos == 1) {
// First bit just sent - record scheduling offset for later use
uint32_t offset = cur - t->timer.waketime;
t->bit_time = offset;
} else if (t->pos >= 5) {
// Last bit of sync nibble just sent - calculate actual baud rate used
uint32_t offset = cur - t->timer.waketime, start_offset = t->bit_time;
int32_t diff = offset - start_offset;
t->bit_time = t->cfg_bit_time + DIV_ROUND_CLOSEST(diff, 4);
t->timer.func = tmcuart_send_event;
t->timer.waketime += diff + t->bit_time;
return SF_RESCHEDULE;
}
t->timer.waketime += t->cfg_bit_time;
return SF_RESCHEDULE;
}
void
command_config_tmcuart(uint32_t *args)
{
struct tmcuart_s *t = oid_alloc(args[0], command_config_tmcuart
, sizeof(*t));
uint8_t pull_up = args[2];
uint32_t rx_pin = args[1], tx_pin = args[3];
t->rx_pin = gpio_in_setup(rx_pin, !!pull_up);
t->tx_pin = gpio_out_setup(tx_pin, 1);
t->cfg_bit_time = args[4];
t->flags = (TU_LINE_HIGH | (pull_up ? TU_PULLUP : 0)
| (rx_pin == tx_pin ? TU_SINGLE_WIRE : 0));
}
DECL_COMMAND(command_config_tmcuart,
"config_tmcuart oid=%c rx_pin=%u pull_up=%c"
" tx_pin=%u bit_time=%u");
// Parse and schedule a TMC UART transmission request
void
command_tmcuart_send(uint32_t *args)
{
struct tmcuart_s *t = oid_lookup(args[0], command_config_tmcuart);
if (t->flags & TU_ACTIVE)
// Uart is busy - silently drop this request (host should retransmit)
return;
uint8_t write_len = args[1];
uint8_t *write = command_decode_ptr(args[2]);
uint8_t read_len = args[3];
if (write_len > sizeof(t->data) || read_len > sizeof(t->data))
shutdown("tmcuart data too large");
memcpy(t->data, write, write_len);
t->pos = 0;
t->flags = (t->flags & (TU_LINE_HIGH|TU_PULLUP|TU_SINGLE_WIRE)) | TU_ACTIVE;
t->write_count = write_len * 8;
t->read_count = read_len * 8;
if (write_len >= 1 && (t->data[0] & 0x3f) == 0x2a) {
t->timer.func = tmcuart_send_sync_event;
} else {
t->bit_time = t->cfg_bit_time;
t->timer.func = tmcuart_send_event;
}
irq_disable();
t->timer.waketime = timer_read_time() + timer_from_us(200);
sched_add_timer(&t->timer);
irq_enable();
}
DECL_COMMAND(command_tmcuart_send, "tmcuart_send oid=%c write=%*s read=%c");
// Report completed response message back to host
void
tmcuart_task(void)
{
if (!sched_check_wake(&tmcuart_wake))
return;
uint8_t oid;
struct tmcuart_s *t;
foreach_oid(oid, t, command_config_tmcuart) {
if (!(t->flags & TU_REPORT))
continue;
irq_disable();
t->flags &= ~TU_REPORT;
irq_enable();
sendf("tmcuart_response oid=%c read=%*s"
, oid, t->read_count / 8, t->data);
}
}
DECL_TASK(tmcuart_task);
void
tmcuart_shutdown(void)
{
uint8_t i;
struct tmcuart_s *t;
foreach_oid(i, t, command_config_tmcuart) {
tmcuart_reset_line(t);
}
}
DECL_SHUTDOWN(tmcuart_shutdown);