-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy patheeprom_programmer_lib.h
More file actions
639 lines (537 loc) · 17.9 KB
/
eeprom_programmer_lib.h
File metadata and controls
639 lines (537 loc) · 17.9 KB
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
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
// TODO: https://docs.arduino.cc/learn/contributions/arduino-creating-library-guide/
#ifndef __eeprom_programmer_lib_h__
#define __eeprom_programmer_lib_h__
#include <limits.h>
#include "chip_wiring.h"
using namespace ChipWiring;
namespace EepromProgrammerLibrary {
// Error Codes
enum ErrorCode : short {
SUCCESS = 0,
// connection and pins
INVALID_BOARD_WIRING_TYPE = 11,
PINS_NOT_INITIALIZED = 12,
// chip
CHIP_NOT_SUPPORTED = 21,
CHIP_ALREADY_INITIALIZED = 22,
CHIP_NOT_INITIALIZED = 23,
// address
INVALID_PAGE_SIZE = 31,
INVALID_PAGE_NO = 32,
INVALID_ADDRESS = 33,
// read
READ_MODE_DISABLED = 41,
READ_FAILED = 42,
// write
WRITE_MODE_DISABLED = 51,
WRITE_FAILED = 52,
// unknown
UNKNOWN_ERROR = SHRT_MAX
};
// EEPROM Programmer
class EepromProgrammer {
public:
EepromProgrammer(const BoardWiringType board_wiring_type);
// init
ErrorCode init_programmer();
ErrorCode init_chip(const String& chip_type);
// settings
inline uint32_t get_board_wiring_type() {
switch (_chip_wiring_controller.get_board_wiring_type()) {
case BoardWiringType::DIP28:
return 28;
case BoardWiringType::DIP24:
return 24;
default:
return 0;
}
}
inline uint32_t get_memory_size_bytes() {
return _memory_size_bytes;
}
inline uint32_t get_page_size_bytes() {
return _page_size_bytes;
}
inline uint32_t get_max_page_size() {
return _MAX_PAGE_SIZE;
}
// read
ErrorCode set_read_mode(const uint32_t page_size_bytes);
ErrorCode read_page(const int page_no, uint8_t* bytes);
// write
ErrorCode set_write_mode(const uint32_t page_size_bytes);
ErrorCode write_page(const int page_no, const uint8_t* bytes, const size_t bytes_size);
// debugging
void get_read_byte_usec_for_page(unsigned int* read_byte_usec_for_page, const size_t buffer_size) {
if (buffer_size <= 0 || buffer_size > _MAX_PAGE_SIZE) {
return;
}
for (int i = 0; i < buffer_size; i++) {
read_byte_usec_for_page[i] = _read_byte_usec_for_page[i];
}
}
void get_write_byte_usec_for_page(unsigned int* write_byte_usec_for_page, const size_t buffer_size) {
if (buffer_size <= 0 || buffer_size > _MAX_PAGE_SIZE) {
return;
}
for (int i = 0; i < buffer_size; i++) {
write_byte_usec_for_page[i] = _write_byte_usec_for_page[i];
}
}
// helpers
static String address_to_binary_string(const uint32_t address, const size_t address_bus_size) {
bool b_address[address_bus_size];
_address_to_bits_array(address, b_address, address_bus_size);
String result = "";
for (int i = 0; i < address_bus_size; i++) {
// print in reverse order, since the printed A0 should be the last bit
result += b_address[address_bus_size - 1 - i] ? 1 : 0;
}
return result;
}
static String address_to_hex_string(const uint32_t address) {
char buf[8 + 1];
sprintf(buf, "%08x", address);
return String(buf);
}
static String data_to_hex_string(const uint8_t data) {
char buf[2 + 1];
sprintf(buf, "%02x", data);
return String(buf);
}
private:
static const uint32_t _MAX_PAGE_SIZE = 64;
ErrorCode _read_byte(const uint32_t address, uint8_t& byte);
ErrorCode _write_byte(const uint32_t address, const uint8_t data);
enum _DataBusMode {
READ,
WRITE,
};
void _set_address_bus_mode();
void _set_data_bus_mode(const _DataBusMode mode);
void _write_address(const uint32_t address);
uint8_t _read_data();
void _write_data(const uint8_t data);
void _polling(const uint8_t data);
void _rdy_busy_polling();
void _data_polling(const uint8_t data);
// wiring controller
ChipWiringController _chip_wiring_controller;
// PINS
// address bus
PIN_NO _address_bus_pins[ChipWiringController::MAX_ADDRESS_BUS_SIZE];
size_t _address_bus_size;
// data bus
PIN_NO _data_bus_pins[ChipWiringController::MAX_DATA_BUS_SIZE];
size_t _data_bus_size;
// management
PIN_NO _chip_enable_pin; // !CE
PIN_NO _output_enable_pin; // !OE
PIN_NO _write_enable_pin; // !WE
PIN_NO _rdy_busy_pin; // RDY / !BUSY
// inner
bool _pins_initialized;
bool _chip_ready;
// chip settings
unsigned int _write_polling_time_usec;
bool _can_write_pages;
// optimizations
bool _current_address[ChipWiringController::MAX_ADDRESS_BUS_SIZE];
bool _current_data[ChipWiringController::MAX_DATA_BUS_SIZE];
// modes
uint32_t _memory_size_bytes;
uint32_t _page_size_bytes;
bool _read_mode;
bool _write_mode;
// debugging
unsigned int _read_byte_usec_for_page[_MAX_PAGE_SIZE];
unsigned int _write_byte_usec_for_page[_MAX_PAGE_SIZE];
// bit operations
// Most Significant Bit First ordering
// { 0,0,0,0,0,0,0,1 } == 1
// { 1,0,0,0,0,0,0,0 } == 128
// Less Significant Bit First ordering
// { 1,0,0,0,0,0,0,0 } == 1
// { 0,0,0,0,0,0,0,1 } == 128
static void _address_to_bits_array(uint32_t address, bool* b_address, const size_t address_bus_size) {
// ensure address is within the memory size range
const uint32_t memory_size_bytes = (uint32_t)(1) << address_bus_size;
if (address >= memory_size_bytes) {
return;
}
for (int i = 0; i < address_bus_size; ++i) {
// MSB order
// b_address[address_bus_size - 1 - i] = (address >> i) & 1;
// LSB order
b_address[i] = (address >> i) & 1;
}
}
static void _data_to_bits_array(uint8_t data, bool* b_data, const size_t data_bus_size) {
// MSB order
for (int i = 0; i < data_bus_size; i++) {
// MSP order
// b_data[data_bus_size - 1 - i] = bitRead(data, i);
// LSB order
b_data[i] = (data >> i) & 1;
}
}
static uint8_t _bits_array_to_data(const bool* b_data, const size_t data_bus_size) {
// MSB order
uint8_t data = 0;
for (int i = 0; i < data_bus_size; i++) {
data = (data << 1) | b_data[data_bus_size - 1 - i];
}
return data;
}
};
EepromProgrammer::EepromProgrammer(const BoardWiringType board_wiring_type)
: _chip_wiring_controller(board_wiring_type) {
// inner
_pins_initialized = false;
_chip_ready = false;
// pins
_address_bus_size = 0;
_data_bus_size = 0;
// chip settings
_write_polling_time_usec = 0;
_can_write_pages = false;
// mode
_memory_size_bytes = 0;
_page_size_bytes = 0;
_read_mode = false;
_write_mode = false;
// performance
for (int i = 0; i < _MAX_PAGE_SIZE; i++) {
_read_byte_usec_for_page[i] = 0;
_write_byte_usec_for_page[i] = 0;
}
}
ErrorCode EepromProgrammer::init_programmer() {
PIN_NO board_bus_pins[ChipWiringController::MAX_BOARD_BUS_SIZE];
const size_t board_bus_size = _chip_wiring_controller.get_board_bus_pins(board_bus_pins, ChipWiringController::MAX_BOARD_BUS_SIZE);
if (board_bus_size == 0) {
return ErrorCode::PINS_NOT_INITIALIZED;
}
// set all pins as NC
for (size_t i = 0; i < board_bus_size; i++) {
const PIN_NO pin_no = board_bus_pins[i];
if (pin_no == 0) { // VCC or GND
continue;
}
pinMode(pin_no, INPUT_PULLUP);
// pinMode(pin_no, OUTPUT);
// digitalWrite(pin_no, LOW);
}
_pins_initialized = true;
return ErrorCode::SUCCESS;
}
ErrorCode EepromProgrammer::init_chip(const String& chip_name) {
if (!_pins_initialized) {
return ErrorCode::PINS_NOT_INITIALIZED;
}
if (_chip_ready) {
return ErrorCode::CHIP_ALREADY_INITIALIZED;
}
_chip_wiring_controller.set_chip_type(chip_name_to_type(chip_name));
ChipType chip_type = _chip_wiring_controller.get_chip_type();
if (chip_type == ChipType::UNKNOWN) {
return ErrorCode::CHIP_NOT_SUPPORTED;
}
// address bus
_address_bus_size = _chip_wiring_controller.get_address_bus_pins(_address_bus_pins, ChipWiringController::MAX_ADDRESS_BUS_SIZE);
if (_address_bus_size == 0) {
return ErrorCode::PINS_NOT_INITIALIZED;
}
_memory_size_bytes = (uint32_t)(1) << _address_bus_size;
_set_address_bus_mode();
// data bus
_data_bus_size = _chip_wiring_controller.get_data_bus_pins(_data_bus_pins, ChipWiringController::MAX_DATA_BUS_SIZE);
if (_data_bus_size == 0) {
return ErrorCode::PINS_NOT_INITIALIZED;
}
_set_data_bus_mode(_DataBusMode::READ);
// management pins
// !CE, !OE, !WE, [!BSY]
PIN_NO management_pins[ChipWiringController::MAX_MANAGEMENT_SIZE];
const size_t management_size = _chip_wiring_controller.get_management_pins(management_pins, ChipWiringController::MAX_MANAGEMENT_SIZE);
if (management_size == 0) {
return ErrorCode::PINS_NOT_INITIALIZED;
}
// !CE
_chip_enable_pin = management_pins[0];
pinMode(_chip_enable_pin, OUTPUT);
digitalWrite(_chip_enable_pin, LOW); // always ON
// !OE
_output_enable_pin = management_pins[1];
pinMode(_output_enable_pin, OUTPUT);
digitalWrite(_output_enable_pin, HIGH);
// !WE
_write_enable_pin = management_pins[2];
pinMode(_write_enable_pin, OUTPUT);
digitalWrite(_write_enable_pin, HIGH);
// RDY/!BUSY
_rdy_busy_pin = management_pins[3];
if (_rdy_busy_pin > 0) {
// open drain
pinMode(_rdy_busy_pin, INPUT_PULLUP);
}
// chip settings
// tune this constant if write is not working
// if the waiting is insufficient, data propagation may be incomplete
// AT28C04 write time is about 950 us
// AT28C16 write time is about 3800 us
// AT28C64 write time is about 4100 us
// AT28C256 write time is about 6500 us
_write_polling_time_usec = 20000;
switch (chip_type) {
case ChipType::AT28C04:
break;
case ChipType::AT28C16:
break;
case ChipType::AT28C64:
break;
case ChipType::AT28C256:
// doesn't work on Arduino MEGA
// enable if use Arduino DUE only
_can_write_pages = true;
break;
default:
break;
}
_chip_ready = true;
return ErrorCode::SUCCESS;
}
ErrorCode EepromProgrammer::set_read_mode(const uint32_t page_size_bytes) {
if (!_pins_initialized) {
return ErrorCode::PINS_NOT_INITIALIZED;
}
if (!_chip_ready) {
return ErrorCode::CHIP_NOT_INITIALIZED;
}
if (page_size_bytes < 1 || page_size_bytes > _MAX_PAGE_SIZE) {
return ErrorCode::INVALID_PAGE_SIZE;
}
_read_mode = true;
_page_size_bytes = page_size_bytes;
// initial READ waveforms state
digitalWrite(_output_enable_pin, HIGH); // off
digitalWrite(_write_enable_pin, HIGH); // not in use
// switch data pins to READ mode
_set_data_bus_mode(_DataBusMode::READ);
return ErrorCode::SUCCESS;
}
ErrorCode EepromProgrammer::read_page(const int page_no, uint8_t* bytes) {
if (!_pins_initialized) {
return ErrorCode::PINS_NOT_INITIALIZED;
}
if (!_chip_ready) {
return ErrorCode::CHIP_NOT_INITIALIZED;
}
if (!_read_mode) {
return ErrorCode::READ_MODE_DISABLED;
}
const uint32_t max_page_no = _memory_size_bytes / _page_size_bytes;
if (page_no < 0 || page_no >= max_page_no) {
return ErrorCode::INVALID_PAGE_NO;
}
const uint32_t start_address = page_no * _page_size_bytes;
for (int i = 0; i < _page_size_bytes; i++) {
const unsigned long read_byte_start_usec = micros();
uint8_t byte = -1;
ErrorCode code = _read_byte(start_address + i, byte);
if (code != ErrorCode::SUCCESS) {
return ErrorCode::READ_FAILED;
}
bytes[i] = byte;
_read_byte_usec_for_page[i] = (unsigned int)(micros() - read_byte_start_usec);
}
return ErrorCode::SUCCESS;
}
ErrorCode EepromProgrammer::set_write_mode(const uint32_t page_size_bytes) {
if (!_pins_initialized) {
return ErrorCode::PINS_NOT_INITIALIZED;
}
if (!_chip_ready) {
return ErrorCode::CHIP_NOT_INITIALIZED;
}
if (page_size_bytes < 1 || page_size_bytes > _MAX_PAGE_SIZE) {
return ErrorCode::INVALID_PAGE_SIZE;
}
_write_mode = true;
_page_size_bytes = page_size_bytes;
// initial WRITE waveforms state (!WE controlled)
digitalWrite(_output_enable_pin, HIGH); // not in use
digitalWrite(_write_enable_pin, HIGH); // off
// switch data pins to WRITE mode
_set_data_bus_mode(_DataBusMode::WRITE);
return ErrorCode::SUCCESS;
}
ErrorCode EepromProgrammer::write_page(const int page_no, const uint8_t* bytes, const size_t bytes_size) {
if (!_pins_initialized) {
return ErrorCode::PINS_NOT_INITIALIZED;
}
if (!_chip_ready) {
return ErrorCode::CHIP_NOT_INITIALIZED;
}
if (!_write_mode) {
return ErrorCode::WRITE_MODE_DISABLED;
}
if (bytes_size <= 0 || bytes_size > _page_size_bytes) {
return ErrorCode::INVALID_PAGE_SIZE;
}
const uint32_t max_page_no = _memory_size_bytes / _page_size_bytes;
if (page_no < 0 || page_no >= max_page_no) {
return ErrorCode::INVALID_PAGE_NO;
}
const uint32_t start_address = page_no * _page_size_bytes;
for (int i = 0; i < bytes_size; i++) {
const unsigned long write_byte_start_usec = micros();
ErrorCode code = _write_byte(start_address + i, bytes[i]);
if (code != ErrorCode::SUCCESS) {
return ErrorCode::WRITE_FAILED;
}
// poll only last byte for the page write mode
if (!_can_write_pages || (_can_write_pages && i == bytes_size - 1)) {
_polling(bytes[i]);
}
_write_byte_usec_for_page[i] = (unsigned int)(micros() - write_byte_start_usec);
}
return ErrorCode::SUCCESS;
}
ErrorCode EepromProgrammer::_read_byte(const uint32_t address, uint8_t& byte) {
if (address >= _memory_size_bytes) {
return ErrorCode::INVALID_ADDRESS;
}
// (1) set address
_write_address(address);
// (2) output enable
digitalWrite(_output_enable_pin, LOW);
// (3) !OE to Output Delay (delta between OE and data ready) == 100 ns MAX
delayMicroseconds(1); // arduino cannot delay in ns, only us
// (4) read data
byte = _read_data();
// (5) output disable
digitalWrite(_output_enable_pin, HIGH);
return ErrorCode::SUCCESS;
}
ErrorCode EepromProgrammer::_write_byte(const uint32_t address, const uint8_t data) {
if (address >= _memory_size_bytes) {
return ErrorCode::INVALID_ADDRESS;
}
// (1) set address
_write_address(address);
// (2) set data
_write_data(data);
// (3) wrtie enable
digitalWrite(_write_enable_pin, LOW);
// (4) wrtie disable (initiates the data flush)
digitalWrite(_write_enable_pin, HIGH);
return ErrorCode::SUCCESS;
}
void EepromProgrammer::_set_address_bus_mode() {
for (int i = 0; i < _address_bus_size; i++) {
pinMode(_address_bus_pins[i], OUTPUT);
digitalWrite(_address_bus_pins[i], 0);
_current_address[i] = 0;
}
}
void EepromProgrammer::_set_data_bus_mode(const EepromProgrammer::_DataBusMode mode) {
if (mode == EepromProgrammer::_DataBusMode::READ) {
for (int i = 0; i < _data_bus_size; i++) {
pinMode(_data_bus_pins[i], INPUT_PULLUP);
}
} else if (mode == EepromProgrammer::_DataBusMode::WRITE) {
for (int i = 0; i < _data_bus_size; i++) {
pinMode(_data_bus_pins[i], OUTPUT);
digitalWrite(_data_bus_pins[i], 0);
_current_data[i] = 0;
}
}
}
void EepromProgrammer::_write_address(const uint32_t address) {
const size_t c_address_bus_size = _address_bus_size;
bool b_address[c_address_bus_size];
_address_to_bits_array(address, b_address, c_address_bus_size);
for (int i = 0; i < c_address_bus_size; i++) {
if (_current_address[i] != b_address[i]) {
digitalWrite(_address_bus_pins[i], b_address[i]);
_current_address[i] = b_address[i];
}
}
}
uint8_t EepromProgrammer::_read_data() {
const size_t c_data_bus_size = _data_bus_size;
bool b_data[c_data_bus_size];
for (int i = 0; i < c_data_bus_size; i++) {
b_data[i] = digitalRead(_data_bus_pins[i]) == HIGH ? 1 : 0;
}
return _bits_array_to_data(b_data, c_data_bus_size);
}
void EepromProgrammer::_write_data(const uint8_t data) {
const size_t c_data_bus_size = _data_bus_size;
bool b_data[c_data_bus_size];
_data_to_bits_array(data, b_data, c_data_bus_size);
for (int i = 0; i < c_data_bus_size; i++) {
if (_current_data[i] != b_data[i]) {
digitalWrite(_data_bus_pins[i], b_data[i]);
_current_data[i] = b_data[i];
}
}
}
void EepromProgrammer::_polling(const uint8_t data) {
if (_rdy_busy_pin > 0) {
_rdy_busy_polling();
} else {
_data_polling(data);
}
}
void EepromProgrammer::_rdy_busy_polling() {
const unsigned long polling_start_usec = micros();
// wait until device switches to !BUSY state, if chip has the RDY/!BUSY pin
// Time to Device Busy (delta between WE and !BUSY) == 50 ms MAX (spec)
delayMicroseconds(1); // arduino cannot delay in ns, only us
int currBusyState = digitalRead(_rdy_busy_pin);
// wait until !BUSY state switches to READY state (1 ms MAX)
// or just wait for the Write Cycle Time MAX
if (currBusyState == LOW) {
// device is in !BUSY state
// use the READY/!BUSY pin status to wait for the Write Cycle End
const unsigned int delay_usec = 100;
int prevBusyState = currBusyState;
while ((micros() - polling_start_usec) < _write_polling_time_usec) {
delayMicroseconds(delay_usec);
prevBusyState = currBusyState;
currBusyState = digitalRead(_rdy_busy_pin);
if (prevBusyState == LOW && currBusyState == HIGH) { // rising edge
break;
}
}
} else {
// device not in !BUSY state
// use generic delay
delayMicroseconds(_write_polling_time_usec);
}
}
void EepromProgrammer::_data_polling(const uint8_t data) {
const unsigned long polling_start_usec = micros();
// use !DATA polling, if chip doesn't have the RDY/!BUSY pin
// following the data poll waveforms, the data is read in a loop until the value matches the one written
// during the write procedure, the data pins remain in a metastable state.
_set_data_bus_mode(_DataBusMode::READ);
const unsigned int delay_usec = 50;
while ((micros() - polling_start_usec) < _write_polling_time_usec) {
delayMicroseconds(delay_usec);
// !DATA polling waveforms require to switch !CE and !OE for every attempt
digitalWrite(_output_enable_pin, LOW);
// !OE to Output Delay (delta between OE and data ready) == 100 ns MAX
delayMicroseconds(1); // arduino cannot delay in ns, only us
uint8_t read_result = _read_data();
digitalWrite(_output_enable_pin, HIGH);
if (read_result == data) {
break;
}
}
_set_data_bus_mode(_DataBusMode::WRITE);
}
} // EepromProgrammerLibrary
#endif // !__eeprom_programmer_lib_h__