-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathUPDI_Programmer.c
499 lines (408 loc) · 13.3 KB
/
UPDI_Programmer.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
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
// UPDI_Programmer.c
// STK500 compatible UPDI programmer firmware for Arduino
// Version 1.4
#include <avr/io.h>
#include <util/delay.h>
#include <avr/interrupt.h>
#include <string.h>
// Define UPDI communication pins
#define UPDI_PORT PORTB
#define UPDI_DDR DDRB
#define UPDI_PIN PINB
#define UPDI_BIT PB0
// UPDI communication constants
#define UPDI_BAUD 225000 // UPDI baud rate
#define UPDI_BIT_TIME (1000000 / UPDI_BAUD)
// UPDI constants
#define UPDI_SYNCH 0x55
#define UPDI_ACK 0x40
// UPDI commands
#define UPDI_CMD_LDS 0x00
#define UPDI_CMD_STS 0x40
#define UPDI_CMD_LD 0x20
#define UPDI_CMD_ST 0x60
#define UPDI_CMD_LDCS 0x80
#define UPDI_CMD_STCS 0xC0
#define UPDI_CMD_REPEAT 0xA0
// UPDI control/status registers
#define UPDI_CS_STATUSA 0x00
#define UPDI_CS_STATUSB 0x01
#define UPDI_CS_CTRLA 0x02
#define UPDI_CS_CTRLB 0x03
// STK500 constants
#define MESSAGE_START 0x1B
#define TOKEN 0x0E
// STK500 commands
#define CMD_SIGN_ON 0x01
#define CMD_GET_PARAMETER 0x41
#define CMD_SET_PARAMETER 0x42
#define CMD_ENTER_PROGMODE 0x50
#define CMD_LEAVE_PROGMODE 0x51
#define CMD_LOAD_ADDRESS 0x06
#define CMD_READ_FLASH 0x74
#define CMD_WRITE_FLASH 0x64
// STK500 status constants
#define STATUS_CMD_OK 0x00
#define STATUS_CMD_FAILED 0xC0
// Buffer sizes
#define MAX_BUFFER_SIZE 275
// Error handling constants
#define MAX_RETRIES 3
#define RETRY_DELAY_MS 100
// Function prototypes
void uart_init(void);
void uart_send_byte(uint8_t data);
uint8_t uart_receive_byte(void);
void updi_init(void);
void updi_send_byte(uint8_t data);
uint8_t updi_receive_byte(void);
void updi_send_break(void);
uint8_t updi_sync(void);
void enable_system_clock_output(void);
void handle_sync_error(uint8_t attempt);
void process_stk500_command(void);
void stk500_send_response(uint8_t status, uint8_t *data, uint16_t len);
void updi_write_cs(uint8_t address, uint8_t value);
uint8_t updi_read_cs(uint8_t address);
void updi_write_data(uint32_t address, uint8_t *data, uint16_t len);
void updi_read_data(uint32_t address, uint8_t *data, uint16_t len);
uint8_t updi_write_data_with_retry(uint32_t address, uint8_t *data, uint16_t len);
uint8_t updi_read_data_with_retry(uint32_t address, uint8_t *data, uint16_t len);
void log_error(const char *message);
// Global variables
uint8_t rx_buffer[MAX_BUFFER_SIZE];
uint8_t tx_buffer[MAX_BUFFER_SIZE];
uint32_t current_address = 0;
// Main function
int main(void) {
// Initialize UART for communication with PC
uart_init();
// Initialize UPDI interface
updi_init();
// Main program loop
while (1) {
process_stk500_command();
}
return 0;
}
// Initialize UART
void uart_init(void) {
// Set baud rate
UBRR0H = (F_CPU / 16 / 115200 - 1) >> 8;
UBRR0L = (F_CPU / 16 / 115200 - 1);
// Enable receiver and transmitter
UCSR0B = (1 << RXEN0) | (1 << TXEN0);
// Set frame format: 8 data bits, 1 stop bit, no parity
UCSR0C = (1 << UCSZ01) | (1 << UCSZ00);
}
// Send a byte over UART
void uart_send_byte(uint8_t data) {
while (!(UCSR0A & (1 << UDRE0)));
UDR0 = data;
}
// Receive a byte over UART
uint8_t uart_receive_byte(void) {
while (!(UCSR0A & (1 << RXC0)));
return UDR0;
}
// Initialize UPDI interface
void updi_init(void) {
// Set UPDI pin as output
UPDI_DDR |= (1 << UPDI_BIT);
// Set UPDI pin high (idle state)
UPDI_PORT |= (1 << UPDI_BIT);
}
// Send a byte over UPDI
void updi_send_byte(uint8_t data) {
// Start bit
UPDI_PORT &= ~(1 << UPDI_BIT);
_delay_us(UPDI_BIT_TIME);
// Data bits
for (uint8_t i = 0; i < 8; i++) {
if (data & 0x01) {
UPDI_PORT |= (1 << UPDI_BIT);
} else {
UPDI_PORT &= ~(1 << UPDI_BIT);
}
data >>= 1;
_delay_us(UPDI_BIT_TIME);
}
// Stop bits (2)
UPDI_PORT |= (1 << UPDI_BIT);
_delay_us(2 * UPDI_BIT_TIME);
}
// Receive a byte over UPDI
uint8_t updi_receive_byte(void) {
uint8_t data = 0;
// Wait for start bit
while (UPDI_PIN & (1 << UPDI_BIT));
_delay_us(UPDI_BIT_TIME / 2);
// Read data bits
for (uint8_t i = 0; i < 8; i++) {
_delay_us(UPDI_BIT_TIME);
data >>= 1;
if (UPDI_PIN & (1 << UPDI_BIT)) {
data |= 0x80;
}
}
// Wait for stop bits
_delay_us(2 * UPDI_BIT_TIME);
return data;
}
// Send a UPDI break signal
void updi_send_break(void) {
// Set UPDI pin as output
UPDI_DDR |= (1 << UPDI_BIT);
// Pull UPDI line low for 24.6 ms (minimum break time: 24.576 ms)
UPDI_PORT &= ~(1 << UPDI_BIT);
_delay_ms(24.6);
// Release UPDI line
UPDI_PORT |= (1 << UPDI_BIT);
// Wait for 2 bit times (required idle periods)
_delay_us(2 * UPDI_BIT_TIME);
}
// Perform UPDI synchronization
uint8_t updi_sync(void) {
uint8_t response;
uint8_t attempts = 0;
const uint8_t max_attempts = 5;
while (attempts < max_attempts) {
// Send break
updi_send_break();
// Send synchronization character
updi_send_byte(UPDI_SYNCH);
// Receive response
response = updi_receive_byte();
if (response == UPDI_ACK) {
return 1; // Synchronization successful
}
attempts++;
handle_sync_error(attempts);
}
return 0; // Synchronization failed
}
// Helper function to enable system clock output (for debugging)
void enable_system_clock_output(void) {
// Enable system clock output on PB0 (Arduino Uno pin 8)
DDRB |= (1 << PB0);
ASSR &= ~(1 << EXCLK);
ASSR |= (1 << EXTCLK);
}
// Advanced error handling function using Exponential Backoff
void handle_sync_error(uint8_t attempt) {
uint16_t backoff_time = (1 << attempt) * 100; // Exponential backoff in milliseconds
if (backoff_time > 5000) backoff_time = 5000; // Cap at 5 seconds
log_error("UPDI synchronization failed");
_delay_ms(backoff_time);
}
// Process STK500 commands
void process_stk500_command(void) {
uint8_t seq_num, cmd;
uint16_t rx_len = 0;
// Wait for message start
while (uart_receive_byte() != MESSAGE_START);
// Receive sequence number
seq_num = uart_receive_byte();
// Receive message length
rx_len = uart_receive_byte() << 8;
rx_len |= uart_receive_byte();
// Receive token
if (uart_receive_byte() != TOKEN) {
log_error("Invalid STK500 token received");
return;
}
// Receive command
cmd = uart_receive_byte();
// Receive message body
for (uint16_t i = 0; i < rx_len; i++) {
rx_buffer[i] = uart_receive_byte();
}
// Process command
switch (cmd) {
case CMD_SIGN_ON:
{
uint8_t sign_on_response[] = {0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 'A', 'V', 'R', 'I', 'S', 'P', '_', 'M', 'K', '2'};
stk500_send_response(STATUS_CMD_OK, sign_on_response, sizeof(sign_on_response));
}
break;
case CMD_GET_PARAMETER:
// TODO: Implement parameter retrieval
stk500_send_response(STATUS_CMD_FAILED, NULL, 0);
break;
case CMD_SET_PARAMETER:
// TODO: Implement parameter setting
stk500_send_response(STATUS_CMD_FAILED, NULL, 0);
break;
case CMD_ENTER_PROGMODE:
if (updi_sync()) {
stk500_send_response(STATUS_CMD_OK, NULL, 0);
} else {
log_error("Failed to enter programming mode");
stk500_send_response(STATUS_CMD_FAILED, NULL, 0);
}
break;
case CMD_LEAVE_PROGMODE:
// TODO: Implement leaving programming mode
stk500_send_response(STATUS_CMD_OK, NULL, 0);
break;
case CMD_LOAD_ADDRESS:
current_address = (rx_buffer[0] << 24) | (rx_buffer[1] << 16) | (rx_buffer[2] << 8) | rx_buffer[3];
stk500_send_response(STATUS_CMD_OK, NULL, 0);
break;
case CMD_READ_FLASH:
{
uint16_t num_bytes = (rx_buffer[0] << 8) | rx_buffer[1];
if (updi_read_data_with_retry(current_address, tx_buffer, num_bytes)) {
stk500_send_response(STATUS_CMD_OK, tx_buffer, num_bytes);
current_address += num_bytes;
} else {
log_error("Failed to read flash memory");
stk500_send_response(STATUS_CMD_FAILED, NULL, 0);
}
}
break;
case CMD_WRITE_FLASH:
{
uint16_t num_bytes = (rx_buffer[0] << 8) | rx_buffer[1];
if (updi_write_data_with_retry(current_address, &rx_buffer[2], num_bytes)) {
stk500_send_response(STATUS_CMD_OK, NULL, 0);
current_address += num_bytes;
} else {
log_error("Failed to write flash memory");
stk500_send_response(STATUS_CMD_FAILED, NULL, 0);
}
}
break;
default:
log_error("Unknown STK500 command received");
stk500_send_response(STATUS_CMD_FAILED, NULL, 0);
break;
}
}
// Send STK500 response
void stk500_send_response(uint8_t status, uint8_t *data, uint16_t len) {
uint16_t tx_len = len + 2; // status + data + 2 bytes for len itself
uart_send_byte(MESSAGE_START);
uart_send_byte(0); // Sequence number, always 0 for responses
uart_send_byte(tx_len >> 8);
uart_send_byte(tx_len & 0xFF);
uart_send_byte(TOKEN);
uart_send_byte(status);
for (uint16_t i = 0; i < len; i++) {
uart_send_byte(data[i]);
}
}
// Write to UPDI Control/Status register
void updi_write_cs(uint8_t address, uint8_t value) {
updi_send_byte(UPDI_CMD_STCS | address);
updi_send_byte(value);
}
// Read from UPDI Control/Status register
uint8_t updi_read_cs(uint8_t address) {
updi_send_byte(UPDI_CMD_LDCS | address);
return updi_receive_byte();
}
// Write data to UPDI
void updi_write_data(uint32_t address, uint8_t *data, uint16_t len) {
// Send STS command
updi_send_byte(UPDI_CMD_STS | 0x04);
// Send address (little-endian)
updi_send_byte(address & 0xFF);
updi_send_byte((address >> 8) & 0xFF);
updi_send_byte((address >> 16) & 0xFF);
updi_send_byte((address >> 24) & 0xFF);
// Send data
for (uint16_t i = 0; i < len; i++) {
updi_send_byte(data[i]);
}
}
// Read data from UPDI
void updi_read_data(uint32_t address, uint8_t *data, uint16_t len) {
// Send LDS command
updi_send_byte(UPDI_CMD_LDS | 0x04);
// Send address (little-endian)
updi_send_byte(address & 0xFF);
updi_send_byte((address >> 8) & 0xFF);
updi_send_byte((address >> 16) & 0xFF);
updi_send_byte((address >> 24) & 0xFF);
// Receive data
for (uint16_t i = 0; i < len; i++) {
data[i] = updi_receive_byte();
}
}
// Write data to UPDI with retry mechanism
uint8_t updi_write_data_with_retry(uint32_t address, uint8_t *data, uint16_t len) {
uint8_t retries = 0;
uint8_t success = 0;
while (retries < MAX_RETRIES && !success) {
updi_write_data(address, data, len);
// Verify written data
uint8_t verify_buffer[len];
updi_read_data(address, verify_buffer, len);
if (memcmp(data, verify_buffer, len) == 0) {
success = 1;
} else {
retries++;
log_error("UPDI write verification failed, retrying...");
_delay_ms(RETRY_DELAY_MS);
}
}
return success;
}
// Read data from UPDI with retry mechanism
uint8_t updi_read_data_with_retry(uint32_t address, uint8_t *data, uint16_t len) {
uint8_t retries = 0;
uint8_t success = 0;
while (retries < MAX_RETRIES && !success) {
updi_read_data(address, data, len);
// Verify read data (read twice and compare)
uint8_t verify_buffer[len];
updi_read_data(address, verify_buffer, len);
if (memcmp(data, verify_buffer, len) == 0) {
success = 1;
} else {
retries++;
log_error("UPDI read verification failed, retrying...");
_delay_ms(RETRY_DELAY_MS);
}
}
return success;
}
// Log error message (placeholder implementation)
void log_error(const char *message) {
// For now, we'll just send it over UART for debugging purposes.
while (*message) {
uart_send_byte(*message++);
}
uart_send_byte('\r');
uart_send_byte('\n');
}
// CRC16 calculation for error checking
uint16_t calculate_crc16(uint8_t *data, uint16_t length) {
uint16_t crc = 0xFFFF;
for (uint16_t i = 0; i < length; i++) {
crc ^= (uint16_t)data[i];
for (uint8_t j = 0; j < 8; j++) {
if (crc & 0x0001) {
crc = (crc >> 1) ^ 0xA001;
} else {
crc >>= 1;
}
}
}
return crc;
}
// Advanced error recovery function
void perform_error_recovery(void) {
log_error("Performing error recovery...");
// Reset UPDI interface
updi_send_break();
// Re-synchronize
if (!updi_sync()) {
log_error("Error recovery failed: unable to re-synchronize UPDI");
return;
}
// Reset device (if applicable)
// TODO: Implement device-specific reset procedure
log_error("Error recovery complete");
}