forked from robokoding/STK500
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSTK500.ino
142 lines (122 loc) · 3.87 KB
/
STK500.ino
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
int resetPin = 7;
// RX and TX should be connected to TX and RX on the target board, respectively.
// If the target board is already powered, then just connect their two
// GND pins together, otherwise connect their +5V pins together as well
int began = 0, groove = 50, i, j, start, end, address, laddress, haddress, error = 0, a, b, c, d, e, f, buff[128], buffLength, k, readBuff[16], preBuff, readBuffLength;
// Reads bytes until there's nothing left to read
// Then sticks 'em in readBuff, and sets readBuffLength
void readBytes() {
readBuffLength = 0;
while (Serial.available() > 0) {
preBuff = Serial.read();
// this if statement is necessary because for some reason, the target
// board replies with 0xFC instead of 0x10
// i've got no idea whether it's an error code, but simply substituting
// 0xFC for 0x10 in the readBuff seems to do the trick
if (preBuff == 0xFC) {
preBuff = 0x10;
}
readBuff[readBuffLength] = preBuff;
readBuffLength++;
}
}
// Read bytes until there's nothing left to read
// The only difference to the above function is that
// this one doesn't substitute 0xFC for 0x10
void readBytess() {
readBuffLength = 0;
while (Serial.available() > 0) {
readBuff[readBuffLength] = Serial.read();
readBuffLength++;
}
}
int copier() {
if (!began) {
Serial.begin(57600);
pinMode(resetPin, OUTPUT);
began = 1;
}
// Reset the target board
digitalWrite(resetPin, LOW);
delay(100);
digitalWrite(resetPin, HIGH);
delay(100);
// Get in sync with the AVR
for (i = 0; i < 25; i++) {
Serial.print(0x30, BYTE); // STK_GET_SYNC
Serial.print(0x20, BYTE); // STK_CRC_EOP
delay(groove);
}
readBytes();
// We expect to receive 0x14 (STK_INSYNC) and 0x10 (STK_OK)
// Every response must start with 0x14 and end with 0x10, otherwise
// something's wrong
if (readBuffLength < 2 || readBuff[0] != 0x14 || readBuff[1] != 0x10) {
return 0;
}
Serial.flush();
// Enter programming mode
Serial.print(0x50, BYTE);
Serial.print(0x20, BYTE);
delay(groove);
readBytes();
if (readBuffLength != 2 || readBuff[0] != 0x14 || readBuff[1] != 0x10) {
return 0;
}
// Now comes the interesting part
// We pour blocks of data from sketch[] and into buff, ready to write
// to the AVR's flash
// We only write in blocks of 128 or less bytes
address = 0;
for (i = 0; i < sketchLength; i += 128) {
start = i;
end = i + 127;
if (sketchLength <= end) {
end = sketchLength - 1;
}
buffLength = end - start + 1;
for (j = 0; j < buffLength; j++) {
buff[j] = pgm_read_byte(sketch+i+j);
}
// The buffer is now filled with the appropriate bytes
// Set the address of the avr's flash memory to write to
haddress = address / 256;
laddress = address % 256;
address += 64; // For the next iteration
Serial.print(0x55, BYTE);
Serial.print(laddress, BYTE);
Serial.print(haddress, BYTE);
Serial.print(0x20, BYTE);
delay(groove);
readBytes();
if (readBuffLength != 2 || readBuff[0] != 0x14 || readBuff[1] != 0x10) {
return 0;
}
// Write the block
Serial.print(0x64, BYTE);
Serial.print(0x00, BYTE);
Serial.print(buffLength, BYTE);
Serial.print(0x46, BYTE);
for (j = 0; j < buffLength; j++) {
Serial.print(buff[j], BYTE);
}
Serial.print(0x20, BYTE);
delay(groove);
readBytes();
if (readBuffLength != 2 || readBuff[0] != 0x14 || readBuff[1] != 0x10) {
return 0;
}
}
// Leave programming mode
Serial.print(0x51, BYTE);
Serial.print(0x20, BYTE);
delay(groove);
readBytes();
if (readBuffLength != 2 || readBuff[0] != 0x14 || readBuff[1] != 0x10) {
return 0;
}
// If we've reached here, then all is well
// But please note: no sketch verification is done by the Copier sketch!
return 1;
}
#endif