From d66de12bff07d14ed4f79a8d9b5bd93509ba6f07 Mon Sep 17 00:00:00 2001 From: Marzogh Date: Sat, 4 Nov 2017 21:36:22 +1000 Subject: [PATCH] Revert "Pull final code changes for v3.0.0" --- .travis.yml | 2 +- README.md | 10 +- examples/FlashDiagnostics/.trinket.test.skip | 1 + .../Diagnostics_functions.ino | 533 ----- .../FlashDiagnostic_functions.ino | 863 +++++++ .../FlashDiagnostics/FlashDiagnostics.ino | 140 +- examples/Struct_writer/.trinket.test.skip | 1 + examples/Struct_writer/Struct_writer.ino | 30 +- examples/TestFlash/.trinket.test.skip | 1 + examples/TestFlash/TestFlash.ino | 227 +- examples/getAddressEx/.trinket.test.skip | 1 + examples/getAddressEx/getAddressEx.ino | 1 - examples/readWriteString/.trinket.test.skip | 1 + examples/readWriteString/readWriteString.ino | 45 +- .../tinyFlashDiagnostics/Attiny_functions.ino | 279 +++ .../NonAttiny_functions.ino | 463 ++++ examples/tinyFlashDiagnostics/Notes.cpp | 46 + .../tinyFlashDiagnostics.ino | 80 + examples/tinyFlashDiagnostics/vars.h | 96 + examples/tinyFlashTester/tinyFlashTester.ino | 69 + extras/Changes.log | 59 - extras/DMASAMD.cpp | 158 -- extras/Library speed comparisons.xlsx | Bin 48504 -> 47493 bytes extras/SPI pinouts.md | 0 ...nbond Flash Instructions - Comparison.xlsx | Bin keywords.txt | 27 +- library.properties | 4 +- src/DMASAMD.h | 140 -- src/{SAM_DMASPI.cpp => DMASPI.cpp} | 0 src/FLASHIO.cpp | 491 ---- src/SPIFlash.cpp | 2061 ++++++++++++----- src/SPIFlash.h | 522 ++--- src/defines.h | 135 +- src/tinySPI.cpp | 53 + src/troubleshoot.cpp | 258 ++- 35 files changed, 4185 insertions(+), 2612 deletions(-) create mode 100644 examples/FlashDiagnostics/.trinket.test.skip delete mode 100644 examples/FlashDiagnostics/Diagnostics_functions.ino create mode 100644 examples/FlashDiagnostics/FlashDiagnostic_functions.ino create mode 100644 examples/Struct_writer/.trinket.test.skip create mode 100644 examples/TestFlash/.trinket.test.skip create mode 100644 examples/getAddressEx/.trinket.test.skip create mode 100644 examples/readWriteString/.trinket.test.skip create mode 100644 examples/tinyFlashDiagnostics/Attiny_functions.ino create mode 100644 examples/tinyFlashDiagnostics/NonAttiny_functions.ino create mode 100644 examples/tinyFlashDiagnostics/Notes.cpp create mode 100644 examples/tinyFlashDiagnostics/tinyFlashDiagnostics.ino create mode 100644 examples/tinyFlashDiagnostics/vars.h create mode 100644 examples/tinyFlashTester/tinyFlashTester.ino mode change 100755 => 100644 extras/Changes.log delete mode 100644 extras/DMASAMD.cpp mode change 100755 => 100644 extras/Library speed comparisons.xlsx mode change 100755 => 100644 extras/SPI pinouts.md mode change 100755 => 100644 extras/Winbond Flash Instructions - Comparison.xlsx mode change 100755 => 100644 keywords.txt delete mode 100644 src/DMASAMD.h rename src/{SAM_DMASPI.cpp => DMASPI.cpp} (100%) delete mode 100644 src/FLASHIO.cpp create mode 100644 src/tinySPI.cpp diff --git a/.travis.yml b/.travis.yml index 71b4eaf..261a945 100644 --- a/.travis.yml +++ b/.travis.yml @@ -4,6 +4,7 @@ before_install: - source <(curl -SLs https://raw.githubusercontent.com/Marzogh/Travis-CI/master/install.sh) script: - build_main_platforms + - build_platform trinket - build_platform rtl8195a #- build_platform uno #- build_platform due @@ -15,7 +16,6 @@ script: #- build_platform mega #- build_platform fio #- build_platform micro - #- build_platform trinket notifications: email: on_success: change diff --git a/README.md b/README.md index 312e006..944efb7 100644 --- a/README.md +++ b/README.md @@ -1,9 +1,4 @@ -# SPIFlash [![Build Status](https://travis-ci.org/Marzogh/SPIFlash.svg?branch=dev)](https://travis-ci.org/Marzogh/SPIFlash) [![DOI](https://zenodo.org/badge/18908/Marzogh/SPIFlash.svg)](https://zenodo.org/badge/latestdoi/18908/Marzogh/SPIFlash) -[![GitHub release](https://img.shields.io/github/release/Marzogh/SPIFlash.svg)](https://github.com/Marzogh/SPIFlash) -[![GitHub commits](https://img.shields.io/github/commits-since/Marzogh/SPIFlash/v2.7.0.svg)](https://github.com/Marzogh/SPIFlash/compare/v2.7.0...development) -[![GitHub issues](https://img.shields.io/github/issues/Marzogh/SPIFlash.svg)](https://github.com/Marzogh/SPIFlash) -[![GitHub pull requests](https://img.shields.io/github/issues-pr/Marzogh/SPIFlash.svg)](https://github.com/Marzogh/SPIFlash) -[![license](https://img.shields.io/github/license/Marzogh/SPIFlash.svg)](https://github.com/Marzogh/SPIFlash) +# SPIFlash [![Build Status](https://travis-ci.org/Marzogh/SPIFlash.svg?branch=master)](https://travis-ci.org/Marzogh/SPIFlash) [![DOI](https://zenodo.org/badge/18908/Marzogh/SPIFlash.svg)](https://zenodo.org/badge/latestdoi/18908/Marzogh/SPIFlash) ### Arduino library for Winbond Flash Memory Chips Download the latest stable release (v2.6.0) from here. Please report any bugs in issues. @@ -18,7 +13,7 @@ This Arduino library is for use with Winbond serial flash memory chips. In its c - IDE v1.5.x - IDE v1.6.0-v1.6.5 - IDE v1.6.9-v1.6.12 -- IDE v1.8.1-v1.8.3 +- IDE v1.8.2 ##### Boards @@ -33,6 +28,7 @@ This Arduino library is for use with Winbond serial flash memory chips. In its c - Arduino Mega - Arduino Micro - Arduino Fio +- Attiny85 Boards ###### In BETA - ESP32 Boards (Tested on the Adafruit Esp32 Feather) The library is known to work with the ESP32 core as of the current commit 9618eec on 02.08.2017. ```ESP32 support will remain in beta till the ESP32 core can be installed via the Arduino boards manager.``` diff --git a/examples/FlashDiagnostics/.trinket.test.skip b/examples/FlashDiagnostics/.trinket.test.skip new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/examples/FlashDiagnostics/.trinket.test.skip @@ -0,0 +1 @@ + diff --git a/examples/FlashDiagnostics/Diagnostics_functions.ino b/examples/FlashDiagnostics/Diagnostics_functions.ino deleted file mode 100644 index b5d8e1a..0000000 --- a/examples/FlashDiagnostics/Diagnostics_functions.ino +++ /dev/null @@ -1,533 +0,0 @@ -/* - |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| - | FlashDiagnostic_functions.ino | - | SPIFlash library | - | v 3.0.0 | - |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| - | Marzogh | - | 04.11.2017 | - |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| - | | - | For a full diagnostics rundown - with error codes and details of the errors | - | uncomment #define RUNDIAGNOSTIC in SPIFlash.cpp in the library before compiling | - | and loading this application onto your Arduino. | - | | - |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| -*/ -void printLine() { - Serial.println(); - for (uint8_t i = 0; i < 230; i++) { - Serial.print(F("-")); - } - Serial.println(); -} - -void clearprintBuffer(char *bufPtr) { - for (uint8_t i = 0; i < 128; i++) { - //printBuffer[i] = 0; - *bufPtr++ = 0; - } -} - -void printTime(uint32_t _wTime, uint32_t _rTime) { - if (_rTime != 0) { - Serial.print(F("\t\tWrite Time: ")); - printTimer(_wTime); - Serial.print(F(",\tRead Time: ")); - printTimer(_rTime); - } - else { - Serial.print(F("\t\tTime: ")); - printTimer(_wTime); - } -} - -void printTimer(uint32_t _us) { - - if (_us > 1000000) { - float _s = _us / (float)1000000; - Serial.print(_s, 3); - Serial.print(" s"); - } - else if (_us > 1000) { - float _ms = _us / (float)1000; - Serial.print(_ms, 3); - Serial.print(" ms"); - } - else { - Serial.print(_us); - Serial.print(F(" us")); - } - delay(20); -} - -void pass(bool _status) { - Serial.print(F("\tData I/O test ")); - if (_status) { - Serial.print(F("PASS\t")); - } - else { - Serial.print(F("FAIL\t")); - } -} - -void printUniqueID(void) { - Serial.print("Unique ID: "); - long long _uniqueID = flash.getUniqueID(); - Serial.print(uint32_t(_uniqueID/1000000L)); - Serial.print(uint32_t(_uniqueID%1000000L)); - Serial.print(", "); - Serial.print(uint32_t(_uniqueID >> 32), HEX); - Serial.print(uint32_t(_uniqueID), HEX); -} - -void getID() { - char printBuffer[128]; - printLine(); - for (uint8_t i = 0; i < 68; i++) { - Serial.print(F(" ")); - } - Serial.print(F("SPIFlash Library version")); -#ifdef LIBVER - uint8_t _ver, _subver, _bugfix; - flash.libver(&_ver, &_subver, &_bugfix); - clearprintBuffer(&printBuffer[1]); - sprintf(printBuffer, ": %d.%d.%d", _ver, _subver, _bugfix); - Serial.print(printBuffer); -#else - Serial.print(F("< 2.5.0")); -#endif - printLine(); - - for (uint8_t i = 0; i < 80; i++) { - Serial.print(F(" ")); - } - Serial.print(F("Get ID")); - printLine(); - uint8_t b1, b2; - //uint16_t b3; - uint32_t JEDEC = flash.getJEDECID(); - uint32_t maxPage = flash.getMaxPage(); - uint32_t capacity = flash.getCapacity(); - b1 = (JEDEC >> 16); - b2 = (JEDEC >> 8); - //b3 = (JEDEC >> 0); - - - printLine(); - //---------------------------------------------------------------------------------------------// - - clearprintBuffer(&printBuffer[1]); - #if defined (ARDUINO_ARCH_ESP32) - sprintf(printBuffer, "\t\t\tJEDEC ID: %04xh", JEDEC); - #else - sprintf(printBuffer, "\t\t\tJEDEC ID: %04lxh", JEDEC); - #endif - Serial.println(printBuffer); - //Serial.print(F("\t\t\tJEDEC ID: ")); - //Serial.print(JEDEC, HEX); - //Serial.println(F("xh")); - clearprintBuffer(&printBuffer[1]); - #if defined (ARDUINO_ARCH_ESP32) - sprintf(printBuffer, "\t\t\tManufacturer ID: %02xh\n\t\t\tMemory Type: %02xh\n\t\t\tCapacity: %u bytes\n\t\t\tMaximum pages: %u", b1, b2, capacity, maxPage); - #else - sprintf(printBuffer, "\t\t\tManufacturer ID: %02xh\n\t\t\tMemory Type: %02xh\n\t\t\tCapacity: %lu bytes\n\t\t\tMaximum pages: %lu", b1, b2, capacity, maxPage); - #endif - Serial.print(printBuffer); - Serial.print("\n\t\t\t"); - printUniqueID(); - printLine(); -} - -void byteTest() { - Serial.println(); - uint32_t wTime = 0; - uint32_t rTime, addr; - uint8_t _data, _d; - _d = 35; - - addr = random(0, 0xFFFFF); - - if (flash.writeByte(addr, _d)) { - wTime = flash.functionRunTime(); - } - - - _data = flash.readByte(addr); - rTime = flash.functionRunTime(); - - Serial.print ("\t\t\tByte: \t\t"); - if (_data == _d) { - pass(TRUE); - } - else { - pass(FALSE); - } - printTime(wTime, rTime); -} - -void charTest() { - Serial.println(); - uint32_t wTime = 0; - uint32_t rTime, addr; - int8_t _data, _d; - _d = -110; - - addr = random(0, 0xFFFFF); - - if (flash.writeChar(addr, _d)) { - wTime = flash.functionRunTime(); - } - - - _data = flash.readChar(addr); - rTime = flash.functionRunTime(); - - - Serial.print ("\t\t\tChar: \t\t"); - if (_data == _d) { - pass(TRUE); - } - else { - pass(FALSE); - } - printTime(wTime, rTime); -} - -void wordTest() { - Serial.println(); - uint32_t wTime = 0; - uint32_t rTime, addr; - uint16_t _data, _d; - _d = 4520; - - addr = random(0, 0xFFFFF); - - if (flash.writeWord(addr, _d)) { - wTime = flash.functionRunTime(); - } - - - _data = flash.readWord(addr); - rTime = flash.functionRunTime(); - - - Serial.print ("\t\t\tWord: \t\t"); - if (_data == _d) { - pass(TRUE); - } - else { - pass(FALSE); - } - printTime(wTime, rTime); -} - -void shortTest() { - Serial.println(); - uint32_t wTime = 0; - uint32_t rTime, addr; - int16_t _data, _d; - _d = -1250; - - addr = random(0, 0xFFFFF); - - if (flash.writeShort(addr, _d)) { - wTime = flash.functionRunTime(); - } - - - _data = flash.readShort(addr); - rTime = flash.functionRunTime(); - - - Serial.print ("\t\t\tShort: \t\t"); - if (_data == _d) { - pass(TRUE); - } - else { - pass(FALSE); - } - printTime(wTime, rTime); -} - -void uLongTest() { - Serial.println(); - uint32_t wTime = 0; - uint32_t rTime, addr; - uint32_t _data, _d; - _d = 876532; - - addr = random(0, 0xFFFFF); - - if (flash.writeULong(addr, _d)) { - wTime = flash.functionRunTime(); - } - - - _data = flash.readULong(addr); - rTime = flash.functionRunTime(); - - - Serial.print ("\t\t\tULong: \t\t"); - if (_data == _d) { - pass(TRUE); - } - else { - pass(FALSE); - } - printTime(wTime, rTime); -} - -void longTest() { - Serial.println(); - uint32_t wTime = 0; - uint32_t rTime, addr; - int32_t _data, _d; - _d = -10959; - - addr = random(0, 0xFFFFF); - - if (flash.writeLong(addr, _d)) { - wTime = flash.functionRunTime(); - } - - - _data = flash.readLong(addr); - rTime = flash.functionRunTime(); - - - Serial.print ("\t\t\tLong: \t\t"); - if (_data == _d) { - pass(TRUE); - } - else { - pass(FALSE); - } - printTime(wTime, rTime); -} - -void floatTest() { - Serial.println(); - uint32_t wTime = 0; - uint32_t rTime, addr; - float _data, _d; - _d = 3.14; - - addr = random(0, 0xFFFFF); - - if (flash.writeFloat(addr, _d)) { - wTime = flash.functionRunTime(); - } - - - _data = flash.readFloat(addr); - rTime = flash.functionRunTime(); - - - Serial.print ("\t\t\tFloat: \t\t"); - if (_data == _d) { - pass(TRUE); - } - else { - pass(FALSE); - } - printTime(wTime, rTime); -} - -void stringTest() { - uint32_t wTime = 0; - uint32_t rTime, addr; - String _data, _d; - _d = "This is a test String 123!@#"; - - addr = random(0, 0xFFFFF); - - if (flash.writeStr(addr, _d)) { - wTime = flash.functionRunTime(); - } - - flash.readStr(addr, _data); - rTime = flash.functionRunTime(); - - - Serial.print ("\t\t\tString: \t"); - if (_data == _d) { - pass(TRUE); - } - else { - pass(FALSE); - } - printTime(wTime, rTime); - -#if defined (ARDUINO_ARCH_SAM) || defined (ARDUINO_ARCH_ESP8266) - Serial.println(); - printLine(); - if (!flash.functionRunTime()) { - Serial.println(F("Please uncomment RUNDIAGNOSTIC in SPIFlash.h to see the time taken by each function to run.")); - } -#endif -} - -void structTest() { - Serial.println(); - struct Test { - uint16_t s1; - float s2; - int32_t s3; - bool s4; - uint8_t s5; - }; - Test _d; - Test _data; - - _d.s1 = 31325; - _d.s2 = 4.84; - _d.s3 = 880932; - _d.s4 = true; - _d.s5 = 5; - - uint32_t wTime = 0; - uint32_t addr, rTime; - - addr = random(0, 0xFFFFF); - - if (flash.writeAnything(addr, _d)) { - wTime = flash.functionRunTime(); - } - - - flash.readAnything(addr, _data); - rTime = flash.functionRunTime(); - - - Serial.print ("\t\t\tStruct: \t"); - if ((_d.s1 == _data.s1) && (_d.s2 == _data.s2) && (_d.s3 == _data.s3) && (_d.s4 == _data.s4) && (_d.s5 == _data.s5)) { - pass(TRUE); - } - else { - pass(FALSE); - } - printTime(wTime, rTime); -} - -void arrayTest() { - Serial.println(); - uint32_t wTime = 0; - uint32_t rTime, addr; - uint8_t _d[256], _data[256]; - - for (uint16_t i = 0; i < 256; i++) { - _d[i] = i; - } - - addr = random(0, 0xFFFFF); - - if (flash.writeByteArray(addr, _d, 256)) { - wTime = flash.functionRunTime(); - } - - - flash.readByteArray(addr, _data, 256); - rTime = flash.functionRunTime(); - - - Serial.print ("\t\t\tByte Array: \t"); - for (uint16_t i = 0; i < 256; i++) { - if (_data[i] != i) { - pass(FALSE); - } - } - pass(TRUE); - printTime(wTime, rTime); - Serial.println(); -} - -void powerDownTest() { - Serial.println(); - uint32_t _time; - Serial.print(F("\t\t\tPower Down: \t")); - if (flash.powerDown()) { - _time = flash.functionRunTime(); - pass(TRUE); - printTime(_time, 0); - } - else { - pass(FALSE); - Serial.print(F("\t\tNot all chips support power down. Please check your datasheet.")); - } -} - -void powerUpTest() { - Serial.println(); - uint32_t _time; - Serial.print(F("\t\t\tPower Up: \t")); - if (flash.powerUp()) { - _time = flash.functionRunTime(); - pass(TRUE); - printTime(_time, 0); - } - else { - pass(FALSE); - } -} - -void eraseSectorTest() { - Serial.println(); - uint32_t _time, _addr; - _addr = random(0, 0xFFFFF); - Serial.print(F("\t\t\tErase 4KB Sector: ")); - if (flash.eraseSector(_addr)) { - _time = flash.functionRunTime(); - pass(TRUE); - printTime(_time, 0); - } - else { - pass(FALSE); - } - Serial.println(); -} - -void eraseBlock32KTest() { - Serial.println(); - uint32_t _time, _addr; - _addr = random(0, 0xFFFFF); - Serial.print(F("\t\t\tErase 32KB Block: ")); - if (flash.eraseBlock32K(_addr)) { - _time = flash.functionRunTime(); - pass(TRUE); - printTime(_time, 0); - } - else { - pass(FALSE); - } -} - -void eraseBlock64KTest() { - Serial.println(); - uint32_t _time, _addr; - _addr = random(0, 0xFFFFF); - Serial.print(F("\t\t\tErase 64KB Block: ")); - if (flash.eraseBlock64K(_addr)) { - _time = flash.functionRunTime(); - pass(TRUE); - printTime(_time, 0); - } - else { - pass(FALSE); - } -} - -void eraseChipTest() { - Serial.println(); - uint32_t _time; - Serial.print(F("\t\t\tErase Chip: \t")); - if (flash.eraseChip()) { - _time = flash.functionRunTime(); - pass(TRUE); - printTime(_time, 0); - } - else { - pass(FALSE); - } -} - diff --git a/examples/FlashDiagnostics/FlashDiagnostic_functions.ino b/examples/FlashDiagnostics/FlashDiagnostic_functions.ino new file mode 100644 index 0000000..399a231 --- /dev/null +++ b/examples/FlashDiagnostics/FlashDiagnostic_functions.ino @@ -0,0 +1,863 @@ +/* + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | FlashDiagnostic_functions.ino | + | SPIFlash library | + | v 2.6.0 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Marzogh | + | 13.11.2016 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | | + | For a full diagnostics rundown - with error codes and details of the errors | + | uncomment #define RUNDIAGNOSTIC in SPIFlash.cpp in the library before compiling | + | and loading this application onto your Arduino. | + | | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| +*/ + +void getID() { + char printBuffer[128]; + printLine(); + for (uint8_t i = 0; i < 68; i++) { + Serial.print(F(" ")); + } + Serial.print(F("SPIFlash Library version")); +#ifdef LIBVER + uint8_t _ver, _subver, _bugfix; + flash.libver(&_ver, &_subver, &_bugfix); + clearprintBuffer(&printBuffer[1]); + sprintf(printBuffer, ": %d.%d.%d", _ver, _subver, _bugfix); + Serial.println(printBuffer); +#else + Serial.println(F("< 2.5.0")); +#endif + printLine(); + + for (uint8_t i = 0; i < 80; i++) { + Serial.print(F(" ")); + } + Serial.println(F("Get ID")); + printLine(); + uint8_t b1, b2; + uint16_t b3; + uint32_t JEDEC = flash.getJEDECID(); + uint32_t maxPage = flash.getMaxPage(); + uint32_t capacity = flash.getCapacity(); + b1 = (JEDEC >> 16); + b2 = (JEDEC >> 8); + b3 = (JEDEC >> 0); + + + printLine(); + //---------------------------------------------------------------------------------------------// + + clearprintBuffer(&printBuffer[1]); + sprintf(printBuffer, "\t\t\tJEDEC ID: %04lxh", JEDEC); + Serial.println(printBuffer); + clearprintBuffer(&printBuffer[1]); + sprintf(printBuffer, "\t\t\tManufacturer ID: %02xh\n\t\t\tMemory Type: %02xh\n\t\t\tCapacity: %lu bytes\n\t\t\tMaximum pages: %lu", b1, b2, capacity, maxPage); + Serial.println(printBuffer); +} + +bool checkPage(uint8_t *data_buffer) { + for (int i = 0; i < 256; i++) { + if (data_buffer[i] != i) + return false; + } + return true; +} + +void diagnose() { + printLine(); + for (uint8_t i = 0; i < 79; i++) { + Serial.print(F(" ")); + } + Serial.println(F("Data Check")); + printLine(); + + Serial.println(F("\tData Written\t||\tData Read\t||\tResult\t\t||\tWrite Time\t||\tRead Time\t||\tWrite Time\t||\tFast Read Time")); + Serial.print(F("\t\t\t||\t\t\t||\t\t\t||\t\t\t||\t\t\t||")); + Serial.println(F("\t(No Error Chk)\t||")); + printLine(); + byteDiag(); + charDiag(); + wordDiag(); + shortDiag(); + uLongDiag(); + longDiag(); + floatDiag(); + stringDiag(); + structDiag(); + pageDiag(); + powerFuncDiag(); + +} + +void byteDiag(void) { + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Byte // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + uint8_t _byte = 35; + uint8_t _b; + float startTime; + uint32_t addr, wTime, rTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeByte(addr, _byte); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _b = flash.readByte(addr); + rTime = micros() - startTime; + //Print result + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(_byte); + printTab(2, 1); + Serial.print(_b); + printTab(2, 1); + if (_byte == _b) + printPass(); + else + printFail(); + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeByte(addr, _byte, false); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _b = flash.readByte(addr, true); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void charDiag(void) { + //-----------------------------------------------------------------------------------------------------------------------------------------------------// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Char // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + int8_t _char = -110; + int8_t _c; + float startTime; + uint32_t addr, wTime, rTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeChar(addr, _char); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _c = flash.readChar(addr); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(_char); + printTab(2, 1); + Serial.print(_c); + printTab(2, 1); + if (_char == _c) + printPass(); + else + printFail(); + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeChar(addr, _char, false); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _c = flash.readChar(addr, true); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void wordDiag(void) { + //-----------------------------------------------------------------------------------------------------------------------------------------------------// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Word // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + uint16_t _word = 4520; + uint16_t _w; + float startTime; + uint32_t addr, wTime, rTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeWord(addr, _word); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _w = flash.readWord(addr); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(_word); + printTab(2, 1); + Serial.print(_w); + printTab(2, 1); + if (_word == _w) + printPass(); + else + printFail(); + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeWord(addr, _word, false); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _w = flash.readWord(addr, true); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void shortDiag(void) { + //-----------------------------------------------------------------------------------------------------------------------------------------------------// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Short // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + int16_t _short = -1250; + int16_t _s; + float startTime; + uint32_t addr, wTime, rTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeShort(addr, _short); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _s = flash.readShort(addr); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(_short); + printTab(2, 1); + Serial.print(_s); + printTab(2, 1); + if (_short == _s) + printPass(); + else + printFail(); + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeShort(addr, _short, false); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _s = flash.readShort(addr, true); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void uLongDiag(void) { + //-----------------------------------------------------------------------------------------------------------------------------------------------------// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Ulong // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + uint32_t _uLong = 876532; + uint32_t _uL; + float startTime; + uint32_t addr, wTime, rTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeULong(addr, _uLong); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _uL = flash.readULong(addr); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(_uLong); + printTab(2, 1); + Serial.print(_uL); + printTab(2, 1); + if (_uLong == _uL) + printPass(); + else + printFail(); + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeULong(addr, _uLong, false); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _uL = flash.readULong(addr, true); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void longDiag(void) { + //-----------------------------------------------------------------------------------------------------------------------------------------------------// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Long // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + int32_t _long = -10959; + int32_t _l; + float startTime; + uint32_t addr, wTime, rTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeLong(addr, _long); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _l = flash.readLong(addr); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(_long); + printTab(2, 1); + Serial.print(_l); + printTab(2, 1); + if (_long == _l) + printPass(); + else + printFail(); + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeLong(addr, _long, false); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _l = flash.readLong(addr, true); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void floatDiag(void) { + //-----------------------------------------------------------------------------------------------------------------------------------------------------// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Float // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + float _float = 3.1415; + float _f; + float startTime; + uint32_t addr, wTime, rTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + if (flash.writeFloat(addr, _float)) { + wTime = micros() - startTime; + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _f = flash.readFloat(addr); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(_float); + printTab(2, 1); + Serial.print(_f); + printTab(2, 1); + if (_float == _f) + printPass(); + else + printFail(); + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + if (flash.writeFloat(addr, _float, false)) { + wTime = micros() - startTime; + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + _f = flash.readFloat(addr, true); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void stringDiag(void) { + //-----------------------------------------------------------------------------------------------------------------------------------------------------// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // String // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + String _string = "123 Test !@#"; + String _str = ""; + float startTime; + uint32_t addr, wTime, rTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + if (flash.writeStr(addr, _string)) { + wTime = micros() - startTime; + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + if (flash.readStr(addr, _str)) { + rTime = micros() - startTime; + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(_string); + printTab(1, 1); + Serial.print(_str); + printTab(1, 1); + if (_string == _str) + printPass(); + else + printFail(); + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + if (flash.writeStr(addr, _string, false)) { + wTime = micros() - startTime; + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + if (flash.readStr(addr, _str, true)) { + rTime = micros() - startTime; + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void structDiag(void) { + //-----------------------------------------------------------------------------------------------------------------------------------------------------// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Struct // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + struct Test { + word s1; + float s2; + long s3; + bool s4; + byte s5; + }; + Test inputStruct; + Test outputStruct; + + inputStruct.s1 = 31325; + inputStruct.s2 = 4.84; + inputStruct.s3 = 880932; + inputStruct.s4 = true; + inputStruct.s5 = 5; + + float startTime; + uint32_t addr, wTime, rTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeAnything(addr, inputStruct); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + flash.readAnything(addr, outputStruct); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("inputStruct")); + printTab(1, 1); + Serial.print(F("outputStruct")); + printTab(1, 1); + if (inputStruct.s1 == outputStruct.s1 && inputStruct.s2 == outputStruct.s2 && inputStruct.s3 == outputStruct.s3 && inputStruct.s4 == outputStruct.s4 && inputStruct.s5 == outputStruct.s5) + printPass(); + else + printFail(); + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, 0xFFFFF); + startTime = micros(); + flash.writeAnything(addr, inputStruct, false); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + startTime = micros(); + flash.readAnything(addr, outputStruct, true); + rTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void pageDiag(void) { + //-----------------------------------------------------------------------------------------------------------------------------------------------------// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Page // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + float startTime; + uint32_t addr, wTime, rTime; + uint8_t pageBuffer[PAGESIZE]; + + for (int i = 0; i < PAGESIZE; ++i) { + pageBuffer[i] = i; + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + addr = random(0, flash.getMaxPage()); + startTime = micros(); + while (!flash.writeByteArray(addr, pageBuffer, PAGESIZE)); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read & Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + for (int i = 0; i < PAGESIZE; ++i) { + pageBuffer[i] = 0; + } + startTime = micros(); + printTab(1, 0); + Serial.print(F("0 - ")); + Serial.print(PAGESIZE - 1); + printTab(2, 1); + startTime = micros(); + + flash.readByteArray(addr, pageBuffer, PAGESIZE); + rTime = micros() - startTime; + bool _pass; + for (uint16_t i = 0; i < 256; i++) { + if (pageBuffer [i] != i) { + _pass = false; + break; + } + else { + _pass = true; + } + } + + if (_pass) { + Serial.print(F("0 - ")); + Serial.print(PAGESIZE - 1); + printTab(2, 1); + printPass(); + } + else { + Serial.print(F("Unknown")); + printTab(2, 1); + printFail(); + } + printTime(wTime, rTime); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write (No Error) // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + for (int i = 0; i < 256; ++i) { + pageBuffer[i] = i; + } + addr = random(0, flash.getMaxPage()); + startTime = micros(); + flash.writeByteArray(addr, pageBuffer, PAGESIZE, false); + wTime = micros() - startTime; + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Fast Read & Print Result // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + for (int i = 0; i < 256; ++i) { + pageBuffer[i] = 0; + } + startTime = micros(); + flash.readByteArray(addr, pageBuffer, PAGESIZE, true); + rTime = micros() - startTime; + printTime(wTime, rTime); + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void powerFuncDiag(void) { + String _string = "123 Test !@#"; + float wTime; + + printLine(); + for (uint8_t i = 0; i < 72; i++) { + Serial.print(" "); + } + Serial.println(F("Check Other Functions")); + printLine(); + Serial.println(F("\t\t\t\t\tFunction\t\t||\t\tResult\t\t\t||\t\tTime")); + printLine(); + Serial.flush(); + + uint32_t capacity = flash.getCapacity(); + if (!Serial) + Serial.begin(115200); + uint32_t stringAddress1 = random(0, capacity); + uint32_t stringAddress2 = random(0, capacity); + uint32_t stringAddress3 = random(0, capacity); + + printTab(5, 0); + Serial.print(F("powerDown")); + printTab(2, 2); + //if (flash.writeStr(stringAddress1, _string)) { + wTime = micros(); + if (flash.powerDown()) { + wTime = micros() - wTime; + printPass(); + } + else { + wTime = micros() - wTime; + printFail(); + } + //} + printTab(3, 2); + printTimer(wTime); + Serial.println(); + + printTab(5, 0); + Serial.print(F("powerUp")); + printTab(3, 2); + wTime = micros(); + if (flash.powerUp()) { + wTime = micros() - wTime; + //if (flash.writeStr(stringAddress3, _string)) { + printPass(); + } + else { + printFail(); + } + //} + printTab(3, 2); + printTimer(wTime); + Serial.println(); + + printTab(5, 0); + Serial.print(F("eraseSector")); + wTime = micros(); + printTab(2, 2); + if (flash.eraseSector(stringAddress1)) { + wTime = micros() - wTime; + printPass(); + } + else { + printFail(); + } + wTime = wTime / 3; + printTab(3, 2); + printTimer(wTime); + Serial.println(); + + printTab(5, 0); + Serial.print(F("eraseBlock32K")); + wTime = micros(); + printTab(2, 2); + if (flash.eraseBlock32K(stringAddress2)) { + wTime = micros() - wTime; + printPass(); + } + else { + printFail(); + } + wTime = wTime / 3; + printTab(3, 2); + printTimer(wTime); + Serial.println(); + + printTab(5, 0); + Serial.print(F("eraseBlock64K")); + wTime = micros(); + printTab(2, 2); + if (flash.eraseBlock64K(stringAddress3)) { + wTime = micros() - wTime; + printPass(); + } + else { + printFail(); + } + wTime = wTime / 3; + printTab(3, 2); + printTimer(wTime); + Serial.println(); + + printTab(5, 0); + Serial.print(F("eraseChip")); + printTab(2, 2); + wTime = micros(); + if (flash.eraseChip()) { + wTime = micros() - wTime; + printPass(); + } + else { + printFail(); + } + + + printTab(3, 2); + printTimer(wTime); + Serial.println(); + + printLine(); +} diff --git a/examples/FlashDiagnostics/FlashDiagnostics.ino b/examples/FlashDiagnostics/FlashDiagnostics.ino index f153eab..c58dfbb 100644 --- a/examples/FlashDiagnostics/FlashDiagnostics.ino +++ b/examples/FlashDiagnostics/FlashDiagnostics.ino @@ -2,10 +2,10 @@ |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| | FlashDiagnostics.ino | | SPIFlash library | - | v 3.0.0 | + | v 2.6.0 | |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| | Marzogh | - | 04.11.2017 | + | 16.04.2017 | |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| | | | For a full diagnostics rundown - with error codes and details of the errors | @@ -17,94 +17,110 @@ #include +//Define a flash memory size (if using non-Winbond memory) according to the list in defines.h +//#define CHIPSIZE MB64 + #if defined(ARDUINO_SAMD_ZERO) && defined(SERIAL_PORT_USBVIRTUAL) // Required for Serial on Zero based boards #define Serial SERIAL_PORT_USBVIRTUAL #endif #if defined (SIMBLEE) - #define BAUD_RATE 250000 - #define RANDPIN 1 +#define BAUD_RATE 250000 +#define RANDPIN 1 #else - #define BAUD_RATE 115200 - #if defined(ARCH_STM32) - #define RANDPIN PA0 - #else - #define RANDPIN A0 - #endif +#define BAUD_RATE 115200 +#define RANDPIN A0 #endif -#define TRUE 1 -#define FALSE 0 - -//SPIFlash flash(SS1, &SPI1); //Use this constructor if using an SPI bus other than the default SPI. Only works with chips with more than one hardware SPI bus SPIFlash flash; void setup() { Serial.begin(BAUD_RATE); - #if defined (ARDUINO_ARCH_SAMD) || (__AVR_ATmega32U4__) || defined(ARCH_STM32) - while (!Serial) ; // Wait for Serial monitor to open - #endif - delay(50); //Time to terminal get connected +#if defined (ARDUINO_ARCH_SAMD) || (__AVR_ATmega32U4__) + while (!Serial) ; // Wait for Serial monitor to open +#endif Serial.print(F("Initialising Flash memory")); - for (uint8_t i = 0; i < 10; ++i) + for (int i = 0; i < 10; ++i) { Serial.print(F(".")); } Serial.println(); - randomSeed(analogRead(RANDPIN)); +#if defined (CHIPSIZE) + flash.begin(CHIPSIZE); //use flash.begin(CHIPSIZE) if using non-Winbond flash (Refer to '#define CHIPSIZE' above) +#else flash.begin(); - //To use a custom flash memory size (if using memory from manufacturers not officially supported by the library) - declare a size variable according to the list in defines.h - //flash.begin(MB(1)); +#endif Serial.println(); Serial.println(); - + randomSeed(analogRead(RANDPIN)); getID(); - eraseChipTest(); - eraseBlock64KTest(); - eraseBlock32KTest(); - eraseSectorTest(); - -#if defined(ARDUINO_ARCH_SAM) || defined(ARDUINO_ARCH_ESP8266) - delay(10); - powerDownTest(); - powerUpTest(); - Serial.println(); -#endif + diagnose(); +} - byteTest(); - charTest(); - wordTest(); - shortTest(); - uLongTest(); -#if defined(ARDUINO_ARCH_SAM) || defined(ARDUINO_ARCH_ESP8266) - delay(10); -#endif - longTest(); - floatTest(); - structTest(); - arrayTest(); - stringTest(); - -#if !defined(ARDUINO_ARCH_SAM) || !defined(ARDUINO_ARCH_ESP8266) - powerDownTest(); - powerUpTest(); -#endif - printLine(); - if (!flash.functionRunTime()) { - Serial.println(F("Please uncomment RUNDIAGNOSTIC in SPIFlash.h to see the time taken by each function to run.")); +void loop() { + +} + + +//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~Serial Print Functions~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + +void clearprintBuffer(char *bufPtr) +{ + for (uint8_t i = 0; i < 128; i++) { + //printBuffer[i] = 0; + *bufPtr++ = 0; } } -void loop() { +void printLine() { + for (uint8_t i = 0; i < 230; i++) { + Serial.print(F("-")); + } + Serial.println(); +} +void printPass() { + Serial.print(F("Pass")); } -void longBlink() { - pinMode(LED_BUILTIN, OUTPUT); - digitalWrite(LED_BUILTIN, HIGH); - delay(3000); - digitalWrite(LED_BUILTIN, LOW); - delay(2000); +void printFail() { + Serial.print(F("Fail")); } +void printTab(uint8_t a, uint8_t b) { + for (uint8_t i = 0; i < a; i++) { + Serial.print(F("\t")); + } + if (b > 0) { + Serial.print("||"); + for (uint8_t i = 0; i < b; i++) { + Serial.print(F("\t")); + } + } +} + +void printTime(uint32_t _wTime, uint32_t _rTime) { + printTab(2, 1); + printTimer(_wTime); + printTab(2, 1); + printTimer(_rTime); +} + +void printTimer(uint32_t _us) { + + if (_us > 1000000) { + float _s = _us / (float)1000000; + Serial.print(_s, 4); + Serial.print(" s"); + } + else if (_us > 10000) { + float _ms = _us / (float)1000; + Serial.print(_ms, 4); + Serial.print(" ms"); + } + else { + Serial.print(_us); + Serial.print(F(" us")); + } +} diff --git a/examples/Struct_writer/.trinket.test.skip b/examples/Struct_writer/.trinket.test.skip new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/examples/Struct_writer/.trinket.test.skip @@ -0,0 +1 @@ + diff --git a/examples/Struct_writer/Struct_writer.ino b/examples/Struct_writer/Struct_writer.ino index 23fd488..26f1129 100644 --- a/examples/Struct_writer/Struct_writer.ino +++ b/examples/Struct_writer/Struct_writer.ino @@ -45,7 +45,6 @@ -//SPIFlash flash(SS1, &SPI1); //Use this constructor if using an SPI bus other than the default SPI. Only works with chips with more than one hardware SPI bus SPIFlash flash; @@ -55,7 +54,6 @@ struct Configuration { float RLDR; // Resistance calculation of potential divider with LDR bool light; uint8_t adc; - uint8_t arr[8]; }; Configuration configuration; @@ -83,17 +81,12 @@ void setup() { configuration.RLDR = 889.32; configuration.light = true; configuration.adc = 5; - for (uint8_t i = 0; i < 8; i++) { - configuration.arr[i] = i; - } #endif #ifdef SENSOR readLDR(); #endif - if (flash.eraseChip()) { - Serial.println("Chip has been erased"); - } + if (flash.writeAnything(_addr, configuration)) Serial.println ("Data write successful"); else @@ -104,20 +97,13 @@ void setup() { Serial.println(configuration.RLDR); Serial.println(configuration.light); Serial.println(configuration.adc); - for (uint8_t i = 0; i < 8; i++) { - Serial.print(configuration.arr[i]); - Serial.print(", "); - } - Serial.println(); + Serial.println("Saved!"); configuration.lux = 0; configuration.vOut = 0; configuration.RLDR = 0; configuration.light = 0; configuration.adc = 0; - for (uint8_t i = 0; i < 8; i++) { - configuration.arr[i] = 0; - } Serial.println(); Serial.println("Local values set to 0"); Serial.println(configuration.lux); @@ -125,14 +111,9 @@ void setup() { Serial.println(configuration.RLDR); Serial.println(configuration.light); Serial.println(configuration.adc); - for (uint8_t i = 0; i < 8; i++) { - Serial.print(configuration.arr[i]); - Serial.print(", "); - } - Serial.println(); Serial.println(); flash.readAnything(_addr, configuration); - flash.eraseSector(_addr); + flash.eraseSector(_addr, 0); Serial.println("After reading"); Serial.println(configuration.lux); @@ -140,11 +121,6 @@ void setup() { Serial.println(configuration.RLDR); Serial.println(configuration.light); Serial.println(configuration.adc); - for (uint8_t i = 0; i < 8; i++) { - Serial.print(configuration.arr[i]); - Serial.print(", "); - } - Serial.println(); } diff --git a/examples/TestFlash/.trinket.test.skip b/examples/TestFlash/.trinket.test.skip new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/examples/TestFlash/.trinket.test.skip @@ -0,0 +1 @@ + diff --git a/examples/TestFlash/TestFlash.ino b/examples/TestFlash/TestFlash.ino index 1cc0977..6af1df5 100644 --- a/examples/TestFlash/TestFlash.ino +++ b/examples/TestFlash/TestFlash.ino @@ -16,46 +16,46 @@ | 1. getID | | '1' gets the JEDEC ID of the chip | | | - | 2. writeByte [address] [byte] | - | '2' followed by '2435' and then by '224' writes the byte 224 to address 2435 | + | 2. writeByte [page] [offset] [byte] | + | '2' followed by '100' and then by '20' and then by '224' writes the byte 224 to page 100 position 20 | | | - | 3. readByte [address] | - | '3' followed by '2435' returns the byte from address '2435' | + | 3. readByte [page] [offset] | + | '3' followed by '100' and then by '20' returns the byte from page 100 position 20 | | | - | 4. writeWord [address] [word] | - | '4' followed by '5948' and then by '633' writes the int 633 to address 5948 | + | 4. writeWord [page] [offset] | + | '4' followed by '55' and then by '35' and then by '633' writes the int 633 to page 5 position 35 | | | - | 5. readWord [address] | - | '5' followed by '5948' returns the int from address 5948 | + | 5. readWord [page] [offset] | + | '5' followed by '200' and then by '30' returns the int from page 200 position 30 | | | - | 6. writeStr [address] [inputString] | - | '6' followed by '345736' and then by 'Test String 1!' writes the String 'Test String 1! to address 345736 | + | 6. writeStr [page] [offset] [inputString] | + | '6' followed by '345' and then by '65' and then by 'Test String 1!' writes the String 'Test String 1! to page 345 position 65 | | | - | 7. readStr [address] [outputString] | - | '7' followed by '345736' reads the String from address 345736 into the outputString | + | 7. readStr [page] [offset] [outputString] | + | '7' followed by '2050' and then by '73' reds the String from page 2050 position 73 into the outputString | | | | 8. writePage [page] | - | '8' followed by '33' writes bytes from 0 to 255 sequentially to fill a page (256 bytes) starting with address 33 | + | '8' followed by '33' writes bytes from 0 to 255 sequentially to fill page 33 | | | | 9. printPage [page] | - | '9' followed by 33 reads & prints a page (256 bytes) starting with address 33. To just read a page to a data buffer, refer | + | '9' followed by 33 reads & prints page 33. To just read a page to a data buffer, refer | | to 'ReadMe.md' in the library folder. | | | - | 10. printAllData | - | '10' reads the entire chip and outputs the data as a byte array to the serial console | + | 10. printAllPages | + | '10' reads all 4096 pages and outputs them to the serial console | | This function is to extract data from a flash chip onto a computer as a text file. | | Refer to 'Read me.md' in the library for details. | | | | 11. Erase 4KB sector | - | '11' followed by 2 erases a 4KB sector containing the address be erased | + | '11' followed by 2 erases a 4KB sector containing the page to be erased | | Page 0-15 --> Sector 0; Page 16-31 --> Sector 1;......Page 4080-4095 --> Sector 255 | | | | 12. Erase 32KB block | - | '12' followed by 2 erases a 32KB block containing the address to be erased | + | '12' followed by 2 erases a 32KB block containing the page to be erased | | Page 0-15 --> Sector 0; Page 16-31 --> Sector 1;......Page 4080-4095 --> Sector 255 | | | | 13. Erase 64KB block | - | '13' followed by 2 erases a 64KB block containing the address to be erased | + | '13' followed by 2 erases a 64KB block containing the page to be erased | | Page 0-15 --> Sector 0; Page 16-31 --> Sector 1;......Page 4080-4095 --> Sector 255 | | | | 14. Erase Chip | @@ -70,14 +70,11 @@ uint8_t pageBuffer[256]; String serialCommand; char printBuffer[128]; -uint32_t addr; -uint8_t dataByte; +uint16_t page; +uint8_t offset, dataByte; uint16_t dataInt; String inputString, outputString; -//Define a flash memory size (if using non-Winbond memory) according to the list in defines.h -//#define CHIPSIZE MB64 - #if defined(ARDUINO_SAMD_ZERO) && defined(SERIAL_PORT_USBVIRTUAL) // Required for Serial on Zero based boards #define Serial SERIAL_PORT_USBVIRTUAL @@ -89,7 +86,6 @@ String inputString, outputString; #define BAUD_RATE 115200 #endif -//SPIFlash flash(SS1, &SPI1); //Use this constructor if using an SPI bus other than the default SPI. Only works with chips with more than one hardware SPI bus SPIFlash flash; void setup() { @@ -101,11 +97,7 @@ void setup() { Serial.print(F(".")); } Serial.println(); -#if defined (CHIPSIZE) - flash.begin(CHIPSIZE); //use flash.begin(CHIPSIZE) if using non-Winbond flash (Refer to '#define CHIPSIZE' above) -#else flash.begin(); -#endif Serial.println(); Serial.println(); commandList(); @@ -142,15 +134,24 @@ void loop() { Serial.println(F(" Function 2 : Write Byte ")); printSplash(); printLine(); - inputAddress(); + Serial.print(F("Please enter the number of the page you wish to modify: ")); + while (!Serial.available()) { + } + page = Serial.parseInt(); + Serial.println(page); + Serial.print(F("Please enter the position on the page (0-255) you wish to modify: ")); + while (!Serial.available()) { + } + offset = Serial.parseInt(); + Serial.println(offset); Serial.print(F("Please enter the value of the byte (0-255) you wish to save: ")); while (!Serial.available()) { } dataByte = Serial.parseInt(); Serial.println(dataByte); - if (flash.writeByte(addr, dataByte)) { + if (flash.writeByte(page, offset, dataByte)) { clearprintBuffer(); - sprintf(printBuffer, "%d has been written to address %d", dataByte, addr); + sprintf(printBuffer, "%d has been written to position %d on page %d", dataByte, offset, page); Serial.println(printBuffer); } else { @@ -164,11 +165,20 @@ void loop() { Serial.println(F(" Function 3 : Read Byte ")); printSplash(); printLine(); - inputAddress(); + Serial.print(F("Please enter the number of the page the byte you wish to read is on: ")); + while (!Serial.available()) { + } + page = Serial.parseInt(); + Serial.println(page); + Serial.print(F("Please enter the position of the byte on the page (0-255) you wish to read: ")); + while (!Serial.available()) { + } + offset = Serial.parseInt(); + Serial.println(offset); clearprintBuffer(); - sprintf(printBuffer, "The byte at address %d is: ", addr); + sprintf(printBuffer, "The byte at position %d on page %d is: ", offset, page); Serial.print(printBuffer); - Serial.println(flash.readByte(addr)); + Serial.println(flash.readByte(page, offset)); printLine(); printNextCMD(); } @@ -177,15 +187,24 @@ void loop() { Serial.println(F(" Function 4 : Write Word ")); printSplash(); printLine(); - inputAddress(); + Serial.print(F("Please enter the number of the page you wish to modify: ")); + while (!Serial.available()) { + } + page = Serial.parseInt(); + Serial.println(page); + Serial.print(F("Please enter the position on the page (0-255) you wish to modify: ")); + while (!Serial.available()) { + } + offset = Serial.parseInt(); + Serial.println(offset); Serial.print(F("Please enter the value of the word (>255) you wish to save: ")); while (!Serial.available()) { } dataInt = Serial.parseInt(); Serial.println(dataInt); - if (flash.writeWord(addr, dataInt)) { + if (flash.writeWord(page, offset, dataInt)) { clearprintBuffer(); - sprintf(printBuffer, "%d has been written to address %d", dataInt, addr); + sprintf(printBuffer, "%d has been written to position %d on page %d", dataInt, offset, page); Serial.println(printBuffer); } else { @@ -199,11 +218,20 @@ void loop() { Serial.println(F(" Function 5 : Read Word ")); printSplash(); printLine(); - inputAddress(); + Serial.print(F("Please enter the number of the page the byte you wish to read is on: ")); + while (!Serial.available()) { + } + page = Serial.parseInt(); + Serial.println(page); + Serial.print(F("Please enter the position of the word on the page (0-255) you wish to read: ")); + while (!Serial.available()) { + } + offset = Serial.parseInt(); + Serial.println(offset); clearprintBuffer(); - sprintf(printBuffer, "The unsigned int at address %d is: ", addr); + sprintf(printBuffer, "The unsigned int at position %d on page %d is: ", offset, page); Serial.print(printBuffer); - Serial.println(flash.readWord(addr)); + Serial.println(flash.readWord(page, offset)); printLine(); printNextCMD(); } @@ -213,16 +241,25 @@ void loop() { printSplash(); printLine(); Serial.println(F("This function will write a String of your choice to the page selected.")); - inputAddress(); + Serial.print(F("Please enter the number of the page you wish to write to: ")); + while (!Serial.available()) { + } + page = Serial.parseInt(); + Serial.println(page); + Serial.print(F("Please enter the position on the page (0-255) you wish to write to: ")); + while (!Serial.available()) { + } + offset = Serial.parseInt(); + Serial.println(offset); Serial.println(F("Please enter the String you wish to save: ")); while (!Serial.available()) { } readSerialStr(inputString); - if (flash.writeStr(addr, inputString)) { + if (flash.writeStr(page, offset, inputString)) { clearprintBuffer(); Serial.print(F("String '")); Serial.print(inputString); - sprintf(printBuffer, "' has been written to address %d", addr); + sprintf(printBuffer, "' has been written to position %d on page %d", offset, page); Serial.println(printBuffer); } else { @@ -236,12 +273,20 @@ void loop() { Serial.println(F(" Function 7 : Read String ")); printSplash(); printLine(); - Serial.print(F("This function will read a string from your address of choice: ")); - inputAddress(); + Serial.print(F("Please enter the number of the page the String you wish to read is on: ")); + while (!Serial.available()) { + } + page = Serial.parseInt(); + Serial.println(page); + Serial.print(F("Please enter the position of the String on the page (0-255) you wish to read: ")); + while (!Serial.available()) { + } + offset = Serial.parseInt(); + Serial.println(offset); clearprintBuffer(); - sprintf(printBuffer, "The String at address %d is: ", addr); + sprintf(printBuffer, "The String at position %d on page %d is: ", offset, page); Serial.print(printBuffer); - flash.readStr(addr, outputString); + flash.readStr(page, offset, outputString); Serial.println(outputString); printLine(); printNextCMD(); @@ -251,18 +296,18 @@ void loop() { Serial.println(F(" Function 8 : Write Page ")); printSplash(); printLine(); - Serial.println(F("This function will write a sequence of bytes (0-255) starting from your address of choice")); - Serial.print(F("Please enter the address you wish to write to: ")); + Serial.println(F("This function will write a sequence of bytes (0-255) to the page selected.")); + Serial.print(F("Please enter the number of the page you wish to write to: ")); while (!Serial.available()) { } - addr = Serial.parseInt(); - Serial.println(addr); - for (uint16_t i = 0; i < SPI_PAGESIZE; ++i) { + page = Serial.parseInt(); + Serial.println(page); + for (uint16_t i = 0; i < PAGESIZE; ++i) { pageBuffer[i] = i; } - if (flash.writeByteArray(addr, &pageBuffer[0], SPI_PAGESIZE)) { + if (flash.writeByteArray(page, 0, &pageBuffer[0], PAGESIZE)) { clearprintBuffer(); - sprintf(printBuffer, "Values from 0 to 255 have been written starting from the address %d", addr); + sprintf(printBuffer, "Values from 0 to 255 have been written to the page %d", page); Serial.println(printBuffer); printReadChoice(); while (!Serial.available()) { @@ -275,7 +320,7 @@ void loop() { } uint8_t outputType = Serial.parseInt(); Serial.println(outputType); - printPage(addr, outputType); + printPage(page, outputType); } } else { @@ -289,18 +334,18 @@ void loop() { Serial.println(F(" Function 9 : Read Page ")); printSplash(); printLine(); - Serial.println(F("This function will read 256 bytes from the address selected.")); - Serial.print(F("Please enter the address you wish to read: ")); + Serial.println(F("This function will read the entire page selected.")); + Serial.print(F("Please enter the number of the page you wish to read: ")); while (!Serial.available()) { } - addr = Serial.parseInt(); - Serial.println(addr); + page = Serial.parseInt(); + Serial.println(page); printOutputChoice(); while (!Serial.available()) { } uint8_t outputType = Serial.parseInt(); Serial.println(outputType); - printPage(addr, outputType); + printPage(page, outputType); printLine(); printNextCMD(); } @@ -332,14 +377,14 @@ void loop() { printSplash(); printLine(); Serial.println(F("This function will erase a 4KB sector.")); - Serial.print(F("Please enter the address you wish to erase: ")); + Serial.print(F("Please enter the number of the page you wish to erase: ")); while (!Serial.available()) { } - addr = Serial.parseInt(); - Serial.println(addr); - flash.eraseSector(addr); + page = Serial.parseInt(); + Serial.println(page); + flash.eraseSector(page, 0); clearprintBuffer(); - sprintf(printBuffer, "A 4KB sector containing address %d has been erased", addr); + sprintf(printBuffer, "A 4KB sector containing page %d has been erased", page); Serial.println(printBuffer); printReadChoice(); while (!Serial.available()) { @@ -352,7 +397,7 @@ void loop() { } uint8_t outputType = Serial.parseInt(); Serial.println(outputType); - printPage(addr, outputType); + printPage(page, outputType); } printLine(); printNextCMD(); @@ -363,14 +408,14 @@ void loop() { printSplash(); printLine(); Serial.println(F("This function will erase a 32KB block.")); - Serial.print(F("Please enter the address you wish to erase: ")); + Serial.print(F("Please enter the number of the page you wish to erase: ")); while (!Serial.available()) { } - addr = Serial.parseInt(); - Serial.println(addr); - flash.eraseBlock32K(addr); + page = Serial.parseInt(); + Serial.println(page); + flash.eraseBlock32K(page, 0); clearprintBuffer(); - sprintf(printBuffer, "A 32KB block containing address %d has been erased", addr); + sprintf(printBuffer, "A 32KB block containing page %d has been erased", page); Serial.println(printBuffer); printReadChoice(); while (!Serial.available()) { @@ -383,7 +428,7 @@ void loop() { } uint8_t outputType = Serial.parseInt(); Serial.println(outputType); - printPage(addr, outputType); + printPage(page, outputType); } printLine(); printNextCMD(); @@ -394,14 +439,14 @@ void loop() { printSplash(); printLine(); Serial.println(F("This function will erase a 64KB block.")); - Serial.print(F("Please enter the address you wish to erase: ")); + Serial.print(F("Please enter the number of the page you wish to erase: ")); while (!Serial.available()) { } - addr = Serial.parseInt(); - Serial.println(addr); - flash.eraseBlock64K(addr); + page = Serial.parseInt(); + Serial.println(page); + flash.eraseBlock64K(page, 0); clearprintBuffer(); - sprintf(printBuffer, "A 64KB block containing address %d has been erased", addr); + sprintf(printBuffer, "A 64KB block containing page %d has been erased", page); Serial.println(printBuffer); printReadChoice(); while (!Serial.available()) { @@ -414,7 +459,7 @@ void loop() { } uint8_t outputType = Serial.parseInt(); Serial.println(outputType); - printPage(addr, outputType); + printPage(page, outputType); } printLine(); printNextCMD(); @@ -484,16 +529,16 @@ void _printPageBytes(uint8_t *data_buffer, uint8_t outputType) { } //Reads a page of data and prints it to Serial stream. Make sure the sizeOf(uint8_t data_buffer[]) == 256. -void printPage(uint32_t _address, uint8_t outputType) { +void printPage(uint16_t page_number, uint8_t outputType) { if (!Serial) Serial.begin(115200); char buffer[24]; - sprintf(buffer, "Reading address (%04x)", _address); + sprintf(buffer, "Reading page (%04x)", page_number); Serial.println(buffer); - uint8_t data_buffer[SPI_PAGESIZE]; - flash.readByteArray(_address, &data_buffer[0], SPI_PAGESIZE); + uint8_t data_buffer[PAGESIZE]; + flash.readByteArray(page_number, 0, &data_buffer[0], PAGESIZE); _printPageBytes(data_buffer, outputType); } @@ -506,11 +551,10 @@ void printAllPages(uint8_t outputType) { Serial.println("Reading all pages"); uint8_t data_buffer[256]; - uint32_t maxAddr = flash.getCapacity(); - for (int a = 0; a < maxAddr; a++) { - flash.readByteArray(a, &data_buffer[0], 256); + uint32_t maxPage = flash.getMaxPage(); + for (int a = 0; a < maxPage; a++) { + flash.readByteArray(a, 0, &data_buffer[0], 256); _printPageBytes(data_buffer, outputType); - delay(100); } } @@ -550,12 +594,3 @@ void writeFail() { Serial.println("Data write failed"); } - -void inputAddress(void) { - Serial.print(F("Please enter the address (0 - CAPACITY) you wish to access: ")); - while (!Serial.available()) { - } - addr = Serial.parseInt(); - Serial.println(addr); -} - diff --git a/examples/getAddressEx/.trinket.test.skip b/examples/getAddressEx/.trinket.test.skip new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/examples/getAddressEx/.trinket.test.skip @@ -0,0 +1 @@ + diff --git a/examples/getAddressEx/getAddressEx.ino b/examples/getAddressEx/getAddressEx.ino index 6a84572..d391801 100644 --- a/examples/getAddressEx/getAddressEx.ino +++ b/examples/getAddressEx/getAddressEx.ino @@ -41,7 +41,6 @@ byte testByte[] = { 3, 245, 84, 100 }; -//SPIFlash flash(SS1, &SPI1); //Use this constructor if using an SPI bus other than the default SPI. Only works with chips with more than one hardware SPI bus SPIFlash flash; void getAddresses(); diff --git a/examples/readWriteString/.trinket.test.skip b/examples/readWriteString/.trinket.test.skip new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/examples/readWriteString/.trinket.test.skip @@ -0,0 +1 @@ + diff --git a/examples/readWriteString/readWriteString.ino b/examples/readWriteString/readWriteString.ino index 64c6572..84ed6a2 100644 --- a/examples/readWriteString/readWriteString.ino +++ b/examples/readWriteString/readWriteString.ino @@ -2,10 +2,10 @@ |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| | readWriteString.ino | | SPIFlash library | - | v 3.0.0 | + | v 2.5.0 | |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| | Marzogh | - | 29.05.2017 | + | 16.11.2016 | |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| | | | This program shows the method of reading a string from the console and saving it to flash memory | @@ -14,7 +14,8 @@ */ #include -uint32_t strAddr; +int strPage, strSize; +byte strOffset; #if defined(ARDUINO_SAMD_ZERO) && defined(SERIAL_PORT_USBVIRTUAL) // Required for Serial on Zero based boards @@ -29,41 +30,56 @@ uint32_t strAddr; #define RANDPIN A0 #endif -//SPIFlash flash(SS1, &SPI1); //Use this constructor if using an SPI bus other than the default SPI. Only works with chips with more than one hardware SPI bus SPIFlash flash; bool readSerialStr(String &inputStr); void setup() { +#ifndef __AVR_ATtiny85__ Serial.begin(BAUD_RATE); +#endif #if defined (ARDUINO_SAMD_ZERO) || (__AVR_ATmega32U4__) while (!Serial) ; // Wait for Serial monitor to open #endif flash.begin(); +#if defined __AVR_ATtiny85__ + randomSeed(65535537); +#else randomSeed(analogRead(RANDPIN)); - strAddr = random(0, flash.getCapacity()); +#endif + strPage = random(0, 4095); + strOffset = random(0, 255); String inputString = "This is a test String"; - flash.writeStr(strAddr, inputString); + flash.writeStr(strPage, strOffset, inputString); +#ifndef __AVR_ATtiny85__ Serial.print(F("Written string: ")); - Serial.println(inputString); - Serial.print(F("To address: ")); - Serial.println(strAddr); + Serial.print(inputString); + Serial.print(F(" to page ")); + Serial.print(strPage); + Serial.print(F(", at offset ")); + Serial.println(strOffset); +#endif String outputString = ""; - if (flash.readStr(strAddr, outputString)) { + if (flash.readStr(strPage, strOffset, outputString)) { +#ifndef __AVR_ATtiny85__ Serial.print(F("Read string: ")); - Serial.println(outputString); - Serial.print(F("From address: ")); - Serial.println(strAddr); + Serial.print(outputString); + Serial.print(F(" from page ")); + Serial.print(strPage); + Serial.print(F(", at offset ")); + Serial.println(strOffset); +#endif } - while (!flash.eraseSector(strAddr)); + while (!flash.eraseSector(strPage, 0)); } void loop() { } +#ifndef __AVR_ATtiny85__ //Reads a string from Serial bool readSerialStr(String &inputStr) { if (!Serial) @@ -75,3 +91,4 @@ bool readSerialStr(String &inputStr) { } return false; } +#endif diff --git a/examples/tinyFlashDiagnostics/Attiny_functions.ino b/examples/tinyFlashDiagnostics/Attiny_functions.ino new file mode 100644 index 0000000..1091824 --- /dev/null +++ b/examples/tinyFlashDiagnostics/Attiny_functions.ino @@ -0,0 +1,279 @@ +/* + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Attiny_functions.ino | + | SPIFlash library | + | v 2.7.0 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Marzogh | + | 26.04.2017 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | | + | For a full diagnostics rundown - with error codes and details of the errors | + | uncomment #define RUNDIAGNOSTIC in SPIFlash.cpp in the library before compiling | + | and loading this application onto your Arduino. | + | | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| +*/ + +#if defined (__AVR_ATtiny85__) + +void startup(void) { +#if defined (CHIPSIZE) + flash.begin(CHIPSIZE); //use flash.begin(CHIPSIZE) if using non-Winbond flash +#else + flash.begin(); +#endif +} + +void setWrittenStatus(void) { + dataPacket.writeStatus |= PASS; + dataPacket.writeStatus |= ATTINY85; +} + +void saveResults() { + flash.writeAnything(addr, dataPacket); +} + +void setTest(uint8_t _t) { + dataPacket.test |= _t; +} + +void intDiag(uint32_t _addr) { + //Set variables + word _data, _d; + _data = 4520; + setTest(INT); + + //Test & time Write function + if (flash.writeWord(_addr, _data)) { + dataPacket.Status |= iW; + } + else + { + dataPacket.Status &= !iW; + } + + + //Test & time Read function + _d = flash.readWord(_addr); + if (_d == _data) { + dataPacket.Status |= iR; + } + else + { + dataPacket.Status &= !iR; + } + + //Erase the sector previously written to + flash.eraseSector(_addr); +} + +void floatDiag(uint32_t _addr) { + //Set variables + float _data, _d; + _data = 3.1412; + setTest(FLOAT); + + //Test & time Write function + if (flash.writeFloat(_addr, _data)) { + dataPacket.Status |= fW; + } + else + { + dataPacket.Status &= !fW; + } + + + //Test & time Read function + _d = flash.readFloat(_addr); + if (_d == _data) { + dataPacket.Status |= fR; + } + else + { + dataPacket.Status &= !fR; + } + + //Erase the sector previously written to + flash.eraseSector(_addr); +} + +void structDiag(uint32_t _addr) { + //Set variables + struct Configuration { // Voltage ouput fR;om potential divider to Analog input + float RLDR; // Resistance calculation of potential divider with LDR + bool light; + uint8_t adc; + }; + Configuration _data, _d; + _data.RLDR = 89.32; + _data.light = true; + _data.adc = 5; + setTest(STRUCT); + + //Test & time Write function + if (flash.writeAnything(_addr, _data)) { + dataPacket.Status |= scW; + } + else + { + dataPacket.Status &= !scW; + } + + //Test & time Read function + if (flash.readAnything(_addr, _d)) { + if (_d.RLDR == _data.RLDR && _d.light == _data.light && _d.adc == _data.adc) { + dataPacket.Status |= scR; + } + else + { + dataPacket.Status &= !scR; + } + } + else { + dataPacket.Status &= !scR; + } + + //Erase the sector previously written to + flash.eraseSector(_addr); +} + +void stringDiag(uint32_t _addr) { + //Set variables + String _d = ""; + String _data = "1Ab# D"; + setTest(STRING); + + //Test & time Write function + if (flash.writeStr(_addr, _data)) { + dataPacket.Status |= sgW; + } + else + { + dataPacket.Status &= !sgW; + } + + + //Test & time Read function + if (flash.readStr(_addr, _d)) { + if (_d == _data) { + dataPacket.Status |= sgR; + } + else + { + dataPacket.Status &= !sgR; + } + //Erase the sector previously written to + flash.eraseSector(_addr); + } +} + +void arrayDiag(uint32_t _addr) { + //Set variables + uint8_t _data[20], _d[20]; + setTest(ARRAY); + + for (uint8_t i = 0; i < 21; i++) { + _data[i] = i; + } + + //Test & time Write function + if (flash.writeByteArray(_addr, _data, 20)) { + dataPacket.Status |= aW; + } + else + { + dataPacket.Status &= !aW; + } + + + //Test & time Read function + if (flash.readByteArray(_addr, _d, 20)) { + for (uint8_t i = 0; i < 21; i++) + if (_d[i] != _data[i]) { + dataPacket.Status &= aR; + break; + } + dataPacket.Status |= aR; + } + //Erase the sector previously written to + flash.eraseSector(_addr); +} + +void eraseDiag(uint32_t _addr) { + setTest(ERASE); + + //Test & time eraseBlock32K function + if (flash.eraseBlock32K(_addr)) { + dataPacket.Status |= eB; + } + else + { + dataPacket.Status &= !eB; + } + + //Test & time eraseChip function + if (flash.eraseChip()) { + dataPacket.Status |= eC; + } + else + { + dataPacket.Status &= !eC; + } +} + +void powerDiag(void) { + setTest(ERASE); + + //Test & time powerDown function + if (flash.powerDown()) { + dataPacket.Status |= pOFF; + } + else + { + dataPacket.Status &= !pOFF; + } + + //Test & time powerUp function + if (flash.powerUp()) { + dataPacket.Status |= pON; + } + else + { + dataPacket.Status &= !pON; + } +} + +void diagnose(void) { + do { + flash.eraseChip(); + } while (prevWritten()); + + setWrittenStatus(); + + uint32_t testaddr = random(STARTADDR, 0xFFFFF); +#if defined INTTEST + intDiag(testaddr); +#endif +#if defined FLOATTEST + floatDiag(testaddr); +#endif +#if defined STRUCTTEST + structDiag(testaddr); +#endif +#if defined STRINGTEST + stringDiag(testaddr); +#endif +#if defined ARRAYTEST + arrayDiag(testaddr); +#endif +#if defined ERASETEST + eraseDiag(testaddr); +#endif +#if defined POWERTEST + powerDiag(); +#endif + saveResults(); +} + +#endif diff --git a/examples/tinyFlashDiagnostics/NonAttiny_functions.ino b/examples/tinyFlashDiagnostics/NonAttiny_functions.ino new file mode 100644 index 0000000..7b364cc --- /dev/null +++ b/examples/tinyFlashDiagnostics/NonAttiny_functions.ino @@ -0,0 +1,463 @@ +/* + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | NonAttiny_functions.ino | + | SPIFlash library | + | v 2.7.0 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Marzogh | + | 26.04.2017 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | | + | For a full diagnostics rundown - with error codes and details of the errors | + | uncomment #define RUNDIAGNOSTIC in SPIFlash.cpp in the library before compiling | + | and loading this application onto your Arduino. | + | | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| +*/ + +#if !defined (__AVR_ATtiny85__) +void startup(void) { + Serial.begin(BAUD_RATE); +#if defined (ARDUINO_ARCH_SAMD) || (__AVR_ATmega32U4__) + while (!Serial) ; // Wait for Serial monitor to open +#endif + Serial.print(F("Initialising Flash memory")); + for (int i = 0; i < 10; ++i) + { + Serial.print(F(".")); + } + Serial.println(); +#if defined (CHIPSIZE) + flash.begin(CHIPSIZE); //use flash.begin(CHIPSIZE) if using non-Winbond flash (Refer to '#define CHIPSIZE' above) +#else + flash.begin(); +#endif + Serial.println(); + Serial.println(); + +#if defined (ARDUINO_ARCH_ESP32) + randomSeed(65535537); +#else + randomSeed(analogRead(RANDPIN)); +#endif +} + +void clearprintBuffer(char *bufPtr) +{ + for (uint8_t i = 0; i < 128; i++) { + //printBuffer[i] = 0; + *bufPtr++ = 0; + } +} + +void printLine() { + for (uint8_t i = 0; i < 230; i++) { + Serial.print(F("-")); + } + Serial.println(); +} + +void printPass() { + Serial.print(F("Pass")); +} + +void printFail() { + Serial.print(F("Fail")); +} + +void printTab(uint8_t a, uint8_t b) { + for (uint8_t i = 0; i < a; i++) { + Serial.print(F("\t")); + } + if (b > 0) { + Serial.print("||"); + for (uint8_t i = 0; i < b; i++) { + Serial.print(F("\t")); + } + } +} + +void printTime(uint32_t _wTime, uint32_t _rTime) { + printTab(2, 1); + printTimer(_wTime); + printTab(2, 1); + printTimer(_rTime); +} + +void printTimer(uint32_t _us) { + + if (_us > 1000000) { + float _s = _us / (float)1000000; + Serial.print(_s, 4); + Serial.print(" s"); + } + else if (_us > 10000) { + float _ms = _us / (float)1000; + Serial.print(_ms, 4); + Serial.print(" ms"); + } + else { + Serial.print(_us); + Serial.print(F(" us")); + } +} + +void getID() { + char printBuffer[128]; + printLine(); + for (uint8_t i = 0; i < 68; i++) { + Serial.print(F(" ")); + } + Serial.print(F("SPIFlash Library version")); +#ifdef LIBVER + uint8_t _ver, _subver, _bugfix; + flash.libver(&_ver, &_subver, &_bugfix); + clearprintBuffer(&printBuffer[1]); + sprintf(printBuffer, ": %d.%d.%d", _ver, _subver, _bugfix); + Serial.println(printBuffer); +#else + Serial.println(F("< 2.5.0")); +#endif + printLine(); + + for (uint8_t i = 0; i < 80; i++) { + Serial.print(F(" ")); + } + Serial.println(F("Get ID")); + printLine(); + uint8_t b1, b2; + uint16_t b3; + uint32_t JEDEC = flash.getJEDECID(); + uint32_t maxPage = flash.getMaxPage(); + uint32_t capacity = flash.getCapacity(); + b1 = (JEDEC >> 16); + b2 = (JEDEC >> 8); + b3 = (JEDEC >> 0); + + + printLine(); + //---------------------------------------------------------------------------------------------// + + clearprintBuffer(&printBuffer[1]); + sprintf(printBuffer, "\t\t\tJEDEC ID: %04lxh", JEDEC); + Serial.println(printBuffer); + clearprintBuffer(&printBuffer[1]); + sprintf(printBuffer, "\t\t\tManufacturer ID: %02xh\n\t\t\tMemory Type: %02xh\n\t\t\tCapacity: %lu bytes\n\t\t\tMaximum pages: %lu", b1, b2, capacity, maxPage); + Serial.println(printBuffer); +} + +bool checkPage(uint8_t *data_buffer) { + for (int i = 0; i < 256; i++) { + if (data_buffer[i] != i) + return false; + } + return true; +} + +void printHeader(uint8_t _t) { + if (_t == DATA_FUNCTION && data_header == false) { + printLine(); + + for (uint8_t i = 0; i < 79; i++) { + Serial.print(F(" ")); + } + Serial.println(F("Data Check")); + printLine(); + + Serial.println(F("\tData Type\t||\tWrite Check\t||\tRead Check\t")); + printLine(); + + data_header = true; + } + if (_t == OTHER_FUNCTION && power_header == false) { + printLine(); + + for (uint8_t i = 0; i < 74; i++) { + Serial.print(F(" ")); + } + Serial.println(F("Other Function Check")); + printLine(); + + Serial.println(F("\tTest Type\t||\tStatus")); + printLine(); + + power_header = true; + } +} +void diagnose() { + getID(); + + uint16_t _stat; + if (prevWritten) { + if (dataPacket.test & INT) { + printHeader(DATA_FUNCTION); + intDiag(); + } + if (dataPacket.test & FLOAT) { + printHeader(DATA_FUNCTION); + floatDiag(); + } + if (dataPacket.test & STRING) { + printHeader(DATA_FUNCTION); + stringDiag(); + } + if (dataPacket.test & STRUCT) { + printHeader(DATA_FUNCTION); + structDiag(); + } + if (dataPacket.test & ARRAY) { + printHeader(DATA_FUNCTION); + arrayDiag(); + } + if (dataPacket.test & ERASE) { + printHeader(OTHER_FUNCTION); + eraseDiag(); + } + if (dataPacket.test & POWER) { + printHeader(OTHER_FUNCTION); + powerDiag(); + } + } + else { + Serial.println(F("No data found on chip. Please run this sketch on a an ATTiny85 connected to this flash chip first")); + } +} + +void intDiag(void) { + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Integer // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("Integer")); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & iW) { + printPass(); + } + else { + printFail(); + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & iR) { + printPass(); + } + else { + printFail(); + } + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void floatDiag(void) { + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Float // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("Float")); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & fW) { + printPass(); + } + else { + printFail(); + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & fR) { + printPass(); + } + else { + printFail(); + } + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void stringDiag(void) { + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Integer // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("String")); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & sgW) { + printPass(); + } + else { + printFail(); + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & sgR) { + printPass(); + } + else { + printFail(); + } + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void structDiag(void) { + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Integer // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("Integer")); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & scW) { + printPass(); + } + else { + printFail(); + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & scR) { + printPass(); + } + else { + printFail(); + } + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void arrayDiag(void) { + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Integer // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("Array")); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Write // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & aW) { + printPass(); + } + else { + printFail(); + } + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Read // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(2, 1); + if (dataPacket.Status & aR) { + printPass(); + } + else { + printFail(); + } + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void eraseDiag(void) { + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Erase Block // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("Erase Block")); + printTab(2, 1); + if (dataPacket.Status & eB) { + printPass(); + } + else { + printFail(); + } + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Erase Chip // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("Erase Chip")); + printTab(2, 1); + if (dataPacket.Status & eC) { + printPass(); + } + else { + printFail(); + } + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} + +void powerDiag(void) { + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Power on // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("Power off")); + printTab(2, 1); + if (dataPacket.Status & pOFF) { + printPass(); + } + else { + printFail(); + } + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Power off // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + printTab(1, 0); + Serial.print(F("Power on")); + printTab(2, 1); + if (dataPacket.Status & pON) { + printPass(); + } + else { + printFail(); + } + Serial.println(); + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + //-----------------------------------------------------------------------------------------------------------------------------------------------------// +} +#endif diff --git a/examples/tinyFlashDiagnostics/Notes.cpp b/examples/tinyFlashDiagnostics/Notes.cpp new file mode 100644 index 0000000..1abbda2 --- /dev/null +++ b/examples/tinyFlashDiagnostics/Notes.cpp @@ -0,0 +1,46 @@ +/* + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Notes.cpp | + | SPIFlash library | + | v 2.7.0 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Marzogh | + | 26.04.2017 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | | + | For a full diagnostics rundown - with error codes and details of the errors | + | uncomment #define RUNDIAGNOSTIC in SPIFlash.cpp in the library before compiling | + | and loading this application onto your Arduino. | + | | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| +*/ + +//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// +// // +// Structure of the status integer // +// _____________________________________________________________________________________ // +// | Bit number | // +// |_____________________________________________________________________________________| // +// | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | // +// |-------------------------------------------------------------------------------------| // +// | RES | RES | pOFF | pON | eC | eB | aR | aW | // +// |_____________________________________________________________________________________| // +// | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | // +// |-------------------------------------------------------------------------------------| // +// | scR | scW | sgR | sgW | fR | fW | iR | iW | // +// |_____________________________________________________________________________________| // +// // +// // +//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// +// // +// Structure of the test byte // +// _____________________________________________________________________________________ // +// | Bit number | // +// |_____________________________________________________________________________________| // +// | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | // +// |-------------------------------------------------------------------------------------| // +// | RES | POWER | ERASE | ARRAY | STRUCT | STRING | FLOAT | INT | // +// |_____________________________________________________________________________________| // +// // +// // +//!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!// diff --git a/examples/tinyFlashDiagnostics/tinyFlashDiagnostics.ino b/examples/tinyFlashDiagnostics/tinyFlashDiagnostics.ino new file mode 100644 index 0000000..4eb58c7 --- /dev/null +++ b/examples/tinyFlashDiagnostics/tinyFlashDiagnostics.ino @@ -0,0 +1,80 @@ +/* + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | tinyFlashDiagnostics.ino | + | SPIFlash library | + | v 2.6.0 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Marzogh | + | 16.04.2017 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | | + | For a full diagnostics rundown - with error codes and details of the errors | + | uncomment #define RUNDIAGNOSTIC in SPIFlash.cpp in the library before compiling | + | and loading this application onto your Arduino. | + | | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| +*/ + +#include "vars.h" + +#include + +//Define a flash memory size (if using non-Winbond memory) according to the list in defines.h +//#define CHIPSIZE MB64 + + //!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!// + // // + // ___ _____ _____ _ _____ _____ // + // / _ \_ _|_ _(_) | _ || ___| // + // / /_\ \| | | | _ _ __ _ _ \ V / |___ \ // + // | _ || | | | | | '_ \| | | |/ _ \ \ \ // + // | | | || | | | | | | | | |_| | |_| |/\__/ / // + // \_| |_/\_/ \_/ |_|_| |_|\__, \_____/\____/ // + // __/ | // + // |___/ // + // // + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + // Uncomment any of the tests below as required. // + // Remember, the ATTiny has a limited amount of SRAM, so only a limited number // + // of tests can be run concurrently. // + // // + //|------------------------------------------//----------------------------------------------|// + //| List of tests // _status bits |// + //|------------------------------------------//----------------------------------------------|// + #define INTTEST // iW, iR |// + #define FLOATTEST // fW, fR |// + //#define STRINGTEST // sgW, sgR |// + #define STRUCTTEST // scW, scR |// + //#define ARRAYTEST // aW, aR |// + #define ERASETEST // eB, eC |// + #define POWERTEST // pON, pOFF |// + //|------------------------------------------//----------------------------------------------|// + +SPIFlash flash; + +void setup() { + startup(); + diagnose(); +} + +void loop() { + +} + +//************************************************************************************************// +// // +// Non-board specific code // +// // +//************************************************************************************************// + +bool prevWritten() { + addr = flash.getAddress(sizeof(dataPacket)); + flash.readAnything(addr, dataPacket); + if (dataPacket.writeStatus != 0xFF) { + return true; + } + else { + return false; + } +} +//************************************************************************************************// diff --git a/examples/tinyFlashDiagnostics/vars.h b/examples/tinyFlashDiagnostics/vars.h new file mode 100644 index 0000000..378a8ab --- /dev/null +++ b/examples/tinyFlashDiagnostics/vars.h @@ -0,0 +1,96 @@ +/* + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | vars.h | + | SPIFlash library | + | v 2.7.0 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Marzogh | + | 26.04.2017 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | | + | For a full diagnostics rundown - with error codes and details of the errors | + | uncomment #define RUNDIAGNOSTIC in SPIFlash.cpp in the library before compiling | + | and loading this application onto your Arduino. | + | | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| +*/ +//************************************************************************************************// +// // +// Non-board specific code // +// // +//************************************************************************************************// +#define PASS 0x01 +#define FAIL 0x00 +#define ATTINY85 0x02 + +#define INT 0x01 +#define FLOAT 0x02 +#define STRING 0x04 +#define STRUCT 0x08 +#define ARRAY 0x10 +#define ERASE 0x20 +#define POWER 0x40 + + +#define iW 0x0001 +#define iR 0x0002 +#define fW 0x0004 +#define fR 0x0008 +#define sgW 0x0010 +#define sgR 0x0020 +#define scW 0x0040 +#define scR 0x0080 +#define aW 0x0100 +#define aR 0x0200 +#define eB 0x0400 +#define eC 0x0800 +#define pON 0x1000 +#define pOFF 0x2000 + +struct _dataPacket { + uint8_t writeStatus; + uint8_t test; + uint16_t Status; +}; +_dataPacket dataPacket; +uint32_t addr; + +//************************************************************************************************// +// // +// If using an ATTiny85 board // +// // +//************************************************************************************************// + +#if defined (__AVR_ATtiny85__) + +#define STARTADDR ((sizeof(dataPacket))*2) + +#endif + +//************************************************************************************************// +// // +// If using a non ATTiny85 board // +// // +//************************************************************************************************// +#if !defined (__AVR_ATtiny85__) + +#if defined(ARDUINO_SAMD_ZERO) && defined(SERIAL_PORT_USBVIRTUAL) +// Required for Serial on Zero based boards +#define Serial SERIAL_PORT_USBVIRTUAL +#endif + +#if defined (SIMBLEE) +#define BAUD_RATE 250000 +#define RANDPIN 1 +#else +#define BAUD_RATE 115200 +#define RANDPIN A0 +#endif + +#define DATA_FUNCTION 0x03 +#define OTHER_FUNCTION 0x04 + +bool data_header, power_header = false; + +#endif + diff --git a/examples/tinyFlashTester/tinyFlashTester.ino b/examples/tinyFlashTester/tinyFlashTester.ino new file mode 100644 index 0000000..173c20e --- /dev/null +++ b/examples/tinyFlashTester/tinyFlashTester.ino @@ -0,0 +1,69 @@ +/* + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Struct_writer.ino | + | SPIFlash library | + | v 2.7.0 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | Marzogh | + | 19.04.2017 | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| + | | + | This program writes a struct to a random location on your flash memory chip and reads it back. | + | This particular program is designed to test the library with an ATTiny85 chip. Follow the steps below to make sure the test runs as it should | + | 1. Compile and run this program on an ATTiny85 board to which an SPIFlash memory has been wired in. | + | 2. Swap Flash chip over to another supported board (with Serial IO) and compile and run this program again | + | 3. Check serial output to see if data has been written/read properly. | + | | + |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~| +*/ + +#include + +SPIFlash flash; + +struct Configuration { + float lux; + float vOut; // Voltage ouput from potential divider to Analog input + float RLDR; // Resistance calculation of potential divider with LDR + bool light; + uint8_t adc; +}; +Configuration configuration; + +void setup() { + flash.begin(); + uint32_t _addr = 34235; + +#if defined (__AVR_ATtiny85__) + configuration.lux = 98.43; + configuration.vOut = 4.84; + configuration.RLDR = 889.32; + configuration.light = true; + configuration.adc = 5; + flash.writeAnything(_addr, configuration); +#else + Serial.println("Data Written to flash was: "); + Serial.println(configuration.lux); + Serial.println(configuration.vOut); + Serial.println(configuration.RLDR); + Serial.println(configuration.light); + Serial.println(configuration.adc); + configuration.lux = 0; + configuration.vOut = 0; + configuration.RLDR = 0; + configuration.light = 0; + configuration.adc = 0; + flash.readAnything(_addr, configuration); + flash.eraseSector(_addr, 0); + Serial.println("After reading"); + Serial.println(configuration.lux); + Serial.println(configuration.vOut); + Serial.println(configuration.RLDR); + Serial.println(configuration.light); + Serial.println(configuration.adc); +#endif + +} + +void loop() { +} diff --git a/extras/Changes.log b/extras/Changes.log old mode 100755 new mode 100644 index 8d070c1..a45333e --- a/extras/Changes.log +++ b/extras/Changes.log @@ -2,65 +2,6 @@ // SPIFlash Library // // Changes log // //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -// Version 3.0.0 // -// Author: Prajwal Bhattaram // -// 09.08.2017 // -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -To check: -Bugs Squashed: ---> The writeByteArray() & writeCharArray() bug that occurred when writing arrays that spanned page boundaries (squashed in v2.5.0), stayed around to haunt the other functions. Writing any data larger than a single byte that spanned page boundaries would cause the data to wrap around to the beginning of the page. The likelihood of this occurring was slim - no one has reported it to date. However, just in case, this has now been squashed in this release. - -Deprecations: ---> Going forward the ATTiny85 is no longer officially supported. ---> The library no longer supports using the page number + offset combination instead of addresses. If your code requires you to use a page number + offset combination, use the following code to help - address = (pagenumber << 8) + offset. - _____________________________________ - (32 bit) | (16 bit) | (8 bit) ---> The constructor no longer takes the pageOverflow variable as an argument. Page overflow is globally enabled by default and can be disabled by including a "#define DISABLEOVERFLOW" at the beginning of the user code. - -New Boards supported: ---> Nucleo-F091RC ---> Adafruit Feather M0 - -New Flash manufacturers supported: ---> Microchip (SST25 & SST26 series) ---> Cypress ---> Spansion - -Enhancements: ---> Confirmed to work with SPANSION/CYPRESS & MICROCHIP (Both SST25 & SST26 series). ---> If not using an officially supported chip, use the following variation of flash.begin() : - flash.begin(flashChipSize); - - where flashChipSize is indicated in Bytes, Kilobytes or Megabytes. (Refer to the next two items in this change log) ---> Including 'flashChipSize' in flash.begin() compiles more efficiently than in prvious versions. ---> The way memory size is indicated by the users in flash.begin(flashChipSize) has changed - please refer to defines.h or the wiki for further information. The new method enables users to define any custom size unlike the previous version where only a limited number of predetermined sizes were available to use. ---> Library faster than before (Refer to Library speed comparison in the extras folder for timing details): - - Improvements in speed in v3.0.0 when compared to v2.7.0 (values in percentage of time v3.0.0 is faster than v2.7.0) - (+) writeByte -> +3% - (+) writeChar -> +6% - (+) writeWord -> +3% - (+) writeShort -> +5% - (+) writeULong -> +7% - (+) writeLong | readLong -> +7% | +7% - (+) writeFloat -> +7% - (+) writeStr | readStr -> +57% | +61% - (+) writeAnything | readAnything (struct) -> +25% | +20% - (+) writeByteArray | readByteArray (256 bytes) -> +18% | +45% - (+) eraseSector -> +99% - (+) eraseBlock32K -> +99.4% - (+) eraseBlock64K -> +99.5% - (+) eraseChip -> +12% - ---> Constructor changed to enable user to choose one of multiple SPI ports - if available. Look at wiki for further info ---> When RUNDIAGNOSTIC is uncommented in SPIFlash.h, users now have access to a new function called flash.functionRunTime() which can be called after any library I/O function is run. flash.functionRunTime() returns the time taken by the previous function to run, in microseconds (as a float). An example use case can be found when the FlashDiagnostics sketch is run with RUNDIAGNOSTIC uncommented. ---> _notBusy() is faster ---> Completely re-written FlashDiagnostics - uses fewer resources, compatible with more chips and boards ---> All functions except the Byte/Char Array Read/Write functions now call an internal _write/_read function for increased ease of compilation, execution and troubleshooting ---> Restructured the internal _troubleshoot() function to be better human readable and faster to compile. ---> Added a function `getUniqueID()` which returns the unique ID of the flash memory chip as a 64-bit integer. -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// // Version 2.7.0 // // Author: Prajwal Bhattaram // // 17.04.2017 // diff --git a/extras/DMASAMD.cpp b/extras/DMASAMD.cpp deleted file mode 100644 index 678627b..0000000 --- a/extras/DMASAMD.cpp +++ /dev/null @@ -1,158 +0,0 @@ -/* Arduino SPIFlash Library v.2.6.0 - * Copyright (C) 2017 by Prajwal Bhattaram - * Created by Prajwal Bhattaram - 30/09/2016 - * Modified by Prajwal Bhattaram - 14/04/2017 - * Original code from @manitou48 - * - * This file is part of the Arduino SPIFlash Library. This library is for - * Winbond NOR flash memory modules. In its current form it enables reading - * and writing individual data variables, structs and arrays from and to various locations; - * reading and writing pages; continuous read functions; sector, block and chip erase; - * suspending and resuming programming/erase and powering down for low power operation. - * - * This Library is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This Library is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License v3.0 - * along with the Arduino SPIFlash Library. If not, see - * . - */ - -#include "SPIFlash.h" - -//--------------------------- Private Arduino Zero Variables ----------------------------// - Sercom *sercom = (Sercom *)ZERO_SPISERCOM; //Set SPI SERCOM - // DMA 12 channels - typedef struct { - uint16_t btctrl; - uint16_t btcnt; - uint32_t srcaddr; - uint32_t dstaddr; - uint32_t descaddr; - } dmacdescriptor ; - volatile dmacdescriptor wrb[12] __attribute__ ((aligned (16))); - dmacdescriptor descriptor_section[12] __attribute__ ((aligned (16))); - dmacdescriptor descriptor __attribute__ ((aligned (16))); - - static uint32_t chnltx = 0, chnlrx = 1; // DMA channels - enum XfrType { DoTX, DoRX, DoTXRX}; - static XfrType xtype; - static uint8_t rxsink[1], txsrc[1] = {0xff}; - volatile uint32_t dmadone; - -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -// Private functions used by Arduino Zero DMA operations // -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// - -void SPIFlash::_zeroDMAC_Handler(void) { - // interrupts DMAC_CHINTENCLR_TERR DMAC_CHINTENCLR_TCMPL DMAC_CHINTENCLR_SUSP - uint8_t active_channel; - - // disable irqs ? - __disable_irq(); - active_channel = DMAC->INTPEND.reg & DMAC_INTPEND_ID_Msk; // get channel number - DMAC->CHID.reg = DMAC_CHID_ID(active_channel); - dmadone = DMAC->CHINTFLAG.reg; - DMAC->CHINTFLAG.reg = DMAC_CHINTENCLR_TCMPL; // clear - DMAC->CHINTFLAG.reg = DMAC_CHINTENCLR_TERR; - DMAC->CHINTFLAG.reg = DMAC_CHINTENCLR_SUSP; - __enable_irq(); -} - -void SPIFlash::_zeroDma_init() { - // probably on by default - PM->AHBMASK.reg |= PM_AHBMASK_DMAC ; - PM->APBBMASK.reg |= PM_APBBMASK_DMAC ; - NVIC_EnableIRQ( DMAC_IRQn ) ; - - DMAC->BASEADDR.reg = (uint32_t)descriptor_section; - DMAC->WRBADDR.reg = (uint32_t)wrb; - DMAC->CTRL.reg = DMAC_CTRL_DMAENABLE | DMAC_CTRL_LVLEN(0xf); -} - -void SPIFlash::_zeroSpi_xfr(void *txdata, void *rxdata, size_t n) { - uint32_t temp_CHCTRLB_reg; - - // set up transmit channel - DMAC->CHID.reg = DMAC_CHID_ID(chnltx); - DMAC->CHCTRLA.reg &= ~DMAC_CHCTRLA_ENABLE; - DMAC->CHCTRLA.reg = DMAC_CHCTRLA_SWRST; - DMAC->SWTRIGCTRL.reg &= (uint32_t)(~(1 << chnltx)); - temp_CHCTRLB_reg = DMAC_CHCTRLB_LVL(0) | - DMAC_CHCTRLB_TRIGSRC(SERCOM4_DMAC_ID_TX) | DMAC_CHCTRLB_TRIGACT_BEAT; - DMAC->CHCTRLB.reg = temp_CHCTRLB_reg; - DMAC->CHINTENSET.reg = DMAC_CHINTENSET_MASK ; // enable all 3 interrupts - descriptor.descaddr = 0; - descriptor.dstaddr = (uint32_t) &sercom->SPI.DATA.reg; - descriptor.btcnt = n; - descriptor.srcaddr = (uint32_t)txdata; - descriptor.btctrl = DMAC_BTCTRL_VALID; - if (xtype != DoRX) { - descriptor.srcaddr += n; - descriptor.btctrl |= DMAC_BTCTRL_SRCINC; - } - memcpy(&descriptor_section[chnltx],&descriptor, sizeof(dmacdescriptor)); - - // rx channel enable interrupts - DMAC->CHID.reg = DMAC_CHID_ID(chnlrx); - DMAC->CHCTRLA.reg &= ~DMAC_CHCTRLA_ENABLE; - DMAC->CHCTRLA.reg = DMAC_CHCTRLA_SWRST; - DMAC->SWTRIGCTRL.reg &= (uint32_t)(~(1 << chnlrx)); - temp_CHCTRLB_reg = DMAC_CHCTRLB_LVL(0) | - DMAC_CHCTRLB_TRIGSRC(SERCOM4_DMAC_ID_RX) | DMAC_CHCTRLB_TRIGACT_BEAT; - DMAC->CHCTRLB.reg = temp_CHCTRLB_reg; - DMAC->CHINTENSET.reg = DMAC_CHINTENSET_MASK ; // enable all 3 interrupts - dmadone = 0; - descriptor.descaddr = 0; - descriptor.srcaddr = (uint32_t) &sercom->SPI.DATA.reg; - descriptor.btcnt = n; - descriptor.dstaddr = (uint32_t)rxdata; - descriptor.btctrl = DMAC_BTCTRL_VALID; - if (xtype != DoTX) { - descriptor.dstaddr += n; - descriptor.btctrl |= DMAC_BTCTRL_DSTINC; - } - memcpy(&descriptor_section[chnlrx],&descriptor, sizeof(dmacdescriptor)); - - // start both channels ? order matter ? - DMAC->CHID.reg = DMAC_CHID_ID(chnltx); - Serial.println("A"); - DMAC->CHCTRLA.reg |= DMAC_CHCTRLA_ENABLE; - system_interrupt_leave_critical_section(); - Serial.println("B"); - DMAC->CHID.reg = DMAC_CHID_ID(chnlrx); - Serial.println("C"); - DMAC->CHCTRLA.reg |= DMAC_CHCTRLA_ENABLE; - Serial.println("D"); - - while(!dmadone); // await DMA done isr - - DMAC->CHID.reg = DMAC_CHID_ID(chnltx); //disable DMA to allow lib SPI - DMAC->CHCTRLA.reg &= ~DMAC_CHCTRLA_ENABLE; - DMAC->CHID.reg = DMAC_CHID_ID(chnlrx); - DMAC->CHCTRLA.reg &= ~DMAC_CHCTRLA_ENABLE; -} - -void SPIFlash::_zeroSpi_write(void *data, size_t n) { - xtype = DoTX; - Serial.println("SPI_write started"); - _zeroSpi_xfr(data,rxsink,n); - Serial.println("WRITE DONE"); -} -void SPIFlash::_zeroSpi_read(void *data, size_t n) { - xtype = DoRX; - _zeroSpi_xfr(txsrc,data,n); - Serial.println("READ DONE"); -} -void SPIFlash::_zeroSpi_transfer(void *txdata, void *rxdata, size_t n) { - xtype = DoTXRX; - _zeroSpi_xfr(txdata,rxdata,n); - Serial.println("transfer done"); -} diff --git a/extras/Library speed comparisons.xlsx b/extras/Library speed comparisons.xlsx old mode 100755 new mode 100644 index 4ac6b3d61a97fd11bfc24320e8569b52842d378d..1c4586c43743a90c3334f3c741ae8e47e270b4c6 GIT binary patch literal 47493 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

SoztOPhy3&)%JfULiN(h;E3W!0 zsHCgDV2y4zv~;-?izNE4LGS)eBY$2+{@8YO0(invWa1M|>jaoXS}rGtH_j(K zs7HNS^Ytwti{$SP3JY5R(7!qdA^^@g8;GSKD=QUfm({@TaTMjx%|zJ+@K0eL=K4Oa zws5oKb_&(_6Hs&WGRF>Jr3o}_;y6eCK9jQ-V^l^09O{^J+zp>@)kyo6&p(bQ8)K=ug*Gq#Ee!1Q%a0fLAV0LI+*%(*Tm z^v`+HDG~&dDguEpXHA&am@en3J%8y*k^XQ%F|GgW3SBJwpH%3?mBWPo=?0t@2n6D* d_*;ek(%Xd)~e&Z{{U2PGJ603 literal 48504 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 diff --git a/extras/SPI pinouts.md b/extras/SPI pinouts.md old mode 100755 new mode 100644 diff --git a/extras/Winbond Flash Instructions - Comparison.xlsx b/extras/Winbond Flash Instructions - Comparison.xlsx old mode 100755 new mode 100644 diff --git a/keywords.txt b/keywords.txt old mode 100755 new mode 100644 index f10adaa..92b31c3 --- a/keywords.txt +++ b/keywords.txt @@ -1,9 +1,9 @@ ####################################### -# Syntax Coloring Map SPIFlash +# Syntax Coloring Map SPI ####################################### ####################################### -# Class (KEYWORD1) +# Datatypes (KEYWORD1) ####################################### SPIFlash KEYWORD1 @@ -13,16 +13,14 @@ SPIFlash KEYWORD1 ####################################### begin KEYWORD2 setClock KEYWORD2 -libver KEYWORD2 error KEYWORD2 -getManID KEYWORD2 +libver KEYWORD2 getJEDECID KEYWORD2 -getUniqueID KEYWORD2 +getManID KEYWORD2 getAddress KEYWORD2 -sizeofStr KEYWORD2 getCapacity KEYWORD2 getMaxPage KEYWORD2 -functionRunTime KEYWORD2 +sizeofStr KEYWORD2 readByte KEYWORD2 readByteArray KEYWORD2 readChar KEYWORD2 @@ -54,20 +52,7 @@ resumeProg KEYWORD2 powerUp KEYWORD2 powerDown KEYWORD2 -####################################### -# Constants (LITERAL1) -####################################### -WINBOND_MANID LITERAL1 -MICROCHIP_MANID LITERAL1 -CYPRESS_MANID LITERAL1 -ADESTO_MANID LITERAL1 -MICRON_MANID LITERAL1 -NULLBYTE LITERAL1 -NULLINT LITERAL1 -BYTE LITERAL1 -KiB LITERAL1 -MiB LITERAL1 ####################################### -# Built-in variables (LITERAL2) +# Constants (LITERAL1) ####################################### diff --git a/library.properties b/library.properties index d3ad686..08fc05b 100644 --- a/library.properties +++ b/library.properties @@ -1,10 +1,10 @@ name=SPIFlash -version=3.0.0 +version=2.7.0 author=Prajwal Bhattaram maintainer=Prajwal Bhattaram sentence=Winbond SPI flash library for Arduino. paragraph=This library enables read, write, erase and power functions on the following Winbond NOR Flash chips - W25X05CL, W25X10BV, W25X20BV, W25X40BV, W25Q80BV, W25Q16BV, W25Q32BV, W25Q64BV & W25Q128BV. All other Winbond flash chips can also be used with this library from v2.6.0 onwards. Refer to change log for further information about this release. category=Data Storage url=https://github.com/Marzogh/SPIFlash -architectures=avr,sam,samd,esp8266,esp32,Simblee,ameba +architectures=avr,sam,samd,esp8266,esp32,simblee,ameba includes=SPIFlash.h diff --git a/src/DMASAMD.h b/src/DMASAMD.h deleted file mode 100644 index a319555..0000000 --- a/src/DMASAMD.h +++ /dev/null @@ -1,140 +0,0 @@ -// DMA memory to memory ZERO -// ch 18 beat burst block -// xdk sam0/drivers/dma/dma.c -// packages/arduino/tools/CMSIS/4.0.0-atmel/Device/ATMEL/samd21/include/component/dmac.h -// http://asf.atmel.com/docs/3.16.0/samd21/html/asfdoc_sam0_sercom_spi_dma_use_case.html -// assume normal SPI setup, then we take over with DMA -#ifdef ARDUINO_ARCH_SAMD -#include - -#define PRREG(x) Serial.print(#x" 0x"); Serial.println(x,HEX) - - -#define BYTES 1024 -char txbuf[BYTES], rxbuf[BYTES]; - -/*void prmbs(char *lbl,unsigned long us,int bits) { - float mbs = (float)bits/us; - Serial.print(mbs,2); Serial.print(" mbs "); - Serial.print(us); Serial.print(" us "); - Serial.println(lbl); -}*/ - -// DMA 12 channels -typedef struct { - uint16_t btctrl; - uint16_t btcnt; - uint32_t srcaddr; - uint32_t dstaddr; - uint32_t descaddr; -} dmacdescriptor ; -volatile dmacdescriptor wrb[12] __attribute__ ((aligned (16))); -dmacdescriptor descriptor_section[12] __attribute__ ((aligned (16))); -dmacdescriptor descriptor __attribute__ ((aligned (16))); - -static uint32_t chnltx = 0, chnlrx = 1; // DMA channels -enum XfrType { DoTX, DoRX, DoTXRX}; -static XfrType xtype; -static uint8_t rxsink[1], txsrc[1] = {0xFF}; -volatile uint32_t dmadone; - -void DMAC_Handler() { - // interrupts DMAC_CHINTENCLR_TERR DMAC_CHINTENCLR_TCMPL DMAC_CHINTENCLR_SUSP - uint8_t active_channel; - - // disable irqs ? - __disable_irq(); - active_channel = DMAC->INTPEND.reg & DMAC_INTPEND_ID_Msk; // get channel number - DMAC->CHID.reg = DMAC_CHID_ID(active_channel); - dmadone = DMAC->CHINTFLAG.reg; - DMAC->CHINTFLAG.reg = DMAC_CHINTENCLR_TCMPL; // clear - DMAC->CHINTFLAG.reg = DMAC_CHINTENCLR_TERR; - DMAC->CHINTFLAG.reg = DMAC_CHINTENCLR_SUSP; - __enable_irq(); -} - -void dma_init() { - // probably on by default - PM->AHBMASK.reg |= PM_AHBMASK_DMAC ; - PM->APBBMASK.reg |= PM_APBBMASK_DMAC ; - NVIC_EnableIRQ( DMAC_IRQn ) ; - - DMAC->BASEADDR.reg = (uint32_t)descriptor_section; - DMAC->WRBADDR.reg = (uint32_t)wrb; - DMAC->CTRL.reg = DMAC_CTRL_DMAENABLE | DMAC_CTRL_LVLEN(0xf); -} - - -Sercom *sercom = (Sercom *)SERCOM4; // SPI SERCOM - -void spi_xfr(void *txdata, void *rxdata, size_t n) { - uint32_t temp_CHCTRLB_reg; - - // set up transmit channel - DMAC->CHID.reg = DMAC_CHID_ID(chnltx); - DMAC->CHCTRLA.reg &= ~DMAC_CHCTRLA_ENABLE; - DMAC->CHCTRLA.reg = DMAC_CHCTRLA_SWRST; - DMAC->SWTRIGCTRL.reg &= (uint32_t)(~(1 << chnltx)); - temp_CHCTRLB_reg = DMAC_CHCTRLB_LVL(0) | - DMAC_CHCTRLB_TRIGSRC(SERCOM4_DMAC_ID_TX) | DMAC_CHCTRLB_TRIGACT_BEAT; - DMAC->CHCTRLB.reg = temp_CHCTRLB_reg; - DMAC->CHINTENSET.reg = DMAC_CHINTENSET_MASK ; // enable all 3 interrupts - descriptor.descaddr = 0; - descriptor.dstaddr = (uint32_t) &sercom->SPI.DATA.reg; - descriptor.btcnt = n; - descriptor.srcaddr = (uint32_t)txdata; - descriptor.btctrl = DMAC_BTCTRL_VALID; - if (xtype != DoRX) { - descriptor.srcaddr += n; - descriptor.btctrl |= DMAC_BTCTRL_SRCINC; - } - memcpy(&descriptor_section[chnltx],&descriptor, sizeof(dmacdescriptor)); - - // rx channel enable interrupts - DMAC->CHID.reg = DMAC_CHID_ID(chnlrx); - DMAC->CHCTRLA.reg &= ~DMAC_CHCTRLA_ENABLE; - DMAC->CHCTRLA.reg = DMAC_CHCTRLA_SWRST; - DMAC->SWTRIGCTRL.reg &= (uint32_t)(~(1 << chnlrx)); - temp_CHCTRLB_reg = DMAC_CHCTRLB_LVL(0) | - DMAC_CHCTRLB_TRIGSRC(SERCOM4_DMAC_ID_RX) | DMAC_CHCTRLB_TRIGACT_BEAT; - DMAC->CHCTRLB.reg = temp_CHCTRLB_reg; - DMAC->CHINTENSET.reg = DMAC_CHINTENSET_MASK ; // enable all 3 interrupts - dmadone = 0; - descriptor.descaddr = 0; - descriptor.srcaddr = (uint32_t) &sercom->SPI.DATA.reg; - descriptor.btcnt = n; - descriptor.dstaddr = (uint32_t)rxdata; - descriptor.btctrl = DMAC_BTCTRL_VALID; - if (xtype != DoTX) { - descriptor.dstaddr += n; - descriptor.btctrl |= DMAC_BTCTRL_DSTINC; - } - memcpy(&descriptor_section[chnlrx],&descriptor, sizeof(dmacdescriptor)); - - // start both channels ? order matter ? - DMAC->CHID.reg = DMAC_CHID_ID(chnltx); - DMAC->CHCTRLA.reg |= DMAC_CHCTRLA_ENABLE; - DMAC->CHID.reg = DMAC_CHID_ID(chnlrx); - DMAC->CHCTRLA.reg |= DMAC_CHCTRLA_ENABLE; - - while(!dmadone); // await DMA done isr - - DMAC->CHID.reg = DMAC_CHID_ID(chnltx); //disable DMA to allow lib SPI - DMAC->CHCTRLA.reg &= ~DMAC_CHCTRLA_ENABLE; - DMAC->CHID.reg = DMAC_CHID_ID(chnlrx); - DMAC->CHCTRLA.reg &= ~DMAC_CHCTRLA_ENABLE; -} - -void spi_write(void *data, size_t n) { - xtype = DoTX; - spi_xfr(data,rxsink,n); -} -void spi_read(void *data, size_t n) { - xtype = DoRX; - spi_xfr(txsrc,data,n); -} -void spi_transfer(void *txdata, void *rxdata, size_t n) { - xtype = DoTXRX; - spi_xfr(txdata,rxdata,n); -} -#endif diff --git a/src/SAM_DMASPI.cpp b/src/DMASPI.cpp similarity index 100% rename from src/SAM_DMASPI.cpp rename to src/DMASPI.cpp diff --git a/src/FLASHIO.cpp b/src/FLASHIO.cpp deleted file mode 100644 index d3f97af..0000000 --- a/src/FLASHIO.cpp +++ /dev/null @@ -1,491 +0,0 @@ -/* Arduino SPIFlash Library v.3.0.0 - * Copyright (C) 2017 by Prajwal Bhattaram - * Created by Prajwal Bhattaram - 04/11/2017 - * - * This file is part of the Arduino SPIFlash Library. This library is for - * Winbond NOR flash memory modules. In its current form it enables reading - * and writing individual data variables, structs and arrays from and to various locations; - * reading and writing pages; continuous read functions; sector, block and chip erase; - * suspending and resuming programming/erase and powering down for low power operation. - * - * This Library is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This Library is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License v3.0 - * along with the Arduino SPIFlash Library. If not, see - * . - */ - -#include "SPIFlash.h" - -#include "DMASAMD.h" -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -// Private functions used by read, write and erase operations // -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// - -//Double checks all parameters before calling a read or write. Comes in two variants -//Takes address and returns the address if true, else returns false. Throws an error if there is a problem. -bool SPIFlash::_prep(uint8_t opcode, uint32_t _addr, uint32_t size) { - switch (opcode) { - case PAGEPROG: - //Serial.print(F("Address being prepped: ")); - //Serial.println(_addr); - #ifndef HIGHSPEED - if(!_addressCheck(_addr, size) || !_notPrevWritten(_addr, size) || !_notBusy() || !_writeEnable()) { - return false; - } - #else - if (!_addressCheck(_addr, size) || !_notBusy() || !_writeEnable()) { - return false; - } - #endif - return true; - break; - - case ERASEFUNC: - _currentAddress = _addr; - if(!_notBusy()||!_writeEnable()) { - return false; - } - return true; - break; - - default: - if (!_addressCheck(_addr, size) || !_notBusy()) { - return false; - } - #ifdef ENABLEZERODMA - _delay_us(3500L); - #endif - return true; - break; - } -} - -// Transfer Address. -bool SPIFlash::_transferAddress(void) { - _nextByte(WRITE, Higher(_currentAddress)); - _nextByte(WRITE, Hi(_currentAddress)); - _nextByte(WRITE, Lo(_currentAddress)); - /*_nextByte(_currentAddress >> 16); - _nextByte(_currentAddress >> 8); - _nextByte(_currentAddress);*/ - return true; -} - -bool SPIFlash::_startSPIBus(void) { - #ifndef SPI_HAS_TRANSACTION - noInterrupts(); - #endif - - #if defined (ARDUINO_ARCH_SAM) - _dueSPIInit(DUE_SPI_CLK); - #elif defined (ARDUINO_ARCH_SAMD) - #ifdef SPI_HAS_TRANSACTION - _spi->beginTransaction(_settings); - #else - _spi->setClockDivider(SPI_CLOCK_DIV_4); - _spi->setDataMode(SPI_MODE0); - _spi->setBitOrder(MSBFIRST); - #endif - #if defined ENABLEZERODMA - dma_init(); - #endif - #else - #if defined (ARDUINO_ARCH_AVR) - //save current SPI settings - _SPCR = SPCR; - _SPSR = SPSR; - #endif - #ifdef SPI_HAS_TRANSACTION - SPI.beginTransaction(_settings); - #else - SPI.setClockDivider(SPI_CLOCK_DIV4); - SPI.setDataMode(SPI_MODE0); - SPI.setBitOrder(MSBFIRST); - #endif - #endif - SPIBusState = true; - return true; -} - -//Initiates SPI operation - but data is not transferred yet. Always call _prep() before this function (especially when it involves writing or reading to/from an address) -bool SPIFlash::_beginSPI(uint8_t opcode) { - if (!SPIBusState) { - _startSPIBus(); - } - CHIP_SELECT - switch (opcode) { - case READDATA: - _nextByte(WRITE, opcode); - _transferAddress(); - break; - - case PAGEPROG: - _nextByte(WRITE, opcode); - _transferAddress(); - break; - - case FASTREAD: - _nextByte(WRITE, opcode); - _nextByte(WRITE, DUMMYBYTE); - _transferAddress(); - - case SECTORERASE: - _nextByte(WRITE, opcode); - _transferAddress(); - - case BLOCK32ERASE: - _nextByte(WRITE, opcode); - _transferAddress(); - - case BLOCK64ERASE: - _nextByte(WRITE, opcode); - _transferAddress(); - - default: - _nextByte(WRITE, opcode); - break; - } - return true; -} -//SPI data lines are left open until _endSPI() is called - -//Reads/Writes next byte. Call 'n' times to read/write 'n' number of bytes. Should be called after _beginSPI() -uint8_t SPIFlash::_nextByte(char IOType, uint8_t data) { -#if defined (ARDUINO_ARCH_SAMD) - #ifdef ENABLEZERODMA - union { - uint8_t dataBuf[1]; - uint8_t val; - } rxData, txData; - txData.val = data; - spi_transfer(txData.dataBuf, rxData.dataBuf, 1); - return rxData.val; - #else - return xfer(data); - #endif -#else - return xfer(data); -#endif -} - -//Reads/Writes next int. Call 'n' times to read/write 'n' number of bytes. Should be called after _beginSPI() -uint16_t SPIFlash::_nextInt(uint16_t data) { -#if defined (ARDUINO_ARCH_SAMD) - return _spi->transfer16(data); -#else - return SPI.transfer16(data); -#endif -} - -//Reads/Writes next data buffer. Call 'n' times to read/write 'n' number of bytes. Should be called after _beginSPI() -void SPIFlash::_nextBuf(uint8_t opcode, uint8_t *data_buffer, uint32_t size) { - uint8_t *_dataAddr = &(*data_buffer); - switch (opcode) { - case READDATA: - #if defined (ARDUINO_ARCH_SAM) - _dueSPIRecByte(&(*data_buffer), size); - #elif defined (ARDUINO_ARCH_SAMD) - #ifdef ENABLEZERODMA - spi_read(&(*data_buffer), size); - #else - _spi->transfer(&data_buffer[0], size); - #endif - #elif defined (ARDUINO_ARCH_AVR) - SPI.transfer(&(*data_buffer), size); - #else - for (uint16_t i = 0; i < size; i++) { - *_dataAddr = xfer(NULLBYTE); - _dataAddr++; - } - #endif - break; - - case PAGEPROG: - #if defined (ARDUINO_ARCH_SAM) - _dueSPISendByte(&(*data_buffer), size); - #elif defined (ARDUINO_ARCH_SAMD) - #ifdef ENABLEZERODMA - spi_write(&(*data_buffer), size); - #else - _spi->transfer(&(*data_buffer), size); - #endif - #elif defined (ARDUINO_ARCH_AVR) - SPI.transfer(&(*data_buffer), size); - #else - for (uint16_t i = 0; i < size; i++) { - xfer(*_dataAddr); - _dataAddr++; - } - #endif - break; - } -} - -//Stops all operations. Should be called after all the required data is read/written from repeated _nextByte() calls -void SPIFlash::_endSPI(void) { - CHIP_DESELECT -#ifdef SPI_HAS_TRANSACTION - #if defined (ARDUINO_ARCH_SAMD) - _spi->endTransaction(); - #else - SPI.endTransaction(); - #endif -#else - interrupts(); -#endif -#if defined (ARDUINO_ARCH_AVR) - SPCR = _SPCR; - SPSR = _SPSR; -#endif - SPIBusState = false; -} - -// Checks if status register 1 can be accessed - used to check chip status, during powerdown and power up and for debugging -uint8_t SPIFlash::_readStat1(void) { - _beginSPI(READSTAT1); - stat1 = _nextByte(READ); - CHIP_DESELECT - return stat1; -} - -// Checks if status register 2 can be accessed, if yes, reads and returns it -uint8_t SPIFlash::_readStat2(void) { - _beginSPI(READSTAT2); - uint8_t stat2 = _nextByte(READ); - stat2 = _nextByte(READ); - CHIP_DESELECT - return stat2; -} - -// Checks the erase/program suspend flag before enabling/disabling a program/erase suspend operation -bool SPIFlash::_noSuspend(void) { - switch (_chip.manufacturerID) { - case WINBOND_MANID: - if(_readStat2() & SUS) { - _troubleshoot(SYSSUSPEND); - return false; - } - return true; - break; - - case MICROCHIP_MANID: - _readStat1(); - if(stat1 & WSE || stat1 & WSP) { - _troubleshoot(SYSSUSPEND); - return false; - } - } - return true; -} - -// Polls the status register 1 until busy flag is cleared or timeout -bool SPIFlash::_notBusy(uint32_t timeout) { - _delay_us(WINBOND_WRITE_DELAY); - uint32_t _time = micros(); - - do { - _readStat1(); - if (!(stat1 & BUSY)) - { - return true; - } - _time++; - } while ((micros() - _time) < timeout); - if ((micros() - _time) == timeout) { - return false; - } - return true; -} - -//Enables writing to chip by setting the WRITEENABLE bit -bool SPIFlash::_writeEnable(bool _troubleshootEnable) { - _beginSPI(WRITEENABLE); - CHIP_DESELECT - if (!(_readStat1() & WRTEN)) { - if (_troubleshootEnable) { - _troubleshoot(CANTENWRITE); - } - return false; - } - return true; -} - -//Disables writing to chip by setting the Write Enable Latch (WEL) bit in the Status Register to 0 -//_writeDisable() is not required under the following conditions because the Write Enable Latch (WEL) flag is cleared to 0 -// i.e. to write disable state: -// Power-up, Write Disable, Page Program, Quad Page Program, Sector Erase, Block Erase, Chip Erase, Write Status Register, -// Erase Security Register and Program Security register -bool SPIFlash::_writeDisable(void) { - _beginSPI(WRITEDISABLE); - CHIP_DESELECT - return true; -} - -//Checks the device ID to establish storage parameters -bool SPIFlash::_getManId(uint8_t *b1, uint8_t *b2) { - if(!_notBusy()) - return false; - _beginSPI(MANID); - _nextByte(READ); - _nextByte(READ); - _nextByte(READ); - *b1 = _nextByte(READ); - *b2 = _nextByte(READ); - CHIP_DESELECT - return true; -} - -//Checks for presence of chip by requesting JEDEC ID -bool SPIFlash::_getJedecId(void) { - if(!_notBusy()) { - return false; - } - _beginSPI(JEDECID); - _chip.manufacturerID = _nextByte(READ); // manufacturer id - _chip.memoryTypeID = _nextByte(READ); // memory type - _chip.capacityID = _nextByte(READ); // capacity - CHIP_DESELECT - if (!_chip.manufacturerID) { - _troubleshoot(NORESPONSE); - return false; - } - else { - return true; - } -} - -bool SPIFlash::_getSFDP(void) { - if(!_notBusy()) { - return false; - } - _currentAddress = 0x00; - _beginSPI(READSFDP); - _transferAddress(); - _nextByte(WRITE, DUMMYBYTE); - /*for (uint8_t i = 0; i < 4; i++) { - _chip.sfdp += (_nextByte(READ) << (8*i)); - }*/ - // Modification suggested by @VitorBoss on 13.08.2017. - // Ref issue 76 on Github here --> - // https://github.com/Marzogh/SPIFlash/issues/76 - Lo(_chip.sfdp) = _nextByte(READ); - Hi(_chip.sfdp) = _nextByte(READ); - Higher(_chip.sfdp) = _nextByte(READ); - Highest(_chip.sfdp) = _nextByte(READ); - CHIP_DESELECT - if (_chip.sfdp == 0x50444653) { - //Serial.print("_chip.sfdp: "); - //Serial.println(_chip.sfdp, HEX); - return true; - } - else { - return false; - } -} - -bool SPIFlash::_disableGlobalBlockProtect(void) { - if (_chip.memoryTypeID == SST25) { - _readStat1(); - stat1 &= 0xC3; - _beginSPI(WRITESTATEN); - CHIP_DESELECT - _beginSPI(WRITESTAT); - _nextByte(WRITE, stat1); - CHIP_DESELECT - } - else if (_chip.memoryTypeID == SST26) { - if(!_notBusy()) { - return false; - } - _writeEnable(); - _delay_us(10); - _beginSPI(ULBPR); - CHIP_DESELECT - _delay_us(50); - _writeDisable(); - } - return true; -} - -//Identifies the chip -bool SPIFlash::_chipID(void) { - //Get Manfucturer/Device ID so the library can identify the chip - //_getSFDP(); - if (!_getJedecId()) { - return false; - } - - if (_chip.manufacturerID == MICROCHIP_MANID) { - _disableGlobalBlockProtect(); - } - - if (!_chip.capacity) { - if (_chip.manufacturerID == WINBOND_MANID || _chip.manufacturerID == MICROCHIP_MANID || _chip.manufacturerID == CYPRESS_MANID || _chip.manufacturerID == ADESTO_MANID || _chip.manufacturerID == MICRON_MANID) { - //Identify capacity - for (uint8_t i = 0; i < sizeof(_capID); i++) { - if (_chip.capacityID == _capID[i]) { - _chip.capacity = (_memSize[i]); - _chip.supported = true; - return true; - } - } - if (!_chip.capacity) { - _troubleshoot(UNKNOWNCAP); - return false; - } - } - else { - _troubleshoot(UNKNOWNCHIP); //Error code for unidentified capacity - return false; - } - } - return true; -} - - -//Checks to see if page overflow is permitted and assists with determining next address to read/write. -//Sets the global address variable -bool SPIFlash::_addressCheck(uint32_t _addr, uint32_t size) { - if (errorcode == UNKNOWNCAP || errorcode == NORESPONSE) { - return false; - } - if (!_chip.capacity) { - _troubleshoot(CALLBEGIN); - return false; - } - - //for (uint32_t i = 0; i < size; i++) { - if (_addr + size >= _chip.capacity) { - #ifdef DISABLEOVERFLOW - _troubleshoot(OUTOFBOUNDS); - return false; // At end of memory - (!pageOverflow) - #else - _currentAddress = 0x00; - return true; // At end of memory - (pageOverflow) - #endif - } - //} - _currentAddress = _addr; - return true; // Not at end of memory if (address < _chip.capacity) -} - -bool SPIFlash::_notPrevWritten(uint32_t _addr, uint32_t size) { - _beginSPI(READDATA); - for (uint32_t i = 0; i < size; i++) { - if (_nextByte(READ) != 0xFF) { - CHIP_DESELECT; - return false; - } - } - CHIP_DESELECT - return true; -} diff --git a/src/SPIFlash.cpp b/src/SPIFlash.cpp index df805c8..b9da54f 100644 --- a/src/SPIFlash.cpp +++ b/src/SPIFlash.cpp @@ -1,8 +1,8 @@ -/* Arduino SPIFlash Library v.3.0.0 +/* Arduino SPIFlash Library v.2.7.0 * Copyright (C) 2017 by Prajwal Bhattaram * Created by Prajwal Bhattaram - 19/05/2015 * Modified by @boseji - 02/03/2017 - * Modified by Prajwal Bhattaram - 04/11/2017 + * Modified by Prajwal Bhattaram - 02/08/2017 * * This file is part of the Arduino SPIFlash Library. This library is for * Winbond NOR flash memory modules. In its current form it enables reading @@ -28,85 +28,476 @@ #include "SPIFlash.h" // Constructor -//If board has multiple SPI interfaces, this constructor lets the user choose between them -// Adding Low level HAL API to initialize the Chip select pinMode on RTL8195A - @boseji 2nd March 2017 #if defined (ARDUINO_ARCH_AVR) -SPIFlash::SPIFlash(uint8_t cs) { +SPIFlash::SPIFlash(uint8_t cs, bool overflow) { csPin = cs; +#ifndef __AVR_ATtiny85__ + cs_port = portOutputRegister(digitalPinToPort(csPin)); +#endif cs_mask = digitalPinToBitMask(csPin); + pageOverflow = overflow; pinMode(csPin, OUTPUT); - CHIP_DESELECT -} -#elif defined (ARDUINO_ARCH_SAMD) || defined(ARCH_STM32) -SPIFlash::SPIFlash(uint8_t cs, SPIClass *spiinterface) { - _spi = spiinterface; //Sets SPI interface - if no user selection is made, this defaults to SPI - csPin = cs; - pinMode(csPin, OUTPUT); - CHIP_DESELECT } +// Adding Low level HAL API to initialize the Chip select pinMode on RTL8195A - @boseji 2nd March 2017 #elif defined (BOARD_RTL8195A) -SPIFlash::SPIFlash(PinName cs) { +SPIFlash::SPIFlash(PinName cs, bool overflow) { gpio_init(&csPin, cs); gpio_dir(&csPin, PIN_OUTPUT); gpio_mode(&csPin, PullNone); gpio_write(&csPin, 1); - CHIP_DESELECT + pageOverflow = overflow; } #else -SPIFlash::SPIFlash(uint8_t cs) { +SPIFlash::SPIFlash(uint8_t cs, bool overflow) { csPin = cs; + pageOverflow = overflow; pinMode(csPin, OUTPUT); - CHIP_DESELECT } #endif - - //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -// Public functions used for read, write and erase operations // +// Private functions used by read, write and erase operations // //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -//Identifies chip and establishes parameters -bool SPIFlash::begin(size_t flashChipSize) { -#ifdef RUNDIAGNOSTIC - Serial.println("Chip Diagnostics initiated."); - Serial.println(); -#endif -#ifdef HIGHSPEED - Serial.println("Highspeed mode initiated."); - Serial.println(); +//Double checks all parameters before calling a read or write. Comes in two variants +//Variant A: Takes address and returns the address if true, else returns false. Throws an error if there is a problem. +bool SPIFlash::_prep(uint8_t opcode, uint32_t address, uint32_t size) { + switch (opcode) { + case PAGEPROG: + if (!_addressCheck(address, size)) { + return false; + } + if(!_notBusy() || !_writeEnable()){ + return false; + } + #ifndef HIGHSPEED + if(!_notPrevWritten(address, size)) { + return false; + } + #endif + return true; + break; + + default: + if (!_addressCheck(address, size)) { + return false; + } + if (!_notBusy()){ + return false; + } + return true; + break; + } +} + +//Variant B: Take the opcode, page number, offset and size of data block as arguments +bool SPIFlash::_prep(uint8_t opcode, uint32_t page_number, uint8_t offset, uint32_t size) { + uint32_t address = _getAddress(page_number, offset); + return _prep(opcode, address, size); +} + +bool SPIFlash::_transferAddress(void) { + _nextByte(_currentAddress >> 16); + _nextByte(_currentAddress >> 8); + _nextByte(_currentAddress); +} + +bool SPIFlash::_startSPIBus(void) { +#ifndef __AVR_ATtiny85__ + #ifndef SPI_HAS_TRANSACTION + noInterrupts(); + #endif + + #if defined (ARDUINO_ARCH_SAM) + _dueSPIInit(DUE_SPI_CLK); + #else + #if defined (ARDUINO_ARCH_AVR) + //save current SPI settings + _SPCR = SPCR; + _SPSR = SPSR; + #endif + #ifdef SPI_HAS_TRANSACTION + SPI.beginTransaction(_settings); + #else + SPI.setClockDivider(SPI_CLOCK_DIV_4) + SPI.setDataMode(SPI_MODE0); + SPI.setBitOrder(MSBFIRST); + #endif + #endif +#else + #endif - BEGIN_SPI -#ifdef SPI_HAS_TRANSACTION - //Define the settings to be used by the SPI bus - _settings = SPISettings(SPI_CLK, MSBFIRST, SPI_MODE0); + SPIBusState = true; + return true; +} + +//Initiates SPI operation - but data is not transferred yet. Always call _prep() before this function (especially when it involves writing or reading to/from an address) +bool SPIFlash::_beginSPI(uint8_t opcode) { + if (!SPIBusState) { + _startSPIBus(); + } + CHIP_SELECT + switch (opcode) { + case FASTREAD: + _nextByte(opcode); + _nextByte(DUMMYBYTE); + _transferAddress(); + break; + + case READDATA: + _nextByte(opcode); + _transferAddress(); + break; + + case PAGEPROG: + _nextByte(opcode); + _transferAddress(); + break; + + default: + _nextByte(opcode); + break; + } + return true; +} +//SPI data lines are left open until _endSPI() is called + +//Reads/Writes next byte. Call 'n' times to read/write 'n' number of bytes. Should be called after _beginSPI() +uint8_t SPIFlash::_nextByte(uint8_t data) { +#if defined (ARDUINO_ARCH_SAM) + return _dueSPITransfer(data); +#else + return xfer(data); #endif -// If no capacity is defined in user code - if (!flashChipSize) { +} + +//Reads/Writes next int. Call 'n' times to read/write 'n' number of bytes. Should be called after _beginSPI() +uint16_t SPIFlash::_nextInt(uint16_t data) { + #ifndef __AVR_ATtiny85__ + return SPI.transfer16(data); + #else + uint16_t _data; + _data = xfer(data >> 0); + data = (_data << 8); + _data += xfer(data >> 8); + return _data; + #endif +} + +//Reads/Writes next data buffer. Call 'n' times to read/write 'n' number of bytes. Should be called after _beginSPI() +void SPIFlash::_nextBuf(uint8_t opcode, uint8_t *data_buffer, uint32_t size) { + uint8_t *_dataAddr = &(*data_buffer); + switch (opcode) { + case READDATA: + #if defined (ARDUINO_ARCH_SAM) + _dueSPIRecByte(&(*data_buffer), size); + #elif defined (ARDUINO_ARCH_AVR) && !defined (__AVR_ATtiny85__) + SPI.transfer(&data_buffer[0], size); + #else + for (uint16_t i = 0; i < size; i++) { + *_dataAddr = xfer(NULLBYTE); + _dataAddr++; + } + #endif + break; + + case PAGEPROG: + #if defined (ARDUINO_ARCH_SAM) + _dueSPISendByte(&(*data_buffer), size); + #elif defined (ARDUINO_ARCH_AVR) && !defined (__AVR_ATtiny85__) + SPI.transfer(&(*data_buffer), size); + #else + for (uint16_t i = 0; i < size; i++) { + xfer(*_dataAddr); + _dataAddr++; + } + #endif + break; + } +} + +//Stops all operations. Should be called after all the required data is read/written from repeated _nextByte() calls +void SPIFlash::_endSPI(void) { + CHIP_DESELECT + #ifdef SPI_HAS_TRANSACTION + SPI.endTransaction(); + #else + interrupts(); + #endif + + #if defined (ARDUINO_ARCH_AVR) && !defined (__AVR_ATtiny85__) + SPCR = _SPCR; + SPSR = _SPSR; + #endif + SPIBusState = false; +} + +// Checks if status register 1 can be accessed - used during powerdown and power up and for debugging +uint8_t SPIFlash::_readStat1(void) { + _beginSPI(READSTAT1); + uint8_t stat1 = _nextByte(); + CHIP_DESELECT + return stat1; +} + +// Checks if status register 2 can be accessed, if yes, reads and returns it +uint8_t SPIFlash::_readStat2(void) { + _beginSPI(READSTAT2); + uint8_t stat2 = _nextByte(); + stat2 = _nextByte(); + CHIP_DESELECT + return stat2; +} + +// Checks the erase/program suspend flag before enabling/disabling a program/erase suspend operation +bool SPIFlash::_noSuspend(void) { + switch (_chip.manufacturerID) { + case WINBOND_MANID: + if(_readStat2() & SUS) { + errorcode = SYSSUSPEND; + return false; + } + return true; + break; + + case MICROCHIP_MANID: + if(_readStat1() & WSE || _readStat1() & WSP) { + errorcode = SYSSUSPEND; + return false; + } + return true; + } + +} + +// Polls the status register 1 until busy flag is cleared or timeout +bool SPIFlash::_notBusy(uint32_t timeout) { + uint32_t startTime = millis(); + + do { + state = _readStat1(); + if((millis()-startTime) > timeout){ + errorcode = CHIPBUSY; + #ifdef RUNDIAGNOSTIC + _troubleshoot(); + #endif + _endSPI(); + return false; + } + } while(state & BUSY); + return true; +} + +//Enables writing to chip by setting the WRITEENABLE bit +bool SPIFlash::_writeEnable(uint32_t timeout) { + uint32_t startTime = millis(); + if (!(state & WRTEN)) { + do { + _beginSPI(WRITEENABLE); + CHIP_DESELECT + state = _readStat1(); + if((millis()-startTime) > timeout) { + errorcode = CANTENWRITE; + #ifdef RUNDIAGNOSTIC + _troubleshoot(); + #endif + _endSPI(); + return false; + } + } while (!(state & WRTEN)); + } + return true; +} + +//Disables writing to chip by setting the Write Enable Latch (WEL) bit in the Status Register to 0 +//_writeDisable() is not required under the following conditions because the Write Enable Latch (WEL) flag is cleared to 0 +// i.e. to write disable state: +// Power-up, Write Disable, Page Program, Quad Page Program, Sector Erase, Block Erase, Chip Erase, Write Status Register, +// Erase Security Register and Program Security register +bool SPIFlash::_writeDisable(void) { + _beginSPI(WRITEDISABLE); + CHIP_DESELECT + return true; +} + +//Gets address from page number and offset. Takes two arguments: +// 1. page_number --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +uint32_t SPIFlash::_getAddress(uint16_t page_number, uint8_t offset) { + uint32_t address = page_number; + return ((address << 8) + offset); +} + +//Checks the device ID to establish storage parameters +bool SPIFlash::_getManId(uint8_t *b1, uint8_t *b2) { + if(!_notBusy()) + return false; + _beginSPI(MANID); + _nextByte(); + _nextByte(); + _nextByte(); + *b1 = _nextByte(); + *b2 = _nextByte(); + CHIP_DESELECT + return true; +} + +//Checks for presence of chip by requesting JEDEC ID +bool SPIFlash::_getJedecId(void) { + if(!_notBusy()) { + return false; + } + _beginSPI(JEDECID); + _chip.manufacturerID = _nextByte(NULLBYTE); // manufacturer id + _chip.memoryTypeID = _nextByte(NULLBYTE); // memory type + _chip.capacityID = _nextByte(NULLBYTE); // capacity + CHIP_DESELECT + if (!_chip.manufacturerID || !_chip.memoryTypeID || !_chip.capacityID) { + errorcode = NORESPONSE; #ifdef RUNDIAGNOSTIC - Serial.println("No Chip size defined by user"); + _troubleshoot(); #endif - bool retVal = _chipID(); - _endSPI(); - return retVal; + return false; + } + else { + return true; + } +} + +bool SPIFlash::_getSFDP(void) { + if(!_notBusy()) { + return false; + } + _beginSPI(READSFDP); + _currentAddress = 0x00; + _transferAddress(); + _nextByte(DUMMYBYTE); + for (uint8_t i = 0; i < 4; i++) { + _chip.sfdp += (_nextByte() << (8*i)); + } + CHIP_DESELECT + if (_chip.sfdp = 0x50444653) { + //Serial.print("_chip.sfdp: "); + //Serial.println(_chip.sfdp, HEX); + return true; + } + else { + return false; + } +} + +//Identifies the chip +bool SPIFlash::_chipID(void) { + //Get Manfucturer/Device ID so the library can identify the chip + _getSFDP(); + if (!_getJedecId()) { + return false; + } + + // If no capacity is defined in user code + if (!_chip.capacity) { + _chip.supported = _chip.manufacturerID; + if (_chip.supported == WINBOND_MANID || _chip.supported == MICROCHIP_MANID) { + //Identify capacity + for (uint8_t i = 0; i < sizeof(_capID); i++) + { + if (_chip.capacityID == _capID[i]) { + _chip.capacity = (_memSize[i]); + _chip.eraseTime = _eraseTime[i]; + } + } + return true; + } + else { + errorcode = UNKNOWNCAP; //Error code for unidentified capacity + #ifdef RUNDIAGNOSTIC + _troubleshoot(); + #endif + return false; + } } else { - _getJedecId(); // If a custom chip size is defined + _chip.eraseTime = _chip.capacity/KB8; + _chip.supported = false;// This chip is not officially supported + errorcode = UNKNOWNCHIP; //Error code for unidentified chip #ifdef RUNDIAGNOSTIC - Serial.println("Custom Chipsize defined"); + _troubleshoot(); #endif - _chip.capacity = flashChipSize; - _chip.supported = false; + //while(1); //Enable this if usercode is not meant to be run on unsupported chips } - _endSPI(); + return true; +} + +//Checks to see if pageOverflow is permitted and assists with determining next address to read/write. +//Sets the global address variable +bool SPIFlash::_addressCheck(uint32_t address, uint32_t size) { + if (errorcode == UNKNOWNCAP || errorcode == NORESPONSE) { + #ifdef RUNDIAGNOSTIC + _troubleshoot(); + #endif + return false; + } + if (!_chip.eraseTime) { + errorcode = CALLBEGIN; + #ifdef RUNDIAGNOSTIC + _troubleshoot(); + #endif + return false; + } - if (_chip.manufacturerID == CYPRESS_MANID) { - setClock(SPI_CLK/4); + for (uint32_t i = 0; i < size; i++) { + if (address + i >= _chip.capacity) { + if (!pageOverflow) { + errorcode = OUTOFBOUNDS; + #ifdef RUNDIAGNOSTIC + _troubleshoot(); + #endif + return false; // At end of memory - (!pageOverflow) + } + else { + _currentAddress = 0x00; + return true; // At end of memory - (pageOverflow) + } + } } + _currentAddress = address; + return true; // Not at end of memory if (address < _chip.capacity) +} +bool SPIFlash::_notPrevWritten(uint32_t address, uint32_t size) { + _beginSPI(READDATA); + for (uint16_t i = 0; i < size; i++) { + if (_nextByte() != 0xFF) { + CHIP_DESELECT; + return false; + } + } + CHIP_DESELECT return true; } +//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// +// Public functions used for read, write and erase operations // +//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + +//Identifies chip and establishes parameters +bool SPIFlash::begin(uint32_t _chipSize) { + if (_chipSize) { + _chip.capacity = _chipSize/8; + } + BEGIN_SPI +#ifdef SPI_HAS_TRANSACTION + //Define the settings to be used by the SPI bus + _settings = SPISettings(SPI_CLK, MSBFIRST, SPI_MODE0); +#endif + if(!_chipID()) { + #ifdef RUNDIAGNOSTIC + _troubleshoot(); + #endif + return false; + } +} + //Allows the setting of a custom clock speed for the SPI bus to communicate with the chip. //Only works if the SPI library in use supports SPI Transactions #ifdef SPI_HAS_TRANSACTION @@ -120,7 +511,7 @@ uint8_t SPIFlash::error(bool _verbosity) { return errorcode; } else { - _troubleshoot(errorcode, PRINTOVERRIDE); + _troubleshoot(); return errorcode; } } @@ -132,12 +523,7 @@ uint32_t SPIFlash::getCapacity(void) { //Returns maximum number of pages uint32_t SPIFlash::getMaxPage(void) { - return (_chip.capacity / SPI_PAGESIZE); -} - -//Returns the time taken to run a function. Must be called immediately after a function is run as the variable returned is overwritten eachtime a function from this library is called. Primarily used in the diagnostics sketch included in the library to track function time. -float SPIFlash::functionRunTime(void) { - return _spifuncruntime; + return (_chip.capacity / PAGESIZE); } //Returns the library version as a string @@ -165,45 +551,32 @@ uint32_t SPIFlash::getJEDECID(void) { return id; } -// Returns a 64-bit Unique ID that is unique to each flash memory chip -uint64_t SPIFlash::getUniqueID(void) { - if(!_notBusy()) { - return false; - } - _beginSPI(UNIQUEID); - for (uint8_t i = 0; i < 4; i++) { - _nextByte(DUMMYBYTE); - } - - for (uint8_t i = 0; i < 8; i++) { - _uniqueID[i] = _nextByte(READ); - } - CHIP_DESELECT - - long long _uid; - for (uint8_t i = 0; i < 8; i++) { - _uid += _uniqueID[i]; - _uid = _uid << 8; - } - return _uid; -} - //Gets the next available address for use. Has two variants: // A. Takes the size of the data as an argument and returns a 32-bit address // B. Takes a three variables, the size of the data and two other variables to return a page number value & an offset into. // All addresses in the in the sketch must be obtained via this function or not at all. // Variant A uint32_t SPIFlash::getAddress(uint16_t size) { - if (!_addressCheck(currentAddress, size)){ - _troubleshoot(OUTOFBOUNDS); + if (!_addressCheck(currentAddress, size)){ + errorcode = OUTOFBOUNDS; + #ifdef RUNDIAGNOSTIC + _troubleshoot(); + #endif return false; } else { - uint32_t _addr = currentAddress; + uint32_t address = currentAddress; currentAddress+=size; - return _addr; + return address; } } +// Variant B +bool SPIFlash::getAddress(uint16_t size, uint16_t &page_number, uint8_t &offset) { + uint32_t address = getAddress(size); + offset = (address >> 0); + page_number = (address >> 8); + return true; +} //Function for returning the size of the string (only to be used for the getAddress() function) uint16_t SPIFlash::sizeofStr(String &inputStr) { @@ -215,51 +588,97 @@ uint16_t SPIFlash::sizeofStr(String &inputStr) { } // Reads a byte of data from a specific location in a page. -// Takes two arguments - -// 1. _addr --> Any address from 0 to capacity -// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true -uint8_t SPIFlash::readByte(uint32_t _addr, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to capacity +// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes three arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +uint8_t SPIFlash::readByte(uint32_t address, bool fastRead) { uint8_t data; - _read(_addr, data, sizeof(data), fastRead); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif + + if (!_prep(READDATA, address, sizeof(data))) { + return false; + } + switch (fastRead) { + case false: + _beginSPI(READDATA); + break; + + case true: + _beginSPI(FASTREAD); + break; + + default: + break; + } + data = _nextByte(); + _endSPI(); return data; } +// Variant B +uint8_t SPIFlash::readByte(uint16_t page_number, uint8_t offset, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); + + return readByte(address, fastRead); +} // Reads a char of data from a specific location in a page. -// Takes two arguments - -// 1. _addr --> Any address from 0 to capacity -// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true -int8_t SPIFlash::readChar(uint32_t _addr, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - int8_t data; - _read(_addr, data, sizeof(data), fastRead); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to capacity +// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes three arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +int8_t SPIFlash::readChar(uint32_t address, bool fastRead) { + int8_t data; + if (!_prep(READDATA, address, sizeof(data))) { + return false; + } + switch (fastRead) { + case false: + _beginSPI(READDATA); + break; + + case true: + _beginSPI(FASTREAD); + break; + + default: + break; + } + data = _nextByte(); + _endSPI(); return data; } +// Variant B +int8_t SPIFlash::readChar(uint16_t page_number, uint8_t offset, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); -// Reads an array of bytes starting from a specific location in a page. -// Takes four arguments -// 1. _addr --> Any address from 0 to capacity -// 2. data_buffer --> The array of bytes to be read from the flash memory - starting at the address indicated -// 3. bufferSize --> The size of the buffer - in number of bytes. -// 4. fastRead --> defaults to false - executes _beginFastRead() if set to true + return readChar(address, fastRead); +} -bool SPIFlash::readByteArray(uint32_t _addr, uint8_t *data_buffer, size_t bufferSize, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - if (!_prep(READDATA, _addr, bufferSize)) { +// Reads an array of bytes starting from a specific location in a page.// Has two variants: +// A. Takes three arguments +// 1. address --> Any address from 0 to capacity +// 2. data_buffer --> The array of bytes to be read from the flash memory - starting at the address indicated +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes four arguments +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data_buffer --> The array of bytes to be read from the flash memory - starting at the offset on the page indicated +// 4. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +bool SPIFlash::readByteArray(uint32_t address, uint8_t *data_buffer, uint16_t bufferSize, bool fastRead) { + if (!_prep(READDATA, address, bufferSize)) { return false; - } + } if(fastRead) { _beginSPI(FASTREAD); } @@ -268,23 +687,28 @@ bool SPIFlash::readByteArray(uint32_t _addr, uint8_t *data_buffer, size_t buffe } _nextBuf(READDATA, &(*data_buffer), bufferSize); _endSPI(); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif return true; } +// Variant B +bool SPIFlash::readByteArray(uint16_t page_number, uint8_t offset, uint8_t *data_buffer, uint16_t bufferSize, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); -// Reads an array of chars starting from a specific location in a page.. -// Takes four arguments -// 1. _addr --> Any address from 0 to capacity -// 2. data_buffer --> The array of bytes to be read from the flash memory - starting at the address indicated -// 3. bufferSize --> The size of the buffer - in number of bytes. -// 4. fastRead --> defaults to false - executes _beginFastRead() if set to true -bool SPIFlash::readCharArray(uint32_t _addr, char *data_buffer, size_t bufferSize, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - if (!_prep(READDATA, _addr, bufferSize)) { + return readByteArray(address, data_buffer, bufferSize, fastRead); +} + +// Reads an array of chars starting from a specific location in a page.// Has two variants: +// A. Takes three arguments +// 1. address --> Any address from 0 to capacity +// 2. data_buffer --> The array of bytes to be read from the flash memory - starting at the address indicated +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes four arguments +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data_buffer --> The array of bytes to be read from the flash memory - starting at the offset on the page indicated +// 4. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +bool SPIFlash::readCharArray(uint32_t address, char *data_buffer, uint16_t bufferSize, bool fastRead) { + if (!_prep(READDATA, address, bufferSize)) { return false; } if(fastRead) { @@ -295,500 +719,1043 @@ bool SPIFlash::readCharArray(uint32_t _addr, char *data_buffer, size_t bufferSi } _nextBuf(READDATA, (uint8_t*) &(*data_buffer), bufferSize); _endSPI(); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif return true; } +// Variant B +bool SPIFlash::readCharArray(uint16_t page_number, uint8_t offset, char *data_buffer, uint16_t bufferSize, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); + + return readCharArray(address, data_buffer, bufferSize, fastRead); +} // Reads an unsigned int of data from a specific location in a page. -// Takes two arguments - -// 1. _addr --> Any address from 0 to capacity -// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true -uint16_t SPIFlash::readWord(uint32_t _addr, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - uint16_t data; - _read(_addr, data, sizeof(data), fastRead); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif - return data; +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to capacity +// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes three arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +uint16_t SPIFlash::readWord(uint32_t address, bool fastRead) { + const uint8_t size = sizeof(uint16_t); + union + { + uint8_t b[size]; + uint16_t I; + } data; + if (!_prep(READDATA, address, size)) { + return false; + } + switch (fastRead) { + case false: + _beginSPI(READDATA); + break; + + case true: + _beginSPI(FASTREAD); + break; + + default: + break; + } + _nextBuf(READDATA, &data.b[0], size); + _endSPI(); + return data.I; +} +// Variant B +uint16_t SPIFlash::readWord(uint16_t page_number, uint8_t offset, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); + + return readWord(address, fastRead); } // Reads a signed int of data from a specific location in a page. -// Takes two arguments - -// 1. _addr --> Any address from 0 to capacity -// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true -int16_t SPIFlash::readShort(uint32_t _addr, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - int16_t data; - _read(_addr, data, sizeof(data), fastRead); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif - return data; +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to capacity +// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes three arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +int16_t SPIFlash::readShort(uint32_t address, bool fastRead) { + const uint8_t size = sizeof(int16_t); + union + { + byte b[size]; + int16_t s; + } data; + + if (!_prep(READDATA, address, size)) { + return false; + } + switch (fastRead) { + case false: + _beginSPI(READDATA); + break; + + case true: + _beginSPI(FASTREAD); + break; + + default: + break; + } + _nextBuf(READDATA, &data.b[0], size); + _endSPI(); + return data.s; +} +// Variant B +int16_t SPIFlash::readShort(uint16_t page_number, uint8_t offset, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); + + return readShort(address, fastRead); } // Reads an unsigned long of data from a specific location in a page. -// Takes two arguments - -// 1. _addr --> Any address from 0 to capacity -// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true -uint32_t SPIFlash::readULong(uint32_t _addr, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - uint32_t data; - _read(_addr, data, sizeof(data), fastRead); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif - return data; +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to capacity +// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes three arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +uint32_t SPIFlash::readULong(uint32_t address, bool fastRead) { + const uint8_t size = (sizeof(uint32_t)); + union + { + uint8_t b[size]; + uint32_t l; + } data; + + if (!_prep(READDATA, address, size)) { + return false; + } + switch (fastRead) { + case false: + _beginSPI(READDATA); + break; + + case true: + _beginSPI(FASTREAD); + break; + + default: + break; + } + _nextBuf(READDATA, &data.b[0], size); + _endSPI(); + return data.l; +} +// Variant B +uint32_t SPIFlash::readULong(uint16_t page_number, uint8_t offset, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); + + return readULong(address, fastRead); } // Reads a signed long of data from a specific location in a page. -// Takes two arguments - -// 1. _addr --> Any address from 0 to capacity -// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true -int32_t SPIFlash::readLong(uint32_t _addr, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - int32_t data; - _read(_addr, data, sizeof(data), fastRead); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif - return data; +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to capacity +// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes three arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +int32_t SPIFlash::readLong(uint32_t address, bool fastRead) { + const uint8_t size = (sizeof(int32_t)); + union + { + byte b[size]; + int32_t l; + } data; + + if (!_prep(READDATA, address, size)) { + return false; + } + switch (fastRead) { + case false: + _beginSPI(READDATA); + break; + + case true: + _beginSPI(FASTREAD); + break; + + default: + break; + } + _nextBuf(READDATA, &data.b[0], size); + _endSPI(); + return data.l; } +// Variant B +int32_t SPIFlash::readLong(uint16_t page_number, uint8_t offset, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); -// Reads a float of data from a specific location in a page. -// Takes two arguments - -// 1. _addr --> Any address from 0 to capacity -// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true -float SPIFlash::readFloat(uint32_t _addr, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - float data; - _read(_addr, data, sizeof(data), fastRead); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif - return data; + return readLong(address, fastRead); +} + +// Reads a signed long of data from a specific location in a page. +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to capacity +// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes three arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +float SPIFlash::readFloat(uint32_t address, bool fastRead) { + const uint8_t size = (sizeof(float)); + union + { + byte b[size]; + float f; + } data; + + if (!_prep(READDATA, address, size)) { + return false; + } + + switch (fastRead) { + case false: + _beginSPI(READDATA); + break; + + case true: + _beginSPI(FASTREAD); + break; + + default: + break; + } + _nextBuf(READDATA, &data.b[0], size); + _endSPI(); + return data.f; +} +// Variant B +float SPIFlash::readFloat(uint16_t page_number, uint8_t offset, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); + + return readFloat(address, fastRead); } // Reads a string from a specific location on a page. -// Takes three arguments -// 1. _addr --> Any address from 0 to capacity -// 2. outputString --> String variable to write the output to -// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true -bool SPIFlash::readStr(uint32_t _addr, String &data, bool fastRead) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - bool _retVal = _read(_addr, data, sizeof(data), fastRead); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return _read(_addr, data, sizeof(data), fastRead); - #endif +// Has two variants: +// A. Takes three arguments +// 1. address --> Any address from 0 to capacity +// 2. outputString --> String variable to write the output to +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes four arguments +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. outputString --> String variable to write the output to +// 4. fastRead --> defaults to false - executes _beginFastRead() if set to true +// This function first reads a short from the address to figure out the size of the String object stored and +// then reads the String object data +// Variant A +bool SPIFlash::readStr(uint32_t address, String &outStr, bool fastRead) { + uint16_t strLen; + //_delay_us(20); + strLen = readWord(address); + address+=(sizeof(strLen)); + char outputChar[strLen]; + + readCharArray(address, outputChar, strLen, fastRead); + + outStr = String(outputChar); + return true; +} +// Variant B +bool SPIFlash::readStr(uint16_t page_number, uint8_t offset, String &outStr, bool fastRead) { + uint32_t address = _getAddress(page_number, offset); + return readStr(address, outStr, fastRead); +} + +// Writes a byte of data to a specific location in a page. +// Has two variants: +// A. Takes three arguments - +// 1. address --> Any address - from 0 to capacity +// 2. data --> One byte of data to be written to a particular location on a page +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data --> One byte of data to be written to a particular location on a page +// 4. errorCheck --> Turned on by default. Checks for writing errors +// WARNING: You can only write to previously erased memory locations (see datasheet). +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// Variant A +bool SPIFlash::writeByte(uint32_t address, uint8_t data, bool errorCheck) { + if(!_prep(PAGEPROG, address, sizeof(data))) { + return false; + } + + _beginSPI(PAGEPROG); + _nextByte(data); + CHIP_DESELECT + + if (!errorCheck) { + _endSPI(); + return true; + } + else { + return _writeErrorCheck(address, data); + } } +// Variant B +bool SPIFlash::writeByte(uint16_t page_number, uint8_t offset, uint8_t data, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); + + return writeByte(address, data, errorCheck); +} + +// Writes a char of data to a specific location in a page. +// Has two variants: +// A. Takes three arguments - +// 1. address --> Any address - from 0 to capacity +// 2. data --> One char of data to be written to a particular location on a page +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data --> One char of data to be written to a particular location on a page +// 4. errorCheck --> Turned on by default. Checks for writing errors +// WARNING: You can only write to previously erased memory locations (see datasheet). +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// Variant A +bool SPIFlash::writeChar(uint32_t address, int8_t data, bool errorCheck) { + if(!_prep(PAGEPROG, address, sizeof(data))) { + return false; + } + + _beginSPI(PAGEPROG); + _nextByte(data); + CHIP_DESELECT + + if (!errorCheck) { + _endSPI(); + return true; + } + else { + return _writeErrorCheck(address, data); + } +} +// Variant B +bool SPIFlash::writeChar(uint16_t page_number, uint8_t offset, int8_t data, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); + + return writeChar(address, data, errorCheck); +} + +// Writes an array of bytes starting from a specific location in a page. +// Has two variants: +// A. Takes three arguments - +// 1. address --> Any address - from 0 to capacity +// 2. data --> An array of bytes to be written to a particular location on a page +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data --> An array of bytes to be written to a particular location on a page +// 4. errorCheck --> Turned on by default. Checks for writing errors +// WARNING: You can only write to previously erased memory locations (see datasheet). +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// Variant A +bool SPIFlash::writeByteArray(uint32_t address, uint8_t *data_buffer, uint16_t bufferSize, bool errorCheck) { + if (!_prep(PAGEPROG, address, bufferSize)) { + return false; + } + uint16_t maxBytes = PAGESIZE-(address % PAGESIZE); // Force the first set of bytes to stay within the first page + uint16_t length = bufferSize; + uint16_t writeBufSz; + uint16_t data_offset = 0; + + while (length > 0) + { + writeBufSz = (length<=maxBytes) ? length : maxBytes; + + if(!_notBusy() || !_writeEnable()){ + return false; + } + + _beginSPI(PAGEPROG); + + for (uint16_t i = 0; i < writeBufSz; ++i) { + _nextByte(data_buffer[data_offset + i]); + } + _currentAddress += writeBufSz; + data_offset += writeBufSz; + length -= writeBufSz; + maxBytes = 256; // Now we can do up to 256 bytes per loop + CHIP_DESELECT + } + + if (!errorCheck) { + _endSPI(); + return true; + } + else { + if (!_notBusy()) { + return false; + } + _currentAddress = address; + CHIP_SELECT + _nextByte(READDATA); + _transferAddress(); + for (uint16_t j = 0; j < bufferSize; j++) { + if (_nextByte(NULLBYTE) != data_buffer[j]) { + return false; + } + } + _endSPI(); + return true; + } +} +// Variant B +bool SPIFlash::writeByteArray(uint16_t page_number, uint8_t offset, uint8_t *data_buffer, uint16_t bufferSize, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); + + return writeByteArray(address, data_buffer, bufferSize, errorCheck); +} + +// Writes an array of bytes starting from a specific location in a page. +// Has two variants: +// A. Takes three arguments - +// 1. address --> Any address - from 0 to capacity +// 2. data --> An array of chars to be written to a particular location on a page +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data --> An array of chars to be written to a particular location on a page +// 4. errorCheck --> Turned on by default. Checks for writing errors +// WARNING: You can only write to previously erased memory locations (see datasheet). +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// Variant A +bool SPIFlash::writeCharArray(uint32_t address, char *data_buffer, uint16_t bufferSize, bool errorCheck) { + uint16_t writeBufSz; + uint16_t maxBytes = PAGESIZE-(address % PAGESIZE); // Force the first set of bytes to stay within the first page + uint16_t data_offset = 0; + uint16_t length = bufferSize; + if (!_prep(PAGEPROG, address, bufferSize)) { + return false; + } + + while (length > 0) + { + writeBufSz = (length<=maxBytes) ? length : maxBytes; + + if(!_notBusy() || !_writeEnable()){ + return false; + } + + _beginSPI(PAGEPROG); + + for (uint16_t i = 0; i < writeBufSz; ++i) { + _nextByte(data_buffer[data_offset + i]); + } + _currentAddress += writeBufSz; + data_offset += writeBufSz; + length -= writeBufSz; + maxBytes = 256; // Now we can do up to 256 bytes per loop + CHIP_DESELECT + } + + if (!errorCheck) { + _endSPI(); + return true; + } + else { + if (!_notBusy()) { + return false; + } + _currentAddress = address; + CHIP_SELECT + _nextByte(READDATA); + _transferAddress(); + for (uint16_t j = 0; j < bufferSize; j++) { + if (_nextByte(NULLBYTE) != data_buffer[j]) { + return false; + } + } + _endSPI(); + return true; + } +} +// Variant B +bool SPIFlash::writeCharArray(uint16_t page_number, uint8_t offset, char *data_buffer, uint16_t bufferSize, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); + + return writeCharArray(address, data_buffer, bufferSize, errorCheck); +} + +// Writes an unsigned int as two bytes starting from a specific location in a page. +// Has two variants: +// A. Takes three arguments - +// 1. address --> Any address - from 0 to capacity +// 2. data --> One unsigned int of data to be written to a particular location on a page +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data --> One unsigned int of data to be written to a particular location on a page +// 4. errorCheck --> Turned on by default. Checks for writing errors +// WARNING: You can only write to previously erased memory locations (see datasheet). +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// Variant A +bool SPIFlash::writeWord(uint32_t address, uint16_t data, bool errorCheck) { + const uint8_t size = sizeof(uint16_t); + + if(!_prep(PAGEPROG, address, size)) { + return false; + } + + union + { + uint8_t b[size]; + uint16_t w; + } var; + var.w = data; + + uint16_t maxBytes = PAGESIZE-(address % PAGESIZE); // Force the first set of bytes to stay within the first page + if (maxBytes > size) { + _beginSPI(PAGEPROG); + _nextBuf(PAGEPROG, &var.b[0], size); + CHIP_DESELECT + } + else { + uint16_t writeBufSz; + uint16_t data_offset = 0; + uint16_t _sz = size; + + while (_sz > 0) + { + writeBufSz = (_sz<=maxBytes) ? _sz : maxBytes; + if(!_notBusy() || !_writeEnable()){ + return false; + } + + _beginSPI(PAGEPROG); + for (uint16_t i = 0; i < writeBufSz; ++i) { + _nextByte(var.b[data_offset + i]); + } + _currentAddress += writeBufSz; + data_offset += writeBufSz; + _sz -= writeBufSz; + maxBytes = 256; // Now we can do up to 256 bytes per loop + CHIP_DESELECT + } + } -// Writes a byte of data to a specific location in a page. -// Takes three arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data --> One byte to be written to a particular location on a page -// 3. errorCheck --> Turned on by default. Checks for writing errors -// WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeByte(uint32_t _addr, uint8_t data, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - bool _retVal = _write(_addr, data, sizeof(data), errorCheck); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return _write(_addr, data, sizeof(data), errorCheck); - #endif + if (!errorCheck) { + _endSPI(); + return true; + } + else + return _writeErrorCheck(address, data); } +// Variant B +bool SPIFlash::writeWord(uint16_t page_number, uint8_t offset, uint16_t data, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); -// Writes a char of data to a specific location in a page. -// Takes three arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data --> One char to be written to a particular location on a page -// 3. errorCheck --> Turned on by default. Checks for writing errors -// WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeChar(uint32_t _addr, int8_t data, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - bool _retVal = _write(_addr, data, sizeof(data), errorCheck); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return _write(_addr, data, sizeof(data), errorCheck); - #endif + return writeWord(address, data, errorCheck); } -// Writes an array of bytes starting from a specific location in a page. -// Takes four arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data_buffer --> The pointer to the array of bytes be written to a particular location on a page -// 3. bufferSize --> Size of the array of bytes - in number of bytes -// 4. errorCheck --> Turned on by default. Checks for writing errors +// Writes a signed int as two bytes starting from a specific location in a page. +// Has two variants: +// A. Takes three arguments - +// 1. address --> Any address - from 0 to capacity +// 2. data --> One signed int of data to be written to a particular location on a page +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data --> One signed int of data to be written to a particular location on a page +// 4. errorCheck --> Turned on by default. Checks for writing errors // WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeByteArray(uint32_t _addr, uint8_t *data_buffer, size_t bufferSize, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - if (!_prep(PAGEPROG, _addr, bufferSize)) { +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// Variant A +bool SPIFlash::writeShort(uint32_t address, int16_t data, bool errorCheck) { + const uint8_t size = sizeof(data); + if(!_prep(PAGEPROG, address, size)) { return false; } - uint16_t maxBytes = SPI_PAGESIZE-(_addr % SPI_PAGESIZE); // Force the first set of bytes to stay within the first page - if (bufferSize <= maxBytes) { - CHIP_SELECT - _nextByte(WRITE, PAGEPROG); - _transferAddress(); - _nextBuf(PAGEPROG, &data_buffer[0], bufferSize); + union + { + uint8_t b[size]; + int16_t s; + } var; + var.s = data; + + uint16_t maxBytes = PAGESIZE-(address % PAGESIZE); // Force the first set of bytes to stay within the first page + if (maxBytes > size) { + _beginSPI(PAGEPROG); + _nextBuf(PAGEPROG, &var.b[0], size); CHIP_DESELECT } else { - uint16_t length = bufferSize; uint16_t writeBufSz; uint16_t data_offset = 0; + uint16_t _sz = size; - do { - writeBufSz = (length<=maxBytes) ? length : maxBytes; + while (_sz > 0) + { + writeBufSz = (_sz<=maxBytes) ? _sz : maxBytes; + if(!_notBusy() || !_writeEnable()){ + return false; + } - CHIP_SELECT - _nextByte(WRITE, PAGEPROG); - _transferAddress(); + _beginSPI(PAGEPROG); for (uint16_t i = 0; i < writeBufSz; ++i) { - _nextByte(WRITE, data_buffer[data_offset + i]); + _nextByte(var.b[data_offset + i]); } - CHIP_DESELECT - _currentAddress += writeBufSz; data_offset += writeBufSz; - length -= writeBufSz; + _sz -= writeBufSz; maxBytes = 256; // Now we can do up to 256 bytes per loop - - if(!_notBusy() || !_writeEnable()){ - return false; - } - - } while (length > 0); + CHIP_DESELECT + } } - if (!errorCheck) { + if (!errorCheck) { _endSPI(); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif return true; + } + else + return _writeErrorCheck(address, data); +} +// Variant B +bool SPIFlash::writeShort(uint16_t page_number, uint8_t offset, int16_t data, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); + + return writeShort(address, data, errorCheck); +} + +// Writes an unsigned long as four bytes starting from a specific location in a page. +// Has two variants: +// A. Takes three arguments - +// 1. address --> Any address - from 0 to capacity +// 2. data --> One unsigned long of data to be written to a particular location on a page +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data --> One unsigned long of data to be written to a particular location on a page +// 4. errorCheck --> Turned on by default. Checks for writing errors +// WARNING: You can only write to previously erased memory locations (see datasheet). +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// Variant A +bool SPIFlash::writeULong(uint32_t address, uint32_t data, bool errorCheck) { + const uint8_t size = (sizeof(data)); + + if(!_prep(PAGEPROG, address, size)) { + return false; + } + + union + { + uint8_t b[size]; + uint32_t l; + } var; + var.l = data; + + uint16_t maxBytes = PAGESIZE-(address % PAGESIZE); // Force the first set of bytes to stay within the first page + if (maxBytes > size) { + _beginSPI(PAGEPROG); + _nextBuf(PAGEPROG, &var.b[0], size); + CHIP_DESELECT } else { - if (!_notBusy()) { - return false; - } - _currentAddress = _addr; - CHIP_SELECT - _nextByte(WRITE, READDATA); - _transferAddress(); - for (uint16_t j = 0; j < bufferSize; j++) { - if (_nextByte(READ) != data_buffer[j]) { + uint16_t writeBufSz; + uint16_t data_offset = 0; + uint16_t _sz = size; + + while (_sz > 0) + { + writeBufSz = (_sz<=maxBytes) ? _sz : maxBytes; + if(!_notBusy() || !_writeEnable()){ return false; } + + _beginSPI(PAGEPROG); + for (uint16_t i = 0; i < writeBufSz; ++i) { + _nextByte(var.b[data_offset + i]); + } + _currentAddress += writeBufSz; + data_offset += writeBufSz; + _sz -= writeBufSz; + maxBytes = 256; // Now we can do up to 256 bytes per loop + CHIP_DESELECT } + } + + if (!errorCheck){ _endSPI(); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif - return true; + return true; + } + else { + return _writeErrorCheck(address, data); } } +// Variant B +bool SPIFlash::writeULong(uint16_t page_number, uint8_t offset, uint32_t data, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); -// Writes an array of bytes starting from a specific location in a page. -// Takes four arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data_buffer --> The pointer to the array of chars be written to a particular location on a page -// 3. bufferSize --> Size of the array of chars - in number of bytes -// 4. errorCheck --> Turned on by default. Checks for writing errors + return writeULong(address, data, errorCheck); +} + +// Writes a signed long as four bytes starting from a specific location in a page. +// Has two variants: +// A. Takes three arguments - +// 1. address --> Any address - from 0 to capacity +// 2. data --> One signed long of data to be written to a particular location on a page +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data --> One signed long of data to be written to a particular location on a page +// 4. errorCheck --> Turned on by default. Checks for writing errors // WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeCharArray(uint32_t _addr, char *data_buffer, size_t bufferSize, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - if (!_prep(PAGEPROG, _addr, bufferSize)) { +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// Variant A +bool SPIFlash::writeLong(uint32_t address, int32_t data, bool errorCheck) { +const uint8_t size = sizeof(data); + + if(!_prep(PAGEPROG, address, size)) { return false; } - uint16_t maxBytes = SPI_PAGESIZE-(_addr % SPI_PAGESIZE); // Force the first set of bytes to stay within the first page - if (bufferSize <= maxBytes) { - CHIP_SELECT - _nextByte(WRITE, PAGEPROG); - _transferAddress(); - _nextBuf(PAGEPROG, (uint8_t*) &data_buffer[0], bufferSize); + union + { + uint8_t b[size]; + int32_t l; + } var; + var.l = data; + + uint16_t maxBytes = PAGESIZE-(address % PAGESIZE); // Force the first set of bytes to stay within the first page + if (maxBytes > size) { + _beginSPI(PAGEPROG); + _nextBuf(PAGEPROG, &var.b[0], size); CHIP_DESELECT } else { - uint16_t length = bufferSize; uint16_t writeBufSz; uint16_t data_offset = 0; + uint16_t _sz = size; - do { - writeBufSz = (length<=maxBytes) ? length : maxBytes; + while (_sz > 0) + { + writeBufSz = (_sz<=maxBytes) ? _sz : maxBytes; + if(!_notBusy() || !_writeEnable()){ + return false; + } - CHIP_SELECT - _nextByte(WRITE, PAGEPROG); - _transferAddress(); + _beginSPI(PAGEPROG); for (uint16_t i = 0; i < writeBufSz; ++i) { - _nextByte(WRITE, data_buffer[data_offset + i]); + _nextByte(var.b[data_offset + i]); } - CHIP_DESELECT - _currentAddress += writeBufSz; data_offset += writeBufSz; - length -= writeBufSz; + _sz -= writeBufSz; maxBytes = 256; // Now we can do up to 256 bytes per loop + CHIP_DESELECT + } + } + + if (!errorCheck){ + _endSPI(); + return true; + } + else { + return _writeErrorCheck(address, data); + } +} +// Variant B +bool SPIFlash::writeLong(uint16_t page_number, uint8_t offset, int32_t data, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); + return writeLong(address, data, errorCheck); +} + +// Writes a float as four bytes starting from a specific location in a page. +// Has two variants: +// A. Takes three arguments - +// 1. address --> Any address - from 0 to capacity +// 2. data --> One float of data to be written to a particular location on a page +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. data --> One float of data to be written to a particular location on a page +// 4. errorCheck --> Turned on by default. Checks for writing errors +// WARNING: You can only write to previously erased memory locations (see datasheet). +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// Variant A +bool SPIFlash::writeFloat(uint32_t address, float data, bool errorCheck) { + const uint8_t size = (sizeof(data)); + + if(!_prep(PAGEPROG, address, size)) { + return false; + } + union + { + uint8_t b[size]; + float f; + } var; + var.f = data; + + uint16_t maxBytes = PAGESIZE-(address % PAGESIZE); // Force the first set of bytes to stay within the first page + if (maxBytes > size) { + _beginSPI(PAGEPROG); + _nextBuf(PAGEPROG, &var.b[0], size); + CHIP_DESELECT + } + else { + uint16_t writeBufSz; + uint16_t data_offset = 0; + uint16_t _sz = size; + + while (_sz > 0) + { + writeBufSz = (_sz<=maxBytes) ? _sz : maxBytes; if(!_notBusy() || !_writeEnable()){ return false; } - } while (length > 0); + _beginSPI(PAGEPROG); + for (uint16_t i = 0; i < writeBufSz; ++i) { + _nextByte(var.b[data_offset + i]); + } + _currentAddress += writeBufSz; + data_offset += writeBufSz; + _sz -= writeBufSz; + maxBytes = 256; // Now we can do up to 256 bytes per loop + CHIP_DESELECT + } } if (!errorCheck) { _endSPI(); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif return true; } else { - if (!_notBusy()) { - return false; - } - _currentAddress = _addr; - CHIP_SELECT - _nextByte(WRITE, READDATA); - _transferAddress(); - for (uint16_t j = 0; j < bufferSize; j++) { - if (_nextByte(READ) != data_buffer[j]) { - return false; - } - } - _endSPI(); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif - return true; + return _writeErrorCheck(address, data); } } +// Variant B +bool SPIFlash::writeFloat(uint16_t page_number, uint8_t offset, float data, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); -// Writes an unsigned int as two bytes starting from a specific location in a page. -// Takes three arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data --> One word to be written to a particular location on a page -// 3. errorCheck --> Turned on by default. Checks for writing errors -// WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeWord(uint32_t _addr, uint16_t data, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - bool _retVal = _write(_addr, data, sizeof(data), errorCheck); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return _write(_addr, data, sizeof(data), errorCheck); - #endif + return writeFloat(address, data, errorCheck); } -// Writes a signed int as two bytes starting from a specific location in a page -// Takes three arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data --> One short to be written to a particular location on a page -// 3. errorCheck --> Turned on by default. Checks for writing errors +// Reads a string from a specific location on a page. +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to capacity +// 2. inputString --> String variable to write the data from +// 3. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes four arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. inputString --> String variable to write the data from +// 4. errorCheck --> Turned on by default. Checks for writing errors // WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeShort(uint32_t _addr, int16_t data, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - bool _retVal = _write(_addr, data, sizeof(data), errorCheck); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return _write(_addr, data, sizeof(data), errorCheck); - #endif -} +// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) +// This function first writes the size of the string as an unsigned int to the address to figure out the size of the String object stored and +// then writes the String object data. Therefore it takes up two bytes more than the size of the String itself. +// Variant A +bool SPIFlash::writeStr(uint32_t address, String &inputStr, bool errorCheck) { + uint16_t inStrLen = inputStr.length() +1; + if(!_prep(PAGEPROG, address, inStrLen)) { + return false; + } -// Writes an unsigned long as four bytes starting from a specific location in a page. -// Takes three arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data --> One unsigned long to be written to a particular location on a page -// 3. errorCheck --> Turned on by default. Checks for writing errors -// WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeULong(uint32_t _addr, uint32_t data, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - bool _retVal = _write(_addr, data, sizeof(data), errorCheck); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return _write(_addr, data, sizeof(data), errorCheck); - #endif -} + const uint16_t size = sizeof(inStrLen); + union + { + uint8_t b[size]; + uint16_t w; + } var; + + var.w = inStrLen; + char inputChar[inStrLen]; + inputStr.toCharArray(inputChar, inStrLen); + uint16_t maxBytes = PAGESIZE-(address % PAGESIZE); // Force the first set of bytes to stay within the first page + if (maxBytes > inStrLen) { + _beginSPI(PAGEPROG); + _nextBuf(PAGEPROG, &var.b[0], size); + _nextBuf(PAGEPROG, (uint8_t*)&inputChar, inStrLen); + CHIP_DESELECT + } + else { + uint16_t writeBufSz; + uint16_t data_offset = 0; + bool strLenWritten = false; -// Writes a signed long as four bytes starting from a specific location in a page -// Takes three arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data --> One signed long to be written to a particular location on a page -// 3. errorCheck --> Turned on by default. Checks for writing errors -// WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeLong(uint32_t _addr, int32_t data, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - bool _retVal = _write(_addr, data, sizeof(data), errorCheck); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return _write(_addr, data, sizeof(data), errorCheck); - #endif -} + while (inStrLen > 0) + { + writeBufSz = (inStrLen<=maxBytes) ? inStrLen : maxBytes; + if(!_notBusy() || !_writeEnable()){ + return false; + } -// Writes a float as four bytes starting from a specific location in a page -// Takes three arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data --> One float to be written to a particular location on a page -// 3. errorCheck --> Turned on by default. Checks for writing errors -// WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeFloat(uint32_t _addr, float data, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - bool _retVal = _write(_addr, data, sizeof(data), errorCheck); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return _write(_addr, data, sizeof(data), errorCheck); - #endif -} + _beginSPI(PAGEPROG); + for (uint16_t i = 0; i < writeBufSz; ++i) { + if(!strLenWritten) { + for (uint8_t j = 0; j < size; j++) { + _nextByte(var.b[j]); + } + strLenWritten = true; + } + _nextByte(inputChar[data_offset + i]); + } + _currentAddress += writeBufSz; + data_offset += writeBufSz; + inStrLen -= writeBufSz; + maxBytes = 256; // Now we can do up to 256 bytes per loop + CHIP_DESELECT + } + } -// Writes a string to a specific location on a page -// Takes three arguments - -// 1. _addr --> Any address - from 0 to capacity -// 2. data --> One String to be written to a particular location on a page -// 3. errorCheck --> Turned on by default. Checks for writing errors -// WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -bool SPIFlash::writeStr(uint32_t _addr, String &data, bool errorCheck) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - bool _retVal = _write(_addr, data, sizeof(data), errorCheck); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return _write(_addr, data, sizeof(data), errorCheck); - #endif + if (!errorCheck) { + _endSPI(); + return true; + } + else { + String tempStr; + readStr(address, tempStr); + return inputStr.equals(tempStr); + } +} +// Variant B +bool SPIFlash::writeStr(uint16_t page_number, uint8_t offset, String &inputStr, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); + return writeStr(address, inputStr, errorCheck); } -// Erases one 4k sector. -// Takes an address as the argument and erases the block containing the address. -bool SPIFlash::eraseSector(uint32_t _addr) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - if (!_prep(ERASEFUNC, _addr)) { - return false; - } - _beginSPI(SECTORERASE); //The address is transferred as a part of this function + +//Erases one 4k sector. Has two variants: +// A. Takes the address as the argument and erases the sector containing the address. +// B. Takes page to be erased as the argument and erases the sector containing the page. +// The sectors are numbered 0 - 255 containing 16 pages each. +// Page 0-15 --> Sector 0; Page 16-31 --> Sector 1;......Page 4080-4095 --> Sector 255 +// Variant A +bool SPIFlash::eraseSector(uint32_t address) { + if(!_notBusy()||!_writeEnable()) + return false; + + _beginSPI(SECTORERASE); + _nextByte(address >> 16); + _nextByte(address >> 8); + _nextByte(0); _endSPI(); - if(!_notBusy(500L)) { - return false; //Datasheet says erasing a sector takes 400ms max - } - //_writeDisable(); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif + if(!_notBusy(500L)) + return false; //Datasheet says erasing a sector takes 400ms max + + //_writeDisable(); //_writeDisable() is not required because the Write Enable Latch (WEL) flag is cleared to 0 + // i.e. to write disable state upon the following conditions: + // Power-up, Write Disable, Page Program, Quad Page Program, ``Sector Erase``, Block Erase, Chip Erase, Write Status Register, + // Erase Security Register and Program Security register return true; } +// Variant B +bool SPIFlash::eraseSector(uint16_t page_number, uint8_t offset) { + uint32_t address = _getAddress(page_number, offset); + return eraseSector(address); +} -// Erases one 32k block. -// Takes an address as the argument and erases the block containing the address. -bool SPIFlash::eraseBlock32K(uint32_t _addr) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - if (!_prep(ERASEFUNC, _addr)) { - return false; +//Erases one 32k block. Has two variants: +// A. Takes the address as the argument and erases the block containing the address. +// B. Takes page to be erased as the argument and erases the block containing the page. +// The blocks are numbered 0 - 31 containing 128 pages each. +// Page 0-127 --> Block 0; Page 128-255 --> Block 1;......Page 3968-4095 --> Block 31 +// Variant A +bool SPIFlash::eraseBlock32K(uint32_t address) { + if(!_notBusy()||!_writeEnable()) { + return false; } _beginSPI(BLOCK32ERASE); + _nextByte(address >> 16); + _nextByte(address >> 8); + _nextByte(0); _endSPI(); - if(!_notBusy(1*S)) { - return false; //Datasheet says erasing a sector takes 400ms max - } - _writeDisable(); - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif + if(!_notBusy(1*S)) + return false; //Datasheet says erasing a sector takes 400ms max + + //_writeDisable(); //_writeDisable() is not required because the Write Enable Latch (WEL) flag is cleared to 0 + // i.e. to write disable state upon the following conditions: + // Power-up, Write Disable, Page Program, Quad Page Program, Sector Erase, ``Block Erase``, Chip Erase, Write Status Register, + // Erase Security Register and Program Security register return true; } +// Variant B +bool SPIFlash::eraseBlock32K(uint16_t page_number, uint8_t offset) { + uint32_t address = _getAddress(page_number, offset); + return eraseBlock32K(address); +} -// Erases one 64k block. -// Takes an address as the argument and erases the block containing the address. -bool SPIFlash::eraseBlock64K(uint32_t _addr) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - if (!_prep(ERASEFUNC, _addr)) { - return false; +//Erases one 64k block. Has two variants: +// A. Takes the address as the argument and erases the block containing the address. +// B. Takes page to be erased as the argument and erases the block containing the page. +// The blocks are numbered 0 - 15 containing 256 pages each. +// Page 0-255 --> Block 0; Page 256-511 --> Block 1;......Page 3840-4095 --> Block 15 +// Variant A +bool SPIFlash::eraseBlock64K(uint32_t address) { + if(!_notBusy()||!_writeEnable()) { + return false; } - _beginSPI(BLOCK64ERASE); + _nextByte(address >> 16); + _nextByte(address >> 8); + _nextByte(0); _endSPI(); - if(!_notBusy(1200L)) { - return false; //Datasheet says erasing a sector takes 400ms max - } - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif + if(!_notBusy(1200L)) + return false; //Datasheet says erasing a sector takes 400ms max + + //_writeDisable(); //_writeDisable() is not required because the Write Enable Latch (WEL) flag is cleared to 0 + // i.e. to write disable state upon the following conditions: + // Power-up, Write Disable, Page Program, Quad Page Program, Sector Erase, ``Block Erase``, Chip Erase, Write Status Register, + // Erase Security Register and Program Security register + return true; } +// Variant B +bool SPIFlash::eraseBlock64K(uint16_t page_number, uint8_t offset) { + uint32_t address = _getAddress(page_number, offset); + return eraseBlock64K(address); +} //Erases whole chip. Think twice before using. bool SPIFlash::eraseChip(void) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - if(!_notBusy() || !_writeEnable()) { - return false; - } + if(!_notBusy()||!_writeEnable()) + return false; _beginSPI(CHIPERASE); - _endSPI(); + _endSPI(); + if(!_notBusy(_chip.eraseTime)) + return false; //Datasheet says erasing chip takes 6s max - while(_readStat1() & BUSY) { - //_delay_us(30000L); - } - _endSPI(); + //_writeDisable(); //_writeDisable() is not required because the Write Enable Latch (WEL) flag is cleared to 0 + // i.e. to write disable state upon the following conditions: + // Power-up, Write Disable, Page Program, Quad Page Program, Sector Erase, Block Erase, ``Chip Erase``, Write Status Register, + // Erase Security Register and Program Security register - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif return true; } @@ -798,37 +1765,27 @@ bool SPIFlash::eraseChip(void) { //Erase suspend is only allowed during Block/Sector erase. //Program suspend is only allowed during Page/Quad Page Program bool SPIFlash::suspendProg(void) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif if(_notBusy()) { return false; } if(!_noSuspend()) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif return true; } - _beginSPI(SUSPEND); - _endSPI(); - _delay_us(20); - if(!_notBusy(50) || _noSuspend()) { - return false; + else { + _beginSPI(SUSPEND); + _endSPI(); + _delay_us(20); + if(!_notBusy(50) || _noSuspend()) { //Max suspend Enable time according to datasheet + return false; + } + return true; } - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif - return true; } //Resumes previously suspended Block Erase/Sector Erase/Page Program. bool SPIFlash::resumeProg(void) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif if(!_notBusy() || _noSuspend()) { return false; } @@ -841,66 +1798,48 @@ bool SPIFlash::resumeProg(void) { if(_notBusy(10) || !_noSuspend()) { return false; } - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros() - _spifuncruntime; - #endif - return true; + return true; } //Puts device in low power state. Good for battery powered operations. +//Typical current consumption during power-down is 1mA with a maximum of 5mA. (Datasheet 7.4) //In powerDown() the chip will only respond to powerUp() bool SPIFlash::powerDown(void) { - if (_chip.manufacturerID != MICROCHIP_MANID) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif - if(!_notBusy(20)) - return false; + if(!_notBusy(20)) + return false; - _beginSPI(POWERDOWN); - _endSPI(); + _beginSPI(POWERDOWN); + _endSPI(); - _delay_us(5); + _delay_us(5); - #ifdef RUNDIAGNOSTIC - bool _retVal = !_writeEnable(false); - _spifuncruntime = micros() - _spifuncruntime; - return _retVal; - #else - return !_writeEnable(false); - #endif + _beginSPI(WRITEENABLE); + CHIP_DESELECT + if (_readStat1() & WRTEN) { + _endSPI(); + return false; } else { - _troubleshoot(UNSUPPORTEDFUNCTION); - return false; + + return true; } } //Wakes chip from low power state. bool SPIFlash::powerUp(void) { - #ifdef RUNDIAGNOSTIC - _spifuncruntime = micros(); - #endif _beginSPI(RELEASE); _endSPI(); _delay_us(3); //Max release enable time according to the Datasheet - #ifdef RUNDIAGNOSTIC - if (_writeEnable(false)) { - _writeDisable(); - _spifuncruntime = micros() - _spifuncruntime; - return true; - } - return false; - #else - if (_writeEnable(false)) { - _writeDisable(); + _beginSPI(WRITEENABLE); + CHIP_DESELECT + if (_readStat1() & WRTEN) { + _endSPI(); return true; } - return false; - #endif -} + else { -/* Note: _writeDisable() is not required at the end of any function that writes to the Flash memory because the Write Enable Latch (WEL) flag is cleared to 0 i.e. to write disable state upon the following conditions being completed: -Power-up, Write Disable, Page Program, Quad Page Program, Sector Erase, Block Erase, Chip Erase, Write Status Register, Erase Security Register and Program Security register */ + return false; + } +} diff --git a/src/SPIFlash.h b/src/SPIFlash.h index 49705e9..e40aa96 100644 --- a/src/SPIFlash.h +++ b/src/SPIFlash.h @@ -1,8 +1,8 @@ -/* Arduino SPIFlash Library v.3.0.0 +/* Arduino SPIFlash Library v.2.7.0 * Copyright (C) 2017 by Prajwal Bhattaram * Created by Prajwal Bhattaram - 19/05/2015 * Modified by @boseji - 02/03/2017 - * Modified by Prajwal Bhattaram - 04/11/2017 + * Modified by Prajwal Bhattaram - 02/08/2017 * * This file is part of the Arduino SPIFlash Library. This library is for * Winbond NOR flash memory modules. In its current form it enables reading @@ -27,14 +27,17 @@ #ifndef SPIFLASH_H #define SPIFLASH_H +//ATTiny85 does not have enough pins to support Serial. So, the basic troubleshooting functions of this library are not applicable. It is up to the end user to come up with a diagnostic routine for the ATTiny85. +#ifndef __AVR_ATtiny85__ //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// // Uncomment the code below to run a diagnostic if your flash // // does not respond // // // // Error codes will be generated and returned on functions // //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -#define RUNDIAGNOSTIC // +//#define RUNDIAGNOSTIC // //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// +#endif //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// // Uncomment the code below to increase the speed of the library // @@ -44,19 +47,8 @@ //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// //#define HIGHSPEED // //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// - -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -// Comment out the code below to disable DMA mode on SAMD based // -// platforms // -// // -// Change the ZERO_SPISERCOM define below to use other SPI ports // -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -//#define ENABLEZERODMA // -//#define ZERO_SPISERCOM SERCOM4 // -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// - #include - #include "defines.h" - #include +#include +#include "defines.h" #if defined (ARDUINO_ARCH_SAM) #include @@ -64,153 +56,173 @@ #include #endif -#if defined (BOARD_RTL8195A) - #ifdef __cplusplus - extern "C" { - #endif - - #include "gpio_api.h" - #include "PinNames.h" - - #ifdef __cplusplus - } - #endif +#ifndef __AVR_ATtiny85__ + #include #endif -#ifndef ARCH_STM32 - #if defined(ARDUINO_ARCH_STM32) || defined(__STM32F1__) || defined(STM32F1) || defined(STM32F3) || defined(STM32F4) || defined(STM32F0xx) - #define ARCH_STM32 - #endif -#endif -#if defined (ARDUINO_ARCH_SAM) || defined (ARDUINO_ARCH_SAMD) || defined (ARDUINO_ARCH_ESP8266) || defined (SIMBLEE) || defined (ARDUINO_ARCH_ESP32) || defined (BOARD_RTL8195A) || defined(ARCH_STM32) +#if defined (ARDUINO_ARCH_SAM) || defined (ARDUINO_ARCH_SAMD) || defined (ARDUINO_ARCH_ESP8266) || defined (SIMBLEE) || defined (ARDUINO_ARCH_ESP32) || defined (BOARD_RTL8195A) // RTL8195A included - @boseji 02.03.17 - #define _delay_us(us) delayMicroseconds(us) + #define _delay_us(us) delayMicroseconds(us) #else - #include + #include +#endif +// Includes specific to RTL8195A to access GPIO HAL - @boseji 02.03.17 +#if defined (BOARD_RTL8195A) +#ifdef __cplusplus +extern "C" { #endif -// Defines and variables specific to SAM architecture -#if defined (ARDUINO_ARCH_SAM) - #define CHIP_SELECT digitalWrite(csPin, LOW); - #define CHIP_DESELECT digitalWrite(csPin, HIGH); - #define xfer _dueSPITransfer - #define BEGIN_SPI _dueSPIBegin(); - extern char _end; - extern "C" char *sbrk(int i); - //char *ramstart=(char *)0x20070000; - //char *ramend=(char *)0x20088000; +#include "gpio_api.h" +#include "PinNames.h" -// Specific access configuration for Chip select pin. Includes specific to RTL8195A to access GPIO HAL - @boseji 02.03.17 -#elif defined (BOARD_RTL8195A) - #ifdef __cplusplus - extern "C" { - #endif +#ifdef __cplusplus +} +#endif - #include "gpio_api.h" - #include "PinNames.h" +#endif - #ifdef __cplusplus - } +#ifdef ARDUINO_ARCH_AVR + #ifdef __AVR_ATtiny85__ + #define CHIP_SELECT PORTB &= ~cs_mask; + #define CHIP_DESELECT PORTB |= cs_mask; + #define SPIBIT \ + USICR = ((1<transfer(n) - #define BEGIN_SPI _spi->begin(); -#else +#elif defined (ARDUINO_ARCH_SAM) + #define CHIP_SELECT digitalWrite(csPin, LOW); + #define CHIP_DESELECT digitalWrite(csPin, HIGH); + #define xfer _dueSPITransfer + #define BEGIN_SPI _dueSPIBegin(); +// Specific access configuration for Chip select pin - @boseji 02.03.17 +#elif defined (BOARD_RTL8195A) + #define CHIP_SELECT gpio_write(&csPin, 0); + #define CHIP_DESELECT gpio_write(&csPin, 1); + #define xfer(n) SPI.transfer(n) + #define BEGIN_SPI SPI.begin(); +#else //#elif defined (ARDUINO_ARCH_ESP8266) || defined (ARDUINO_ARCH_SAMD) #define CHIP_SELECT digitalWrite(csPin, LOW); #define CHIP_DESELECT digitalWrite(csPin, HIGH); #define xfer(n) SPI.transfer(n) #define BEGIN_SPI SPI.begin(); #endif -#define LIBVER 3 -#define LIBSUBVER 0 +#define LIBVER 2 +#define LIBSUBVER 7 #define BUGFIXVER 0 +#if defined (ARDUINO_ARCH_SAM) + extern char _end; + extern "C" char *sbrk(int i); + //char *ramstart=(char *)0x20070000; + //char *ramend=(char *)0x20088000; +#endif + class SPIFlash { public: - //------------------------------------ Constructor ------------------------------------// + //----------------------------------------------Constructor----------------------------------------------- //New Constructor to Accept the PinNames as a Chip select Parameter - @boseji 02.03.17 - #if defined (ARDUINO_ARCH_SAMD) || defined(ARCH_STM32) - SPIFlash(uint8_t cs = CS, SPIClass *spiinterface=&SPI); - #elif defined (BOARD_RTL8195A) - SPIFlash(PinName cs = CS); + #if !defined (BOARD_RTL8195A) + SPIFlash(uint8_t cs = CS, bool overflow = true); #else - SPIFlash(uint8_t cs = CS); + SPIFlash(PinName cs = CS, bool overflow = true); #endif - //----------------------------- Initial / Chip Functions ------------------------------// - bool begin(size_t flashChipSize = 0); + //----------------------------------------Initial / Chip Functions----------------------------------------// + bool begin(uint32_t _chipSize = 0); void setClock(uint32_t clockSpeed); bool libver(uint8_t *b1, uint8_t *b2, uint8_t *b3); uint8_t error(bool verbosity = false); uint16_t getManID(void); uint32_t getJEDECID(void); - uint64_t getUniqueID(void); + bool getAddress(uint16_t size, uint16_t &page_number, uint8_t &offset); uint32_t getAddress(uint16_t size); uint16_t sizeofStr(String &inputStr); uint32_t getCapacity(void); uint32_t getMaxPage(void); - float functionRunTime(void); - //-------------------------------- Write / Read Bytes ---------------------------------// - bool writeByte(uint32_t _addr, uint8_t data, bool errorCheck = true); - uint8_t readByte(uint32_t _addr, bool fastRead = false); - //----------------------------- Write / Read Byte Arrays ------------------------------// - bool writeByteArray(uint32_t _addr, uint8_t *data_buffer, size_t bufferSize, bool errorCheck = true); - bool readByteArray(uint32_t _addr, uint8_t *data_buffer, size_t bufferSize, bool fastRead = false); - //-------------------------------- Write / Read Chars ---------------------------------// - bool writeChar(uint32_t _addr, int8_t data, bool errorCheck = true); - int8_t readChar(uint32_t _addr, bool fastRead = false); - //------------------------------ Write / Read Char Arrays -----------------------------// - bool writeCharArray(uint32_t _addr, char *data_buffer, size_t bufferSize, bool errorCheck = true); - bool readCharArray(uint32_t _addr, char *data_buffer, size_t buffer_size, bool fastRead = false); - //-------------------------------- Write / Read Shorts --------------------------------// - bool writeShort(uint32_t _addr, int16_t data, bool errorCheck = true); - int16_t readShort(uint32_t _addr, bool fastRead = false); - //-------------------------------- Write / Read Words ---------------------------------// - bool writeWord(uint32_t _addr, uint16_t data, bool errorCheck = true); - uint16_t readWord(uint32_t _addr, bool fastRead = false); - //-------------------------------- Write / Read Longs ---------------------------------// - bool writeLong(uint32_t _addr, int32_t data, bool errorCheck = true); - int32_t readLong(uint32_t _addr, bool fastRead = false); - //--------------------------- Write / Read Unsigned Longs -----------------------------// - bool writeULong(uint32_t _addr, uint32_t data, bool errorCheck = true); - uint32_t readULong(uint32_t _addr, bool fastRead = false); - //-------------------------------- Write / Read Floats --------------------------------// - bool writeFloat(uint32_t _addr, float data, bool errorCheck = true); - float readFloat(uint32_t _addr, bool fastRead = false); - //-------------------------------- Write / Read Strings -------------------------------// - bool writeStr(uint32_t _addr, String &data, bool errorCheck = true); - bool readStr(uint32_t _addr, String &data, bool fastRead = false); - //------------------------------- Write / Read Anything -------------------------------// - - template bool writeAnything(uint32_t _addr, const T& data, bool errorCheck = true); - template bool readAnything(uint32_t _addr, T& data, bool fastRead = false); - //-------------------------------- Erase functions ------------------------------------// - bool eraseSector(uint32_t _addr); - bool eraseBlock32K(uint32_t _addr); - bool eraseBlock64K(uint32_t _addr); + //-------------------------------------------Write / Read Bytes-------------------------------------------// + bool writeByte(uint32_t address, uint8_t data, bool errorCheck = true); + bool writeByte(uint16_t page_number, uint8_t offset, uint8_t data, bool errorCheck = true); + uint8_t readByte(uint16_t page_number, uint8_t offset, bool fastRead = false); + uint8_t readByte(uint32_t address, bool fastRead = false); + //----------------------------------------Write / Read Byte Arrays----------------------------------------// + bool writeByteArray(uint32_t address, uint8_t *data_buffer, uint16_t bufferSize, bool errorCheck = true); + bool writeByteArray(uint16_t page_number, uint8_t offset, uint8_t *data_buffer, uint16_t bufferSize, bool errorCheck = true); + bool readByteArray(uint32_t address, uint8_t *data_buffer, uint16_t bufferSize, bool fastRead = false); + bool readByteArray(uint16_t page_number, uint8_t offset, uint8_t *data_buffer, uint16_t bufferSize, bool fastRead = false); + //-------------------------------------------Write / Read Chars-------------------------------------------// + bool writeChar(uint32_t address, int8_t data, bool errorCheck = true); + bool writeChar(uint16_t page_number, uint8_t offset, int8_t data, bool errorCheck = true); + int8_t readChar(uint32_t address, bool fastRead = false); + int8_t readChar(uint16_t page_number, uint8_t offset, bool fastRead = false); + //----------------------------------------Write / Read Char Arrays----------------------------------------// + bool writeCharArray(uint32_t address, char *data_buffer, uint16_t bufferSize, bool errorCheck = true); + bool writeCharArray(uint16_t page_number, uint8_t offset, char *data_buffer, uint16_t bufferSize, bool errorCheck = true); + bool readCharArray(uint32_t address, char *data_buffer, uint16_t buffer_size, bool fastRead = false); + bool readCharArray(uint16_t page_number, uint8_t offset, char *data_buffer, uint16_t buffer_size, bool fastRead = false); + //------------------------------------------Write / Read Shorts------------------------------------------// + bool writeShort(uint32_t address, int16_t data, bool errorCheck = true); + bool writeShort(uint16_t page_number, uint8_t offset, int16_t data, bool errorCheck = true); + int16_t readShort(uint32_t address, bool fastRead = false); + int16_t readShort(uint16_t page_number, uint8_t offset, bool fastRead = false); + //-------------------------------------------Write / Read Words-------------------------------------------// + bool writeWord(uint32_t address, uint16_t data, bool errorCheck = true); + bool writeWord(uint16_t page_number, uint8_t offset, uint16_t data, bool errorCheck = true); + uint16_t readWord(uint32_t address, bool fastRead = false); + uint16_t readWord(uint16_t page_number, uint8_t offset, bool fastRead = false); + //-------------------------------------------Write / Read Longs-------------------------------------------// + bool writeLong(uint32_t address, int32_t data, bool errorCheck = true); + bool writeLong(uint16_t page_number, uint8_t offset, int32_t data, bool errorCheck = true); + int32_t readLong(uint32_t address, bool fastRead = false); + int32_t readLong(uint16_t page_number, uint8_t offset, bool fastRead = false); + //--------------------------------------Write / Read Unsigned Longs---------------------------------------// + bool writeULong(uint32_t address, uint32_t data, bool errorCheck = true); + bool writeULong(uint16_t page_number, uint8_t offset, uint32_t data, bool errorCheck = true); + uint32_t readULong(uint32_t address, bool fastRead = false); + uint32_t readULong(uint16_t page_number, uint8_t offset, bool fastRead = false); + //-------------------------------------------Write / Read Floats------------------------------------------// + bool writeFloat(uint32_t address, float data, bool errorCheck = true); + bool writeFloat(uint16_t page_number, uint8_t offset, float data, bool errorCheck = true); + float readFloat(uint32_t address, bool fastRead = false); + float readFloat(uint16_t page_number, uint8_t offset, bool fastRead = false); + //------------------------------------------Write / Read Strings------------------------------------------// + bool writeStr(uint32_t address, String &inputStr, bool errorCheck = true); + bool writeStr(uint16_t page_number, uint8_t offset, String &inputStr, bool errorCheck = true); + bool readStr(uint32_t address, String &outStr, bool fastRead = false); + bool readStr(uint16_t page_number, uint8_t offset, String &outStr, bool fastRead = false); + //------------------------------------------Write / Read Anything-----------------------------------------// + template bool writeAnything(uint32_t address, const T& value, bool errorCheck = true); + template bool writeAnything(uint16_t page_number, uint8_t offset, const T& value, bool errorCheck = true); + template bool readAnything(uint32_t address, T& value, bool fastRead = false); + template bool readAnything(uint16_t page_number, uint8_t offset, T& value, bool fastRead = false); + //--------------------------------------------Erase functions---------------------------------------------// + bool eraseSector(uint32_t address); + bool eraseSector(uint16_t page_number, uint8_t offset); + bool eraseBlock32K(uint32_t address); + bool eraseBlock32K(uint16_t page_number, uint8_t offset); + bool eraseBlock64K(uint32_t address); + bool eraseBlock64K(uint16_t page_number, uint8_t offset); bool eraseChip(void); - //-------------------------------- Power functions ------------------------------------// + //---------------------------------------------Power functions--------------------------------------------// bool suspendProg(void); bool resumeProg(void); bool powerDown(void); bool powerUp(void); - //-------------------------- Public Arduino Due Functions -----------------------------// + //-------------------------------------Public Arduino Due Functions-------------------------------------// //#if defined (ARDUINO_ARCH_SAM) //uint32_t freeRAM(void); //#endif - //------------------------------- Public variables ------------------------------------// + //-------------------------------------------Public variables-------------------------------------// private: #if defined (ARDUINO_ARCH_SAM) - //-------------------------- Private Arduino Due Functions ----------------------------// + //-------------------------------------Private Arduino Due Functions--------------------------------------// void _dmac_disable(void); void _dmac_enable(void); void _dmac_channel_disable(uint32_t ul_num); @@ -232,41 +244,42 @@ class SPIFlash { void _dueSPISendChar(char b); void _dueSPISendChar(const char* buf, size_t len); #endif - //------------------------------- Private functions -----------------------------------// - void _troubleshoot(uint8_t _code, bool printoverride = false); +//-------------------------------------Private ATTiny85 Functions-------------------------------------// +#if defined (__AVR_ATtiny85__) + static uint8_t _tinySPItransfer(uint8_t _data); + void _tinySPIbegin(); +#endif + //----------------------------------------Private functions----------------------------------------// + void _troubleshoot(void); void _printErrorCode(void); void _printSupportLink(void); void _endSPI(void); - bool _disableGlobalBlockProtect(void); - bool _prep(uint8_t opcode, uint32_t _addr, uint32_t size = 0); + bool _prep(uint8_t opcode, uint32_t address, uint32_t size); + bool _prep(uint8_t opcode, uint32_t page_number, uint8_t offset, uint32_t size); bool _startSPIBus(void); bool _beginSPI(uint8_t opcode); bool _noSuspend(void); bool _notBusy(uint32_t timeout = BUSY_TIMEOUT); - bool _notPrevWritten(uint32_t _addr, uint32_t size = 1); - bool _writeEnable(bool _troubleshootEnable = true); + bool _notPrevWritten(uint32_t address, uint32_t size = 1); + bool _writeEnable(uint32_t timeout = 10L); bool _writeDisable(void); bool _getJedecId(void); bool _getManId(uint8_t *b1, uint8_t *b2); bool _getSFDP(void); bool _chipID(void); bool _transferAddress(void); - bool _addressCheck(uint32_t _addr, uint32_t size = 1); - uint8_t _nextByte(char IOType, uint8_t data = NULLBYTE); + bool _addressCheck(uint32_t address, uint32_t size = 1); + uint8_t _nextByte(uint8_t data = NULLBYTE); uint16_t _nextInt(uint16_t = NULLINT); void _nextBuf(uint8_t opcode, uint8_t *data_buffer, uint32_t size); uint8_t _readStat1(void); uint8_t _readStat2(void); - template bool _write(uint32_t _addr, const T& value, uint32_t _sz, bool errorCheck); - template bool _read(uint32_t _addr, T& value, uint32_t _sz, bool fastRead = false); - template bool _writeErrorCheck(uint32_t _addr, const T& value); - template bool _writeErrorCheck(uint32_t _addr, const T& value, uint32_t _sz); - //-------------------------------- Private variables ----------------------------------// + uint32_t _getAddress(uint16_t page_number, uint8_t offset = 0); + template bool _writeErrorCheck(uint32_t address, const T& value); + //-------------------------------------------Private variables------------------------------------------// #ifdef SPI_HAS_TRANSACTION SPISettings _settings; #endif - //If multiple SPI ports are available this variable is used to choose between them (SPI, SPI1, SPI2 etc.) - SPIClass *_spi; #if !defined (BOARD_RTL8195A) uint8_t csPin; #else @@ -275,175 +288,154 @@ class SPIFlash { #endif volatile uint8_t *cs_port; bool pageOverflow, SPIBusState; - uint8_t cs_mask, errorcode, stat1, _SPCR, _SPSR, _a0, _a1, _a2; - char READ = 'R'; - char WRITE = 'W'; - float _spifuncruntime = 0; + uint8_t cs_mask, errorcode, state, _SPCR, _SPSR; struct chipID { - bool supported; uint8_t manufacturerID; uint8_t memoryTypeID; uint8_t capacityID; + uint16_t supported; uint32_t sfdp; uint32_t capacity; uint32_t eraseTime; }; chipID _chip; uint32_t currentAddress, _currentAddress = 0; - uint8_t _uniqueID[8]; - const uint8_t _capID[14] = - {0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x43, 0x4B, 0x00, 0x01}; - - const uint32_t _memSize[14] = - {64L * KiB, 128L * KiB, 256L * KiB, 512L * KiB, 1L * MiB, 2L * MiB, 4L * MiB, 8L * MiB, 6L * MiB, 32L * MiB, 8L * MiB, 8L * MiB, 256L * KiB, 512L * KiB}; + const uint8_t _capID[11] = {0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x43}; + const uint32_t _memSize[11] = {64L * K, 128L * K, 256L * K, 512L * K, 1L * M, 2L * M, 4L * M, 8L * M, + 16L * M, 32L * M, 8L * M}; + const uint32_t _eraseTime[11] = {1L * S, 2L * S, 2L * S, 4L * S, 6L * S, 10L * S, 15L * S, 100L * S, 200L * S, 400L * S, 50L}; //Erase time in milliseconds }; -//--------------------------------- Public Templates ------------------------------------// +//--------------------------------------------Templates-------------------------------------------// // Writes any type of data to a specific location in the flash memory. -// Takes three arguments - -// 1. _addr --> Any address from 0 to maxAddress -// 2. T& value --> Variable to write -// 3. errorCheck --> Turned on by default. Checks for writing errors +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to maxAddress +// 2. T& value --> Variable to write data from +// 4. errorCheck --> Turned on by default. Checks for writing errors +// B. Takes three arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. const T& value --> Variable with the data to be written +// 4. errorCheck --> Turned on by default. Checks for writing errors // WARNING: You can only write to previously erased memory locations (see datasheet). // Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) -template bool SPIFlash::writeAnything(uint32_t _addr, const T& data, bool errorCheck) { - return _write(_addr, data, sizeof(data), errorCheck); -} - -// Reads any type of data from a specific location in the flash memory. -// Takes three arguments - -// 1. _addr --> Any address from 0 to maxAddress -// 2. T& value --> Variable to return data into -// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true -template bool SPIFlash::readAnything(uint32_t _addr, T& data, bool fastRead) { - return _read(_addr, data, sizeof(data), fastRead); -} - -//---------------------------------- Private Templates ----------------------------------// - -// Private template to check for errors in writing to flash memory -// Takes three arguments - -// 1. _addr --> Any address from 0 to maxAddress -// 2. const T& value --> Variable with the data to be error checked -// 3. _sz --> Size of the data variable to be error checked, in bytes (1 byte = 8 bits) -template bool SPIFlash::_writeErrorCheck(uint32_t _addr, const T& value, uint32_t _sz) { - if (!_notBusy()) { +// Variant A +template bool SPIFlash::writeAnything(uint32_t address, const T& value, bool errorCheck) { + if (!_prep(PAGEPROG, address, sizeof(value))) { return false; } - //Serial.print(F("Address being error checked: ")); - //Serial.println(_addr); - _currentAddress = _addr; - const uint8_t* p = (const uint8_t*)(const void*)&value; - CHIP_SELECT - _nextByte(WRITE, READDATA); - _transferAddress(); - for (uint16_t i = 0; i < _sz; i++) { - if (*p++ != _nextByte(READ)) { - _troubleshoot(ERRORCHKFAIL); - _endSPI(); - return false; + uint16_t maxBytes = PAGESIZE-(address % PAGESIZE); // Force the first set of bytes to stay within the first page + uint16_t length = sizeof(value); + + //if (maxBytes > length) { + uint32_t writeBufSz; + uint16_t data_offset = 0; + const uint8_t* p = ((const uint8_t*)(const void*)&value); + + if (!SPIBusState) { + _startSPIBus(); } - } - _endSPI(); - return true; -} + while (length > 0) + { + writeBufSz = (length<=maxBytes) ? length : maxBytes; -// Writes any type of data to a specific location in the flash memory. -// Takes four arguments - -// 1. _addr --> Any address from 0 to maxAddress -// 2. T& value --> Variable to write -// 3. _sz --> Size of variable in bytes (1 byte = 8 bits) -// 4. errorCheck --> Turned on by default. Checks for writing errors -// WARNING: You can only write to previously erased memory locations (see datasheet). -// Use the eraseSector()/eraseBlock32K/eraseBlock64K commands to first clear memory (write 0xFFs) + if(!_notBusy() || !_writeEnable()){ + return false; + } -template bool SPIFlash::_write(uint32_t _addr, const T& value, uint32_t _sz, bool errorCheck) { - if (!_prep(PAGEPROG, _addr, _sz)) { - return false; - } - const uint8_t* p = ((const uint8_t*)(const void*)&value); -//Serial.print(F("Address being written to: ")); -//Serial.println(_addr); - if (!SPIBusState) { - _startSPIBus(); - } - CHIP_SELECT - _nextByte(WRITE, PAGEPROG); - _transferAddress(); - //If data is only one byte (8 bits) long - if (_sz == 0x01) { - _nextByte(WRITE, *p); - CHIP_DESELECT - } - else { //If data is longer than one byte (8 bits) - uint32_t length = _sz; - uint16_t maxBytes = SPI_PAGESIZE-(_addr % SPI_PAGESIZE); // Force the first set of bytes to stay within the first page + CHIP_SELECT + (void)xfer(PAGEPROG); + _transferAddress(); - if (maxBytes > length) { - for (uint16_t i = 0; i < length; ++i) { - _nextByte(WRITE, *p++); + for (uint16_t i = 0; i < writeBufSz; ++i) { + _nextByte(*p++); } + _currentAddress += writeBufSz; + data_offset += writeBufSz; + length -= writeBufSz; + maxBytes = 256; // Now we can do up to 256 bytes per loop CHIP_DESELECT } - else { - uint32_t writeBufSz; - uint16_t data_offset = 0; - - do { - writeBufSz = (length<=maxBytes) ? length : maxBytes; - for (uint16_t i = 0; i < writeBufSz; ++i) { - _nextByte(WRITE, *p++); - } - CHIP_DESELECT - _currentAddress += writeBufSz; - data_offset += writeBufSz; - length -= writeBufSz; - maxBytes = 256; // Now we can do up to 256 bytes per loop - if(!_notBusy() || !_writeEnable()) { - return false; - } - } while (length > 0); - } - } if (!errorCheck) { _endSPI(); return true; } else { - //Serial.print(F("Address sent to error check: ")); - //Serial.println(_addr); - return _writeErrorCheck(_addr, value, _sz); + return _writeErrorCheck(address, value); } } +// Variant B +template bool SPIFlash::writeAnything(uint16_t page_number, uint8_t offset, const T& value, bool errorCheck) { + uint32_t address = _getAddress(page_number, offset); + return writeAnything(address, value, errorCheck); +} // Reads any type of data from a specific location in the flash memory. -// Takes four arguments - -// 1. _addr --> Any address from 0 to maxAddress -// 2. T& value --> Variable to return data into -// 3. _sz --> Size of the variable in bytes (1 byte = 8 bits) -// 4. fastRead --> defaults to false - executes _beginFastRead() if set to true -template bool SPIFlash::_read(uint32_t _addr, T& value, uint32_t _sz, bool fastRead) { - if (_prep(READDATA, _addr, _sz)) { +// Has two variants: +// A. Takes two arguments - +// 1. address --> Any address from 0 to maxAddress +// 2. T& value --> Variable to return data into +// 2. fastRead --> defaults to false - executes _beginFastRead() if set to true +// B. Takes three arguments - +// 1. page --> Any page number from 0 to maxPage +// 2. offset --> Any offset within the page - from 0 to 255 +// 3. T& value --> Variable to return data into +// 3. fastRead --> defaults to false - executes _beginFastRead() if set to true +// Variant A +template bool SPIFlash::readAnything(uint32_t address, T& value, bool fastRead) { + if (!_prep(READDATA, address, sizeof(value))) + return false; + uint8_t* p = (uint8_t*)(void*)&value; - CHIP_SELECT - if (fastRead) { - _nextByte(WRITE, FASTREAD); - } - else { - _nextByte(WRITE, READDATA); + if(!fastRead) + _beginSPI(READDATA); + else + _beginSPI(FASTREAD); + for (uint16_t i = 0; i < sizeof(value); i++) { + *p++ =_nextByte(); + } + _endSPI(); + return true; +} +// Variant B +template bool SPIFlash::readAnything(uint16_t page_number, uint8_t offset, T& value, bool fastRead) +{ + uint32_t address = _getAddress(page_number, offset); + return readAnything(address, value, fastRead); +} + +// Private template to check for errors in writing to flash memory +template bool SPIFlash::_writeErrorCheck(uint32_t address, const T& value) { +if (!_prep(READDATA, address, sizeof(value)) && !_notBusy()) { + return false; +} + + const uint8_t* p = (const uint8_t*)(const void*)&value; + _beginSPI(READDATA); + uint8_t _v; + for(uint16_t i = 0; i < sizeof(value);i++) + { +#if defined (ARDUINO_ARCH_SAM) + if(*p++ != _dueSPIRecByte()) + { + return false; } - _transferAddress(); - for (uint16_t i = 0; i < _sz; i++) { - *p++ =_nextByte(READ); +#else + if(*p++ != _nextByte()) + { + errorcode = ERRORCHKFAIL; + #ifdef RUNDIAGNOSTIC + _troubleshoot(); + #endif + return false; } - _endSPI(); - return true; - } - else { - return false; +#endif } + _endSPI(); + return true; } #endif // _SPIFLASH_H_ diff --git a/src/defines.h b/src/defines.h index ff02333..df629e7 100644 --- a/src/defines.h +++ b/src/defines.h @@ -1,7 +1,7 @@ -/* Arduino SPIFlash Library v.3.0.0 +/* Arduino SPIFlash Library v.2.7.0 * Copyright (C) 2017 by Prajwal Bhattaram * Created by Prajwal Bhattaram - 19/05/2015 - * Modified by Prajwal Bhattaram - 04/11/2017 + * Modified by Prajwal Bhattaram - 02/08/2017 * * This file is part of the Arduino SPIFlash Library. This library is for * Winbond NOR flash memory modules. In its current form it enables reading @@ -35,62 +35,74 @@ #define WRITEDISABLE 0x04 #define READSTAT1 0x05 #define READSTAT2 0x35 -#define WRITESTATEN 0x50 #define WRITESTAT 0x01 #define WRITEENABLE 0x06 #define SECTORERASE 0x20 #define BLOCK32ERASE 0x52 -#define BLOCK64ERASE 0xD8 -#define CHIPERASE 0x60 +#define CHIPERASE 0xC7 #define SUSPEND 0x75 #define ID 0x90 #define RESUME 0x7A #define JEDECID 0x9F -#define POWERDOWN 0xB9 #define RELEASE 0xAB +#define POWERDOWN 0xB9 +#define BLOCK64ERASE 0xD8 #define READSFDP 0x5A -#define UNIQUEID 0x4B //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// // General size definitions // -// B = Bytes; KiB = Kilo Bytes; MiB = Mega Bytes // +// B = Bytes; KB = Kilo bits; MB = Mega bits // //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -#define B(x) size_t(x*BYTE) -#define KB(x) size_t(x*KiB) -#define MB(x) size_t(x*MiB) +#define B1 1L +#define B2 2L +#define B4 4L +#define B8 8L +#define B16 16L +#define B32 32L +#define B64 64L +#define B80 80L +#define B128 128L +#define B256 256L +#define B512 512L +#define KB1 B1 * K +#define KB2 B2 * K +#define KB4 B4 * K +#define KB8 B8 * K +#define KB16 B16 * K +#define KB32 B32 * K +#define KB64 B64 * K +#define KB128 B128 * K +#define KB256 B256 * K +#define KB512 B512 * K +#define MB1 B1 * M +#define MB2 B2 * M +#define MB4 B4 * M +#define MB8 B8 * M +#define MB16 B16 * M +#define MB32 B32 * M +#define MB64 B64 * M +#define MB128 B128 * M +#define MB256 B256 * M +#define MB512 B512 * M //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// // Chip specific instructions // //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -//~~~~~~~~~~~~~~~~~~~~~~~~~ Winbond ~~~~~~~~~~~~~~~~~~~~~~~~~// - #define WINBOND_MANID 0xEF - #define SPI_PAGESIZE 0x100 - #define WINBOND_WRITE_DELAY 0x02 - #define WINBOND_WREN_TIMEOUT 10L - -//~~~~~~~~~~~~~~~~~~~~~~~~ Microchip ~~~~~~~~~~~~~~~~~~~~~~~~// - #define MICROCHIP_MANID 0xBF - #define SST25 0x25 - #define SST26 0x26 - #define ULBPR 0x98 //Global Block Protection Unlock (Ref sections 4.1.1 & 5.37 of datasheet) - -//~~~~~~~~~~~~~~~~~~~~~~~~ Cypress ~~~~~~~~~~~~~~~~~~~~~~~~// - #define CYPRESS_MANID 0x01 + //~~~~~~~~~~~~~~~~~~~~~~~~~ Winbond ~~~~~~~~~~~~~~~~~~~~~~~~~// + #define WINBOND_MANID 0xEF + #define PAGESIZE 0x100 -//~~~~~~~~~~~~~~~~~~~~~~~~ Adesto ~~~~~~~~~~~~~~~~~~~~~~~~// - #define ADESTO_MANID 0x1F - -//~~~~~~~~~~~~~~~~~~~~~~~~ Micron ~~~~~~~~~~~~~~~~~~~~~~~~// - #define MICRON_MANID 0x20 + //~~~~~~~~~~~~~~~~~~~~~~~~ Microchip ~~~~~~~~~~~~~~~~~~~~~~~~// + #define MICROCHIP_MANID 0xBF //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// // Definitions // //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// #define BUSY 0x01 #if defined (ARDUINO_ARCH_ESP32) -#define SPI_CLK 20000000 //Hz equivalent of 20MHz +#define SPI_CLK 20000000 #else -#define SPI_CLK 104000000 //Hz equivalent of 104MHz +#define SPI_CLK 104000000 //Hex equivalent of 104MHz #endif #define WRTEN 0x02 #define SUS 0x80 @@ -105,26 +117,23 @@ #define NOOVERFLOW false #define NOERRCHK false #define VERBOSE true -#define PRINTOVERRIDE true -#define ERASEFUNC 0xEF #if defined (SIMBLEE) #define BUSY_TIMEOUT 100L -#elif defined ENABLEZERODMA -#define BUSY_TIMEOUT 3500L #else -#define BUSY_TIMEOUT 1000L +#define BUSY_TIMEOUT 10L #endif #define arrayLen(x) (sizeof(x) / sizeof(*x)) #define lengthOf(x) (sizeof(x))/sizeof(byte) -#define BYTE 1L -#define KiB 1024L -#define MiB KiB * KiB +#define K 1024L +#define M K * K #define S 1000L #if defined (ARDUINO_ARCH_ESP8266) #define CS 15 #elif defined (ARDUINO_ARCH_SAMD) #define CS 10 +#elif defined __AVR_ATtiny85__ +#define CS 5 /********************************************************************************************* // Declaration of the Default Chip select pin name for RTL8195A // Note: This has been shifted due to a bug identified in the HAL layer SPI driver @@ -164,35 +173,19 @@ // List of Error codes // //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// - #define SUCCESS 0x00 - #define CALLBEGIN 0x01 - #define UNKNOWNCHIP 0x02 - #define UNKNOWNCAP 0x03 - #define CHIPBUSY 0x04 - #define OUTOFBOUNDS 0x05 - #define CANTENWRITE 0x06 - #define PREVWRITTEN 0x07 - #define LOWRAM 0x08 - #define SYSSUSPEND 0x09 - #define UNSUPPORTED 0x0A - #define ERRORCHKFAIL 0x0B - #define NORESPONSE 0x0C - #define UNSUPPORTEDFUNCTION 0x0D - #define UNKNOWNERROR 0xFE - -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -// Bit shift macros // -// Thanks to @VitorBoss // -// https://github.com/Marzogh/SPIFlash/issues/76 // -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// -#define Lo(param) ((char *)¶m)[0] //0x000y -#define Hi(param) ((char *)¶m)[1] //0x00y0 -#define Higher(param) ((char *)¶m)[2] //0x0y00 -#define Highest(param) ((char *)¶m)[3] //0xy000 -#define Low(param) ((int *)¶m)[0] //0x00yy -#define Top(param) ((int *)¶m)[1] //0xyy00 -//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// + #define SUCCESS 0x00 + #define CALLBEGIN 0x01 + #define UNKNOWNCHIP 0x02 + #define UNKNOWNCAP 0x03 + #define CHIPBUSY 0x04 + #define OUTOFBOUNDS 0x05 + #define CANTENWRITE 0x06 + #define PREVWRITTEN 0x07 + #define LOWRAM 0x08 + #define SYSSUSPEND 0x09 + #define UNSUPPORTED 0x0A + #define ERRORCHKFAIL 0x0B + #define NORESPONSE 0x0C + #define UNKNOWNERROR 0xFE -#ifndef LED_BUILTIN //fix for boards without that definition - #define LED_BUILTIN 13 -#endif + //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~// diff --git a/src/tinySPI.cpp b/src/tinySPI.cpp new file mode 100644 index 0000000..78e1303 --- /dev/null +++ b/src/tinySPI.cpp @@ -0,0 +1,53 @@ +/* Arduino SPIFlash Library v.2.7.0 + * Copyright (C) 2017 by Prajwal Bhattaram + * Created by Prajwal Bhattaram - 19/04/2017 + * Modified by Prajwal Bhattaram - 19/04/2017 + * Original code from @manitou48 + * + * This file is part of the Arduino SPIFlash Library. This library is for + * Winbond NOR flash memory modules. In its current form it enables reading + * and writing individual data variables, structs and arrays from and to various locations; + * reading and writing pages; continuous read functions; sector, block and chip erase; + * suspending and resuming programming/erase and powering down for low power operation. + * + * This Library is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This Library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License v3.0 + * along with the Arduino SPIFlash Library. If not, see + * . + */ + #include "SPIFlash.h" + +#if defined (__AVR_ATtiny85__) +void SPIFlash::_tinySPIbegin() { + PORTB &= ~(_BV(PORTB0) | _BV(PORTB1) | _BV(PORTB2) | cs_mask); + DDRB &= ~_BV(PORTB0); // DI (NOT MISO) + DDRB |= _BV(PORTB1) // DO (NOT MOSI) + | _BV(PORTB2) // SCK + | cs_mask; // CS +} + +static uint8_t SPIFlash::_tinySPItransfer(uint8_t _data) { + USIDR = _data; + for (uint8_t i = 0; i < 8; i++) { + SPIBIT + } + /*SPIBIT + SPIBIT + SPIBIT + SPIBIT + SPIBIT + SPIBIT + SPIBIT + SPIBIT*/ + return USIDR; +} +#endif diff --git a/src/troubleshoot.cpp b/src/troubleshoot.cpp index ce71a9f..41e4edc 100644 --- a/src/troubleshoot.cpp +++ b/src/troubleshoot.cpp @@ -1,4 +1,4 @@ -/* Arduino SPIFlash Library v.3.0.0 +/* Arduino SPIFlash Library v.2.7.0 * Copyright (C) 2017 by Prajwal Bhattaram * Created by Prajwal Bhattaram - 14/11/2016 * Modified by @boseji - 02/03/2017 @@ -25,110 +25,162 @@ * . */ -#include "SPIFlash.h" - -//Subfunctions for troubleshooting function -void SPIFlash::_printErrorCode(void) { - Serial.print("Error code: 0x"); - if (errorcode < 0x10) { - Serial.print("0"); - } - Serial.println(errorcode, HEX); -} - -void SPIFlash::_printSupportLink(void) { - Serial.print("If this does not help resolve/clarify this issue, "); - Serial.println("please raise an issue at http://www.github.com/Marzogh/SPIFlash/issues with the details of what your were doing when this error occurred"); -} -//Troubleshooting function. Called when #ifdef RUNDIAGNOSTIC is uncommented at the top of this file. -void SPIFlash::_troubleshoot(uint8_t _code, bool printoverride) { - bool _printoverride; - errorcode = _code; -#ifdef RUNDIAGNOSTIC - _printoverride = true; + #include "SPIFlash.h" + + #if !defined (__AVR_ATtiny85__) + //Subfunctions for troubleshooting function + void SPIFlash::_printErrorCode(void) { + Serial.print("Error code: 0x"); + if (errorcode < 0x10) { + Serial.print("0"); + } + Serial.println(errorcode, HEX); + } + + void SPIFlash::_printSupportLink(void) { + Serial.print("If this does not help resolve/clarify this issue, "); + Serial.println("please raise an issue at http://www.github.com/Marzogh/SPIFlash/issues with the details of what your were doing when this error occurred"); + } + //Troubleshooting function. Called when #ifdef RUNDIAGNOSTIC is uncommented at the top of this file. + void SPIFlash::_troubleshoot(void) { + + switch (errorcode) { + case SUCCESS: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("Action completed successfully"); + #endif + break; + + case NORESPONSE: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("Check your wiring. Flash chip is non-responsive."); + _printSupportLink(); + #endif + break; + + case CALLBEGIN: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("*constructor_of_choice*.begin() was not called in void setup()"); + _printSupportLink(); + #endif + break; + + case UNKNOWNCHIP: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + Serial.print("Error code: 0x0"); + Serial.println(UNKNOWNCHIP, HEX); + #else + Serial.println("Unable to identify chip. Are you sure this chip is supported?"); + _printSupportLink(); + #endif + Serial.println("Chip details:"); + Serial.print("manufacturer ID: 0x"); Serial.println(_chip.manufacturerID, HEX); + Serial.print("capacity ID: 0x");Serial.println(_chip.memoryTypeID, HEX); + Serial.print("device ID: 0x");Serial.println(_chip.capacityID, HEX); + break; + + case UNKNOWNCAP: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("Unable to identify capacity. Is this chip officially supported? If not, please define a `CAPACITY` constant and include it in flash.begin(CAPACITY)."); + _printSupportLink(); + #endif + break; + + case CHIPBUSY: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("Chip is busy."); + Serial.println("Make sure all pins have been connected properly"); + _printSupportLink(); + #endif + break; + + case OUTOFBOUNDS: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("Page overflow has been disabled and the address called exceeds the memory"); + _printSupportLink(); + #endif + break; + + case CANTENWRITE: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("Unable to Enable Writing to chip."); + Serial.println("Please make sure the HOLD & WRITEPROTECT pins are pulled up to VCC"); + _printSupportLink(); + #endif + break; + + case PREVWRITTEN: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("This sector already contains data."); + Serial.println("Please make sure the sectors being written to are erased."); + _printSupportLink(); + #endif + break; + + case LOWRAM: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("You are running low on SRAM. Please optimise your program for better RAM usage"); + /*#if defined (ARDUINO_ARCH_SAM) + Serial.print("Current Free SRAM: "); + Serial.println(freeRAM()); + #endif*/ + _printSupportLink(); + #endif + break; + + case SYSSUSPEND: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("Unable to suspend/resume operation."); + _printSupportLink(); + #endif + break; + + case UNSUPPORTED: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); + #else + Serial.println("This function is not supported by the current flash IC."); + _printSupportLink(); + #endif + break; + + case ERRORCHKFAIL: +#if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) + _printErrorCode(); #else - _printoverride = printoverride; + Serial.println("Write Function has failed errorcheck."); + _printSupportLink(); #endif - if (_printoverride) { - #if defined (ARDUINO_ARCH_AVR) + break; + + default: + #if defined (ARDUINO_ARCH_AVR) || defined (__AVR_ATtiny85__) _printErrorCode(); #else - Serial.println(); - switch (_code) { - case SUCCESS: - Serial.println("Function executed successfully"); - break; - - case NORESPONSE: - Serial.println("Check your wiring. Flash chip is non-responsive."); - break; - - case CALLBEGIN: - Serial.println("*constructor_of_choice*.begin() was not called in void setup()"); - break; - - case UNKNOWNCHIP: - Serial.println("Unable to identify chip. Are you sure this chip is supported?"); - Serial.println("Chip details:"); - Serial.print("manufacturer ID: 0x"); Serial.println(_chip.manufacturerID, HEX); - Serial.print("capacity ID: 0x"); Serial.println(_chip.memoryTypeID, HEX); - Serial.print("device ID: 0x"); Serial.println(_chip.capacityID, HEX); - break; - - case UNKNOWNCAP: - Serial.println("Unable to identify capacity. Is this chip officially supported? If not, please define a `CAPACITY` constant and include it in flash.begin(CAPACITY)."); - break; - - case CHIPBUSY: - Serial.println("Chip is busy."); - Serial.println("Make sure all pins have been connected properly"); - break; - - case OUTOFBOUNDS: - Serial.println("Page overflow has been disabled and the address called exceeds the memory"); - break; - - case CANTENWRITE: - Serial.println("Unable to Enable Writing to chip."); - Serial.println("Please make sure the HOLD & WRITEPROTECT pins are pulled up to VCC"); - break; - - case PREVWRITTEN: - Serial.println("This sector already contains data."); - Serial.println("Please make sure the sectors being written to are erased."); - break; - - case LOWRAM: - Serial.println("You are running low on SRAM. Please optimise your program for better RAM usage"); - /*#if defined (ARDUINO_ARCH_SAM) - Serial.print("Current Free SRAM: "); - Serial.println(freeRAM()); - #endif*/ - break; - - case UNSUPPORTEDFUNCTION: - Serial.println("This function is not supported by the flash memory hardware."); - break; - - case SYSSUSPEND: - Serial.println("Unable to suspend/resume operation."); - break; - - case UNSUPPORTED: - Serial.println("This function is not supported by the current flash IC."); - break; - - case ERRORCHKFAIL: - Serial.println("Write Function has failed errorcheck."); - break; - - default: - Serial.println("Unknown error"); - break; - } - if (_code != SUCCESS) { - _printSupportLink(); - } + Serial.println("Unknown error"); + _printSupportLink(); #endif - } -} + break; + } + } + #endif