diff --git a/.gitignore b/.gitignore
index e37ce9534..f5ee135e6 100644
--- a/.gitignore
+++ b/.gitignore
@@ -66,6 +66,7 @@ client/traces/*
 armsrc/TEMP EMV/*
 tools/mf_nonce_brute/*
 tools/andrew/*
+tools/jtag_openocd/openocd_configuration
 ppls patches/*
 *- Copy.*
 
@@ -73,3 +74,4 @@ client/lualibs/mf_default_keys.lua
 client/lualibs/pm3_cmd.lua
 # recompiled
 fpga_version_info.c
+
diff --git a/.travis.yml b/.travis.yml
index be68e761a..f42dd8f1c 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -1,35 +1,25 @@
 # Travis-CI Build for RfidResearchGroup/Proxmark3
 language: c
 
-#default linux build env is: Ubuntu 14.04 trusty
+#default linux build env is: xenial
 compiler: gcc
 
 # Test on Linux and MacOS
 matrix:
   include:
-#    - os: osx
-#      osx_image: xcode7.3	# OS X 10.11
-#    - os: osx
-#      osx_image: xcode8.3   # OS X 10.12
-#    - os: osx
-#      osx_image: xcode9     # OS X 10.13
     - os: osx
-      osx_image: xcode9.1   # OS X 10.13.1
+      osx_image: xcode11
     - os: linux
-      dist: trusty
+      dist: xenial
       sudo: required
 
-before_install:
-## Install ARM toolchain on Linux.
-## add our homebrew tap for MacOS
-## Note: all dependencies on MacOS should be resolved by the brew install command
-  if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then 
-    sudo apt-get update -qq;
-    sudo apt-get install -y gcc-arm-none-eabi;
-  elif [[ "$TRAVIS_OS_NAME" == "osx" ]]; then
-    brew update;
-    brew tap RfidResearchGroup/proxmark3;
-  fi
+addons:
+  apt:
+    packages:
+      - gcc-arm-none-eabi
+      - libnewlib-dev
+  homebrew:
+    taps: RfidResearchGroup/proxmark3
 
 install:
   if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then 
@@ -40,8 +30,6 @@ install:
     make all;
   fi
 
-before_script:
-
 script: 
 ## start and run a test script
   if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then 
diff --git a/CHANGELOG.md b/CHANGELOG.md
index d46c8222b..c9da6eba9 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -3,6 +3,36 @@ All notable changes to this project will be documented in this file.
 This project uses the changelog in accordance with [keepchangelog](http://keepachangelog.com/). Please use this to write notable changes, which is not the same as git commit log...
 
 ## [unreleased][unreleased]
+ - Fix T55x7 Downlink timings backward compatible (@mwalker33)
+ - Add proper Makefile halting when using incompatible STANDALONE and PLATFORM vars (@doegox)
+ - Add T55x7 Downlink mode support (@mwalker33)
+ - Add SPIFFS Flash filesystem support (@cjbrigato)
+ - Fix support for flashing 512K units with old bootrom (@slurdge/@doegox)
+ - Fix `hf mf sim` - wrong access rights to write key B in trailer (@McEloff)
+ - Add option -i to flasher to query Pm3 for its memory size (@doegox)
+ - Add support for flashing 512K units (@slurdge)
+ - Add a simple python tool to check the elf sizes (@slurdge)
+ - Change: new keys for Vigik badges in default_keys.dict (@luminouw)
+ - Add `hw standalone` to jump to standalone mode from command line or script (@doegox)
+ - Add to `hf 14a apdu` print apdu and compose apdu (@merlokk)
+ - Change: buggy `mem read` removed, `mem save` renamed `mem dump`, can now display too (@doegox)
+ - Fix: timeout for mem wipe was too short, thanks @cjbrigato (@doegox)
+ - Fix `hf mf sim` - Mifare Classic simulation more flexible anti-collision check (@McEloff)
+ - Change: `hf mf sim` - Mifare Classic simulation not respond NACK on invalid authentication request (@McEloff)
+ - Change: `read_pwd_mem.lua` now handles Mifare Classic dictionaries large than 4096 bytes (@iceman1001)
+ - Change: Do not clear trace log during `hf mf chk`, to save whole process history (@McEloff)
+ - Add `msleep` command, for pauses in scripts (@doegox)
+ - Add support for WSL in proxmark.sh (@doegox)
+ - Add documentation for usage of Proxmark3 under WSL (@doegox)
+ - Change: replace aes.c with mbedtls version (@slurdge)
+ - Change: replace ukbhit by kbd_enter_pressed, not requiring tcgetattr (@xianglin1998/@doegox)
+ - Add config for RaspberryPi in JTAG tools (@doegox)
+ - Add config for FTDI C232HM-DDHSL-0 in JTAG tools (@doegox)
+ - Fix compilation under MacOSX with binutils (@matrix)
+ - Add dynamic report of the chipID for flashing purposes (@slurdge)
+ - Fix Clang warnings (@matrix)
+ - Fix EMVGPO bug (@matrix)
+ - Add hitag2 write password auth (@ViRb3)
  - Add check if bootloader segment is within bounds (@slurdge)
  - Add `hf 15 csetuid` - set UID on ISO-15693 Magic tags (@t0m4-null)
  - Change: Print help if unknown arg for hitag reader/writer (@ViRb3)
@@ -10,58 +40,58 @@ This project uses the changelog in accordance with [keepchangelog](http://keepac
  - Add compiler info in client & ARM sections (@slurdge)
  - Add support for automatic COM detection on Windows (@slurdge)
  - Add support for compilation on RaspberryPiZero (armv6) (@doegox)
- - Change: updates to README (@iceman)
- - Change: hf mf/mfu dbg => hw dbg (@doegox)
+ - Change: updates to README (@iceman1001)
+ - Change: `hf mf/mfu dbg` => `hw dbg` (@doegox)
  - Change: replace usb_poll_validate_length() by data_available() that supports USART too (@doegox)
  - Make sure standalone modes can be launched when connected on USB without client (@doegox)
- - Change: cleaner makefile execution, use 'make V=1' if you want to see full lines (@doegox)
+ - Change: cleaner makefile execution, use `make V=1` if you want to see full lines (@doegox)
  - Change: automate make clean when platform definitions are changed (@doegox)
  - Add STANDALONE option to Makefile.hal (@Fl0-0)
- - Change: mem info - production public key to verify rdv4.0 flash  signature (@iceman)
+ - Change: mem info - production public key to verify rdv4.0 flash  signature (@iceman1001)
  - Fix specify that we need TCP and not UDP connection (@phcoder)
- - Change: lf cotag demod - adjusted error trigger (@iceman)
- - Add documentation on BT add-on (@iceman/@doegox)
+ - Change: lf cotag demod - adjusted error trigger (@iceman1001)
+ - Add documentation on BT add-on (@iceman1001/@doegox)
  - Change: new button behaviour in bootloader mode, no need to keep it pressed, press again to interrupt (@doegox)
- - Change: new keys in dicts, new mem layout to accomodate them (@iceman/various)
- - Fix lf sim -  if called with empty graphbuffer all strange things happend, like turning on HF field (@iceman)
+ - Change: new keys in dicts, new mem layout to accomodate them (@iceman1001/various)
+ - Fix lf sim -  if called with empty graphbuffer all strange things happend, like turning on HF field (@iceman1001)
  - Change: hf 14a sim / hf mf sim - check buttonpress/usb frame fewer times in order not to disrupt simulation (@McEloff)
- - Change: data convertbitstream - converts bit to max/min in order to facilitate demodulation of simulation data (@iceman)
- - Change: lf em 410x_demod - now can demod the simulation data (@iceman)
+ - Change: data convertbitstream - converts bit to max/min in order to facilitate demodulation of simulation data (@iceman1001)
+ - Change: lf em 410x_demod - now can demod the simulation data (@iceman1001)
  - Add HC-06 scripts for BT add-on (@doegox)
- - Fix lf nedap sim - error when adding parity (@iceman)
- - Add documentation on UART and baudrates (@doegox/@iceman)
- - Change: prompt now shows which channel is used (@iceman)
- - Change: USART baudrates computation, up to 6Mbps (@iceman/@doegox)
+ - Fix lf nedap sim - error when adding parity (@iceman1001)
+ - Add documentation on UART and baudrates (@doegox/@iceman1001)
+ - Change: prompt now shows which channel is used (@iceman1001)
+ - Change: USART baudrates computation, up to 6Mbps (@iceman1001/@doegox)
  - Change: hf mf nack - keep sync if started without card over antenna
  - Add usart btfactory - to reset a misconfigured BT add-on (@doegox)
- - Change: hw status - now prints number of dictionary keys loaded (@iceman)
+ - Change: hw status - now prints number of dictionary keys loaded (@iceman1001)
  - Add home, end, pageup, and pagedown keybinds to the plot GUI. Also fix paged movement in GUI. (@mcd1992)
- - Change legic.lua saves data in EML and BIN formats (@iceman)
+ - Change legic.lua saves data in EML and BIN formats (@iceman1001)
  - Change hf tune - is now synchronous (for BT add-on) and can be interrupted by kbd (@doegox)
  - Change: update macOS install instruction (@ Uli Heilmeier)
  - Add trace ouput in hexdump format for Wireshark import (@ Uli Heilmeier)
  - Add usart btpin - to change BT add-on PIN (@doegox)
- - Add reconnection support (@iceman/@doegox)
+ - Add reconnection support (@iceman1001/@doegox)
  - Add usart tx/rx/... - USART developer commands (@doegox)
  - Add PLATFORM_EXTRAS, WITH_FPC_USART_HOST, BTADDON Makefile configuration (@doegox)
- - Fix slow reconfigure on mingw of serial port (@iceman)
- - Fix cross thread communictions of timeout variable (@iceman)
+ - Fix slow reconfigure on mingw of serial port (@iceman1001)
+ - Fix cross thread communictions of timeout variable (@iceman1001)
  - Change: client is now "universal", adapting to Proxmark3 capabilities (@doegox)
- - Add disconnect support to Lua (@iceman)
+ - Add disconnect support to Lua (@iceman1001)
  - Change: handles FPC/FLASH FW more gracefully on non-RDV4 pm3 (@doegox)
  - Add JTAG support for Shikra (@NinjaStyle82)
  - Change: smart color handling: only if linux and on real term (@doegox)
- - Change: reconfigure uart timeouts when compiled for FPC and connecting over USB (@iceman)
- - Change: fast push for many commands (@iceman/@doegox)
- - Add: fast push for Lua (@iceman)
- - Add NDEF parser in Lua (@iceman)
- - Change: improve NDEF parser (@iceman)
- - Change: all commands got migrated to MIX/NG packet format (@iceman/@doegox)
+ - Change: reconfigure uart timeouts when compiled for FPC and connecting over USB (@iceman1001)
+ - Change: fast push for many commands (@iceman1001/@doegox)
+ - Add: fast push for Lua (@iceman1001)
+ - Add NDEF parser in Lua (@iceman1001)
+ - Change: improve NDEF parser (@iceman1001)
+ - Change: all commands got migrated to MIX/NG packet format (@iceman1001/@doegox)
  - Fix: Mifare Ultralight read block missing bytes (@doegox)
- - Add support new frame format in all Lua scripts (@iceman)
+ - Add support new frame format in all Lua scripts (@iceman1001)
  - Add CMD_CAPABILITIES for pm3 to inform dynamically the client (@doegox)
- - Change baudrate handling, make it clear it's only indicative for USB-CDC & BT (@doegox)
- - Change: new progressive light scheme for 'hw detectreader' (@doegox)
+ - Change baudrate handling, make it clear it is only indicative for USB-CDC & BT (@doegox)
+ - Change: new progressive light scheme for `hw detectreader` (@doegox)
  - Add common error definitions system for retvals (@doegox)
  - Change USART RX & TX code and fix delays handling to make it more robust, especially over BT (@doegox)
  - Add support for new frames format, speedup & huge changes, see doc/new_frame_format.txt (@doegox)
@@ -74,218 +104,219 @@ This project uses the changelog in accordance with [keepchangelog](http://keepac
  - Add color support to Dbprintf & alike and rework Dbprintf flags (@doegox)
  - Change: archive (and fix) hid-flasher (@doegox)
  - Add standalone placeholder to simplify new standalone integration (@doegox)
- - Change: refactor standalone mode info string (@iceman)
- - Add iceman skeleton standalone mode for ppl to use as base for their new modes (@iceman)
+ - Change: refactor standalone mode info string (@iceman1001)
+ - Add iceman skeleton standalone mode for ppl to use as base for their new modes (@iceman1001)
  - Change: move compilation options to Makefile.hal (@doegox)
- - Fix compilation under OSX (@iceman)
+ - Fix compilation under OSX (@iceman1001)
  - Add openocd config files for JLink (@doegox)
  - Fix compilation dependencies for recovery (@doegox)
  - Fix segfault when loading a file (@doegox)
  - Change/Add new dump format for Ultralight/NTAG, counters support, simulation (@mceloff)
- - Add 'hf mf sim' full-byte split anticollision support (@mceloff)
- - Fix/Add 'hf mf sim' bugs fix, RATS support, etc (@mceloff)
+ - Add `hf mf sim` full-byte split anticollision support (@mceloff)
+ - Fix/Add `hf mf sim` bugs fix, RATS support, etc (@mceloff)
  - Fix serial of FPC. (@ryan)
- - Fix 'data shiftgraphzero' corrupting end of GraphBuffer (@doegox)
- - Fix 'hf legic info' - unsegmented card now uses card size to calc remaining length (@iceman)
- - Add 36bit HID format, extend calcWiegand() to include oem bits ((@davidbeauchamp)
- - Fix 'hf mf hardnested' - not verify key when reading nonce file (@iceman)
- - Change optimizations for ask/bi (@iceman)
- - Fix 'hf mf sim' - bugs fix, refactoring (@mceloff)
+ - Fix `data shiftgraphzero` corrupting end of GraphBuffer (@doegox)
+ - Fix `hf legic info` - unsegmented card now uses card size to calc remaining length (@iceman1001)
+ - Add 36bit HID format, extend calcWiegand() to include oem bits (@davidbeauchamp)
+ - Fix `hf mf hardnested` - not verify key when reading nonce file (@iceman1001)
+ - Change optimizations for ask/bi (@iceman1001)
+ - Fix `hf mf sim` - bugs fix, refactoring (@mceloff)
  - Add WRITE and COMPATIBLE_WRITE support to Mifare Ultralight/NTAG simulation (@mceloff)
  - Change installation instructions and add video links (@5w0rdfish)
- - Change 'hf mf sim' to support more types (@vratiskol)
- - Change better strong wave detection for biphase (@iceman)
- - Add 'script run test_t55x7' (@iceman)
- - Add new lua scripting support for some t55xx commands (@iceman)
+ - Change `hf mf sim` to support more types (@vratiskol)
+ - Change better strong wave detection for biphase (@iceman1001)
+ - Add `script run test_t55x7` (@iceman1001)
+ - Add new lua scripting support for some t55xx commands (@iceman1001)
  - Add FPC USART for BT add-on with pm3 client. (@doegox)
- - Add '-b baudrate' option to the pm3 client. (@doegox)
- - Change 'lf t55xx info': tell if known configuration block0. (@iceman)
+ - Add `-b baudrate` option to the pm3 client. (@doegox)
+ - Change `lf t55xx info`: tell if known configuration block0. (@iceman1001)
  - Fix/Add FPC usart: fix TX, bring RX, full speed. (@doegox)
- - Change 'lf t55xx config' options: allow to toggle on/off i/q5/st
- - Change 'lf t55xx info': support offline block0, Q5, fix extended, add warns. (@doegox)
+ - Change `lf t55xx config` options: allow to toggle on/off i/q5/st
+ - Change `lf t55xx info`: support offline block0, Q5, fix extended, add warns. (@doegox)
  - Avoid race condition when flasher finds the not yet closed pm3 port. (@doegox)
- - Fix 'lf t55xx trace': read the proper block. (@doegox)
+ - Fix `lf t55xx trace`: read the proper block. (@doegox)
  - Fix Indala 64 on T55xx: use PSK1. (@doegox)
  - Force proper Linefeed (LF) handling in ProxSpace. (@vratiskol)
- - Fix Makefiles race conditions to allow parallel compilation, e.g. 'make -j8'. (@doegox)
+ - Fix Makefiles race conditions to allow parallel compilation, e.g. `make -j8`. (@doegox)
  - Add - dictionary key file for MFU. (not in use at the moment) (@mazodude)
- - Change 'lf fdx demod - better biphase maxerrors. (@MalteHillmann)
- - Change 'hf mf sim' - now works better against android (@mceloff)
- - Fix 'lf t55xx brute' - now works after aquiredata adaptations (@iceman)
- - Fix 'lf t55xx chk' - now works after aquiredata adaptations (@iceman)
- - Fix 'lf t55xx recoverpwd' - now works after aquiredata adaptations (@iceman)
- - Fix 'data detect p' - reverted bad clock detection (@iceman)
- - Change 'data detect a' - better clock detection (@iceman)
- - Add 'hf 14a info' - now detects some magic card Gen2 (@iceman)
- - Removed 'LCD' code in armsrc compilation (@iceman)
- - Change - Generic fixes of codestyle (@doegox) (@iceman)
- - Change 'lf indala demod' - refactoring (@iceman)
+ - Change `lf fdx demod` - better biphase maxerrors. (@MalteHillmann)
+ - Change `hf mf sim` - now works better against android (@mceloff)
+ - Fix `lf t55xx brute` - now works after aquiredata adaptations (@iceman1001)
+ - Fix `lf t55xx chk` - now works after aquiredata adaptations (@iceman1001)
+ - Fix `lf t55xx recoverpwd` - now works after aquiredata adaptations (@iceman1001)
+ - Fix `data detect p` - reverted bad clock detection (@iceman1001)
+ - Change `data detect a` - better clock detection (@iceman1001)
+ - Add `hf 14a info` - now detects some magic card Gen2 (@iceman1001)
+ - Removed `LCD` code in armsrc compilation (@iceman1001)
+ - Change - Generic fixes of codestyle (@doegox) (@iceman1001)
+ - Change `lf indala demod` - refactoring (@iceman1001)
  - Change - handling fault bit markers (7) and partial nibbles in hex printing (@doegox)
  - Change - printing of fault bit markers (7) using a dot (@doegox)
- - Change 'sc upgrade' - firmware file integrity check (@piwi)
- - Fix 'data rawdemod am' - last bit was missing (@doegox)
- - Fix 'hf 15 dump f' - also selects tag first (@iceman)
- - Fix 'hf iclass clone' - missing fileclose (@iceman)
- - Add 'trace list hitag' - old hitag annotations now use the new trace (@iceman)
- - Change 'lf hitag sim' - loads bin/eml/json (@iceman)
- - Change 'lf hitag reader 21' - saves in bin/eml/json (@iceman)
- - Change 'lf hitag' - refactoring (@iceman)
- - Change 'lf hitag' - refactoring (@piwi)
- - Fix 'lf hitag' - generic fix for missing clock init (@piwi)
+ - Change `sc upgrade` - firmware file integrity check (@piwi)
+ - Fix `data rawdemod am` - last bit was missing (@doegox)
+ - Fix `hf 15 dump f` - also selects tag first (@iceman1001)
+ - Fix `hf iclass clone` - missing fileclose (@iceman1001)
+ - Add `trace list hitag` - old hitag annotations now use the new trace (@iceman1001)
+ - Change `lf hitag sim` - loads bin/eml/json (@iceman1001)
+ - Change `lf hitag reader 21` - saves in bin/eml/json (@iceman1001)
+ - Change `lf hitag` - refactoring (@iceman1001)
+ - Change `lf hitag` - refactoring (@piwi)
+ - Fix `lf hitag` - generic fix for missing clock init (@piwi)
  - Fix fsk sim operations on deviceside - avoid division by zero (@doegox)
- - Fix 'hf mf fchk' - condition always false (@doegox)
- - Fix 'lf t55xx recoverpw' - shift as u32 (@doegox)
- - Fix 'lf ti demod' - shift as u32 (@doegox)
- - Fix 'lf ti read' - shift as u32 (@doegox)
- - Fix 'lf t55xx chk' - condition always false (@doegox) 
- - Change 'lf sim' - ledcontrol refactoring (@doegox)
- - Fix 'hf mf nack' - signedness bug (@doegox)
- - Fix 'hf epa cnonce' - check return value (@doegox)
- - Fix 'lf hitag write' - condition always true (@doegox)
- - Fix 'mem write' - added extra check (@doegox)
- - Fix 'iso15693' - bad string cpy (@doegox)
- - Fix 'make style' - EOF LF support (@doegox)
- - Add 'hf 14b raw' - added -t for timeout  (@iceman)
- - Rename 'lf hitag snoop' - renamed to 'lf hitag sniff' (@iceman)
- - Rename 'lf snoop' - renamed to 'lf sniff' (@iceman)
- - Rename 'hf snoop' - renamed to 'hf sniff' (@iceman)
- - Fix 'hf mfp wrbl' - more blocks available (@merlokk)
- - Add 'make platform' - compile for non-rdv4 devices made simpler (@doegox)
+ - Fix `hf mf fchk` - condition always false (@doegox)
+ - Fix `lf t55xx recoverpw` - shift as u32 (@doegox)
+ - Fix `lf ti demod` - shift as u32 (@doegox)
+ - Fix `lf ti read` - shift as u32 (@doegox)
+ - Fix `lf t55xx chk` - condition always false (@doegox)
+ - Change `lf sim` - ledcontrol refactoring (@doegox)
+ - Fix `hf mf nack` - signedness bug (@doegox)
+ - Fix `hf epa cnonce` - check return value (@doegox)
+ - Fix `lf hitag write` - condition always true (@doegox)
+ - Fix `mem write` - added extra check (@doegox)
+ - Fix `iso15693` - bad string cpy (@doegox)
+ - Fix `make style` - EOF LF support (@doegox)
+ - Add `hf 14b raw` - added -t for timeout  (@iceman1001)
+ - Rename `lf hitag snoop` - renamed to `lf hitag sniff` (@iceman1001)
+ - Rename `lf snoop` - renamed to `lf sniff` (@iceman1001)
+ - Rename `hf snoop` - renamed to `hf sniff` (@iceman1001)
+ - Fix `hf mfp wrbl` - more blocks available (@merlokk)
+ - Add `make platform` - compile for non-rdv4 devices made simpler (@doegox)
  - Change Makefiles optimizations when recompiling (@doegox)
- - Fix 'data load' - loads TITEST.txt again (@iceman)
- - Change 'lf search' - now detects TI (@iceman)
+ - Fix `data load` - loads TITEST.txt again (@iceman1001)
+ - Change `lf search` - now detects TI (@iceman1001)
  - Change fixing signal cleaning for LF (@doegox)
- - Fix 'lf paradox demod' - wrong check (@iceman)
- - Change 'lf t55xx' - aquiredata uses getsamples (@iceman)
- - Fix 'lf search' - chipset detection restore demod buffer again (@iceman)
- - Add 'make style' (@doegox)
+ - Fix `lf paradox demod` - wrong check (@iceman1001)
+ - Change `lf t55xx` - aquiredata uses getsamples (@iceman1001)
+ - Fix `lf search` - chipset detection restore demod buffer again (@iceman1001)
+ - Add `make style` (@doegox)
  - Fix mixed tabs vs spaces. Now only use 4 space as tab. (@doegox)
- - Fix 'lf visa2000 read' - too few samples (@iceman)
- - Fix 'lf t55xx bruteforce' - infinity loop (@doegox)
- - Fix 'analyse nuid' - correct crc (@doegox)
+ - Fix `lf visa2000 read` - too few samples (@iceman1001)
+ - Fix `lf t55xx bruteforce` - infinity loop (@doegox)
+ - Fix `analyse nuid` - correct crc (@doegox)
  - Add command history not repeating logged commands (@doegox)
  - Fix path for aidjson (@doegox)
  - Fix missing init i2x (@doegox)
- - Fix '14b select card' - (@doegox)
- - Add 'hf mf ndef' - parsing of NDEF messages (@merlokk)
- - Add 'hf mf mad' - parsing of Mifare Application Directory (@merlokk)
- - Rename 'lf snoop' -> 'lf sniff' (@iceman)
- - Rename 'hf snoop' -> 'hf sniff' (@iceman)
- - Change generally added more colors (@iceman)
- - Change 'sc upgrade' updated firmware v3.11 (RDV40) (@sentiprox)
- - Change 'data autocorrelate' - better visual representation and added extra peak detection (@iceman)
- - Fix 'lf search' - false positive indala identification fixed (@iceman)
- - Add 'lf keri' - basic support for Keri tags (@iceman)
- - Add 'hf mf list' - re-added it again (@iceman)
- - Fix - A lot of bugfixes, like memory leaks (@iceman)
- - Change 'hf 14a antifuzz' - original implementation (@asfabw),  reworked a bit
- - Fix 'hf mf fchk' (@iceman)
- - Fix 'usb slow on posix based systems' (@fl0-0)
- - Change 'lf pcf7931' - improved read code (@sguerrini97)
- - Change 'hf felica list' - started with some FeliCa annotations (@iceman)
- - Fix 'hf tune' - now works as expected (@iceman)
- - Add 'option to use flash memory to upload dictionary files' (RDV40) (@iceman)
- - Fix 'printing percentage now standard compliant' (@fabled)
- - Add 'emv roca' - command to test for ROCA vuln in public RSA modulus (@merlokk / @iceman)
+ - Fix `14b select card` - (@doegox)
+ - Add `hf mf ndef` - parsing of NDEF messages (@merlokk)
+ - Add `hf mf mad` - parsing of Mifare Application Directory (@merlokk)
+ - Rename `lf snoop` -> `lf sniff` (@iceman1001)
+ - Rename `hf snoop` -> `hf sniff` (@iceman1001)
+ - Change generally added more colors (@iceman1001)
+ - Change `sc upgrade` updated firmware v3.11 (RDV40) (@sentiprox)
+ - Change `data autocorrelate` - better visual representation and added extra peak detection (@iceman1001)
+ - Fix `lf search` - false positive indala identification fixed (@iceman1001)
+ - Add `lf keri` - basic support for Keri tags (@iceman1001)
+ - Add `hf mf list` - re-added it again (@iceman1001)
+ - Fix - A lot of bugfixes, like memory leaks (@iceman1001)
+ - Change `hf 14a antifuzz` - original implementation (@asfabw),  reworked a bit
+ - Fix `hf mf fchk` (@iceman1001)
+ - Fix `usb slow on posix based systems` (@fl0-0)
+ - Change `lf pcf7931` - improved read code (@sguerrini97)
+ - Change `hf felica list` - started with some FeliCa annotations (@iceman1001)
+ - Fix `hf tune` - now works as expected (@iceman1001)
+ - Add `option to use flash memory to upload dictionary files` (RDV40) (@iceman1001)
+ - Fix `printing percentage now standard compliant` (@fabled)
+ - Add `emv roca` - command to test for ROCA vuln in public RSA modulus (@merlokk / @iceman1001)
  - Added TCP ports support (on linux) (@phcoder)
  - Added HF sniff standalone mode with optional storing of ULC/NTAG/ULEV1 authentication attempts (@bogiton)
- - Fix 'Lining up plot and control window' (@anticat)
- - Fix 'annoying focus behaviour' on OSX  (@Anticat)
+ - Fix `Lining up plot and control window` (@anticat)
+ - Fix `annoying focus behaviour` on OSX  (@Anticat)
  - Implemented AppNap API, fixing #283 and #627 OSX USB comm issues (@AntiCat)
- - Added 'sc brute' - a naive SFI bruteforcer for contact smartcards (RDV40) (@iceman)
- - Change 'lf t55xx detectconfig' - now optional to persist settings to flashmem (RDV40) (@iceman)
- - Change 'hf mf csave' - now saves both EML/BIN formats (@iceman)
- - Change 'hf mf esave' - now saves both EML/BIN formats (@iceman)
- - Fix 'compiler warning on macos and gcc7.1 or higher'  (@TomHarkness)
- - Fix 'crash on Bionic libc if CloseProxmark is called twice' (@micolous)
- - Change 'lf hid' - got an  updated to Kastle format (@xilni)
- - Added 'lf t55xx deviceconfig' - enables custom t55xx timing settings. (RDV40) (@iceman)
- - Chg adaptations for FPC communications (work in progress)  (@iceman)
- - Fix 'stand-alone Colin' - remake to benefit from flashmem for persistence. (@cjbrigato)
- - Fix 'LEGIC SIM' - remake of legic sim (@drandreas)
- - Changed 'proxmark3 client threading'  - remake from official repo (@micolous)
- - Add 'rem' - new command that adds a line to the log file (@didierStevens)
- - Fix 'EM410xdemod  empty tag id in lfops.c' (@Defensor7)  
- - Fix 'usb device descriptor' - some android phones will enumerate better when iSerialnumber isn't a multiple of 8 (@micolous, @megabug)
- - Fix 'StandaloneMode LF' -  when collecting signal, justNoise detection is needed (@didierStevens, @Megabug)
- - Fix 'StandAloneMode Colin' - mifare1ksim called with right params (@cjbrigato)
- - Improved 'install.sh' to install dependencies for Ubuntu 18.04 and using max number of processors during compilation (@joanbono)				
- - Modified 'install.sh' script to work in macOS and Linux + added the 'update.sh' and 'proxmark3.sh' from joanbono (@TomHarkness)	 
- - Fix 'hf emv' - some cards need to have Le=0x00, some don't need to have (@merlokk)
- - Fix 'hf legic'  enhancement of rx / tx in legic commands (@drandreas)
- - Fix 'data buffclear' - now frees bigbuff also (@iceman)
+ - Added `sc brute` - a naive SFI bruteforcer for contact smartcards (RDV40) (@iceman1001)
+ - Change `lf t55xx detectconfig` - now optional to persist settings to flashmem (RDV40) (@iceman1001)
+ - Change `hf mf csave` - now saves both EML/BIN formats (@iceman1001)
+ - Change `hf mf esave` - now saves both EML/BIN formats (@iceman1001)
+ - Fix `compiler warning on macos and gcc7.1 or higher`  (@TomHarkness)
+ - Fix `crash on Bionic libc if CloseProxmark is called twice` (@micolous)
+ - Change `lf hid` - got an  updated to Kastle format (@xilni)
+ - Added `lf t55xx deviceconfig` - enables custom t55xx timing settings. (RDV40) (@iceman1001)
+ - Chg adaptations for FPC communications (work in progress)  (@iceman1001)
+ - Fix `stand-alone Colin` - remake to benefit from flashmem for persistence. (@cjbrigato)
+ - Fix `LEGIC SIM` - remake of legic sim (@drandreas)
+ - Changed `proxmark3 client threading`  - remake from official repo (@micolous)
+ - Add `rem` - new command that adds a line to the log file (@didierStevens)
+ - Fix `EM410xdemod  empty tag id in lfops.c` (@Defensor7)
+ - Fix `usb device descriptor` - some android phones will enumerate better when iSerialnumber is not a multiple of 8 (@micolous, @megabug)
+ - Fix `StandaloneMode LF` -  when collecting signal, justNoise detection is needed (@didierStevens, @Megabug)
+ - Fix `StandAloneMode Colin` - mifare1ksim called with right params (@cjbrigato)
+ - Improved `install.sh` to install dependencies for Ubuntu 18.04 and using max number of processors during compilation (@joanbono)
+ - Modified `install.sh` script to work in macOS and Linux + added the `update.sh` and `proxmark3.sh` from joanbono (@TomHarkness)
+ - Fix `hf emv` - some cards need to have Le=0x00, some do not need to have (@merlokk)
+ - Fix `hf legic`  enhancement of rx / tx in legic commands (@drandreas)
+ - Fix `data buffclear` - now frees bigbuff also (@iceman1001)
  - Fix GET_TICKS  and signess while shifting (@drandreas)
- - Added 'hf 14b dump' - now dumps to file (bin & eml)  (@iceman)
+ - Added `hf 14b dump` - now dumps to file (bin & eml)  (@iceman1001)
  - Fix fixed xcorrelation for strong signal (@drandreas)
- - Fix 'hf mf chk' - keytype was reversed (@TomHarkness)
- - Added strange vid/pid found in wild.  Could be pm3 easy clones. (@iceman)
- - Fix 'make udev'  -  udev filename could be in conflict,  renamed.  (@blshkv)
- - Fix 'lf t55xx config'  - wrong sized array disabled FSK1a/FSK2a as options (@grauerfuchs)
- - Added more default keys  (@j8048188) (@iceman)
- - Added 'sc list/info/raw/reader/upgrade' - (RDV40) smart card module functionality  (@iceman)
- - Fix 'download eml buffer' (@drandreas)
- - Changed 'exclusion of floatingpoint lib' (@pwpiwi)
- - Changed 'lua scripts bit32 calls' (@iceman)
- - Changed 'hw version'  (@pwpiwi),   adapted to iceman fork ( @iceman)  
- - Added 'amiibo functionality'  (@jamchamb),   adapted to iceman fork ( @iceman)  
- - Fix 'hf legic'  (RDV40) adaptations to FPGA HF enhanched reading distance (@iceman)   Thanks to @drandreas!
- - Added 'script run mifare_acces' - script to decode Mifare classic accessbits (@Neuromancer)
- - Added 'mem load/save/wipe' - commands to upload / download to new RDV40 onboard flashmemory (@iceman)
- - Added 'script run mifareplus" - script to communicate with a mifare plus tag (@dceliano)
- - Added FlashMemory functionality  (RDV40)  (Thanks @willok)
- - Fix 'hf mfu dump' - partial reads lead to corrupt data (Thanks @elafargue for pointing it out)
- - Changed 'hf mfu dump / read'  - now retries five times. (@jamchamb)
+ - Fix `hf mf chk` - keytype was reversed (@TomHarkness)
+ - Added strange vid/pid found in wild.  Could be pm3 easy clones. (@iceman1001)
+ - Fix `make udev`  -  udev filename could be in conflict,  renamed.  (@blshkv)
+ - Fix `lf t55xx config`  - wrong sized array disabled FSK1a/FSK2a as options (@grauerfuchs)
+ - Added more default keys  (@j8048188) (@iceman1001)
+ - Added `sc list/info/raw/reader/upgrade` - (RDV40) smart card module functionality  (@iceman1001)
+ - Fix `download eml buffer` (@drandreas)
+ - Changed `exclusion of floatingpoint lib` (@pwpiwi)
+ - Changed `lua scripts bit32 calls` (@iceman1001)
+ - Changed `hw version`  (@pwpiwi),   adapted to iceman fork (@iceman1001)
+ - Added `amiibo functionality`  (@jamchamb),   adapted to iceman fork (@iceman1001)
+ - Fix `hf legic`  (RDV40) adaptations to FPGA HF enhanched reading distance (@iceman1001)   Thanks to @drandreas!
+ - Added `script run mifare_acces` - script to decode Mifare classic accessbits (@Neuromancer)
+ - Added `mem load/save/wipe` - commands to upload / download to new RDV40 onboard flashmemory (@iceman1001)
+ - Added `script run mifareplus` - script to communicate with a mifare plus tag (@dceliano)
+ - Added FlashMemory functionality  (RDV40)  (@willok)
+ - Fix `hf mfu dump` - partial reads lead to corrupt data (Thanks @elafargue for pointing it out)
+ - Changed `hf mfu dump / read`  - now retries five times. (@jamchamb)
  - Added `hf list mf` - deciphers crypto1 stream and works with first authentication and weak nested authentications (@Merlok)
  - Adjusted `lf cmdread` to respond to client when complete and the client will then automatically call `data samples` (@marshmellow42)
  - Added a bitbang mode to `lf cmdread` if delay is 0 the cmd bits turn off and on the antenna with 0 and 1 respectively (@marshmellow42)
  - dump / restore now uses custom filenames (@brianpow)
- - Removed 'hf mf sniff' ,  (@iceman),  use HF 14A SNIFF instead
- - Added 'hf iclass lookup' (@iceman)
- - Added 'hf iclass chk' (@iceman)
- - Fixed ADC mux all closed push-pull state (@iceman)
- - Fix 'hf mf darkside' - speed fixes (@pwpiwi)
- - Fix 'hw tune'  -  now compensates for 3% error in output,  also measure full 140v using ADC channel 5 and 7.  (@iceman)
+ - Removed `hf mf sniff` ,  (@iceman1001),  use HF 14A SNIFF instead
+ - Added `hf iclass lookup` (@iceman1001)
+ - Added `hf iclass chk` (@iceman1001)
+ - Fixed ADC mux all closed push-pull state (@iceman1001)
+ - Fix `hf mf darkside` - speed fixes (@pwpiwi)
+ - Fix `hw tune`  -  now compensates for 3% error in output,  also measure full 140v using ADC channel 5 and 7.  (@iceman1001)
  - Updated loclass gpl license (@holiman)
- - Fix Antenna on after changed FPGA Mode. (@iceman)
- - Added 'hf mf nack' - Mifare NACK bug detection (@iceman) (@doegox)
- - Fix 'hf mf mifare' - zero parity works, no more double runs for normal darkside (@iceman)
- - Added 'hf mf fchk' - the fastest check keys implementation tothisday (@iceman)
- - Fix 'hf iclass' - more stable demod (@iceman)
- - Added 'hf iclass chk'  - check keys from default_iclass_keys.dic file (@iceman)
- - Fix 'hf 15 dump' - no more crc faults (@iceman)
- - Fix 'hf 15 read' - no more crc faults (@iceman)
- - Fix 'hf 15 readmulti' - no more crc faults (@iceman)
+ - Fix Antenna on after changed FPGA Mode. (@iceman1001)
+ - Added `hf mf nack` - Mifare NACK bug detection (@iceman1001) (@doegox)
+ - Fix `hf mf mifare` - zero parity works, no more double runs for normal darkside (@iceman1001)
+ - Added `hf mf fchk` - the fastest check keys implementation tothisday (@iceman1001)
+ - Fix `hf iclass` - more stable demod (@iceman1001)
+ - Added `hf iclass chk`  - check keys from default_iclass_keys.dic file (@iceman1001)
+ - Fix `hf 15 dump` - no more crc faults (@iceman1001)
+ - Fix `hf 15 read` - no more crc faults (@iceman1001)
+ - Fix `hf 15 readmulti` - no more crc faults (@iceman1001)
  - Changed proxmark command line parameter `flush` to `-f` or `-flush` (@merlokk)
  - Added to proxmark command line parameters `w` - wait 20s for serial port (@merlokk)
  - Added to proxmark command line parameters `c` and `l` - execute command and lua script from command line (@merlokk)
- - Added to proxmark ability to execute commands from stdin (pipe) ((@merlokk)
+ - Added to proxmark ability to execute commands from stdin (pipe) (@merlokk)
  - Added new standalone mode "HF Mifare ultra fast sniff/sim/clone - aka VIGIKPWN"  (@cjbrigato)
  - Added to `hf 14a apdu` - exchange apdu via iso1443-4 (@merlokk)
  - Added to `hf 14a apdu` - apdu and tlv results parser (@merlokk)
- - Added 'hf emv' commands  (@merlokk)
+ - Added `hf emv` commands  (@merlokk)
  - lots of bug fixes (many many)
- - Changed hf mfp security. Now it works in all the modes. (drHatson)
+ - Changed hf mfp security. Now it works in all the modes. (@drHatson)
  - Added `hf fido` commands that work with FIDO U2F authenticators (@merlokk)
  - Added mbedtls instead of old polarssl (@merlokk)
  - Added jansson (@merlokk)
- - Added `hf emv scan` - save card's data to json file (@merlokk)
+ - Added `hf emv scan` - save card data to json file (@merlokk)
  - Added `hf emv` `gpo`, `readrec`, `genac`, `challenge`, `intauth` - separate commands from `hf emc exec` (@merlokk)
  - Added  `hf fido` `assert` and `make` commands from fido2 protocol (authenticatorMakeCredential and authenticatorGetAssertion) (@merlokk)
  - Added trailer block decoding to `hf mf rdbl` and `hf mf cgetbl` (@merlokk)
  - Added `hf mf mad` and `hf mfp mad` MAD decode, check and print commands (@merlokk)
- 
+ - Added T55x7 downlink mode support r <mode> 0 Default, 1 Long Leading 0, 2 Leading 0, 3 1 of 4 and 4 (in some commands) try all.
+
 ### Fixed
-- Changed driver file proxmark3.inf to support both old and new Product/Vendor IDs (piwi)
-- Changed start sequence in Qt mode (fix: short commands hangs main Qt thread) (Merlok)
+ - Changed driver file proxmark3.inf to support both old and new Product/Vendor IDs (@pwpiwi)
+ - Changed start sequence in Qt mode (fix: short commands hangs main Qt thread) (@merlokk)
 
 ## [ice.3.1.0][2017-09-26]
-  - proxmark3 client can reconnect to device without restart (iceman)
-  - lots of bug fixes (many many)
-  - trace/securakey-64169.pm3 - trace of a scecurakey (atyppo)
-  - 'hf mf decrypt' - got some longer input and helptext parameter (iceman)  
-  - Updated the Reveng 1.51 sourcecode to 1.52 from Reveng project homepage (iceman)
-  - 'hf 14a read' - disconnects when failing to read tag (iceman)
-  - 'hf mf csave' - renamed parameter 'i' to 'o' as in output (iceman)
+ - proxmark3 client can reconnect to device without restart (@iceman1001)
+ - lots of bug fixes (many many)
+ - trace/securakey-64169.pm3 - trace of a scecurakey (@atyppo)
+ - `hf mf decrypt` - got some longer input and helptext parameter (@iceman1001)
+ - Updated the Reveng 1.51 sourcecode to 1.52 from Reveng project homepage (@iceman1001)
+ - `hf 14a read` - disconnects when failing to read tag (@iceman1001)
+ - `hf mf csave` - renamed parameter `i` to `o` as in output (@iceman1001)
 
 ## [3.0.0][2017-08-29]
   Notes on this release
@@ -293,267 +324,265 @@ This project uses the changelog in accordance with [keepchangelog](http://keepac
   There is a lot of changes,  command breaking changes, which is the cause for the JUMP in version number.
   It is set to v3.0.0 to show that it is on par with PM3 Offical v3 release.
 
-  - Updated 'mkversion.pl' to write a date based on file when repo is downloaded as a zip file from github (iceman)
-  - Update 'readme.md' to fit GitHubs markup(joanbono)
-  - Added 'script run ul_uid', try to change UID on a magic UL-card. (iceman)
-  - Fixed 'hf snoop' bug,  of wrong bool value (ikarus23)
-  - Fixed fullimage.s19, wrong offsets (doegox)
-  - Updated '77-mm-usb-device-blacklist.rules' for the pid/vid (iceman)
-  - 'hf 14a sim' now follows Mifare UL-EV1 protocol better (iceman)
-  - Updated 'fpga_hf.bit' file (piwi)
-  - Added more card detections to 'hf mfu info' (iceman)
-  - Fixed 'hf mfu restore/dump' to use the right struct values in special data in dumpfile. (iceman)
-  - Added 'hf mfu restore r' new parameter to use the new pwd for all further auths needed when executing (iceman)
-  - Added 'default_keys_dic2lua.awk' script to generate default_keys file in client/lualibs (iceman)
-  - Fixes to lots of lua scripts, among others
-    - 'mifare_autopwn', now uses PRNG detection (iceman)
-	- 'mfkeys', fixed bug which only tested the first key (iceman)
-	- 'dumptoemul', removed last newline (iceman)
-	- ...
-  - Added USB/SERIAL communication enhancements (micolous)
-  - Change 'hf 14a cuids', to be interrupted with keyboard press (iceman)
-  - Change debugstatements for LF to show which function more unified (iceman)
-  - Added 'script run calc_di' , to calculate some Mifare keys (iceman)
-  - Fixed iclass commands never shut down antenna afterwards (iceman)
-  - Change 512kb detection when flashing (iceman)
-  - Fixed compilation GCC4.9 or higher detection (winguru)
-  - Fixed compiler warnings in Ubuntu 17.04 (iceman)
-  - Ripped out 'standalone' code into separete folder to be continued. (iceman)
-  - 'hf mf nested', added key validation to entered key (merlokk)
-  - 'hf mf hardnested', added key validation to enterd key (iceman)
-  - Change a lot of help texts (iceman)
-  - Fixed 'hf mf chk' - keyblock bug, limited keys to 256. (iceman)
-  - Change 'hf mf dump' retries three times now before giving up (marshmellow)
-  - Fixed 'mfu authentication', with pack-len error (iceman)
-  - 'Script list', change sortorder to alphabetic order (iceman)
-  - Change 'hf mfu gen' to read taguid (iceman)
-  - Change 'hf mfu pwdgen' to read taguid (iceman)
-  - Added 'hf mf setmod' sets Mifare Classic EV1 load modulation strength to card (angelsl)
-  - Added 'hf 14a read'  Mifare PRNG detection based on @doegox LIBNFC impl (iceman)
-  - Added 'hf mf nonces', collects Mifare Classic nonces for analysing of PRNG (iceman)		
-  - Added new CSNS in 'hf iclass sim 2' attack (iceman)						
-  - Added more default keys (iceman)						
-  - Added analyse nuid, enable creation of Mifare NUID (iceman) 																					  
-  - Updated the Reveng 1.44 sourcecode to 1.51 from Reveng project homepage (iceman)
-  - script run formatMifare - got an option to execute the generate strings (iceman)
-  - Fix 'hf mf cgetsc' (iceman)
-  - Fix 'hf legic info' (iceman)
-  - Change version output (iceman)
-  - Added PAC/Stanley detection to lf search (marshmellow)
-  - Added lf pac demod and lf pac read - extracts the raw blocks from a PAC/Stanley tag (marshmellow)
-  - Added hf mf csave commands compatibity for 4k (Fl0-0)
-  - Added data fsktonrz, a fsk cleaning/demodulating routine for weak fsk signal. Note: follow this up with a `data rawdemod nr` to finish demoding your signal. (marshmellow)
-  - Added lf em 410xbrute, LF EM410x reader bruteforce attack by simulating UIDs from a file (Fl0-0)
-  - Compiles on OS X
-  - Compiles with gcc 4.9
-  - Compiles for non-Intel CPUs
-  - Added lf hitag write 24, the command writes a block to hitag2 tags in crypto mode (henjo)
-  - Added the improved 'hf mf hardnested', an attack working for hardened Mifare cards (EV1, Mifare Plus SL1)
-  - Added experimental testmode write option for t55xx (danger) (marshmellow)
-  - Added t55xx p1detect to `lf search` chip detections (marshmellow)
-  - Added lf t55xx p1detect, detect page 1 of a t55xx tag based on E015 mfg code (marshmellow)
-  - Added lf noralsy demod, read, clone, sim commands (iceman)
-  - Added lf jablotron demod, read, clone, sim commands (iceman)
-  - Added lf nexwatch read   - reads a nexwatch tag from the antenna
-  - Added lf paradox read    - reads a paradox tag from the antenna
-  - Added lf fdx sim (iceman)
-  - Added lf fdx clone       - clones an fdx-b animal tag to t55x7 or q5 (iceman)
-  - Added lf fdx read        - reads a fdx-b tag from the antenna (iceman)
-  - Added lf gproxii read    - reads a gproxii tag from the antenna (marshmellow)
-  - Added lf indala read     - reads an indala tag from the antenna (marshmellow)
-  - Added lf visa2000 demod, read, clone, sim commands (iceman)
+ - Updated `mkversion.pl` to write a date based on file when repo is downloaded as a zip file from github (@iceman1001)
+ - Update `readme.md` to fit GitHubs markup(@joanbono)
+ - Added `script run ul_uid`, try to change UID on a magic UL-card. (@iceman1001)
+ - Fixed `hf snoop` bug,  of wrong bool value (@ikarus23)
+ - Fixed fullimage.s19, wrong offsets (@doegox)
+ - Updated `77-mm-usb-device-blacklist.rules` for the pid/vid (@iceman1001)
+ - `hf 14a sim` now follows Mifare UL-EV1 protocol better (@iceman1001)
+ - Updated `fpga_hf.bit` file (@pwpiwi)
+ - Added more card detections to `hf mfu info` (@iceman1001)
+ - Fixed `hf mfu restore/dump` to use the right struct values in special data in dumpfile. (@iceman1001)
+ - Added `hf mfu restore r` new parameter to use the new pwd for all further auths needed when executing (@iceman1001)
+ - Added `default_keys_dic2lua.awk` script to generate default_keys file in client/lualibs (@iceman1001)
+ - Fixes to lots of lua scripts, among others
+    - `mifare_autopwn`, now uses PRNG detection (@iceman1001)
+    - `mfkeys`, fixed bug which only tested the first key (@iceman1001)
+    - `dumptoemul`, removed last newline (@iceman1001)
+    - ...
+ - Added USB/SERIAL communication enhancements (@micolous)
+ - Change `hf 14a cuids`, to be interrupted with keyboard press (@iceman1001)
+ - Change debugstatements for LF to show which function more unified (@iceman1001)
+ - Added `script run calc_di` , to calculate some Mifare keys (@iceman1001)
+ - Fixed iclass commands never shut down antenna afterwards (@iceman1001)
+ - Change 512kb detection when flashing (@iceman1001)
+ - Fixed compilation GCC4.9 or higher detection (@winguru)
+ - Fixed compiler warnings in Ubuntu 17.04 (@iceman1001)
+ - Ripped out `standalone` code into separete folder to be continued. (@iceman1001)
+ - `hf mf nested`, added key validation to entered key (@merlokk)
+ - `hf mf hardnested`, added key validation to enterd key (@iceman1001)
+ - Change a lot of help texts (@iceman1001)
+ - Fixed `hf mf chk` - keyblock bug, limited keys to 256. (@iceman1001)
+ - Change `hf mf dump` retries three times now before giving up (@marshmellow42)
+ - Fixed `mfu authentication`, with pack-len error (@iceman1001)
+ - `Script list`, change sortorder to alphabetic order (@iceman1001)
+ - Change `hf mfu gen` to read taguid (@iceman1001)
+ - Change `hf mfu pwdgen` to read taguid (@iceman1001)
+ - Added `hf mf setmod` sets Mifare Classic EV1 load modulation strength to card (@angelsl)
+ - Added `hf 14a read`  Mifare PRNG detection based on @doegox LIBNFC impl (@iceman1001)
+ - Added `hf mf nonces`, collects Mifare Classic nonces for analysing of PRNG (@iceman1001)
+ - Added new CSNS in `hf iclass sim 2` attack (@iceman1001)
+ - Added more default keys (@iceman1001)
+ - Added analyse nuid, enable creation of Mifare NUID (@iceman1001)
+ - Updated the Reveng 1.44 sourcecode to 1.51 from Reveng project homepage (@iceman1001)
+ - script run formatMifare - got an option to execute the generate strings (@iceman1001)
+ - Fix `hf mf cgetsc` (@iceman1001)
+ - Fix `hf legic info` (@iceman1001)
+ - Change version output (@iceman1001)
+ - Added PAC/Stanley detection to lf search (@marshmellow42)
+ - Added lf pac demod and lf pac read - extracts the raw blocks from a PAC/Stanley tag (@marshmellow42)
+ - Added hf mf csave commands compatibity for 4k (@Fl0-0)
+ - Added data fsktonrz, a fsk cleaning/demodulating routine for weak fsk signal. Note: follow this up with a `data rawdemod nr` to finish demoding your signal. (@marshmellow42)
+ - Added lf em 410xbrute, LF EM410x reader bruteforce attack by simulating UIDs from a file (@Fl0-0)
+ - Compiles on OS X
+ - Compiles with gcc 4.9
+ - Compiles for non-Intel CPUs
+ - Added lf hitag write 24, the command writes a block to hitag2 tags in crypto mode (@henjo)
+ - Added the improved `hf mf hardnested`, an attack working for hardened Mifare cards (EV1, Mifare Plus SL1)
+ - Added experimental testmode write option for t55xx (danger) (@marshmellow42)
+ - Added t55xx p1detect to `lf search` chip detections (@marshmellow42)
+ - Added lf t55xx p1detect, detect page 1 of a t55xx tag based on E015 mfg code (@marshmellow42)
+ - Added lf noralsy demod, read, clone, sim commands (@iceman1001)
+ - Added lf jablotron demod, read, clone, sim commands (@iceman1001)
+ - Added lf nexwatch read   - reads a nexwatch tag from the antenna
+ - Added lf paradox read    - reads a paradox tag from the antenna
+ - Added lf fdx sim (@iceman1001)
+ - Added lf fdx clone       - clones an fdx-b animal tag to t55x7 or q5 (@iceman1001)
+ - Added lf fdx read        - reads a fdx-b tag from the antenna (@iceman1001)
+ - Added lf gproxii read    - reads a gproxii tag from the antenna (@marshmellow42)
+ - Added lf indala read     - reads an indala tag from the antenna (@marshmellow42)
+ - Added lf visa2000 demod, read, clone, sim commands (@iceman1001)
 
 ## [1.7.0 iceman fork] [2017-03-07]
-  - hf mf dump - added retry loops to try each read attempt up to 3 times.  makes getting a complete dump easier with many antennas. (marshmellow)
-
-  - Added markers in the graph around found Sequence Terminator after askmandemod. (marshmellow)
-  - Added data mtrim <start> <stop> command to trim out samples between start and stop. (marshmellow)
-  - Added data setgraphmarkers <orange> <blue> command to set two extra markers on the graph (marshmellow)
-  - added json support in lua (vitorio)
-  - added a buspirate settings file for at91sam7s512 (adamlaurie)
-  - `lf read` timeouts is now depended on what threshold level you set in `lf config`  (marshmellow)
-  - `hf mf sim` fixed a bug which made sim fail auths. (iceman)
-  - `hf 14a read` added magic tag generation 1a and 1b detection  (iceman)
-  - correctly using stdtypes.h printf and scanf format string macros (PRIx64 et al) (pwpivi)
-  - fix linker warning re missing entry point when linking fullimage.elf (pwpivi)
-  - small changes to lf psk and fsk demods to improve results when the trace begins with noise or the chip isn't broadcasting yet (marshmellow)
-  - NOTE CHANGED ALL `lf em4x em*` cmds to simpler `lf em ` - example: `lf em4x em410xdemod` is now `lf em 410xdemod`
-  - Renamed and rebuilt `lf em readword` && readwordpwd to `lf em 4x05read` - it now demods and outputs the read block (marshmellow/iceman)
-  - Renamed and rebuilt `lf em writeword` && writewordpwd to `lf em 4x05write` - it now also reads validation output from the tag (marshmellow/iceman)
-  - Fixed bug in lf sim and continuous demods not turning off antenna when finished
-  - Added lua script path fixes (pwpivi)
-  - `lf search` - Added EM4x05/EM4x69 chip detection (marshmellow)
-  - Added lf em 4x05dump command to read and output all the blocks of the chip (marshmellow)
-  - Added lf em 4x05info command to read and display information about the chip (marshmellow)
-  - `lf em4x em4x50***` refactoring of em4x50 commands. (iceman)
+ - hf mf dump - added retry loops to try each read attempt up to 3 times.  makes getting a complete dump easier with many antennas. (@marshmellow42)
+ - Added markers in the graph around found Sequence Terminator after askmandemod. (@marshmellow42)
+ - Added data mtrim <start> <stop> command to trim out samples between start and stop. (@marshmellow42)
+ - Added data setgraphmarkers <orange> <blue> command to set two extra markers on the graph (@marshmellow42)
+ - added json support in lua (@vitorio)
+ - added a buspirate settings file for at91sam7s512 (@adamlaurie)
+ - `lf read` timeouts is now depended on what threshold level you set in `lf config`  (@marshmellow42)
+ - `hf mf sim` fixed a bug which made sim fail auths. (@iceman1001)
+ - `hf 14a read` added magic tag generation 1a and 1b detection  (@iceman1001)
+ - correctly using stdtypes.h printf and scanf format string macros (PRIx64 et al) (@pwpiwi)
+ - fix linker warning re missing entry point when linking fullimage.elf (@pwpiwi)
+ - small changes to lf psk and fsk demods to improve results when the trace begins with noise or the chip is not broadcasting yet (@marshmellow42)
+ - NOTE CHANGED ALL `lf em4x em*` cmds to simpler `lf em ` - example: `lf em4x em410xdemod` is now `lf em 410xdemod`
+ - Renamed and rebuilt `lf em readword` && readwordpwd to `lf em 4x05read` - it now demods and outputs the read block (@marshmellow42/@iceman1001)
+ - Renamed and rebuilt `lf em writeword` && writewordpwd to `lf em 4x05write` - it now also reads validation output from the tag (@marshmellow42/@iceman1001)
+ - Fixed bug in lf sim and continuous demods not turning off antenna when finished
+ - Added lua script path fixes (@pwpiwi)
+ - `lf search` - Added EM4x05/EM4x69 chip detection (@marshmellow42)
+ - Added lf em 4x05dump command to read and output all the blocks of the chip (@marshmellow42)
+ - Added lf em 4x05info command to read and display information about the chip (@marshmellow42)
+ - `lf em4x em4x50***` refactoring of em4x50 commands. (@iceman1001)
 
 ## [1.6.9 iceman fork] [2017-02-06]
-  - Serial speedup,  if possible 408600baud otherwise default to 115200baud (iceman)
-  - `hf emv` - Added Peter Fillmore's EMV branch now compiles on iceman fork.  See seperate issue. (iceman)
-  - `hf 14a reader` - Aztek detection. (iceman)
-  - `standalone mode` - added more detection of tags and refactored (iceman)
-  - `script run ufodump` - dumps an Aztek tag. (iceman)
-  - `script run hard_autopwn` - runs hardnested attack against all sectors on tag (iceman)
-  - Added lf cotag read, and added it to lf search (iceman)
-  - Added hitag2 read UID only and added that to lf search (marshmellow)
-  - `lf search` -  check for if signal is only noice (marshmellow)
-  - `hf 14a reader` - fixed a bug when card has sak 0x00 but still is not UL/NTAG etc. (iceman)
-  - `hf mf sim` / `hf 14a sim` - use random nonce. (micolous)
-  - `hw tune` - only prints out if voltage is detected from antenna. (iceman)
-  - `hf iclass decrypt` - only decrypt Application1 (iceman)
-  - `lf t55xx detect` - when finding multiple possible config blocks, see if a known configblock exists and select. (iceman)
+ - Serial speedup,  if possible 408600baud otherwise default to 115200baud (@iceman1001)
+ - `hf emv` - Added Peter Fillmore EMV branch now compiles on iceman fork.  See seperate issue. (@iceman1001)
+ - `hf 14a reader` - Aztek detection. (@iceman1001)
+ - `standalone mode` - added more detection of tags and refactored (@iceman1001)
+ - `script run ufodump` - dumps an Aztek tag. (@iceman1001)
+ - `script run hard_autopwn` - runs hardnested attack against all sectors on tag (@iceman1001)
+ - Added lf cotag read, and added it to lf search (@iceman1001)
+ - Added hitag2 read UID only and added that to lf search (@marshmellow42)
+ - `lf search` -  check for if signal is only noice (@marshmellow42)
+ - `hf 14a reader` - fixed a bug when card has sak 0x00 but still is not UL/NTAG etc. (@iceman1001)
+ - `hf mf sim` / `hf 14a sim` - use random nonce. (@micolous)
+ - `hw tune` - only prints out if voltage is detected from antenna. (@iceman1001)
+ - `hf iclass decrypt` - only decrypt Application1 (@iceman1001)
+ - `lf t55xx detect` - when finding multiple possible config blocks, see if a known configblock exists and select. (@iceman1001)
 
 ## [1.6.7 iceman fork] [2017-01-05]
-  - `lf animal` - FDX-B animal commands (iceman)
-  -  Fixed bugs in `lf sim` and other lf continuous demods not turning off antenna when finished (marshmellow)
-  -  `hf iclass write` - fixed bugs, added crc. (?)
-  -  `hf iclass dump` - changed layout in dump (iceman)
-  -  Changed - debug statements are more clear (iceman)
-  -  `lf search` - fixed the silent option when acquire data. (iceman)
-  -  `lf search` - added presco, visa2000, noralsy detection (iceman)
-  -  `lf precso` - fixed some bitsgeneration in precso bits (iceman)
-  -  Added `lf noralsy` -  adds demod/clone/sime of Noralsy LF tags. (iceman)
-  -  Added `lf visa2000` - adds demod/clone/sim of Visa2000 lF tags. (iceman)
-  -  Added `hf mf key_brute` - adds J-Runs 2nd phase bruteforce ref: https://github.com/J-Run/mf_key_brute   (iceman)
-  -  Added `lf jablotron` - adds demod/clone/sim of Jablotron LF tags. (iceman)
-  -  Added `lf t55xx recoverpw` - adds a new password recovery using bitflips and partial flips if password write went bad. (alexgrin)
-  - `hf legic` - added improved legic data mapping. (jason)
-  - `hf mf mifare` - added possibility to target key A|B (douniwan5788)
-  -  Added `analyse lcr` - added a new main command group,  to help analysing bytes & bits & nibbles. (iceman)
-  -  Added `lf nedap` - added identification of a NEDAP tag. (iceman)
-  - `lf viking clone` - fixed a bug. (iceman)
-  -  Added bitsliced bruteforce solver in `hf mf hardnested` (Aczid)
-  - `hf mf chk` speedup (iceman)
-  - `hf 14a/mf sim x` attack mode,  now uses also moebius version of mfkey32 to try finding the key. (iceman)
-  - `hf 14a sim` Added emulation of Mifare cards with 10byte UID length. (iceman)
-  - `hf mf sim` Added emulation of Mifare cards with 10byte UID length. (iceman)
-  -  Added `lf guard clone/sim` (iceman)
-  -  Added `lf pyramd clone/sim` (iceman)
-  - trying to fix `hf 14b` command to be able to read CALYPSO card.	 (iceman)
-  - `hf legic load`, it now loads faster and a casting bug is gone. (iceman)
-  -  Added `hf legic calccrc8` added a method to calculate the legic crc-8 value (iceman)
-  - `hf legic decode` fixed the output overflow bugs, better printing (iceman)
-  - Coverity Scan fixes a lot of resource leaks, etc (iceman)
-  -  Added `lf presco *` commands started (iceman)
-  -  Added `lf hid wiegand` added a method to calculate WIEGAND in different formats, (iceman)
-  - `hf mf chkkeys` better printing, same table output as nested, faster execution and added Adam Lauries "try to read Key B if Key A is found" (iceman)
-  - `hf mf nested` better printing and added Adam Lauries "try to read Key B if Key A is found" (iceman)
-  - `hf mf mifare` fixing the zero parity path, which doesn't got called. (iceman)
-  - Updated the @blapost's Crapto1 implementation to v3.3 (blapost)
-  - `hf mf c*` updated the calling structure and refactored of the chinese magic commands (iceman, marshmellow)
-  - Started to add Peter Fillmore's  EMV fork into Iceman fork. ref: https://github.com/peterfillmore/proxmark3  (peter fillmore,  iceman)
-  - Added Travis-CI automatic build integration with GitHub fork. (iceman)
-  - Updated the Reveng 1.30 sourcecode to 1.31 from Reveng project homepage (iceman)
-  - Updated the Reveng 1.31 sourcecode to 1.40 from Reveng project homepage (iceman)
-
-  - Added possibility to write direct to a Legic Prime Tag (MIM256/1024) without using values from the 'BigBuffer' -> 'hf legic writeRaw <addr> <value>' (icsom)
-  - Added possibility to decrease DCF values at address 0x05 & 0x06 on a Legic Prime Tag
-		DCF-value will be pulled from the BigBuffer (address 0x05 & 0x06) so you have to
-		load the data into the BigBuffer before with 'hf legic load <path/to/legic.dump>' & then
-		write the DCF-Values (both at once) with 'hf legic write 0x05 0x02'  (icsom)
-  - Added script `legic.lua` for display and edit Data of Legic-Prime Tags (icsom)
-  - Added the experimental HITAG_S support (spenneb)
-  - Added topaz detection to `hf search` (iceman)
-  - Fixed the silent mode for 14b to be used inside `hf search` (iceman)
+ - `lf animal` - FDX-B animal commands (@iceman1001)
+ - Fixed bugs in `lf sim` and other lf continuous demods not turning off antenna when finished (@marshmellow42)
+ - `hf iclass write` - fixed bugs, added crc. (?)
+ - `hf iclass dump` - changed layout in dump (@iceman1001)
+ - Changed - debug statements are more clear (@iceman1001)
+ - `lf search` - fixed the silent option when acquire data. (@iceman1001)
+ - `lf search` - added presco, visa2000, noralsy detection (@iceman1001)
+ - `lf precso` - fixed some bitsgeneration in precso bits (@iceman1001)
+ - Added `lf noralsy` -  adds demod/clone/sime of Noralsy LF tags. (@iceman1001)
+ - Added `lf visa2000` - adds demod/clone/sim of Visa2000 lF tags. (@iceman1001)
+ - Added `hf mf key_brute` - adds J-Runs 2nd phase bruteforce ref: https://github.com/J-Run/mf_key_brute   (@iceman1001)
+ - Added `lf jablotron` - adds demod/clone/sim of Jablotron LF tags. (@iceman1001)
+ - Added `lf t55xx recoverpw` - adds a new password recovery using bitflips and partial flips if password write went bad. (@alexgrin)
+ - `hf legic` - added improved legic data mapping. (jason)
+ - `hf mf mifare` - added possibility to target key A|B (@douniwan5788)
+ - Added `analyse lcr` - added a new main command group,  to help analysing bytes & bits & nibbles. (@iceman1001)
+ - Added `lf nedap` - added identification of a NEDAP tag. (@iceman1001)
+ - `lf viking clone` - fixed a bug. (@iceman1001)
+ - Added bitsliced bruteforce solver in `hf mf hardnested` (@Aczid)
+ - `hf mf chk` speedup (@iceman1001)
+ - `hf 14a/mf sim x` attack mode,  now uses also moebius version of mfkey32 to try finding the key. (@iceman1001)
+ - `hf 14a sim` Added emulation of Mifare cards with 10byte UID length. (@iceman1001)
+ - `hf mf sim` Added emulation of Mifare cards with 10byte UID length. (@iceman1001)
+ - Added `lf guard clone/sim` (@iceman1001)
+ - Added `lf pyramd clone/sim` (@iceman1001)
+ - trying to fix `hf 14b` command to be able to read CALYPSO card. (@iceman1001)
+ - `hf legic load`, it now loads faster and a casting bug is gone. (@iceman1001)
+ - Added `hf legic calccrc8` added a method to calculate the legic crc-8 value (@iceman1001)
+ - `hf legic decode` fixed the output overflow bugs, better printing (@iceman1001)
+ - Coverity Scan fixes a lot of resource leaks, etc (@iceman1001)
+ - Added `lf presco *` commands started (@iceman1001)
+ - Added `lf hid wiegand` added a method to calculate WIEGAND in different formats, (@iceman1001)
+ - `hf mf chkkeys` better printing, same table output as nested, faster execution and added Adam Lauries "try to read Key B if Key A is found" (@iceman1001)
+ - `hf mf nested` better printing and added Adam Lauries "try to read Key B if Key A is found" (@iceman1001)
+ - `hf mf mifare` fixing the zero parity path, which did not get called. (@iceman1001)
+ - Updated the @blapost Crapto1 implementation to v3.3 (blapost)
+ - `hf mf c*` updated the calling structure and refactored of the chinese magic commands (@iceman1001, @marshmellow42)
+ - Started to add Peter Fillmore  EMV fork into Iceman fork. ref: https://github.com/peterfillmore/proxmark3  (@peterfillmore,  @iceman1001)
+ - Added Travis-CI automatic build integration with GitHub fork. (@iceman1001)
+ - Updated the Reveng 1.30 sourcecode to 1.31 from Reveng project homepage (@iceman1001)
+ - Updated the Reveng 1.31 sourcecode to 1.40 from Reveng project homepage (@iceman1001)
+ - Added possibility to write direct to a Legic Prime Tag (MIM256/1024) without using values from the `BigBuffer` -> `hf legic writeRaw <addr> <value>` (@icsom)
+ - Added possibility to decrease DCF values at address 0x05 & 0x06 on a Legic Prime Tag  
+   DCF-value will be pulled from the BigBuffer (address 0x05 & 0x06) so you have to 
+   load the data into the BigBuffer before with `hf legic load <path/to/legic.dump>` & then 
+   write the DCF-Values (both at once) with `hf legic write 0x05 0x02`  (@icsom)
+ - Added script `legic.lua` for display and edit Data of Legic-Prime Tags (@icsom)
+ - Added the experimental HITAG_S support (@spenneb)
+ - Added topaz detection to `hf search` (@iceman1001)
+ - Fixed the silent mode for 14b to be used inside `hf search` (@iceman1001)
 
 ### Added
-- Added a LF ASK Sequence Terminator detection option to the standard ask demod - and applied it to `lf search u`, `lf t55xx detect`, and `data rawdemod am s` (marshmellow)
-- `lf awid bruteforce <facilitycode>` - Simple bruteforce attack against a AWID reader.
-- `lf t55xx bruteforce <start password> <end password> [i <*.dic>]` - Simple bruteforce attack to find password - (iceman and others)
-- `lf viking clone`- clone viking tag to t55x7 or Q5 from 4byte hex ID input
-- `lf viking sim`  - sim full viking tag from 4byte hex ID input
-- `lf viking read` - read viking tag and output ID
-- `lf t55xx wipe`  - sets t55xx back to factory defaults
-- Added viking demod to `lf search` (marshmellow)
-- `data askvikingdemod` demod viking id tag from graphbuffer (marshmellow)
-- `lf t55xx resetread` added reset then read command - should allow determining start of stream transmissions (marshmellow)
-- `lf t55xx wakeup` added wake with password (AOR) to allow lf search or standard lf read after (iceman, marshmellow)
-- `hf mf eload u` added an ultralight/ntag option. (marshmellow)
-- `hf iclass managekeys` to save, load and manage iclass keys.  (adjusted most commands to accept a loaded key in memory) (marshmellow)
-- `hf iclass readblk` to select, authenticate, and read 1 block from an iclass card (marshmellow)
-- `hf iclass writeblk` to select, authenticate, and write 1 block to an iclass card (or picopass) (marshmellow + others)
-- `hf iclass clone` to take a saved dump file and clone selected blocks to a new tag (marshmellow + others)
-- `hf iclass calcnewkey` - to calculate the div_key change to change a key - (experimental) (marshmellow + others)
-- `hf iclass encryptblk` - to encrypt a data block hex to prep for writing that block (marshmellow)
-- ISO14443a stand-alone operation with ARM CFLAG="WITH_ISO14443a_StandAlone". This code can read & emulate two banks of 14a tag UIDs and write to "magic" cards  (Craig Young)
-- AWID26 command context added as 'lf awid' containing realtime demodulation as well as cloning/simulation based on tag numbers (Craig Young)
-- Added 'hw status'. This command makes the ARM print out some runtime information. (holiman)
-- Added 'hw ping'. This command just sends a usb packets and checks if the pm3 is responsive. Can be used to abort certain operations which supports abort over usb. (holiman)
-- Added `data hex2bin` and `data bin2hex` for command line conversion between binary and hexadecimal (holiman)
-- Added 'hf snoop'. This command take digitalized signal from FPGA and put in BigBuffer. (pwpiwi + enio)
-- Added Topaz (NFC type 1) protocol support ('hf topaz reader', 'hf list topaz', 'hf 14a raw -T', 'hf topaz snoop'). (piwi)
-- Added option c to 'hf list' (mark CRC bytes) (piwi)
+ - Added a LF ASK Sequence Terminator detection option to the standard ask demod - and applied it to `lf search u`, `lf t55xx detect`, and `data rawdemod am s` (@marshmellow42)
+ - `lf awid bruteforce <facilitycode>` - Simple bruteforce attack against a AWID reader.
+ - `lf t55xx bruteforce <start password> <end password> [i <*.dic>]` - Simple bruteforce attack to find password - (@iceman1001 and others)
+ - `lf viking clone`- clone viking tag to t55x7 or Q5 from 4byte hex ID input
+ - `lf viking sim`  - sim full viking tag from 4byte hex ID input
+ - `lf viking read` - read viking tag and output ID
+ - `lf t55xx wipe`  - sets t55xx back to factory defaults
+ - Added viking demod to `lf search` (@marshmellow42)
+ - `data askvikingdemod` demod viking id tag from graphbuffer (@marshmellow42)
+ - `lf t55xx resetread` added reset then read command - should allow determining start of stream transmissions (@marshmellow42)
+ - `lf t55xx wakeup` added wake with password (AOR) to allow lf search or standard lf read after (@iceman1001, @marshmellow42)
+ - `hf mf eload u` added an ultralight/ntag option. (@marshmellow42)
+ - `hf iclass managekeys` to save, load and manage iclass keys.  (adjusted most commands to accept a loaded key in memory) (@marshmellow42)
+ - `hf iclass readblk` to select, authenticate, and read 1 block from an iclass card (@marshmellow42)
+ - `hf iclass writeblk` to select, authenticate, and write 1 block to an iclass card (or picopass) (@marshmellow42 + others)
+ - `hf iclass clone` to take a saved dump file and clone selected blocks to a new tag (@marshmellow42 + others)
+ - `hf iclass calcnewkey` - to calculate the div_key change to change a key - (experimental) (@marshmellow42 + others)
+ - `hf iclass encryptblk` - to encrypt a data block hex to prep for writing that block (@marshmellow42)
+ - ISO14443a stand-alone operation with ARM CFLAG="WITH_ISO14443a_StandAlone". This code can read & emulate two banks of 14a tag UIDs and write to "magic" cards  (Craig Young)
+ - AWID26 command context added as `lf awid` containing realtime demodulation as well as cloning/simulation based on tag numbers (Craig Young)
+ - Added `hw status`. This command makes the ARM print out some runtime information. (@holiman)
+ - Added `hw ping`. This command just sends a usb packets and checks if the pm3 is responsive. Can be used to abort certain operations which supports abort over usb. (@holiman)
+ - Added `data hex2bin` and `data bin2hex` for command line conversion between binary and hexadecimal (@holiman)
+ - Added `hf snoop`. This command take digitalized signal from FPGA and put in BigBuffer. (@pwpiwi + enio)
+ - Added Topaz (NFC type 1) protocol support (`hf topaz reader`, `hf list topaz`, `hf 14a raw -T`, `hf topaz snoop`). (@pwpiwi)
+ - Added option c to `hf list` (mark CRC bytes) (@pwpiwi)
 
-### Changed																		
-- Added `[l] <length>` option to data printdemodbuffer
-- Adjusted lf awid clone to optionally clone to Q5 tags
-- Adjusted lf t55xx detect to find Q5 tags (t5555) instead of just t55x7
-- Adjusted all lf NRZ demods - works more accurately and consistently (as long as you have strong signal)
-- Adjusted lf pskindalademod to reduce false positive reads.
-- Small adjustments to psk, nrz, and ask clock detect routines - more reliable.
-- Adjusted lf em410x em410xsim to accept a clock argument
-- Adjusted lf t55xx dump to allow overriding the safety check and warning text (marshmellow)
-- Adjusted lf t55xx write input variables (marshmellow)
-- Adjusted lf t55xx read with password safety check and warning text and adjusted the input variables (marshmellow & iceman)
-- Adjusted LF FSK demod to account for cross threshold fluctuations (898 count waves will adjust the 9 to 8 now...) more accurate. (marshmellow)
-- Adjusted timings for t55xx commands.  more reliable now. (marshmellow & iceman)
-- `lf cmdread` adjusted input methods and added help text (marshmellow & iceman)
-- changed `lf config t <threshold>` to be 0 - 128 and will trigger on + or - threshold value (marshmellow)
-- `hf iclass dump` cli options - can now dump AA1 and AA2 with different keys in one run (does not go to multiple pages for the larger tags yet) (marshmellow)
-- Revised workflow for StandAloneMode14a (Craig Young)
-- EPA functions (`hf epa`) now support both ISO 14443-A and 14443-B cards (frederikmoellers)
-- 'hw version' only talks to ARM at startup, after that the info is cached. (pwpiwi)
-- Added `r` option to iclass functions - allows key to be provided in raw block 3/4 format
+### Changed
+ - Added `[l] <length>` option to data printdemodbuffer
+ - Adjusted lf awid clone to optionally clone to Q5 tags
+ - Adjusted lf t55xx detect to find Q5 tags (t5555) instead of just t55x7
+ - Adjusted all lf NRZ demods - works more accurately and consistently (as long as you have strong signal)
+ - Adjusted lf pskindalademod to reduce false positive reads.
+ - Small adjustments to psk, nrz, and ask clock detect routines - more reliable.
+ - Adjusted lf em410x em410xsim to accept a clock argument
+ - Adjusted lf t55xx dump to allow overriding the safety check and warning text (@marshmellow42)
+ - Adjusted lf t55xx write input variables (@marshmellow42)
+ - Adjusted lf t55xx read with password safety check and warning text and adjusted the input variables (@marshmellow42 & @iceman1001)
+ - Adjusted LF FSK demod to account for cross threshold fluctuations (898 count waves will adjust the 9 to 8 now...) more accurate. (@marshmellow42)
+ - Adjusted timings for t55xx commands.  more reliable now. (@marshmellow42 & @iceman1001)
+ - `lf cmdread` adjusted input methods and added help text (@marshmellow42 & @iceman1001)
+ - changed `lf config t <threshold>` to be 0 - 128 and will trigger on + or - threshold value (@marshmellow42)
+ - `hf iclass dump` cli options - can now dump AA1 and AA2 with different keys in one run (does not go to multiple pages for the larger tags yet) (@marshmellow42)
+ - Revised workflow for StandAloneMode14a (Craig Young)
+ - EPA functions (`hf epa`) now support both ISO 14443-A and 14443-B cards (@frederikmoellers)
+ - `hw version` only talks to ARM at startup, after that the info is cached. (@pwpiwi)
+ - Added `r` option to iclass functions - allows key to be provided in raw block 3/4 format
 
 ## [2.2.0][2015-07-12]
 
 ### Changed
-- Added `hf 14b raw -s` option to auto select a 14b std tag before raw command
-- Changed `hf 14b write` to `hf 14b sriwrite` as it only applied to sri tags (marshmellow)
-- Added `hf 14b info` to `hf search` (marshmellow)
-- Added compression of fpga config and data, *BOOTROM REFLASH REQUIRED* (piwi)
-- Implemented better detection of mifare-tags that are not vulnerable to classic attacks (`hf mf mifare`, `hf mf nested`) (piwi)
+ - Added `hf 14b raw -s` option to auto select a 14b std tag before raw command
+ - Changed `hf 14b write` to `hf 14b sriwrite` as it only applied to sri tags (@marshmellow42)
+ - Added `hf 14b info` to `hf search` (@marshmellow42)
+ - Added compression of fpga config and data, *BOOTROM REFLASH REQUIRED* (@pwpiwi)
+ - Implemented better detection of mifare-tags that are not vulnerable to classic attacks (`hf mf mifare`, `hf mf nested`) (@pwpiwi)
 
 ### Added
-- Add `hf 14b reader` to find and print general info about known 14b tags (marshmellow)
-- Add `hf 14b info` to find and print info about std 14b tags and sri tags (using 14b raw commands in the client)  (marshmellow)
-- Add PACE replay functionality (frederikmoellers)
+ - Add `hf 14b reader` to find and print general info about known 14b tags (@marshmellow42)
+ - Add `hf 14b info` to find and print info about std 14b tags and sri tags (using 14b raw commands in the client)  (@marshmellow42)
+ - Add PACE replay functionality (@frederikmoellers)
 
 ### Fixed
-- t55xx write timing (marshmellow)
+ - t55xx write timing (@marshmellow42)
 
 
 ## [2.1.0][2015-06-23]
 
 ### Changed
-- Added ultralight/ntag tag type detection to `hf 14a read` (marshmellow)
-- Improved ultralight dump command to auto detect tag type, take authentication, and dump full memory (or subset specified) of known tag types (iceman1001 / marshmellow)
-- Combined ultralight read/write commands and added authentication (iceman1001)
-- Improved LF manchester and biphase demodulation and ask clock detection especially for reads with heavy clipping. (marshmellow)
-- Iclass read, `hf iclass read` now also reads tag config and prints configuration. (holiman)
-- *bootrom* needs to be flashed, due to new address boundaries between os and fpga, after a size optimization (piwi)
+ - Added ultralight/ntag tag type detection to `hf 14a read` (@marshmellow42)
+ - Improved ultralight dump command to auto detect tag type, take authentication, and dump full memory (or subset specified) of known tag types (@iceman1001 / @marshmellow42)
+ - Combined ultralight read/write commands and added authentication (@iceman1001)
+ - Improved LF manchester and biphase demodulation and ask clock detection especially for reads with heavy clipping. (@marshmellow42)
+ - Iclass read, `hf iclass read` now also reads tag config and prints configuration. (@holiman)
+ - *bootrom* needs to be flashed, due to new address boundaries between os and fpga, after a size optimization (@pwpiwi)
 
 ### Fixed
-- Fixed EM4x50 read/demod of the tags broadcasted memory blocks. 'lf em4x em4x50read' (not page read) (marshmellow)
-- Fixed issue #19, problems with LF T55xx commands (iceman1001, marshmellow)
-- Fixed various problems with iso14443b, issue #103 (piwi, marshmellow)
+ - Fixed EM4x50 read/demod of the tags broadcasted memory blocks. `lf em4x em4x50read` (not page read) (@marshmellow42)
+ - Fixed issue #19, problems with LF T55xx commands (@iceman1001, @marshmellow42)
+ - Fixed various problems with iso14443b, issue #103 (@pwpiwi, @marshmellow42)
 
 ### Added
-- Added `hf search` - currently tests for 14443a tags, iclass tags, and 15693 tags (marshmellow)
-- Added `hf mfu info` Ultralight/NTAG info command - reads tag configuration and info, allows authentication if needed (iceman1001, marshmellow)
-- Added Mifare Ultralight C and Ultralight EV1/NTAG authentication. (iceman1001)
-- Added changelog			 
-- Added `data fdxbdemod` - Demodulate a FDX-B ISO11784/85 Biphase tag from GraphBuffer aka ANIMAL TAG (marshmellow, iceman1001)
+ - Added `hf search` - currently tests for 14443a tags, iclass tags, and 15693 tags (@marshmellow42)
+ - Added `hf mfu info` Ultralight/NTAG info command - reads tag configuration and info, allows authentication if needed (@iceman1001, @marshmellow42)
+ - Added Mifare Ultralight C and Ultralight EV1/NTAG authentication. (@iceman1001)
+ - Added changelog
+ - Added `data fdxbdemod` - Demodulate a FDX-B ISO11784/85 Biphase tag from GraphBuffer aka ANIMAL TAG (@marshmellow42, @iceman1001)
 
 ## [2.0.0] - 2015-03-25
 ### Changed
-- LF sim operations now abort when new commands arrive over the USB - not required to push the device button anymore.
+ - LF sim operations now abort when new commands arrive over the USB - not required to push the device button anymore.
 
 ### Fixed
-- Mifare simulation, `hf mf sim` (was broken a long time) (pwpiwi)
-- Major improvements in LF area and data operations. (marshmellow, iceman1001)
-- Issues regarding LF simulation (pwpiwi)
+ - Mifare simulation, `hf mf sim` (was broken a long time) (@pwpiwi)
+ - Major improvements in LF area and data operations. (@marshmellow42, @iceman1001)
+ - Issues regarding LF simulation (@pwpiwi)
 
 ### Added
-- iClass functionality: full simulation of iclass tags, so tags can be simulated with data (not only CSN). Not yet support for write/update, but readers don't seem to enforce update. (holiman).
-- iClass decryption. Proxmark can now decrypt data on an iclass tag, but requires you to have the HID decryption key locally on your computer, as this is not bundled with the sourcecode.
+ - iClass functionality: full simulation of iclass tags, so tags can be simulated with data (not only CSN). Not yet support for write/update, but readers do not seem to enforce update. (@holiman).
+ - iClass decryption. Proxmark can now decrypt data on an iclass tag, but requires you to have the HID decryption key locally on your computer, as this is not bundled with the sourcecode.
diff --git a/LICENSE.txt b/LICENSE.txt
index b26a158fa..7f48bec21 100644
--- a/LICENSE.txt
+++ b/LICENSE.txt
@@ -1,13 +1,13 @@
 
-		    GNU GENERAL PUBLIC LICENSE
-		       Version 2, June 1991
+            GNU GENERAL PUBLIC LICENSE
+                Version 2, June 1991
 
  Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
  51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  Everyone is permitted to copy and distribute verbatim copies
  of this license document, but changing it is not allowed.
 
-			    Preamble
+                Preamble
 
   The licenses for most software are designed to take away your
 freedom to share and change it.  By contrast, the GNU General Public
@@ -57,7 +57,7 @@ patent must be licensed for everyone's free use or not licensed at all.
   The precise terms and conditions for copying, distribution and
 modification follow.
 
-		    GNU GENERAL PUBLIC LICENSE
+            GNU GENERAL PUBLIC LICENSE
    TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
 
   0. This License applies to any program or other work which contains
@@ -256,7 +256,7 @@ make exceptions for this.  Our decision will be guided by the two goals
 of preserving the free status of all derivatives of our free software and
 of promoting the sharing and reuse of software generally.
 
-			    NO WARRANTY
+                NO WARRANTY
 
   11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
 FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW.  EXCEPT WHEN
@@ -278,4 +278,4 @@ YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
 PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
 POSSIBILITY OF SUCH DAMAGES.
 
-		     END OF TERMS AND CONDITIONS
+            END OF TERMS AND CONDITIONS
diff --git a/Makefile b/Makefile
index 910aa62f6..587fdf09d 100644
--- a/Makefile
+++ b/Makefile
@@ -145,21 +145,24 @@ style:
 	# Make sure astyle is installed
 	@which astyle >/dev/null || ( echo "Please install 'astyle' package first" ; exit 1 )
 	# Remove spaces & tabs at EOL, add LF at EOF if needed on *.c, *.h, *.cpp. *.lua, *.py, *.pl, Makefile
-	find . \( -name "*.[ch]" -or -name "*.cpp" -or -name "*.lua" -or -name "*.py" -or -name "*.pl" -or -name "Makefile" \) \
+	find . \( -name "*.[ch]" -or \( -name "*.cpp" -and -not -name "*.moc.cpp" \) -or -name "*.lua" -or -name "*.py" -or -name "*.pl" -or -name "Makefile" \) \
 	    -exec perl -pi -e 's/[ \t]+$$//' {} \; \
 	    -exec sh -c "tail -c1 {} | xxd -p | tail -1 | grep -q -v 0a$$" \; \
 	    -exec sh -c "echo >> {}" \;
 	# Apply astyle on *.c, *.h, *.cpp
-	find . \( -name "*.[ch]" -or -name "*.cpp" \) -exec astyle --formatted --mode=c --suffix=none \
+	find . \( -name "*.[ch]" -or \( -name "*.cpp" -and -not -name "*.moc.cpp" \) \) -exec astyle --formatted --mode=c --suffix=none \
 	    --indent=spaces=4 --indent-switches \
 	    --keep-one-line-blocks --max-instatement-indent=60 \
 	    --style=google --pad-oper --unpad-paren --pad-header \
 	    --align-pointer=name {} \;
 
-# Detecting weird codepages.
+# Detecting weird codepages and tabs.
 checks:
 	find . \( -name "*.[ch]" -or -name "*.cpp" -or -name "*.lua" -or -name "*.py" -or -name "*.pl" -or -name "Makefile" \) \
 	      -exec sh -c "cat {} |recode utf8.. >/dev/null || echo {}" \;
+	find . \( -name "*.[ch]" -or \( -name "*.cpp" -and -not -name "*.moc.cpp" \) -or -name "*.lua" -or -name "*.py" -or -name "*.pl" -or -name "*.md" -or -name "*.txt" -or -name "*.awk" \) \
+	      -exec grep -lP '\t' {} \;
+# to remove tabs within lines, one can try with vi: :set tabstop=4   :set et|retab
 
 # Dummy target to test for GNU make availability
 _test:
diff --git a/Makefile.platform.sample b/Makefile.platform.sample
index b0d6cf5fe..9d0e2d965 100644
--- a/Makefile.platform.sample
+++ b/Makefile.platform.sample
@@ -1,4 +1,7 @@
 # If you want to use it, copy this file as Makefile.platform and adjust it to your needs
+# Run 'make PLATFORM=' to get an exhaustive list of possible parameters for this file.
+
 PLATFORM=PM3RDV4
+# If you want more than one PLATFORM_EXTRAS option, separate them by spaces:
 #PLATFORM_EXTRAS=BTADDON
 #STANDALONE=LF_SAMYRUN
diff --git a/README.md b/README.md
index e63e41ae2..50813868e 100644
--- a/README.md
+++ b/README.md
@@ -1,9 +1,11 @@
 # Proxmark3 RDV4.0 Dedicated Github
 
 This repo is based on iceman fork for Proxmark3. It is dedicated to bringing the most out of the new features for Proxmark3 RDV4.0 new hardware and design.
+Note that it also supports other Proxmark3 platforms as well!
 
-[![Build status](https://ci.appveyor.com/api/projects/status/ct5blik2wa96bv0x/branch/master?svg=true)](https://ci.appveyor.com/project/iceman1001/proxmark3-ji4wj/branch/master)
-[![Latest release](https://img.shields.io/github/release/RfidResearchGroup/proxmark3.svg)](https://github.com/RfidResearchGroup/proxmark3/releases/latest)
+| Releases     | Linux & OSX CI       | Windows CI |
+| ------------------- |:-------------------:| -------------------:|
+| [![Latest release](https://img.shields.io/github/release/RfidResearchGroup/proxmark3.svg)](https://github.com/RfidResearchGroup/proxmark3/releases/latest) | [![Build status](https://travis-ci.org/RfidResearchGroup/proxmark3.svg?branch=master)](https://travis-ci.org/RfidResearchGroup/proxmark3) | [![Build status](https://ci.appveyor.com/api/projects/status/b4gwrhq3nc876cuu/branch/master?svg=true)](https://ci.appveyor.com/project/RfidResearchGroup/proxmark3/branch/master) |
 
 ---
 
@@ -15,9 +17,10 @@ This repo is based on iceman fork for Proxmark3. It is dedicated to bringing the
 |[Development](#development) | [Important notes on ModemManager for Linux users](/doc/md/Installation_Instructions/ModemManager-Must-Be-Discarded.md) | [Validating proxmark client functionality](/doc/md/Use_of_Proxmark/1_Validation.md) |
 |[Why didn't you base it on official PM3 Master?](#why-didnt-you-base-it-on-official-pm3-master)| [Homebrew (Mac OS X) & Upgrading HomeBrew Tap Formula](/doc/md/Installation_Instructions/Mac-OS-X-Homebrew-Installation-Instructions.md) | [First Use and Verification](/doc/md/Use_of_Proxmark/2_Configuration-and-Verification.md)|
 |[PM3 GUI](#pm3-gui)|[Setup and build for Windows](/doc/md/Installation_Instructions/Windows-Installation-Instructions.md)|[Commands & Features](/doc/md/Use_of_Proxmark/3_Commands-and-Features.md)|
-|[Issues](#issues)|[Blue shark manual](/doc/bt_manual_v10.md) ||
+|[Issues](#issues)|[Blue shark manual](/doc/bt_manual_v10.md) |[Advanced compilation parameters](/doc/md/Use_of_Proxmark/4_Advanced-compilation-parameters.md)|
 |[Notes on UART](/doc/uart_notes.md)|||
 |[Notes on Frame format](/doc/new_frame_format.md)|||
+|[Notes on external flash](/doc/ext_flash_notes.md)|||
 |[Notes on Termux / Android](/doc/termux_notes.md)|||
 |[Developing standalone mode](/armsrc/Standalone/readme.md)|[Wiki about standalone mode](https://github.com/RfidResearchGroup/proxmark3/wiki/Standalone-mode) ||
 |[Donations](#Donations)|||
diff --git a/armsrc/Makefile b/armsrc/Makefile
index 7fda76b99..105a880a7 100644
--- a/armsrc/Makefile
+++ b/armsrc/Makefile
@@ -30,10 +30,11 @@ SRC_ISO15693 = iso15693.c iso15693tools.c
 SRC_ISO14443a = iso14443a.c mifareutil.c mifarecmd.c epa.c mifaresim.c
 SRC_ISO14443b = iso14443b.c
 SRC_FELICA = felica.c
-SRC_CRAPTO1 = crypto1.c des.c aes.c desfire_key.c desfire_crypto.c mifaredesfire.c
+SRC_CRAPTO1 = crypto1.c des.c desfire_key.c desfire_crypto.c mifaredesfire.c aes.c platform_util.c
 SRC_CRC = crc.c crc16.c crc32.c
 SRC_ICLASS = iclass.c optimized_cipher.c
 SRC_LEGIC = legicrf.c legicrfsim.c legic_prng.c
+SRC_SPIFFS = spiffs.c spiffs_cache.c spiffs_check.c spiffs_gc.c spiffs_nucleus.c spiffs_hydrogen.c
 #   SRC_BEE = bee.c
 
 # RDV40 related hardware support
@@ -68,41 +69,7 @@ else
 endif
 
 # Generic standalone Mode injection of source code
-
-
-SRC_STANDALONE = placeholder.c
-# WITH_STANDALONE_LF_ICERUN
-ifneq (,$(findstring WITH_STANDALONE_LF_ICERUN,$(APP_CFLAGS)))
-    SRC_STANDALONE = lf_icerun.c
-endif
-# WITH_STANDALONE_LF_SAMYRUN
-ifneq (,$(findstring WITH_STANDALONE_LF_SAMYRUN,$(APP_CFLAGS)))
-    SRC_STANDALONE = lf_samyrun.c
-endif
-# WITH_STANDALONE_LF_PROXBRUTE
-ifneq (,$(findstring WITH_STANDALONE_LF_PROXBRUTE,$(APP_CFLAGS)))
-    SRC_STANDALONE = lf_proxbrute.c
-endif
-# WITH_STANDALONE_LF_HIDBRUTE
-ifneq (,$(findstring WITH_STANDALONE_LF_HIDBRUTE,$(APP_CFLAGS)))
-    SRC_STANDALONE = lf_hidbrute.c
-endif
-# WITH_STANDALONE_HF_YOUNG
-ifneq (,$(findstring WITH_STANDALONE_HF_YOUNG,$(APP_CFLAGS)))
-    SRC_STANDALONE = hf_young.c
-endif
-# WITH_STANDALONE_HF_MATTYRUN
-ifneq (,$(findstring WITH_STANDALONE_HF_MATTYRUN,$(APP_CFLAGS)))
-    SRC_STANDALONE = hf_mattyrun.c
-endif
-# WITH_STANDALONE_HF_COLIN
-ifneq (,$(findstring WITH_STANDALONE_HF_COLIN,$(APP_CFLAGS)))
-    SRC_STANDALONE = vtsend.c hf_colin.c
-endif
-# WITH_STANDALONE_HF_BOG
-ifneq (,$(findstring WITH_STANDALONE_HF_BOG,$(APP_CFLAGS)))
-    SRC_STANDALONE = hf_bog.c
-endif
+include Standalone/Makefile.inc
 
 #the FPGA bitstream files. Note: order matters!
 FPGA_BITSTREAMS = fpga_lf.bit fpga_hf.bit
@@ -131,6 +98,7 @@ THUMBSRC = start.c \
     $(SRC_SMARTCARD) \
     $(SRC_FPC) \
     $(SRC_HITAG) \
+    $(SRC_SPIFFS) \
     appmain.c \
     printf.c \
     commonutil.c \
diff --git a/armsrc/Standalone/Makefile.hal b/armsrc/Standalone/Makefile.hal
new file mode 100644
index 000000000..5d3904794
--- /dev/null
+++ b/armsrc/Standalone/Makefile.hal
@@ -0,0 +1,55 @@
+# Default standalone if no standalone specified
+DEFAULT_STANDALONE=LF_SAMYRUN
+HELP_EXAMPLE_STANDALONE=HF_COLIN
+# (you can set explicitly STANDALONE= to disable standalone modes)
+STANDALONE?=$(DEFAULT_STANDALONE)
+STANDALONE_REQ_DEFS=
+
+define KNOWN_STANDALONE_DEFINITIONS
++==========================================================+
+| STANDALONE      | DESCRIPTION                            |
++==========================================================+
+| (empty)         | No standalone mode                     |
++----------------------------------------------------------+
+| LF_SAMYRUN      | HID26 read/clone/sim                   |
+| (default)       | - Samy Kamkar                          |
++----------------------------------------------------------+
+| LF_ICERUN       | standalone mode skeleton               |
+|                 | - iceman                               |
++----------------------------------------------------------+
+| LF_PROXBRUTE    | HID ProxII bruteforce                  |
+|                 | - Brad Antoniewicz                     |
++----------------------------------------------------------+
+| LF_HIDBRUTE     | HID corporate 1000 bruteforce          |
+|                 | - Federico dotta & Maurizio Agazzini   |
++----------------------------------------------------------+
+| HF_YOUNG        | Mifare sniff/simulation                |
+|                 | - Craig Young                          |
++----------------------------------------------------------+
+| HF_MATTYRUN     | Mifare sniff/clone                     |
+|                 | - Matías A. Ré Medina                  |
++----------------------------------------------------------+
+| HF_COLIN        | Mifare ultra fast sniff/sim/clone      |
+| (RDV4 only)     | - Colin Brigato                        |
++----------------------------------------------------------+
+| HF_BOG          | 14a sniff with ULC/ULEV1/NTAG auth     |
+| (RDV4 only)     | storing in flashmem - Bogito           |
++----------------------------------------------------------+
+
+endef
+
+STANDALONE_MODES := LF_SAMYRUN LF_ICERUN LF_PROXBRUTE LF_HIDBRUTE
+STANDALONE_MODES += HF_YOUNG HF_MATTYRUN HF_COLIN HF_BOG
+STANDALONE_MODES_REQ_SMARTCARD :=
+STANDALONE_MODES_REQ_FLASH := HF_COLIN HF_BOG
+ifneq ($(filter $(STANDALONE),$(STANDALONE_MODES)),)
+    STANDALONE_PLATFORM_DEFS += -DWITH_STANDALONE_$(STANDALONE)
+    ifneq ($(filter $(STANDALONE),$(STANDALONE_MODES_REQ_SMARTCARD)),)
+        STANDALONE_REQ_DEFS += -DWITH_SMARTCARD
+    endif
+    ifneq ($(filter $(STANDALONE),$(STANDALONE_MODES_REQ_FLASH)),)
+        STANDALONE_REQ_DEFS += -DWITH_FLASH
+    endif
+else ifneq ($(STANDALONE),)
+    $(error Invalid STANDALONE: $(STANDALONE). $(KNOWN_DEFINITIONS))
+endif
diff --git a/armsrc/Standalone/Makefile.inc b/armsrc/Standalone/Makefile.inc
new file mode 100644
index 000000000..cc7221a73
--- /dev/null
+++ b/armsrc/Standalone/Makefile.inc
@@ -0,0 +1,36 @@
+# Generic standalone Mode injection of source code
+
+SRC_STANDALONE = placeholder.c
+# WITH_STANDALONE_LF_ICERUN
+ifneq (,$(findstring WITH_STANDALONE_LF_ICERUN,$(APP_CFLAGS)))
+    SRC_STANDALONE = lf_icerun.c
+endif
+# WITH_STANDALONE_LF_SAMYRUN
+ifneq (,$(findstring WITH_STANDALONE_LF_SAMYRUN,$(APP_CFLAGS)))
+    SRC_STANDALONE = lf_samyrun.c
+endif
+# WITH_STANDALONE_LF_PROXBRUTE
+ifneq (,$(findstring WITH_STANDALONE_LF_PROXBRUTE,$(APP_CFLAGS)))
+    SRC_STANDALONE = lf_proxbrute.c
+endif
+# WITH_STANDALONE_LF_HIDBRUTE
+ifneq (,$(findstring WITH_STANDALONE_LF_HIDBRUTE,$(APP_CFLAGS)))
+    SRC_STANDALONE = lf_hidbrute.c
+endif
+# WITH_STANDALONE_HF_YOUNG
+ifneq (,$(findstring WITH_STANDALONE_HF_YOUNG,$(APP_CFLAGS)))
+    SRC_STANDALONE = hf_young.c
+endif
+# WITH_STANDALONE_HF_MATTYRUN
+ifneq (,$(findstring WITH_STANDALONE_HF_MATTYRUN,$(APP_CFLAGS)))
+    SRC_STANDALONE = hf_mattyrun.c
+endif
+# WITH_STANDALONE_HF_COLIN
+ifneq (,$(findstring WITH_STANDALONE_HF_COLIN,$(APP_CFLAGS)))
+    SRC_STANDALONE = vtsend.c hf_colin.c
+endif
+# WITH_STANDALONE_HF_BOG
+ifneq (,$(findstring WITH_STANDALONE_HF_BOG,$(APP_CFLAGS)))
+    SRC_STANDALONE = hf_bog.c
+endif
+
diff --git a/armsrc/Standalone/hf_bog.c b/armsrc/Standalone/hf_bog.c
index 093bc841f..f8f832220 100644
--- a/armsrc/Standalone/hf_bog.c
+++ b/armsrc/Standalone/hf_bog.c
@@ -15,7 +15,7 @@ The retrieved sniffing session can be acquired by connecting the device
 to a client that supports the reconnect capability and issue 'hf 14a list'.
 
 In order to view the grabbed authentication attempts in the flash mem,
-you can simply run 'script run read_pwd_mem' or just 'mem read l 256'
+you can simply run 'script run read_pwd_mem' or just 'mem dump p l 256'
 from the client to view the stored quadlets.
 */
 
@@ -27,35 +27,7 @@ from the client to view the stored quadlets.
 // Maximum number of auth attempts per standalone session
 #define MAX_PWDS_PER_SESSION 64
 
-uint8_t FindOffsetInFlash() {
-    uint8_t mem[4] = { 0x00, 0x00, 0x00, 0x00 };
-    uint8_t eom[4] = { 0xFF, 0xFF, 0xFF, 0xFF };
-    uint8_t memcnt = 0;
-
-    while (memcnt < 0xFF) {
-        Flash_ReadData(memcnt, mem, 4);
-        if (memcmp(mem, eom, 4) == 0) {
-            return memcnt;
-        }
-        memcnt += 4;
-    }
-
-    return 0; // wrap-around
-}
-
-void EraseMemory() {
-    if (!FlashInit()) {
-        return;
-    }
-
-    Flash_CheckBusy(BUSY_TIMEOUT);
-    Flash_WriteEnable();
-    Flash_Erase4k(0, 0);
-
-    if (DBGLEVEL > 1) Dbprintf("[!] Erased flash!");
-    FlashStop();
-    SpinDelay(100);
-}
+#define HF_BOG_LOGFILE "hf_bog.log"
 
 // This is actually copied from SniffIso14443a
 void RAMFUNC SniffAndStore(uint8_t param) {
@@ -96,8 +68,9 @@ void RAMFUNC SniffAndStore(uint8_t param) {
     UartInit(receivedCmd, receivedCmdPar);
 
     // Setup and start DMA.
-    if (!FpgaSetupSscDma((uint8_t *) dmaBuf, DMA_BUFFER_SIZE)) {
-        if (DBGLEVEL > 1) Dbprintf("FpgaSetupSscDma failed. Exiting");
+    if (!FpgaSetupSscDma((uint8_t *)dmaBuf, DMA_BUFFER_SIZE)) {
+        if (DBGLEVEL > 1)
+            Dbprintf("FpgaSetupSscDma failed. Exiting");
         return;
     }
 
@@ -134,17 +107,18 @@ void RAMFUNC SniffAndStore(uint8_t param) {
             Dbprintf("[!] blew circular buffer! | datalen %u", dataLen);
             break;
         }
-        if (dataLen < 1) continue;
+        if (dataLen < 1)
+            continue;
 
         // primary buffer was stopped( <-- we lost data!
         if (!AT91C_BASE_PDC_SSC->PDC_RCR) {
-            AT91C_BASE_PDC_SSC->PDC_RPR = (uint32_t) dmaBuf;
+            AT91C_BASE_PDC_SSC->PDC_RPR = (uint32_t)dmaBuf;
             AT91C_BASE_PDC_SSC->PDC_RCR = DMA_BUFFER_SIZE;
-            //Dbprintf("[-] RxEmpty ERROR | data length %d", dataLen); // temporary
+            // Dbprintf("[-] RxEmpty ERROR | data length %d", dataLen); // temporary
         }
         // secondary buffer sets as primary, secondary buffer was stopped
         if (!AT91C_BASE_PDC_SSC->PDC_RNCR) {
-            AT91C_BASE_PDC_SSC->PDC_RNPR = (uint32_t) dmaBuf;
+            AT91C_BASE_PDC_SSC->PDC_RNPR = (uint32_t)dmaBuf;
             AT91C_BASE_PDC_SSC->PDC_RNCR = DMA_BUFFER_SIZE;
         }
 
@@ -159,23 +133,24 @@ void RAMFUNC SniffAndStore(uint8_t param) {
                     LED_C_ON();
 
                     // check - if there is a short 7bit request from reader
-                    if ((!triggered) && (param & 0x02) && (uart->len == 1) && (uart->bitCount == 7)) triggered = true;
+                    if ((!triggered) && (param & 0x02) && (uart->len == 1) && (uart->bitCount == 7))
+                        triggered = true;
 
                     if (triggered) {
-                        if ((receivedCmd) && ((receivedCmd[0] == MIFARE_ULEV1_AUTH) || (receivedCmd[0] == MIFARE_ULC_AUTH_1))) {
-                            if (DBGLEVEL > 1) Dbprintf("PWD-AUTH KEY: 0x%02x%02x%02x%02x", receivedCmd[1], receivedCmd[2], receivedCmd[3], receivedCmd[4]);
+                        if ((receivedCmd) &&
+                                ((receivedCmd[0] == MIFARE_ULEV1_AUTH) || (receivedCmd[0] == MIFARE_ULC_AUTH_1))) {
+                            if (DBGLEVEL > 1)
+                                Dbprintf("PWD-AUTH KEY: 0x%02x%02x%02x%02x", receivedCmd[1], receivedCmd[2],
+                                         receivedCmd[3], receivedCmd[4]);
 
                             // temporarily save the captured pwd in our array
                             memcpy(&capturedPwds[4 * auth_attempts], receivedCmd + 1, 4);
                             auth_attempts++;
                         }
 
-                        if (!LogTrace(receivedCmd,
-                                      uart->len,
-                                      uart->startTime * 16 - DELAY_READER_AIR2ARM_AS_SNIFFER,
-                                      uart->endTime * 16 - DELAY_READER_AIR2ARM_AS_SNIFFER,
-                                      uart->parity,
-                                      true)) break;
+                        if (!LogTrace(receivedCmd, uart->len, uart->startTime * 16 - DELAY_READER_AIR2ARM_AS_SNIFFER,
+                                      uart->endTime * 16 - DELAY_READER_AIR2ARM_AS_SNIFFER, uart->parity, true))
+                            break;
                     }
                     /* ready to receive another command. */
                     UartReset();
@@ -193,20 +168,18 @@ void RAMFUNC SniffAndStore(uint8_t param) {
                 if (ManchesterDecoding(tagdata, 0, (my_rsamples - 1) * 4)) {
                     LED_B_ON();
 
-                    if (!LogTrace(receivedResp,
-                                  demod->len,
-                                  demod->startTime * 16 - DELAY_TAG_AIR2ARM_AS_SNIFFER,
-                                  demod->endTime * 16 - DELAY_TAG_AIR2ARM_AS_SNIFFER,
-                                  demod->parity,
-                                  false)) break;
+                    if (!LogTrace(receivedResp, demod->len, demod->startTime * 16 - DELAY_TAG_AIR2ARM_AS_SNIFFER,
+                                  demod->endTime * 16 - DELAY_TAG_AIR2ARM_AS_SNIFFER, demod->parity, false))
+                        break;
 
-                    if ((!triggered) && (param & 0x01)) triggered = true;
+                    if ((!triggered) && (param & 0x01))
+                        triggered = true;
 
                     // ready to receive another response.
                     DemodReset();
                     // reset the Miller decoder including its (now outdated) input buffer
                     UartReset();
-                    //UartInit(receivedCmd, receivedCmdPar);
+                    // UartInit(receivedCmd, receivedCmdPar);
                     LED_C_OFF();
                 }
                 TagIsActive = (demod->state != DEMOD_UNSYNCD);
@@ -228,57 +201,25 @@ void RAMFUNC SniffAndStore(uint8_t param) {
 
     SpinDelay(200);
 
-    // Write stuff to flash
+    // Write stuff to spiffs logfile
     if (auth_attempts > 0) {
-        if (DBGLEVEL > 1) Dbprintf("[!] Authentication attempts = %u", auth_attempts);
+        if (DBGLEVEL > 1)
+            Dbprintf("[!] Authentication attempts = %u", auth_attempts);
+        size_t size = 4 * auth_attempts;
+        uint8_t *data = BigBuf_malloc(size);
 
-        // Setting the SPI Baudrate to 48MHz to avoid the bit-flip issue (https://github.com/RfidResearchGroup/proxmark3/issues/34)
-        FlashmemSetSpiBaudrate(48000000);
-
-        // Find the offset in flash mem to continue writing the auth attempts
-        uint8_t memoffset = FindOffsetInFlash();
-        if (DBGLEVEL > 1) Dbprintf("[!] Memory offset = %u", memoffset);
-
-        if ((memoffset + 4 * auth_attempts) > 0xFF) {
-            // We opt to keep the new data only
-            memoffset = 0;
-            if (DBGLEVEL > 1) Dbprintf("[!] Size of total data > 256 bytes. Discarding the old data.");
+        if (!exists_in_spiffs((char *)HF_BOG_LOGFILE)) {
+            rdv40_spiffs_write((char *)HF_BOG_LOGFILE, (uint8_t *)data, size, RDV40_SPIFFS_SAFETY_SAFE);
+        } else {
+            rdv40_spiffs_append((char *)HF_BOG_LOGFILE, (uint8_t *)data, size, RDV40_SPIFFS_SAFETY_SAFE);
         }
-
-        // Get previous data from flash mem
-        uint8_t *previousdata = BigBuf_malloc(memoffset);
-        if (memoffset > 0) {
-            uint16_t readlen = Flash_ReadData(0, previousdata, memoffset);
-            if (DBGLEVEL > 1) Dbprintf("[!] Read %u bytes from flash mem", readlen);
-        }
-
-        // create new bigbuf to hold all data
-        size_t total_size = memoffset + 4 * auth_attempts;
-        uint8_t *total_data = BigBuf_malloc(total_size);
-
-        // Add the previousdata array into total_data array
-        memcpy(total_data, previousdata, memoffset);
-
-        // Copy bytes of capturedPwds immediately following bytes of previousdata
-        memcpy(total_data + memoffset, capturedPwds, 4 * auth_attempts);
-
-        // Erase first page of flash mem
-        EraseMemory();
-
-        // Write total data to flash mem
-        uint16_t writelen = Flash_WriteData(0, total_data, memoffset + 4 * auth_attempts);
-        if (DBGLEVEL > 1) Dbprintf("[!] Wrote %u bytes into flash mem", writelen);
-
-        // If pwd saved successfully, blink led A three times
-        if (writelen > 0) {
-            SpinErr(0, 200, 5); // blink led A
-        }
-
-        SpinDelay(100);
-
-        // Reset the SPI Baudrate to the default value (24MHz)
-        FlashmemSetSpiBaudrate(24000000);
     }
+
+    if (DBGLEVEL > 1)
+        Dbprintf("[!] Wrote %u Authentification attempts into logfile", auth_attempts);
+
+    SpinErr(0, 200, 5); // blink led A
+    SpinDelay(100);
 }
 
 void ModInfo(void) {
diff --git a/armsrc/Standalone/hf_bog.h b/armsrc/Standalone/hf_bog.h
index 3dd68b3e2..50bdf2df8 100644
--- a/armsrc/Standalone/hf_bog.h
+++ b/armsrc/Standalone/hf_bog.h
@@ -21,6 +21,7 @@
 #include "apps.h"
 #include "printf.h"
 #include "parity.h"
+#include "spiffs.h"
 
 
 #endif /* __HF_BOG_H */
diff --git a/armsrc/Standalone/hf_colin.c b/armsrc/Standalone/hf_colin.c
index 63366259e..5eeb2a42a 100644
--- a/armsrc/Standalone/hf_colin.c
+++ b/armsrc/Standalone/hf_colin.c
@@ -1,5 +1,5 @@
 //-----------------------------------------------------------------------------
-// Colin Brigato, 2016, 2017
+// Colin Brigato, 2016, 2017, 2018, 2019
 // Christian Herrmann, 2017
 //
 // This code is licensed to you under the terms of the GNU GPL, version 2 or,
@@ -13,6 +13,7 @@
 #define MF1KSZ 1024
 #define MF1KSZSIZE 64
 #define AUTHENTICATION_TIMEOUT 848
+#define HFCOLIN_LASTTAG_SYMLINK "hf_colin/lasttag.bin"
 
 uint8_t cjuid[10];
 uint32_t cjcuid;
@@ -26,39 +27,78 @@ int curlline;
 
 // Colin's VIGIKPWN sniff/simulate/clone repeat routine for HF Mifare
 
-/*
-void cjPrintBigArray(const char *bigar, int len, uint8_t newlines, uint8_t debug)
-{
-    uint32_t chunksize = (PM3_CMD_DATA_SIZE / 4);
-    uint8_t totalchunks = len / chunksize;
-    uint8_t last_chunksize = len - (totalchunks * chunksize);
-    char chunk[chunksize + 1];
-    memset(chunk, 0x00, sizeof(chunk));
-    if (debug > 0)
-    {
-        Dbprintf("len : %d", len);
-        Dbprintf("chunksize : %d bytes", chunksize);
-        Dbprintf("totalchunks : %d", totalchunks);
-        Dbprintf("last_chunksize: %d", last_chunksize);
+typedef struct MFC1KSchema {
+    uint8_t name[32];
+    uint64_t trigger;
+    uint64_t keysA[16];
+    uint64_t keysB[16];
+} MFC1KSchema;
+
+#define MAX_SCHEMAS 4
+
+MFC1KSchema Schemas[MAX_SCHEMAS];
+
+MFC1KSchema Noralsy = {
+    .name = "Noralsy",
+    .trigger = 0x414c41524f4e,
+    .keysA = {
+        0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e,
+        0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e,
+        0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e, 0x414c41524f4e
+    },
+    .keysB = {
+        0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e,
+        0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e,
+        0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e, 0x424c41524f4e
     }
-    for (uint8_t i = 0; i < totalchunks; i++)
-    {
-        memset(chunk, 0x00, sizeof(chunk));
-        memcpy(chunk, &bigar[i * chunksize], chunksize);
-        DbprintfEx(FLAG_RAWPRINT, "%s", chunk);
+};
+
+MFC1KSchema InfiHexact = {.name = "Infineon/Hexact",
+                          .trigger = 0x484558414354,
+                          .keysA = {0x484558414354, 0x484558414354, 0x484558414354, 0x484558414354, 0x484558414354,
+                                    0x484558414354, 0x484558414354, 0x484558414354, 0x484558414354, 0x484558414354,
+                                    0x484558414354, 0x484558414354, 0x484558414354, 0x484558414354, 0x484558414354,
+                                    0x484558414354
+                                   },
+                          .keysB = {0xa22ae129c013, 0x49fae4e3849f, 0x38fcf33072e0, 0x8ad5517b4b18, 0x509359f131b1,
+                                    0x6c78928e1317, 0xaa0720018738, 0xa6cac2886412, 0x62d0c424ed8e, 0xe64a986a5d94,
+                                    0x8fa1d601d0a2, 0x89347350bd36, 0x66d2b7dc39ef, 0x6bc1e1ae547d, 0x22729a9bd40f
+                                   }
+                         };
+
+MFC1KSchema UrmetCaptive = {
+    .name = "Urmet Captive",
+    .trigger = 0x8829da9daf76,
+    .keysA = {
+        0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76,
+        0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76,
+        0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76
+    },
+    .keysB = {
+        0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76,
+        0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76,
+        0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76, 0x8829da9daf76
     }
-    if (last_chunksize > 0)
-    {
-        memset(chunk, 0x00, sizeof(chunk));
-        memcpy(chunk, &bigar[totalchunks * chunksize], last_chunksize);
-        DbprintfEx(FLAG_RAWPRINT, "%s", chunk);
-    }
-    if (newlines > 0)
-    {
-        DbprintfEx(FLAG_NEWLINE, " ");
+};
+
+int total_schemas = 0;
+
+void add_schema(MFC1KSchema *p, MFC1KSchema a, int *schemas_counter) {
+    if (*schemas_counter < MAX_SCHEMAS) {
+        p[*schemas_counter] = a;
+        *schemas_counter += 1;
+    }
+}
+
+void delete_schema(MFC1KSchema *p, int *schemas_counter, int index) {
+    if (*schemas_counter > 0 && index < *schemas_counter && index > -1) {
+        int last_index = *schemas_counter - 1;
+        for (int i = index; i < last_index; i++) {
+            p[i] = p[i + 1];
+        }
+        *schemas_counter -= 1;
     }
 }
-*/
 
 void cjSetCursFRight() {
     vtsend_cursor_position(NULL, 98, (currfline));
@@ -80,9 +120,8 @@ void cjTabulize() { DbprintfEx(FLAG_RAWPRINT, "\t\t\t"); }
 /*
 void cjPrintKey(uint64_t key, uint8_t *foundKey, uint16_t sectorNo, uint8_t type) {
     char tosendkey[13];
-    sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[0], foundKey[1], foundKey[2], foundKey[3], foundKey[4], foundKey[5]);
-    cjSetCursRight();
-    DbprintfEx(FLAG_NEWLINE, "SEC: %02x | KEY : %s | TYP: %d", sectorNo, tosendkey, type);
+    sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[0], foundKey[1], foundKey[2], foundKey[3], foundKey[4],
+foundKey[5]); cjSetCursRight(); DbprintfEx(FLAG_NEWLINE, "SEC: %02x | KEY : %s | TYP: %d", sectorNo, tosendkey, type);
 }
 */
 
@@ -92,47 +131,26 @@ void ReadLastTagFromFlash() {
     LED_B_ON();
     LED_C_ON();
     LED_D_ON();
-    uint32_t startidx = 0;
     uint16_t len = 1024;
+    size_t size = len;
 
     DbprintfEx(FLAG_NEWLINE, "Button HELD ! Using LAST Known TAG for Simulation...");
     cjSetCursLeft();
 
-    size_t size = len;
     uint8_t *mem = BigBuf_malloc(size);
 
-    if (!FlashInit()) {
-        return;
-    }
-    Flash_CheckBusy(BUSY_TIMEOUT);
+    // this one will handle filetype (symlink or not) and resolving by itself
+    rdv40_spiffs_read_as_filetype((char *)HFCOLIN_LASTTAG_SYMLINK, (uint8_t *)mem, len, RDV40_SPIFFS_SAFETY_SAFE);
 
-    uint32_t start_time = GetTickCount();
-    uint32_t delta_time = 0;
+    emlSetMem(mem, 0, 64);
 
-    for (size_t i = 0; i < len; i += size) {
-        len = MIN((len - i), size);
-        uint16_t isok = Flash_ReadDataCont(startidx + i, mem, len);
-        if (isok == len) {
-            emlSetMem(mem, 0, 64);
-        } else {
-            DbprintfEx(FLAG_NEWLINE, "FlashMem reading failed | %d | %d", len, isok);
-            cjSetCursLeft();
-            FlashStop();
-            SpinOff(100);
-            return;
-        }
-    }
-    delta_time = GetTickCountDelta(start_time);
     DbprintfEx(FLAG_NEWLINE, "[OK] Last tag recovered from FLASHMEM set to emulator");
     cjSetCursLeft();
-    DbprintfEx(FLAG_NEWLINE, "%s[IN]%s %s%dms%s for TAG_FLASH_READ", _XGREEN_, _XWHITE_, _XYELLOW_, delta_time, _XWHITE_);
-    cjSetCursLeft();
-    FlashStop();
     SpinOff(0);
     return;
 }
 
-void WriteTagToFlash(uint8_t index, size_t size) {
+void WriteTagToFlash(uint32_t uid, size_t size) {
     SpinOff(0);
     LED_A_ON();
     LED_B_ON();
@@ -140,68 +158,37 @@ void WriteTagToFlash(uint8_t index, size_t size) {
     LED_D_ON();
 
     uint32_t len = size;
-    uint32_t bytes_sent = 0;
-    uint32_t bytes_remaining = len;
-
     uint8_t data[(size * (16 * 64)) / 1024];
-    uint8_t buff[PAGESIZE];
 
     emlGetMem(data, 0, (size * 64) / 1024);
 
-    if (!FlashInit()) {
-        return;
-    }
+    char dest[SPIFFS_OBJ_NAME_LEN];
+    uint8_t buid[4];
+    num_to_bytes(uid, 4, buid);
+    sprintf(dest, "hf_colin/mf_%02x%02x%02x%02x.bin", buid[0], buid[1], buid[2], buid[3]);
 
-    Flash_CheckBusy(BUSY_TIMEOUT);
-    Flash_WriteEnable();
-    Flash_Erase4k(0, 0);
+    // TODO : by using safe function for multiple writes we are both breaking cache mecanisms and making useless and
+    // unoptimized mount operations we should manage at out level the mount status before and after the whole standalone
+    // mode
+    rdv40_spiffs_write((char *)dest, (uint8_t *)data, len, RDV40_SPIFFS_SAFETY_SAFE);
+    // lastag will only contain filename/path to last written tag file so we don't loose time or space.
+    rdv40_spiffs_make_symlink((char *)dest, (char *)HFCOLIN_LASTTAG_SYMLINK, RDV40_SPIFFS_SAFETY_SAFE);
 
-    uint32_t start_time = GetTickCount();
-    uint32_t delta_time = 0;
-
-    while (bytes_remaining > 0) {
-        Flash_CheckBusy(BUSY_TIMEOUT);
-        Flash_WriteEnable();
-
-        uint32_t bytes_in_packet = MIN(FLASH_MEM_BLOCK_SIZE, bytes_remaining);
-
-        memcpy(buff, data + bytes_sent, bytes_in_packet);
-
-        bytes_remaining -= bytes_in_packet;
-        uint16_t res = Flash_WriteDataCont(bytes_sent + (index * size), buff, bytes_in_packet);
-        bytes_sent += bytes_in_packet;
-
-        uint8_t isok = (res == bytes_in_packet) ? 1 : 0;
-
-        if (!isok) {
-            DbprintfEx(FLAG_NEWLINE, "FlashMem write FAILEd [offset %u]", bytes_sent);
-            cjSetCursLeft();
-            SpinOff(100);
-            return;
-        }
-
-        LED_A_INV();
-        LED_B_INV();
-        LED_C_INV();
-        LED_D_INV();
-    }
-    delta_time = GetTickCountDelta(start_time);
-
-    DbprintfEx(FLAG_NEWLINE, "[OK] TAG WRITTEN TO FLASH ! [0-to offset %u]", bytes_sent);
+    DbprintfEx(FLAG_NEWLINE, "[OK] TAG WRITTEN TO FLASH !");
     cjSetCursLeft();
-    DbprintfEx(FLAG_NEWLINE, "%s[IN]%s %s%dms%s for TAG_FLASH_WRITE", _XGREEN_, _XWHITE_, _XYELLOW_, delta_time, _XWHITE_);
-    cjSetCursLeft();
-    FlashStop();
     SpinOff(0);
     return;
 }
 
-void ModInfo(void) {
-    DbpString("  HF Mifare ultra fast sniff/sim/clone - aka VIGIKPWN (Colin Brigato)");
-}
+void ModInfo(void) { DbpString("  HF Mifare ultra fast sniff/sim/clone - aka VIGIKPWN (Colin Brigato)"); }
 
 void RunMod() {
     StandAloneMode();
+
+    add_schema(Schemas, Noralsy, &total_schemas);
+    add_schema(Schemas, InfiHexact, &total_schemas);
+    add_schema(Schemas, UrmetCaptive, &total_schemas);
+
     FpgaDownloadAndGo(FPGA_BITSTREAM_HF);
 
     currline = 20;
@@ -210,7 +197,7 @@ void RunMod() {
     memset(cjuid, 0, sizeof(cjuid));
     cjcuid = 0;
     uint8_t sectorsCnt = (MF1KSZ / MF1KSZSIZE);
-    uint64_t key64;           // Defines current key
+    uint64_t key64;    // Defines current key
     uint8_t *keyBlock; // Where the keys will be held in memory.
 
     /* VIGIK EXPIRED DUMP FOR STUDY
@@ -249,11 +236,11 @@ void RunMod() {
     ACCBITS : 796788[00]+VALUE
     */
 
-    //----------------------------
-    //   Set of keys to be used.
-    //  This should cover ~98% of
-    //  French VIGIK system @2017
-    //----------------------------
+//----------------------------
+//   Set of keys to be used.
+//  This should cover ~98% of
+//  French VIGIK system @2017
+//----------------------------
 
 #define STKEYS 37
 
@@ -350,7 +337,7 @@ failtag:
     SpinOff(50);
     LED_A_ON();
     uint8_t ticker = 0;
-    //while (!BUTTON_PRESS() && !iso14443a_select_card(cjuid, NULL, &cjcuid, true, 0, true))
+    // while (!BUTTON_PRESS() && !iso14443a_select_card(cjuid, NULL, &cjcuid, true, 0, true))
     while (!iso14443a_select_card(cjuid, &p_card, &cjcuid, true, 0, true)) {
         WDT_HIT();
 
@@ -424,7 +411,7 @@ failtag:
     //-----------------------------------------------------------------------------
     // also we could avoid first UID check for every block
 
-    // then let’s expose this “optimal case” of “well known vigik schemes” :
+    // then let's expose this optimal case of well known vigik schemes :
     for (uint8_t type = 0; type < 2 && !err && !trapped; type++) {
         for (int sec = 0; sec < sectorsCnt && !err && !trapped; ++sec) {
             key = cjat91_saMifareChkKeys(sec * 4, type, NULL, size, &keyBlock[0], &key64);
@@ -432,8 +419,8 @@ failtag:
             if (key == -1) {
                 err = 1;
                 allKeysFound = false;
-                // used in “portable” imlementation on microcontroller: it reports back the fail and open the standalone lock
-                // reply_old(CMD_CJB_FSMSTATE_MENU, 0, 0, 0, 0, 0);
+                // used in portable imlementation on microcontroller: it reports back the fail and open the standalone
+                // lock reply_old(CMD_CJB_FSMSTATE_MENU, 0, 0, 0, 0, 0);
                 break;
             } else if (key == -2) {
                 err = 1; // Can't select card.
@@ -448,10 +435,10 @@ failtag:
                 cjSetCursRight();
                 DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %012" PRIx64 " ; TYP: %i", sec, key64, type);
                 /*reply_old(CMD_CJB_INFORM_CLIENT_KEY, 12, sec, type, tosendkey, 12);*/
-                switch (key64) {
-                    /////////////////////////////////////////////////////////
-                    // COMMON SCHEME 1  : INFINITRON/HEXACT
-                    case 0x484558414354:
+
+                for (int i = 0; i < total_schemas; i++) {
+                    if (key64 == Schemas[i].trigger) {
+
                         cjSetCursLeft();
                         DbprintfEx(FLAG_NEWLINE, "%s>>>>>>>>>>>>!*STOP*!<<<<<<<<<<<<<<%s", _XRED_, _XWHITE_);
                         cjSetCursLeft();
@@ -459,232 +446,36 @@ failtag:
                         DbprintfEx(FLAG_NEWLINE, "    .TAG SEEMS %sDETERMINISTIC%s.     ", _XGREEN_, _XWHITE_);
                         cjSetCursLeft();
 
-                        DbprintfEx(FLAG_NEWLINE, "%sDetected: %s INFI_HEXACT_VIGIK_TAG%s", _XORANGE_, _XCYAN_, _XWHITE_);
+                        DbprintfEx(FLAG_NEWLINE, "%sDetected: %s %s%s", _XORANGE_, _XCYAN_, Schemas[i].name, _XWHITE_);
                         cjSetCursLeft();
 
-                        DbprintfEx(FLAG_NEWLINE, "...%s[%sKey_derivation_schemeTest%s]%s...", _XYELLOW_, _XGREEN_, _XYELLOW_, _XGREEN_);
+                        DbprintfEx(FLAG_NEWLINE, "...%s[%sKey_derivation_schemeTest%s]%s...", _XYELLOW_, _XGREEN_,
+                                   _XYELLOW_, _XGREEN_);
                         cjSetCursLeft();
 
                         DbprintfEx(FLAG_NEWLINE, "%s>>>>>>>>>>>>!*DONE*!<<<<<<<<<<<<<<%s", _XGREEN_, _XWHITE_);
-                        ;
-                        // Type 0 / A first
+
                         uint16_t t = 0;
                         for (uint16_t s = 0; s < sectorsCnt; s++) {
-                            num_to_bytes(0x484558414354, 6, foundKey[t][s]);
-                            sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][s][0], foundKey[t][s][1], foundKey[t][s][2],
-                                    foundKey[t][s][3], foundKey[t][s][4], foundKey[t][s][5]);
+                            num_to_bytes(Schemas[i].keysA[s], 6, foundKey[t][s]);
+                            sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][s][0], foundKey[t][s][1],
+                                    foundKey[t][s][2], foundKey[t][s][3], foundKey[t][s][4], foundKey[t][s][5]);
                             cjSetCursRight();
                             DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", s, tosendkey, t);
                         }
                         t = 1;
-                        uint16_t sectorNo = 0;
-                        num_to_bytes(0xa22ae129c013, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 1;
-                        num_to_bytes(0x49fae4e3849f, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 2;
-                        num_to_bytes(0x38fcf33072e0, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 3;
-                        num_to_bytes(0x8ad5517b4b18, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 4;
-                        num_to_bytes(0x509359f131b1, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 5;
-                        num_to_bytes(0x6c78928e1317, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 6;
-                        num_to_bytes(0xaa0720018738, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 7;
-                        num_to_bytes(0xa6cac2886412, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 8;
-                        num_to_bytes(0x62d0c424ed8e, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 9;
-                        num_to_bytes(0xe64a986a5d94, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 10;
-                        num_to_bytes(0x8fa1d601d0a2, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 11;
-                        num_to_bytes(0x89347350bd36, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 12;
-                        num_to_bytes(0x66d2b7dc39ef, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 13;
-                        num_to_bytes(0x6bc1e1ae547d, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 14;
-                        num_to_bytes(0x22729a9bd40f, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        sectorNo = 15;
-                        num_to_bytes(0x484558414354, 6, foundKey[t][sectorNo]);
-                        sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][sectorNo][0], foundKey[t][sectorNo][1], foundKey[t][sectorNo][2],
-                                foundKey[t][sectorNo][3], foundKey[t][sectorNo][4], foundKey[t][sectorNo][5]);
-                        cjSetCursRight();
-
-                        DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", sectorNo, tosendkey, t);
-                        trapped = 1;
-                        break;
-                    ////////////////END OF SCHEME 1//////////////////////////////
-
-                    ///////////////////////////////////////
-                    // COMMON SCHEME 2  : URMET CAPTIVE / COGELEC!/?
-                    case 0x8829da9daf76:
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "%s>>>>>>>>>>>>!*STOP*!<<<<<<<<<<<<<<%s", _XRED_, _XWHITE_);
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "    .TAG SEEMS %sDETERMINISTIC%s.     ", _XGREEN_, _XWHITE_);
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "%sDetected :%sURMET_CAPTIVE_VIGIK_TAG%s", _XORANGE_, _XCYAN_, _XWHITE_);
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "...%s[%sKey_derivation_schemeTest%s]%s...", _XYELLOW_, _XGREEN_, _XYELLOW_, _XGREEN_);
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "%s>>>>>>>>>>>>!*DONE*!<<<<<<<<<<<<<<%s", _XGREEN_, _XWHITE_);
-                        cjSetCursLeft();
-
-                        // emlClearMem();
-                        // A very weak one...
-                        for (uint16_t i = 0; i < 2; i++) {
-                            for (uint16_t s = 0; s < sectorsCnt; s++) {
-                                num_to_bytes(key64, 6, foundKey[i][s]);
-                                sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x",
-                                        foundKey[i][s][0],
-                                        foundKey[i][s][1],
-                                        foundKey[i][s][2],
-                                        foundKey[i][s][3],
-                                        foundKey[i][s][4],
-                                        foundKey[i][s][5]
-                                       );
-                                cjSetCursRight();
-                                DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", s, tosendkey, i);
-                            }
-                        }
-                        trapped = 1;
-                        break;
-                    ////////////////END OF SCHEME 2//////////////////////////////
-
-                    ///////////////////////////////////////
-                    // COMMON SCHEME 3  : NORALSY "A-LARON & B-LARON . . . NORAL-B & NORAL-A"
-                    case 0x414c41524f4e: // Thumbs up to the guy who had the idea of such a "mnemotechnical" key pair
-                    case 0x424c41524f4e:
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "%s>>>>>>>>>>>>!*STOP*!<<<<<<<<<<<<<<%s", _XRED_, _XWHITE_);
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "    .TAG SEEMS %sDETERMINISTIC%s.     ", _XGREEN_, _XWHITE_);
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "%s  Detected :%sNORALSY_VIGIK_TAG %s", _XORANGE_, _XCYAN_, _XWHITE_);
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "...%s[%sKey_derivation_schemeTest%s]%s...", _XYELLOW_, _XGREEN_, _XYELLOW_, _XGREEN_);
-                        cjSetCursLeft();
-
-                        DbprintfEx(FLAG_NEWLINE, "%s>>>>>>>>>>>>!*DONE*!<<<<<<<<<<<<<<%s", _XGREEN_, _XWHITE_);
-
-                        t = 0;
                         for (uint16_t s = 0; s < sectorsCnt; s++) {
-                            num_to_bytes(0x414c41524f4e, 6, foundKey[t][s]);
-                            sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x",
-                                    foundKey[t][s][0],
-                                    foundKey[t][s][1],
-                                    foundKey[t][s][2],
-                                    foundKey[t][s][3],
-                                    foundKey[t][s][4],
-                                    foundKey[t][s][5]);
-                            cjSetCursRight();
-                            DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", s, tosendkey, t);
-                        }
-
-                        t = 1;
-                        for (uint16_t s = 0; s < sectorsCnt; s++) {
-                            num_to_bytes(0x424c41524f4e, 6, foundKey[t][s]);
-                            sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x",
-                                    foundKey[t][s][0],
-                                    foundKey[t][s][1],
-                                    foundKey[t][s][2],
-                                    foundKey[t][s][3],
-                                    foundKey[t][s][4],
-                                    foundKey[t][s][5]);
+                            num_to_bytes(Schemas[i].keysB[s], 6, foundKey[t][s]);
+                            sprintf(tosendkey, "%02x%02x%02x%02x%02x%02x", foundKey[t][s][0], foundKey[t][s][1],
+                                    foundKey[t][s][2], foundKey[t][s][3], foundKey[t][s][4], foundKey[t][s][5]);
                             cjSetCursRight();
                             DbprintfEx(FLAG_NEWLINE, "SEC: %02x ; KEY : %s ; TYP: %d", s, tosendkey, t);
                         }
                         trapped = 1;
                         break;
-                        ////////////////END OF SCHEME 3//////////////////////////////
+                    }
                 }
+
                 /* etc etc for testing schemes quick schemes */
             }
         }
@@ -740,7 +531,8 @@ failtag:
     delta_time = GetTickCountDelta(start_time);
     cjSetCursLeft();
 
-    DbprintfEx(FLAG_NEWLINE, "%s>>%s Time for VIGIK break :%s%dms%s", _XGREEN_, _XWHITE_, _XYELLOW_, delta_time, _XWHITE_);
+    DbprintfEx(FLAG_NEWLINE, "%s>>%s Time for VIGIK break :%s%dms%s", _XGREEN_, _XWHITE_, _XYELLOW_, delta_time,
+               _XWHITE_);
 
     vtsend_cursor_position_save(NULL);
     vtsend_set_attribute(NULL, 1);
@@ -750,10 +542,9 @@ failtag:
     cjSetCursLeft();
     cjSetCursLeft();
 
-    WriteTagToFlash(0, 1024);
+    WriteTagToFlash(cjcuid, 1024);
 
 readysim:
-    // SIM ?
     cjSetCursLeft();
 
     DbprintfEx(FLAG_NEWLINE, "-> We launch Emulation ->");
@@ -762,8 +553,9 @@ readysim:
     DbprintfEx(FLAG_NEWLINE, "%s!> HOLD ON : %s When you'll click, simm will stop", _XRED_, _XWHITE_);
     cjSetCursLeft();
 
-    DbprintfEx(FLAG_NEWLINE, "Then %s immediately %s we'll try to %s dump our emulator state%s \r\nin a %s chinese tag%s", _XRED_, _XWHITE_, _XYELLOW_, _XWHITE_,
-               _XCYAN_, _XWHITE_);
+    DbprintfEx(FLAG_NEWLINE,
+               "Then %s immediately %s we'll try to %s dump our emulator state%s \r\nin a %s chinese tag%s", _XRED_,
+               _XWHITE_, _XYELLOW_, _XWHITE_, _XCYAN_, _XWHITE_);
     cjSetCursLeft();
     cjSetCursLeft();
 
@@ -783,10 +575,18 @@ readysim:
         case 7:
             flags = FLAG_7B_UID_IN_DATA;
             break;
-        default:
+        case 4:
             flags = FLAG_4B_UID_IN_DATA;
             break;
+        default:
+            flags = FLAG_UID_IN_EMUL;
+            break;
     }
+
+    // Use UID, SAK, ATQA from EMUL, if uid not defined
+    // if ((flags & (FLAG_4B_UID_IN_DATA | FLAG_7B_UID_IN_DATA | FLAG_10B_UID_IN_DATA)) == 0) {
+    flags |= FLAG_UID_IN_EMUL;
+    //}
     Mifare1ksim(flags | FLAG_MF_1K, 0, cjuid);
     LED_C_OFF();
     SpinOff(50);
@@ -902,9 +702,10 @@ int e_MifareECardLoad(uint32_t numofsectors, uint8_t keytype) {
     return (isOK) ? PM3_SUCCESS : PM3_EUNDEF;
 }
 
-/* the chk function is a piwi’ed(tm) check that will try all keys for
+/* the chk function is a piwi'ed(tm) check that will try all keys for
 a particular sector. also no tracing no dbg */
-int cjat91_saMifareChkKeys(uint8_t blockNo, uint8_t keyType, bool clearTrace, uint8_t keyCount, uint8_t *datain, uint64_t *key) {
+int cjat91_saMifareChkKeys(uint8_t blockNo, uint8_t keyType, bool clearTrace, uint8_t keyCount, uint8_t *datain,
+                           uint64_t *key) {
     DBGLEVEL = DBG_NONE;
     iso14443a_setup(FPGA_HF_ISO14443A_READER_LISTEN);
     set_tracing(false);
@@ -1009,6 +810,7 @@ void saMifareMakeTag(void) {
     }
 }
 
+// TODO : make this work either for a Gen1a or for a block 0 direct write all transparently
 //-----------------------------------------------------------------------------
 // Matt's StandAlone mod.
 // Work with "magic Chinese" card (email him: ouyangweidaxian@live.cn)
@@ -1095,7 +897,8 @@ int saMifareCSetBlock(uint32_t arg0, uint32_t arg1, uint32_t arg2, uint8_t *data
             };
         }
 
-        if ((mifare_sendcmd_short(NULL, 0, 0xA0, blockNo, receivedAnswer, receivedAnswerPar, NULL) != 1) || (receivedAnswer[0] != 0x0a)) {
+        if ((mifare_sendcmd_short(NULL, 0, 0xA0, blockNo, receivedAnswer, receivedAnswerPar, NULL) != 1) ||
+                (receivedAnswer[0] != 0x0a)) {
             DbprintfEx(FLAG_NEWLINE, "write block send command error");
             break;
         };
diff --git a/armsrc/Standalone/hf_colin.h b/armsrc/Standalone/hf_colin.h
index 8c8e391e1..e669f0417 100644
--- a/armsrc/Standalone/hf_colin.h
+++ b/armsrc/Standalone/hf_colin.h
@@ -22,6 +22,7 @@
 #include "iso14443a.h"
 #include "protocols.h"
 #include "util.h"
+#include "pmflash.h"
 #include "standalone.h" // standalone definitions
 #include <stdbool.h>    // for bool
 #include <stdio.h>
@@ -29,6 +30,7 @@
 #include "vtsend.h"
 #include "apps.h"
 #include "printf.h"
+#include "spiffs.h"
 
 #define _XRED_ "\x1b[31m"
 #define _XGREEN_ "\x1b[32m"
@@ -43,7 +45,7 @@ int cjat91_saMifareChkKeys(uint8_t blockNo, uint8_t keyType, bool clearTrace, ui
 int e_MifareECardLoad(uint32_t numofsectors, uint8_t keytype);
 void saMifareMakeTag(void);
 int saMifareCSetBlock(uint32_t arg0, uint32_t arg1, uint32_t arg2, uint8_t *datain);
-void WriteTagToFlash(uint8_t index, size_t size);
+void WriteTagToFlash(uint32_t uid, size_t size);
 
 const char clearTerm[8] = {0x1b, 0x5b, 0x48, 0x1b, 0x5b, 0x32, 0x4a, '\0'};
 
diff --git a/armsrc/Standalone/hf_mattyrun.c b/armsrc/Standalone/hf_mattyrun.c
index 7747a49da..9ea0ba524 100644
--- a/armsrc/Standalone/hf_mattyrun.c
+++ b/armsrc/Standalone/hf_mattyrun.c
@@ -223,6 +223,7 @@ void RunMod() {
     */
     bool printKeys = false;         // Prints keys
     bool transferToEml = true;      // Transfer keys to emulator memory
+    bool ecfill = true;             // Fill emulator memory with cards content.
     bool simulation = true;         // Simulates an exact copy of the target tag
     bool fillFromEmulator = false;  // Dump emulator memory.
 
diff --git a/armsrc/Standalone/hf_young.c b/armsrc/Standalone/hf_young.c
index 3ef9e37cf..906d0b36c 100644
--- a/armsrc/Standalone/hf_young.c
+++ b/armsrc/Standalone/hf_young.c
@@ -8,14 +8,16 @@
 //-----------------------------------------------------------------------------
 // main code for HF standalone mode Mifare /sniff/emulation by Craig Young
 //-----------------------------------------------------------------------------
+
 #include "hf_young.h"
+#include "common.h"
 
 typedef struct {
     uint8_t uid[10];
     uint8_t uidlen;
     uint8_t atqa[2];
     uint8_t sak;
-} __attribute__((__packed__)) card_clone_t;
+} PACKED card_clone_t;
 
 
 void ModInfo(void) {
diff --git a/armsrc/Standalone/lf_icerun.c b/armsrc/Standalone/lf_icerun.c
index abc2daa4e..eafd4ca7d 100644
--- a/armsrc/Standalone/lf_icerun.c
+++ b/armsrc/Standalone/lf_icerun.c
@@ -13,7 +13,6 @@ void ModInfo(void) {
     DbpString("  LF skeleton mode -  aka IceRun (iceman)");
 }
 
-// samy's sniff and repeat routine for LF
 void RunMod() {
     StandAloneMode();
     Dbprintf("[=] LF skeleton code a.k.a IceRun started");
diff --git a/armsrc/Standalone/readme.md b/armsrc/Standalone/readme.md
index 8d2420def..9b9d45688 100644
--- a/armsrc/Standalone/readme.md
+++ b/armsrc/Standalone/readme.md
@@ -1,6 +1,6 @@
 # Standalone Modes
 
-This contains functionality for different StandAlone modes. The fullimage will be built given the correct compiler flags used. Build targets for these files are contained in `armsrc/Makefile` and `common/Makefile.hal`
+This contains functionality for different StandAlone modes. The fullimage will be built given the correct compiler flags used. Build targets for these files are contained in `Makefile.inc` and `Makefile.hal`
 
 If you want to implement a new standalone mode, you need to implement the methods provided in `standalone.h`.
 Have a look at the skeleton standalone mode called IceRun, in the files `lf_icerun.c lf_icerun.h`.
@@ -40,8 +40,6 @@ void RunMod(void) {
     }
 ````
 
-Each standalone mode needs to have its own compiler flag to be added in `armsrc/Makefile`.
-
 ## Naming your standalone mode
 
 We suggest that you follow these guidelines:
@@ -58,9 +56,9 @@ This leads to your next step, your DEFINE name needed in Makefile.
 `WITH_STANDALONE_LF_FOO`
 
 
-## Update COMMON/MAKEFILE.HAL
+## Update MAKEFILE.HAL
 
-Add your mode to the `common/Makefile.hal` help and modes list:
+Add your mode to the `Makefile.hal` help and modes list:
 ```
 +==========================================================+
 | STANDALONE      | DESCRIPTION                            |
@@ -74,8 +72,15 @@ STANDALONE_MODES := LF_SAMYRUN LF_ICERUN LF_PROXBRUTE LF_HIDBRUTE LF_FOO
 STANDALONE_MODES += HF_YOUNG HF_MATTYRUN HF_COLIN HF_BOG
 ```
 
-## Update ARMSRC/MAKEFILE
-Add your source code files like the following sample in the `armsrc/Makefile`
+If your mode is using one of the unique features of the RDV4, add it to the proper list:
+
+```
+STANDALONE_MODES_REQ_SMARTCARD :=
+STANDALONE_MODES_REQ_FLASH := HF_COLIN HF_BOG
+```
+
+## Update MAKEFILE.INC
+Add your source code files like the following sample in the `Makefile.inc`
 
 ```
 # WITH_STANDALONE_LF_ICERUN
@@ -100,7 +105,7 @@ void ModInfo(void) {
 ````
 
 ## Compiling your standalone mode
-Once all this is done, you and others can now easily compile different standalone modes by just selecting one of the standalone modes (list in `common/Makefile.hal` or ) , e.g.:
+Once all this is done, you and others can now easily compile different standalone modes by just selecting one of the standalone modes (list in `Makefile.hal` or ) , e.g.:
 
 - rename  Makefile.platform.sample -> Makefile.platform
 - edit the "STANDALONE" row inside Makefile.platform.  You need to uncomment it and add your standalone mode name
diff --git a/armsrc/aes.c b/armsrc/aes.c
deleted file mode 100644
index f8a36f28c..000000000
--- a/armsrc/aes.c
+++ /dev/null
@@ -1,1170 +0,0 @@
-#include "aes.h"
-
-static const unsigned int Te0[256] = {
-    0xc66363a5UL, 0xf87c7c84UL, 0xee777799UL, 0xf67b7b8dUL,
-    0xfff2f20dUL, 0xd66b6bbdUL, 0xde6f6fb1UL, 0x91c5c554UL,
-    0x60303050UL, 0x02010103UL, 0xce6767a9UL, 0x562b2b7dUL,
-    0xe7fefe19UL, 0xb5d7d762UL, 0x4dababe6UL, 0xec76769aUL,
-    0x8fcaca45UL, 0x1f82829dUL, 0x89c9c940UL, 0xfa7d7d87UL,
-    0xeffafa15UL, 0xb25959ebUL, 0x8e4747c9UL, 0xfbf0f00bUL,
-    0x41adadecUL, 0xb3d4d467UL, 0x5fa2a2fdUL, 0x45afafeaUL,
-    0x239c9cbfUL, 0x53a4a4f7UL, 0xe4727296UL, 0x9bc0c05bUL,
-    0x75b7b7c2UL, 0xe1fdfd1cUL, 0x3d9393aeUL, 0x4c26266aUL,
-    0x6c36365aUL, 0x7e3f3f41UL, 0xf5f7f702UL, 0x83cccc4fUL,
-    0x6834345cUL, 0x51a5a5f4UL, 0xd1e5e534UL, 0xf9f1f108UL,
-    0xe2717193UL, 0xabd8d873UL, 0x62313153UL, 0x2a15153fUL,
-    0x0804040cUL, 0x95c7c752UL, 0x46232365UL, 0x9dc3c35eUL,
-    0x30181828UL, 0x379696a1UL, 0x0a05050fUL, 0x2f9a9ab5UL,
-    0x0e070709UL, 0x24121236UL, 0x1b80809bUL, 0xdfe2e23dUL,
-    0xcdebeb26UL, 0x4e272769UL, 0x7fb2b2cdUL, 0xea75759fUL,
-    0x1209091bUL, 0x1d83839eUL, 0x582c2c74UL, 0x341a1a2eUL,
-    0x361b1b2dUL, 0xdc6e6eb2UL, 0xb45a5aeeUL, 0x5ba0a0fbUL,
-    0xa45252f6UL, 0x763b3b4dUL, 0xb7d6d661UL, 0x7db3b3ceUL,
-    0x5229297bUL, 0xdde3e33eUL, 0x5e2f2f71UL, 0x13848497UL,
-    0xa65353f5UL, 0xb9d1d168UL, 0x00000000UL, 0xc1eded2cUL,
-    0x40202060UL, 0xe3fcfc1fUL, 0x79b1b1c8UL, 0xb65b5bedUL,
-    0xd46a6abeUL, 0x8dcbcb46UL, 0x67bebed9UL, 0x7239394bUL,
-    0x944a4adeUL, 0x984c4cd4UL, 0xb05858e8UL, 0x85cfcf4aUL,
-    0xbbd0d06bUL, 0xc5efef2aUL, 0x4faaaae5UL, 0xedfbfb16UL,
-    0x864343c5UL, 0x9a4d4dd7UL, 0x66333355UL, 0x11858594UL,
-    0x8a4545cfUL, 0xe9f9f910UL, 0x04020206UL, 0xfe7f7f81UL,
-    0xa05050f0UL, 0x783c3c44UL, 0x259f9fbaUL, 0x4ba8a8e3UL,
-    0xa25151f3UL, 0x5da3a3feUL, 0x804040c0UL, 0x058f8f8aUL,
-    0x3f9292adUL, 0x219d9dbcUL, 0x70383848UL, 0xf1f5f504UL,
-    0x63bcbcdfUL, 0x77b6b6c1UL, 0xafdada75UL, 0x42212163UL,
-    0x20101030UL, 0xe5ffff1aUL, 0xfdf3f30eUL, 0xbfd2d26dUL,
-    0x81cdcd4cUL, 0x180c0c14UL, 0x26131335UL, 0xc3ecec2fUL,
-    0xbe5f5fe1UL, 0x359797a2UL, 0x884444ccUL, 0x2e171739UL,
-    0x93c4c457UL, 0x55a7a7f2UL, 0xfc7e7e82UL, 0x7a3d3d47UL,
-    0xc86464acUL, 0xba5d5de7UL, 0x3219192bUL, 0xe6737395UL,
-    0xc06060a0UL, 0x19818198UL, 0x9e4f4fd1UL, 0xa3dcdc7fUL,
-    0x44222266UL, 0x542a2a7eUL, 0x3b9090abUL, 0x0b888883UL,
-    0x8c4646caUL, 0xc7eeee29UL, 0x6bb8b8d3UL, 0x2814143cUL,
-    0xa7dede79UL, 0xbc5e5ee2UL, 0x160b0b1dUL, 0xaddbdb76UL,
-    0xdbe0e03bUL, 0x64323256UL, 0x743a3a4eUL, 0x140a0a1eUL,
-    0x924949dbUL, 0x0c06060aUL, 0x4824246cUL, 0xb85c5ce4UL,
-    0x9fc2c25dUL, 0xbdd3d36eUL, 0x43acacefUL, 0xc46262a6UL,
-    0x399191a8UL, 0x319595a4UL, 0xd3e4e437UL, 0xf279798bUL,
-    0xd5e7e732UL, 0x8bc8c843UL, 0x6e373759UL, 0xda6d6db7UL,
-    0x018d8d8cUL, 0xb1d5d564UL, 0x9c4e4ed2UL, 0x49a9a9e0UL,
-    0xd86c6cb4UL, 0xac5656faUL, 0xf3f4f407UL, 0xcfeaea25UL,
-    0xca6565afUL, 0xf47a7a8eUL, 0x47aeaee9UL, 0x10080818UL,
-    0x6fbabad5UL, 0xf0787888UL, 0x4a25256fUL, 0x5c2e2e72UL,
-    0x381c1c24UL, 0x57a6a6f1UL, 0x73b4b4c7UL, 0x97c6c651UL,
-    0xcbe8e823UL, 0xa1dddd7cUL, 0xe874749cUL, 0x3e1f1f21UL,
-    0x964b4bddUL, 0x61bdbddcUL, 0x0d8b8b86UL, 0x0f8a8a85UL,
-    0xe0707090UL, 0x7c3e3e42UL, 0x71b5b5c4UL, 0xcc6666aaUL,
-    0x904848d8UL, 0x06030305UL, 0xf7f6f601UL, 0x1c0e0e12UL,
-    0xc26161a3UL, 0x6a35355fUL, 0xae5757f9UL, 0x69b9b9d0UL,
-    0x17868691UL, 0x99c1c158UL, 0x3a1d1d27UL, 0x279e9eb9UL,
-    0xd9e1e138UL, 0xebf8f813UL, 0x2b9898b3UL, 0x22111133UL,
-    0xd26969bbUL, 0xa9d9d970UL, 0x078e8e89UL, 0x339494a7UL,
-    0x2d9b9bb6UL, 0x3c1e1e22UL, 0x15878792UL, 0xc9e9e920UL,
-    0x87cece49UL, 0xaa5555ffUL, 0x50282878UL, 0xa5dfdf7aUL,
-    0x038c8c8fUL, 0x59a1a1f8UL, 0x09898980UL, 0x1a0d0d17UL,
-    0x65bfbfdaUL, 0xd7e6e631UL, 0x844242c6UL, 0xd06868b8UL,
-    0x824141c3UL, 0x299999b0UL, 0x5a2d2d77UL, 0x1e0f0f11UL,
-    0x7bb0b0cbUL, 0xa85454fcUL, 0x6dbbbbd6UL, 0x2c16163aUL,
-};
-static const unsigned int Te1[256] = {
-    0xa5c66363UL, 0x84f87c7cUL, 0x99ee7777UL, 0x8df67b7bUL,
-    0x0dfff2f2UL, 0xbdd66b6bUL, 0xb1de6f6fUL, 0x5491c5c5UL,
-    0x50603030UL, 0x03020101UL, 0xa9ce6767UL, 0x7d562b2bUL,
-    0x19e7fefeUL, 0x62b5d7d7UL, 0xe64dababUL, 0x9aec7676UL,
-    0x458fcacaUL, 0x9d1f8282UL, 0x4089c9c9UL, 0x87fa7d7dUL,
-    0x15effafaUL, 0xebb25959UL, 0xc98e4747UL, 0x0bfbf0f0UL,
-    0xec41adadUL, 0x67b3d4d4UL, 0xfd5fa2a2UL, 0xea45afafUL,
-    0xbf239c9cUL, 0xf753a4a4UL, 0x96e47272UL, 0x5b9bc0c0UL,
-    0xc275b7b7UL, 0x1ce1fdfdUL, 0xae3d9393UL, 0x6a4c2626UL,
-    0x5a6c3636UL, 0x417e3f3fUL, 0x02f5f7f7UL, 0x4f83ccccUL,
-    0x5c683434UL, 0xf451a5a5UL, 0x34d1e5e5UL, 0x08f9f1f1UL,
-    0x93e27171UL, 0x73abd8d8UL, 0x53623131UL, 0x3f2a1515UL,
-    0x0c080404UL, 0x5295c7c7UL, 0x65462323UL, 0x5e9dc3c3UL,
-    0x28301818UL, 0xa1379696UL, 0x0f0a0505UL, 0xb52f9a9aUL,
-    0x090e0707UL, 0x36241212UL, 0x9b1b8080UL, 0x3ddfe2e2UL,
-    0x26cdebebUL, 0x694e2727UL, 0xcd7fb2b2UL, 0x9fea7575UL,
-    0x1b120909UL, 0x9e1d8383UL, 0x74582c2cUL, 0x2e341a1aUL,
-    0x2d361b1bUL, 0xb2dc6e6eUL, 0xeeb45a5aUL, 0xfb5ba0a0UL,
-    0xf6a45252UL, 0x4d763b3bUL, 0x61b7d6d6UL, 0xce7db3b3UL,
-    0x7b522929UL, 0x3edde3e3UL, 0x715e2f2fUL, 0x97138484UL,
-    0xf5a65353UL, 0x68b9d1d1UL, 0x00000000UL, 0x2cc1ededUL,
-    0x60402020UL, 0x1fe3fcfcUL, 0xc879b1b1UL, 0xedb65b5bUL,
-    0xbed46a6aUL, 0x468dcbcbUL, 0xd967bebeUL, 0x4b723939UL,
-    0xde944a4aUL, 0xd4984c4cUL, 0xe8b05858UL, 0x4a85cfcfUL,
-    0x6bbbd0d0UL, 0x2ac5efefUL, 0xe54faaaaUL, 0x16edfbfbUL,
-    0xc5864343UL, 0xd79a4d4dUL, 0x55663333UL, 0x94118585UL,
-    0xcf8a4545UL, 0x10e9f9f9UL, 0x06040202UL, 0x81fe7f7fUL,
-    0xf0a05050UL, 0x44783c3cUL, 0xba259f9fUL, 0xe34ba8a8UL,
-    0xf3a25151UL, 0xfe5da3a3UL, 0xc0804040UL, 0x8a058f8fUL,
-    0xad3f9292UL, 0xbc219d9dUL, 0x48703838UL, 0x04f1f5f5UL,
-    0xdf63bcbcUL, 0xc177b6b6UL, 0x75afdadaUL, 0x63422121UL,
-    0x30201010UL, 0x1ae5ffffUL, 0x0efdf3f3UL, 0x6dbfd2d2UL,
-    0x4c81cdcdUL, 0x14180c0cUL, 0x35261313UL, 0x2fc3ececUL,
-    0xe1be5f5fUL, 0xa2359797UL, 0xcc884444UL, 0x392e1717UL,
-    0x5793c4c4UL, 0xf255a7a7UL, 0x82fc7e7eUL, 0x477a3d3dUL,
-    0xacc86464UL, 0xe7ba5d5dUL, 0x2b321919UL, 0x95e67373UL,
-    0xa0c06060UL, 0x98198181UL, 0xd19e4f4fUL, 0x7fa3dcdcUL,
-    0x66442222UL, 0x7e542a2aUL, 0xab3b9090UL, 0x830b8888UL,
-    0xca8c4646UL, 0x29c7eeeeUL, 0xd36bb8b8UL, 0x3c281414UL,
-    0x79a7dedeUL, 0xe2bc5e5eUL, 0x1d160b0bUL, 0x76addbdbUL,
-    0x3bdbe0e0UL, 0x56643232UL, 0x4e743a3aUL, 0x1e140a0aUL,
-    0xdb924949UL, 0x0a0c0606UL, 0x6c482424UL, 0xe4b85c5cUL,
-    0x5d9fc2c2UL, 0x6ebdd3d3UL, 0xef43acacUL, 0xa6c46262UL,
-    0xa8399191UL, 0xa4319595UL, 0x37d3e4e4UL, 0x8bf27979UL,
-    0x32d5e7e7UL, 0x438bc8c8UL, 0x596e3737UL, 0xb7da6d6dUL,
-    0x8c018d8dUL, 0x64b1d5d5UL, 0xd29c4e4eUL, 0xe049a9a9UL,
-    0xb4d86c6cUL, 0xfaac5656UL, 0x07f3f4f4UL, 0x25cfeaeaUL,
-    0xafca6565UL, 0x8ef47a7aUL, 0xe947aeaeUL, 0x18100808UL,
-    0xd56fbabaUL, 0x88f07878UL, 0x6f4a2525UL, 0x725c2e2eUL,
-    0x24381c1cUL, 0xf157a6a6UL, 0xc773b4b4UL, 0x5197c6c6UL,
-    0x23cbe8e8UL, 0x7ca1ddddUL, 0x9ce87474UL, 0x213e1f1fUL,
-    0xdd964b4bUL, 0xdc61bdbdUL, 0x860d8b8bUL, 0x850f8a8aUL,
-    0x90e07070UL, 0x427c3e3eUL, 0xc471b5b5UL, 0xaacc6666UL,
-    0xd8904848UL, 0x05060303UL, 0x01f7f6f6UL, 0x121c0e0eUL,
-    0xa3c26161UL, 0x5f6a3535UL, 0xf9ae5757UL, 0xd069b9b9UL,
-    0x91178686UL, 0x5899c1c1UL, 0x273a1d1dUL, 0xb9279e9eUL,
-    0x38d9e1e1UL, 0x13ebf8f8UL, 0xb32b9898UL, 0x33221111UL,
-    0xbbd26969UL, 0x70a9d9d9UL, 0x89078e8eUL, 0xa7339494UL,
-    0xb62d9b9bUL, 0x223c1e1eUL, 0x92158787UL, 0x20c9e9e9UL,
-    0x4987ceceUL, 0xffaa5555UL, 0x78502828UL, 0x7aa5dfdfUL,
-    0x8f038c8cUL, 0xf859a1a1UL, 0x80098989UL, 0x171a0d0dUL,
-    0xda65bfbfUL, 0x31d7e6e6UL, 0xc6844242UL, 0xb8d06868UL,
-    0xc3824141UL, 0xb0299999UL, 0x775a2d2dUL, 0x111e0f0fUL,
-    0xcb7bb0b0UL, 0xfca85454UL, 0xd66dbbbbUL, 0x3a2c1616UL,
-};
-static const unsigned int Te2[256] = {
-    0x63a5c663UL, 0x7c84f87cUL, 0x7799ee77UL, 0x7b8df67bUL,
-    0xf20dfff2UL, 0x6bbdd66bUL, 0x6fb1de6fUL, 0xc55491c5UL,
-    0x30506030UL, 0x01030201UL, 0x67a9ce67UL, 0x2b7d562bUL,
-    0xfe19e7feUL, 0xd762b5d7UL, 0xabe64dabUL, 0x769aec76UL,
-    0xca458fcaUL, 0x829d1f82UL, 0xc94089c9UL, 0x7d87fa7dUL,
-    0xfa15effaUL, 0x59ebb259UL, 0x47c98e47UL, 0xf00bfbf0UL,
-    0xadec41adUL, 0xd467b3d4UL, 0xa2fd5fa2UL, 0xafea45afUL,
-    0x9cbf239cUL, 0xa4f753a4UL, 0x7296e472UL, 0xc05b9bc0UL,
-    0xb7c275b7UL, 0xfd1ce1fdUL, 0x93ae3d93UL, 0x266a4c26UL,
-    0x365a6c36UL, 0x3f417e3fUL, 0xf702f5f7UL, 0xcc4f83ccUL,
-    0x345c6834UL, 0xa5f451a5UL, 0xe534d1e5UL, 0xf108f9f1UL,
-    0x7193e271UL, 0xd873abd8UL, 0x31536231UL, 0x153f2a15UL,
-    0x040c0804UL, 0xc75295c7UL, 0x23654623UL, 0xc35e9dc3UL,
-    0x18283018UL, 0x96a13796UL, 0x050f0a05UL, 0x9ab52f9aUL,
-    0x07090e07UL, 0x12362412UL, 0x809b1b80UL, 0xe23ddfe2UL,
-    0xeb26cdebUL, 0x27694e27UL, 0xb2cd7fb2UL, 0x759fea75UL,
-    0x091b1209UL, 0x839e1d83UL, 0x2c74582cUL, 0x1a2e341aUL,
-    0x1b2d361bUL, 0x6eb2dc6eUL, 0x5aeeb45aUL, 0xa0fb5ba0UL,
-    0x52f6a452UL, 0x3b4d763bUL, 0xd661b7d6UL, 0xb3ce7db3UL,
-    0x297b5229UL, 0xe33edde3UL, 0x2f715e2fUL, 0x84971384UL,
-    0x53f5a653UL, 0xd168b9d1UL, 0x00000000UL, 0xed2cc1edUL,
-    0x20604020UL, 0xfc1fe3fcUL, 0xb1c879b1UL, 0x5bedb65bUL,
-    0x6abed46aUL, 0xcb468dcbUL, 0xbed967beUL, 0x394b7239UL,
-    0x4ade944aUL, 0x4cd4984cUL, 0x58e8b058UL, 0xcf4a85cfUL,
-    0xd06bbbd0UL, 0xef2ac5efUL, 0xaae54faaUL, 0xfb16edfbUL,
-    0x43c58643UL, 0x4dd79a4dUL, 0x33556633UL, 0x85941185UL,
-    0x45cf8a45UL, 0xf910e9f9UL, 0x02060402UL, 0x7f81fe7fUL,
-    0x50f0a050UL, 0x3c44783cUL, 0x9fba259fUL, 0xa8e34ba8UL,
-    0x51f3a251UL, 0xa3fe5da3UL, 0x40c08040UL, 0x8f8a058fUL,
-    0x92ad3f92UL, 0x9dbc219dUL, 0x38487038UL, 0xf504f1f5UL,
-    0xbcdf63bcUL, 0xb6c177b6UL, 0xda75afdaUL, 0x21634221UL,
-    0x10302010UL, 0xff1ae5ffUL, 0xf30efdf3UL, 0xd26dbfd2UL,
-    0xcd4c81cdUL, 0x0c14180cUL, 0x13352613UL, 0xec2fc3ecUL,
-    0x5fe1be5fUL, 0x97a23597UL, 0x44cc8844UL, 0x17392e17UL,
-    0xc45793c4UL, 0xa7f255a7UL, 0x7e82fc7eUL, 0x3d477a3dUL,
-    0x64acc864UL, 0x5de7ba5dUL, 0x192b3219UL, 0x7395e673UL,
-    0x60a0c060UL, 0x81981981UL, 0x4fd19e4fUL, 0xdc7fa3dcUL,
-    0x22664422UL, 0x2a7e542aUL, 0x90ab3b90UL, 0x88830b88UL,
-    0x46ca8c46UL, 0xee29c7eeUL, 0xb8d36bb8UL, 0x143c2814UL,
-    0xde79a7deUL, 0x5ee2bc5eUL, 0x0b1d160bUL, 0xdb76addbUL,
-    0xe03bdbe0UL, 0x32566432UL, 0x3a4e743aUL, 0x0a1e140aUL,
-    0x49db9249UL, 0x060a0c06UL, 0x246c4824UL, 0x5ce4b85cUL,
-    0xc25d9fc2UL, 0xd36ebdd3UL, 0xacef43acUL, 0x62a6c462UL,
-    0x91a83991UL, 0x95a43195UL, 0xe437d3e4UL, 0x798bf279UL,
-    0xe732d5e7UL, 0xc8438bc8UL, 0x37596e37UL, 0x6db7da6dUL,
-    0x8d8c018dUL, 0xd564b1d5UL, 0x4ed29c4eUL, 0xa9e049a9UL,
-    0x6cb4d86cUL, 0x56faac56UL, 0xf407f3f4UL, 0xea25cfeaUL,
-    0x65afca65UL, 0x7a8ef47aUL, 0xaee947aeUL, 0x08181008UL,
-    0xbad56fbaUL, 0x7888f078UL, 0x256f4a25UL, 0x2e725c2eUL,
-    0x1c24381cUL, 0xa6f157a6UL, 0xb4c773b4UL, 0xc65197c6UL,
-    0xe823cbe8UL, 0xdd7ca1ddUL, 0x749ce874UL, 0x1f213e1fUL,
-    0x4bdd964bUL, 0xbddc61bdUL, 0x8b860d8bUL, 0x8a850f8aUL,
-    0x7090e070UL, 0x3e427c3eUL, 0xb5c471b5UL, 0x66aacc66UL,
-    0x48d89048UL, 0x03050603UL, 0xf601f7f6UL, 0x0e121c0eUL,
-    0x61a3c261UL, 0x355f6a35UL, 0x57f9ae57UL, 0xb9d069b9UL,
-    0x86911786UL, 0xc15899c1UL, 0x1d273a1dUL, 0x9eb9279eUL,
-    0xe138d9e1UL, 0xf813ebf8UL, 0x98b32b98UL, 0x11332211UL,
-    0x69bbd269UL, 0xd970a9d9UL, 0x8e89078eUL, 0x94a73394UL,
-    0x9bb62d9bUL, 0x1e223c1eUL, 0x87921587UL, 0xe920c9e9UL,
-    0xce4987ceUL, 0x55ffaa55UL, 0x28785028UL, 0xdf7aa5dfUL,
-    0x8c8f038cUL, 0xa1f859a1UL, 0x89800989UL, 0x0d171a0dUL,
-    0xbfda65bfUL, 0xe631d7e6UL, 0x42c68442UL, 0x68b8d068UL,
-    0x41c38241UL, 0x99b02999UL, 0x2d775a2dUL, 0x0f111e0fUL,
-    0xb0cb7bb0UL, 0x54fca854UL, 0xbbd66dbbUL, 0x163a2c16UL,
-};
-static const unsigned int Te3[256] = {
-    0x6363a5c6UL, 0x7c7c84f8UL, 0x777799eeUL, 0x7b7b8df6UL,
-    0xf2f20dffUL, 0x6b6bbdd6UL, 0x6f6fb1deUL, 0xc5c55491UL,
-    0x30305060UL, 0x01010302UL, 0x6767a9ceUL, 0x2b2b7d56UL,
-    0xfefe19e7UL, 0xd7d762b5UL, 0xababe64dUL, 0x76769aecUL,
-    0xcaca458fUL, 0x82829d1fUL, 0xc9c94089UL, 0x7d7d87faUL,
-    0xfafa15efUL, 0x5959ebb2UL, 0x4747c98eUL, 0xf0f00bfbUL,
-    0xadadec41UL, 0xd4d467b3UL, 0xa2a2fd5fUL, 0xafafea45UL,
-    0x9c9cbf23UL, 0xa4a4f753UL, 0x727296e4UL, 0xc0c05b9bUL,
-    0xb7b7c275UL, 0xfdfd1ce1UL, 0x9393ae3dUL, 0x26266a4cUL,
-    0x36365a6cUL, 0x3f3f417eUL, 0xf7f702f5UL, 0xcccc4f83UL,
-    0x34345c68UL, 0xa5a5f451UL, 0xe5e534d1UL, 0xf1f108f9UL,
-    0x717193e2UL, 0xd8d873abUL, 0x31315362UL, 0x15153f2aUL,
-    0x04040c08UL, 0xc7c75295UL, 0x23236546UL, 0xc3c35e9dUL,
-    0x18182830UL, 0x9696a137UL, 0x05050f0aUL, 0x9a9ab52fUL,
-    0x0707090eUL, 0x12123624UL, 0x80809b1bUL, 0xe2e23ddfUL,
-    0xebeb26cdUL, 0x2727694eUL, 0xb2b2cd7fUL, 0x75759feaUL,
-    0x09091b12UL, 0x83839e1dUL, 0x2c2c7458UL, 0x1a1a2e34UL,
-    0x1b1b2d36UL, 0x6e6eb2dcUL, 0x5a5aeeb4UL, 0xa0a0fb5bUL,
-    0x5252f6a4UL, 0x3b3b4d76UL, 0xd6d661b7UL, 0xb3b3ce7dUL,
-    0x29297b52UL, 0xe3e33eddUL, 0x2f2f715eUL, 0x84849713UL,
-    0x5353f5a6UL, 0xd1d168b9UL, 0x00000000UL, 0xeded2cc1UL,
-    0x20206040UL, 0xfcfc1fe3UL, 0xb1b1c879UL, 0x5b5bedb6UL,
-    0x6a6abed4UL, 0xcbcb468dUL, 0xbebed967UL, 0x39394b72UL,
-    0x4a4ade94UL, 0x4c4cd498UL, 0x5858e8b0UL, 0xcfcf4a85UL,
-    0xd0d06bbbUL, 0xefef2ac5UL, 0xaaaae54fUL, 0xfbfb16edUL,
-    0x4343c586UL, 0x4d4dd79aUL, 0x33335566UL, 0x85859411UL,
-    0x4545cf8aUL, 0xf9f910e9UL, 0x02020604UL, 0x7f7f81feUL,
-    0x5050f0a0UL, 0x3c3c4478UL, 0x9f9fba25UL, 0xa8a8e34bUL,
-    0x5151f3a2UL, 0xa3a3fe5dUL, 0x4040c080UL, 0x8f8f8a05UL,
-    0x9292ad3fUL, 0x9d9dbc21UL, 0x38384870UL, 0xf5f504f1UL,
-    0xbcbcdf63UL, 0xb6b6c177UL, 0xdada75afUL, 0x21216342UL,
-    0x10103020UL, 0xffff1ae5UL, 0xf3f30efdUL, 0xd2d26dbfUL,
-    0xcdcd4c81UL, 0x0c0c1418UL, 0x13133526UL, 0xecec2fc3UL,
-    0x5f5fe1beUL, 0x9797a235UL, 0x4444cc88UL, 0x1717392eUL,
-    0xc4c45793UL, 0xa7a7f255UL, 0x7e7e82fcUL, 0x3d3d477aUL,
-    0x6464acc8UL, 0x5d5de7baUL, 0x19192b32UL, 0x737395e6UL,
-    0x6060a0c0UL, 0x81819819UL, 0x4f4fd19eUL, 0xdcdc7fa3UL,
-    0x22226644UL, 0x2a2a7e54UL, 0x9090ab3bUL, 0x8888830bUL,
-    0x4646ca8cUL, 0xeeee29c7UL, 0xb8b8d36bUL, 0x14143c28UL,
-    0xdede79a7UL, 0x5e5ee2bcUL, 0x0b0b1d16UL, 0xdbdb76adUL,
-    0xe0e03bdbUL, 0x32325664UL, 0x3a3a4e74UL, 0x0a0a1e14UL,
-    0x4949db92UL, 0x06060a0cUL, 0x24246c48UL, 0x5c5ce4b8UL,
-    0xc2c25d9fUL, 0xd3d36ebdUL, 0xacacef43UL, 0x6262a6c4UL,
-    0x9191a839UL, 0x9595a431UL, 0xe4e437d3UL, 0x79798bf2UL,
-    0xe7e732d5UL, 0xc8c8438bUL, 0x3737596eUL, 0x6d6db7daUL,
-    0x8d8d8c01UL, 0xd5d564b1UL, 0x4e4ed29cUL, 0xa9a9e049UL,
-    0x6c6cb4d8UL, 0x5656faacUL, 0xf4f407f3UL, 0xeaea25cfUL,
-    0x6565afcaUL, 0x7a7a8ef4UL, 0xaeaee947UL, 0x08081810UL,
-    0xbabad56fUL, 0x787888f0UL, 0x25256f4aUL, 0x2e2e725cUL,
-    0x1c1c2438UL, 0xa6a6f157UL, 0xb4b4c773UL, 0xc6c65197UL,
-    0xe8e823cbUL, 0xdddd7ca1UL, 0x74749ce8UL, 0x1f1f213eUL,
-    0x4b4bdd96UL, 0xbdbddc61UL, 0x8b8b860dUL, 0x8a8a850fUL,
-    0x707090e0UL, 0x3e3e427cUL, 0xb5b5c471UL, 0x6666aaccUL,
-    0x4848d890UL, 0x03030506UL, 0xf6f601f7UL, 0x0e0e121cUL,
-    0x6161a3c2UL, 0x35355f6aUL, 0x5757f9aeUL, 0xb9b9d069UL,
-    0x86869117UL, 0xc1c15899UL, 0x1d1d273aUL, 0x9e9eb927UL,
-    0xe1e138d9UL, 0xf8f813ebUL, 0x9898b32bUL, 0x11113322UL,
-    0x6969bbd2UL, 0xd9d970a9UL, 0x8e8e8907UL, 0x9494a733UL,
-    0x9b9bb62dUL, 0x1e1e223cUL, 0x87879215UL, 0xe9e920c9UL,
-    0xcece4987UL, 0x5555ffaaUL, 0x28287850UL, 0xdfdf7aa5UL,
-    0x8c8c8f03UL, 0xa1a1f859UL, 0x89898009UL, 0x0d0d171aUL,
-    0xbfbfda65UL, 0xe6e631d7UL, 0x4242c684UL, 0x6868b8d0UL,
-    0x4141c382UL, 0x9999b029UL, 0x2d2d775aUL, 0x0f0f111eUL,
-    0xb0b0cb7bUL, 0x5454fca8UL, 0xbbbbd66dUL, 0x16163a2cUL,
-};
-static const unsigned int Te4[256] = {
-    0x63636363UL, 0x7c7c7c7cUL, 0x77777777UL, 0x7b7b7b7bUL,
-    0xf2f2f2f2UL, 0x6b6b6b6bUL, 0x6f6f6f6fUL, 0xc5c5c5c5UL,
-    0x30303030UL, 0x01010101UL, 0x67676767UL, 0x2b2b2b2bUL,
-    0xfefefefeUL, 0xd7d7d7d7UL, 0xababababUL, 0x76767676UL,
-    0xcacacacaUL, 0x82828282UL, 0xc9c9c9c9UL, 0x7d7d7d7dUL,
-    0xfafafafaUL, 0x59595959UL, 0x47474747UL, 0xf0f0f0f0UL,
-    0xadadadadUL, 0xd4d4d4d4UL, 0xa2a2a2a2UL, 0xafafafafUL,
-    0x9c9c9c9cUL, 0xa4a4a4a4UL, 0x72727272UL, 0xc0c0c0c0UL,
-    0xb7b7b7b7UL, 0xfdfdfdfdUL, 0x93939393UL, 0x26262626UL,
-    0x36363636UL, 0x3f3f3f3fUL, 0xf7f7f7f7UL, 0xccccccccUL,
-    0x34343434UL, 0xa5a5a5a5UL, 0xe5e5e5e5UL, 0xf1f1f1f1UL,
-    0x71717171UL, 0xd8d8d8d8UL, 0x31313131UL, 0x15151515UL,
-    0x04040404UL, 0xc7c7c7c7UL, 0x23232323UL, 0xc3c3c3c3UL,
-    0x18181818UL, 0x96969696UL, 0x05050505UL, 0x9a9a9a9aUL,
-    0x07070707UL, 0x12121212UL, 0x80808080UL, 0xe2e2e2e2UL,
-    0xebebebebUL, 0x27272727UL, 0xb2b2b2b2UL, 0x75757575UL,
-    0x09090909UL, 0x83838383UL, 0x2c2c2c2cUL, 0x1a1a1a1aUL,
-    0x1b1b1b1bUL, 0x6e6e6e6eUL, 0x5a5a5a5aUL, 0xa0a0a0a0UL,
-    0x52525252UL, 0x3b3b3b3bUL, 0xd6d6d6d6UL, 0xb3b3b3b3UL,
-    0x29292929UL, 0xe3e3e3e3UL, 0x2f2f2f2fUL, 0x84848484UL,
-    0x53535353UL, 0xd1d1d1d1UL, 0x00000000UL, 0xededededUL,
-    0x20202020UL, 0xfcfcfcfcUL, 0xb1b1b1b1UL, 0x5b5b5b5bUL,
-    0x6a6a6a6aUL, 0xcbcbcbcbUL, 0xbebebebeUL, 0x39393939UL,
-    0x4a4a4a4aUL, 0x4c4c4c4cUL, 0x58585858UL, 0xcfcfcfcfUL,
-    0xd0d0d0d0UL, 0xefefefefUL, 0xaaaaaaaaUL, 0xfbfbfbfbUL,
-    0x43434343UL, 0x4d4d4d4dUL, 0x33333333UL, 0x85858585UL,
-    0x45454545UL, 0xf9f9f9f9UL, 0x02020202UL, 0x7f7f7f7fUL,
-    0x50505050UL, 0x3c3c3c3cUL, 0x9f9f9f9fUL, 0xa8a8a8a8UL,
-    0x51515151UL, 0xa3a3a3a3UL, 0x40404040UL, 0x8f8f8f8fUL,
-    0x92929292UL, 0x9d9d9d9dUL, 0x38383838UL, 0xf5f5f5f5UL,
-    0xbcbcbcbcUL, 0xb6b6b6b6UL, 0xdadadadaUL, 0x21212121UL,
-    0x10101010UL, 0xffffffffUL, 0xf3f3f3f3UL, 0xd2d2d2d2UL,
-    0xcdcdcdcdUL, 0x0c0c0c0cUL, 0x13131313UL, 0xececececUL,
-    0x5f5f5f5fUL, 0x97979797UL, 0x44444444UL, 0x17171717UL,
-    0xc4c4c4c4UL, 0xa7a7a7a7UL, 0x7e7e7e7eUL, 0x3d3d3d3dUL,
-    0x64646464UL, 0x5d5d5d5dUL, 0x19191919UL, 0x73737373UL,
-    0x60606060UL, 0x81818181UL, 0x4f4f4f4fUL, 0xdcdcdcdcUL,
-    0x22222222UL, 0x2a2a2a2aUL, 0x90909090UL, 0x88888888UL,
-    0x46464646UL, 0xeeeeeeeeUL, 0xb8b8b8b8UL, 0x14141414UL,
-    0xdedededeUL, 0x5e5e5e5eUL, 0x0b0b0b0bUL, 0xdbdbdbdbUL,
-    0xe0e0e0e0UL, 0x32323232UL, 0x3a3a3a3aUL, 0x0a0a0a0aUL,
-    0x49494949UL, 0x06060606UL, 0x24242424UL, 0x5c5c5c5cUL,
-    0xc2c2c2c2UL, 0xd3d3d3d3UL, 0xacacacacUL, 0x62626262UL,
-    0x91919191UL, 0x95959595UL, 0xe4e4e4e4UL, 0x79797979UL,
-    0xe7e7e7e7UL, 0xc8c8c8c8UL, 0x37373737UL, 0x6d6d6d6dUL,
-    0x8d8d8d8dUL, 0xd5d5d5d5UL, 0x4e4e4e4eUL, 0xa9a9a9a9UL,
-    0x6c6c6c6cUL, 0x56565656UL, 0xf4f4f4f4UL, 0xeaeaeaeaUL,
-    0x65656565UL, 0x7a7a7a7aUL, 0xaeaeaeaeUL, 0x08080808UL,
-    0xbabababaUL, 0x78787878UL, 0x25252525UL, 0x2e2e2e2eUL,
-    0x1c1c1c1cUL, 0xa6a6a6a6UL, 0xb4b4b4b4UL, 0xc6c6c6c6UL,
-    0xe8e8e8e8UL, 0xddddddddUL, 0x74747474UL, 0x1f1f1f1fUL,
-    0x4b4b4b4bUL, 0xbdbdbdbdUL, 0x8b8b8b8bUL, 0x8a8a8a8aUL,
-    0x70707070UL, 0x3e3e3e3eUL, 0xb5b5b5b5UL, 0x66666666UL,
-    0x48484848UL, 0x03030303UL, 0xf6f6f6f6UL, 0x0e0e0e0eUL,
-    0x61616161UL, 0x35353535UL, 0x57575757UL, 0xb9b9b9b9UL,
-    0x86868686UL, 0xc1c1c1c1UL, 0x1d1d1d1dUL, 0x9e9e9e9eUL,
-    0xe1e1e1e1UL, 0xf8f8f8f8UL, 0x98989898UL, 0x11111111UL,
-    0x69696969UL, 0xd9d9d9d9UL, 0x8e8e8e8eUL, 0x94949494UL,
-    0x9b9b9b9bUL, 0x1e1e1e1eUL, 0x87878787UL, 0xe9e9e9e9UL,
-    0xcecececeUL, 0x55555555UL, 0x28282828UL, 0xdfdfdfdfUL,
-    0x8c8c8c8cUL, 0xa1a1a1a1UL, 0x89898989UL, 0x0d0d0d0dUL,
-    0xbfbfbfbfUL, 0xe6e6e6e6UL, 0x42424242UL, 0x68686868UL,
-    0x41414141UL, 0x99999999UL, 0x2d2d2d2dUL, 0x0f0f0f0fUL,
-    0xb0b0b0b0UL, 0x54545454UL, 0xbbbbbbbbUL, 0x16161616UL,
-};
-static const unsigned int Td0[256] = {
-    0x51f4a750UL, 0x7e416553UL, 0x1a17a4c3UL, 0x3a275e96UL,
-    0x3bab6bcbUL, 0x1f9d45f1UL, 0xacfa58abUL, 0x4be30393UL,
-    0x2030fa55UL, 0xad766df6UL, 0x88cc7691UL, 0xf5024c25UL,
-    0x4fe5d7fcUL, 0xc52acbd7UL, 0x26354480UL, 0xb562a38fUL,
-    0xdeb15a49UL, 0x25ba1b67UL, 0x45ea0e98UL, 0x5dfec0e1UL,
-    0xc32f7502UL, 0x814cf012UL, 0x8d4697a3UL, 0x6bd3f9c6UL,
-    0x038f5fe7UL, 0x15929c95UL, 0xbf6d7aebUL, 0x955259daUL,
-    0xd4be832dUL, 0x587421d3UL, 0x49e06929UL, 0x8ec9c844UL,
-    0x75c2896aUL, 0xf48e7978UL, 0x99583e6bUL, 0x27b971ddUL,
-    0xbee14fb6UL, 0xf088ad17UL, 0xc920ac66UL, 0x7dce3ab4UL,
-    0x63df4a18UL, 0xe51a3182UL, 0x97513360UL, 0x62537f45UL,
-    0xb16477e0UL, 0xbb6bae84UL, 0xfe81a01cUL, 0xf9082b94UL,
-    0x70486858UL, 0x8f45fd19UL, 0x94de6c87UL, 0x527bf8b7UL,
-    0xab73d323UL, 0x724b02e2UL, 0xe31f8f57UL, 0x6655ab2aUL,
-    0xb2eb2807UL, 0x2fb5c203UL, 0x86c57b9aUL, 0xd33708a5UL,
-    0x302887f2UL, 0x23bfa5b2UL, 0x02036abaUL, 0xed16825cUL,
-    0x8acf1c2bUL, 0xa779b492UL, 0xf307f2f0UL, 0x4e69e2a1UL,
-    0x65daf4cdUL, 0x0605bed5UL, 0xd134621fUL, 0xc4a6fe8aUL,
-    0x342e539dUL, 0xa2f355a0UL, 0x058ae132UL, 0xa4f6eb75UL,
-    0x0b83ec39UL, 0x4060efaaUL, 0x5e719f06UL, 0xbd6e1051UL,
-    0x3e218af9UL, 0x96dd063dUL, 0xdd3e05aeUL, 0x4de6bd46UL,
-    0x91548db5UL, 0x71c45d05UL, 0x0406d46fUL, 0x605015ffUL,
-    0x1998fb24UL, 0xd6bde997UL, 0x894043ccUL, 0x67d99e77UL,
-    0xb0e842bdUL, 0x07898b88UL, 0xe7195b38UL, 0x79c8eedbUL,
-    0xa17c0a47UL, 0x7c420fe9UL, 0xf8841ec9UL, 0x00000000UL,
-    0x09808683UL, 0x322bed48UL, 0x1e1170acUL, 0x6c5a724eUL,
-    0xfd0efffbUL, 0x0f853856UL, 0x3daed51eUL, 0x362d3927UL,
-    0x0a0fd964UL, 0x685ca621UL, 0x9b5b54d1UL, 0x24362e3aUL,
-    0x0c0a67b1UL, 0x9357e70fUL, 0xb4ee96d2UL, 0x1b9b919eUL,
-    0x80c0c54fUL, 0x61dc20a2UL, 0x5a774b69UL, 0x1c121a16UL,
-    0xe293ba0aUL, 0xc0a02ae5UL, 0x3c22e043UL, 0x121b171dUL,
-    0x0e090d0bUL, 0xf28bc7adUL, 0x2db6a8b9UL, 0x141ea9c8UL,
-    0x57f11985UL, 0xaf75074cUL, 0xee99ddbbUL, 0xa37f60fdUL,
-    0xf701269fUL, 0x5c72f5bcUL, 0x44663bc5UL, 0x5bfb7e34UL,
-    0x8b432976UL, 0xcb23c6dcUL, 0xb6edfc68UL, 0xb8e4f163UL,
-    0xd731dccaUL, 0x42638510UL, 0x13972240UL, 0x84c61120UL,
-    0x854a247dUL, 0xd2bb3df8UL, 0xaef93211UL, 0xc729a16dUL,
-    0x1d9e2f4bUL, 0xdcb230f3UL, 0x0d8652ecUL, 0x77c1e3d0UL,
-    0x2bb3166cUL, 0xa970b999UL, 0x119448faUL, 0x47e96422UL,
-    0xa8fc8cc4UL, 0xa0f03f1aUL, 0x567d2cd8UL, 0x223390efUL,
-    0x87494ec7UL, 0xd938d1c1UL, 0x8ccaa2feUL, 0x98d40b36UL,
-    0xa6f581cfUL, 0xa57ade28UL, 0xdab78e26UL, 0x3fadbfa4UL,
-    0x2c3a9de4UL, 0x5078920dUL, 0x6a5fcc9bUL, 0x547e4662UL,
-    0xf68d13c2UL, 0x90d8b8e8UL, 0x2e39f75eUL, 0x82c3aff5UL,
-    0x9f5d80beUL, 0x69d0937cUL, 0x6fd52da9UL, 0xcf2512b3UL,
-    0xc8ac993bUL, 0x10187da7UL, 0xe89c636eUL, 0xdb3bbb7bUL,
-    0xcd267809UL, 0x6e5918f4UL, 0xec9ab701UL, 0x834f9aa8UL,
-    0xe6956e65UL, 0xaaffe67eUL, 0x21bccf08UL, 0xef15e8e6UL,
-    0xbae79bd9UL, 0x4a6f36ceUL, 0xea9f09d4UL, 0x29b07cd6UL,
-    0x31a4b2afUL, 0x2a3f2331UL, 0xc6a59430UL, 0x35a266c0UL,
-    0x744ebc37UL, 0xfc82caa6UL, 0xe090d0b0UL, 0x33a7d815UL,
-    0xf104984aUL, 0x41ecdaf7UL, 0x7fcd500eUL, 0x1791f62fUL,
-    0x764dd68dUL, 0x43efb04dUL, 0xccaa4d54UL, 0xe49604dfUL,
-    0x9ed1b5e3UL, 0x4c6a881bUL, 0xc12c1fb8UL, 0x4665517fUL,
-    0x9d5eea04UL, 0x018c355dUL, 0xfa877473UL, 0xfb0b412eUL,
-    0xb3671d5aUL, 0x92dbd252UL, 0xe9105633UL, 0x6dd64713UL,
-    0x9ad7618cUL, 0x37a10c7aUL, 0x59f8148eUL, 0xeb133c89UL,
-    0xcea927eeUL, 0xb761c935UL, 0xe11ce5edUL, 0x7a47b13cUL,
-    0x9cd2df59UL, 0x55f2733fUL, 0x1814ce79UL, 0x73c737bfUL,
-    0x53f7cdeaUL, 0x5ffdaa5bUL, 0xdf3d6f14UL, 0x7844db86UL,
-    0xcaaff381UL, 0xb968c43eUL, 0x3824342cUL, 0xc2a3405fUL,
-    0x161dc372UL, 0xbce2250cUL, 0x283c498bUL, 0xff0d9541UL,
-    0x39a80171UL, 0x080cb3deUL, 0xd8b4e49cUL, 0x6456c190UL,
-    0x7bcb8461UL, 0xd532b670UL, 0x486c5c74UL, 0xd0b85742UL,
-};
-static const unsigned int Td1[256] = {
-    0x5051f4a7UL, 0x537e4165UL, 0xc31a17a4UL, 0x963a275eUL,
-    0xcb3bab6bUL, 0xf11f9d45UL, 0xabacfa58UL, 0x934be303UL,
-    0x552030faUL, 0xf6ad766dUL, 0x9188cc76UL, 0x25f5024cUL,
-    0xfc4fe5d7UL, 0xd7c52acbUL, 0x80263544UL, 0x8fb562a3UL,
-    0x49deb15aUL, 0x6725ba1bUL, 0x9845ea0eUL, 0xe15dfec0UL,
-    0x02c32f75UL, 0x12814cf0UL, 0xa38d4697UL, 0xc66bd3f9UL,
-    0xe7038f5fUL, 0x9515929cUL, 0xebbf6d7aUL, 0xda955259UL,
-    0x2dd4be83UL, 0xd3587421UL, 0x2949e069UL, 0x448ec9c8UL,
-    0x6a75c289UL, 0x78f48e79UL, 0x6b99583eUL, 0xdd27b971UL,
-    0xb6bee14fUL, 0x17f088adUL, 0x66c920acUL, 0xb47dce3aUL,
-    0x1863df4aUL, 0x82e51a31UL, 0x60975133UL, 0x4562537fUL,
-    0xe0b16477UL, 0x84bb6baeUL, 0x1cfe81a0UL, 0x94f9082bUL,
-    0x58704868UL, 0x198f45fdUL, 0x8794de6cUL, 0xb7527bf8UL,
-    0x23ab73d3UL, 0xe2724b02UL, 0x57e31f8fUL, 0x2a6655abUL,
-    0x07b2eb28UL, 0x032fb5c2UL, 0x9a86c57bUL, 0xa5d33708UL,
-    0xf2302887UL, 0xb223bfa5UL, 0xba02036aUL, 0x5ced1682UL,
-    0x2b8acf1cUL, 0x92a779b4UL, 0xf0f307f2UL, 0xa14e69e2UL,
-    0xcd65daf4UL, 0xd50605beUL, 0x1fd13462UL, 0x8ac4a6feUL,
-    0x9d342e53UL, 0xa0a2f355UL, 0x32058ae1UL, 0x75a4f6ebUL,
-    0x390b83ecUL, 0xaa4060efUL, 0x065e719fUL, 0x51bd6e10UL,
-    0xf93e218aUL, 0x3d96dd06UL, 0xaedd3e05UL, 0x464de6bdUL,
-    0xb591548dUL, 0x0571c45dUL, 0x6f0406d4UL, 0xff605015UL,
-    0x241998fbUL, 0x97d6bde9UL, 0xcc894043UL, 0x7767d99eUL,
-    0xbdb0e842UL, 0x8807898bUL, 0x38e7195bUL, 0xdb79c8eeUL,
-    0x47a17c0aUL, 0xe97c420fUL, 0xc9f8841eUL, 0x00000000UL,
-    0x83098086UL, 0x48322bedUL, 0xac1e1170UL, 0x4e6c5a72UL,
-    0xfbfd0effUL, 0x560f8538UL, 0x1e3daed5UL, 0x27362d39UL,
-    0x640a0fd9UL, 0x21685ca6UL, 0xd19b5b54UL, 0x3a24362eUL,
-    0xb10c0a67UL, 0x0f9357e7UL, 0xd2b4ee96UL, 0x9e1b9b91UL,
-    0x4f80c0c5UL, 0xa261dc20UL, 0x695a774bUL, 0x161c121aUL,
-    0x0ae293baUL, 0xe5c0a02aUL, 0x433c22e0UL, 0x1d121b17UL,
-    0x0b0e090dUL, 0xadf28bc7UL, 0xb92db6a8UL, 0xc8141ea9UL,
-    0x8557f119UL, 0x4caf7507UL, 0xbbee99ddUL, 0xfda37f60UL,
-    0x9ff70126UL, 0xbc5c72f5UL, 0xc544663bUL, 0x345bfb7eUL,
-    0x768b4329UL, 0xdccb23c6UL, 0x68b6edfcUL, 0x63b8e4f1UL,
-    0xcad731dcUL, 0x10426385UL, 0x40139722UL, 0x2084c611UL,
-    0x7d854a24UL, 0xf8d2bb3dUL, 0x11aef932UL, 0x6dc729a1UL,
-    0x4b1d9e2fUL, 0xf3dcb230UL, 0xec0d8652UL, 0xd077c1e3UL,
-    0x6c2bb316UL, 0x99a970b9UL, 0xfa119448UL, 0x2247e964UL,
-    0xc4a8fc8cUL, 0x1aa0f03fUL, 0xd8567d2cUL, 0xef223390UL,
-    0xc787494eUL, 0xc1d938d1UL, 0xfe8ccaa2UL, 0x3698d40bUL,
-    0xcfa6f581UL, 0x28a57adeUL, 0x26dab78eUL, 0xa43fadbfUL,
-    0xe42c3a9dUL, 0x0d507892UL, 0x9b6a5fccUL, 0x62547e46UL,
-    0xc2f68d13UL, 0xe890d8b8UL, 0x5e2e39f7UL, 0xf582c3afUL,
-    0xbe9f5d80UL, 0x7c69d093UL, 0xa96fd52dUL, 0xb3cf2512UL,
-    0x3bc8ac99UL, 0xa710187dUL, 0x6ee89c63UL, 0x7bdb3bbbUL,
-    0x09cd2678UL, 0xf46e5918UL, 0x01ec9ab7UL, 0xa8834f9aUL,
-    0x65e6956eUL, 0x7eaaffe6UL, 0x0821bccfUL, 0xe6ef15e8UL,
-    0xd9bae79bUL, 0xce4a6f36UL, 0xd4ea9f09UL, 0xd629b07cUL,
-    0xaf31a4b2UL, 0x312a3f23UL, 0x30c6a594UL, 0xc035a266UL,
-    0x37744ebcUL, 0xa6fc82caUL, 0xb0e090d0UL, 0x1533a7d8UL,
-    0x4af10498UL, 0xf741ecdaUL, 0x0e7fcd50UL, 0x2f1791f6UL,
-    0x8d764dd6UL, 0x4d43efb0UL, 0x54ccaa4dUL, 0xdfe49604UL,
-    0xe39ed1b5UL, 0x1b4c6a88UL, 0xb8c12c1fUL, 0x7f466551UL,
-    0x049d5eeaUL, 0x5d018c35UL, 0x73fa8774UL, 0x2efb0b41UL,
-    0x5ab3671dUL, 0x5292dbd2UL, 0x33e91056UL, 0x136dd647UL,
-    0x8c9ad761UL, 0x7a37a10cUL, 0x8e59f814UL, 0x89eb133cUL,
-    0xeecea927UL, 0x35b761c9UL, 0xede11ce5UL, 0x3c7a47b1UL,
-    0x599cd2dfUL, 0x3f55f273UL, 0x791814ceUL, 0xbf73c737UL,
-    0xea53f7cdUL, 0x5b5ffdaaUL, 0x14df3d6fUL, 0x867844dbUL,
-    0x81caaff3UL, 0x3eb968c4UL, 0x2c382434UL, 0x5fc2a340UL,
-    0x72161dc3UL, 0x0cbce225UL, 0x8b283c49UL, 0x41ff0d95UL,
-    0x7139a801UL, 0xde080cb3UL, 0x9cd8b4e4UL, 0x906456c1UL,
-    0x617bcb84UL, 0x70d532b6UL, 0x74486c5cUL, 0x42d0b857UL,
-};
-static const unsigned int Td2[256] = {
-    0xa75051f4UL, 0x65537e41UL, 0xa4c31a17UL, 0x5e963a27UL,
-    0x6bcb3babUL, 0x45f11f9dUL, 0x58abacfaUL, 0x03934be3UL,
-    0xfa552030UL, 0x6df6ad76UL, 0x769188ccUL, 0x4c25f502UL,
-    0xd7fc4fe5UL, 0xcbd7c52aUL, 0x44802635UL, 0xa38fb562UL,
-    0x5a49deb1UL, 0x1b6725baUL, 0x0e9845eaUL, 0xc0e15dfeUL,
-    0x7502c32fUL, 0xf012814cUL, 0x97a38d46UL, 0xf9c66bd3UL,
-    0x5fe7038fUL, 0x9c951592UL, 0x7aebbf6dUL, 0x59da9552UL,
-    0x832dd4beUL, 0x21d35874UL, 0x692949e0UL, 0xc8448ec9UL,
-    0x896a75c2UL, 0x7978f48eUL, 0x3e6b9958UL, 0x71dd27b9UL,
-    0x4fb6bee1UL, 0xad17f088UL, 0xac66c920UL, 0x3ab47dceUL,
-    0x4a1863dfUL, 0x3182e51aUL, 0x33609751UL, 0x7f456253UL,
-    0x77e0b164UL, 0xae84bb6bUL, 0xa01cfe81UL, 0x2b94f908UL,
-    0x68587048UL, 0xfd198f45UL, 0x6c8794deUL, 0xf8b7527bUL,
-    0xd323ab73UL, 0x02e2724bUL, 0x8f57e31fUL, 0xab2a6655UL,
-    0x2807b2ebUL, 0xc2032fb5UL, 0x7b9a86c5UL, 0x08a5d337UL,
-    0x87f23028UL, 0xa5b223bfUL, 0x6aba0203UL, 0x825ced16UL,
-    0x1c2b8acfUL, 0xb492a779UL, 0xf2f0f307UL, 0xe2a14e69UL,
-    0xf4cd65daUL, 0xbed50605UL, 0x621fd134UL, 0xfe8ac4a6UL,
-    0x539d342eUL, 0x55a0a2f3UL, 0xe132058aUL, 0xeb75a4f6UL,
-    0xec390b83UL, 0xefaa4060UL, 0x9f065e71UL, 0x1051bd6eUL,
-    0x8af93e21UL, 0x063d96ddUL, 0x05aedd3eUL, 0xbd464de6UL,
-    0x8db59154UL, 0x5d0571c4UL, 0xd46f0406UL, 0x15ff6050UL,
-    0xfb241998UL, 0xe997d6bdUL, 0x43cc8940UL, 0x9e7767d9UL,
-    0x42bdb0e8UL, 0x8b880789UL, 0x5b38e719UL, 0xeedb79c8UL,
-    0x0a47a17cUL, 0x0fe97c42UL, 0x1ec9f884UL, 0x00000000UL,
-    0x86830980UL, 0xed48322bUL, 0x70ac1e11UL, 0x724e6c5aUL,
-    0xfffbfd0eUL, 0x38560f85UL, 0xd51e3daeUL, 0x3927362dUL,
-    0xd9640a0fUL, 0xa621685cUL, 0x54d19b5bUL, 0x2e3a2436UL,
-    0x67b10c0aUL, 0xe70f9357UL, 0x96d2b4eeUL, 0x919e1b9bUL,
-    0xc54f80c0UL, 0x20a261dcUL, 0x4b695a77UL, 0x1a161c12UL,
-    0xba0ae293UL, 0x2ae5c0a0UL, 0xe0433c22UL, 0x171d121bUL,
-    0x0d0b0e09UL, 0xc7adf28bUL, 0xa8b92db6UL, 0xa9c8141eUL,
-    0x198557f1UL, 0x074caf75UL, 0xddbbee99UL, 0x60fda37fUL,
-    0x269ff701UL, 0xf5bc5c72UL, 0x3bc54466UL, 0x7e345bfbUL,
-    0x29768b43UL, 0xc6dccb23UL, 0xfc68b6edUL, 0xf163b8e4UL,
-    0xdccad731UL, 0x85104263UL, 0x22401397UL, 0x112084c6UL,
-    0x247d854aUL, 0x3df8d2bbUL, 0x3211aef9UL, 0xa16dc729UL,
-    0x2f4b1d9eUL, 0x30f3dcb2UL, 0x52ec0d86UL, 0xe3d077c1UL,
-    0x166c2bb3UL, 0xb999a970UL, 0x48fa1194UL, 0x642247e9UL,
-    0x8cc4a8fcUL, 0x3f1aa0f0UL, 0x2cd8567dUL, 0x90ef2233UL,
-    0x4ec78749UL, 0xd1c1d938UL, 0xa2fe8ccaUL, 0x0b3698d4UL,
-    0x81cfa6f5UL, 0xde28a57aUL, 0x8e26dab7UL, 0xbfa43fadUL,
-    0x9de42c3aUL, 0x920d5078UL, 0xcc9b6a5fUL, 0x4662547eUL,
-    0x13c2f68dUL, 0xb8e890d8UL, 0xf75e2e39UL, 0xaff582c3UL,
-    0x80be9f5dUL, 0x937c69d0UL, 0x2da96fd5UL, 0x12b3cf25UL,
-    0x993bc8acUL, 0x7da71018UL, 0x636ee89cUL, 0xbb7bdb3bUL,
-    0x7809cd26UL, 0x18f46e59UL, 0xb701ec9aUL, 0x9aa8834fUL,
-    0x6e65e695UL, 0xe67eaaffUL, 0xcf0821bcUL, 0xe8e6ef15UL,
-    0x9bd9bae7UL, 0x36ce4a6fUL, 0x09d4ea9fUL, 0x7cd629b0UL,
-    0xb2af31a4UL, 0x23312a3fUL, 0x9430c6a5UL, 0x66c035a2UL,
-    0xbc37744eUL, 0xcaa6fc82UL, 0xd0b0e090UL, 0xd81533a7UL,
-    0x984af104UL, 0xdaf741ecUL, 0x500e7fcdUL, 0xf62f1791UL,
-    0xd68d764dUL, 0xb04d43efUL, 0x4d54ccaaUL, 0x04dfe496UL,
-    0xb5e39ed1UL, 0x881b4c6aUL, 0x1fb8c12cUL, 0x517f4665UL,
-    0xea049d5eUL, 0x355d018cUL, 0x7473fa87UL, 0x412efb0bUL,
-    0x1d5ab367UL, 0xd25292dbUL, 0x5633e910UL, 0x47136dd6UL,
-    0x618c9ad7UL, 0x0c7a37a1UL, 0x148e59f8UL, 0x3c89eb13UL,
-    0x27eecea9UL, 0xc935b761UL, 0xe5ede11cUL, 0xb13c7a47UL,
-    0xdf599cd2UL, 0x733f55f2UL, 0xce791814UL, 0x37bf73c7UL,
-    0xcdea53f7UL, 0xaa5b5ffdUL, 0x6f14df3dUL, 0xdb867844UL,
-    0xf381caafUL, 0xc43eb968UL, 0x342c3824UL, 0x405fc2a3UL,
-    0xc372161dUL, 0x250cbce2UL, 0x498b283cUL, 0x9541ff0dUL,
-    0x017139a8UL, 0xb3de080cUL, 0xe49cd8b4UL, 0xc1906456UL,
-    0x84617bcbUL, 0xb670d532UL, 0x5c74486cUL, 0x5742d0b8UL,
-};
-static const unsigned int Td3[256] = {
-    0xf4a75051UL, 0x4165537eUL, 0x17a4c31aUL, 0x275e963aUL,
-    0xab6bcb3bUL, 0x9d45f11fUL, 0xfa58abacUL, 0xe303934bUL,
-    0x30fa5520UL, 0x766df6adUL, 0xcc769188UL, 0x024c25f5UL,
-    0xe5d7fc4fUL, 0x2acbd7c5UL, 0x35448026UL, 0x62a38fb5UL,
-    0xb15a49deUL, 0xba1b6725UL, 0xea0e9845UL, 0xfec0e15dUL,
-    0x2f7502c3UL, 0x4cf01281UL, 0x4697a38dUL, 0xd3f9c66bUL,
-    0x8f5fe703UL, 0x929c9515UL, 0x6d7aebbfUL, 0x5259da95UL,
-    0xbe832dd4UL, 0x7421d358UL, 0xe0692949UL, 0xc9c8448eUL,
-    0xc2896a75UL, 0x8e7978f4UL, 0x583e6b99UL, 0xb971dd27UL,
-    0xe14fb6beUL, 0x88ad17f0UL, 0x20ac66c9UL, 0xce3ab47dUL,
-    0xdf4a1863UL, 0x1a3182e5UL, 0x51336097UL, 0x537f4562UL,
-    0x6477e0b1UL, 0x6bae84bbUL, 0x81a01cfeUL, 0x082b94f9UL,
-    0x48685870UL, 0x45fd198fUL, 0xde6c8794UL, 0x7bf8b752UL,
-    0x73d323abUL, 0x4b02e272UL, 0x1f8f57e3UL, 0x55ab2a66UL,
-    0xeb2807b2UL, 0xb5c2032fUL, 0xc57b9a86UL, 0x3708a5d3UL,
-    0x2887f230UL, 0xbfa5b223UL, 0x036aba02UL, 0x16825cedUL,
-    0xcf1c2b8aUL, 0x79b492a7UL, 0x07f2f0f3UL, 0x69e2a14eUL,
-    0xdaf4cd65UL, 0x05bed506UL, 0x34621fd1UL, 0xa6fe8ac4UL,
-    0x2e539d34UL, 0xf355a0a2UL, 0x8ae13205UL, 0xf6eb75a4UL,
-    0x83ec390bUL, 0x60efaa40UL, 0x719f065eUL, 0x6e1051bdUL,
-    0x218af93eUL, 0xdd063d96UL, 0x3e05aeddUL, 0xe6bd464dUL,
-    0x548db591UL, 0xc45d0571UL, 0x06d46f04UL, 0x5015ff60UL,
-    0x98fb2419UL, 0xbde997d6UL, 0x4043cc89UL, 0xd99e7767UL,
-    0xe842bdb0UL, 0x898b8807UL, 0x195b38e7UL, 0xc8eedb79UL,
-    0x7c0a47a1UL, 0x420fe97cUL, 0x841ec9f8UL, 0x00000000UL,
-    0x80868309UL, 0x2bed4832UL, 0x1170ac1eUL, 0x5a724e6cUL,
-    0x0efffbfdUL, 0x8538560fUL, 0xaed51e3dUL, 0x2d392736UL,
-    0x0fd9640aUL, 0x5ca62168UL, 0x5b54d19bUL, 0x362e3a24UL,
-    0x0a67b10cUL, 0x57e70f93UL, 0xee96d2b4UL, 0x9b919e1bUL,
-    0xc0c54f80UL, 0xdc20a261UL, 0x774b695aUL, 0x121a161cUL,
-    0x93ba0ae2UL, 0xa02ae5c0UL, 0x22e0433cUL, 0x1b171d12UL,
-    0x090d0b0eUL, 0x8bc7adf2UL, 0xb6a8b92dUL, 0x1ea9c814UL,
-    0xf1198557UL, 0x75074cafUL, 0x99ddbbeeUL, 0x7f60fda3UL,
-    0x01269ff7UL, 0x72f5bc5cUL, 0x663bc544UL, 0xfb7e345bUL,
-    0x4329768bUL, 0x23c6dccbUL, 0xedfc68b6UL, 0xe4f163b8UL,
-    0x31dccad7UL, 0x63851042UL, 0x97224013UL, 0xc6112084UL,
-    0x4a247d85UL, 0xbb3df8d2UL, 0xf93211aeUL, 0x29a16dc7UL,
-    0x9e2f4b1dUL, 0xb230f3dcUL, 0x8652ec0dUL, 0xc1e3d077UL,
-    0xb3166c2bUL, 0x70b999a9UL, 0x9448fa11UL, 0xe9642247UL,
-    0xfc8cc4a8UL, 0xf03f1aa0UL, 0x7d2cd856UL, 0x3390ef22UL,
-    0x494ec787UL, 0x38d1c1d9UL, 0xcaa2fe8cUL, 0xd40b3698UL,
-    0xf581cfa6UL, 0x7ade28a5UL, 0xb78e26daUL, 0xadbfa43fUL,
-    0x3a9de42cUL, 0x78920d50UL, 0x5fcc9b6aUL, 0x7e466254UL,
-    0x8d13c2f6UL, 0xd8b8e890UL, 0x39f75e2eUL, 0xc3aff582UL,
-    0x5d80be9fUL, 0xd0937c69UL, 0xd52da96fUL, 0x2512b3cfUL,
-    0xac993bc8UL, 0x187da710UL, 0x9c636ee8UL, 0x3bbb7bdbUL,
-    0x267809cdUL, 0x5918f46eUL, 0x9ab701ecUL, 0x4f9aa883UL,
-    0x956e65e6UL, 0xffe67eaaUL, 0xbccf0821UL, 0x15e8e6efUL,
-    0xe79bd9baUL, 0x6f36ce4aUL, 0x9f09d4eaUL, 0xb07cd629UL,
-    0xa4b2af31UL, 0x3f23312aUL, 0xa59430c6UL, 0xa266c035UL,
-    0x4ebc3774UL, 0x82caa6fcUL, 0x90d0b0e0UL, 0xa7d81533UL,
-    0x04984af1UL, 0xecdaf741UL, 0xcd500e7fUL, 0x91f62f17UL,
-    0x4dd68d76UL, 0xefb04d43UL, 0xaa4d54ccUL, 0x9604dfe4UL,
-    0xd1b5e39eUL, 0x6a881b4cUL, 0x2c1fb8c1UL, 0x65517f46UL,
-    0x5eea049dUL, 0x8c355d01UL, 0x877473faUL, 0x0b412efbUL,
-    0x671d5ab3UL, 0xdbd25292UL, 0x105633e9UL, 0xd647136dUL,
-    0xd7618c9aUL, 0xa10c7a37UL, 0xf8148e59UL, 0x133c89ebUL,
-    0xa927eeceUL, 0x61c935b7UL, 0x1ce5ede1UL, 0x47b13c7aUL,
-    0xd2df599cUL, 0xf2733f55UL, 0x14ce7918UL, 0xc737bf73UL,
-    0xf7cdea53UL, 0xfdaa5b5fUL, 0x3d6f14dfUL, 0x44db8678UL,
-    0xaff381caUL, 0x68c43eb9UL, 0x24342c38UL, 0xa3405fc2UL,
-    0x1dc37216UL, 0xe2250cbcUL, 0x3c498b28UL, 0x0d9541ffUL,
-    0xa8017139UL, 0x0cb3de08UL, 0xb4e49cd8UL, 0x56c19064UL,
-    0xcb84617bUL, 0x32b670d5UL, 0x6c5c7448UL, 0xb85742d0UL,
-};
-static const unsigned int Td4[256] = {
-    0x52525252UL, 0x09090909UL, 0x6a6a6a6aUL, 0xd5d5d5d5UL,
-    0x30303030UL, 0x36363636UL, 0xa5a5a5a5UL, 0x38383838UL,
-    0xbfbfbfbfUL, 0x40404040UL, 0xa3a3a3a3UL, 0x9e9e9e9eUL,
-    0x81818181UL, 0xf3f3f3f3UL, 0xd7d7d7d7UL, 0xfbfbfbfbUL,
-    0x7c7c7c7cUL, 0xe3e3e3e3UL, 0x39393939UL, 0x82828282UL,
-    0x9b9b9b9bUL, 0x2f2f2f2fUL, 0xffffffffUL, 0x87878787UL,
-    0x34343434UL, 0x8e8e8e8eUL, 0x43434343UL, 0x44444444UL,
-    0xc4c4c4c4UL, 0xdedededeUL, 0xe9e9e9e9UL, 0xcbcbcbcbUL,
-    0x54545454UL, 0x7b7b7b7bUL, 0x94949494UL, 0x32323232UL,
-    0xa6a6a6a6UL, 0xc2c2c2c2UL, 0x23232323UL, 0x3d3d3d3dUL,
-    0xeeeeeeeeUL, 0x4c4c4c4cUL, 0x95959595UL, 0x0b0b0b0bUL,
-    0x42424242UL, 0xfafafafaUL, 0xc3c3c3c3UL, 0x4e4e4e4eUL,
-    0x08080808UL, 0x2e2e2e2eUL, 0xa1a1a1a1UL, 0x66666666UL,
-    0x28282828UL, 0xd9d9d9d9UL, 0x24242424UL, 0xb2b2b2b2UL,
-    0x76767676UL, 0x5b5b5b5bUL, 0xa2a2a2a2UL, 0x49494949UL,
-    0x6d6d6d6dUL, 0x8b8b8b8bUL, 0xd1d1d1d1UL, 0x25252525UL,
-    0x72727272UL, 0xf8f8f8f8UL, 0xf6f6f6f6UL, 0x64646464UL,
-    0x86868686UL, 0x68686868UL, 0x98989898UL, 0x16161616UL,
-    0xd4d4d4d4UL, 0xa4a4a4a4UL, 0x5c5c5c5cUL, 0xccccccccUL,
-    0x5d5d5d5dUL, 0x65656565UL, 0xb6b6b6b6UL, 0x92929292UL,
-    0x6c6c6c6cUL, 0x70707070UL, 0x48484848UL, 0x50505050UL,
-    0xfdfdfdfdUL, 0xededededUL, 0xb9b9b9b9UL, 0xdadadadaUL,
-    0x5e5e5e5eUL, 0x15151515UL, 0x46464646UL, 0x57575757UL,
-    0xa7a7a7a7UL, 0x8d8d8d8dUL, 0x9d9d9d9dUL, 0x84848484UL,
-    0x90909090UL, 0xd8d8d8d8UL, 0xababababUL, 0x00000000UL,
-    0x8c8c8c8cUL, 0xbcbcbcbcUL, 0xd3d3d3d3UL, 0x0a0a0a0aUL,
-    0xf7f7f7f7UL, 0xe4e4e4e4UL, 0x58585858UL, 0x05050505UL,
-    0xb8b8b8b8UL, 0xb3b3b3b3UL, 0x45454545UL, 0x06060606UL,
-    0xd0d0d0d0UL, 0x2c2c2c2cUL, 0x1e1e1e1eUL, 0x8f8f8f8fUL,
-    0xcacacacaUL, 0x3f3f3f3fUL, 0x0f0f0f0fUL, 0x02020202UL,
-    0xc1c1c1c1UL, 0xafafafafUL, 0xbdbdbdbdUL, 0x03030303UL,
-    0x01010101UL, 0x13131313UL, 0x8a8a8a8aUL, 0x6b6b6b6bUL,
-    0x3a3a3a3aUL, 0x91919191UL, 0x11111111UL, 0x41414141UL,
-    0x4f4f4f4fUL, 0x67676767UL, 0xdcdcdcdcUL, 0xeaeaeaeaUL,
-    0x97979797UL, 0xf2f2f2f2UL, 0xcfcfcfcfUL, 0xcecececeUL,
-    0xf0f0f0f0UL, 0xb4b4b4b4UL, 0xe6e6e6e6UL, 0x73737373UL,
-    0x96969696UL, 0xacacacacUL, 0x74747474UL, 0x22222222UL,
-    0xe7e7e7e7UL, 0xadadadadUL, 0x35353535UL, 0x85858585UL,
-    0xe2e2e2e2UL, 0xf9f9f9f9UL, 0x37373737UL, 0xe8e8e8e8UL,
-    0x1c1c1c1cUL, 0x75757575UL, 0xdfdfdfdfUL, 0x6e6e6e6eUL,
-    0x47474747UL, 0xf1f1f1f1UL, 0x1a1a1a1aUL, 0x71717171UL,
-    0x1d1d1d1dUL, 0x29292929UL, 0xc5c5c5c5UL, 0x89898989UL,
-    0x6f6f6f6fUL, 0xb7b7b7b7UL, 0x62626262UL, 0x0e0e0e0eUL,
-    0xaaaaaaaaUL, 0x18181818UL, 0xbebebebeUL, 0x1b1b1b1bUL,
-    0xfcfcfcfcUL, 0x56565656UL, 0x3e3e3e3eUL, 0x4b4b4b4bUL,
-    0xc6c6c6c6UL, 0xd2d2d2d2UL, 0x79797979UL, 0x20202020UL,
-    0x9a9a9a9aUL, 0xdbdbdbdbUL, 0xc0c0c0c0UL, 0xfefefefeUL,
-    0x78787878UL, 0xcdcdcdcdUL, 0x5a5a5a5aUL, 0xf4f4f4f4UL,
-    0x1f1f1f1fUL, 0xddddddddUL, 0xa8a8a8a8UL, 0x33333333UL,
-    0x88888888UL, 0x07070707UL, 0xc7c7c7c7UL, 0x31313131UL,
-    0xb1b1b1b1UL, 0x12121212UL, 0x10101010UL, 0x59595959UL,
-    0x27272727UL, 0x80808080UL, 0xececececUL, 0x5f5f5f5fUL,
-    0x60606060UL, 0x51515151UL, 0x7f7f7f7fUL, 0xa9a9a9a9UL,
-    0x19191919UL, 0xb5b5b5b5UL, 0x4a4a4a4aUL, 0x0d0d0d0dUL,
-    0x2d2d2d2dUL, 0xe5e5e5e5UL, 0x7a7a7a7aUL, 0x9f9f9f9fUL,
-    0x93939393UL, 0xc9c9c9c9UL, 0x9c9c9c9cUL, 0xefefefefUL,
-    0xa0a0a0a0UL, 0xe0e0e0e0UL, 0x3b3b3b3bUL, 0x4d4d4d4dUL,
-    0xaeaeaeaeUL, 0x2a2a2a2aUL, 0xf5f5f5f5UL, 0xb0b0b0b0UL,
-    0xc8c8c8c8UL, 0xebebebebUL, 0xbbbbbbbbUL, 0x3c3c3c3cUL,
-    0x83838383UL, 0x53535353UL, 0x99999999UL, 0x61616161UL,
-    0x17171717UL, 0x2b2b2b2bUL, 0x04040404UL, 0x7e7e7e7eUL,
-    0xbabababaUL, 0x77777777UL, 0xd6d6d6d6UL, 0x26262626UL,
-    0xe1e1e1e1UL, 0x69696969UL, 0x14141414UL, 0x63636363UL,
-    0x55555555UL, 0x21212121UL, 0x0c0c0c0cUL, 0x7d7d7d7dUL,
-};
-static const unsigned int rcon[] = {
-    0x01000000UL, 0x02000000UL, 0x04000000UL, 0x08000000UL,
-    0x10000000UL, 0x20000000UL, 0x40000000UL, 0x80000000UL,
-    0x1B000000UL, 0x36000000UL,
-};
-
-#define GETU32(pt) (((unsigned int)(pt)[0] << 24) ^ \
-                    ((unsigned int)(pt)[1] << 16) ^ \
-                    ((unsigned int)(pt)[2] <<  8) ^ \
-                    ((unsigned int)(pt)[3]))
-
-#define PUTU32(ct, st) { (ct)[0] = (unsigned char)((st) >> 24); \
-        (ct)[1] = (unsigned char)((st) >> 16); \
-        (ct)[2] = (unsigned char)((st) >>  8); \
-        (ct)[3] = (unsigned char)(st); }
-
-/*
-* Expand the cipher key into the encryption key schedule and return the
-* number of rounds for the given cipher key size.
-*/
-int aes_setkey_enc(unsigned int rk[], const unsigned char cipherKey[], int keyBytes) {
-    int i = 0;
-    unsigned int temp;
-
-    rk[0] = GETU32(cipherKey);
-    rk[1] = GETU32(cipherKey +  4);
-    rk[2] = GETU32(cipherKey +  8);
-    rk[3] = GETU32(cipherKey + 12);
-    if (keyBytes == 16) { // 128 bits
-        for (;;) {
-            temp  = rk[3];
-            rk[4] = rk[0] ^
-                    (Te4[(temp >> 16) & 0xff] & 0xff000000) ^
-                    (Te4[(temp >>  8) & 0xff] & 0x00ff0000) ^
-                    (Te4[(temp) & 0xff] & 0x0000ff00) ^
-                    (Te4[(temp >> 24)       ] & 0x000000ff) ^
-                    rcon[i];
-            rk[5] = rk[1] ^ rk[4];
-            rk[6] = rk[2] ^ rk[5];
-            rk[7] = rk[3] ^ rk[6];
-            if (++i == 10) {
-                return 10;
-            }
-            rk += 4;
-        }
-    }
-    rk[4] = GETU32(cipherKey + 16);
-    rk[5] = GETU32(cipherKey + 20);
-    if (keyBytes == 24) { // 192 bits
-        for (;;) {
-            temp = rk[ 5];
-            rk[ 6] = rk[ 0] ^
-                     (Te4[(temp >> 16) & 0xff] & 0xff000000) ^
-                     (Te4[(temp >>  8) & 0xff] & 0x00ff0000) ^
-                     (Te4[(temp) & 0xff] & 0x0000ff00) ^
-                     (Te4[(temp >> 24)       ] & 0x000000ff) ^
-                     rcon[i];
-            rk[ 7] = rk[ 1] ^ rk[ 6];
-            rk[ 8] = rk[ 2] ^ rk[ 7];
-            rk[ 9] = rk[ 3] ^ rk[ 8];
-            if (++i == 8) {
-                return 12;
-            }
-            rk[10] = rk[ 4] ^ rk[ 9];
-            rk[11] = rk[ 5] ^ rk[10];
-            rk += 6;
-        }
-    }
-    rk[6] = GETU32(cipherKey + 24);
-    rk[7] = GETU32(cipherKey + 28);
-    if (keyBytes == 32) { // 256 bits
-        for (;;) {
-            temp = rk[ 7];
-            rk[ 8] = rk[ 0] ^
-                     (Te4[(temp >> 16) & 0xff] & 0xff000000) ^
-                     (Te4[(temp >>  8) & 0xff] & 0x00ff0000) ^
-                     (Te4[(temp) & 0xff] & 0x0000ff00) ^
-                     (Te4[(temp >> 24)       ] & 0x000000ff) ^
-                     rcon[i];
-            rk[ 9] = rk[ 1] ^ rk[ 8];
-            rk[10] = rk[ 2] ^ rk[ 9];
-            rk[11] = rk[ 3] ^ rk[10];
-            if (++i == 7) {
-                return 14;
-            }
-            temp = rk[11];
-            rk[12] = rk[ 4] ^
-                     (Te4[(temp >> 24)       ] & 0xff000000) ^
-                     (Te4[(temp >> 16) & 0xff] & 0x00ff0000) ^
-                     (Te4[(temp >>  8) & 0xff] & 0x0000ff00) ^
-                     (Te4[(temp) & 0xff] & 0x000000ff);
-            rk[13] = rk[ 5] ^ rk[12];
-            rk[14] = rk[ 6] ^ rk[13];
-            rk[15] = rk[ 7] ^ rk[14];
-
-            rk += 8;
-        }
-    }
-    return 0;
-}
-
-/*
-* Expand the cipher key into encryption and decryption key schedule and
-* return the number of rounds for the given cipher key size.
-*/
-int AesGenKeySched(unsigned int rk[], unsigned int rrk[], const unsigned char cipherKey[], int keyBytes) {
-    int Nr, i;
-
-    // expand the cipher key
-    Nr = aes_setkey_enc(rk, cipherKey, keyBytes);
-    // invert the order of the first round keys
-    rrk += Nr * 4;
-    rrk[0] = rk[0];
-    rrk[1] = rk[1];
-    rrk[2] = rk[2];
-    rrk[3] = rk[3];
-
-    /*
-     * apply the inverse MixColumn transform to all round keys but the first
-     * and the last
-     */
-    for (i = 1; i < Nr; i++) {
-        rrk -= 4;
-        rk += 4;
-        rrk[0] =
-            Td0[Te4[(rk[0] >> 24)       ] & 0xff] ^
-            Td1[Te4[(rk[0] >> 16) & 0xff] & 0xff] ^
-            Td2[Te4[(rk[0] >>  8) & 0xff] & 0xff] ^
-            Td3[Te4[(rk[0]) & 0xff] & 0xff];
-        rrk[1] =
-            Td0[Te4[(rk[1] >> 24)       ] & 0xff] ^
-            Td1[Te4[(rk[1] >> 16) & 0xff] & 0xff] ^
-            Td2[Te4[(rk[1] >>  8) & 0xff] & 0xff] ^
-            Td3[Te4[(rk[1]) & 0xff] & 0xff];
-        rrk[2] =
-            Td0[Te4[(rk[2] >> 24)       ] & 0xff] ^
-            Td1[Te4[(rk[2] >> 16) & 0xff] & 0xff] ^
-            Td2[Te4[(rk[2] >>  8) & 0xff] & 0xff] ^
-            Td3[Te4[(rk[2]) & 0xff] & 0xff];
-        rrk[3] =
-            Td0[Te4[(rk[3] >> 24)       ] & 0xff] ^
-            Td1[Te4[(rk[3] >> 16) & 0xff] & 0xff] ^
-            Td2[Te4[(rk[3] >>  8) & 0xff] & 0xff] ^
-            Td3[Te4[(rk[3]) & 0xff] & 0xff];
-    }
-    // invert the order of the last round keys
-    rrk -= 4;
-    rk += 4;
-    rrk[0] = rk[0];
-    rrk[1] = rk[1];
-    rrk[2] = rk[2];
-    rrk[3] = rk[3];
-
-    return Nr;
-}
-
-/*
-* Encrypt the plain text into cipher
-*/
-void AesEncBlk(AesCtx *pCtx, const unsigned char pt[], unsigned char ct[]) {
-    unsigned int s0, s1, s2, s3, t0, t1, t2, t3, *iv;
-    const unsigned int *rk;
-    int r;
-
-    rk = pCtx->Ek;
-    iv = pCtx->Iv;
-    /*
-     * map byte array block to cipher state
-     * and add initial round key:
-     */
-    s0 = GETU32(pt) ^ rk[0];
-    s1 = GETU32(pt +  4) ^ rk[1];
-    s2 = GETU32(pt +  8) ^ rk[2];
-    s3 = GETU32(pt + 12) ^ rk[3];
-    if (pCtx->Mode) {
-        s0 = s0 ^ iv[0];
-        s1 = s1 ^ iv[1];
-        s2 = s2 ^ iv[2];
-        s3 = s3 ^ iv[3];
-    }
-    /*
-     * Nr - 1 full rounds:
-     */
-    r = pCtx->Nr >> 1;
-    for (;;) {
-        t0 =
-            Te0[(s0 >> 24)       ] ^
-            Te1[(s1 >> 16) & 0xff] ^
-            Te2[(s2 >>  8) & 0xff] ^
-            Te3[(s3) & 0xff] ^
-            rk[4];
-        t1 =
-            Te0[(s1 >> 24)       ] ^
-            Te1[(s2 >> 16) & 0xff] ^
-            Te2[(s3 >>  8) & 0xff] ^
-            Te3[(s0) & 0xff] ^
-            rk[5];
-        t2 =
-            Te0[(s2 >> 24)       ] ^
-            Te1[(s3 >> 16) & 0xff] ^
-            Te2[(s0 >>  8) & 0xff] ^
-            Te3[(s1) & 0xff] ^
-            rk[6];
-        t3 =
-            Te0[(s3 >> 24)       ] ^
-            Te1[(s0 >> 16) & 0xff] ^
-            Te2[(s1 >>  8) & 0xff] ^
-            Te3[(s2) & 0xff] ^
-            rk[7];
-
-        rk += 8;
-        if (--r == 0) {
-            break;
-        }
-
-        s0 =
-            Te0[(t0 >> 24)       ] ^
-            Te1[(t1 >> 16) & 0xff] ^
-            Te2[(t2 >>  8) & 0xff] ^
-            Te3[(t3) & 0xff] ^
-            rk[0];
-        s1 =
-            Te0[(t1 >> 24)       ] ^
-            Te1[(t2 >> 16) & 0xff] ^
-            Te2[(t3 >>  8) & 0xff] ^
-            Te3[(t0) & 0xff] ^
-            rk[1];
-        s2 =
-            Te0[(t2 >> 24)       ] ^
-            Te1[(t3 >> 16) & 0xff] ^
-            Te2[(t0 >>  8) & 0xff] ^
-            Te3[(t1) & 0xff] ^
-            rk[2];
-        s3 =
-            Te0[(t3 >> 24)       ] ^
-            Te1[(t0 >> 16) & 0xff] ^
-            Te2[(t1 >>  8) & 0xff] ^
-            Te3[(t2) & 0xff] ^
-            rk[3];
-    }
-    /*
-     * apply last round and
-     * map cipher state to byte array block:
-     */
-    s0 =
-        (Te4[(t0 >> 24)       ] & 0xff000000) ^
-        (Te4[(t1 >> 16) & 0xff] & 0x00ff0000) ^
-        (Te4[(t2 >>  8) & 0xff] & 0x0000ff00) ^
-        (Te4[(t3) & 0xff] & 0x000000ff) ^
-        rk[0];
-    PUTU32(ct, s0);
-    s1 =
-        (Te4[(t1 >> 24)       ] & 0xff000000) ^
-        (Te4[(t2 >> 16) & 0xff] & 0x00ff0000) ^
-        (Te4[(t3 >>  8) & 0xff] & 0x0000ff00) ^
-        (Te4[(t0) & 0xff] & 0x000000ff) ^
-        rk[1];
-    PUTU32(ct +  4, s1);
-    s2 =
-        (Te4[(t2 >> 24)       ] & 0xff000000) ^
-        (Te4[(t3 >> 16) & 0xff] & 0x00ff0000) ^
-        (Te4[(t0 >>  8) & 0xff] & 0x0000ff00) ^
-        (Te4[(t1) & 0xff] & 0x000000ff) ^
-        rk[2];
-    PUTU32(ct +  8, s2);
-    s3 =
-        (Te4[(t3 >> 24)       ] & 0xff000000) ^
-        (Te4[(t0 >> 16) & 0xff] & 0x00ff0000) ^
-        (Te4[(t1 >>  8) & 0xff] & 0x0000ff00) ^
-        (Te4[(t2) & 0xff] & 0x000000ff) ^
-        rk[3];
-    PUTU32(ct + 12, s3);
-
-    if (pCtx->Mode) {
-        iv[0] = s0;
-        iv[1] = s1;
-        iv[2] = s2;
-        iv[3] = s3;
-    }
-}
-
-/*
-* Decrypt the cipher into plain text
-*/
-void AesDecBlk(AesCtx *pCtx, const unsigned char ct[], unsigned char pt[]) {
-    unsigned int s0, s1, s2, s3, t0, t1, t2, t3, v0, v1, v2, v3, *iv;
-    const unsigned int *rk;
-    int r;
-
-    rk = pCtx->Dk;
-    iv = pCtx->Iv;
-    /*
-     * map byte array block to cipher state
-     * and add initial round key:
-     */
-    v0 = GETU32(ct);
-    s0 = v0 ^ rk[0];
-    v1 = GETU32(ct +  4);
-    s1 = v1 ^ rk[1];
-    v2 = GETU32(ct +  8);
-    s2 = v2 ^ rk[2];
-    v3 = GETU32(ct + 12);
-    s3 = v3 ^ rk[3];
-    /*
-     * Nr - 1 full rounds:
-     */
-    r = pCtx->Nr >> 1;
-    for (;;) {
-        t0 =
-            Td0[(s0 >> 24)       ] ^
-            Td1[(s3 >> 16) & 0xff] ^
-            Td2[(s2 >>  8) & 0xff] ^
-            Td3[(s1) & 0xff] ^
-            rk[4];
-        t1 =
-            Td0[(s1 >> 24)       ] ^
-            Td1[(s0 >> 16) & 0xff] ^
-            Td2[(s3 >>  8) & 0xff] ^
-            Td3[(s2) & 0xff] ^
-            rk[5];
-        t2 =
-            Td0[(s2 >> 24)       ] ^
-            Td1[(s1 >> 16) & 0xff] ^
-            Td2[(s0 >>  8) & 0xff] ^
-            Td3[(s3) & 0xff] ^
-            rk[6];
-        t3 =
-            Td0[(s3 >> 24)       ] ^
-            Td1[(s2 >> 16) & 0xff] ^
-            Td2[(s1 >>  8) & 0xff] ^
-            Td3[(s0) & 0xff] ^
-            rk[7];
-
-        rk += 8;
-        if (--r == 0) {
-            break;
-        }
-
-        s0 =
-            Td0[(t0 >> 24)       ] ^
-            Td1[(t3 >> 16) & 0xff] ^
-            Td2[(t2 >>  8) & 0xff] ^
-            Td3[(t1) & 0xff] ^
-            rk[0];
-        s1 =
-            Td0[(t1 >> 24)       ] ^
-            Td1[(t0 >> 16) & 0xff] ^
-            Td2[(t3 >>  8) & 0xff] ^
-            Td3[(t2) & 0xff] ^
-            rk[1];
-        s2 =
-            Td0[(t2 >> 24)       ] ^
-            Td1[(t1 >> 16) & 0xff] ^
-            Td2[(t0 >>  8) & 0xff] ^
-            Td3[(t3) & 0xff] ^
-            rk[2];
-        s3 =
-            Td0[(t3 >> 24)       ] ^
-            Td1[(t2 >> 16) & 0xff] ^
-            Td2[(t1 >>  8) & 0xff] ^
-            Td3[(t0) & 0xff] ^
-            rk[3];
-    }
-    /*
-     * apply last round and
-     * map cipher state to byte array block:
-     */
-    s0 =
-        (Td4[(t0 >> 24)       ] & 0xff000000) ^
-        (Td4[(t3 >> 16) & 0xff] & 0x00ff0000) ^
-        (Td4[(t2 >>  8) & 0xff] & 0x0000ff00) ^
-        (Td4[(t1) & 0xff] & 0x000000ff) ^
-        rk[0];
-    s1 =
-        (Td4[(t1 >> 24)       ] & 0xff000000) ^
-        (Td4[(t0 >> 16) & 0xff] & 0x00ff0000) ^
-        (Td4[(t3 >>  8) & 0xff] & 0x0000ff00) ^
-        (Td4[(t2) & 0xff] & 0x000000ff) ^
-        rk[1];
-    s2 =
-        (Td4[(t2 >> 24)       ] & 0xff000000) ^
-        (Td4[(t1 >> 16) & 0xff] & 0x00ff0000) ^
-        (Td4[(t0 >>  8) & 0xff] & 0x0000ff00) ^
-        (Td4[(t3) & 0xff] & 0x000000ff) ^
-        rk[2];
-    s3 =
-        (Td4[(t3 >> 24)       ] & 0xff000000) ^
-        (Td4[(t2 >> 16) & 0xff] & 0x00ff0000) ^
-        (Td4[(t1 >>  8) & 0xff] & 0x0000ff00) ^
-        (Td4[(t0) & 0xff] & 0x000000ff) ^
-        rk[3];
-
-    if (pCtx->Mode) {
-        s0 = s0 ^ iv[0];
-        iv[0] = v0;
-        s1 = s1 ^ iv[1];
-        iv[1] = v1;
-        s2 = s2 ^ iv[2];
-        iv[2] = v2;
-        s3 = s3 ^ iv[3];
-        iv[3] = v3;
-    }
-
-    PUTU32(pt, s0);
-    PUTU32(pt +  4, s1);
-    PUTU32(pt +  8, s2);
-    PUTU32(pt + 12, s3);
-}
-
-//////////////////////////////////////////////////////////////////////////////
-// API functions                                                            //
-//////////////////////////////////////////////////////////////////////////////
-
-/*
-* initialize AES context
-*/
-int AesCtxIni(AesCtx *pCtx, unsigned char *pIV, unsigned char *pKey, unsigned int KeyLen, unsigned char Mode) {
-    if (pKey == 0 || pCtx == 0 || (KeyLen != KEY128 && KeyLen != KEY192 && KeyLen != KEY256))
-        return -1;
-
-    // generate key schedule
-    pCtx->Nr = AesGenKeySched(pCtx->Ek,  pCtx->Dk, pKey, KeyLen);
-
-    // initialize IV
-    if (pIV != 0) {
-        pCtx->Iv[0] = GETU32(pIV);
-        pCtx->Iv[1] = GETU32(pIV + 4);
-        pCtx->Iv[2] = GETU32(pIV + 8);
-        pCtx->Iv[3] = GETU32(pIV + 12);
-    }
-
-    // mode
-    pCtx->Mode = Mode;
-
-    return 0;
-}
-
-/*
-* Encrypt plain text
-*/
-int AesEncrypt(AesCtx *pCtx, unsigned char *pData, unsigned char *pCipher, unsigned int DataLen) {
-    int i;
-
-    if (pData == 0 || pCipher == 0 || pCtx == 0 || (DataLen & 0xf) != 0)
-        return -1;
-
-    for (i = 0; i < DataLen; i += BLOCKSZ) {
-        // encrypt block by block
-        AesEncBlk(pCtx, pData, pCipher);
-        pCipher += BLOCKSZ;
-        pData += BLOCKSZ;
-    }
-    return DataLen;
-}
-
-/*
-* Decrypt cipher
-*/
-int AesDecrypt(AesCtx *pCtx, unsigned char *pCipher, unsigned char *pData, unsigned int CipherLen) {
-    int i;
-
-    if (pData == 0 || pCipher == 0 || pCtx == 0 || (CipherLen & 0xf) != 0)
-        return -1;
-
-    for (i = 0; i < CipherLen; i += BLOCKSZ) {
-        // decrypt block by block
-        AesDecBlk(pCtx, pCipher, pData);
-        pCipher += BLOCKSZ;
-        pData += BLOCKSZ;
-    }
-    return CipherLen;
-}
-
-//////////////////////////////////////////////////////////////////////////////
-// Sample main program                                                      //
-//////////////////////////////////////////////////////////////////////////////
-
-#ifndef EMBEDDED
-
-#include <stdio.h>
-
-int main() {
-    AesCtx ctx;
-    unsigned char iv[] = "INI VECTINI VECT";
-    unsigned char key[] = "This is a sample AESKey";
-    unsigned char databuf[] = "Data : AES Test"; // must be in multiple of 16
-
-    // initialize context and encrypt data at one end
-
-    if (AesCtxIni(&ctx, iv, key, KEY128, CBC) < 0)
-        printf("init error\n");
-
-    if (AesEncrypt(&ctx, databuf, databuf, sizeof(databuf)) < 0)
-        printf("error in encryption\n");
-
-    // initialize context and decrypt cipher at other end
-
-    if (AesCtxIni(&ctx, iv, key, KEY128, CBC) < 0)
-        printf("init error\n");
-
-    if (AesDecrypt(&ctx, databuf, databuf, sizeof(databuf)) < 0)
-        printf("error in decryption\n");
-
-    printf("%s\n", databuf);
-
-    return 0;
-}
-#endif
diff --git a/armsrc/aes.h b/armsrc/aes.h
deleted file mode 100644
index aa9f0c582..000000000
--- a/armsrc/aes.h
+++ /dev/null
@@ -1,34 +0,0 @@
-/*
-* AES Cryptographic Algorithm Header File. Include this header file in
-* your source which uses these given APIs. (This source is kept under
-* public domain)
-*/
-#ifndef __AES_H
-#define __AES_H
-
-// AES context structure
-typedef struct {
-    unsigned int Ek[60];
-    unsigned int Dk[60];
-    unsigned int Iv[4];
-    unsigned char Nr;
-    unsigned char Mode;
-} AesCtx;
-
-// key length in bytes
-#define KEY128 16
-#define KEY192 24
-#define KEY256 32
-// block size in bytes
-#define BLOCKSZ 16
-// mode
-#define EBC 0
-#define CBC 1
-
-// AES API function prototype
-
-int AesCtxIni(AesCtx *pCtx, unsigned char *pIV, unsigned char *pKey, unsigned int KeyLen, unsigned char Mode);
-int AesEncrypt(AesCtx *pCtx, unsigned char *pData, unsigned char *pCipher, unsigned int DataLen);
-int AesDecrypt(AesCtx *pCtx, unsigned char *pCipher, unsigned char *pData, unsigned int CipherLen);
-
-#endif
diff --git a/armsrc/appmain.c b/armsrc/appmain.c
index 3213fbe14..02829c20c 100644
--- a/armsrc/appmain.c
+++ b/armsrc/appmain.c
@@ -13,6 +13,7 @@
 #include <inttypes.h>
 #include "usb_cdc.h"
 #include "proxmark3.h"
+#include "pmflash.h"
 #include "apps.h"
 #include "fpga.h"
 #include "util.h"
@@ -42,6 +43,7 @@
 
 #ifdef WITH_FLASH
 #include "flashmem.h"
+#include "spiffs.h"
 #endif
 
 //=============================================================================
@@ -347,9 +349,9 @@ void SendVersion(void) {
     strncat(VersionString, temp, sizeof(VersionString) - strlen(VersionString) - 1);
 
 #if defined(__clang__)
-        strncat(VersionString, "  compiled with Clang/LLVM "__VERSION__"\n", sizeof(VersionString) - strlen(VersionString) - 1);
+    strncat(VersionString, "  compiled with Clang/LLVM "__VERSION__"\n", sizeof(VersionString) - strlen(VersionString) - 1);
 #elif defined(__GNUC__) || defined(__GNUG__)
-        strncat(VersionString, "  compiled with GCC "__VERSION__"\n", sizeof(VersionString) - strlen(VersionString) - 1);
+    strncat(VersionString, "  compiled with GCC "__VERSION__"\n", sizeof(VersionString) - strlen(VersionString) - 1);
 #endif
 
     strncat(VersionString, "\n [ FPGA ]\n ", sizeof(VersionString) - strlen(VersionString) - 1);
@@ -872,12 +874,13 @@ static void PacketReceived(PacketCommandNG *packet) {
         case CMD_T55XX_READ_BLOCK: {
             struct p {
                 uint32_t password;
-                uint8_t blockno;
-                uint8_t page;
-                bool pwdmode;
+                uint8_t  blockno;
+                uint8_t  page;
+                bool     pwdmode;
+                uint8_t  downlink_mode;
             } PACKED;
             struct p *payload = (struct p *) packet->data.asBytes;
-            T55xxReadBlock(payload->page, payload->pwdmode, false, payload->blockno, payload->password);
+            T55xxReadBlock(payload->page, payload->pwdmode, false, payload->blockno, payload->password, payload->downlink_mode);
             break;
         }
         case CMD_T55XX_WRITE_BLOCK: {
@@ -886,15 +889,15 @@ static void PacketReceived(PacketCommandNG *packet) {
             break;
         }
         case CMD_T55XX_WAKEUP: {
-            T55xxWakeUp(packet->oldarg[0]);
+            T55xxWakeUp(packet->oldarg[0], packet->oldarg[1]);
             break;
         }
         case CMD_T55XX_RESET_READ: {
-            T55xxResetRead();
+            T55xxResetRead(packet->data.asBytes[0] & 0xff);
             break;
         }
         case CMD_T55XX_CHKPWDS: {
-            T55xx_ChkPwds();
+            T55xx_ChkPwds(packet->data.asBytes[0] & 0xff);
             break;
         }
         case CMD_PCF7931_READ: {
@@ -1588,42 +1591,138 @@ static void PacketReceived(PacketCommandNG *packet) {
             break;
         }
 #ifdef WITH_FLASH
-        case CMD_FLASHMEM_SET_SPIBAUDRATE: {
-            FlashmemSetSpiBaudrate(packet->oldarg[0]);
+        case CMD_SPIFFS_TEST: {
+            test_spiffs();
             break;
         }
-        case CMD_FLASHMEM_READ: {
+        case CMD_SPIFFS_MOUNT: {
+            rdv40_spiffs_lazy_mount();
+            break;
+        }
+        case CMD_SPIFFS_UNMOUNT: {
+            rdv40_spiffs_lazy_unmount();
+            break;
+        }
+        case CMD_SPIFFS_PRINT_TREE: {
+            rdv40_spiffs_safe_print_tree(false);
+            break;
+        }
+        case CMD_SPIFFS_PRINT_FSINFO: {
+            rdv40_spiffs_safe_print_fsinfo();
+            break;
+        }
+        case CMD_SPIFFS_DOWNLOAD: {
             LED_B_ON();
-            uint32_t startidx = packet->oldarg[0];
-            uint16_t len = packet->oldarg[1];
+            uint8_t filename[32];
+            uint8_t *pfilename = packet->data.asBytes;
+            memcpy(filename, pfilename, SPIFFS_OBJ_NAME_LEN);
+            if (DBGLEVEL > 1) Dbprintf("> Filename received for spiffs dump : %s", filename);
 
-            Dbprintf("FlashMem read | %d - %d | ", startidx, len);
+            //uint32_t size = 0;
+            //rdv40_spiffs_stat((char *)filename, (uint32_t *)size,RDV40_SPIFFS_SAFETY_SAFE);
+            uint32_t size = packet->oldarg[1];
+            //uint8_t buff[size];
 
-            size_t size = MIN(PM3_CMD_DATA_SIZE, len);
+            uint8_t *buff = BigBuf_malloc(size);
+            rdv40_spiffs_read_as_filetype((char *)filename, (uint8_t *)buff, size, RDV40_SPIFFS_SAFETY_SAFE);
 
-            if (!FlashInit()) {
-                break;
+            // arg0 = filename
+            // arg1 = size
+            // arg2 = RFU
+
+            for (size_t i = 0; i < size; i += PM3_CMD_DATA_SIZE) {
+                size_t len = MIN((size - i), PM3_CMD_DATA_SIZE);
+                int result = reply_old(CMD_SPIFFS_DOWNLOADED, i, len, 0, buff + i, len);
+                if (result != PM3_SUCCESS)
+                    Dbprintf("transfer to client failed ::  | bytes between %d - %d (%d) | result: %d", i, i + len, len, result);
             }
-
-            uint8_t *mem = BigBuf_malloc(size);
-
-            for (size_t i = 0; i < len; i += size) {
-                len = MIN((len - i), size);
-
-                Dbprintf("FlashMem reading  | %d | %d | %d |", startidx + i, i, len);
-                uint16_t isok = Flash_ReadDataCont(startidx + i, mem, len);
-                if (isok == len) {
-                    print_result("Chunk: ", mem, len);
-                } else {
-                    Dbprintf("FlashMem reading failed | %d | %d", len, isok);
-                    break;
-                }
-            }
-            BigBuf_free();
-            FlashStop();
+            // Trigger a finish downloading signal with an ACK frame
+            reply_old(CMD_ACK, 1, 0, 0, 0, 0);
             LED_B_OFF();
             break;
         }
+        case CMD_SPIFFS_STAT: {
+            LED_B_ON();
+            uint8_t filename[32];
+            uint8_t *pfilename = packet->data.asBytes;
+            memcpy(filename, pfilename, SPIFFS_OBJ_NAME_LEN);
+            if (DBGLEVEL > 1) Dbprintf("> Filename received for spiffs STAT : %s", filename);
+            int changed = rdv40_spiffs_lazy_mount();
+            uint32_t size = size_in_spiffs((char *)filename);
+            if (changed) rdv40_spiffs_lazy_unmount();
+            reply_old(CMD_ACK, size, 0, 0, 0, 0);
+            LED_B_OFF();
+            break;
+        }
+        case CMD_SPIFFS_REMOVE: {
+            LED_B_ON();
+            uint8_t filename[32];
+            uint8_t *pfilename = packet->data.asBytes;
+            memcpy(filename, pfilename, SPIFFS_OBJ_NAME_LEN);
+            if (DBGLEVEL > 1) Dbprintf("> Filename received for spiffs REMOVE : %s", filename);
+            rdv40_spiffs_remove((char *) filename, RDV40_SPIFFS_SAFETY_SAFE);
+            LED_B_OFF();
+            break;
+        }
+        case CMD_SPIFFS_RENAME: {
+            LED_B_ON();
+            uint8_t srcfilename[32];
+            uint8_t destfilename[32];
+            uint8_t *pfilename = packet->data.asBytes;
+            char *token;
+            token = strtok((char *)pfilename, ",");
+            strcpy((char *)srcfilename, token);
+            token = strtok(NULL, ",");
+            strcpy((char *)destfilename, token);
+            if (DBGLEVEL > 1) Dbprintf("> Filename received as source for spiffs RENAME : %s", srcfilename);
+            if (DBGLEVEL > 1) Dbprintf("> Filename received as destination for spiffs RENAME : %s", destfilename);
+            rdv40_spiffs_rename((char *) srcfilename, (char *)destfilename, RDV40_SPIFFS_SAFETY_SAFE);
+            LED_B_OFF();
+            break;
+        }
+        case CMD_SPIFFS_COPY: {
+            LED_B_ON();
+            uint8_t srcfilename[32];
+            uint8_t destfilename[32];
+            uint8_t *pfilename = packet->data.asBytes;
+            char *token;
+            token = strtok((char *)pfilename, ",");
+            strcpy((char *)srcfilename, token);
+            token = strtok(NULL, ",");
+            strcpy((char *)destfilename, token);
+            if (DBGLEVEL > 1) Dbprintf("> Filename received as source for spiffs COPY : %s", srcfilename);
+            if (DBGLEVEL > 1) Dbprintf("> Filename received as destination for spiffs COPY : %s", destfilename);
+            rdv40_spiffs_copy((char *) srcfilename, (char *)destfilename, RDV40_SPIFFS_SAFETY_SAFE);
+            LED_B_OFF();
+            break;
+        }
+        case CMD_SPIFFS_WRITE: {
+            LED_B_ON();
+            uint8_t filename[32];
+            uint32_t append = packet->oldarg[0];
+            uint32_t size = packet->oldarg[1];
+            uint8_t *data = packet->data.asBytes;
+
+            //rdv40_spiffs_lazy_mount();
+
+            uint8_t *pfilename = packet->data.asBytes;
+            memcpy(filename, pfilename, SPIFFS_OBJ_NAME_LEN);
+            data += SPIFFS_OBJ_NAME_LEN;
+
+            if (DBGLEVEL > 1) Dbprintf("> Filename received for spiffs WRITE : %s with APPEND SET TO : %d", filename, append);
+            if (!append) {
+                rdv40_spiffs_write((char *) filename, (uint8_t *)data, size, RDV40_SPIFFS_SAFETY_SAFE);
+            } else {
+                rdv40_spiffs_append((char *) filename, (uint8_t *)data, size, RDV40_SPIFFS_SAFETY_SAFE);
+            }
+            reply_old(CMD_ACK, 1, 0, 0, 0, 0);
+            LED_B_OFF();
+            break;
+        }
+        case CMD_FLASHMEM_SET_SPIBAUDRATE: {
+            FlashmemSetSpiBaudrate(packet->oldarg[0]);
+            break;
+        }
         case CMD_FLASHMEM_WRITE: {
             LED_B_ON();
             uint8_t isok = 0;
@@ -1789,6 +1888,10 @@ static void PacketReceived(PacketCommandNG *packet) {
             SendStatus();
             break;
         }
+        case CMD_STANDALONE: {
+            RunMod();
+            break;
+        }
         case CMD_CAPABILITIES: {
             SendCapabilities();
             break;
diff --git a/armsrc/apps.h b/armsrc/apps.h
index 41a1f0271..e3ea8048c 100644
--- a/armsrc/apps.h
+++ b/armsrc/apps.h
@@ -102,12 +102,13 @@ void CopyVikingtoT55xx(uint32_t block1, uint32_t block2, uint8_t Q5);
 void WriteEM410x(uint32_t card, uint32_t id_hi, uint32_t id_lo);
 void CopyIndala64toT55x7(uint32_t hi, uint32_t lo); // Clone Indala 64-bit tag by UID to T55x7
 void CopyIndala224toT55x7(uint32_t uid1, uint32_t uid2, uint32_t uid3, uint32_t uid4, uint32_t uid5, uint32_t uid6, uint32_t uid7); // Clone Indala 224-bit tag by UID to T55x7
-void T55xxResetRead(void);
+void T55xxResetRead(uint8_t flags);
+//id T55xxWriteBlock(uint32_t data, uint8_t blockno, uint32_t pwd, uint8_t flags);
 void T55xxWriteBlock(uint8_t *data);
-void T55xxWriteBlockExt(uint32_t data, uint8_t blockno, uint32_t pwd, uint8_t flags);
-void T55xxReadBlock(uint8_t page, bool pwd_mode, bool brute_mem, uint8_t block, uint32_t pwd);
-void T55xxWakeUp(uint32_t Pwd);
-void T55xx_ChkPwds(void);
+// void T55xxWriteBlockExt(uint32_t data, uint8_t blockno, uint32_t pwd, uint8_t flags);
+void T55xxReadBlock(uint8_t page, bool pwd_mode, bool brute_mem, uint8_t block, uint32_t pwd, uint8_t downlink_mode);
+void T55xxWakeUp(uint32_t Pwd, uint8_t flags);
+void T55xx_ChkPwds(uint8_t flags);
 
 void TurnReadLFOn(uint32_t delay);
 
diff --git a/armsrc/desfire_crypto.c b/armsrc/desfire_crypto.c
index 10ddf4162..2e9d5a8d3 100644
--- a/armsrc/desfire_crypto.c
+++ b/armsrc/desfire_crypto.c
@@ -571,15 +571,19 @@ void mifare_cypher_single_block(desfirekey_t key, uint8_t *data, uint8_t *ivect,
         case T_AES:
             switch (operation) {
                 case MCO_ENCYPHER: {
-                    AesCtx ctx;
-                    AesCtxIni(&ctx, ivect, key->data, KEY128, CBC);
-                    AesEncrypt(&ctx, data, edata, sizeof(edata));
+                    mbedtls_aes_context ctx;
+                    mbedtls_aes_init(&ctx);
+                    mbedtls_aes_setkey_enc(&ctx, key->data, 128);
+                    mbedtls_aes_crypt_cbc(&ctx, MBEDTLS_AES_ENCRYPT, sizeof(edata), ivect, data, edata);
+                    mbedtls_aes_free(&ctx);
                     break;
                 }
                 case MCO_DECYPHER: {
-                    AesCtx ctx;
-                    AesCtxIni(&ctx, ivect, key->data, KEY128, CBC);
-                    AesDecrypt(&ctx, edata, data, sizeof(edata));
+                    mbedtls_aes_context ctx;
+                    mbedtls_aes_init(&ctx);
+                    mbedtls_aes_setkey_dec(&ctx, key->data, 128);
+                    mbedtls_aes_crypt_cbc(&ctx, MBEDTLS_AES_DECRYPT, sizeof(edata), ivect, edata, data);
+                    mbedtls_aes_free(&ctx);
                     break;
                 }
             }
diff --git a/armsrc/flashmem.c b/armsrc/flashmem.c
index 4584d41b2..7bca9ddf9 100644
--- a/armsrc/flashmem.c
+++ b/armsrc/flashmem.c
@@ -1,4 +1,5 @@
 #include "flashmem.h"
+#include "pmflash.h"
 
 /* here: use NCPS2 @ PA10: */
 #define SPI_CSR_NUM      2
@@ -553,6 +554,7 @@ void Flashmem_print_status(void) {
 }
 
 void Flashmem_print_info(void) {
+
     if (!FlashInit()) return;
 
     DbpString(_BLUE_("Flash memory dictionary loaded"));
@@ -565,21 +567,21 @@ void Flashmem_print_info(void) {
     if (isok == 2) {
         num = ((keysum[1] << 8) | keysum[0]);
         if (num != 0xFFFF && num != 0x0)
-            Dbprintf("  Mifare................"_YELLOW_("%d")"keys", num);
+            Dbprintf("  Mifare.................."_YELLOW_("%d")"keys", num);
     }
 
     isok = Flash_ReadDataCont(DEFAULT_T55XX_KEYS_OFFSET, keysum, 2);
     if (isok == 2) {
         num = ((keysum[1] << 8) | keysum[0]);
         if (num != 0xFFFF && num != 0x0)
-            Dbprintf("  T55x7................."_YELLOW_("%d")"keys", num);
+            Dbprintf("  T55x7..................."_YELLOW_("%d")"keys", num);
     }
 
     isok = Flash_ReadDataCont(DEFAULT_ICLASS_KEYS_OFFSET, keysum, 2);
     if (isok == 2) {
         num = ((keysum[1] << 8) | keysum[0]);
         if (num != 0xFFFF && num != 0x0)
-            Dbprintf("  iClass................"_YELLOW_("%d")"keys", num);
+            Dbprintf("  iClass.................."_YELLOW_("%d")"keys", num);
     }
 
     FlashStop();
diff --git a/armsrc/flashmem.h b/armsrc/flashmem.h
index 4016e66bd..26d2d974d 100644
--- a/armsrc/flashmem.h
+++ b/armsrc/flashmem.h
@@ -62,7 +62,7 @@
 #define RESUME          0x7A
 
 // Flash busy timeout: 20ms is the strict minimum when writing 256kb
-#define BUSY_TIMEOUT    50000L
+#define BUSY_TIMEOUT    200000L
 
 #define WINBOND_MANID   0xEF
 #define WINBOND_DEVID   0x11
@@ -142,5 +142,6 @@ uint16_t Flash_WriteData(uint32_t address, uint8_t *in, uint16_t len);
 uint16_t Flash_WriteDataCont(uint32_t address, uint8_t *in, uint16_t len);
 void Flashmem_print_status(void);
 void Flashmem_print_info(void);
+uint16_t FlashSendLastByte(uint32_t data);
 
 #endif
diff --git a/armsrc/hitag2.c b/armsrc/hitag2.c
index 46e9a43dd..682dd950a 100644
--- a/armsrc/hitag2.c
+++ b/armsrc/hitag2.c
@@ -434,7 +434,7 @@ static bool hitag2_password(uint8_t *rx, const size_t rxlen, uint8_t *tx, size_t
             break;
         }
     }
-    
+
     return true;
 }
 
diff --git a/armsrc/iso14443a.c b/armsrc/iso14443a.c
index b5e612ec9..a09978d7a 100644
--- a/armsrc/iso14443a.c
+++ b/armsrc/iso14443a.c
@@ -13,6 +13,8 @@
 
 #define MAX_ISO14A_TIMEOUT 524288
 static uint32_t iso14a_timeout;
+// if iso14443a not active - transmit/receive dont try to execute
+static bool iso14443a_active = false;
 
 uint8_t colpos = 0;
 int rsamples = 0;
@@ -91,7 +93,7 @@ static uint32_t LastProxToAirDuration;
 // Sequence D: 11110000 modulation with subcarrier during first half
 // Sequence E: 00001111 modulation with subcarrier during second half
 // Sequence F: 00000000 no modulation with subcarrier
-// Sequence COLL: 11111111 load modulation over the full bitlenght.
+// Sequence COLL: 11111111 load modulation over the full bitlength.
 //                         Tricks the reader to think that multiple cards answer (at least one card with 1 and at least one card with 0).
 // READER TO CARD - miller
 // Sequence X: 00001100 drop after half a period
@@ -741,7 +743,7 @@ static void Code4bitAnswerAsTag(uint8_t cmd) {
 
 //-----------------------------------------------------------------------------
 // Wait for commands from reader
-// stop when button is pressed
+// stop when button is pressed or client usb connection resets
 // or return TRUE when command is captured
 //-----------------------------------------------------------------------------
 static bool GetIso14443aCommandFromReader(uint8_t *received, uint8_t *par, int *len) {
@@ -1092,7 +1094,7 @@ void SimulateIso14443aTag(uint8_t tagType, uint8_t flags, uint8_t *data) {
 #define ORDER_SELECT_CL2     30
 #define ORDER_EV1_COMP_WRITE 40
 #define ORDER_RATS           70
-    int order = ORDER_NONE;
+    uint8_t order = ORDER_NONE;
 
     int retval = PM3_SUCCESS;
 
@@ -1551,6 +1553,9 @@ void PrepareDelayedTransfer(uint16_t delay) {
 //-------------------------------------------------------------------------------------
 static void TransmitFor14443a(const uint8_t *cmd, uint16_t len, uint32_t *timing) {
 
+    if (!iso14443a_active)
+        return;
+
     FpgaWriteConfWord(FPGA_MAJOR_MODE_HF_ISO14443A | FPGA_HF_ISO14443A_READER_MOD);
 
     if (timing) {
@@ -1578,10 +1583,20 @@ static void TransmitFor14443a(const uint8_t *cmd, uint16_t len, uint32_t *timing
 
     volatile uint8_t b;
     uint16_t c = 0;
+    uint32_t sendtimer = GetTickCount();
+    uint32_t cntr = 0;
     while (c < len) {
         if (AT91C_BASE_SSC->SSC_SR & (AT91C_SSC_TXRDY)) {
             AT91C_BASE_SSC->SSC_THR = cmd[c++];
+            cntr = 0;
+        } else {
+            if (cntr++ > 1000) {
+                cntr = 0;
+                if (GetTickCount() - sendtimer > 100)
+                    break;
+            }
         }
+
         //iceman test
         if (AT91C_BASE_SSC->SSC_SR & (AT91C_SSC_RXRDY)) {
             b = (uint16_t)(AT91C_BASE_SSC->SSC_RHR);
@@ -1923,6 +1938,9 @@ bool EmLogTrace(uint8_t *reader_data, uint16_t reader_len, uint32_t reader_Start
 static int GetIso14443aAnswerFromTag(uint8_t *receivedResponse, uint8_t *receivedResponsePar, uint16_t offset) {
     uint32_t c = 0;
 
+    if (!iso14443a_active)
+        return false;
+
     // Set FPGA mode to "reader listen mode", no modulation (listen
     // only, since we are receiving, not transmitting).
     // Signal field is on with the appropriate LED
@@ -1937,6 +1955,7 @@ static int GetIso14443aAnswerFromTag(uint8_t *receivedResponse, uint8_t *receive
     (void)b;
 
     uint32_t timeout = iso14a_get_timeout();
+    uint32_t receive_timer = GetTickCount();
     for (;;) {
         WDT_HIT();
 
@@ -1949,7 +1968,12 @@ static int GetIso14443aAnswerFromTag(uint8_t *receivedResponse, uint8_t *receive
                 return false;
             }
         }
+
+        // timeout already in ms + 100ms guard time
+        if (GetTickCount() - receive_timer > timeout + 100)
+            break;
     }
+    return false;
 }
 
 void ReaderTransmitBitsPar(uint8_t *frame, uint16_t bits, uint8_t *par, uint32_t *timing) {
@@ -2354,6 +2378,14 @@ void iso14443a_setup(uint8_t fpga_minor_mode) {
     UartReset();
     NextTransferTime = 2 * DELAY_ARM2AIR_AS_READER;
     iso14a_set_timeout(1060); // 106 * 10ms default
+
+    iso14443a_active = true;
+}
+
+void iso14443a_off() {
+    FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
+    LEDsoff();
+    iso14443a_active = false;
 }
 
 /* Peter Fillmore 2015
@@ -2558,9 +2590,8 @@ void ReaderIso14443a(PacketCommandNG *c) {
         return;
 
 OUT:
-    FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
+    iso14443a_off();
     set_tracing(false);
-    LEDsoff();
 }
 
 // Determine the distance between two nonces.
@@ -2854,8 +2885,7 @@ void ReaderMifare(bool first_try, uint8_t block, uint8_t keytype) {
 
     reply_mix(CMD_ACK, isOK, 0, 0, buf, sizeof(buf));
 
-    FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
-    LEDsoff();
+    iso14443a_off();
     set_tracing(false);
 }
 
@@ -3094,7 +3124,6 @@ void DetectNACKbug() {
 
     //reply_mix(CMD_ACK, isOK, num_nacks, i, 0, 0);
     BigBuf_free();
-    FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
-    LEDsoff();
+    iso14443a_off();
     set_tracing(false);
 }
diff --git a/armsrc/iso14443a.h b/armsrc/iso14443a.h
index 5ebc3e645..e5523c598 100644
--- a/armsrc/iso14443a.h
+++ b/armsrc/iso14443a.h
@@ -104,7 +104,7 @@ typedef struct {
 #endif
 
 #ifndef CheckCrc14A
-# define CheckCrc14A(data, len)	check_crc(CRC_14443_A, (data), (len))
+# define CheckCrc14A(data, len) check_crc(CRC_14443_A, (data), (len))
 #endif
 
 void GetParity(const uint8_t *pbtCmd, uint16_t len, uint8_t *par);
diff --git a/armsrc/iso15693.c b/armsrc/iso15693.c
index 374b2be23..e74ca972d 100644
--- a/armsrc/iso15693.c
+++ b/armsrc/iso15693.c
@@ -695,7 +695,7 @@ static void BuildInventoryResponse(uint8_t *cmdout, uint8_t *uid) {
 //  speed ... 0 low speed, 1 hi speed
 //  **recv will return you a pointer to the received data
 //  If you do not need the answer use NULL for *recv[]
-//  return: lenght of received data
+//  return: length of received data
 // logging enabled
 int SendDataTag(uint8_t *send, int sendlen, bool init, int speed, uint8_t *outdata) {
 
diff --git a/armsrc/ldscript b/armsrc/ldscript
index 34da26bcd..80c2a4aed 100644
--- a/armsrc/ldscript
+++ b/armsrc/ldscript
@@ -11,57 +11,57 @@ INCLUDE ../common/ldscript.common
 
 PHDRS
 {
-	text PT_LOAD FLAGS(5);
-	data PT_LOAD;
-	bss PT_LOAD;
+    text PT_LOAD FLAGS(5);
+    data PT_LOAD;
+    bss PT_LOAD;
 }
 
 ENTRY(Vector)
 SECTIONS
 {
-	.start : {
-		*(.startos)
-	} >osimage :text
+    .start : {
+        *(.startos)
+    } >osimage :text
 
-	.text : {
-		KEEP(*(stage1_image))
-		*(.text)
-		*(.text.*)
-		*(.eh_frame)
-		*(.glue_7)
-		*(.glue_7t)
-	} >osimage :text
+    .text : {
+        KEEP(*(stage1_image))
+        *(.text)
+        *(.text.*)
+        *(.eh_frame)
+        *(.glue_7)
+        *(.glue_7t)
+    } >osimage :text
 
-	.rodata : {
-		*(.rodata)
-		*(.rodata.*)
-		*(fpga_all_bit.data)
-		KEEP(*(.version_information))
-		. = ALIGN(8);
-	} >osimage :text
+    .rodata : {
+        *(.rodata)
+        *(.rodata.*)
+        *(fpga_all_bit.data)
+        KEEP(*(.version_information))
+        . = ALIGN(8);
+    } >osimage :text
 
-	.data : {
-		KEEP(*(compressed_data))
-		*(.data)
-		*(.data.*)
-		*(.ramfunc)
-		. = ALIGN(4);
-	} >ram AT>osimage :data
+    .data : {
+        KEEP(*(compressed_data))
+        *(.data)
+        *(.data.*)
+        *(.ramfunc)
+        . = ALIGN(4);
+    } >ram AT>osimage :data
 
-	__data_src_start__ = LOADADDR(.data);
-	__data_start__ = ADDR(.data);
-	__data_end__ = __data_start__ + SIZEOF(.data);
-	__os_size__ = SIZEOF(.text) + SIZEOF(.data) + SIZEOF(.rodata);
-	
-	.bss : {
-		__bss_start__ = .; 
-		*(.bss)
-		*(.bss.*) 
-		. = ALIGN(4);
-		__bss_end__ = .;
-	} >ram AT>ram :bss
+    __data_src_start__ = LOADADDR(.data);
+    __data_start__ = ADDR(.data);
+    __data_end__ = __data_start__ + SIZEOF(.data);
+    __os_size__ = SIZEOF(.text) + SIZEOF(.data) + SIZEOF(.rodata);
+    
+    .bss : {
+        __bss_start__ = .; 
+        *(.bss)
+        *(.bss.*) 
+        . = ALIGN(4);
+        __bss_end__ = .;
+    } >ram AT>ram :bss
 
-	.commonarea (NOLOAD) : {
-		*(.commonarea)
-	} >commonarea :NONE
+    .commonarea (NOLOAD) : {
+        *(.commonarea)
+    } >commonarea :NONE
 }
diff --git a/armsrc/lfops.c b/armsrc/lfops.c
index 09a1632cb..af80e2f6c 100644
--- a/armsrc/lfops.c
+++ b/armsrc/lfops.c
@@ -19,8 +19,11 @@
 #include "protocols.h"
 #include "usb_cdc.h" // for usb_poll_validate_length
 #include "common.h"
+#include "pmflash.h"
 #include "flashmem.h" // persistence on mem
 
+
+
 //#define START_GAP 31*8 // was 250 // SPEC:  1*8 to 50*8 - typ 15*8 (15fc)
 //#define WRITE_GAP 8*8 // 17*8 // was 160 // SPEC:  1*8 to 20*8 - typ 10*8 (10fc)
 //#define WRITE_0   15*8 // 18*8 // was 144 // SPEC: 16*8 to 32*8 - typ 24*8 (24fc)
@@ -45,31 +48,148 @@
 //     = 1us = 1.5ticks
 // 1fc = 8us = 12ticks
 /*
-Default LF T55xx config is set to:
-    startgap = 31*8
-    writegap = 17*8
-    write_0 = 15*8
-    write_1 = 47*8
-    read_gap = 15*8
+  ==========================================================================================================
+       T55x7 Timing
+  ==========================================================================================================
+
+ // t55xx_config t_config = { 29 * 8, 17 * 8, 15 * 8, 47 * 8, 15 * 8 } ;
+
+    ATA5577 Downlink Protocol Timings.
+    Note: All absolute times assume TC = 1 / fC = 8 μs (fC = 125 kHz)
+                                -----------------------------------------------------------------------
+ Fixed-bit-length Protocol     |           Normal Downlink         |           Fast Downlink           |
+ ------------------------------+-----------------------------------+-----------------------------------+------
+| Parameter  | Remark | Symbol |    Min.   |    Typ.   |    Max.   |    Min.   |    Typ.   |    Max.   | Unit |
+|------------+--------+--------+-----------+-----------+-----------+-----------+-----------+-----------+------|
+| Start gap  |        | Sgap   |      8    |     15    |     50    |      8    |     15    |     50    |  Tc  |
+| Write gap  |        | Wgap   |      8    |     10    |     20    |      8    |     10    |     20    |  Tc  |
+|------------+--------+--------+-----------+-----------+-----------+-----------+-----------+-----------+------|
+| coding     | 0 data | d0     |     16    |     24    |     32    |      8    |     12    |     16    |  Tc  |
+|            | 1 data | d1     |     48    |     56    |     64    |     24    |     28    |     32    |  Tc  |
+ -------------------------------------------------------------------------------------------------------------
+
+                                -----------------------------------------------------------------------
+ Long Leading Reference        |           Normal Downlink         |            Fast Downlink          |
+ ------------------------------+-----------------------------------+-----------------------------------+------
+| Parameter | Remark | Symbol  |    Min.   |    Typ.   |    Max.   |    Min.   |    Typ.   |    Max.   | Unit |
+|-----------+--------+---------+-----------+-----------+-----------+-----------+-----------+-----------+------|
+| Start gap |        | Sgap    |     8     |     10    |     50    |      8    |     10    |     50    |  Tc  |
+| Write gap |        | Wgap    |     8     |     10    |     20    |      8    |     10    |     20    |  Tc  |
+|-----------+--------+---------+-----------+-----------+-----------+-----------+-----------+-----------+------|
+|   Write   | Ref    |         |    152    |    160    |    168    |     140   |     144   |    148    |  Tc  |
+|   data    | Pulse  |  dref   |       136 clocks + 0 data bit     |       132 clocks + 0 data bit     |  Tc  |
+|  coding   |--------+---------+-----------------------------------+-----------------------------------+------|
+|           | 0 data | d0      |dref – 143 |dref – 136 |dref – 128 |dref – 135 |dref – 132 |dref – 124 |  Tc  |
+|           | 1 data | d1      |dref – 111 |dref – 104 |dref – 96  |dref – 119 |dref – 116 |dref – 112 |  Tc  |
+ -------------------------------------------------------------------------------------------------------------
+
+                                -----------------------------------------------------------------------
+ Leading-zero Reference        |           Normal Downlink         |            Fast Downlink          |
+ ------------------------------+-----------------------------------+-----------------------------------+------
+| Parameter | Remark | Symbol  |   Min.    |    Typ.   |    Max.   |    Min.   |   Typ.    |    Max.   | Unit |
+|-----------+--------+---------+-----------+-----------+-----------+-----------+-----------+-----------+------|
+| Start gap |        | Sgap    |     8     |      10   |     50    |      8    |     10    |     50    |  Tc  |
+| Write gap |        | Wgap    |     8     |      10   |     20    |      8    |     10    |     20    |  Tc  |
+|-----------+--------+---------+-----------+-----------+-----------+-----------+-----------+-----------+------|
+| Write     | Ref    | dref    |    12     |      –    |     72    |      8    |     –     |     68    |  Tc  |
+| data      | 0 data | d0      | dref – 7  |     dref  | dref + 8  | dref – 3  |   dref    | dref + 4  |  Tc  |
+| coding    | 1 data | d1      | dref + 9  | dref + 16 | dref + 24 | dref + 5  | dref + 8  | dref + 12 |  Tc  |
+ -------------------------------------------------------------------------------------------------------------
+
+                                -----------------------------------------------------------------------
+ 1-of-4 Coding                 |           Normal Downlink         |           Fast Downlink           |
+ ------------------------------+-----------------------------------+-----------------------------------+------
+| Parameter | Remark | Symbol  |   Min.    |   Typ.    |    Max.   |    Min.   |    Typ.   |    Max.   | Unit |
+|-----------+--------+---------+-----------+-----------+-----------+-----------+-----------+-----------+------|
+| Start gap |        | Sgap    |     8     |     10    |     50    |       8   |      10   |     50    |  Tc  |
+| Write gap |        | Wgap    |     8     |     10    |     20    |       8   |      10   |     20    |  Tc  |
+|-----------+--------+---------+-----------+-----------+-----------+-----------+-----------+-----------+------|
+| Write     | Ref 00 | dref    |     8     |     –     |     68    |      12   |      –    |     72    |  Tc  |
+| data      |00 data | d00     | dref – 7  |   dref    | dref + 8  | dref – 3  |    dref   |  dref+ 4  |  Tc  |
+| coding    |01 data | d01     | dref + 9  | dref + 16 | dref + 24 | dref + 5  | dref + 8  | dref + 12 |  Tc  |
+|           |10 data | d10     | dref + 25 | dref + 32 | dref + 40 | dref + 13 | dref + 16 | dref + 20 |  Tc  |
+|           |11 data | d11     | dref + 41 | dref + 48 | dref + 56 | dref + 21 | dref + 24 | dref + 28 |  Tc  |
+ -------------------------------------------------------------------------------------------------------------
 */
-t55xx_config t_config = { 29 * 8, 17 * 8, 15 * 8, 47 * 8, 15 * 8 } ;
+// Initial values if not in flash
+/*
+// Note: Moved * 8 to apply when used. Saving 28 bytes here (- the *8) and 28 bytes flash.
+//                              StartGap WriteGap Bit 0/00 Bit 1/01  Bit 10  Bit 11  ReadGap
+t55xx_config T55xx_Timing  = {{
+        { 29    , 17    , 15    , 47    , 0     , 0     , 15     }, // Default Fixed
+        { 31    , 20    , 18    , 50    , 0     , 0     , 15     }, // Long Leading Ref.
+        { 31    , 20    , 18    , 40    , 0     , 0     , 15     }, // Leading 0
+        { 29    , 17    , 15    , 31    , 47    , 63    , 15     }  // 1 of 4
+    }
+};
+*/
+//      StartGap WriteGap Bit 0/00 Bit 1/01  ReadGap Bit 10  Bit 11  
+t55xx_config T55xx_Timing  = {{
+        { 29 * 8, 17 * 8, 15 * 8, 47 * 8, 15 * 8, 0, 0 },           // Default Fixed
+        { 31 * 8, 20 * 8, 18 * 8, 50 * 8, 15 * 8, 0, 0 },           // Long Leading Ref.
+        { 31 * 8, 20 * 8, 18 * 8, 40 * 8, 15 * 8, 0, 0 },           // Leading 0
+        { 29 * 8, 17 * 8, 15 * 8, 31 * 8, 15 * 8, 47 * 8, 63 * 8 }  // 1 of 4
+    }
+};
+
+
+// Some defines for readability
+#define T55xx_DLMode_Fixed         0 // Default Mode
+#define T55xx_DLMode_LLR           1 // Long Leading Reference
+#define T55xx_DLMode_Leading0      2 // Leading Zero
+#define T55xx_DLMode_1of4          3 // 1 of 4
+#define T55xx_LongLeadingReference 4 // Value to tell Write Bit to send long reference
 
 void printT55xxConfig(void) {
+    int DLMode;
+
     DbpString(_BLUE_("LF T55XX config"));
-    Dbprintf("  [a] startgap............%d*8 (%d)", t_config.start_gap / 8, t_config.start_gap);
-    Dbprintf("  [b] writegap............%d*8 (%d)", t_config.write_gap / 8, t_config.write_gap);
-    Dbprintf("  [c] write_0.............%d*8 (%d)", t_config.write_0 / 8,   t_config.write_0);
-    Dbprintf("  [d] write_1.............%d*8 (%d)", t_config.write_1 / 8,   t_config.write_1);
-    Dbprintf("  [e] readgap.............%d*8 (%d)", t_config.read_gap / 8,  t_config.read_gap);
+    for (DLMode = 0; DLMode < 4; DLMode++) {
+        switch (DLMode) {
+            case T55xx_DLMode_Fixed    :
+                Dbprintf("r 0 fixed bit length (default)");
+                break;
+            case T55xx_DLMode_LLR      :
+                Dbprintf("r 1 long leading reference");
+                break;
+            case T55xx_DLMode_Leading0 :
+                Dbprintf("r 2 leading zero");
+                break;
+            case T55xx_DLMode_1of4     :
+                Dbprintf("r 3 1 of 4 coding reference");
+                break;
+        }
+        Dbprintf("  [a] startgap............%d*8 (%d)", T55xx_Timing.m[DLMode].start_gap / 8, T55xx_Timing.m[DLMode].start_gap);
+        Dbprintf("  [b] writegap............%d*8 (%d)", T55xx_Timing.m[DLMode].write_gap / 8, T55xx_Timing.m[DLMode].write_gap);
+        Dbprintf("  [c] write_0.............%d*8 (%d)", T55xx_Timing.m[DLMode].write_0   / 8, T55xx_Timing.m[DLMode].write_0);
+        Dbprintf("  [d] write_1.............%d*8 (%d)", T55xx_Timing.m[DLMode].write_1   / 8, T55xx_Timing.m[DLMode].write_1);
+        Dbprintf("  [e] readgap.............%d*8 (%d)", T55xx_Timing.m[DLMode].read_gap / 8, T55xx_Timing.m[DLMode].read_gap);
+        if (DLMode ==  T55xx_DLMode_1of4) {
+            Dbprintf("  [f] write_2.............%d*8 (%d)", T55xx_Timing.m[DLMode].write_2 / 8, T55xx_Timing.m[DLMode].write_2);
+            Dbprintf("  [g] write_3.............%d*8 (%d)", T55xx_Timing.m[DLMode].write_3 / 8, T55xx_Timing.m[DLMode].write_3);
+        }
+    }
 }
 
 void setT55xxConfig(uint8_t arg0, t55xx_config *c) {
+    uint8_t DLMode;
+//    uint8_t ClearT55Settings = c->m[0].start_gap & 0xffff; // all values will be ffff if clear requested
 
-    if (c->start_gap != 0) t_config.start_gap = c->start_gap;
-    if (c->write_gap != 0) t_config.write_gap = c->write_gap;
-    if (c->write_0   != 0) t_config.write_0 = c->write_0;
-    if (c->write_1   != 0) t_config.write_1 = c->write_1;
-    if (c->read_gap  != 0) t_config.read_gap = c->read_gap;
+
+    for (DLMode = 0; DLMode < 4; DLMode++) {
+        if (c->m[DLMode].start_gap != 0) T55xx_Timing.m[DLMode].start_gap = c->m[DLMode].start_gap;// * 8;
+        if (c->m[DLMode].write_gap != 0) T55xx_Timing.m[DLMode].write_gap = c->m[DLMode].write_gap;// * 8;
+        if (c->m[DLMode].write_0   != 0) T55xx_Timing.m[DLMode].write_0   = c->m[DLMode].write_0  ;// * 8;
+        if (c->m[DLMode].write_1   != 0) T55xx_Timing.m[DLMode].write_1   = c->m[DLMode].write_1  ;// * 8;
+        if (DLMode ==  T55xx_DLMode_1of4) {
+            if (c->m[DLMode].write_2   != 0) T55xx_Timing.m[DLMode].write_2   = c->m[DLMode].write_2;// * 8;
+            if (c->m[DLMode].write_3   != 0) T55xx_Timing.m[DLMode].write_3   = c->m[DLMode].write_3;// * 8 ;
+        } else {
+            T55xx_Timing.m[DLMode].write_2   = 0x00;
+            T55xx_Timing.m[DLMode].write_3   = 0x00;
+        }
+        if (c->m[DLMode].read_gap  != 0) T55xx_Timing.m[DLMode].read_gap  = c->m[DLMode].read_gap;//* 8;
+    }
 
     printT55xxConfig();
 
@@ -83,6 +203,7 @@ void setT55xxConfig(uint8_t arg0, t55xx_config *c) {
         return;
     }
 
+
     uint8_t *buf = BigBuf_malloc(T55XX_CONFIG_LEN);
     Flash_CheckBusy(BUSY_TIMEOUT);
     uint16_t res = Flash_ReadDataCont(T55XX_CONFIG_OFFSET, buf, T55XX_CONFIG_LEN);
@@ -92,11 +213,12 @@ void setT55xxConfig(uint8_t arg0, t55xx_config *c) {
         return;
     }
 
-    memcpy(buf, &t_config, T55XX_CONFIG_LEN);
+    memcpy(buf, &T55xx_Timing, T55XX_CONFIG_LEN);
 
     Flash_CheckBusy(BUSY_TIMEOUT);
     Flash_WriteEnable();
     Flash_Erase4k(3, 0xD);
+
     res = Flash_Write(T55XX_CONFIG_OFFSET, buf, T55XX_CONFIG_LEN);
 
     if (res == T55XX_CONFIG_LEN && DBGLEVEL > 1) {
@@ -108,11 +230,12 @@ void setT55xxConfig(uint8_t arg0, t55xx_config *c) {
 }
 
 t55xx_config *getT55xxConfig(void) {
-    return &t_config;
+    return &T55xx_Timing;//_FixedBit;
 }
 
 void loadT55xxConfig(void) {
 #ifdef WITH_FLASH
+
     if (!FlashInit()) {
         return;
     }
@@ -134,7 +257,8 @@ void loadT55xxConfig(void) {
         return;
     }
 
-    memcpy((uint8_t *)&t_config, buf, T55XX_CONFIG_LEN);
+    if (buf[0] != 0xFF) // if not set for clear
+        memcpy((uint8_t *)&T55xx_Timing, buf, T55XX_CONFIG_LEN);
 
     if (isok == T55XX_CONFIG_LEN) {
         if (DBGLEVEL > 1) DbpString("T55XX Config load success");
@@ -1353,36 +1477,184 @@ void TurnReadLF_off(uint32_t delay) {
     WaitUS(delay);
 }
 
-// Write one bit to card
-void T55xxWriteBit(int bit) {
-    if (!bit)
-        TurnReadLFOn(t_config.write_0);
-    else
-        TurnReadLFOn(t_config.write_1);
+// Macro for code readability
+#define BitStream_Byte(X) ((X) >> 3)
+#define BitStream_Bit(X)  ((X) &  7)
+#define t55_llr_ref  			(136 * 8)
+#define t55_send_PwdMode      (arg & 0x01)
+#define t55_send_Page        ((arg & 0x02) >> 1)
+#define t55_send_TestMode    ((arg & 0x04) >> 2)
+#define t55_send_RegReadMode ((arg & 0x20) >> 5)
+#define t55_send_ReadCmd     ((arg & 0x40) >> 6)
+#define t55_send_Reset       ((arg & 0x80) >> 7)
+
+// Write one bit to chip
+void T55xxWriteBit(uint8_t bit, uint8_t downlink_idx) {
+
+    // Dbprintf ("%d",bit);
+    // If bit = 4 Send Long Leading Reference which is (138*8) + WRITE_0
+    switch (bit) {
+        case 0 :
+            TurnReadLFOn(T55xx_Timing.m[downlink_idx].write_0);
+            break; // Send bit  0/00
+        case 1 :
+            TurnReadLFOn(T55xx_Timing.m[downlink_idx].write_1);
+            break; // Send bit  1/01
+        case 2 :
+            TurnReadLFOn(T55xx_Timing.m[downlink_idx].write_2);
+            break; // Send bits   10 (1 of 4)
+        case 3 :
+            TurnReadLFOn(T55xx_Timing.m[downlink_idx].write_3);
+            break; // Send bits   11 (1 of 4)
+        case 4 :
+            TurnReadLFOn(T55xx_Timing.m[downlink_idx].write_0 + t55_llr_ref);
+            break; // Send Long Leading Reference
+    }
+
     FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
-    WaitUS(t_config.write_gap);
+    WaitUS(T55xx_Timing.m[downlink_idx].write_gap);
 }
 
-// Send T5577 reset command then read stream (see if we can identify the start of the stream)
-void T55xxResetRead(void) {
-    LED_A_ON();
-    //clear buffer now so it does not interfere with timing later
-    BigBuf_Clear_keep_EM();
+// Function to abstract an Arbitrary length byte array to store bit pattern.
+// bit_array    - Array to hold data/bit pattern
+// start_offset - bit location to start storing new bits.
+// data         - upto 32 bits of data to store
+// num_bits     - how many bits (low x bits of data)  Max 32 bits at a time
+// max_len      - how many bytes can the bit_array hold (ensure no buffer overflow)
+// returns "Next" bit offset / bits stored (for next store)
+uint8_t T55xx_SetBits(uint8_t *BitStream, uint8_t start_offset, uint32_t data, uint8_t num_bits, uint8_t max_len) {
+    int8_t offset;
+    int8_t NextOffset = start_offset;
 
+    // Check if data will fit.
+    if ((start_offset + num_bits) <= (max_len * 8)) {
+        // Loop through the data and store
+        for (offset = (num_bits - 1); offset >= 0; offset--) {
+
+            if ((data >> offset) & 1)  BitStream[BitStream_Byte(NextOffset)] |= (1         << BitStream_Bit(NextOffset));     // Set the bit to 1
+            else                       BitStream[BitStream_Byte(NextOffset)] &= (0xff ^ (1 << BitStream_Bit(NextOffset)));    // Set the bit to 0
+
+            NextOffset++;
+        }
+    } else {
+        // Note: This should never happen unless some code changes cause it.
+        // So short message for coders when testing.
+        Dbprintf("T55 too many bits");
+    }
+    return NextOffset;
+}
+
+// Send one downlink command to the card
+// void T55xx_SendCMD (uint32_t Data, uint8_t Block, uint32_t Pwd, uint8_t arg) {
+void T55xx_SendCMD(uint32_t Data, uint32_t Pwd, uint16_t arg) {
+
+    /*
+    arg bits
+    xxxx xxxxxxx1 0x001 PwdMode
+    xxxx xxxxxx1x 0x002 Page
+    xxxx xxxxx1xx 0x004 testMode
+    xxxx xxx11xxx 0x018 downlink mode
+    xxxx xx1xxxxx 0x020 !reg_readmode
+    xxxx x1xxxxxx 0x040 called for a read, so no data packet
+    xxxx 1xxxxxxx 0x080 reset
+    xxx1 xxxxxxxx 0x100 brute force
+    111x xxxxxxxx 0xE00 Block
+    */
+
+    uint8_t downlink_mode = (arg >> 3) & 0x03;
+    uint8_t i             = 0;
+    uint8_t BitStream[10];  // Max Downlink Command size ~74 bits, so 10 bytes (80 bits)
+    uint8_t BitStreamLen  = 0;
+    uint8_t SendBits;
+    uint8_t start_wait    = 4;
+    bool    brute_mem     = (arg & 0x100);
+    uint8_t Block         = (arg >> 9) & 0x07;
+
+    if (brute_mem) start_wait = 0;
+
+    // Build Bit Stream to send.
+    memset(BitStream, 0x00, sizeof(BitStream));
+
+    BitStreamLen = 0; // Ensure 0 bit index to start.
+
+    // Add Leading 0 and 1 of 4 reference bit
+    if ((downlink_mode == T55xx_DLMode_Leading0) || (downlink_mode == T55xx_DLMode_1of4))
+        BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, 0, 1, sizeof(BitStream));
+
+    // Add extra reference 0 for 1 of 4
+    if (downlink_mode == T55xx_DLMode_1of4)
+        BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, 0, 1, sizeof(BitStream));
+
+    // Add Opcode
+    if (t55_send_Reset) {
+        //  Reset : r*) 00
+        BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, 0, 2, sizeof(BitStream));
+    } else {
+        if (t55_send_TestMode) Dbprintf("TestMODE");
+        BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, t55_send_TestMode ? 0 : 1, 1, sizeof(BitStream));
+        BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, t55_send_TestMode ? 1 : t55_send_Page, 1, sizeof(BitStream));
+        //if (PwdMode) {
+        if (t55_send_PwdMode) {
+            // Leading 0 and 1 of 4 00 fixed bits if passsword used
+            if ((downlink_mode == T55xx_DLMode_Leading0) || (downlink_mode == T55xx_DLMode_1of4)) {
+                BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, 0, 2, sizeof(BitStream));
+            }
+            BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, Pwd, 32, sizeof(BitStream));
+        }
+
+        // Add Lock bit 0
+        if (!t55_send_RegReadMode) BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, 0, 1, sizeof(BitStream));
+
+        // Add Data if a write command
+        if (!t55_send_ReadCmd) BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, Data, 32, sizeof(BitStream));
+
+        // Add Address
+        if (!t55_send_RegReadMode) BitStreamLen = T55xx_SetBits(BitStream, BitStreamLen, Block, 3, sizeof(BitStream));
+    }
+
+    // Send Bits to T55xx
     // Set up FPGA, 125kHz
     LFSetupFPGAForADC(95, true);
+
     // make sure tag is fully powered up...
-    WaitMS(4);
+    WaitMS(start_wait);
 
     // Trigger T55x7 in mode.
     FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
-    WaitUS(t_config.start_gap);
+    WaitUS(T55xx_Timing.m[downlink_mode].start_gap * 8);
 
-    // reset tag - op code 00
-    T55xxWriteBit(0);
-    T55xxWriteBit(0);
+    // If long leading 0 send long reference pulse
+    if (downlink_mode ==  T55xx_DLMode_LLR)
+        T55xxWriteBit(T55xx_LongLeadingReference, downlink_mode);//Timing); // Send Long Leading Start Reference
 
-    TurnReadLFOn(t_config.read_gap);
+    if ((downlink_mode ==  T55xx_DLMode_1of4) && (BitStreamLen > 0)) { // 1 of 4 need to send 2 bits at a time
+        for (i = 0; i < BitStreamLen - 1; i += 2) {
+            SendBits  = (BitStream[BitStream_Byte(i)] >> (BitStream_Bit(i)) & 1) << 1;       // Bit i
+            SendBits += (BitStream[BitStream_Byte(i + 1)] >> (BitStream_Bit(i + 1)) & 1);    // Bit i+1;
+            T55xxWriteBit(SendBits & 3, downlink_mode);//Timing);
+        }
+    } else {
+        for (i = 0; i < BitStreamLen; i++) {
+            SendBits = (BitStream[BitStream_Byte(i)] >> BitStream_Bit(i));
+            T55xxWriteBit(SendBits & 1, downlink_mode);//Timing);
+        }
+    }
+}
+
+// Send T5577 reset command then read stream (see if we can identify the start of the stream)
+void T55xxResetRead(uint8_t flags) {
+
+    uint8_t downlink_mode = ((flags >> 3) & 3);
+    uint8_t arg           = 0x80 | downlink_mode;
+
+    LED_A_ON();
+
+    //clear buffer now so it does not interfere with timing later
+    BigBuf_Clear_keep_EM();
+
+    T55xx_SendCMD(0, 0, arg);
+
+    TurnReadLFOn(T55xx_Timing.m[downlink_mode].read_gap);
 
     // Acquisition
     DoPartialAcquisition(0, true, BigBuf_max_traceLen(), 0);
@@ -1390,54 +1662,34 @@ void T55xxResetRead(void) {
     // Turn the field off
     FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF); // field off
     reply_mix(CMD_ACK, 0, 0, 0, 0, 0);
+
     LED_A_OFF();
 }
 
 // Write one card block in page 0, no lock
-void T55xxWriteBlockExt(uint32_t data, uint8_t blockno, uint32_t pwd, uint8_t flags) {
+//void T55xxWriteBlockExt(uint32_t data, uint8_t blockno, uint32_t pwd, uint8_t flags) {
+void T55xxWriteBlock(uint8_t *data) {
+
+    /*
+    flag bits
+    xxxxxxx1 0x01 PwdMode
+    xxxxxx1x 0x02 Page
+    xxxxx1xx 0x04 testMode
+    xxx11xxx 0x18 downlink mode
+    xx1xxxxx 0x20 !reg_readmode
+    x1xxxxxx 0x40 called for a read, so no data packet
+    1xxxxxxx 0x80 reset
+    */
+
+    t55xx_write_block_t *c = (t55xx_write_block_t *)data;
+    // c->data, c->blockno, c->pwd, c->flags
+
+    bool testMode = ((c->flags & 0x04) == 0x04);
+
+    c->flags &= (0xff ^ 0x40); // Called for a write, so ensure it is clear/0
+
     LED_A_ON();
-    bool pwd_mode = (flags & 0x1);
-    uint8_t page = (flags & 0x2) >> 1;
-    bool test_mode = (flags & 0x4) >> 2;
-    uint32_t i = 0;
-
-    // Set up FPGA, 125kHz
-    LFSetupFPGAForADC(95, true);
-
-    // make sure tag is fully powered up...
-    WaitMS(4);
-
-    // Trigger T55x7 in mode.
-    FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
-    WaitUS(t_config.start_gap);
-
-    if (test_mode) {
-        Dbprintf("T55xx writing with %s", _YELLOW_("test mode enabled"));
-        // undocmented testmode opcode 01
-        T55xxWriteBit(0);
-        T55xxWriteBit(1);
-    } else {
-        // std opcode 10 == page 0
-        // std opcode 11 == page 1
-        T55xxWriteBit(1);
-        T55xxWriteBit(page);
-    }
-
-    if (pwd_mode) {
-        // Send pwd
-        for (i = 0x80000000; i != 0; i >>= 1)
-            T55xxWriteBit(pwd & i);
-    }
-    // Send lock bit
-    T55xxWriteBit(0);
-
-    // Send data
-    for (i = 0x80000000; i != 0; i >>= 1)
-        T55xxWriteBit(data & i);
-
-    // Send block number
-    for (i = 0x04; i != 0; i >>= 1)
-        T55xxWriteBit(blockno & i);
+    T55xx_SendCMD(c->data, c->pwd, c->flags | (c->blockno << 9)) ; //, false);
 
     // Perform write (nominal is 5.6 ms for T55x7 and 18ms for E5550,
     // so wait a little more)
@@ -1446,7 +1698,7 @@ void T55xxWriteBlockExt(uint32_t data, uint8_t blockno, uint32_t pwd, uint8_t fl
     //  - programming takes ~5.6ms for t5577 ~18ms for E5550 or t5567
     //  so we should wait 1 clock + 5.6ms then read response?
     //  but we need to know we are dealing with t5577 vs t5567 vs e5550 (or q5) marshmellow...
-    if (test_mode) {
+    if (testMode) {
         //TESTMODE TIMING TESTS:
         // <566us does nothing
         // 566-568 switches between wiping to 0s and doing nothing
@@ -1456,7 +1708,6 @@ void T55xxWriteBlockExt(uint32_t data, uint8_t blockno, uint32_t pwd, uint8_t fl
 
     } else {
         TurnReadLFOn(20 * 1000);
-
         //could attempt to do a read to confirm write took
         // as the tag should repeat back the new block
         // until it is reset, but to confirm it we would
@@ -1467,69 +1718,63 @@ void T55xxWriteBlockExt(uint32_t data, uint8_t blockno, uint32_t pwd, uint8_t fl
 
         //DoPartialAcquisition(20, true, 12000);
     }
-
     // turn field off
     FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
+
+    // cmd_send(CMD_ACK,0,0,0,0,0);
+    reply_ng(CMD_T55XX_WRITE_BLOCK, PM3_SUCCESS, NULL, 0);
     LED_A_OFF();
 }
 
-// Write one card block in page 0, no lock
+/*
 // uses NG format
 void T55xxWriteBlock(uint8_t *data) {
     t55xx_write_block_t *c = (t55xx_write_block_t *)data;
     T55xxWriteBlockExt(c->data, c->blockno, c->pwd, c->flags);
-    reply_ng(CMD_T55XX_WRITE_BLOCK, PM3_SUCCESS, NULL, 0);
+    // reply_ng(CMD_T55XX_WRITE_BLOCK, PM3_SUCCESS, NULL, 0);
 }
-
+*/
+/*
 // Read one card block in page [page]
-void T55xxReadBlock(uint8_t page, bool pwd_mode, bool brute_mem, uint8_t block, uint32_t pwd) {
-    LED_A_ON();
-    bool regular_readmode = (block == 0xFF);
-    uint8_t start_wait = 4;
+void T55xxReadBlockExt(uint16_t flags, uint8_t block, uint32_t pwd) {
+    / *
+    flag bits
+    xxxx xxxxxxx1 0x0001 PwdMode
+    xxxx xxxxxx1x 0x0002 Page
+    xxxx xxxxx1xx 0x0004 testMode
+    xxxx xxx11xxx 0x0018 downlink mode
+    xxxx xx1xxxxx 0x0020 !reg_readmode
+    xxxx x1xxxxxx 0x0040 called for a read, so no data packet
+    xxxx 1xxxxxxx 0x0080 reset
+    xxx1 xxxxxxxx 0x0100 brute / leave field on
+    * /
     size_t samples = 12000;
-    uint32_t i;
+bool brute_mem = (flags & 0x0100) >> 8;
 
-    if (brute_mem) {
-        start_wait = 0;
-        samples = 1024;
-    }
+    LED_A_ON();
 
-    //clear buffer now so it does not interfere with timing later
-    BigBuf_Clear_keep_EM();
+    if (brute_mem) samples = 1024;
+
+    // Set Read Flag to ensure SendCMD does not add "data" to the packet
+    flags |= 0x40;
+
+    // RegRead Mode true block = 0xff, so read without an address
+    if (block == 0xff) flags |= 0x20;
 
     //make sure block is at max 7
     block &= 0x7;
 
-    // Set up FPGA, 125kHz to power up the tag
-    LFSetupFPGAForADC(95, true);
-    // make sure tag is fully powered up...
-    WaitMS(start_wait);
+    //clear buffer now so it does not interfere with timing later
+    BigBuf_Clear_keep_EM();
 
-    // Trigger T55x7 Direct Access Mode with start gap
-    FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
-    WaitUS(t_config.start_gap);
-
-    // Opcode 1[page]
-    T55xxWriteBit(1);
-    T55xxWriteBit(page); //Page 0
-
-    if (pwd_mode) {
-        // Send Pwd
-        for (i = 0x80000000; i != 0; i >>= 1)
-            T55xxWriteBit(pwd & i);
-    }
-    // Send a zero bit separation
-    T55xxWriteBit(0);
-
-    // Send Block number (if direct access mode)
-    if (!regular_readmode)
-        for (i = 0x04; i != 0; i >>= 1)
-            T55xxWriteBit(block & i);
+    T55xx_SendCMD(0,  pwd, flags | (block << 9));  //, true);
 
     // Turn field on to read the response
     // 137*8 seems to get to the start of data pretty well...
-    // but we want to go past the start and let the repeating data settle in...
-    TurnReadLFOn(150 * 8);
+    //  but we want to go past the start and let the repeating data settle in...
+
+    // TurnReadLFOn(210*8); // issues with block 1 reads so dropping down seemed to help
+    TurnReadLFOn(137 * 8);
 
     // Acquisition
     // Now do the acquisition
@@ -1542,25 +1787,84 @@ void T55xxReadBlock(uint8_t page, bool pwd_mode, bool brute_mem, uint8_t block,
         LED_A_OFF();
     }
 }
+*/
+// Read one card block in page [page]
+void T55xxReadBlock(uint8_t page, bool pwd_mode, bool brute_mem, uint8_t block, uint32_t pwd, uint8_t downlink_mode) {
+    /*
+    flag bits
+    xxxx xxxxxxx1 0x0001 PwdMode
+    xxxx xxxxxx1x 0x0002 Page
+    xxxx xxxxx1xx 0x0004 testMode
+    xxxx xxx11xxx 0x0018 downlink mode
+    xxxx xx1xxxxx 0x0020 !reg_readmode
+    xxxx x1xxxxxx 0x0040 called for a read, so no data packet
+    xxxx 1xxxxxxx 0x0080 reset
+    xxx1 xxxxxxxx 0x0100 brute / leave field on
+    */
+    uint16_t flags        = 0x0040; // read packet
+    if (pwd_mode)  flags |= 0x0001;
+    if (page)      flags |= 0x0002;
+    flags                |= (downlink_mode & 3) << 3;
+    if (brute_mem) flags |= 0x0100;
 
-void T55xx_ChkPwds() {
+//    T55xxReadBlockExt (flags,block,pwd);
+    size_t samples = 12000;
+    // bool brute_mem = (flags & 0x0100) >> 8;
+
+    LED_A_ON();
+
+    if (brute_mem) samples = 1024;
+
+    //-- Set Read Flag to ensure SendCMD does not add "data" to the packet
+    //-- flags |= 0x40;
+
+    // RegRead Mode true block = 0xff, so read without an address
+    if (block == 0xff) flags |= 0x20;
+
+    //make sure block is at max 7
+    block &= 0x7;
+
+    //clear buffer now so it does not interfere with timing later
+    BigBuf_Clear_keep_EM();
+
+    T55xx_SendCMD(0,  pwd, flags | (block << 9));  //, true);
+
+    // Turn field on to read the response
+    // 137*8 seems to get to the start of data pretty well...
+    //  but we want to go past the start and let the repeating data settle in...
+
+    // TurnReadLFOn(210*8); // issues with block 1 reads so dropping down seemed to help
+    TurnReadLFOn(137 * 8);
+
+    // Acquisition
+    // Now do the acquisition
+    DoPartialAcquisition(0, true, samples, 0);
+
+    // Turn the field off
+    if (!brute_mem) {
+        FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
+        reply_ng(CMD_T55XX_READ_BLOCK, PM3_SUCCESS, NULL, 0);
+        LED_A_OFF();
+    }
+
+}
+
+void T55xx_ChkPwds(uint8_t flags) {
 
     DbpString("[+] T55XX Check pwds using flashmemory starting");
 
     uint8_t ret = 0;
     // First get baseline and setup LF mode.
     // tends to mess up BigBuf
-    uint8_t *buf = BigBuf_get_addr();
-
-    uint32_t b1, baseline = 0;
+    uint8_t  *buf           = BigBuf_get_addr();
+    uint32_t b1, baseline   = 0;
+    uint8_t  downlink_mode  = (flags >> 3) & 0x03;
 
     // collect baseline for failed attempt
     uint8_t x = 32;
     while (x--) {
         b1 = 0;
-
-//        T55xxReadBlock(uint8_t page, bool pwd_mode, bool brute_mem, uint8_t block, uint32_t pwd)
-        T55xxReadBlock(0, 0, true, 1, 0);
+        T55xxReadBlock(0, 0, true, 1, 0, downlink_mode);
         for (uint16_t j = 0; j < 1024; ++j)
             b1 += buf[j];
 
@@ -1575,8 +1879,8 @@ void T55xx_ChkPwds() {
     uint8_t *pwds = BigBuf_get_EM_addr();
     uint16_t pwdCount = 0;
     uint32_t candidate = 0;
-
 #ifdef WITH_FLASH
+
     BigBuf_Clear_EM();
     uint16_t isok = 0;
     uint8_t counter[2] = {0x00, 0x00};
@@ -1605,7 +1909,7 @@ void T55xx_ChkPwds() {
 
         pwd = bytes_to_num(pwds + i * 4, 4);
 
-        T55xxReadBlock(0, true, true, 0, pwd);
+        T55xxReadBlock(0, true, true, 0, pwd, downlink_mode);
 
         // calc mean of BigBuf 1024 samples.
         uint32_t sum = 0;
@@ -1622,8 +1926,6 @@ void T55xx_ChkPwds() {
         Dbprintf("[=] Pwd %08X  | ABS %u", pwd, curr);
 
         if (curr > prev) {
-
-
             Dbprintf("[=]  --> ABS %u  Candidate %08X <--", curr, pwd);
             candidate = pwd;
             prev = curr;
@@ -1639,36 +1941,31 @@ OUT:
     LEDsoff();
 }
 
-void T55xxWakeUp(uint32_t Pwd) {
+void T55xxWakeUp(uint32_t Pwd, uint8_t flags) {
+
+    flags |= 0x01 | 0x40 | 0x20; //Password | Read Call (no data) | reg_read no block
     LED_B_ON();
-    uint32_t i = 0;
 
-    // Set up FPGA, 125kHz
-    LFSetupFPGAForADC(95, true);
-    // make sure tag is fully powered up...
-    WaitMS(4);
+    T55xx_SendCMD(0, Pwd, flags);
 
-    // Trigger T55x7 Direct Access Mode
-    FpgaWriteConfWord(FPGA_MAJOR_MODE_OFF);
-    WaitUS(t_config.start_gap);
-
-    // Opcode 10
-    T55xxWriteBit(1);
-    T55xxWriteBit(0); //Page 0
-
-    // Send Pwd
-    for (i = 0x80000000; i != 0; i >>= 1)
-        T55xxWriteBit(Pwd & i);
-
-    // Turn and leave field on to let the begin repeating transmission
+    //-- Turn and leave field on to let the begin repeating transmission
     TurnReadLFOn(20 * 1000);
 }
 
+
 /*-------------- Cloning routines -----------*/
 void WriteT55xx(uint32_t *blockdata, uint8_t startblock, uint8_t numblocks) {
-    // write last block first and config block last (if included)
-    for (uint8_t i = numblocks + startblock; i > startblock; i--)
-        T55xxWriteBlockExt(blockdata[i - 1], i - 1, 0, 0);
+
+    t55xx_write_block_t cmd;
+    cmd.pwd     = 0;
+    cmd.flags   = 0;
+
+    for (uint8_t i = numblocks + startblock; i > startblock; i--) {
+        cmd.data    = blockdata[i - 1];
+        cmd.blockno = i - 1;
+        T55xxWriteBlock((uint8_t *)&cmd);
+    }
+
 }
 
 // Copy HID id to card and setup block 0 config
diff --git a/armsrc/lfsampling.c b/armsrc/lfsampling.c
index c5a075e7a..c88401229 100644
--- a/armsrc/lfsampling.c
+++ b/armsrc/lfsampling.c
@@ -136,11 +136,11 @@ uint32_t DoAcquisition(uint8_t decimation, uint32_t bits_per_sample, bool averag
     uint32_t sample_total_numbers = 0;
     uint32_t sample_total_saved = 0;
     uint32_t cancel_counter = 0;
-    
+
     uint16_t checker = 0;
-  
+
     while (true) {
-        if ( checker == 1000 ) {
+        if (checker == 1000) {
             if (BUTTON_PRESS() || data_available())
                 break;
             else
@@ -148,7 +148,7 @@ uint32_t DoAcquisition(uint8_t decimation, uint32_t bits_per_sample, bool averag
         } else {
             ++checker;
         }
-    
+
         WDT_HIT();
 
         if (AT91C_BASE_SSC->SSC_SR & AT91C_SSC_RXRDY) {
@@ -293,11 +293,11 @@ void doT55x7Acquisition(size_t sample_size) {
     bool startFound = false;
     bool highFound = false;
     bool lowFound = false;
-    
+
     uint16_t checker = 0;
-  
-    while ( skipCnt < 1000 && (i < bufsize)) {
-        if ( checker == 1000 ) {
+
+    while (skipCnt < 1000 && (i < bufsize)) {
+        if (checker == 1000) {
             if (BUTTON_PRESS() || data_available())
                 break;
             else
@@ -371,9 +371,9 @@ void doCotagAcquisition(size_t sample_size) {
     uint16_t noise_counter = 0;
 
     uint16_t checker = 0;
-  
+
     while ((i < bufsize) && (noise_counter < (COTAG_T1 << 1))) {
-        if ( checker == 1000 ) {
+        if (checker == 1000) {
             if (BUTTON_PRESS() || data_available())
                 break;
             else
@@ -431,9 +431,9 @@ uint32_t doCotagAcquisitionManchester() {
     uint8_t curr = 0, prev = 0;
     uint16_t noise_counter = 0;
     uint16_t checker = 0;
-  
+
     while ((sample_counter < bufsize) && (noise_counter < (COTAG_T1 << 1))) {
-        if ( checker == 1000 ) {
+        if (checker == 1000) {
             if (BUTTON_PRESS() || data_available())
                 break;
             else
diff --git a/armsrc/mifarecmd.c b/armsrc/mifarecmd.c
index 26b6583fc..2d56a8a37 100644
--- a/armsrc/mifarecmd.c
+++ b/armsrc/mifarecmd.c
@@ -14,8 +14,11 @@
 //-----------------------------------------------------------------------------
 
 #include "mifarecmd.h"
+
 #include <inttypes.h>
 
+#include "pmflash.h"
+
 #ifndef HARDNESTED_AUTHENTICATION_TIMEOUT
 # define HARDNESTED_AUTHENTICATION_TIMEOUT  848     // card times out 1ms after wrong authentication (according to NXP documentation)
 #endif
diff --git a/armsrc/mifaredesfire.c b/armsrc/mifaredesfire.c
index bda31790d..c57fed866 100644
--- a/armsrc/mifaredesfire.c
+++ b/armsrc/mifaredesfire.c
@@ -433,15 +433,9 @@ void MifareDES_Auth1(uint8_t arg0, uint8_t arg1, uint8_t arg2,  uint8_t *datain)
             desfirekey_t key = &defaultkey;
             Desfire_aes_key_new(keybytes, key);
 
-            AesCtx ctx;
+            mbedtls_aes_context ctx;
             uint8_t IV[16] = {0x00};
-            if (AesCtxIni(&ctx, IV, key->data, KEY128, CBC) < 0) {
-                if (DBGLEVEL >= 4) {
-                    DbpString("AES context failed to init");
-                }
-                OnError(7);
-                return;
-            }
+            mbedtls_aes_init(&ctx);
 
             cmd[0] = AUTHENTICATE_AES;
             cmd[1] = 0x00;  //keynumber
@@ -457,13 +451,27 @@ void MifareDES_Auth1(uint8_t arg0, uint8_t arg1, uint8_t arg2,  uint8_t *datain)
             memcpy(encRndB, resp + 3, 16);
 
             // dekryptera tagnonce.
-            AesDecrypt(&ctx, encRndB, decRndB, 16);
+            if (mbedtls_aes_setkey_dec(&ctx, key->data, 128) != 0) {
+                if (DBGLEVEL >= 4) {
+                    DbpString("mbedtls_aes_setkey_dec failed");
+                }
+                OnError(7);
+                return;
+            }
+            mbedtls_aes_crypt_cbc(&ctx, MBEDTLS_AES_DECRYPT, 16, IV, encRndB, decRndB);
             rol(decRndB, 16);
             uint8_t nonce[16] = {0x00};
             memcpy(both, nonce, 16);
             memcpy(both + 16, decRndB, 16);
             uint8_t encBoth[32] = {0x00};
-            AesEncrypt(&ctx, both, encBoth, 32);
+            if (mbedtls_aes_setkey_enc(&ctx, key->data, 128) != 0) {
+                if (DBGLEVEL >= 4) {
+                    DbpString("mbedtls_aes_setkey_enc failed");
+                }
+                OnError(7);
+                return;
+            }
+            mbedtls_aes_crypt_cbc(&ctx, MBEDTLS_AES_ENCRYPT, 32, IV, both, encBoth);
 
             cmd[0] = ADDITIONAL_FRAME;
             memcpy(cmd + 1, encBoth, 32);
diff --git a/armsrc/mifaresim.c b/armsrc/mifaresim.c
index 714beb284..0e64f00d1 100644
--- a/armsrc/mifaresim.c
+++ b/armsrc/mifaresim.c
@@ -42,30 +42,36 @@ static bool IsTrailerAccessAllowed(uint8_t blockNo, uint8_t keytype, uint8_t act
                  | ((sector_trailer[8] >> 7) & 0x01);
     switch (action) {
         case AC_KEYA_READ: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsTrailerAccessAllowed: AC_KEYA_READ");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsTrailerAccessAllowed: AC_KEYA_READ");
             return false;
         }
         case AC_KEYA_WRITE: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsTrailerAccessAllowed: AC_KEYA_WRITE");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsTrailerAccessAllowed: AC_KEYA_WRITE");
             return ((keytype == AUTHKEYA && (AC == 0x00 || AC == 0x01))
                     || (keytype == AUTHKEYB && (AC == 0x04 || AC == 0x03)));
         }
         case AC_KEYB_READ: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsTrailerAccessAllowed: AC_KEYB_READ");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsTrailerAccessAllowed: AC_KEYB_READ");
             return (keytype == AUTHKEYA && (AC == 0x00 || AC == 0x02 || AC == 0x01));
         }
         case AC_KEYB_WRITE: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsTrailerAccessAllowed: AC_KEYB_WRITE");
-            return ((keytype == AUTHKEYA && (AC == 0x00 || AC == 0x04))
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsTrailerAccessAllowed: AC_KEYB_WRITE");
+            return ((keytype == AUTHKEYA && (AC == 0x00 || AC == 0x01))
                     || (keytype == AUTHKEYB && (AC == 0x04 || AC == 0x03)));
         }
         case AC_AC_READ: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsTrailerAccessAllowed: AC_AC_READ");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsTrailerAccessAllowed: AC_AC_READ");
             return ((keytype == AUTHKEYA)
                     || (keytype == AUTHKEYB && !(AC == 0x00 || AC == 0x02 || AC == 0x01)));
         }
         case AC_AC_WRITE: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsTrailerAccessAllowed: AC_AC_WRITE");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsTrailerAccessAllowed: AC_AC_WRITE");
             return ((keytype == AUTHKEYA && (AC == 0x01))
                     || (keytype == AUTHKEYB && (AC == 0x03 || AC == 0x05)));
         }
@@ -93,46 +99,54 @@ static bool IsDataAccessAllowed(uint8_t blockNo, uint8_t keytype, uint8_t action
             AC = ((sector_trailer[7] >> 2) & 0x04)
                  | ((sector_trailer[8] << 1) & 0x02)
                  | ((sector_trailer[8] >> 4) & 0x01);
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsDataAccessAllowed: case 0x00 - %02x", AC);
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsDataAccessAllowed: case 0x00 - %02x", AC);
             break;
         }
         case 0x01: {
             AC = ((sector_trailer[7] >> 3) & 0x04)
                  | ((sector_trailer[8] >> 0) & 0x02)
                  | ((sector_trailer[8] >> 5) & 0x01);
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsDataAccessAllowed: case 0x01 - %02x", AC);
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsDataAccessAllowed: case 0x01 - %02x", AC);
             break;
         }
         case 0x02: {
             AC = ((sector_trailer[7] >> 4) & 0x04)
                  | ((sector_trailer[8] >> 1) & 0x02)
                  | ((sector_trailer[8] >> 6) & 0x01);
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsDataAccessAllowed: case 0x02  - %02x", AC);
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsDataAccessAllowed: case 0x02  - %02x", AC);
             break;
         }
         default:
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsDataAccessAllowed: Error");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsDataAccessAllowed: Error");
             return false;
     }
 
     switch (action) {
         case AC_DATA_READ: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsDataAccessAllowed - AC_DATA_READ: OK");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsDataAccessAllowed - AC_DATA_READ: OK");
             return ((keytype == AUTHKEYA && !(AC == 0x03 || AC == 0x05 || AC == 0x07))
                     || (keytype == AUTHKEYB && !(AC == 0x07)));
         }
         case AC_DATA_WRITE: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsDataAccessAllowed - AC_DATA_WRITE: OK");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsDataAccessAllowed - AC_DATA_WRITE: OK");
             return ((keytype == AUTHKEYA && (AC == 0x00))
                     || (keytype == AUTHKEYB && (AC == 0x00 || AC == 0x04 || AC == 0x06 || AC == 0x03)));
         }
         case AC_DATA_INC: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("IsDataAccessAllowed - AC_DATA_INC: OK");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("IsDataAccessAllowed - AC_DATA_INC: OK");
             return ((keytype == AUTHKEYA && (AC == 0x00))
                     || (keytype == AUTHKEYB && (AC == 0x00 || AC == 0x06)));
         }
         case AC_DATA_DEC_TRANS_REST: {
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("AC_DATA_DEC_TRANS_REST: OK");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("AC_DATA_DEC_TRANS_REST: OK");
             return ((keytype == AUTHKEYA && (AC == 0x00 || AC == 0x06 || AC == 0x01))
                     || (keytype == AUTHKEYB && (AC == 0x00 || AC == 0x06 || AC == 0x01)));
         }
@@ -160,25 +174,25 @@ static bool MifareSimInit(uint16_t flags, uint8_t *datain, tag_response_info_t *
 
     // SAK
     static uint8_t rSAK_Mini = 0x09;    // mifare Mini
-    static uint8_t rSAK_1k   = 0x08;	// mifare 1k
-    static uint8_t rSAK_2k   = 0x08;	// mifare 2k with RATS support
-    static uint8_t rSAK_4k   = 0x18;	// mifare 4k
+    static uint8_t rSAK_1k   = 0x08;    // mifare 1k
+    static uint8_t rSAK_2k   = 0x08;    // mifare 2k with RATS support
+    static uint8_t rSAK_4k   = 0x18;    // mifare 4k
 
-    static uint8_t rUIDBCC1[]   = {0x00, 0x00, 0x00, 0x00, 0x00};	// UID 1st cascade level
-    static uint8_t rUIDBCC1b4[] = {0x00, 0x00, 0x00, 0x00};	        // UID 1st cascade level, last 4 bytes
-    static uint8_t rUIDBCC1b3[] = {0x00, 0x00, 0x00};	            // UID 1st cascade level, last 3 bytes
-    static uint8_t rUIDBCC1b2[] = {0x00, 0x00};	                    // UID 1st cascade level, last 2 bytes
-    static uint8_t rUIDBCC1b1[] = {0x00};	                        // UID 1st cascade level, last byte
-    static uint8_t rUIDBCC2[]   = {0x00, 0x00, 0x00, 0x00, 0x00};	// UID 2nd cascade level
-    static uint8_t rUIDBCC2b4[] = {0x00, 0x00, 0x00, 0x00};	        // UID 2st cascade level, last 4 bytes
-    static uint8_t rUIDBCC2b3[] = {0x00, 0x00, 0x00};	            // UID 2st cascade level, last 3 bytes
-    static uint8_t rUIDBCC2b2[] = {0x00, 0x00};	                    // UID 2st cascade level, last 2 bytes
-    static uint8_t rUIDBCC2b1[] = {0x00};	                        // UID 2st cascade level, last byte
-    static uint8_t rUIDBCC3[]   = {0x00, 0x00, 0x00, 0x00, 0x00};	// UID 3nd cascade level
-    static uint8_t rUIDBCC3b4[] = {0x00, 0x00, 0x00, 0x00};	        // UID 3st cascade level, last 4 bytes
-    static uint8_t rUIDBCC3b3[] = {0x00, 0x00, 0x00};	            // UID 3st cascade level, last 3 bytes
-    static uint8_t rUIDBCC3b2[] = {0x00, 0x00};	                    // UID 3st cascade level, last 2 bytes
-    static uint8_t rUIDBCC3b1[] = {0x00};	                        // UID 3st cascade level, last byte
+    static uint8_t rUIDBCC1[]   = {0x00, 0x00, 0x00, 0x00, 0x00};   // UID 1st cascade level
+    static uint8_t rUIDBCC1b4[] = {0x00, 0x00, 0x00, 0x00};         // UID 1st cascade level, last 4 bytes
+    static uint8_t rUIDBCC1b3[] = {0x00, 0x00, 0x00};               // UID 1st cascade level, last 3 bytes
+    static uint8_t rUIDBCC1b2[] = {0x00, 0x00};                     // UID 1st cascade level, last 2 bytes
+    static uint8_t rUIDBCC1b1[] = {0x00};                           // UID 1st cascade level, last byte
+    static uint8_t rUIDBCC2[]   = {0x00, 0x00, 0x00, 0x00, 0x00};   // UID 2nd cascade level
+    static uint8_t rUIDBCC2b4[] = {0x00, 0x00, 0x00, 0x00};         // UID 2st cascade level, last 4 bytes
+    static uint8_t rUIDBCC2b3[] = {0x00, 0x00, 0x00};               // UID 2st cascade level, last 3 bytes
+    static uint8_t rUIDBCC2b2[] = {0x00, 0x00};                     // UID 2st cascade level, last 2 bytes
+    static uint8_t rUIDBCC2b1[] = {0x00};                           // UID 2st cascade level, last byte
+    static uint8_t rUIDBCC3[]   = {0x00, 0x00, 0x00, 0x00, 0x00};   // UID 3nd cascade level
+    static uint8_t rUIDBCC3b4[] = {0x00, 0x00, 0x00, 0x00};         // UID 3st cascade level, last 4 bytes
+    static uint8_t rUIDBCC3b3[] = {0x00, 0x00, 0x00};               // UID 3st cascade level, last 3 bytes
+    static uint8_t rUIDBCC3b2[] = {0x00, 0x00};                     // UID 3st cascade level, last 2 bytes
+    static uint8_t rUIDBCC3b1[] = {0x00};                           // UID 3st cascade level, last byte
 
     static uint8_t rATQA[]     = {0x00, 0x00};             // Current ATQA
     static uint8_t rSAK[]      = {0x00, 0x00, 0x00};       // Current SAK, CRC
@@ -254,17 +268,18 @@ static bool MifareSimInit(uint16_t flags, uint8_t *datain, tag_response_info_t *
     }
 
     // Prepare UID arrays
-    if ((flags & FLAG_4B_UID_IN_DATA) == FLAG_4B_UID_IN_DATA) { 	// get UID from datain
+    if ((flags & FLAG_4B_UID_IN_DATA) == FLAG_4B_UID_IN_DATA) { // get UID from datain
         memcpy(rUIDBCC1, datain, 4);
         *uid_len = 4;
-        if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("MifareSimInit - FLAG_4B_UID_IN_DATA => Get UID from datain: %02X - Flag: %02X - UIDBCC1: %02X", FLAG_4B_UID_IN_DATA, flags, rUIDBCC1);
+        if (DBGLEVEL >= DBG_EXTENDED)
+            Dbprintf("MifareSimInit - FLAG_4B_UID_IN_DATA => Get UID from datain: %02X - Flag: %02X - UIDBCC1: %02X", FLAG_4B_UID_IN_DATA, flags, rUIDBCC1);
 
 
         // save CUID
         *cuid = bytes_to_num(rUIDBCC1, 4);
         // BCC
         rUIDBCC1[4] = rUIDBCC1[0] ^ rUIDBCC1[1] ^ rUIDBCC1[2] ^ rUIDBCC1[3];
-        if (DBGLEVEL >= DBG_NONE)	{
+        if (DBGLEVEL >= DBG_NONE) {
             Dbprintf("4B UID: %02x%02x%02x%02x", rUIDBCC1[0], rUIDBCC1[1], rUIDBCC1[2], rUIDBCC1[3]);
         }
 
@@ -275,7 +290,8 @@ static bool MifareSimInit(uint16_t flags, uint8_t *datain, tag_response_info_t *
         memcpy(&rUIDBCC1[1], datain, 3);
         memcpy(rUIDBCC2, datain + 3, 4);
         *uid_len = 7;
-        if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("MifareSimInit - FLAG_7B_UID_IN_DATA => Get UID from datain: %02X - Flag: %02X - UIDBCC1: %02X", FLAG_7B_UID_IN_DATA, flags, rUIDBCC1);
+        if (DBGLEVEL >= DBG_EXTENDED)
+            Dbprintf("MifareSimInit - FLAG_7B_UID_IN_DATA => Get UID from datain: %02X - Flag: %02X - UIDBCC1: %02X", FLAG_7B_UID_IN_DATA, flags, rUIDBCC1);
 
         // save CUID
         *cuid = bytes_to_num(rUIDBCC2, 4);
@@ -284,7 +300,7 @@ static bool MifareSimInit(uint16_t flags, uint8_t *datain, tag_response_info_t *
         // BCC
         rUIDBCC1[4] = rUIDBCC1[0] ^ rUIDBCC1[1] ^ rUIDBCC1[2] ^ rUIDBCC1[3];
         rUIDBCC2[4] = rUIDBCC2[0] ^ rUIDBCC2[1] ^ rUIDBCC2[2] ^ rUIDBCC2[3];
-        if (DBGLEVEL >= DBG_NONE)	{
+        if (DBGLEVEL >= DBG_NONE) {
             Dbprintf("7B UID: %02x %02x %02x %02x %02x %02x %02x",
                      rUIDBCC1[1], rUIDBCC1[2], rUIDBCC1[3], rUIDBCC2[0], rUIDBCC2[1], rUIDBCC2[2], rUIDBCC2[3]);
         }
@@ -297,7 +313,8 @@ static bool MifareSimInit(uint16_t flags, uint8_t *datain, tag_response_info_t *
         memcpy(&rUIDBCC2[1], datain + 3, 3);
         memcpy(rUIDBCC3,    datain + 6, 4);
         *uid_len = 10;
-        if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("MifareSimInit - FLAG_10B_UID_IN_DATA => Get UID from datain: %02X - Flag: %02X - UIDBCC1: %02X", FLAG_10B_UID_IN_DATA, flags, rUIDBCC1);
+        if (DBGLEVEL >= DBG_EXTENDED)
+            Dbprintf("MifareSimInit - FLAG_10B_UID_IN_DATA => Get UID from datain: %02X - Flag: %02X - UIDBCC1: %02X", FLAG_10B_UID_IN_DATA, flags, rUIDBCC1);
 
         // save CUID
         *cuid = bytes_to_num(rUIDBCC3, 4);
@@ -309,7 +326,7 @@ static bool MifareSimInit(uint16_t flags, uint8_t *datain, tag_response_info_t *
         rUIDBCC2[4] = rUIDBCC2[0] ^ rUIDBCC2[1] ^ rUIDBCC2[2] ^ rUIDBCC2[3];
         rUIDBCC3[4] = rUIDBCC3[0] ^ rUIDBCC3[1] ^ rUIDBCC3[2] ^ rUIDBCC3[3];
 
-        if (DBGLEVEL >= DBG_NONE)	{
+        if (DBGLEVEL >= DBG_NONE) {
             Dbprintf("10B UID: %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x",
                      rUIDBCC1[1], rUIDBCC1[2], rUIDBCC1[3],
                      rUIDBCC2[1], rUIDBCC2[2], rUIDBCC2[3],
@@ -347,23 +364,23 @@ static bool MifareSimInit(uint16_t flags, uint8_t *datain, tag_response_info_t *
 
 #define TAG_RESPONSE_COUNT 18
     static tag_response_info_t responses_init[TAG_RESPONSE_COUNT] = {
-        { .response = rATQA,     .response_n = sizeof(rATQA)     },		// Answer to request - respond with card type
+        { .response = rATQA,     .response_n = sizeof(rATQA)     },     // Answer to request - respond with card type
         { .response = rSAK,      .response_n = sizeof(rSAK)      },     //
-        { .response = rSAKuid,   .response_n = sizeof(rSAKuid)   },	    //
+        { .response = rSAKuid,   .response_n = sizeof(rSAKuid)   },     //
         // Do not reorder. Block used via relative index of rUIDBCC1
-        { .response = rUIDBCC1,  .response_n = sizeof(rUIDBCC1)  },	  	// Anticollision cascade1 - respond with first part of uid
+        { .response = rUIDBCC1,  .response_n = sizeof(rUIDBCC1)  },     // Anticollision cascade1 - respond with first part of uid
         { .response = rUIDBCC1b4, .response_n = sizeof(rUIDBCC1b4)},
         { .response = rUIDBCC1b3, .response_n = sizeof(rUIDBCC1b3)},
         { .response = rUIDBCC1b2, .response_n = sizeof(rUIDBCC1b2)},
         { .response = rUIDBCC1b1, .response_n = sizeof(rUIDBCC1b1)},
         // Do not reorder. Block used via relative index of rUIDBCC2
-        { .response = rUIDBCC2,  .response_n = sizeof(rUIDBCC2)  },		// Anticollision cascade2 - respond with 2nd part of uid
+        { .response = rUIDBCC2,  .response_n = sizeof(rUIDBCC2)  },     // Anticollision cascade2 - respond with 2nd part of uid
         { .response = rUIDBCC2b4, .response_n = sizeof(rUIDBCC2b4)},
         { .response = rUIDBCC2b3, .response_n = sizeof(rUIDBCC2b3)},
         { .response = rUIDBCC2b2, .response_n = sizeof(rUIDBCC2b2)},
         { .response = rUIDBCC2b1, .response_n = sizeof(rUIDBCC2b1)},
         // Do not reorder. Block used via relative index of rUIDBCC3
-        { .response = rUIDBCC3,  .response_n = sizeof(rUIDBCC3)  },		// Anticollision cascade3 - respond with 3th part of uid
+        { .response = rUIDBCC3,  .response_n = sizeof(rUIDBCC3)  },     // Anticollision cascade3 - respond with 3th part of uid
         { .response = rUIDBCC3b4, .response_n = sizeof(rUIDBCC3b4)},
         { .response = rUIDBCC3b3, .response_n = sizeof(rUIDBCC3b3)},
         { .response = rUIDBCC3b2, .response_n = sizeof(rUIDBCC3b2)},
@@ -407,11 +424,11 @@ static bool MifareSimInit(uint16_t flags, uint8_t *datain, tag_response_info_t *
 *MIFARE 1K simulate.
 *
 *@param flags :
-*	FLAG_INTERACTIVE - In interactive mode, we are expected to finish the operation with an ACK
+* FLAG_INTERACTIVE - In interactive mode, we are expected to finish the operation with an ACK
 * FLAG_4B_UID_IN_DATA - means that there is a 4-byte UID in the data-section, we're expected to use that
 * FLAG_7B_UID_IN_DATA - means that there is a 7-byte UID in the data-section, we're expected to use that
-* FLAG_10B_UID_IN_DATA	- use 10-byte UID in the data-section not finished
-*	FLAG_NR_AR_ATTACK  - means we should collect NR_AR responses for bruteforcing later
+* FLAG_10B_UID_IN_DATA - use 10-byte UID in the data-section not finished
+* FLAG_NR_AR_ATTACK - means we should collect NR_AR responses for bruteforcing later
 *@param exitAfterNReads, exit simulation after n blocks have been read, 0 is infinite ...
 * (unless reader attack mode enabled then it runs util it gets enough nonces to recover all keys attmpted)
 */
@@ -421,7 +438,7 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
     uint8_t uid_len = 0; // 4,7, 10
     uint32_t cuid = 0;
 
-    int vHf = 0;	// in mV
+    int vHf = 0; // in mV
 
     uint32_t selTimer = 0;
     uint32_t authTimer = 0;
@@ -445,7 +462,7 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
     struct Crypto1State *pcs;
     pcs = &mpcs;
 
-    uint32_t numReads = 0;	//Counts numer of times reader reads a block
+    uint32_t numReads = 0; //Counts numer of times reader reads a block
     uint8_t receivedCmd[MAX_MIFARE_FRAME_SIZE] = {0x00};
     uint8_t receivedCmd_dec[MAX_MIFARE_FRAME_SIZE] = {0x00};
     uint8_t receivedCmd_par[MAX_MIFARE_PARITY_SIZE] = {0x00};
@@ -457,8 +474,6 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
     uint8_t *rats = NULL;
     uint8_t rats_len = 0;
 
-    uint8_t rAUTH_AT[] = {0x00, 0x00, 0x00, 0x00};
-
     //Here, we collect UID,sector,keytype,NT,AR,NR,NT2,AR2,NR2
     // This will be used in the reader-only attack.
 
@@ -469,11 +484,11 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
 
     uint8_t ar_nr_collected[ATTACK_KEY_COUNT * 2]; //*2 for 2nd attack type (moebius)
     memset(ar_nr_collected, 0x00, sizeof(ar_nr_collected));
-    uint8_t	nonce1_count = 0;
-    uint8_t	nonce2_count = 0;
-    uint8_t	moebius_n_count = 0;
+    uint8_t nonce1_count = 0;
+    uint8_t nonce2_count = 0;
+    uint8_t moebius_n_count = 0;
     bool gettingMoebius = false;
-    uint8_t	mM = 0; //moebius_modifier for collection storage
+    uint8_t mM = 0; //moebius_modifier for collection storage
 
     // Authenticate response - nonce
     uint8_t rAUTH_NT[4];
@@ -522,18 +537,21 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
         if (res == 2) { //Field is off!
             LEDsoff();
             cardSTATE = MFEMUL_NOFIELD;
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("cardSTATE = MFEMUL_NOFIELD");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("cardSTATE = MFEMUL_NOFIELD");
             continue;
         } else if (res == 1) { // button pressed
             button_pushed = true;
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("Button pressed");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("Button pressed");
             break;
         }
 
         // WUPA in HALTED state or REQA or WUPA in any other state
         if (receivedCmd_len == 1 && ((receivedCmd[0] == ISO14443A_CMD_REQA && cardSTATE != MFEMUL_HALTED) || receivedCmd[0] == ISO14443A_CMD_WUPA)) {
             selTimer = GetTickCount();
-            if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("EmSendPrecompiledCmd(&responses[ATQA]);");
+            if (DBGLEVEL >= DBG_EXTENDED)
+                Dbprintf("EmSendPrecompiledCmd(&responses[ATQA]);");
             EmSendPrecompiledCmd(&responses[ATQA]);
 
             // init crypto block
@@ -552,12 +570,15 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
 
         switch (cardSTATE) {
             case MFEMUL_NOFIELD:
-                if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("MFEMUL_NOFIELD");
+                if (DBGLEVEL >= DBG_EXTENDED)
+                    Dbprintf("MFEMUL_NOFIELD");
             case MFEMUL_HALTED:
-                if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("MFEMUL_HALTED");
+                if (DBGLEVEL >= DBG_EXTENDED)
+                    Dbprintf("MFEMUL_HALTED");
             case MFEMUL_IDLE: {
                 LogTrace(uart->output, uart->len, uart->startTime * 16 - DELAY_AIR2ARM_AS_TAG, uart->endTime * 16 - DELAY_AIR2ARM_AS_TAG, uart->parity, true);
-                if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("MFEMUL_IDLE");
+                if (DBGLEVEL >= DBG_EXTENDED)
+                    Dbprintf("MFEMUL_IDLE");
                 break;
             }
 
@@ -606,8 +627,8 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
                 if (receivedCmd_len == 9 && receivedCmd[1] == 0x70) {
                     if (memcmp(&receivedCmd[2], responses[uid_index].response, 4) == 0) {
                         bool cl_finished = (uid_len == 4  && uid_index == UIDBCC1) ||
-                                        (uid_len == 7  && uid_index == UIDBCC2) ||
-                                        (uid_len == 10 && uid_index == UIDBCC3);
+                                           (uid_len == 7  && uid_index == UIDBCC2) ||
+                                           (uid_len == 10 && uid_index == UIDBCC3);
                         EmSendPrecompiledCmd(&responses[cl_finished ? SAK : SAKuid]);
                         if (DBGLEVEL >= DBG_EXTENDED) Dbprintf("SELECT CLx %02x%02x%02x%02x received", receivedCmd[2], receivedCmd[3], receivedCmd[4], receivedCmd[5]);
                         if (cl_finished) {
@@ -625,7 +646,8 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
                 }
 
                 // Incoming anti-collision frame
-                if (receivedCmd_len >= 2 && receivedCmd_len <= 6 && receivedCmd[1] == 0x50) {
+                // receivedCmd[1] indicates number of byte and bit collision, supports only for bit collision is zero
+                if (receivedCmd_len >= 3 && receivedCmd_len <= 6 && (receivedCmd[1] & 0x0f) == 0) {
                     // we can process only full-byte frame anti-collision procedure
                     if (memcmp(&receivedCmd[2], responses[uid_index].response, receivedCmd_len - 2) == 0) {
                         // response missing part of UID via relative array index
@@ -650,7 +672,8 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
             // WORK
             case MFEMUL_WORK: {
 
-                if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("[MFEMUL_WORK] Enter in case");
+                if (DBGLEVEL >= DBG_EXTENDED)
+                    Dbprintf("[MFEMUL_WORK] Enter in case");
 
                 if (receivedCmd_len == 0) {
                     if (DBGLEVEL >= DBG_EXTENDED) Dbprintf("[MFEMUL_WORK] NO CMD received");
@@ -712,7 +735,7 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
                         crypto1_word(pcs, cuid ^ nonce, 0);
                         // rAUTH_NT contains prepared nonce for authenticate
                         EmSendCmd(rAUTH_NT, sizeof(rAUTH_NT));
-                        if (DBGLEVEL >= DBG_EXTENDED) Dbprintf("[MFEMUL_WORK] Reader authenticating for block %d (0x%02x) with key %c - nonce: %02X - ciud: %02X", receivedCmd_dec[1], receivedCmd_dec[1], (cardAUTHKEY == 0) ? 'A' : 'B', rAUTH_AT, cuid);
+                        if (DBGLEVEL >= DBG_EXTENDED) Dbprintf("[MFEMUL_WORK] Reader authenticating for block %d (0x%02x) with key %c - nonce: %02X - ciud: %02X", receivedCmd_dec[1], receivedCmd_dec[1], (cardAUTHKEY == 0) ? 'A' : 'B', rAUTH_NT, cuid);
                     } else {
                         // nested authentication
                         /*
@@ -800,20 +823,20 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
                     if (IsSectorTrailer(blockNo)) {
 
                         if (!IsAccessAllowed(blockNo, cardAUTHKEY, AC_KEYA_READ)) {
-                            memset(response, 0x00, 6); 	// keyA can never be read
+                            memset(response, 0x00, 6); // keyA can never be read
                             if (DBGLEVEL >= DBG_EXTENDED) Dbprintf("[MFEMUL_WORK - IsSectorTrailer] keyA can never be read - block %d (0x%02x)", blockNo, blockNo);
                         }
                         if (!IsAccessAllowed(blockNo, cardAUTHKEY, AC_KEYB_READ)) {
-                            memset(response + 10, 0x00, 6); 	// keyB cannot be read
+                            memset(response + 10, 0x00, 6); // keyB cannot be read
                             if (DBGLEVEL >= DBG_EXTENDED) Dbprintf("[MFEMUL_WORK - IsSectorTrailer] keyB cannot be read - block %d (0x%02x)", blockNo, blockNo);
                         }
                         if (!IsAccessAllowed(blockNo, cardAUTHKEY, AC_AC_READ)) {
-                            memset(response + 6, 0x00, 4); 	// AC bits cannot be read
+                            memset(response + 6, 0x00, 4); // AC bits cannot be read
                             if (DBGLEVEL >= DBG_EXTENDED) Dbprintf("[MFEMUL_WORK - IsAccessAllowed] AC bits cannot be read - block %d (0x%02x)", blockNo, blockNo);
                         }
                     } else {
                         if (!IsAccessAllowed(blockNo, cardAUTHKEY, AC_DATA_READ)) {
-                            memset(response, 0x00, 16);		// datablock cannot be read
+                            memset(response, 0x00, 16); // datablock cannot be read
                             if (DBGLEVEL >= DBG_EXTENDED) Dbprintf("[MFEMUL_WORK - IsAccessAllowed] Data block %d (0x%02x) cannot be read", blockNo, blockNo);
                         }
                     }
@@ -899,7 +922,8 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
                     LED_C_OFF();
                     cardSTATE = MFEMUL_HALTED;
                     cardAUTHKEY = AUTHKEYNONE;
-                    if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("[MFEMUL_WORK] cardSTATE = MFEMUL_HALTED");
+                    if (DBGLEVEL >= DBG_EXTENDED)
+                        Dbprintf("[MFEMUL_WORK] cardSTATE = MFEMUL_HALTED");
                     break;
                 }
 
@@ -912,10 +936,12 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
                             EmSendCmdPar(response, rats_len, response_par);
                         } else
                             EmSendCmd(rats, rats_len);
-                        if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("[MFEMUL_WORK] RCV RATS => ACK");
+                        if (DBGLEVEL >= DBG_EXTENDED)
+                            Dbprintf("[MFEMUL_WORK] RCV RATS => ACK");
                     } else {
                         EmSend4bit(encrypted_data ? mf_crypto1_encrypt4bit(pcs, CARD_NACK_NA) : CARD_NACK_NA);
-                        if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("[MFEMUL_WORK] RCV RATS => NACK");
+                        if (DBGLEVEL >= DBG_EXTENDED)
+                            Dbprintf("[MFEMUL_WORK] RCV RATS => NACK");
                     }
                     break;
                 }
@@ -930,28 +956,33 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
                             EmSendCmdPar(response, receivedCmd_len, response_par);
                         } else
                             EmSendCmd(receivedCmd_dec, receivedCmd_len);
-                        if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("[MFEMUL_WORK] RCV NXP DESELECT => ACK");
+                        if (DBGLEVEL >= DBG_EXTENDED)
+                            Dbprintf("[MFEMUL_WORK] RCV NXP DESELECT => ACK");
                     } else {
                         EmSend4bit(encrypted_data ? mf_crypto1_encrypt4bit(pcs, CARD_NACK_NA) : CARD_NACK_NA);
-                        if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("[MFEMUL_WORK] RCV NXP DESELECT => NACK");
+                        if (DBGLEVEL >= DBG_EXTENDED)
+                            Dbprintf("[MFEMUL_WORK] RCV NXP DESELECT => NACK");
                     }
                     break;
                 }
 
                 // case MFEMUL_WORK => command not allowed
-                if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("Received command not allowed, nacking");
+                if (DBGLEVEL >= DBG_EXTENDED)
+                    Dbprintf("Received command not allowed, nacking");
                 EmSend4bit(encrypted_data ? mf_crypto1_encrypt4bit(pcs, CARD_NACK_NA) : CARD_NACK_NA);
                 break;
             }
 
             // AUTH1
             case MFEMUL_AUTH1: {
-                if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("[MFEMUL_AUTH1] Enter case");
+                if (DBGLEVEL >= DBG_EXTENDED)
+                    Dbprintf("[MFEMUL_AUTH1] Enter case");
 
                 if (receivedCmd_len != 8) {
                     cardSTATE_TO_IDLE();
                     LogTrace(uart->output, uart->len, uart->startTime * 16 - DELAY_AIR2ARM_AS_TAG, uart->endTime * 16 - DELAY_AIR2ARM_AS_TAG, uart->parity, true);
-                    if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("MFEMUL_AUTH1: receivedCmd_len != 8 (%d) => cardSTATE_TO_IDLE())", receivedCmd_len);
+                    if (DBGLEVEL >= DBG_EXTENDED)
+                        Dbprintf("MFEMUL_AUTH1: receivedCmd_len != 8 (%d) => cardSTATE_TO_IDLE())", receivedCmd_len);
                     break;
                 }
 
@@ -1035,16 +1066,17 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
                                  , prng_successor(nonce, 64)
                                 );
                     }
-                    cardAUTHKEY = AUTHKEYNONE;	// not authenticated
-                    EmSend4bit(mf_crypto1_encrypt4bit(pcs, CARD_NACK_NA));
+                    cardAUTHKEY = AUTHKEYNONE; // not authenticated
                     cardSTATE_TO_IDLE();
+                    // Really tags not respond NACK on invalid authentication
+                    LogTrace(uart->output, uart->len, uart->startTime * 16 - DELAY_AIR2ARM_AS_TAG, uart->endTime * 16 - DELAY_AIR2ARM_AS_TAG, uart->parity, true);
                     break;
                 }
 
                 ans = prng_successor(nonce, 96);
-                num_to_bytes(ans, 4, rAUTH_AT);
-                mf_crypto1_encrypt(pcs, rAUTH_AT, 4, response_par);
-                EmSendCmdPar(rAUTH_AT, 4, response_par);
+                num_to_bytes(ans, 4, response);
+                mf_crypto1_encrypt(pcs, response, 4, response_par);
+                EmSendCmdPar(response, 4, response_par);
 
                 if (DBGLEVEL >= DBG_EXTENDED) {
                     Dbprintf("[MFEMUL_AUTH1] AUTH COMPLETED for sector %d with key %c. time=%d",
@@ -1067,21 +1099,21 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
                         if (IsSectorTrailer(cardWRBL)) {
                             emlGetMem(response, cardWRBL, 1);
                             if (!IsAccessAllowed(cardWRBL, cardAUTHKEY, AC_KEYA_WRITE)) {
-                                memcpy(receivedCmd_dec, response, 6);	// don't change KeyA
+                                memcpy(receivedCmd_dec, response, 6); // don't change KeyA
                             }
                             if (!IsAccessAllowed(cardWRBL, cardAUTHKEY, AC_KEYB_WRITE)) {
-                                memcpy(receivedCmd_dec + 10, response + 10, 6);	// don't change KeyA
+                                memcpy(receivedCmd_dec + 10, response + 10, 6); // don't change KeyA
                             }
                             if (!IsAccessAllowed(cardWRBL, cardAUTHKEY, AC_AC_WRITE)) {
-                                memcpy(receivedCmd_dec + 6, response + 6, 4);	// don't change AC bits
+                                memcpy(receivedCmd_dec + 6, response + 6, 4); // don't change AC bits
                             }
                         } else {
                             if (!IsAccessAllowed(cardWRBL, cardAUTHKEY, AC_DATA_WRITE)) {
-                                memcpy(receivedCmd_dec, response, 16);	// don't change anything
+                                memcpy(receivedCmd_dec, response, 16); // don't change anything
                             }
                         }
                         emlSetMem(receivedCmd_dec, cardWRBL, 1);
-                        EmSend4bit(mf_crypto1_encrypt4bit(pcs, CARD_ACK));	// always ACK?
+                        EmSend4bit(mf_crypto1_encrypt4bit(pcs, CARD_ACK)); // always ACK?
                         cardSTATE = MFEMUL_WORK;
                         if (DBGLEVEL >= DBG_EXTENDED) Dbprintf("[MFEMUL_WRITEBL2] cardSTATE = MFEMUL_WORK");
                         break;
@@ -1152,7 +1184,7 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
 
     // NR AR ATTACK
     if (((flags & FLAG_NR_AR_ATTACK) == FLAG_NR_AR_ATTACK) && (DBGLEVEL >= DBG_INFO)) {
-        for (uint8_t	i = 0; i < ATTACK_KEY_COUNT; i++) {
+        for (uint8_t i = 0; i < ATTACK_KEY_COUNT; i++) {
             if (ar_nr_collected[i] == 2) {
                 Dbprintf("Collected two pairs of AR/NR which can be used to extract %s from reader for sector %d:", (i < ATTACK_KEY_COUNT / 2) ? "keyA" : "keyB", ar_nr_resp[i].sector);
                 Dbprintf("../tools/mfkey/mfkey32 %08x %08x %08x %08x %08x %08x",
@@ -1167,7 +1199,7 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
         }
     }
 
-    for (uint8_t	i = ATTACK_KEY_COUNT; i < ATTACK_KEY_COUNT * 2; i++) {
+    for (uint8_t i = ATTACK_KEY_COUNT; i < ATTACK_KEY_COUNT * 2; i++) {
         if (ar_nr_collected[i] == 2) {
             Dbprintf("Collected two pairs of AR/NR which can be used to extract %s from reader for sector %d:", (i < ATTACK_KEY_COUNT / 2) ? "keyA" : "keyB", ar_nr_resp[i].sector);
             Dbprintf("../tools/mfkey/mfkey32v2 %08x %08x %08x %08x %08x %08x %08x",
@@ -1182,7 +1214,7 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain) {
         }
     }
 
-    if (DBGLEVEL >= DBG_ERROR)	{
+    if (DBGLEVEL >= DBG_ERROR) {
         Dbprintf("Emulator stopped. Tracing: %d  trace length: %d ", get_tracing(), BigBuf_get_traceLen());
     }
 
diff --git a/armsrc/mifaresim.h b/armsrc/mifaresim.h
index 2dc9650cc..d05b730c9 100644
--- a/armsrc/mifaresim.h
+++ b/armsrc/mifaresim.h
@@ -16,7 +16,7 @@
 #include <stdint.h>
 
 #ifndef CheckCrc14A
-# define CheckCrc14A(data, len)	check_crc(CRC_14443_A, (data), (len))
+# define CheckCrc14A(data, len) check_crc(CRC_14443_A, (data), (len))
 #endif
 
 void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain);
@@ -24,7 +24,7 @@ void Mifare1ksim(uint16_t flags, uint8_t exitAfterNReads, uint8_t *datain);
 #define AC_DATA_READ             0
 #define AC_DATA_WRITE            1
 #define AC_DATA_INC              2
-#define AC_DATA_DEC_TRANS_REST	 3
+#define AC_DATA_DEC_TRANS_REST   3
 #define AC_KEYA_READ             0
 #define AC_KEYA_WRITE            1
 #define AC_KEYB_READ             2
diff --git a/armsrc/mifaresniff.c b/armsrc/mifaresniff.c
index 1d7f576af..4171ab572 100644
--- a/armsrc/mifaresniff.c
+++ b/armsrc/mifaresniff.c
@@ -11,7 +11,7 @@
 #include "mifaresniff.h"
 
 #ifndef CheckCrc14A
-# define CheckCrc14A(data, len)	check_crc(CRC_14443_A, (data), (len))
+# define CheckCrc14A(data, len) check_crc(CRC_14443_A, (data), (len))
 #endif
 
 //static int sniffState = SNF_INIT;
diff --git a/armsrc/mifareutil.c b/armsrc/mifareutil.c
index 6142198a5..b5f7567f9 100644
--- a/armsrc/mifareutil.c
+++ b/armsrc/mifareutil.c
@@ -602,10 +602,12 @@ void emlClearMem(void) {
 
 uint8_t SectorTrailer(uint8_t blockNo) {
     if (blockNo <= MIFARE_2K_MAXBLOCK) {
-        if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("Sector Trailer for block %d : %d", blockNo, (blockNo | 0x03));
+        if (DBGLEVEL >= DBG_EXTENDED)
+            Dbprintf("Sector Trailer for block %d : %d", blockNo, (blockNo | 0x03));
         return (blockNo | 0x03);
     } else {
-        if (DBGLEVEL >= DBG_EXTENDED)	Dbprintf("Sector Trailer for block %d : %d", blockNo, (blockNo | 0x0f));
+        if (DBGLEVEL >= DBG_EXTENDED)
+            Dbprintf("Sector Trailer for block %d : %d", blockNo, (blockNo | 0x0f));
         return (blockNo | 0x0f);
     }
 }
diff --git a/armsrc/spiffs.c b/armsrc/spiffs.c
new file mode 100644
index 000000000..7fd712e7f
--- /dev/null
+++ b/armsrc/spiffs.c
@@ -0,0 +1,607 @@
+//-----------------------------------------------------------------------------
+// Colin J. Brigato, 2019 - [colin@brigato.fr]
+//
+// This code is licensed to you under the terms of the GNU GPL, version 2 or,
+// at your option, any later version. See the LICENSE.txt file for the text of
+// the license.
+//-----------------------------------------------------------------------------
+// SPIFFS api for RDV40 Integration by Colin Brigato
+//-----------------------------------------------------------------------------
+
+#define SPIFFS_CFG_PHYS_SZ (1024 * 128)
+#define SPIFFS_CFG_PHYS_ERASE_SZ (4 * 1024)
+#define SPIFFS_CFG_PHYS_ADDR (0)
+#define SPIFFS_CFG_LOG_PAGE_SZ (256)
+#define SPIFFS_CFG_LOG_BLOCK_SZ (4 * 1024)
+#define LOG_PAGE_SIZE 256
+#define RDV40_SPIFFS_WORKBUF_SZ (LOG_PAGE_SIZE * 2)
+// Experimental : 4 full pages(LOG_PAGE_SIZE + file descript size) of cache for
+// Reading and writing if reading cache is stable, writing cache may need more
+// testing regarding power loss, page consistency checks, Garbage collector
+// Flushing handling... in doubt, use maximal safetylevel as, in most of the
+// case, will ensure a flush by rollbacking to previous Unmounted state
+#define RDV40_SPIFFS_CACHE_SZ ((LOG_PAGE_SIZE + 32) * 4)
+#define SPIFFS_FD_SIZE (32)
+#define RDV40_SPIFFS_MAX_FD (2)
+#define RDV40_SPIFFS_FDBUF_SZ (SPIFFS_FD_SIZE * RDV40_SPIFFS_MAX_FD)
+
+#define RDV40_SPIFFS_LAZY_HEADER                                                                                       \
+    int changed = 0;                                                                                                   \
+    if ((level == RDV40_SPIFFS_SAFETY_LAZY) || (level == RDV40_SPIFFS_SAFETY_SAFE)) {                                  \
+        changed = rdv40_spiffs_lazy_mount();                                                                           \
+    }
+
+#define RDV40_SPIFFS_SAFE_FOOTER                                                                                       \
+    if (level == RDV40_SPIFFS_SAFETY_SAFE) {                                                                           \
+        changed = rdv40_spiffs_lazy_mount_rollback(changed);                                                           \
+    }                                                                                                                  \
+    return changed;
+
+#define RDV40_SPIFFS_SAFE_FUNCTION(RDV40_SPIFFS_LLFUNCT)                                                               \
+    RDV40_SPIFFS_LAZY_HEADER                                                                                           \
+    RDV40_SPIFFS_LLFUNCT                                                                                               \
+    RDV40_SPIFFS_SAFE_FOOTER
+
+#include "spiffs.h"
+
+///// FLASH LEVEL R/W/E operations  for feeding SPIFFS Driver/////////////////
+static s32_t rdv40_spiffs_llread(u32_t addr, u32_t size, u8_t *dst) {
+
+    if (!Flash_ReadData(addr, dst, size)) {
+        return 128;
+    }
+    return SPIFFS_OK;
+}
+
+static s32_t rdv40_spiffs_llwrite(u32_t addr, u32_t size, u8_t *src) {
+
+    if (!FlashInit()) {
+        return 129;
+    }
+    Flash_Write(addr, src, size);
+    return SPIFFS_OK;
+}
+
+static s32_t rdv40_spiffs_llerase(u32_t addr, u32_t size) {
+
+    if (!FlashInit()) {
+        return 130;
+    }
+
+    uint32_t bytes_erased = 0, bytes_remaining = size;
+    while (bytes_remaining > 0) {
+
+        addr += bytes_erased;
+        Flash_CheckBusy(BUSY_TIMEOUT);
+        Flash_WriteEnable();
+        FlashSendByte(SECTORERASE);
+        Flash_TransferAdresse(addr);
+        FlashSendLastByte(0);
+
+        bytes_remaining -= 4096;
+        bytes_erased += 4096;
+    }
+
+    Flash_CheckBusy(BUSY_TIMEOUT);
+    FlashStop();
+
+    return SPIFFS_OK;
+}
+
+////////////////////////////////////////////////////////////////////////////////
+
+////// SPIFFS LOW LEVEL OPERATIONS /////////////////////////////////////////////
+static u8_t spiffs_work_buf[RDV40_SPIFFS_WORKBUF_SZ];
+static u8_t spiffs_fds[RDV40_SPIFFS_FDBUF_SZ];
+static u8_t spiffs_cache_buf[RDV40_SPIFFS_CACHE_SZ];
+
+static spiffs fs;
+
+static enum spiffs_mount_status {
+    RDV40_SPIFFS_UNMOUNTED,
+    RDV40_SPIFFS_MOUNTED,
+    RDV40_SPIFFS_UNKNOWN
+} RDV40_SPIFFS_MOUNT_STATUS;
+
+int rdv40_spiffs_mounted() {
+    int ret = 0;
+
+    switch (RDV40_SPIFFS_MOUNT_STATUS) {
+        case RDV40_SPIFFS_MOUNTED:
+            ret = 1;
+            break;
+        case RDV40_SPIFFS_UNMOUNTED:
+        case RDV40_SPIFFS_UNKNOWN:
+        default:
+            ret = 0;
+    }
+
+    return ret;
+}
+
+int rdv40_spiffs_mount() {
+    if (rdv40_spiffs_mounted()) {
+        Dbprintf("ERR: SPIFFS already mounted !");
+        return SPIFFS_ERR_MOUNTED;
+    }
+
+    spiffs_config cfg;
+    cfg.hal_read_f = rdv40_spiffs_llread;
+    cfg.hal_write_f = rdv40_spiffs_llwrite;
+    cfg.hal_erase_f = rdv40_spiffs_llerase;
+
+    // uncached version
+    // int ret = SPIFFS_mount(&fs, &cfg, spiffs_work_buf, spiffs_fds,
+    // sizeof(spiffs_fds), 0, 0, 0); cached version, experimental
+    int ret = SPIFFS_mount(&fs, &cfg, spiffs_work_buf, spiffs_fds, sizeof(spiffs_fds), spiffs_cache_buf,
+                           sizeof(spiffs_cache_buf), 0);
+    if (ret == SPIFFS_OK) {
+        RDV40_SPIFFS_MOUNT_STATUS = RDV40_SPIFFS_MOUNTED;
+    }
+    return ret;
+}
+
+int rdv40_spiffs_unmount() {
+    if (!rdv40_spiffs_mounted()) {
+        Dbprintf("ERR: SPIFFS not mounted !");
+        return SPIFFS_ERR_NOT_MOUNTED;
+    }
+
+    SPIFFS_clearerr(&fs);
+    SPIFFS_unmount(&fs);
+
+    int ret = SPIFFS_errno(&fs);
+    if (ret == SPIFFS_OK) {
+        RDV40_SPIFFS_MOUNT_STATUS = RDV40_SPIFFS_UNMOUNTED;
+    }
+    return ret;
+}
+////////////////////////////////////////////////////////////////////////////////
+
+///// Base RDV40_SPIFFS_SAFETY_NORMAL operations////////////////////////////////
+
+void write_to_spiffs(const char *filename, uint8_t *src, uint32_t size) {
+    spiffs_file fd = SPIFFS_open(&fs, filename, SPIFFS_CREAT | SPIFFS_TRUNC | SPIFFS_RDWR, 0);
+    if (SPIFFS_write(&fs, fd, src, size) < 0)
+        Dbprintf("errno %i\n", SPIFFS_errno(&fs));
+    SPIFFS_close(&fs, fd);
+}
+
+void append_to_spiffs(const char *filename, uint8_t *src, uint32_t size) {
+    spiffs_file fd = SPIFFS_open(&fs, filename, SPIFFS_APPEND | SPIFFS_RDWR, 0);
+    if (SPIFFS_write(&fs, fd, src, size) < 0)
+        Dbprintf("errno %i\n", SPIFFS_errno(&fs));
+    SPIFFS_close(&fs, fd);
+}
+
+void read_from_spiffs(const char *filename, uint8_t *dst, uint32_t size) {
+    spiffs_file fd = SPIFFS_open(&fs, filename, SPIFFS_RDWR, 0);
+    if (SPIFFS_read(&fs, fd, dst, size) < 0)
+        Dbprintf("errno %i\n", SPIFFS_errno(&fs));
+    SPIFFS_close(&fs, fd);
+}
+
+void rename_in_spiffs(const char *old_filename, const char *new_filename) {
+    if (SPIFFS_rename(&fs, old_filename, new_filename) < 0)
+        Dbprintf("errno %i\n", SPIFFS_errno(&fs));
+}
+
+void remove_from_spiffs(const char *filename) {
+    if (SPIFFS_remove(&fs, filename) < 0)
+        Dbprintf("errno %i\n", SPIFFS_errno(&fs));
+}
+
+spiffs_stat stat_in_spiffs(const char *filename) {
+    spiffs_stat s;
+    if (SPIFFS_stat(&fs, filename, &s) < 0)
+        Dbprintf("errno %i\n", SPIFFS_errno(&fs));
+    return s;
+}
+
+uint32_t size_in_spiffs(const char *filename) {
+    spiffs_stat s = stat_in_spiffs(filename);
+    return s.size;
+}
+
+rdv40_spiffs_fsinfo info_of_spiffs() {
+    rdv40_spiffs_fsinfo fsinfo;
+    fsinfo.blockSize = SPIFFS_CFG_LOG_BLOCK_SZ;
+    fsinfo.pageSize = LOG_PAGE_SIZE;
+    fsinfo.maxOpenFiles = RDV40_SPIFFS_MAX_FD;
+    fsinfo.maxPathLength = SPIFFS_OBJ_NAME_LEN;
+    if (SPIFFS_info(&fs, &fsinfo.totalBytes, &fsinfo.usedBytes) < 0)
+        Dbprintf("errno %i\n", SPIFFS_errno(&fs));
+    fsinfo.freeBytes = fsinfo.totalBytes - fsinfo.usedBytes;
+    // Rounding without float may be improved
+    fsinfo.usedPercent = ((100 * fsinfo.usedBytes) + (fsinfo.totalBytes / 2)) / fsinfo.totalBytes;
+    fsinfo.freePercent = (100 - fsinfo.usedPercent);
+    return fsinfo;
+}
+
+int exists_in_spiffs(const char *filename) {
+    spiffs_stat stat;
+    int rc = SPIFFS_stat(&fs, filename, &stat);
+    return rc == SPIFFS_OK;
+}
+
+RDV40SpiFFSFileType filetype_in_spiffs(const char *filename) {
+    RDV40SpiFFSFileType filetype = RDV40_SPIFFS_FILETYPE_UNKNOWN;
+    char symlinked[SPIFFS_OBJ_NAME_LEN];
+    sprintf(symlinked, "%s.lnk", filename);
+    if (exists_in_spiffs(filename)) {
+        filetype = RDV40_SPIFFS_FILETYPE_REAL;
+    }
+    if (exists_in_spiffs(symlinked)) {
+        if (filetype != RDV40_SPIFFS_FILETYPE_UNKNOWN) {
+            filetype = RDV40_SPIFFS_FILETYPE_BOTH;
+        } else {
+            filetype = RDV40_SPIFFS_FILETYPE_SYMLINK;
+        }
+    }
+    if (DBGLEVEL > 1) {
+        switch (filetype) {
+            case RDV40_SPIFFS_FILETYPE_REAL:
+                Dbprintf("Filetype is : RDV40_SPIFFS_FILETYPE_REAL");
+                break;
+            case RDV40_SPIFFS_FILETYPE_SYMLINK:
+                Dbprintf("Filetype is : RDV40_SPIFFS_FILETYPE_SYMLINK");
+                break;
+            case RDV40_SPIFFS_FILETYPE_BOTH:
+                Dbprintf("Filetype is : RDV40_SPIFFS_FILETYPE_BOTH");
+                break;
+            case RDV40_SPIFFS_FILETYPE_UNKNOWN:
+                Dbprintf("Filetype is : RDV40_SPIFFS_FILETYPE_UNKNOWN");
+                break;
+        }
+    }
+    return filetype;
+}
+
+int is_valid_filename(const char *filename) {
+    if (filename == NULL) {
+        return false;
+    }
+    uint32_t len = strlen(filename);
+    return len > 0 && len < SPIFFS_OBJ_NAME_LEN;
+}
+
+void copy_in_spiffs(const char *src, const char *dst) {
+    uint32_t size = size_in_spiffs((char *)src);
+    uint8_t *mem = BigBuf_malloc(size);
+    read_from_spiffs((char *)src, (uint8_t *)mem, size);
+    write_to_spiffs((char *)dst, (uint8_t *)mem, size);
+}
+
+////////////////////////////////////////////////////////////////////////////////
+
+////// Abstract Operations for base Safetyness /////////////////////////////////
+//
+// mount if not already
+// As an "hint" to the behavior one should adopt after his or her lazyness
+// it will return 0 if the call was a noop, either because it did not need to
+// change OR because it wasn't ABLE to change :)
+//                1 if the mount status actually changed
+// so you know what to do IN CASE you wished to set things "back to previous
+// state"
+int rdv40_spiffs_lazy_mount() {
+    int changed = 0;
+    if (!rdv40_spiffs_mounted()) {
+        changed = rdv40_spiffs_mount();
+        /* if changed = 0 = SPIFFS_OK then all went well then the change
+         * actually occured :)*/
+        changed = !changed;
+    }
+    return changed;
+}
+
+// unmount if not already
+int rdv40_spiffs_lazy_unmount() {
+    int changed = 0;
+    if (rdv40_spiffs_mounted()) {
+        changed = rdv40_spiffs_unmount();
+        changed = !changed;
+    }
+    return changed;
+}
+
+// Before further Reading, it is required to have in mind that UNMOUTING is
+// important in some ways Because it is the ONLY operation which ensure that
+// -> all Caches and writings are flushed to the FS
+// -> all FD are properly closed
+// -> Every best effort has been done to ensure consistency and integrity of the
+// state reputated to be the actual state of the Filesystem.
+//---
+
+// This will "toggle" mount status
+// on "changement" conditional
+// so it is for the former lazy_ mounting function to actually rollback or not
+// depending on the result of the previous This is super lazy implementation as
+// it is either a toggle to previous state or again a noop as everything was a
+// nonevent If you have a function which NEEDS mounting but you want to exit
+// this function in the very state mounting was before your intervention all
+// things can now be transparent like
+/*
+void my_lazy_spiffs_act(){
+  uint8_t changed = rdv40_spiffs_lazy_mount();
+     [..] Do what you have to do with spiffs
+  rdv40_spiffs_lazy_rollback(changed)
+}
+*/
+// The exact same goes for needed unmouting with eventual rollback, you just
+// have to use lazy_unmount insted of lazy mount This way, you can ensure
+// consistency in operation even with complex chain of mounting and unmounting
+// Lets's say you are in a function which needs to mount if not already, and in
+// the middle of itself calls function which indeed will need to unmount if not
+// already. Well you better use safe or wrapped function which are made to
+// rollback to previous state, so you can continue right after to do your
+// things.
+//
+//  As an extreme example: let's imagine that we have a function which is made
+//  to FORMAT the whole SPIFFS if a SPECIFIC content is written in the 4 first
+//  byte of that file. Also in such a case it should quickly create a bunch of
+//  skkeleton to get itself back to a known and wanted state. This behavior has
+//  to be done at every "manual" (not a lazy or internal event) mounting, just
+//  like an action upon boot.
+/*
+   void my_spiffs_boot(){
+     uint8_t resetret[4];
+     // this lazy_mount since needed and can also report back the change on
+state implied by eventual mount, if needed rdv40_spiffs_lazy_read((const char
+*)".SHOULDRESET",(uint8_t *)resetret,4); if( resetret == "YESS" ) { uint8_t
+changed = rdv40_spiffs_lazy_format(); // this will imply change only if we where
+already mounted beforehand, was the case after our reading without further
+rollback rdv40_spiffs_lazy_mount_rollback(changed); // so if we were mounted
+just get back to this state. If not, just don't.
+        [...]
+     }
+     [...]
+}
+*/
+// Again : This will "toggle" spiffs mount status only if a "change" occured
+// (and should be fed by the result of a spiffs_lazy* function) If everything
+// went well, it will return SPIFFS_OK if everything went well, and a report
+// back the chain a SPI_ERRNO if not.
+int rdv40_spiffs_lazy_mount_rollback(int changed) {
+    if (!changed)
+        return SPIFFS_OK;
+    if (rdv40_spiffs_mounted())
+        return rdv40_spiffs_unmount();
+    return rdv40_spiffs_mount();
+}
+///////////////////////////////////////////////////////////////////////////////
+
+// High level functions with SatefetyLevel
+// Beware that different safety level makes different return behavior
+//
+// RDV40_SPIFFS_SAFETY_NORMAL : will operate withtout further change on mount
+// status RDV40_SPIFFS_SAFETY_LAZY : will ensure mount status already being in
+// correct state before ops,
+//                            will return !false if mount state had to change
+// RDV40_SPIFFS_SAFETY_SAFE : will do same as RDV40_SPIFFS_SAFETY_LAZY
+//                            will also safely rollback to previous state IF
+//                            mount state had to change will return SPIFFS_OK /
+//                            0 / false if everything went well
+
+// TODO : this functions are common enought to be unified with a switchcase
+// statement or some function taking function parameters
+// TODO : forbid writing to a filename which already exists as lnk !
+// TODO : forbid writing to a filename.lnk which already exists without lnk !
+int rdv40_spiffs_write(char *filename, uint8_t *src, uint32_t size, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(                                  //
+        write_to_spiffs((char *)filename, (uint8_t *)src, size); //
+    )
+}
+
+int rdv40_spiffs_append(char *filename, uint8_t *src, uint32_t size, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(                                   //
+        append_to_spiffs((char *)filename, (uint8_t *)src, size); //
+    )
+}
+
+// todo integrate reading symlinks transparently
+int rdv40_spiffs_read(char *filename, uint8_t *dst, uint32_t size, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(                                   //
+        read_from_spiffs((char *)filename, (uint8_t *)dst, size); //
+    )
+}
+
+// TODO : forbid writing to a filename which already exists as lnk !
+// TODO : forbid writing to a filename.lnk which already exists without lnk !
+int rdv40_spiffs_rename(char *old_filename, char *new_filename, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(                                       //
+        rename_in_spiffs((char *)old_filename, (char *)new_filename); //
+    )
+}
+int rdv40_spiffs_remove(char *filename, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(               //
+        remove_from_spiffs((char *)filename); //
+    )
+}
+
+int rdv40_spiffs_copy(char *src, char *dst, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(                   //
+        copy_in_spiffs((char *)src, (char *)dst); //
+    )
+}
+
+int rdv40_spiffs_stat(char *filename, uint32_t *buf, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(                      //
+        *buf = size_in_spiffs((char *)filename); //
+    )
+}
+
+int rdv40_spiffs_getfsinfo(rdv40_spiffs_fsinfo *fsinfo, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(         //
+        *fsinfo = info_of_spiffs(); //
+    )
+}
+
+// test for symlink from filename
+int rdv40_spiffs_is_symlink(const char *s) {
+    int ret = 0;
+
+    if (s != NULL) {
+        size_t size = strlen(s);
+
+        if (size >= 4 && s[size - 4] == '.' && s[size - 3] == 'l' && s[size - 2] == 'n' && s[size - 1] == 'k') {
+            ret = 1;
+        }
+    }
+
+    return ret;
+}
+
+// since FILENAME can't be longer than 32Bytes as of hard configuration, we're
+// safe with Such maximum. So the "size" variable is actually the known/intended
+// size of DESTINATION file, may it be known (may we provide a "stat from
+// symlinkk ?")
+// ATTENTION : you must NOT provide the whole filename (so please do not include the .lnk extension)
+// TODO : integrate in read_function
+int rdv40_spiffs_read_as_symlink(char *filename, uint8_t *dst, uint32_t size, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(                 //
+        char linkdest[SPIFFS_OBJ_NAME_LEN];     //
+        char linkfilename[SPIFFS_OBJ_NAME_LEN]; //
+        sprintf(linkfilename, "%s.lnk", filename);
+        if (DBGLEVEL > 1) Dbprintf("Linkk real filename is  destination is : %s", linkfilename);
+        read_from_spiffs((char *)linkfilename, (uint8_t *)linkdest, SPIFFS_OBJ_NAME_LEN);
+        if (DBGLEVEL > 1) Dbprintf("Symlink destination is : %s", linkdest);
+            read_from_spiffs((char *)linkdest, (uint8_t *)dst, size); //
+        )
+        }
+
+// BEWARE ! This function is DESTRUCTIVE as it will UPDATE an existing symlink
+// Since it creates a .lnk extension file it may be minor to mistake the order of arguments
+// Still please use this function with care.
+// Also, remind that it will NOT check if destination filename actually exists
+// As a mnenotechnic, think about the "ln" unix command, which order is the same as "cp" unix command
+// in regard of arguments orders.
+// Eg :
+// rdv40_spiffs_make_symlink((uint8_t *)"hello", (uint8_t *)"world", RDV40_SPIFFS_SAFETY_SAFE)
+//   will generate a file named "world.lnk" with the path to file "hello" written in
+//   wich you can then read back with :
+//   rdv40_spiffs_read_as_symlink((uint8_t *)"world",(uint8_t *) buffer, orig_file_size, RDV40_SPIFFS_SAFETY_SAFE);
+// TODO : FORBID creating a symlink with a basename (before.lnk) which already exists as a file !
+int rdv40_spiffs_make_symlink(char *linkdest, char *filename, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(                 //
+        char linkfilename[SPIFFS_OBJ_NAME_LEN]; //
+        sprintf(linkfilename, "%s.lnk", filename);
+        write_to_spiffs((char *)linkfilename, (uint8_t *)linkdest, SPIFFS_OBJ_NAME_LEN); //
+    )
+}
+
+// filename and filename.lnk will both the existence-checked
+// if filename exists, read will be used, if filename.lnk exists, read_as_symlink will be used
+// Both existence is not handled right now and should not happen or create a default fallback behavior
+// Still, this case won't happend when the write(s) functions will check for both symlink and real file
+// preexistance, avoiding a link being created if filename exists, or avoiding a file being created if
+// symlink exists with same name
+int rdv40_spiffs_read_as_filetype(char *filename, uint8_t *dst, uint32_t size, RDV40SpiFFSSafetyLevel level) {
+    RDV40_SPIFFS_SAFE_FUNCTION(                                              //
+        RDV40SpiFFSFileType filetype = filetype_in_spiffs((char *)filename); //
+    switch (filetype) {
+    case RDV40_SPIFFS_FILETYPE_REAL:
+        rdv40_spiffs_read((char *)filename, (uint8_t *)dst, size, level);
+            break;
+        case RDV40_SPIFFS_FILETYPE_SYMLINK:
+            rdv40_spiffs_read_as_symlink((char *)filename, (uint8_t *)dst, size, level);
+            break;
+        case RDV40_SPIFFS_FILETYPE_BOTH:
+        case RDV40_SPIFFS_FILETYPE_UNKNOWN:
+        default:
+            ;
+    } //
+    )
+}
+
+// TODO regarding reads/write and symlinks :
+// Provide a higher level readFile function which
+//   - don't need a size to be provided, getting it from STAT call and using bigbuff malloc
+//   - send back the whole readed file as return Result
+// Maybe a good think to implement a VFS api here.
+
+////////////////////////////////////////////////////////////////////////////////
+
+///////// MISC HIGH LEVEL FUNCTIONS ////////////////////////////////////////////
+
+void rdv40_spiffs_safe_print_fsinfo() {
+    rdv40_spiffs_fsinfo fsinfo;
+    rdv40_spiffs_getfsinfo(&fsinfo, RDV40_SPIFFS_SAFETY_SAFE);
+    DbpString(_BLUE_("Flash Memory FileSystem Info (SPIFFS)"));
+    Dbprintf("-------------------------------------");
+    Dbprintf("* Filesystem Logical Block Size.........%d bytes", fsinfo.blockSize);
+    Dbprintf("* Filesystem Logical Page Size..........%d bytes", fsinfo.pageSize);
+    Dbprintf("--");
+    Dbprintf("* Filesystem Max Open Files.............%d file descriptors", fsinfo.maxOpenFiles);
+    Dbprintf("* Filesystem Max Path Length............%d chars", fsinfo.maxPathLength);
+    Dbprintf("--");
+    Dbprintf("Filesystem\tSize\tUsed\tAvailable\tUse%\tMounted on");
+    Dbprintf("spiffs\t%dB\t%dB\t%dB\t\t%d%\t/", fsinfo.totalBytes, fsinfo.usedBytes, fsinfo.freeBytes,
+             fsinfo.usedPercent);
+}
+
+// this function is safe and WILL rollback since it is only a PRINTING function,
+// not a function intended to give any sort of struct to manipulate the FS
+// objects
+// TODO : Fake the Directory availability by spliting strings , buffering,
+// maintaining prefix list sorting, unique_checking, THEN outputing precomputed
+// tree Other solutio nwould be to add directory support to SPIFFS, but that we
+// dont want, as prefix are way easier and lighter in every aspect.
+void rdv40_spiffs_safe_print_tree(uint8_t banner) {
+
+    int changed = rdv40_spiffs_lazy_mount();
+    spiffs_DIR d;
+    struct spiffs_dirent e;
+    struct spiffs_dirent *pe = &e;
+    if (banner) {
+        DbpString(_BLUE_("Flash Memory FileSystem tree (SPIFFS)"));
+        Dbprintf("-------------------------------------");
+    }
+    SPIFFS_opendir(&fs, "/", &d);
+    Dbprintf("    \t         \t/");
+    while ((pe = SPIFFS_readdir(&d, pe))) {
+
+        char resolvedlink[11 + SPIFFS_OBJ_NAME_LEN];
+        if (rdv40_spiffs_is_symlink((const char *)pe->name)) {
+            char linkdest[SPIFFS_OBJ_NAME_LEN];
+            read_from_spiffs((char *)pe->name, (uint8_t *)linkdest, SPIFFS_OBJ_NAME_LEN);
+            sprintf(resolvedlink, "(.lnk) --> %s", linkdest);
+            // Kind of stripping the .lnk extension
+            strtok((char *)pe->name, ".");
+        } else {
+            memset(resolvedlink, 0, sizeof(resolvedlink));
+        }
+
+        Dbprintf("[%04x]\t %ibytes \t|-- %s%s", pe->obj_id, pe->size, pe->name, resolvedlink);
+    }
+    SPIFFS_closedir(&d);
+
+    rdv40_spiffs_lazy_mount_rollback(changed);
+}
+
+void test_spiffs() {
+    Dbprintf("---------------------------");
+    Dbprintf("Testing SPIFFS operations");
+    Dbprintf("---------------------------");
+    Dbprintf("(all test are made using lazy safetylevel)");
+    Dbprintf("* Mounting filesystem (lazy).......");
+    int changed = rdv40_spiffs_lazy_mount();
+    Dbprintf("* Printing tree..............");
+    rdv40_spiffs_safe_print_tree(false);
+    Dbprintf("* Writing 'I love Proxmark' in a testspiffs.txt");
+    // Since We lazy_mounted manually before hand, the wrte safety level will
+    // just imply noops
+    rdv40_spiffs_write((char *)"testspiffs.txt", (uint8_t *)"I love Proxmark", 15, RDV40_SPIFFS_SAFETY_SAFE);
+    Dbprintf("* Printing tree again.......");
+    rdv40_spiffs_safe_print_tree(false);
+    Dbprintf("* Making a symlink to testspiffs.txt");
+    rdv40_spiffs_make_symlink((char *)"testspiffs.txt", (char *)"linktotestspiffs.txt", RDV40_SPIFFS_SAFETY_SAFE);
+    Dbprintf("* Printing tree again.......");
+    rdv40_spiffs_safe_print_tree(false);
+    // TODO READBACK, rename,print tree read back, remove, print tree;
+    Dbprintf("* Rollbacking The mount status IF things have changed");
+    rdv40_spiffs_lazy_mount_rollback(changed);
+    Dbprintf("All done");
+    return;
+}
+
+///////////////////////////////////////////////////////////////////////////////
diff --git a/armsrc/spiffs.h b/armsrc/spiffs.h
new file mode 100644
index 000000000..f36bde4b0
--- /dev/null
+++ b/armsrc/spiffs.h
@@ -0,0 +1,861 @@
+/*
+ * spiffs.h
+ *
+ *  Created on: May 26, 2013
+ *      Author: petera
+ */
+
+#ifndef SPIFFS_H_
+#define SPIFFS_H_
+#if defined(__cplusplus)
+extern "C" {
+#endif
+
+#include "spiffs_config.h"
+
+typedef enum spiffs_safety_level { RDV40_SPIFFS_SAFETY_NORMAL, RDV40_SPIFFS_SAFETY_LAZY, RDV40_SPIFFS_SAFETY_SAFE } RDV40SpiFFSSafetyLevel;
+
+typedef enum spiffs_file_type {
+    RDV40_SPIFFS_FILETYPE_REAL,
+    RDV40_SPIFFS_FILETYPE_SYMLINK,
+    RDV40_SPIFFS_FILETYPE_BOTH,
+    RDV40_SPIFFS_FILETYPE_UNKNOWN
+} RDV40SpiFFSFileType;
+
+typedef struct rdv40_spiffs_fsinfo {
+    uint32_t blockSize;
+    uint32_t pageSize;
+    uint32_t maxOpenFiles;
+    uint32_t maxPathLength;
+    uint32_t totalBytes, usedBytes, freeBytes;
+    uint32_t usedPercent, freePercent;
+} rdv40_spiffs_fsinfo;
+
+int rdv40_spiffs_read_as_filetype(char *filename, uint8_t *dst, uint32_t size, RDV40SpiFFSSafetyLevel level);
+
+
+int rdv40_spiffs_lazy_unmount();
+int rdv40_spiffs_lazy_mount();
+int rdv40_spiffs_lazy_mount_rollback(int changed);
+int rdv40_spiffs_write(char *filename, uint8_t *src, uint32_t size, RDV40SpiFFSSafetyLevel level);
+int rdv40_spiffs_read(char *filename, uint8_t *dst, uint32_t size, RDV40SpiFFSSafetyLevel level);
+int rdv40_spiffs_rename(char *old_filename, char *new_filename, RDV40SpiFFSSafetyLevel level);
+int rdv40_spiffs_remove(char *filename, RDV40SpiFFSSafetyLevel level);
+int rdv40_spiffs_read_as_symlink(char *filename, uint8_t *dst, uint32_t size, RDV40SpiFFSSafetyLevel level);
+void write_to_spiffs(const char *filename, uint8_t *src, uint32_t size);
+void read_from_spiffs(const char *filename, uint8_t *dst, uint32_t size);
+void test_spiffs();
+void rdv40_spiffs_safe_print_tree(uint8_t banner);
+int rdv40_spiffs_unmount();
+int rdv40_spiffs_mount();
+int rdv40_spiffs_is_symlink(const char *s);
+void rdv40_spiffs_safe_print_fsinfo();
+int rdv40_spiffs_make_symlink(char *linkdest, char *filename, RDV40SpiFFSSafetyLevel level);
+void append_to_spiffs(const char *filename, uint8_t *src, uint32_t size);
+int rdv40_spiffs_copy(char *src, char *dst, RDV40SpiFFSSafetyLevel level);
+int rdv40_spiffs_append(char *filename, uint8_t *src, uint32_t size, RDV40SpiFFSSafetyLevel level);
+int rdv40_spiffs_stat(char *filename, uint32_t *buf, RDV40SpiFFSSafetyLevel level);
+uint32_t size_in_spiffs(const char *filename);
+int exists_in_spiffs(const char *filename);
+
+#define SPIFFS_OK                       0
+#define SPIFFS_ERR_NOT_MOUNTED          -10000
+#define SPIFFS_ERR_FULL                 -10001
+#define SPIFFS_ERR_NOT_FOUND            -10002
+#define SPIFFS_ERR_END_OF_OBJECT        -10003
+#define SPIFFS_ERR_DELETED              -10004
+#define SPIFFS_ERR_NOT_FINALIZED        -10005
+#define SPIFFS_ERR_NOT_INDEX            -10006
+#define SPIFFS_ERR_OUT_OF_FILE_DESCS    -10007
+#define SPIFFS_ERR_FILE_CLOSED          -10008
+#define SPIFFS_ERR_FILE_DELETED         -10009
+#define SPIFFS_ERR_BAD_DESCRIPTOR       -10010
+#define SPIFFS_ERR_IS_INDEX             -10011
+#define SPIFFS_ERR_IS_FREE              -10012
+#define SPIFFS_ERR_INDEX_SPAN_MISMATCH  -10013
+#define SPIFFS_ERR_DATA_SPAN_MISMATCH   -10014
+#define SPIFFS_ERR_INDEX_REF_FREE       -10015
+#define SPIFFS_ERR_INDEX_REF_LU         -10016
+#define SPIFFS_ERR_INDEX_REF_INVALID    -10017
+#define SPIFFS_ERR_INDEX_FREE           -10018
+#define SPIFFS_ERR_INDEX_LU             -10019
+#define SPIFFS_ERR_INDEX_INVALID        -10020
+#define SPIFFS_ERR_NOT_WRITABLE         -10021
+#define SPIFFS_ERR_NOT_READABLE         -10022
+#define SPIFFS_ERR_CONFLICTING_NAME     -10023
+#define SPIFFS_ERR_NOT_CONFIGURED       -10024
+
+#define SPIFFS_ERR_NOT_A_FS             -10025
+#define SPIFFS_ERR_MOUNTED              -10026
+#define SPIFFS_ERR_ERASE_FAIL           -10027
+#define SPIFFS_ERR_MAGIC_NOT_POSSIBLE   -10028
+
+#define SPIFFS_ERR_NO_DELETED_BLOCKS    -10029
+
+#define SPIFFS_ERR_FILE_EXISTS          -10030
+
+#define SPIFFS_ERR_NOT_A_FILE           -10031
+#define SPIFFS_ERR_RO_NOT_IMPL          -10032
+#define SPIFFS_ERR_RO_ABORTED_OPERATION -10033
+#define SPIFFS_ERR_PROBE_TOO_FEW_BLOCKS -10034
+#define SPIFFS_ERR_PROBE_NOT_A_FS       -10035
+#define SPIFFS_ERR_NAME_TOO_LONG        -10036
+
+#define SPIFFS_ERR_IX_MAP_UNMAPPED      -10037
+#define SPIFFS_ERR_IX_MAP_MAPPED        -10038
+#define SPIFFS_ERR_IX_MAP_BAD_RANGE     -10039
+
+#define SPIFFS_ERR_SEEK_BOUNDS          -10040
+
+
+#define SPIFFS_ERR_INTERNAL             -10050
+
+#define SPIFFS_ERR_TEST                 -10100
+
+
+// spiffs file descriptor index type. must be signed
+typedef s16_t spiffs_file;
+// spiffs file descriptor flags
+typedef u16_t spiffs_flags;
+// spiffs file mode
+typedef u16_t spiffs_mode;
+// object type
+typedef u8_t spiffs_obj_type;
+
+struct spiffs_t;
+
+#if SPIFFS_HAL_CALLBACK_EXTRA
+
+/* spi read call function type */
+typedef s32_t (*spiffs_read)(struct spiffs_t *fs, u32_t addr, u32_t size, u8_t *dst);
+/* spi write call function type */
+typedef s32_t (*spiffs_write)(struct spiffs_t *fs, u32_t addr, u32_t size, u8_t *src);
+/* spi erase call function type */
+typedef s32_t (*spiffs_erase)(struct spiffs_t *fs, u32_t addr, u32_t size);
+
+#else // SPIFFS_HAL_CALLBACK_EXTRA
+
+/* spi read call function type */
+typedef s32_t (*spiffs_read)(u32_t addr, u32_t size, u8_t *dst);
+/* spi write call function type */
+typedef s32_t (*spiffs_write)(u32_t addr, u32_t size, u8_t *src);
+/* spi erase call function type */
+typedef s32_t (*spiffs_erase)(u32_t addr, u32_t size);
+#endif // SPIFFS_HAL_CALLBACK_EXTRA
+
+/* file system check callback report operation */
+typedef enum {
+    SPIFFS_CHECK_LOOKUP = 0,
+    SPIFFS_CHECK_INDEX,
+    SPIFFS_CHECK_PAGE
+} spiffs_check_type;
+
+/* file system check callback report type */
+typedef enum {
+    SPIFFS_CHECK_PROGRESS = 0,
+    SPIFFS_CHECK_ERROR,
+    SPIFFS_CHECK_FIX_INDEX,
+    SPIFFS_CHECK_FIX_LOOKUP,
+    SPIFFS_CHECK_DELETE_ORPHANED_INDEX,
+    SPIFFS_CHECK_DELETE_PAGE,
+    SPIFFS_CHECK_DELETE_BAD_FILE
+} spiffs_check_report;
+
+/* file system check callback function */
+#if SPIFFS_HAL_CALLBACK_EXTRA
+typedef void (*spiffs_check_callback)(struct spiffs_t *fs, spiffs_check_type type, spiffs_check_report report,
+                                      u32_t arg1, u32_t arg2);
+#else // SPIFFS_HAL_CALLBACK_EXTRA
+typedef void (*spiffs_check_callback)(spiffs_check_type type, spiffs_check_report report,
+                                      u32_t arg1, u32_t arg2);
+#endif // SPIFFS_HAL_CALLBACK_EXTRA
+
+/* file system listener callback operation */
+typedef enum {
+    /* the file has been created */
+    SPIFFS_CB_CREATED = 0,
+    /* the file has been updated or moved to another page */
+    SPIFFS_CB_UPDATED,
+    /* the file has been deleted */
+    SPIFFS_CB_DELETED
+} spiffs_fileop_type;
+
+/* file system listener callback function */
+typedef void (*spiffs_file_callback)(struct spiffs_t *fs, spiffs_fileop_type op, spiffs_obj_id obj_id, spiffs_page_ix pix);
+
+#ifndef SPIFFS_DBG
+#define SPIFFS_DBG(...) \
+    printf(__VA_ARGS__)
+#endif
+#ifndef SPIFFS_GC_DBG
+#define SPIFFS_GC_DBG(...) printf(__VA_ARGS__)
+#endif
+#ifndef SPIFFS_CACHE_DBG
+#define SPIFFS_CACHE_DBG(...) printf(__VA_ARGS__)
+#endif
+#ifndef SPIFFS_CHECK_DBG
+#define SPIFFS_CHECK_DBG(...) printf(__VA_ARGS__)
+#endif
+
+/* Any write to the filehandle is appended to end of the file */
+#define SPIFFS_APPEND                   (1<<0)
+#define SPIFFS_O_APPEND                 SPIFFS_APPEND
+/* If the opened file exists, it will be truncated to zero length before opened */
+#define SPIFFS_TRUNC                    (1<<1)
+#define SPIFFS_O_TRUNC                  SPIFFS_TRUNC
+/* If the opened file does not exist, it will be created before opened */
+#define SPIFFS_CREAT                    (1<<2)
+#define SPIFFS_O_CREAT                  SPIFFS_CREAT
+/* The opened file may only be read */
+#define SPIFFS_RDONLY                   (1<<3)
+#define SPIFFS_O_RDONLY                 SPIFFS_RDONLY
+/* The opened file may only be written */
+#define SPIFFS_WRONLY                   (1<<4)
+#define SPIFFS_O_WRONLY                 SPIFFS_WRONLY
+/* The opened file may be both read and written */
+#define SPIFFS_RDWR                     (SPIFFS_RDONLY | SPIFFS_WRONLY)
+#define SPIFFS_O_RDWR                   SPIFFS_RDWR
+/* Any writes to the filehandle will never be cached but flushed directly */
+#define SPIFFS_DIRECT                   (1<<5)
+#define SPIFFS_O_DIRECT                 SPIFFS_DIRECT
+/* If SPIFFS_O_CREAT and SPIFFS_O_EXCL are set, SPIFFS_open() shall fail if the file exists */
+#define SPIFFS_EXCL                     (1<<6)
+#define SPIFFS_O_EXCL                   SPIFFS_EXCL
+
+#define SPIFFS_SEEK_SET                 (0)
+#define SPIFFS_SEEK_CUR                 (1)
+#define SPIFFS_SEEK_END                 (2)
+
+#define SPIFFS_TYPE_FILE                (1)
+#define SPIFFS_TYPE_DIR                 (2)
+#define SPIFFS_TYPE_HARD_LINK           (3)
+#define SPIFFS_TYPE_SOFT_LINK           (4)
+
+#ifndef SPIFFS_LOCK
+#define SPIFFS_LOCK(fs)
+#endif
+
+#ifndef SPIFFS_UNLOCK
+#define SPIFFS_UNLOCK(fs)
+#endif
+
+// phys structs
+
+// spiffs spi configuration struct
+typedef struct {
+    // physical read function
+    spiffs_read hal_read_f;
+    // physical write function
+    spiffs_write hal_write_f;
+    // physical erase function
+    spiffs_erase hal_erase_f;
+#if SPIFFS_SINGLETON == 0
+    // physical size of the spi flash
+    u32_t phys_size;
+    // physical offset in spi flash used for spiffs,
+    // must be on block boundary
+    u32_t phys_addr;
+    // physical size when erasing a block
+    u32_t phys_erase_block;
+
+    // logical size of a block, must be on physical
+    // block size boundary and must never be less than
+    // a physical block
+    u32_t log_block_size;
+    // logical size of a page, must be at least
+    // log_block_size / 8
+    u32_t log_page_size;
+
+#endif
+#if SPIFFS_FILEHDL_OFFSET
+    // an integer offset added to each file handle
+    u16_t fh_ix_offset;
+#endif
+} spiffs_config;
+
+typedef struct spiffs_t {
+    // file system configuration
+    spiffs_config cfg;
+    // number of logical blocks
+    u32_t block_count;
+
+    // cursor for free blocks, block index
+    spiffs_block_ix free_cursor_block_ix;
+    // cursor for free blocks, entry index
+    int free_cursor_obj_lu_entry;
+    // cursor when searching, block index
+    spiffs_block_ix cursor_block_ix;
+    // cursor when searching, entry index
+    int cursor_obj_lu_entry;
+
+    // primary work buffer, size of a logical page
+    u8_t *lu_work;
+    // secondary work buffer, size of a logical page
+    u8_t *work;
+    // file descriptor memory area
+    u8_t *fd_space;
+    // available file descriptors
+    u32_t fd_count;
+
+    // last error
+    s32_t err_code;
+
+    // current number of free blocks
+    u32_t free_blocks;
+    // current number of busy pages
+    u32_t stats_p_allocated;
+    // current number of deleted pages
+    u32_t stats_p_deleted;
+    // flag indicating that garbage collector is cleaning
+    u8_t cleaning;
+    // max erase count amongst all blocks
+    spiffs_obj_id max_erase_count;
+
+#if SPIFFS_GC_STATS
+    u32_t stats_gc_runs;
+#endif
+
+#if SPIFFS_CACHE
+    // cache memory
+    void *cache;
+    // cache size
+    u32_t cache_size;
+#if SPIFFS_CACHE_STATS
+    u32_t cache_hits;
+    u32_t cache_misses;
+#endif
+#endif
+
+    // check callback function
+    spiffs_check_callback check_cb_f;
+    // file callback function
+    spiffs_file_callback file_cb_f;
+    // mounted flag
+    u8_t mounted;
+    // user data
+    void *user_data;
+    // config magic
+    u32_t config_magic;
+} spiffs;
+
+/* spiffs file status struct */
+typedef struct {
+    spiffs_obj_id obj_id;
+    u32_t size;
+    spiffs_obj_type type;
+    spiffs_page_ix pix;
+    u8_t name[SPIFFS_OBJ_NAME_LEN];
+#if SPIFFS_OBJ_META_LEN
+    u8_t meta[SPIFFS_OBJ_META_LEN];
+#endif
+} spiffs_stat;
+
+struct spiffs_dirent {
+    spiffs_obj_id obj_id;
+    u8_t name[SPIFFS_OBJ_NAME_LEN];
+    spiffs_obj_type type;
+    u32_t size;
+    spiffs_page_ix pix;
+#if SPIFFS_OBJ_META_LEN
+    u8_t meta[SPIFFS_OBJ_META_LEN];
+#endif
+};
+
+typedef struct {
+    spiffs *fs;
+    spiffs_block_ix block;
+    int entry;
+} spiffs_DIR;
+
+#if SPIFFS_IX_MAP
+
+typedef struct {
+    // buffer with looked up data pixes
+    spiffs_page_ix *map_buf;
+    // precise file byte offset
+    u32_t offset;
+    // start data span index of lookup buffer
+    spiffs_span_ix start_spix;
+    // end data span index of lookup buffer
+    spiffs_span_ix end_spix;
+} spiffs_ix_map;
+
+#endif
+
+// functions
+
+#if SPIFFS_USE_MAGIC && SPIFFS_USE_MAGIC_LENGTH && SPIFFS_SINGLETON==0
+/**
+ * Special function. This takes a spiffs config struct and returns the number
+ * of blocks this file system was formatted with. This function relies on
+ * that following info is set correctly in given config struct:
+ *
+ * phys_addr, log_page_size, and log_block_size.
+ *
+ * Also, hal_read_f must be set in the config struct.
+ *
+ * One must be sure of the correct page size and that the physical address is
+ * correct in the probed file system when calling this function. It is not
+ * checked if the phys_addr actually points to the start of the file system,
+ * so one might get a false positive if entering a phys_addr somewhere in the
+ * middle of the file system at block boundary. In addition, it is not checked
+ * if the page size is actually correct. If it is not, weird file system sizes
+ * will be returned.
+ *
+ * If this function detects a file system it returns the assumed file system
+ * size, which can be used to set the phys_size.
+ *
+ * Otherwise, it returns an error indicating why it is not regarded as a file
+ * system.
+ *
+ * Note: this function is not protected with SPIFFS_LOCK and SPIFFS_UNLOCK
+ * macros. It returns the error code directly, instead of as read by
+ * SPIFFS_errno.
+ *
+ * @param config        essential parts of the physical and logical
+ *                      configuration of the file system.
+ */
+s32_t SPIFFS_probe_fs(spiffs_config *config);
+#endif // SPIFFS_USE_MAGIC && SPIFFS_USE_MAGIC_LENGTH && SPIFFS_SINGLETON==0
+
+/**
+ * Initializes the file system dynamic parameters and mounts the filesystem.
+ * If SPIFFS_USE_MAGIC is enabled the mounting may fail with SPIFFS_ERR_NOT_A_FS
+ * if the flash does not contain a recognizable file system.
+ * In this case, SPIFFS_format must be called prior to remounting.
+ * @param fs            the file system struct
+ * @param config        the physical and logical configuration of the file system
+ * @param work          a memory work buffer comprising 2*config->log_page_size
+ *                      bytes used throughout all file system operations
+ * @param fd_space      memory for file descriptors
+ * @param fd_space_size memory size of file descriptors
+ * @param cache         memory for cache, may be null
+ * @param cache_size    memory size of cache
+ * @param check_cb_f    callback function for reporting during consistency checks
+ */
+s32_t SPIFFS_mount(spiffs *fs, spiffs_config *config, u8_t *work,
+                   u8_t *fd_space, u32_t fd_space_size,
+                   void *cache, u32_t cache_size,
+                   spiffs_check_callback check_cb_f);
+
+/**
+ * Unmounts the file system. All file handles will be flushed of any
+ * cached writes and closed.
+ * @param fs            the file system struct
+ */
+void SPIFFS_unmount(spiffs *fs);
+
+/**
+ * Creates a new file.
+ * @param fs            the file system struct
+ * @param path          the path of the new file
+ * @param mode          ignored, for posix compliance
+ */
+s32_t SPIFFS_creat(spiffs *fs, const char *path, spiffs_mode mode);
+
+/**
+ * Opens/creates a file.
+ * @param fs            the file system struct
+ * @param path          the path of the new file
+ * @param flags         the flags for the open command, can be combinations of
+ *                      SPIFFS_O_APPEND, SPIFFS_O_TRUNC, SPIFFS_O_CREAT, SPIFFS_O_RDONLY,
+ *                      SPIFFS_O_WRONLY, SPIFFS_O_RDWR, SPIFFS_O_DIRECT, SPIFFS_O_EXCL
+ * @param mode          ignored, for posix compliance
+ */
+spiffs_file SPIFFS_open(spiffs *fs, const char *path, spiffs_flags flags, spiffs_mode mode);
+
+/**
+ * Opens a file by given dir entry.
+ * Optimization purposes, when traversing a file system with SPIFFS_readdir
+ * a normal SPIFFS_open would need to traverse the filesystem again to find
+ * the file, whilst SPIFFS_open_by_dirent already knows where the file resides.
+ * @param fs            the file system struct
+ * @param e             the dir entry to the file
+ * @param flags         the flags for the open command, can be combinations of
+ *                      SPIFFS_APPEND, SPIFFS_TRUNC, SPIFFS_CREAT, SPIFFS_RD_ONLY,
+ *                      SPIFFS_WR_ONLY, SPIFFS_RDWR, SPIFFS_DIRECT.
+ *                      SPIFFS_CREAT will have no effect in this case.
+ * @param mode          ignored, for posix compliance
+ */
+spiffs_file SPIFFS_open_by_dirent(spiffs *fs, struct spiffs_dirent *e, spiffs_flags flags, spiffs_mode mode);
+
+/**
+ * Opens a file by given page index.
+ * Optimization purposes, opens a file by directly pointing to the page
+ * index in the spi flash.
+ * If the page index does not point to a file header SPIFFS_ERR_NOT_A_FILE
+ * is returned.
+ * @param fs            the file system struct
+ * @param page_ix       the page index
+ * @param flags         the flags for the open command, can be combinations of
+ *                      SPIFFS_APPEND, SPIFFS_TRUNC, SPIFFS_CREAT, SPIFFS_RD_ONLY,
+ *                      SPIFFS_WR_ONLY, SPIFFS_RDWR, SPIFFS_DIRECT.
+ *                      SPIFFS_CREAT will have no effect in this case.
+ * @param mode          ignored, for posix compliance
+ */
+spiffs_file SPIFFS_open_by_page(spiffs *fs, spiffs_page_ix page_ix, spiffs_flags flags, spiffs_mode mode);
+
+/**
+ * Reads from given filehandle.
+ * @param fs            the file system struct
+ * @param fh            the filehandle
+ * @param buf           where to put read data
+ * @param len           how much to read
+ * @returns number of bytes read, or -1 if error
+ */
+s32_t SPIFFS_read(spiffs *fs, spiffs_file fh, void *buf, s32_t len);
+
+/**
+ * Writes to given filehandle.
+ * @param fs            the file system struct
+ * @param fh            the filehandle
+ * @param buf           the data to write
+ * @param len           how much to write
+ * @returns number of bytes written, or -1 if error
+ */
+s32_t SPIFFS_write(spiffs *fs, spiffs_file fh, void *buf, s32_t len);
+
+/**
+ * Moves the read/write file offset. Resulting offset is returned or negative if error.
+ * lseek(fs, fd, 0, SPIFFS_SEEK_CUR) will thus return current offset.
+ * @param fs            the file system struct
+ * @param fh            the filehandle
+ * @param offs          how much/where to move the offset
+ * @param whence        if SPIFFS_SEEK_SET, the file offset shall be set to offset bytes
+ *                      if SPIFFS_SEEK_CUR, the file offset shall be set to its current location plus offset
+ *                      if SPIFFS_SEEK_END, the file offset shall be set to the size of the file plus offse, which should be negative
+ */
+s32_t SPIFFS_lseek(spiffs *fs, spiffs_file fh, s32_t offs, int whence);
+
+/**
+ * Removes a file by path
+ * @param fs            the file system struct
+ * @param path          the path of the file to remove
+ */
+s32_t SPIFFS_remove(spiffs *fs, const char *path);
+
+/**
+ * Removes a file by filehandle
+ * @param fs            the file system struct
+ * @param fh            the filehandle of the file to remove
+ */
+s32_t SPIFFS_fremove(spiffs *fs, spiffs_file fh);
+
+/**
+ * Gets file status by path
+ * @param fs            the file system struct
+ * @param path          the path of the file to stat
+ * @param s             the stat struct to populate
+ */
+s32_t SPIFFS_stat(spiffs *fs, const char *path, spiffs_stat *s);
+
+/**
+ * Gets file status by filehandle
+ * @param fs            the file system struct
+ * @param fh            the filehandle of the file to stat
+ * @param s             the stat struct to populate
+ */
+s32_t SPIFFS_fstat(spiffs *fs, spiffs_file fh, spiffs_stat *s);
+
+/**
+ * Flushes all pending write operations from cache for given file
+ * @param fs            the file system struct
+ * @param fh            the filehandle of the file to flush
+ */
+s32_t SPIFFS_fflush(spiffs *fs, spiffs_file fh);
+
+/**
+ * Closes a filehandle. If there are pending write operations, these are finalized before closing.
+ * @param fs            the file system struct
+ * @param fh            the filehandle of the file to close
+ */
+s32_t SPIFFS_close(spiffs *fs, spiffs_file fh);
+
+/**
+ * Renames a file
+ * @param fs            the file system struct
+ * @param old           path of file to rename
+ * @param newPath       new path of file
+ */
+s32_t SPIFFS_rename(spiffs *fs, const char *old, const char *newPath);
+
+#if SPIFFS_OBJ_META_LEN
+/**
+ * Updates file's metadata
+ * @param fs            the file system struct
+ * @param path          path to the file
+ * @param meta          new metadata. must be SPIFFS_OBJ_META_LEN bytes long.
+ */
+s32_t SPIFFS_update_meta(spiffs *fs, const char *name, const void *meta);
+
+/**
+ * Updates file's metadata
+ * @param fs            the file system struct
+ * @param fh            file handle of the file
+ * @param meta          new metadata. must be SPIFFS_OBJ_META_LEN bytes long.
+ */
+s32_t SPIFFS_fupdate_meta(spiffs *fs, spiffs_file fh, const void *meta);
+#endif
+
+/**
+ * Returns last error of last file operation.
+ * @param fs            the file system struct
+ */
+s32_t SPIFFS_errno(spiffs *fs);
+
+/**
+ * Clears last error.
+ * @param fs            the file system struct
+ */
+void SPIFFS_clearerr(spiffs *fs);
+
+/**
+ * Opens a directory stream corresponding to the given name.
+ * The stream is positioned at the first entry in the directory.
+ * On hydrogen builds the name argument is ignored as hydrogen builds always correspond
+ * to a flat file structure - no directories.
+ * @param fs            the file system struct
+ * @param name          the name of the directory
+ * @param d             pointer the directory stream to be populated
+ */
+spiffs_DIR *SPIFFS_opendir(spiffs *fs, const char *name, spiffs_DIR *d);
+
+/**
+ * Closes a directory stream
+ * @param d             the directory stream to close
+ */
+s32_t SPIFFS_closedir(spiffs_DIR *d);
+
+/**
+ * Reads a directory into given spifs_dirent struct.
+ * @param d             pointer to the directory stream
+ * @param e             the dirent struct to be populated
+ * @returns null if error or end of stream, else given dirent is returned
+ */
+struct spiffs_dirent *SPIFFS_readdir(spiffs_DIR *d, struct spiffs_dirent *e);
+
+/**
+ * Runs a consistency check on given filesystem.
+ * @param fs            the file system struct
+ */
+s32_t SPIFFS_check(spiffs *fs);
+
+/**
+ * Returns number of total bytes available and number of used bytes.
+ * This is an estimation, and depends on if there a many files with little
+ * data or few files with much data.
+ * NB: If used number of bytes exceeds total bytes, a SPIFFS_check should
+ * run. This indicates a power loss in midst of things. In worst case
+ * (repeated powerlosses in mending or gc) you might have to delete some files.
+ *
+ * @param fs            the file system struct
+ * @param total         total number of bytes in filesystem
+ * @param used          used number of bytes in filesystem
+ */
+s32_t SPIFFS_info(spiffs *fs, u32_t *total, u32_t *used);
+
+/**
+ * Formats the entire file system. All data will be lost.
+ * The filesystem must not be mounted when calling this.
+ *
+ * NB: formatting is awkward. Due to backwards compatibility, SPIFFS_mount
+ * MUST be called prior to formatting in order to configure the filesystem.
+ * If SPIFFS_mount succeeds, SPIFFS_unmount must be called before calling
+ * SPIFFS_format.
+ * If SPIFFS_mount fails, SPIFFS_format can be called directly without calling
+ * SPIFFS_unmount first.
+ *
+ * @param fs            the file system struct
+ */
+s32_t SPIFFS_format(spiffs *fs);
+
+/**
+ * Returns nonzero if spiffs is mounted, or zero if unmounted.
+ * @param fs            the file system struct
+ */
+u8_t SPIFFS_mounted(spiffs *fs);
+
+/**
+ * Tries to find a block where most or all pages are deleted, and erase that
+ * block if found. Does not care for wear levelling. Will not move pages
+ * around.
+ * If parameter max_free_pages are set to 0, only blocks with only deleted
+ * pages will be selected.
+ *
+ * NB: the garbage collector is automatically called when spiffs needs free
+ * pages. The reason for this function is to give possibility to do background
+ * tidying when user knows the system is idle.
+ *
+ * Use with care.
+ *
+ * Setting max_free_pages to anything larger than zero will eventually wear
+ * flash more as a block containing free pages can be erased.
+ *
+ * Will set err_no to SPIFFS_OK if a block was found and erased,
+ * SPIFFS_ERR_NO_DELETED_BLOCK if no matching block was found,
+ * or other error.
+ *
+ * @param fs             the file system struct
+ * @param max_free_pages maximum number allowed free pages in block
+ */
+s32_t SPIFFS_gc_quick(spiffs *fs, u16_t max_free_pages);
+
+/**
+ * Will try to make room for given amount of bytes in the filesystem by moving
+ * pages and erasing blocks.
+ * If it is physically impossible, err_no will be set to SPIFFS_ERR_FULL. If
+ * there already is this amount (or more) of free space, SPIFFS_gc will
+ * silently return. It is recommended to call SPIFFS_info before invoking
+ * this method in order to determine what amount of bytes to give.
+ *
+ * NB: the garbage collector is automatically called when spiffs needs free
+ * pages. The reason for this function is to give possibility to do background
+ * tidying when user knows the system is idle.
+ *
+ * Use with care.
+ *
+ * @param fs            the file system struct
+ * @param size          amount of bytes that should be freed
+ */
+s32_t SPIFFS_gc(spiffs *fs, u32_t size);
+
+/**
+ * Check if EOF reached.
+ * @param fs            the file system struct
+ * @param fh            the filehandle of the file to check
+ */
+s32_t SPIFFS_eof(spiffs *fs, spiffs_file fh);
+
+/**
+ * Get position in file.
+ * @param fs            the file system struct
+ * @param fh            the filehandle of the file to check
+ */
+s32_t SPIFFS_tell(spiffs *fs, spiffs_file fh);
+
+/**
+ * Registers a callback function that keeps track on operations on file
+ * headers. Do note, that this callback is called from within internal spiffs
+ * mechanisms. Any operations on the actual file system being callbacked from
+ * in this callback will mess things up for sure - do not do this.
+ * This can be used to track where files are and move around during garbage
+ * collection, which in turn can be used to build location tables in ram.
+ * Used in conjuction with SPIFFS_open_by_page this may improve performance
+ * when opening a lot of files.
+ * Must be invoked after mount.
+ *
+ * @param fs            the file system struct
+ * @param cb_func       the callback on file operations
+ */
+s32_t SPIFFS_set_file_callback_func(spiffs *fs, spiffs_file_callback cb_func);
+
+#if SPIFFS_IX_MAP
+
+/**
+ * Maps the first level index lookup to a given memory map.
+ * This will make reading big files faster, as the memory map will be used for
+ * looking up data pages instead of searching for the indices on the physical
+ * medium. When mapping, all affected indicies are found and the information is
+ * copied to the array.
+ * Whole file or only parts of it may be mapped. The index map will cover file
+ * contents from argument offset until and including arguments (offset+len).
+ * It is valid to map a longer range than the current file size. The map will
+ * then be populated when the file grows.
+ * On garbage collections and file data page movements, the map array will be
+ * automatically updated. Do not tamper with the map array, as this contains
+ * the references to the data pages. Modifying it from outside will corrupt any
+ * future readings using this file descriptor.
+ * The map will no longer be used when the file descriptor closed or the file
+ * is unmapped.
+ * This can be useful to get faster and more deterministic timing when reading
+ * large files, or when seeking and reading a lot within a file.
+ * @param fs      the file system struct
+ * @param fh      the file handle of the file to map
+ * @param map     a spiffs_ix_map struct, describing the index map
+ * @param offset  absolute file offset where to start the index map
+ * @param len     length of the mapping in actual file bytes
+ * @param map_buf the array buffer for the look up data - number of required
+ *                elements in the array can be derived from function
+ *                SPIFFS_bytes_to_ix_map_entries given the length
+ */
+s32_t SPIFFS_ix_map(spiffs *fs, spiffs_file fh, spiffs_ix_map *map,
+                    u32_t offset, u32_t len, spiffs_page_ix *map_buf);
+
+/**
+ * Unmaps the index lookup from this filehandle. All future readings will
+ * proceed as normal, requiring reading of the first level indices from
+ * physical media.
+ * The map and map buffer given in function SPIFFS_ix_map will no longer be
+ * referenced by spiffs.
+ * It is not strictly necessary to unmap a file before closing it, as closing
+ * a file will automatically unmap it.
+ * @param fs      the file system struct
+ * @param fh      the file handle of the file to unmap
+ */
+s32_t SPIFFS_ix_unmap(spiffs *fs, spiffs_file fh);
+
+/**
+ * Moves the offset for the index map given in function SPIFFS_ix_map. Parts or
+ * all of the map buffer will repopulated.
+ * @param fs      the file system struct
+ * @param fh      the mapped file handle of the file to remap
+ * @param offset  new absolute file offset where to start the index map
+ */
+s32_t SPIFFS_ix_remap(spiffs *fs, spiffs_file fh, u32_t offs);
+
+/**
+ * Utility function to get number of spiffs_page_ix entries a map buffer must
+ * contain on order to map given amount of file data in bytes.
+ * See function SPIFFS_ix_map and SPIFFS_ix_map_entries_to_bytes.
+ * @param fs      the file system struct
+ * @param bytes   number of file data bytes to map
+ * @return        needed number of elements in a spiffs_page_ix array needed to
+ *                map given amount of bytes in a file
+ */
+s32_t SPIFFS_bytes_to_ix_map_entries(spiffs *fs, u32_t bytes);
+
+/**
+ * Utility function to amount of file data bytes that can be mapped when
+ * mapping a file with buffer having given number of spiffs_page_ix entries.
+ * See function SPIFFS_ix_map and SPIFFS_bytes_to_ix_map_entries.
+ * @param fs      the file system struct
+ * @param map_page_ix_entries   number of entries in a spiffs_page_ix array
+ * @return        amount of file data in bytes that can be mapped given a map
+ *                buffer having given amount of spiffs_page_ix entries
+ */
+s32_t SPIFFS_ix_map_entries_to_bytes(spiffs *fs, u32_t map_page_ix_entries);
+
+#endif // SPIFFS_IX_MAP
+
+
+#if SPIFFS_TEST_VISUALISATION
+/**
+ * Prints out a visualization of the filesystem.
+ * @param fs            the file system struct
+ */
+s32_t SPIFFS_vis(spiffs *fs);
+#endif
+
+#if SPIFFS_BUFFER_HELP
+/**
+ * Returns number of bytes needed for the filedescriptor buffer given
+ * amount of file descriptors.
+ */
+u32_t SPIFFS_buffer_bytes_for_filedescs(spiffs *fs, u32_t num_descs);
+
+#if SPIFFS_CACHE
+/**
+ * Returns number of bytes needed for the cache buffer given
+ * amount of cache pages.
+ */
+u32_t SPIFFS_buffer_bytes_for_cache(spiffs *fs, u32_t num_pages);
+#endif
+#endif
+
+#if SPIFFS_CACHE
+#endif
+#if defined(__cplusplus)
+}
+#endif
+
+#endif /* SPIFFS_H_ */
diff --git a/armsrc/spiffs_cache.c b/armsrc/spiffs_cache.c
new file mode 100644
index 000000000..f86133be4
--- /dev/null
+++ b/armsrc/spiffs_cache.c
@@ -0,0 +1,319 @@
+/*
+ * spiffs_cache.c
+ *
+ *  Created on: Jun 23, 2013
+ *      Author: petera
+ */
+
+#include "spiffs.h"
+#include "spiffs_nucleus.h"
+
+#if SPIFFS_CACHE
+
+// returns cached page for give page index, or null if no such cached page
+static spiffs_cache_page *spiffs_cache_page_get(spiffs *fs, spiffs_page_ix pix) {
+    spiffs_cache *cache = spiffs_get_cache(fs);
+    if ((cache->cpage_use_map & cache->cpage_use_mask) == 0) return 0;
+    int i;
+    for (i = 0; i < cache->cpage_count; i++) {
+        spiffs_cache_page *cp = spiffs_get_cache_page_hdr(fs, cache, i);
+        if ((cache->cpage_use_map & (1 << i)) &&
+                (cp->flags & SPIFFS_CACHE_FLAG_TYPE_WR) == 0 &&
+                cp->ucache.spix.pix == pix) {
+            //SPIFFS_CACHE_DBG("CACHE_GET: have cache page "_SPIPRIi" for "_SPIPRIpg"\n", i, pix);
+            cp->last_access = cache->last_access;
+            return cp;
+        }
+    }
+    //SPIFFS_CACHE_DBG("CACHE_GET: no cache for "_SPIPRIpg"\n", pix);
+    return 0;
+}
+
+// frees cached page
+static s32_t spiffs_cache_page_free(spiffs *fs, int ix, u8_t write_back) {
+    s32_t res = SPIFFS_OK;
+    spiffs_cache *cache = spiffs_get_cache(fs);
+    spiffs_cache_page *cp = spiffs_get_cache_page_hdr(fs, cache, ix);
+    if (cache->cpage_use_map & (1 << ix)) {
+        if (write_back &&
+                (cp->flags & SPIFFS_CACHE_FLAG_TYPE_WR) == 0 &&
+                (cp->flags & SPIFFS_CACHE_FLAG_DIRTY)) {
+            u8_t *mem =  spiffs_get_cache_page(fs, cache, ix);
+            SPIFFS_CACHE_DBG("CACHE_FREE: write cache page "_SPIPRIi" pix "_SPIPRIpg"\n", ix, cp->pix);
+            res = SPIFFS_HAL_WRITE(fs, SPIFFS_PAGE_TO_PADDR(fs, cp->ucache.spix.pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), mem);
+        }
+
+#if SPIFFS_CACHE_WR
+        if (cp->flags & SPIFFS_CACHE_FLAG_TYPE_WR) {
+            SPIFFS_CACHE_DBG("CACHE_FREE: free cache page "_SPIPRIi" objid "_SPIPRIid"\n", ix, cp->obj_id);
+        } else
+#endif
+        {
+            SPIFFS_CACHE_DBG("CACHE_FREE: free cache page "_SPIPRIi" pix "_SPIPRIpg"\n", ix, cp->pix);
+        }
+        cache->cpage_use_map &= ~(1 << ix);
+        cp->flags = 0;
+    }
+
+    return res;
+}
+
+// removes the oldest accessed cached page
+static s32_t spiffs_cache_page_remove_oldest(spiffs *fs, u8_t flag_mask, u8_t flags) {
+    s32_t res = SPIFFS_OK;
+    spiffs_cache *cache = spiffs_get_cache(fs);
+
+    if ((cache->cpage_use_map & cache->cpage_use_mask) != cache->cpage_use_mask) {
+        // at least one free cpage
+        return SPIFFS_OK;
+    }
+
+    // all busy, scan thru all to find the cpage which has oldest access
+    int i;
+    int cand_ix = -1;
+    u32_t oldest_val = 0;
+    for (i = 0; i < cache->cpage_count; i++) {
+        spiffs_cache_page *cp = spiffs_get_cache_page_hdr(fs, cache, i);
+        if ((cache->last_access - cp->last_access) > oldest_val &&
+                (cp->flags & flag_mask) == flags) {
+            oldest_val = cache->last_access - cp->last_access;
+            cand_ix = i;
+        }
+    }
+
+    if (cand_ix >= 0) {
+        res = spiffs_cache_page_free(fs, cand_ix, 1);
+    }
+
+    return res;
+}
+
+// allocates a new cached page and returns it, or null if all cache pages are busy
+static spiffs_cache_page *spiffs_cache_page_allocate(spiffs *fs) {
+    spiffs_cache *cache = spiffs_get_cache(fs);
+    if (cache->cpage_use_map == 0xffffffff) {
+        // out of cache memory
+        return 0;
+    }
+    int i;
+    for (i = 0; i < cache->cpage_count; i++) {
+        if ((cache->cpage_use_map & (1 << i)) == 0) {
+            spiffs_cache_page *cp = spiffs_get_cache_page_hdr(fs, cache, i);
+            cache->cpage_use_map |= (1 << i);
+            cp->last_access = cache->last_access;
+            //SPIFFS_CACHE_DBG("CACHE_ALLO: allocated cache page "_SPIPRIi"\n", i);
+            return cp;
+        }
+    }
+    // out of cache entries
+    return 0;
+}
+
+// drops the cache page for give page index
+void spiffs_cache_drop_page(spiffs *fs, spiffs_page_ix pix) {
+    spiffs_cache_page *cp =  spiffs_cache_page_get(fs, pix);
+    if (cp) {
+        spiffs_cache_page_free(fs, cp->ix, 0);
+    }
+}
+
+// ------------------------------
+
+// reads from spi flash or the cache
+s32_t spiffs_phys_rd(
+    spiffs *fs,
+    u8_t op,
+    spiffs_file fh,
+    u32_t addr,
+    u32_t len,
+    u8_t *dst) {
+    (void)fh;
+    s32_t res = SPIFFS_OK;
+    spiffs_cache *cache = spiffs_get_cache(fs);
+    spiffs_cache_page *cp =  spiffs_cache_page_get(fs, SPIFFS_PADDR_TO_PAGE(fs, addr));
+    cache->last_access++;
+    if (cp) {
+        // we've already got one, you see
+#if SPIFFS_CACHE_STATS
+        fs->cache_hits++;
+#endif
+        cp->last_access = cache->last_access;
+        u8_t *mem =  spiffs_get_cache_page(fs, cache, cp->ix);
+        _SPIFFS_MEMCPY(dst, &mem[SPIFFS_PADDR_TO_PAGE_OFFSET(fs, addr)], len);
+    } else {
+        if ((op & SPIFFS_OP_TYPE_MASK) == SPIFFS_OP_T_OBJ_LU2) {
+            // for second layer lookup functions, we do not cache in order to prevent shredding
+            return SPIFFS_HAL_READ(fs, addr, len, dst);
+        }
+#if SPIFFS_CACHE_STATS
+        fs->cache_misses++;
+#endif
+        // this operation will always free one cache page (unless all already free),
+        // the result code stems from the write operation of the possibly freed cache page
+        res = spiffs_cache_page_remove_oldest(fs, SPIFFS_CACHE_FLAG_TYPE_WR, 0);
+
+        cp = spiffs_cache_page_allocate(fs);
+        if (cp) {
+            cp->flags = SPIFFS_CACHE_FLAG_WRTHRU;
+            cp->ucache.spix.pix = SPIFFS_PADDR_TO_PAGE(fs, addr);
+            SPIFFS_CACHE_DBG("CACHE_ALLO: allocated cache page "_SPIPRIi" for pix "_SPIPRIpg "\n", cp->ix, cp->pix);
+
+            s32_t res2 = SPIFFS_HAL_READ(fs,
+                                         addr - SPIFFS_PADDR_TO_PAGE_OFFSET(fs, addr),
+                                         SPIFFS_CFG_LOG_PAGE_SZ(fs),
+                                         spiffs_get_cache_page(fs, cache, cp->ix));
+            if (res2 != SPIFFS_OK) {
+                // honor read failure before possible write failure (bad idea?)
+                res = res2;
+            }
+            u8_t *mem =  spiffs_get_cache_page(fs, cache, cp->ix);
+            _SPIFFS_MEMCPY(dst, &mem[SPIFFS_PADDR_TO_PAGE_OFFSET(fs, addr)], len);
+        } else {
+            // this will never happen, last resort for sake of symmetry
+            s32_t res2 = SPIFFS_HAL_READ(fs, addr, len, dst);
+            if (res2 != SPIFFS_OK) {
+                // honor read failure before possible write failure (bad idea?)
+                res = res2;
+            }
+        }
+    }
+    return res;
+}
+
+// writes to spi flash and/or the cache
+s32_t spiffs_phys_wr(
+    spiffs *fs,
+    u8_t op,
+    spiffs_file fh,
+    u32_t addr,
+    u32_t len,
+    u8_t *src) {
+    (void)fh;
+    spiffs_page_ix pix = SPIFFS_PADDR_TO_PAGE(fs, addr);
+    spiffs_cache *cache = spiffs_get_cache(fs);
+    spiffs_cache_page *cp =  spiffs_cache_page_get(fs, pix);
+
+    if (cp && (op & SPIFFS_OP_COM_MASK) != SPIFFS_OP_C_WRTHRU) {
+        // have a cache page
+        // copy in data to cache page
+
+        if ((op & SPIFFS_OP_COM_MASK) == SPIFFS_OP_C_DELE &&
+                (op & SPIFFS_OP_TYPE_MASK) != SPIFFS_OP_T_OBJ_LU) {
+            // page is being deleted, wipe from cache - unless it is a lookup page
+            spiffs_cache_page_free(fs, cp->ix, 0);
+            return SPIFFS_HAL_WRITE(fs, addr, len, src);
+        }
+
+        u8_t *mem =  spiffs_get_cache_page(fs, cache, cp->ix);
+        _SPIFFS_MEMCPY(&mem[SPIFFS_PADDR_TO_PAGE_OFFSET(fs, addr)], src, len);
+
+        cache->last_access++;
+        cp->last_access = cache->last_access;
+
+        if (cp->flags & SPIFFS_CACHE_FLAG_WRTHRU) {
+            // page is being updated, no write-cache, just pass thru
+            return SPIFFS_HAL_WRITE(fs, addr, len, src);
+        } else {
+            return SPIFFS_OK;
+        }
+    } else {
+        // no cache page, no write cache - just write thru
+        return SPIFFS_HAL_WRITE(fs, addr, len, src);
+    }
+}
+
+#if SPIFFS_CACHE_WR
+// returns the cache page that this fd refers, or null if no cache page
+spiffs_cache_page *spiffs_cache_page_get_by_fd(spiffs *fs, spiffs_fd *fd) {
+    spiffs_cache *cache = spiffs_get_cache(fs);
+
+    if ((cache->cpage_use_map & cache->cpage_use_mask) == 0) {
+        // all cpages free, no cpage cannot be assigned to obj_id
+        return 0;
+    }
+
+    int i;
+    for (i = 0; i < cache->cpage_count; i++) {
+        spiffs_cache_page *cp = spiffs_get_cache_page_hdr(fs, cache, i);
+        if ((cache->cpage_use_map & (1 << i)) &&
+                (cp->flags & SPIFFS_CACHE_FLAG_TYPE_WR) &&
+                cp->ucache.swrc.obj_id == fd->obj_id) {
+            return cp;
+        }
+    }
+
+    return 0;
+}
+
+// allocates a new cache page and refers this to given fd - flushes an old cache
+// page if all cache is busy
+spiffs_cache_page *spiffs_cache_page_allocate_by_fd(spiffs *fs, spiffs_fd *fd) {
+    // before this function is called, it is ensured that there is no already existing
+    // cache page with same object id
+    spiffs_cache_page_remove_oldest(fs, SPIFFS_CACHE_FLAG_TYPE_WR, 0);
+    spiffs_cache_page *cp = spiffs_cache_page_allocate(fs);
+    if (cp == 0) {
+        // could not get cache page
+        return 0;
+    }
+
+    cp->flags = SPIFFS_CACHE_FLAG_TYPE_WR;
+    cp->ucache.swrc.obj_id = fd->obj_id;
+    fd->cache_page = cp;
+    SPIFFS_CACHE_DBG("CACHE_ALLO: allocated cache page "_SPIPRIi" for fd "_SPIPRIfd ":"_SPIPRIid "\n", cp->ix, fd->file_nbr, fd->obj_id);
+    return cp;
+}
+
+// unrefers all fds that this cache page refers to and releases the cache page
+void spiffs_cache_fd_release(spiffs *fs, spiffs_cache_page *cp) {
+    if (cp == 0) return;
+    u32_t i;
+    spiffs_fd *fds = (spiffs_fd *)fs->fd_space;
+    for (i = 0; i < fs->fd_count; i++) {
+        spiffs_fd *cur_fd = &fds[i];
+        if (cur_fd->file_nbr != 0 && cur_fd->cache_page == cp) {
+            cur_fd->cache_page = 0;
+        }
+    }
+    spiffs_cache_page_free(fs, cp->ix, 0);
+
+    cp->ucache.swrc.obj_id = 0;
+}
+
+#endif
+
+// initializes the cache
+void spiffs_cache_init(spiffs *fs) {
+    if (fs->cache == 0) return;
+    u32_t sz = fs->cache_size;
+    u32_t cache_mask = 0;
+    int i;
+    int cache_entries =
+        (sz - sizeof(spiffs_cache)) / (SPIFFS_CACHE_PAGE_SIZE(fs));
+    if (cache_entries <= 0) return;
+
+    for (i = 0; i < cache_entries; i++) {
+        cache_mask <<= 1;
+        cache_mask |= 1;
+    }
+
+    spiffs_cache cache;
+    memset(&cache, 0, sizeof(spiffs_cache));
+    cache.cpage_count = cache_entries;
+    cache.cpages = (u8_t *)((u8_t *)fs->cache + sizeof(spiffs_cache));
+
+    cache.cpage_use_map = 0xffffffff;
+    cache.cpage_use_mask = cache_mask;
+    _SPIFFS_MEMCPY(fs->cache, &cache, sizeof(spiffs_cache));
+
+    spiffs_cache *c = spiffs_get_cache(fs);
+
+    memset(c->cpages, 0, c->cpage_count * SPIFFS_CACHE_PAGE_SIZE(fs));
+
+    c->cpage_use_map &= ~(c->cpage_use_mask);
+    for (i = 0; i < cache.cpage_count; i++) {
+        spiffs_get_cache_page_hdr(fs, c, i)->ix = i;
+    }
+}
+
+#endif // SPIFFS_CACHE
diff --git a/armsrc/spiffs_check.c b/armsrc/spiffs_check.c
new file mode 100644
index 000000000..90fba8a41
--- /dev/null
+++ b/armsrc/spiffs_check.c
@@ -0,0 +1,1006 @@
+/*
+ * spiffs_check.c
+ *
+ * Contains functionality for checking file system consistency
+ * and mending problems.
+ * Three levels of consistency checks are implemented:
+ *
+ * Look up consistency
+ *   Checks if indices in lookup pages are coherent with page headers
+ * Object index consistency
+ *   Checks if there are any orphaned object indices (missing object index headers).
+ *   If an object index is found but not its header, the object index is deleted.
+ *   This is critical for the following page consistency check.
+ * Page consistency
+ *   Checks for pages that ought to be indexed, ought not to be indexed, are multiple indexed
+ *
+ *
+ *  Created on: Jul 7, 2013
+ *      Author: petera
+ */
+
+
+#include "spiffs.h"
+#include "spiffs_nucleus.h"
+
+#if !SPIFFS_READ_ONLY
+
+#if SPIFFS_HAL_CALLBACK_EXTRA
+#define CHECK_CB(_fs, _type, _rep, _arg1, _arg2) \
+  do { \
+    if ((_fs)->check_cb_f) (_fs)->check_cb_f((_fs), (_type), (_rep), (_arg1), (_arg2)); \
+  } while (0)
+#else
+#define CHECK_CB(_fs, _type, _rep, _arg1, _arg2) \
+  do { \
+    if ((_fs)->check_cb_f) (_fs)->check_cb_f((_type), (_rep), (_arg1), (_arg2)); \
+  } while (0)
+#endif
+
+//---------------------------------------
+// Look up consistency
+
+// searches in the object indices and returns the referenced page index given
+// the object id and the data span index
+// destroys fs->lu_work
+static s32_t spiffs_object_get_data_page_index_reference(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_span_ix data_spix,
+    spiffs_page_ix *pix,
+    spiffs_page_ix *objix_pix) {
+    s32_t res;
+
+    // calculate object index span index for given data page span index
+    spiffs_span_ix objix_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, data_spix);
+
+    // find obj index for obj id and span index
+    res = spiffs_obj_lu_find_id_and_span(fs, obj_id | SPIFFS_OBJ_ID_IX_FLAG, objix_spix, 0, objix_pix);
+    SPIFFS_CHECK_RES(res);
+
+    // load obj index entry
+    u32_t addr = SPIFFS_PAGE_TO_PADDR(fs, *objix_pix);
+    if (objix_spix == 0) {
+        // get referenced page from object index header
+        addr += sizeof(spiffs_page_object_ix_header) + data_spix * sizeof(spiffs_page_ix);
+    } else {
+        // get referenced page from object index
+        addr += sizeof(spiffs_page_object_ix) + SPIFFS_OBJ_IX_ENTRY(fs, data_spix) * sizeof(spiffs_page_ix);
+    }
+
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ, 0, addr, sizeof(spiffs_page_ix), (u8_t *)pix);
+
+    return res;
+}
+
+// copies page contents to a new page
+static s32_t spiffs_rewrite_page(spiffs *fs, spiffs_page_ix cur_pix, spiffs_page_header *p_hdr, spiffs_page_ix *new_pix) {
+    s32_t res;
+    res = spiffs_page_allocate_data(fs, p_hdr->obj_id, p_hdr, 0, 0, 0, 0, new_pix);
+    SPIFFS_CHECK_RES(res);
+    res = spiffs_phys_cpy(fs, 0,
+                          SPIFFS_PAGE_TO_PADDR(fs, *new_pix) + sizeof(spiffs_page_header),
+                          SPIFFS_PAGE_TO_PADDR(fs, cur_pix) + sizeof(spiffs_page_header),
+                          SPIFFS_DATA_PAGE_SIZE(fs));
+    SPIFFS_CHECK_RES(res);
+    return res;
+}
+
+// rewrites the object index for given object id and replaces the
+// data page index to a new page index
+static s32_t spiffs_rewrite_index(spiffs *fs, spiffs_obj_id obj_id, spiffs_span_ix data_spix, spiffs_page_ix new_data_pix, spiffs_page_ix objix_pix) {
+    s32_t res;
+    spiffs_block_ix bix;
+    int entry;
+    spiffs_page_ix free_pix;
+    obj_id |= SPIFFS_OBJ_ID_IX_FLAG;
+
+    // find free entry
+    res = spiffs_obj_lu_find_free(fs, fs->free_cursor_block_ix, fs->free_cursor_obj_lu_entry, &bix, &entry);
+    SPIFFS_CHECK_RES(res);
+    free_pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, entry);
+
+    // calculate object index span index for given data page span index
+    spiffs_span_ix objix_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, data_spix);
+    if (objix_spix == 0) {
+        // calc index in index header
+        entry = data_spix;
+    } else {
+        // calc entry in index
+        entry = SPIFFS_OBJ_IX_ENTRY(fs, data_spix);
+
+    }
+    // load index
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                     0, SPIFFS_PAGE_TO_PADDR(fs, objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+    SPIFFS_CHECK_RES(res);
+    spiffs_page_header *objix_p_hdr = (spiffs_page_header *)fs->lu_work;
+
+    // be ultra safe, double check header against provided data
+    if (objix_p_hdr->obj_id != obj_id) {
+        spiffs_page_delete(fs, free_pix);
+        return SPIFFS_ERR_CHECK_OBJ_ID_MISM;
+    }
+    if (objix_p_hdr->span_ix != objix_spix) {
+        spiffs_page_delete(fs, free_pix);
+        return SPIFFS_ERR_CHECK_SPIX_MISM;
+    }
+    if ((objix_p_hdr->flags & (SPIFFS_PH_FLAG_USED | SPIFFS_PH_FLAG_IXDELE | SPIFFS_PH_FLAG_INDEX |
+                               SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_DELET)) !=
+            (SPIFFS_PH_FLAG_IXDELE | SPIFFS_PH_FLAG_DELET)) {
+        spiffs_page_delete(fs, free_pix);
+        return SPIFFS_ERR_CHECK_FLAGS_BAD;
+    }
+
+    // rewrite in mem
+    if (objix_spix == 0) {
+        ((spiffs_page_ix *)((u8_t *)fs->lu_work + sizeof(spiffs_page_object_ix_header)))[data_spix] = new_data_pix;
+    } else {
+        ((spiffs_page_ix *)((u8_t *)fs->lu_work + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, data_spix)] = new_data_pix;
+    }
+
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                     0, SPIFFS_PAGE_TO_PADDR(fs, free_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+    SPIFFS_CHECK_RES(res);
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_UPDT,
+                     0, SPIFFS_BLOCK_TO_PADDR(fs, SPIFFS_BLOCK_FOR_PAGE(fs, free_pix)) + SPIFFS_OBJ_LOOKUP_ENTRY_FOR_PAGE(fs, free_pix) * sizeof(spiffs_page_ix),
+                     sizeof(spiffs_obj_id),
+                     (u8_t *)&obj_id);
+    SPIFFS_CHECK_RES(res);
+    res = spiffs_page_delete(fs, objix_pix);
+
+    return res;
+}
+
+// deletes an object just by marking object index header as deleted
+static s32_t spiffs_delete_obj_lazy(spiffs *fs, spiffs_obj_id obj_id) {
+    spiffs_page_ix objix_hdr_pix;
+    s32_t res;
+    res = spiffs_obj_lu_find_id_and_span(fs, obj_id, 0, 0, &objix_hdr_pix);
+    if (res == SPIFFS_ERR_NOT_FOUND) {
+        return SPIFFS_OK;
+    }
+    SPIFFS_CHECK_RES(res);
+    u8_t flags = 0xff;
+#if SPIFFS_NO_BLIND_WRITES
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                     0, SPIFFS_PAGE_TO_PADDR(fs, objix_hdr_pix) + offsetof(spiffs_page_header, flags),
+                     sizeof(flags), &flags);
+    SPIFFS_CHECK_RES(res);
+#endif
+    flags &= ~SPIFFS_PH_FLAG_IXDELE;
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_UPDT,
+                     0, SPIFFS_PAGE_TO_PADDR(fs, objix_hdr_pix) + offsetof(spiffs_page_header, flags),
+                     sizeof(flags), &flags);
+    return res;
+}
+
+// validates the given look up entry
+static s32_t spiffs_lookup_check_validate(spiffs *fs, spiffs_obj_id lu_obj_id, spiffs_page_header *p_hdr,
+                                          spiffs_page_ix cur_pix, spiffs_block_ix cur_block, int cur_entry, int *reload_lu) {
+    (void)cur_block;
+    (void)cur_entry;
+    u8_t delete_page = 0;
+    s32_t res = SPIFFS_OK;
+    spiffs_page_ix objix_pix;
+    spiffs_page_ix ref_pix;
+    // check validity, take actions
+    if (((lu_obj_id == SPIFFS_OBJ_ID_DELETED) && (p_hdr->flags & SPIFFS_PH_FLAG_DELET)) ||
+            ((lu_obj_id == SPIFFS_OBJ_ID_FREE) && (p_hdr->flags & SPIFFS_PH_FLAG_USED) == 0)) {
+        // look up entry deleted / free but used in page header
+        SPIFFS_CHECK_DBG("LU: pix "_SPIPRIpg" deleted/free in lu but not on page\n", cur_pix);
+        *reload_lu = 1;
+        delete_page = 1;
+        if (p_hdr->flags & SPIFFS_PH_FLAG_INDEX) {
+            // header says data page
+            // data page can be removed if not referenced by some object index
+            res = spiffs_object_get_data_page_index_reference(fs, p_hdr->obj_id, p_hdr->span_ix, &ref_pix, &objix_pix);
+            if (res == SPIFFS_ERR_NOT_FOUND) {
+                // no object with this id, so remove page safely
+                res = SPIFFS_OK;
+            } else {
+                SPIFFS_CHECK_RES(res);
+                if (ref_pix == cur_pix) {
+                    // data page referenced by object index but deleted in lu
+                    // copy page to new place and re-write the object index to new place
+                    spiffs_page_ix new_pix;
+                    res = spiffs_rewrite_page(fs, cur_pix, p_hdr, &new_pix);
+                    SPIFFS_CHECK_DBG("LU: FIXUP: data page not found elsewhere, rewriting "_SPIPRIpg" to new page "_SPIPRIpg"\n", cur_pix, new_pix);
+                    SPIFFS_CHECK_RES(res);
+                    *reload_lu = 1;
+                    SPIFFS_CHECK_DBG("LU: FIXUP: "_SPIPRIpg" rewritten to "_SPIPRIpg", affected objix_pix "_SPIPRIpg"\n", cur_pix, new_pix, objix_pix);
+                    res = spiffs_rewrite_index(fs, p_hdr->obj_id, p_hdr->span_ix, new_pix, objix_pix);
+                    if (res <= _SPIFFS_ERR_CHECK_FIRST && res > _SPIFFS_ERR_CHECK_LAST) {
+                        // index bad also, cannot mend this file
+                        SPIFFS_CHECK_DBG("LU: FIXUP: index bad "_SPIPRIi", cannot mend!\n", res);
+                        res = spiffs_page_delete(fs, new_pix);
+                        SPIFFS_CHECK_RES(res);
+                        res = spiffs_delete_obj_lazy(fs, p_hdr->obj_id);
+                        CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_DELETE_BAD_FILE, p_hdr->obj_id, 0);
+                    } else {
+                        CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_FIX_INDEX, p_hdr->obj_id, p_hdr->span_ix);
+                    }
+                    SPIFFS_CHECK_RES(res);
+                }
+            }
+        } else {
+            // header says index page
+            // index page can be removed if other index with same obj_id and spanix is found
+            res = spiffs_obj_lu_find_id_and_span(fs, p_hdr->obj_id | SPIFFS_OBJ_ID_IX_FLAG, p_hdr->span_ix, cur_pix, 0);
+            if (res == SPIFFS_ERR_NOT_FOUND) {
+                // no such index page found, check for a data page amongst page headers
+                // lu cannot be trusted
+                res = spiffs_obj_lu_find_id_and_span_by_phdr(fs, p_hdr->obj_id | SPIFFS_OBJ_ID_IX_FLAG, 0, 0, 0);
+                if (res == SPIFFS_OK) { // ignore other errors
+                    // got a data page also, assume lu corruption only, rewrite to new page
+                    spiffs_page_ix new_pix;
+                    res = spiffs_rewrite_page(fs, cur_pix, p_hdr, &new_pix);
+                    SPIFFS_CHECK_DBG("LU: FIXUP: ix page with data not found elsewhere, rewriting "_SPIPRIpg" to new page "_SPIPRIpg"\n", cur_pix, new_pix);
+                    SPIFFS_CHECK_RES(res);
+                    *reload_lu = 1;
+                    CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_FIX_LOOKUP, p_hdr->obj_id, p_hdr->span_ix);
+                }
+            } else {
+                SPIFFS_CHECK_RES(res);
+            }
+        }
+    }
+    if (lu_obj_id != SPIFFS_OBJ_ID_FREE && lu_obj_id != SPIFFS_OBJ_ID_DELETED) {
+        // look up entry used
+        if ((p_hdr->obj_id | SPIFFS_OBJ_ID_IX_FLAG) != (lu_obj_id | SPIFFS_OBJ_ID_IX_FLAG)) {
+            SPIFFS_CHECK_DBG("LU: pix "_SPIPRIpg" differ in obj_id lu:"_SPIPRIid" ph:"_SPIPRIid"\n", cur_pix, lu_obj_id, p_hdr->obj_id);
+            delete_page = 1;
+            if ((p_hdr->flags & SPIFFS_PH_FLAG_DELET) == 0 ||
+                    (p_hdr->flags & SPIFFS_PH_FLAG_FINAL) ||
+                    (p_hdr->flags & (SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_IXDELE)) == 0) {
+                // page deleted or not finalized, just remove it
+            } else {
+                if (p_hdr->flags & SPIFFS_PH_FLAG_INDEX) {
+                    // if data page, check for reference to this page
+                    res = spiffs_object_get_data_page_index_reference(fs, p_hdr->obj_id, p_hdr->span_ix, &ref_pix, &objix_pix);
+                    if (res == SPIFFS_ERR_NOT_FOUND) {
+                        // no object with this id, so remove page safely
+                        res = SPIFFS_OK;
+                    } else {
+                        SPIFFS_CHECK_RES(res);
+                        //   if found, rewrite page with object id, update index, and delete current
+                        if (ref_pix == cur_pix) {
+                            spiffs_page_ix new_pix;
+                            res = spiffs_rewrite_page(fs, cur_pix, p_hdr, &new_pix);
+                            SPIFFS_CHECK_RES(res);
+                            res = spiffs_rewrite_index(fs, p_hdr->obj_id, p_hdr->span_ix, new_pix, objix_pix);
+                            if (res <= _SPIFFS_ERR_CHECK_FIRST && res > _SPIFFS_ERR_CHECK_LAST) {
+                                // index bad also, cannot mend this file
+                                SPIFFS_CHECK_DBG("LU: FIXUP: index bad "_SPIPRIi", cannot mend!\n", res);
+                                res = spiffs_page_delete(fs, new_pix);
+                                SPIFFS_CHECK_RES(res);
+                                res = spiffs_delete_obj_lazy(fs, p_hdr->obj_id);
+                                *reload_lu = 1;
+                                CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_DELETE_BAD_FILE, p_hdr->obj_id, 0);
+                            }
+                            SPIFFS_CHECK_RES(res);
+                        }
+                    }
+                } else {
+                    // else if index, check for other pages with both obj_id's and spanix
+                    spiffs_page_ix objix_pix_lu, objix_pix_ph;
+                    // see if other object index page exists for lookup obj id and span index
+                    res = spiffs_obj_lu_find_id_and_span(fs, lu_obj_id | SPIFFS_OBJ_ID_IX_FLAG, p_hdr->span_ix, 0, &objix_pix_lu);
+                    if (res == SPIFFS_ERR_NOT_FOUND) {
+                        res = SPIFFS_OK;
+                        objix_pix_lu = 0;
+                    }
+                    SPIFFS_CHECK_RES(res);
+                    // see if other object index exists for page header obj id and span index
+                    res = spiffs_obj_lu_find_id_and_span(fs, p_hdr->obj_id | SPIFFS_OBJ_ID_IX_FLAG, p_hdr->span_ix, 0, &objix_pix_ph);
+                    if (res == SPIFFS_ERR_NOT_FOUND) {
+                        res = SPIFFS_OK;
+                        objix_pix_ph = 0;
+                    }
+                    SPIFFS_CHECK_RES(res);
+                    //   if both obj_id's found, just delete current
+                    if (objix_pix_ph == 0 || objix_pix_lu == 0) {
+                        // otherwise try finding first corresponding data pages
+                        spiffs_page_ix data_pix_lu, data_pix_ph;
+                        // see if other data page exists for look up obj id and span index
+                        res = spiffs_obj_lu_find_id_and_span(fs, lu_obj_id & ~SPIFFS_OBJ_ID_IX_FLAG, 0, 0, &data_pix_lu);
+                        if (res == SPIFFS_ERR_NOT_FOUND) {
+                            res = SPIFFS_OK;
+                            objix_pix_lu = 0;
+                        }
+                        SPIFFS_CHECK_RES(res);
+                        // see if other data page exists for page header obj id and span index
+                        res = spiffs_obj_lu_find_id_and_span(fs, p_hdr->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG, 0, 0, &data_pix_ph);
+                        if (res == SPIFFS_ERR_NOT_FOUND) {
+                            res = SPIFFS_OK;
+                            objix_pix_ph = 0;
+                        }
+                        SPIFFS_CHECK_RES(res);
+
+                        spiffs_page_header new_ph;
+                        new_ph.flags = 0xff & ~(SPIFFS_PH_FLAG_USED | SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_FINAL);
+                        new_ph.span_ix = p_hdr->span_ix;
+                        spiffs_page_ix new_pix;
+                        if ((objix_pix_lu && data_pix_lu && data_pix_ph && objix_pix_ph == 0) ||
+                                (objix_pix_lu == 0 && data_pix_ph && objix_pix_ph == 0)) {
+                            //   got a data page for page header obj id
+                            //   rewrite as obj_id_ph
+                            new_ph.obj_id = p_hdr->obj_id | SPIFFS_OBJ_ID_IX_FLAG;
+                            res = spiffs_rewrite_page(fs, cur_pix, &new_ph, &new_pix);
+                            SPIFFS_CHECK_DBG("LU: FIXUP: rewrite page "_SPIPRIpg" as "_SPIPRIid" to pix "_SPIPRIpg"\n", cur_pix, new_ph.obj_id, new_pix);
+                            CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_FIX_LOOKUP, p_hdr->obj_id, p_hdr->span_ix);
+                            SPIFFS_CHECK_RES(res);
+                            *reload_lu = 1;
+                        } else if ((objix_pix_ph && data_pix_ph && data_pix_lu && objix_pix_lu == 0) ||
+                                   (objix_pix_ph == 0 && data_pix_lu && objix_pix_lu == 0)) {
+                            //   got a data page for look up obj id
+                            //   rewrite as obj_id_lu
+                            new_ph.obj_id =  lu_obj_id | SPIFFS_OBJ_ID_IX_FLAG;
+                            SPIFFS_CHECK_DBG("LU: FIXUP: rewrite page "_SPIPRIpg" as "_SPIPRIid"\n", cur_pix, new_ph.obj_id);
+                            CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_FIX_LOOKUP, p_hdr->obj_id, p_hdr->span_ix);
+                            res = spiffs_rewrite_page(fs, cur_pix, &new_ph, &new_pix);
+                            SPIFFS_CHECK_RES(res);
+                            *reload_lu = 1;
+                        } else {
+                            // cannot safely do anything
+                            SPIFFS_CHECK_DBG("LU: FIXUP: nothing to do, just delete\n", NULL);
+                        }
+                    }
+                }
+            }
+        } else if (((lu_obj_id & SPIFFS_OBJ_ID_IX_FLAG) && (p_hdr->flags & SPIFFS_PH_FLAG_INDEX)) ||
+                   ((lu_obj_id & SPIFFS_OBJ_ID_IX_FLAG) == 0 && (p_hdr->flags & SPIFFS_PH_FLAG_INDEX) == 0)) {
+            SPIFFS_CHECK_DBG("LU: "_SPIPRIpg" lu/page index marking differ\n", cur_pix);
+            spiffs_page_ix data_pix, objix_pix_d;
+            // see if other data page exists for given obj id and span index
+            res = spiffs_obj_lu_find_id_and_span(fs, lu_obj_id & ~SPIFFS_OBJ_ID_IX_FLAG, p_hdr->span_ix, cur_pix, &data_pix);
+            if (res == SPIFFS_ERR_NOT_FOUND) {
+                res = SPIFFS_OK;
+                data_pix = 0;
+            }
+            SPIFFS_CHECK_RES(res);
+            // see if other object index exists for given obj id and span index
+            res = spiffs_obj_lu_find_id_and_span(fs, lu_obj_id | SPIFFS_OBJ_ID_IX_FLAG, p_hdr->span_ix, cur_pix, &objix_pix_d);
+            if (res == SPIFFS_ERR_NOT_FOUND) {
+                res = SPIFFS_OK;
+                objix_pix_d = 0;
+            }
+            SPIFFS_CHECK_RES(res);
+
+            delete_page = 1;
+            // if other data page exists and object index exists, just delete page
+            if (data_pix && objix_pix_d) {
+                SPIFFS_CHECK_DBG("LU: FIXUP: other index and data page exists, simply remove\n", NULL);
+            } else
+                // if only data page exists, make this page index
+                if (data_pix && objix_pix_d == 0) {
+                    SPIFFS_CHECK_DBG("LU: FIXUP: other data page exists, make this index\n", NULL);
+                    CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_FIX_INDEX, lu_obj_id, p_hdr->span_ix);
+                    spiffs_page_header new_ph;
+                    spiffs_page_ix new_pix;
+                    new_ph.flags = 0xff & ~(SPIFFS_PH_FLAG_USED | SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_INDEX);
+                    new_ph.obj_id = lu_obj_id | SPIFFS_OBJ_ID_IX_FLAG;
+                    new_ph.span_ix = p_hdr->span_ix;
+                    res = spiffs_page_allocate_data(fs, new_ph.obj_id, &new_ph, 0, 0, 0, 1, &new_pix);
+                    SPIFFS_CHECK_RES(res);
+                    res = spiffs_phys_cpy(fs, 0, SPIFFS_PAGE_TO_PADDR(fs, new_pix) + sizeof(spiffs_page_header),
+                                          SPIFFS_PAGE_TO_PADDR(fs, cur_pix) + sizeof(spiffs_page_header),
+                                          SPIFFS_CFG_LOG_PAGE_SZ(fs) - sizeof(spiffs_page_header));
+                    SPIFFS_CHECK_RES(res);
+                } else
+                    // if only index exists, make data page
+                    if (data_pix == 0 && objix_pix_d) {
+                        SPIFFS_CHECK_DBG("LU: FIXUP: other index page exists, make this data\n", NULL);
+                        CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_FIX_LOOKUP, lu_obj_id, p_hdr->span_ix);
+                        spiffs_page_header new_ph;
+                        spiffs_page_ix new_pix;
+                        new_ph.flags = 0xff & ~(SPIFFS_PH_FLAG_USED | SPIFFS_PH_FLAG_FINAL);
+                        new_ph.obj_id = lu_obj_id & ~SPIFFS_OBJ_ID_IX_FLAG;
+                        new_ph.span_ix = p_hdr->span_ix;
+                        res = spiffs_page_allocate_data(fs, new_ph.obj_id, &new_ph, 0, 0, 0, 1, &new_pix);
+                        SPIFFS_CHECK_RES(res);
+                        res = spiffs_phys_cpy(fs, 0, SPIFFS_PAGE_TO_PADDR(fs, new_pix) + sizeof(spiffs_page_header),
+                                              SPIFFS_PAGE_TO_PADDR(fs, cur_pix) + sizeof(spiffs_page_header),
+                                              SPIFFS_CFG_LOG_PAGE_SZ(fs) - sizeof(spiffs_page_header));
+                        SPIFFS_CHECK_RES(res);
+                    } else {
+                        // if nothing exists, we cannot safely make a decision - delete
+                    }
+        } else if ((p_hdr->flags & SPIFFS_PH_FLAG_DELET) == 0) {
+            SPIFFS_CHECK_DBG("LU: pix "_SPIPRIpg" busy in lu but deleted on page\n", cur_pix);
+            delete_page = 1;
+        } else if ((p_hdr->flags & SPIFFS_PH_FLAG_FINAL)) {
+            SPIFFS_CHECK_DBG("LU: pix "_SPIPRIpg" busy but not final\n", cur_pix);
+            // page can be removed if not referenced by object index
+            *reload_lu = 1;
+            res = spiffs_object_get_data_page_index_reference(fs, lu_obj_id, p_hdr->span_ix, &ref_pix, &objix_pix);
+            if (res == SPIFFS_ERR_NOT_FOUND) {
+                // no object with this id, so remove page safely
+                res = SPIFFS_OK;
+                delete_page = 1;
+            } else {
+                SPIFFS_CHECK_RES(res);
+                if (ref_pix != cur_pix) {
+                    SPIFFS_CHECK_DBG("LU: FIXUP: other finalized page is referred, just delete\n", NULL);
+                    delete_page = 1;
+                } else {
+                    // page referenced by object index but not final
+                    // just finalize
+                    SPIFFS_CHECK_DBG("LU: FIXUP: unfinalized page is referred, finalizing\n", NULL);
+                    CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_FIX_LOOKUP, p_hdr->obj_id, p_hdr->span_ix);
+                    u8_t flags = 0xff;
+#if SPIFFS_NO_BLIND_WRITES
+                    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_READ,
+                                     0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix) + offsetof(spiffs_page_header, flags),
+                                     sizeof(flags), &flags);
+                    SPIFFS_CHECK_RES(res);
+#endif
+                    flags &= ~SPIFFS_PH_FLAG_FINAL;
+                    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                                     0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix) + offsetof(spiffs_page_header, flags),
+                                     sizeof(flags), &flags);
+                }
+            }
+        }
+    }
+
+    if (delete_page) {
+        SPIFFS_CHECK_DBG("LU: FIXUP: deleting page "_SPIPRIpg"\n", cur_pix);
+        CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_DELETE_PAGE, cur_pix, 0);
+        res = spiffs_page_delete(fs, cur_pix);
+        SPIFFS_CHECK_RES(res);
+    }
+
+    return res;
+}
+
+static s32_t spiffs_lookup_check_v(spiffs *fs, spiffs_obj_id obj_id, spiffs_block_ix cur_block, int cur_entry,
+                                   const void *user_const_p, void *user_var_p) {
+    (void)user_const_p;
+    (void)user_var_p;
+    s32_t res = SPIFFS_OK;
+    spiffs_page_header p_hdr;
+    spiffs_page_ix cur_pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, cur_block, cur_entry);
+
+    CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_PROGRESS,
+             (cur_block * 256) / fs->block_count, 0);
+
+    // load header
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                     0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix), sizeof(spiffs_page_header), (u8_t *)&p_hdr);
+    SPIFFS_CHECK_RES(res);
+
+    int reload_lu = 0;
+
+    res = spiffs_lookup_check_validate(fs, obj_id, &p_hdr, cur_pix, cur_block, cur_entry, &reload_lu);
+    SPIFFS_CHECK_RES(res);
+
+    if (res == SPIFFS_OK) {
+        return reload_lu ? SPIFFS_VIS_COUNTINUE_RELOAD : SPIFFS_VIS_COUNTINUE;
+    }
+    return res;
+}
+
+
+// Scans all object look up. For each entry, corresponding page header is checked for validity.
+// If an object index header page is found, this is also checked
+s32_t spiffs_lookup_consistency_check(spiffs *fs, u8_t check_all_objects) {
+    (void)check_all_objects;
+    s32_t res = SPIFFS_OK;
+
+    CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_PROGRESS, 0, 0);
+
+    res = spiffs_obj_lu_find_entry_visitor(fs, 0, 0, 0, 0, spiffs_lookup_check_v, 0, 0, 0, 0);
+
+    if (res == SPIFFS_VIS_END) {
+        res = SPIFFS_OK;
+    }
+
+    if (res != SPIFFS_OK) {
+        CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_ERROR, res, 0);
+    }
+
+    CHECK_CB(fs, SPIFFS_CHECK_LOOKUP, SPIFFS_CHECK_PROGRESS, 256, 0);
+
+    return res;
+}
+
+//---------------------------------------
+// Page consistency
+
+// Scans all pages (except lu pages), reserves 4 bits in working memory for each page
+// bit 0: 0 == FREE|DELETED, 1 == USED
+// bit 1: 0 == UNREFERENCED, 1 == REFERENCED
+// bit 2: 0 == NOT_INDEX,    1 == INDEX
+// bit 3: unused
+// A consistent file system will have only pages being
+//  * x000 free, unreferenced, not index
+//  * x011 used, referenced only once, not index
+//  * x101 used, unreferenced, index
+// The working memory might not fit all pages so several scans might be needed
+static s32_t spiffs_page_consistency_check_i(spiffs *fs) {
+    const u32_t bits = 4;
+    const spiffs_page_ix pages_per_scan = SPIFFS_CFG_LOG_PAGE_SZ(fs) * 8 / bits;
+
+    s32_t res = SPIFFS_OK;
+    spiffs_page_ix pix_offset = 0;
+
+    // for each range of pages fitting into work memory
+    while (pix_offset < SPIFFS_PAGES_PER_BLOCK(fs) * fs->block_count) {
+        // set this flag to abort all checks and rescan the page range
+        u8_t restart = 0;
+        memset(fs->work, 0, SPIFFS_CFG_LOG_PAGE_SZ(fs));
+
+        spiffs_block_ix cur_block = 0;
+        // build consistency bitmap for id range traversing all blocks
+        while (!restart && cur_block < fs->block_count) {
+            CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_PROGRESS,
+                     (pix_offset * 256) / (SPIFFS_PAGES_PER_BLOCK(fs) * fs->block_count) +
+                     ((((cur_block * pages_per_scan * 256) / (SPIFFS_PAGES_PER_BLOCK(fs) * fs->block_count))) / fs->block_count),
+                     0);
+            // traverse each page except for lookup pages
+            spiffs_page_ix cur_pix = SPIFFS_OBJ_LOOKUP_PAGES(fs) + SPIFFS_PAGES_PER_BLOCK(fs) * cur_block;
+            while (!restart && cur_pix < SPIFFS_PAGES_PER_BLOCK(fs) * (cur_block + 1)) {
+                //if ((cur_pix & 0xff) == 0)
+                //  SPIFFS_CHECK_DBG("PA: processing pix "_SPIPRIpg", block "_SPIPRIbl" of pix "_SPIPRIpg", block "_SPIPRIbl"\n",
+                //      cur_pix, cur_block, SPIFFS_PAGES_PER_BLOCK(fs) * fs->block_count, fs->block_count);
+
+                // read header
+                spiffs_page_header p_hdr;
+                res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                                 0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix), sizeof(spiffs_page_header), (u8_t *)&p_hdr);
+                SPIFFS_CHECK_RES(res);
+
+                u8_t within_range = (cur_pix >= pix_offset && cur_pix < pix_offset + pages_per_scan);
+                const u32_t pix_byte_ix = (cur_pix - pix_offset) / (8 / bits);
+                const u8_t pix_bit_ix = (cur_pix & ((8 / bits) - 1)) * bits;
+
+                if (within_range &&
+                        (p_hdr.flags & SPIFFS_PH_FLAG_DELET) && (p_hdr.flags & SPIFFS_PH_FLAG_USED) == 0) {
+                    // used
+                    fs->work[pix_byte_ix] |= (1 << (pix_bit_ix + 0));
+                }
+                if ((p_hdr.flags & SPIFFS_PH_FLAG_DELET) &&
+                        (p_hdr.flags & SPIFFS_PH_FLAG_IXDELE) &&
+                        (p_hdr.flags & (SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_USED)) == 0) {
+                    // found non-deleted index
+                    if (within_range) {
+                        fs->work[pix_byte_ix] |= (1 << (pix_bit_ix + 2));
+                    }
+
+                    // load non-deleted index
+                    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                                     0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+                    SPIFFS_CHECK_RES(res);
+
+                    // traverse index for referenced pages
+                    spiffs_page_ix *object_page_index;
+                    spiffs_page_header *objix_p_hdr = (spiffs_page_header *)fs->lu_work;
+
+                    int entries;
+                    int i;
+                    spiffs_span_ix data_spix_offset;
+                    if (p_hdr.span_ix == 0) {
+                        // object header page index
+                        entries = SPIFFS_OBJ_HDR_IX_LEN(fs);
+                        data_spix_offset = 0;
+                        object_page_index = (spiffs_page_ix *)((u8_t *)fs->lu_work + sizeof(spiffs_page_object_ix_header));
+                    } else {
+                        // object page index
+                        entries = SPIFFS_OBJ_IX_LEN(fs);
+                        data_spix_offset = SPIFFS_OBJ_HDR_IX_LEN(fs) + SPIFFS_OBJ_IX_LEN(fs) * (p_hdr.span_ix - 1);
+                        object_page_index = (spiffs_page_ix *)((u8_t *)fs->lu_work + sizeof(spiffs_page_object_ix));
+                    }
+
+                    // for all entries in index
+                    for (i = 0; !restart && i < entries; i++) {
+                        spiffs_page_ix rpix = object_page_index[i];
+                        u8_t rpix_within_range = rpix >= pix_offset && rpix < pix_offset + pages_per_scan;
+
+                        if ((rpix != (spiffs_page_ix) - 1 && rpix > SPIFFS_MAX_PAGES(fs))
+                                || (rpix_within_range && SPIFFS_IS_LOOKUP_PAGE(fs, rpix))) {
+
+                            // bad reference
+                            SPIFFS_CHECK_DBG("PA: pix "_SPIPRIpg"x bad pix / LU referenced from page "_SPIPRIpg"\n",
+                                             rpix, cur_pix);
+                            // check for data page elsewhere
+                            spiffs_page_ix data_pix;
+                            res = spiffs_obj_lu_find_id_and_span(fs, objix_p_hdr->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG,
+                                                                 data_spix_offset + i, 0, &data_pix);
+                            if (res == SPIFFS_ERR_NOT_FOUND) {
+                                res = SPIFFS_OK;
+                                data_pix = 0;
+                            }
+                            SPIFFS_CHECK_RES(res);
+                            if (data_pix == 0) {
+                                // if not, allocate free page
+                                spiffs_page_header new_ph;
+                                new_ph.flags = 0xff & ~(SPIFFS_PH_FLAG_USED | SPIFFS_PH_FLAG_FINAL);
+                                new_ph.obj_id = objix_p_hdr->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG;
+                                new_ph.span_ix = data_spix_offset + i;
+                                res = spiffs_page_allocate_data(fs, new_ph.obj_id, &new_ph, 0, 0, 0, 1, &data_pix);
+                                SPIFFS_CHECK_RES(res);
+                                SPIFFS_CHECK_DBG("PA: FIXUP: found no existing data page, created new @ "_SPIPRIpg"\n", data_pix);
+                            }
+                            // remap index
+                            SPIFFS_CHECK_DBG("PA: FIXUP: rewriting index pix "_SPIPRIpg"\n", cur_pix);
+                            res = spiffs_rewrite_index(fs, objix_p_hdr->obj_id | SPIFFS_OBJ_ID_IX_FLAG,
+                                                       data_spix_offset + i, data_pix, cur_pix);
+                            if (res <= _SPIFFS_ERR_CHECK_FIRST && res > _SPIFFS_ERR_CHECK_LAST) {
+                                // index bad also, cannot mend this file
+                                SPIFFS_CHECK_DBG("PA: FIXUP: index bad "_SPIPRIi", cannot mend - delete object\n", res);
+                                CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_DELETE_BAD_FILE, objix_p_hdr->obj_id, 0);
+                                // delete file
+                                res = spiffs_page_delete(fs, cur_pix);
+                            } else {
+                                CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_FIX_INDEX, objix_p_hdr->obj_id, objix_p_hdr->span_ix);
+                            }
+                            SPIFFS_CHECK_RES(res);
+                            restart = 1;
+
+                        } else if (rpix_within_range) {
+
+                            // valid reference
+                            // read referenced page header
+                            spiffs_page_header rp_hdr;
+                            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                                             0, SPIFFS_PAGE_TO_PADDR(fs, rpix), sizeof(spiffs_page_header), (u8_t *)&rp_hdr);
+                            SPIFFS_CHECK_RES(res);
+
+                            // cross reference page header check
+                            if (rp_hdr.obj_id != (p_hdr.obj_id & ~SPIFFS_OBJ_ID_IX_FLAG) ||
+                                    rp_hdr.span_ix != data_spix_offset + i ||
+                                    (rp_hdr.flags & (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_USED)) !=
+                                    (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_INDEX)) {
+                                SPIFFS_CHECK_DBG("PA: pix "_SPIPRIpg" has inconsistent page header ix id/span:"_SPIPRIid"/"_SPIPRIsp", ref id/span:"_SPIPRIid"/"_SPIPRIsp" flags:"_SPIPRIfl"\n",
+                                                 rpix, p_hdr.obj_id & ~SPIFFS_OBJ_ID_IX_FLAG, data_spix_offset + i,
+                                                 rp_hdr.obj_id, rp_hdr.span_ix, rp_hdr.flags);
+                                // try finding correct page
+                                spiffs_page_ix data_pix;
+                                res = spiffs_obj_lu_find_id_and_span(fs, p_hdr.obj_id & ~SPIFFS_OBJ_ID_IX_FLAG,
+                                                                     data_spix_offset + i, rpix, &data_pix);
+                                if (res == SPIFFS_ERR_NOT_FOUND) {
+                                    res = SPIFFS_OK;
+                                    data_pix = 0;
+                                }
+                                SPIFFS_CHECK_RES(res);
+                                if (data_pix == 0) {
+                                    // not found, this index is badly borked
+                                    SPIFFS_CHECK_DBG("PA: FIXUP: index bad, delete object id "_SPIPRIid"\n", p_hdr.obj_id);
+                                    CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_DELETE_BAD_FILE, p_hdr.obj_id, 0);
+                                    res = spiffs_delete_obj_lazy(fs, p_hdr.obj_id);
+                                    SPIFFS_CHECK_RES(res);
+                                    break;
+                                } else {
+                                    // found it, so rewrite index
+                                    SPIFFS_CHECK_DBG("PA: FIXUP: found correct data pix "_SPIPRIpg", rewrite ix pix "_SPIPRIpg" id "_SPIPRIid"\n",
+                                                     data_pix, cur_pix, p_hdr.obj_id);
+                                    res = spiffs_rewrite_index(fs, p_hdr.obj_id, data_spix_offset + i, data_pix, cur_pix);
+                                    if (res <= _SPIFFS_ERR_CHECK_FIRST && res > _SPIFFS_ERR_CHECK_LAST) {
+                                        // index bad also, cannot mend this file
+                                        SPIFFS_CHECK_DBG("PA: FIXUP: index bad "_SPIPRIi", cannot mend!\n", res);
+                                        CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_DELETE_BAD_FILE, p_hdr.obj_id, 0);
+                                        res = spiffs_delete_obj_lazy(fs, p_hdr.obj_id);
+                                    } else {
+                                        CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_FIX_INDEX, p_hdr.obj_id, p_hdr.span_ix);
+                                    }
+                                    SPIFFS_CHECK_RES(res);
+                                    restart = 1;
+                                }
+                            } else {
+                                // mark rpix as referenced
+                                const u32_t rpix_byte_ix = (rpix - pix_offset) / (8 / bits);
+                                const u8_t rpix_bit_ix = (rpix & ((8 / bits) - 1)) * bits;
+                                if (fs->work[rpix_byte_ix] & (1 << (rpix_bit_ix + 1))) {
+                                    SPIFFS_CHECK_DBG("PA: pix "_SPIPRIpg" multiple referenced from page "_SPIPRIpg"\n",
+                                                     rpix, cur_pix);
+                                    // Here, we should have fixed all broken references - getting this means there
+                                    // must be multiple files with same object id. Only solution is to delete
+                                    // the object which is referring to this page
+                                    SPIFFS_CHECK_DBG("PA: FIXUP: removing object "_SPIPRIid" and page "_SPIPRIpg"\n",
+                                                     p_hdr.obj_id, cur_pix);
+                                    CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_DELETE_BAD_FILE, p_hdr.obj_id, 0);
+                                    res = spiffs_delete_obj_lazy(fs, p_hdr.obj_id);
+                                    SPIFFS_CHECK_RES(res);
+                                    // extra precaution, delete this page also
+                                    res = spiffs_page_delete(fs, cur_pix);
+                                    SPIFFS_CHECK_RES(res);
+                                    restart = 1;
+                                }
+                                fs->work[rpix_byte_ix] |= (1 << (rpix_bit_ix + 1));
+                            }
+                        }
+                    } // for all index entries
+                } // found index
+
+                // next page
+                cur_pix++;
+            }
+            // next block
+            cur_block++;
+        }
+        // check consistency bitmap
+        if (!restart) {
+            spiffs_page_ix objix_pix;
+            spiffs_page_ix rpix;
+
+            u32_t byte_ix;
+            u8_t bit_ix;
+            for (byte_ix = 0; !restart && byte_ix < SPIFFS_CFG_LOG_PAGE_SZ(fs); byte_ix++) {
+                for (bit_ix = 0; !restart && bit_ix < 8 / bits; bit_ix ++) {
+                    u8_t bitmask = (fs->work[byte_ix] >> (bit_ix * bits)) & 0x7;
+                    spiffs_page_ix cur_pix = pix_offset + byte_ix * (8 / bits) + bit_ix;
+
+                    // 000 ok - free, unreferenced, not index
+
+                    if (bitmask == 0x1) {
+
+                        // 001
+                        SPIFFS_CHECK_DBG("PA: pix "_SPIPRIpg" USED, UNREFERENCED, not index\n", cur_pix);
+
+                        u8_t rewrite_ix_to_this = 0;
+                        u8_t delete_page = 0;
+                        // check corresponding object index entry
+                        spiffs_page_header p_hdr;
+                        res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                                         0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix), sizeof(spiffs_page_header), (u8_t *)&p_hdr);
+                        SPIFFS_CHECK_RES(res);
+
+                        res = spiffs_object_get_data_page_index_reference(fs, p_hdr.obj_id, p_hdr.span_ix,
+                                                                          &rpix, &objix_pix);
+                        if (res == SPIFFS_OK) {
+                            if (((rpix == (spiffs_page_ix) - 1 || rpix > SPIFFS_MAX_PAGES(fs)) || (SPIFFS_IS_LOOKUP_PAGE(fs, rpix)))) {
+                                // pointing to a bad page altogether, rewrite index to this
+                                rewrite_ix_to_this = 1;
+                                SPIFFS_CHECK_DBG("PA: corresponding ref is bad: "_SPIPRIpg", rewrite to this "_SPIPRIpg"\n", rpix, cur_pix);
+                            } else {
+                                // pointing to something else, check what
+                                spiffs_page_header rp_hdr;
+                                res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                                                 0, SPIFFS_PAGE_TO_PADDR(fs, rpix), sizeof(spiffs_page_header), (u8_t *)&rp_hdr);
+                                SPIFFS_CHECK_RES(res);
+                                if (((p_hdr.obj_id & ~SPIFFS_OBJ_ID_IX_FLAG) == rp_hdr.obj_id) &&
+                                        ((rp_hdr.flags & (SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_USED | SPIFFS_PH_FLAG_FINAL)) ==
+                                         (SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_DELET))) {
+                                    // pointing to something else valid, just delete this page then
+                                    SPIFFS_CHECK_DBG("PA: corresponding ref is good but different: "_SPIPRIpg", delete this "_SPIPRIpg"\n", rpix, cur_pix);
+                                    delete_page = 1;
+                                } else {
+                                    // pointing to something weird, update index to point to this page instead
+                                    if (rpix != cur_pix) {
+                                        SPIFFS_CHECK_DBG("PA: corresponding ref is weird: "_SPIPRIpg" %s%s%s%s, rewrite this "_SPIPRIpg"\n", rpix,
+                                                         (rp_hdr.flags & SPIFFS_PH_FLAG_INDEX) ? "" : "INDEX ",
+                                                         (rp_hdr.flags & SPIFFS_PH_FLAG_DELET) ? "" : "DELETED ",
+                                                         (rp_hdr.flags & SPIFFS_PH_FLAG_USED) ? "NOTUSED " : "",
+                                                         (rp_hdr.flags & SPIFFS_PH_FLAG_FINAL) ? "NOTFINAL " : "",
+                                                         cur_pix);
+                                        rewrite_ix_to_this = 1;
+                                    } else {
+                                        // should not happen, destined for fubar
+                                    }
+                                }
+                            }
+                        } else if (res == SPIFFS_ERR_NOT_FOUND) {
+                            SPIFFS_CHECK_DBG("PA: corresponding ref not found, delete "_SPIPRIpg"\n", cur_pix);
+                            delete_page = 1;
+                            res = SPIFFS_OK;
+                        }
+
+                        if (rewrite_ix_to_this) {
+                            // if pointing to invalid page, redirect index to this page
+                            SPIFFS_CHECK_DBG("PA: FIXUP: rewrite index id "_SPIPRIid" data spix "_SPIPRIsp" to point to this pix: "_SPIPRIpg"\n",
+                                             p_hdr.obj_id, p_hdr.span_ix, cur_pix);
+                            res = spiffs_rewrite_index(fs, p_hdr.obj_id, p_hdr.span_ix, cur_pix, objix_pix);
+                            if (res <= _SPIFFS_ERR_CHECK_FIRST && res > _SPIFFS_ERR_CHECK_LAST) {
+                                // index bad also, cannot mend this file
+                                SPIFFS_CHECK_DBG("PA: FIXUP: index bad "_SPIPRIi", cannot mend!\n", res);
+                                CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_DELETE_BAD_FILE, p_hdr.obj_id, 0);
+                                res = spiffs_page_delete(fs, cur_pix);
+                                SPIFFS_CHECK_RES(res);
+                                res = spiffs_delete_obj_lazy(fs, p_hdr.obj_id);
+                            } else {
+                                CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_FIX_INDEX, p_hdr.obj_id, p_hdr.span_ix);
+                            }
+                            SPIFFS_CHECK_RES(res);
+                            restart = 1;
+                            continue;
+                        } else if (delete_page) {
+                            SPIFFS_CHECK_DBG("PA: FIXUP: deleting page "_SPIPRIpg"\n", cur_pix);
+                            CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_DELETE_PAGE, cur_pix, 0);
+                            res = spiffs_page_delete(fs, cur_pix);
+                        }
+                        SPIFFS_CHECK_RES(res);
+                    }
+                    if (bitmask == 0x2) {
+
+                        // 010
+                        SPIFFS_CHECK_DBG("PA: pix "_SPIPRIpg" FREE, REFERENCED, not index\n", cur_pix);
+
+                        // no op, this should be taken care of when checking valid references
+                    }
+
+                    // 011 ok - busy, referenced, not index
+
+                    if (bitmask == 0x4) {
+
+                        // 100
+                        SPIFFS_CHECK_DBG("PA: pix "_SPIPRIpg" FREE, unreferenced, INDEX\n", cur_pix);
+
+                        // this should never happen, major fubar
+                    }
+
+                    // 101 ok - busy, unreferenced, index
+
+                    if (bitmask == 0x6) {
+
+                        // 110
+                        SPIFFS_CHECK_DBG("PA: pix "_SPIPRIpg" FREE, REFERENCED, INDEX\n", cur_pix);
+
+                        // no op, this should be taken care of when checking valid references
+                    }
+                    if (bitmask == 0x7) {
+
+                        // 111
+                        SPIFFS_CHECK_DBG("PA: pix "_SPIPRIpg" USED, REFERENCED, INDEX\n", cur_pix);
+
+                        // no op, this should be taken care of when checking valid references
+                    }
+                }
+            }
+        }
+
+        SPIFFS_CHECK_DBG("PA: processed "_SPIPRIpg", restart "_SPIPRIi"\n", pix_offset, restart);
+        // next page range
+        if (!restart) {
+            pix_offset += pages_per_scan;
+        }
+    } // while page range not reached end
+    return res;
+}
+
+// Checks consistency amongst all pages and fixes irregularities
+s32_t spiffs_page_consistency_check(spiffs *fs) {
+    CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_PROGRESS, 0, 0);
+    s32_t res = spiffs_page_consistency_check_i(fs);
+    if (res != SPIFFS_OK) {
+        CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_ERROR, res, 0);
+    }
+    CHECK_CB(fs, SPIFFS_CHECK_PAGE, SPIFFS_CHECK_PROGRESS, 256, 0);
+    return res;
+}
+
+//---------------------------------------
+// Object index consistency
+
+// searches for given object id in temporary object id index,
+// returns the index or -1
+static int spiffs_object_index_search(spiffs *fs, spiffs_obj_id obj_id) {
+    u32_t i;
+    spiffs_obj_id *obj_table = (spiffs_obj_id *)fs->work;
+    obj_id &= ~SPIFFS_OBJ_ID_IX_FLAG;
+    for (i = 0; i < SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(spiffs_obj_id); i++) {
+        if ((obj_table[i] & ~SPIFFS_OBJ_ID_IX_FLAG) == obj_id) {
+            return i;
+        }
+    }
+    return -1;
+}
+
+static s32_t spiffs_object_index_consistency_check_v(spiffs *fs, spiffs_obj_id obj_id, spiffs_block_ix cur_block,
+                                                     int cur_entry, const void *user_const_p, void *user_var_p) {
+    (void)user_const_p;
+    s32_t res_c = SPIFFS_VIS_COUNTINUE;
+    s32_t res = SPIFFS_OK;
+    u32_t *log_ix = (u32_t *)user_var_p;
+    spiffs_obj_id *obj_table = (spiffs_obj_id *)fs->work;
+
+    CHECK_CB(fs, SPIFFS_CHECK_INDEX, SPIFFS_CHECK_PROGRESS,
+             (cur_block * 256) / fs->block_count, 0);
+
+    if (obj_id != SPIFFS_OBJ_ID_FREE && obj_id != SPIFFS_OBJ_ID_DELETED && (obj_id & SPIFFS_OBJ_ID_IX_FLAG)) {
+        spiffs_page_header p_hdr;
+        spiffs_page_ix cur_pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, cur_block, cur_entry);
+
+        // load header
+        res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                         0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix), sizeof(spiffs_page_header), (u8_t *)&p_hdr);
+        SPIFFS_CHECK_RES(res);
+
+        if (p_hdr.span_ix == 0 &&
+                (p_hdr.flags & (SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_IXDELE)) ==
+                (SPIFFS_PH_FLAG_DELET)) {
+            SPIFFS_CHECK_DBG("IX: pix "_SPIPRIpg", obj id:"_SPIPRIid" spix:"_SPIPRIsp" header not fully deleted - deleting\n",
+                             cur_pix, obj_id, p_hdr.span_ix);
+            CHECK_CB(fs, SPIFFS_CHECK_INDEX, SPIFFS_CHECK_DELETE_PAGE, cur_pix, obj_id);
+            res = spiffs_page_delete(fs, cur_pix);
+            SPIFFS_CHECK_RES(res);
+            return res_c;
+        }
+
+        if ((p_hdr.flags & (SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_IXDELE)) ==
+                (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_IXDELE)) {
+            return res_c;
+        }
+
+        if (p_hdr.span_ix == 0) {
+            // objix header page, register objid as reachable
+            int r = spiffs_object_index_search(fs, obj_id);
+            if (r == -1) {
+                // not registered, do it
+                obj_table[*log_ix] = obj_id & ~SPIFFS_OBJ_ID_IX_FLAG;
+                (*log_ix)++;
+                if (*log_ix >= SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(spiffs_obj_id)) {
+                    *log_ix = 0;
+                }
+            }
+        } else { // span index
+            // objix page, see if header can be found
+            int r = spiffs_object_index_search(fs, obj_id);
+            u8_t delete = 0;
+            if (r == -1) {
+                // not in temporary index, try finding it
+                spiffs_page_ix objix_hdr_pix;
+                res = spiffs_obj_lu_find_id_and_span(fs, obj_id | SPIFFS_OBJ_ID_IX_FLAG, 0, 0, &objix_hdr_pix);
+                res_c = SPIFFS_VIS_COUNTINUE_RELOAD;
+                if (res == SPIFFS_OK) {
+                    // found, register as reachable
+                    obj_table[*log_ix] = obj_id & ~SPIFFS_OBJ_ID_IX_FLAG;
+                } else if (res == SPIFFS_ERR_NOT_FOUND) {
+                    // not found, register as unreachable
+                    delete = 1;
+                    obj_table[*log_ix] = obj_id | SPIFFS_OBJ_ID_IX_FLAG;
+                } else {
+                    SPIFFS_CHECK_RES(res);
+                }
+                (*log_ix)++;
+                if (*log_ix >= SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(spiffs_obj_id)) {
+                    *log_ix = 0;
+                }
+            } else {
+                // in temporary index, check reachable flag
+                if ((obj_table[r] & SPIFFS_OBJ_ID_IX_FLAG)) {
+                    // registered as unreachable
+                    delete = 1;
+                }
+            }
+
+            if (delete) {
+                SPIFFS_CHECK_DBG("IX: FIXUP: pix "_SPIPRIpg", obj id:"_SPIPRIid" spix:"_SPIPRIsp" is orphan index - deleting\n",
+                                 cur_pix, obj_id, p_hdr.span_ix);
+                CHECK_CB(fs, SPIFFS_CHECK_INDEX, SPIFFS_CHECK_DELETE_ORPHANED_INDEX, cur_pix, obj_id);
+                res = spiffs_page_delete(fs, cur_pix);
+                SPIFFS_CHECK_RES(res);
+            }
+        } // span index
+    } // valid object index id
+
+    return res_c;
+}
+
+// Removes orphaned and partially deleted index pages.
+// Scans for index pages. When an index page is found, corresponding index header is searched for.
+// If no such page exists, the index page cannot be reached as no index header exists and must be
+// deleted.
+s32_t spiffs_object_index_consistency_check(spiffs *fs) {
+    s32_t res = SPIFFS_OK;
+    // impl note:
+    // fs->work is used for a temporary object index memory, listing found object ids and
+    // indicating whether they can be reached or not. Acting as a fifo if object ids cannot fit.
+    // In the temporary object index memory, SPIFFS_OBJ_ID_IX_FLAG bit is used to indicate
+    // a reachable/unreachable object id.
+    memset(fs->work, 0, SPIFFS_CFG_LOG_PAGE_SZ(fs));
+    u32_t obj_id_log_ix = 0;
+    CHECK_CB(fs, SPIFFS_CHECK_INDEX, SPIFFS_CHECK_PROGRESS, 0, 0);
+    res = spiffs_obj_lu_find_entry_visitor(fs, 0, 0, 0, 0, spiffs_object_index_consistency_check_v, 0, &obj_id_log_ix,
+                                           0, 0);
+    if (res == SPIFFS_VIS_END) {
+        res = SPIFFS_OK;
+    }
+    if (res != SPIFFS_OK) {
+        CHECK_CB(fs, SPIFFS_CHECK_INDEX, SPIFFS_CHECK_ERROR, res, 0);
+    }
+    CHECK_CB(fs, SPIFFS_CHECK_INDEX, SPIFFS_CHECK_PROGRESS, 256, 0);
+    return res;
+}
+
+#endif // !SPIFFS_READ_ONLY
diff --git a/armsrc/spiffs_config.h b/armsrc/spiffs_config.h
new file mode 100644
index 000000000..99557f41f
--- /dev/null
+++ b/armsrc/spiffs_config.h
@@ -0,0 +1,388 @@
+/*
+ * spiffs_config.h
+ *
+ *  Created on: Jul 3, 2013
+ *      Author: petera
+ */
+
+#ifndef SPIFFS_CONFIG_H_
+#define SPIFFS_CONFIG_H_
+
+// ----------- 8< ------------
+// Following includes are for the linux test build of spiffs
+// These may/should/must be removed/altered/replaced in your target
+//#include <stdio.h>
+//#include <stdlib.h>
+//
+#include "printf.h"
+#include "string.h"
+#include "flashmem.h"
+
+void Dbprintf(const char *fmt, ...);
+
+//#include <stddef.h>
+//#include <unistd.h>
+// ----------- >8 ------------
+
+
+typedef int s32_t;
+typedef uint32_t u32_t;
+typedef int16_t  s16_t;
+typedef uint16_t u16_t;
+typedef int8_t s8_t;
+typedef uint8_t u8_t;
+
+// compile time switches
+
+// Set generic spiffs debug output call.
+#ifndef SPIFFS_DBG
+#define SPIFFS_DBG(_f, ...) //Dbprintf(_f, ## __VA_ARGS__)
+#define SPIFFS_DBGF(str) SPIFFS_DBG(str,NULL)
+#endif
+// Set spiffs debug output call for garbage collecting.
+#ifndef SPIFFS_GC_DBG
+#define SPIFFS_GC_DBG(_f, ...)
+#define SPIFFS_GC_DBGF(str) SPIFFS_GC_DBG(str,NULL)
+#endif
+// Set spiffs debug output call for caching.
+#ifndef SPIFFS_CACHE_DBG
+#define SPIFFS_CACHE_DBG(_f, ...) //Dbprintf(_f, ## __VA_ARGS__)
+#define SPIFFS_CACHE_DBGF(str) SPIFFS_CACHE_DBG(str,NULL)
+#endif
+// Set spiffs debug output call for system consistency checks.
+#ifndef SPIFFS_CHECK_DBG
+#define SPIFFS_CHECK_DBG(_f, ...) //SPIFFS_CHECK_DBG(_f, ## __VA_ARGS__)
+#define SPIFFS_CHECK_DBGF(str) SPIFFS_CHECK_DBG(str,NULL)
+#endif
+// Set spiffs debug output call for all api invocations.
+#ifndef SPIFFS_API_DBG
+#define SPIFFS_API_DBG(_f, ...) //Dbprintf(_f, ## __VA_ARGS__)
+#define SPIFFS_API_DBGF(str) SPIFFS_API_DBG(str,NULL)
+#endif
+
+
+
+// Defines spiffs debug print formatters
+// some general signed number
+#ifndef _SPIPRIi
+#define _SPIPRIi   "%d"
+#endif
+// address
+#ifndef _SPIPRIad
+#define _SPIPRIad  "%08x"
+#endif
+// block
+#ifndef _SPIPRIbl
+#define _SPIPRIbl  "%04x"
+#endif
+// page
+#ifndef _SPIPRIpg
+#define _SPIPRIpg  "%04x"
+#endif
+// span index
+#ifndef _SPIPRIsp
+#define _SPIPRIsp  "%04x"
+#endif
+// file descriptor
+#ifndef _SPIPRIfd
+#define _SPIPRIfd  "%d"
+#endif
+// file object id
+#ifndef _SPIPRIid
+#define _SPIPRIid  "%04x"
+#endif
+// file flags
+#ifndef _SPIPRIfl
+#define _SPIPRIfl  "%02x"
+#endif
+
+
+// Enable/disable API functions to determine exact number of bytes
+// for filedescriptor and cache buffers. Once decided for a configuration,
+// this can be disabled to reduce flash.
+#ifndef SPIFFS_BUFFER_HELP
+#define SPIFFS_BUFFER_HELP              0
+#endif
+
+// Enables/disable memory read caching of nucleus file system operations.
+// If enabled, memory area must be provided for cache in SPIFFS_mount.
+#ifndef  SPIFFS_CACHE
+#define SPIFFS_CACHE                    1
+#endif
+#if SPIFFS_CACHE
+// Enables memory write caching for file descriptors in hydrogen
+#ifndef  SPIFFS_CACHE_WR
+#define SPIFFS_CACHE_WR                 1
+#endif
+
+// Enable/disable statistics on caching. Debug/test purpose only.
+#ifndef  SPIFFS_CACHE_STATS
+#define SPIFFS_CACHE_STATS              0
+#endif
+#endif
+
+// Always check header of each accessed page to ensure consistent state.
+// If enabled it will increase number of reads, will increase flash.
+#ifndef SPIFFS_PAGE_CHECK
+#define SPIFFS_PAGE_CHECK               0
+#endif
+
+// Define maximum number of gc runs to perform to reach desired free pages.
+#ifndef SPIFFS_GC_MAX_RUNS
+#define SPIFFS_GC_MAX_RUNS              5
+#endif
+
+// Enable/disable statistics on gc. Debug/test purpose only.
+#ifndef SPIFFS_GC_STATS
+#define SPIFFS_GC_STATS                 0
+#endif
+
+// Garbage collecting examines all pages in a block which and sums up
+// to a block score. Deleted pages normally gives positive score and
+// used pages normally gives a negative score (as these must be moved).
+// To have a fair wear-leveling, the erase age is also included in score,
+// whose factor normally is the most positive.
+// The larger the score, the more likely it is that the block will
+// picked for garbage collection.
+
+// Garbage collecting heuristics - weight used for deleted pages.
+#ifndef SPIFFS_GC_HEUR_W_DELET
+#define SPIFFS_GC_HEUR_W_DELET          (5)
+#endif
+// Garbage collecting heuristics - weight used for used pages.
+#ifndef SPIFFS_GC_HEUR_W_USED
+#define SPIFFS_GC_HEUR_W_USED           (-1)
+#endif
+// Garbage collecting heuristics - weight used for time between
+// last erased and erase of this block.
+#ifndef SPIFFS_GC_HEUR_W_ERASE_AGE
+#define SPIFFS_GC_HEUR_W_ERASE_AGE      (50)
+#endif
+
+// Object name maximum length. Note that this length include the
+// zero-termination character, meaning maximum string of characters
+// can at most be SPIFFS_OBJ_NAME_LEN - 1.
+#ifndef SPIFFS_OBJ_NAME_LEN
+#define SPIFFS_OBJ_NAME_LEN             (32)
+#endif
+
+// Maximum length of the metadata associated with an object.
+// Setting to non-zero value enables metadata-related API but also
+// changes the on-disk format, so the change is not backward-compatible.
+//
+// Do note: the meta length must never exceed
+// logical_page_size - (SPIFFS_OBJ_NAME_LEN + 64)
+//
+// This is derived from following:
+// logical_page_size - (SPIFFS_OBJ_NAME_LEN + sizeof(spiffs_page_header) +
+// spiffs_object_ix_header fields + at least some LUT entries)
+#ifndef SPIFFS_OBJ_META_LEN
+#define SPIFFS_OBJ_META_LEN             (0)
+#endif
+
+// Size of buffer allocated on stack used when copying data.
+// Lower value generates more read/writes. No meaning having it bigger
+// than logical page size.
+#ifndef SPIFFS_COPY_BUFFER_STACK
+#define SPIFFS_COPY_BUFFER_STACK        (256)
+#endif
+
+// Enable this to have an identifiable spiffs filesystem. This will look for
+// a magic in all sectors to determine if this is a valid spiffs system or
+// not on mount point. If not, SPIFFS_format must be called prior to mounting
+// again.
+#ifndef SPIFFS_USE_MAGIC
+#define SPIFFS_USE_MAGIC                (0)
+#endif
+
+#if SPIFFS_USE_MAGIC
+// Only valid when SPIFFS_USE_MAGIC is enabled. If SPIFFS_USE_MAGIC_LENGTH is
+// enabled, the magic will also be dependent on the length of the filesystem.
+// For example, a filesystem configured and formatted for 4 megabytes will not
+// be accepted for mounting with a configuration defining the filesystem as 2
+// megabytes.
+#ifndef SPIFFS_USE_MAGIC_LENGTH
+#define SPIFFS_USE_MAGIC_LENGTH         (0)
+#endif
+#endif
+
+// SPIFFS_LOCK and SPIFFS_UNLOCK protects spiffs from reentrancy on api level
+// These should be defined on a multithreaded system
+
+// define this to enter a mutex if you're running on a multithreaded system
+#ifndef SPIFFS_LOCK
+#define SPIFFS_LOCK(fs)
+#endif
+// define this to exit a mutex if you're running on a multithreaded system
+#ifndef SPIFFS_UNLOCK
+#define SPIFFS_UNLOCK(fs)
+#endif
+
+// Enable if only one spiffs instance with constant configuration will exist
+// on the target. This will reduce calculations, flash and memory accesses.
+// Parts of configuration must be defined below instead of at time of mount.
+#ifndef SPIFFS_SINGLETON
+#define SPIFFS_SINGLETON (1)
+#endif
+
+#if SPIFFS_SINGLETON
+// Instead of giving parameters in config struct, singleton build must
+// give parameters in defines below.
+#ifndef SPIFFS_CFG_PHYS_SZ
+#define SPIFFS_CFG_PHYS_SZ(ignore)        (1024*128)
+#endif
+#ifndef SPIFFS_CFG_PHYS_ERASE_SZ
+#define SPIFFS_CFG_PHYS_ERASE_SZ(ignore)  (4*1024)
+#endif
+#ifndef SPIFFS_CFG_PHYS_ADDR
+#define SPIFFS_CFG_PHYS_ADDR(ignore)      (0)
+#endif
+#ifndef SPIFFS_CFG_LOG_PAGE_SZ
+#define SPIFFS_CFG_LOG_PAGE_SZ(ignore)    (256)
+#endif
+#ifndef SPIFFS_CFG_LOG_BLOCK_SZ
+#define SPIFFS_CFG_LOG_BLOCK_SZ(ignore)   (4*1024)
+#endif
+#endif
+
+// Enable this if your target needs aligned data for index tables
+#ifndef SPIFFS_ALIGNED_OBJECT_INDEX_TABLES
+#define SPIFFS_ALIGNED_OBJECT_INDEX_TABLES       1
+#endif
+
+// Enable this if you want the HAL callbacks to be called with the spiffs struct
+#ifndef SPIFFS_HAL_CALLBACK_EXTRA
+#define SPIFFS_HAL_CALLBACK_EXTRA         0
+#endif
+
+// Enable this if you want to add an integer offset to all file handles
+// (spiffs_file). This is useful if running multiple instances of spiffs on
+// same target, in order to recognise to what spiffs instance a file handle
+// belongs.
+// NB: This adds config field fh_ix_offset in the configuration struct when
+// mounting, which must be defined.
+#ifndef SPIFFS_FILEHDL_OFFSET
+#define SPIFFS_FILEHDL_OFFSET                 0
+#endif
+
+// Enable this to compile a read only version of spiffs.
+// This will reduce binary size of spiffs. All code comprising modification
+// of the file system will not be compiled. Some config will be ignored.
+// HAL functions for erasing and writing to spi-flash may be null. Cache
+// can be disabled for even further binary size reduction (and ram savings).
+// Functions modifying the fs will return SPIFFS_ERR_RO_NOT_IMPL.
+// If the file system cannot be mounted due to aborted erase operation and
+// SPIFFS_USE_MAGIC is enabled, SPIFFS_ERR_RO_ABORTED_OPERATION will be
+// returned.
+// Might be useful for e.g. bootloaders and such.
+#ifndef SPIFFS_READ_ONLY
+#define SPIFFS_READ_ONLY                      0
+#endif
+
+// Enable this to add a temporal file cache using the fd buffer.
+// The effects of the cache is that SPIFFS_open will find the file faster in
+// certain cases. It will make it a lot easier for spiffs to find files
+// opened frequently, reducing number of readings from the spi flash for
+// finding those files.
+// This will grow each fd by 6 bytes. If your files are opened in patterns
+// with a degree of temporal locality, the system is optimized.
+// Examples can be letting spiffs serve web content, where one file is the css.
+// The css is accessed for each html file that is opened, meaning it is
+// accessed almost every second time a file is opened. Another example could be
+// a log file that is often opened, written, and closed.
+// The size of the cache is number of given file descriptors, as it piggybacks
+// on the fd update mechanism. The cache lives in the closed file descriptors.
+// When closed, the fd know the whereabouts of the file. Instead of forgetting
+// this, the temporal cache will keep handling updates to that file even if the
+// fd is closed. If the file is opened again, the location of the file is found
+// directly. If all available descriptors become opened, all cache memory is
+// lost.
+#ifndef SPIFFS_TEMPORAL_FD_CACHE
+#define SPIFFS_TEMPORAL_FD_CACHE              1
+#endif
+
+// Temporal file cache hit score. Each time a file is opened, all cached files
+// will lose one point. If the opened file is found in cache, that entry will
+// gain SPIFFS_TEMPORAL_CACHE_HIT_SCORE points. One can experiment with this
+// value for the specific access patterns of the application. However, it must
+// be between 1 (no gain for hitting a cached entry often) and 255.
+#ifndef SPIFFS_TEMPORAL_CACHE_HIT_SCORE
+#define SPIFFS_TEMPORAL_CACHE_HIT_SCORE       4
+#endif
+
+// Enable to be able to map object indices to memory.
+// This allows for faster and more deterministic reading if cases of reading
+// large files and when changing file offset by seeking around a lot.
+// When mapping a file's index, the file system will be scanned for index pages
+// and the info will be put in memory provided by user. When reading, the
+// memory map can be looked up instead of searching for index pages on the
+// medium. This way, user can trade memory against performance.
+// Whole, parts of, or future parts not being written yet can be mapped. The
+// memory array will be owned by spiffs and updated accordingly during garbage
+// collecting or when modifying the indices. The latter is invoked by when the
+// file is modified in some way. The index buffer is tied to the file
+// descriptor.
+#ifndef SPIFFS_IX_MAP
+#define SPIFFS_IX_MAP                         1
+#endif
+
+// By default SPIFFS in some cases relies on the property of NOR flash that bits
+// cannot be set from 0 to 1 by writing and that controllers will ignore such
+// bit changes. This results in fewer reads as SPIFFS can in some cases perform
+// blind writes, with all bits set to 1 and only those it needs reset set to 0.
+// Most of the chips and controllers allow this behavior, so the default is to
+// use this technique. If your controller is one of the rare ones that don't,
+// turn this option on and SPIFFS will perform a read-modify-write instead.
+#ifndef SPIFFS_NO_BLIND_WRITES
+#define SPIFFS_NO_BLIND_WRITES                0
+#endif
+
+// Set SPIFFS_TEST_VISUALISATION to non-zero to enable SPIFFS_vis function
+// in the api. This function will visualize all filesystem using given printf
+// function.
+#ifndef SPIFFS_TEST_VISUALISATION
+#define SPIFFS_TEST_VISUALISATION         0
+#endif
+#if SPIFFS_TEST_VISUALISATION
+#ifndef spiffs_printf
+#define spiffs_printf(...)                Dbprintf(__VA_ARGS__)
+#endif
+// spiffs_printf argument for a free page
+#ifndef SPIFFS_TEST_VIS_FREE_STR
+#define SPIFFS_TEST_VIS_FREE_STR          "_"
+#endif
+// spiffs_printf argument for a deleted page
+#ifndef SPIFFS_TEST_VIS_DELE_STR
+#define SPIFFS_TEST_VIS_DELE_STR          "/"
+#endif
+// spiffs_printf argument for an index page for given object id
+#ifndef SPIFFS_TEST_VIS_INDX_STR
+#define SPIFFS_TEST_VIS_INDX_STR(id)      "i"
+#endif
+// spiffs_printf argument for a data page for given object id
+#ifndef SPIFFS_TEST_VIS_DATA_STR
+#define SPIFFS_TEST_VIS_DATA_STR(id)      "d"
+#endif
+#endif
+
+// Types depending on configuration such as the amount of flash bytes
+// given to spiffs file system in total (spiffs_file_system_size),
+// the logical block size (log_block_size), and the logical page size
+// (log_page_size)
+
+// Block index type. Make sure the size of this type can hold
+// the highest number of all blocks - i.e. spiffs_file_system_size / log_block_size
+typedef u16_t spiffs_block_ix;
+// Page index type. Make sure the size of this type can hold
+// the highest page number of all pages - i.e. spiffs_file_system_size / log_page_size
+typedef u16_t spiffs_page_ix;
+// Object id type - most significant bit is reserved for index flag. Make sure the
+// size of this type can hold the highest object id on a full system,
+// i.e. 2 + (spiffs_file_system_size / (2*log_page_size))*2
+typedef u16_t spiffs_obj_id;
+// Object span index type. Make sure the size of this type can
+// hold the largest possible span index on the system -
+// i.e. (spiffs_file_system_size / log_page_size) - 1
+typedef u16_t spiffs_span_ix;
+
+#endif /* SPIFFS_CONFIG_H_ */
diff --git a/armsrc/spiffs_gc.c b/armsrc/spiffs_gc.c
new file mode 100644
index 000000000..76ff094c9
--- /dev/null
+++ b/armsrc/spiffs_gc.c
@@ -0,0 +1,606 @@
+#include "spiffs.h"
+#include "spiffs_nucleus.h"
+
+#if !SPIFFS_READ_ONLY
+
+// Erases a logical block and updates the erase counter.
+// If cache is enabled, all pages that might be cached in this block
+// is dropped.
+static s32_t spiffs_gc_erase_block(
+    spiffs *fs,
+    spiffs_block_ix bix) {
+    s32_t res;
+
+    SPIFFS_GC_DBG("gc: erase block "_SPIPRIbl"\n", bix);
+    res = spiffs_erase_block(fs, bix);
+    SPIFFS_CHECK_RES(res);
+
+#if SPIFFS_CACHE
+    {
+        u32_t i;
+        for (i = 0; i < SPIFFS_PAGES_PER_BLOCK(fs); i++) {
+            spiffs_cache_drop_page(fs, SPIFFS_PAGE_FOR_BLOCK(fs, bix) + i);
+        }
+    }
+#endif
+    return res;
+}
+
+// Searches for blocks where all entries are deleted - if one is found,
+// the block is erased. Compared to the non-quick gc, the quick one ensures
+// that no updates are needed on existing objects on pages that are erased.
+s32_t spiffs_gc_quick(
+    spiffs *fs, u16_t max_free_pages) {
+    s32_t res = SPIFFS_OK;
+    u32_t blocks = fs->block_count;
+    spiffs_block_ix cur_block = 0;
+    u32_t cur_block_addr = 0;
+    int cur_entry = 0;
+    spiffs_obj_id *obj_lu_buf = (spiffs_obj_id *)fs->lu_work;
+
+    SPIFFS_GC_DBGF("gc_quick: running\n");
+#if SPIFFS_GC_STATS
+    fs->stats_gc_runs++;
+#endif
+
+    int entries_per_page = (SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(spiffs_obj_id));
+
+    // find fully deleted blocks
+    // check each block
+    while (res == SPIFFS_OK && blocks--) {
+        u16_t deleted_pages_in_block = 0;
+        u16_t free_pages_in_block = 0;
+
+        int obj_lookup_page = 0;
+        // check each object lookup page
+        while (res == SPIFFS_OK && obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs)) {
+            int entry_offset = obj_lookup_page * entries_per_page;
+            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                             0, cur_block_addr + SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+            // check each entry
+            while (res == SPIFFS_OK &&
+                    cur_entry - entry_offset < entries_per_page &&
+                    cur_entry < (int)(SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs))) {
+                spiffs_obj_id obj_id = obj_lu_buf[cur_entry - entry_offset];
+                if (obj_id == SPIFFS_OBJ_ID_DELETED) {
+                    deleted_pages_in_block++;
+                } else if (obj_id == SPIFFS_OBJ_ID_FREE) {
+                    // kill scan, go for next block
+                    free_pages_in_block++;
+                    if (free_pages_in_block > max_free_pages) {
+                        obj_lookup_page = SPIFFS_OBJ_LOOKUP_PAGES(fs);
+                        res = 1; // kill object lu loop
+                        break;
+                    }
+                }  else {
+                    // kill scan, go for next block
+                    obj_lookup_page = SPIFFS_OBJ_LOOKUP_PAGES(fs);
+                    res = 1; // kill object lu loop
+                    break;
+                }
+                cur_entry++;
+            } // per entry
+            obj_lookup_page++;
+        } // per object lookup page
+        if (res == 1) res = SPIFFS_OK;
+
+        if (res == SPIFFS_OK &&
+                deleted_pages_in_block + free_pages_in_block == SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs) &&
+                free_pages_in_block <= max_free_pages) {
+            // found a fully deleted block
+            fs->stats_p_deleted -= deleted_pages_in_block;
+            res = spiffs_gc_erase_block(fs, cur_block);
+            return res;
+        }
+
+        cur_entry = 0;
+        cur_block++;
+        cur_block_addr += SPIFFS_CFG_LOG_BLOCK_SZ(fs);
+    } // per block
+
+    if (res == SPIFFS_OK) {
+        res = SPIFFS_ERR_NO_DELETED_BLOCKS;
+    }
+    return res;
+}
+
+// Checks if garbage collecting is necessary. If so a candidate block is found,
+// cleansed and erased
+s32_t spiffs_gc_check(
+    spiffs *fs,
+    u32_t len) {
+    s32_t res;
+    s32_t free_pages =
+        (SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs)) * (fs->block_count - 2)
+        - fs->stats_p_allocated - fs->stats_p_deleted;
+    int tries = 0;
+
+    if (fs->free_blocks > 3 &&
+            (s32_t)len < free_pages * (s32_t)SPIFFS_DATA_PAGE_SIZE(fs)) {
+        return SPIFFS_OK;
+    }
+
+    u32_t needed_pages = (len + SPIFFS_DATA_PAGE_SIZE(fs) - 1) / SPIFFS_DATA_PAGE_SIZE(fs);
+//  if (fs->free_blocks <= 2 && (s32_t)needed_pages > free_pages) {
+//    SPIFFS_GC_DBG("gc: full freeblk:"_SPIPRIi" needed:"_SPIPRIi" free:"_SPIPRIi" dele:"_SPIPRIi"\n", fs->free_blocks, needed_pages, free_pages, fs->stats_p_deleted);
+//    return SPIFFS_ERR_FULL;
+//  }
+    if ((s32_t)needed_pages > (s32_t)(free_pages + fs->stats_p_deleted)) {
+        SPIFFS_GC_DBG("gc_check: full freeblk:"_SPIPRIi" needed:"_SPIPRIi" free:"_SPIPRIi" dele:"_SPIPRIi"\n", fs->free_blocks, needed_pages, free_pages, fs->stats_p_deleted);
+        return SPIFFS_ERR_FULL;
+    }
+
+    do {
+        SPIFFS_GC_DBG("\ngc_check #"_SPIPRIi": run gc free_blocks:"_SPIPRIi" pfree:"_SPIPRIi" pallo:"_SPIPRIi" pdele:"_SPIPRIi" ["_SPIPRIi"] len:"_SPIPRIi" of "_SPIPRIi"\n",
+                      tries,
+                      fs->free_blocks, free_pages, fs->stats_p_allocated, fs->stats_p_deleted, (free_pages + fs->stats_p_allocated + fs->stats_p_deleted),
+                      len, (u32_t)(free_pages * SPIFFS_DATA_PAGE_SIZE(fs)));
+
+        spiffs_block_ix *cands;
+        int count;
+        spiffs_block_ix cand;
+        s32_t prev_free_pages = free_pages;
+        // if the fs is crammed, ignore block age when selecting candidate - kind of a bad state
+        res = spiffs_gc_find_candidate(fs, &cands, &count, free_pages <= 0);
+        SPIFFS_CHECK_RES(res);
+        if (count == 0) {
+            SPIFFS_GC_DBGF("gc_check: no candidates, return\n");
+            return (s32_t)needed_pages < free_pages ? SPIFFS_OK : SPIFFS_ERR_FULL;
+        }
+#if SPIFFS_GC_STATS
+        fs->stats_gc_runs++;
+#endif
+        cand = cands[0];
+        fs->cleaning = 1;
+        //SPIFFS_GC_DBG("gcing: cleaning block "_SPIPRIi"\n", cand);
+        res = spiffs_gc_clean(fs, cand);
+        fs->cleaning = 0;
+        if (res < 0) {
+            SPIFFS_GC_DBG("gc_check: cleaning block "_SPIPRIi", result "_SPIPRIi"\n", cand, res);
+        } else {
+            SPIFFS_GC_DBG("gc_check: cleaning block "_SPIPRIi", result "_SPIPRIi"\n", cand, res);
+        }
+        SPIFFS_CHECK_RES(res);
+
+        res = spiffs_gc_erase_page_stats(fs, cand);
+        SPIFFS_CHECK_RES(res);
+
+        res = spiffs_gc_erase_block(fs, cand);
+        SPIFFS_CHECK_RES(res);
+
+        free_pages =
+            (SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs)) * (fs->block_count - 2)
+            - fs->stats_p_allocated - fs->stats_p_deleted;
+
+        if (prev_free_pages <= 0 && prev_free_pages == free_pages) {
+            // abort early to reduce wear, at least tried once
+            SPIFFS_GC_DBGF("gc_check: early abort, no result on gc when fs crammed\n");
+            break;
+        }
+
+    } while (++tries < SPIFFS_GC_MAX_RUNS && (fs->free_blocks <= 2 ||
+                                              (s32_t)len > free_pages * (s32_t)SPIFFS_DATA_PAGE_SIZE(fs)));
+
+    free_pages =
+        (SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs)) * (fs->block_count - 2)
+        - fs->stats_p_allocated - fs->stats_p_deleted;
+    if ((s32_t)len > free_pages * (s32_t)SPIFFS_DATA_PAGE_SIZE(fs)) {
+        res = SPIFFS_ERR_FULL;
+    }
+
+    SPIFFS_GC_DBG("gc_check: finished, "_SPIPRIi" dirty, blocks "_SPIPRIi" free, "_SPIPRIi" pages free, "_SPIPRIi" tries, res "_SPIPRIi"\n",
+                  fs->stats_p_allocated + fs->stats_p_deleted,
+                  fs->free_blocks, free_pages, tries, res);
+
+    return res;
+}
+
+// Updates page statistics for a block that is about to be erased
+s32_t spiffs_gc_erase_page_stats(
+    spiffs *fs,
+    spiffs_block_ix bix) {
+    s32_t res = SPIFFS_OK;
+    int obj_lookup_page = 0;
+    int entries_per_page = (SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(spiffs_obj_id));
+    spiffs_obj_id *obj_lu_buf = (spiffs_obj_id *)fs->lu_work;
+    int cur_entry = 0;
+    u32_t dele = 0;
+    u32_t allo = 0;
+
+    // check each object lookup page
+    while (res == SPIFFS_OK && obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs)) {
+        int entry_offset = obj_lookup_page * entries_per_page;
+        res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                         0, bix * SPIFFS_CFG_LOG_BLOCK_SZ(fs) + SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+        // check each entry
+        while (res == SPIFFS_OK &&
+                cur_entry - entry_offset < entries_per_page && cur_entry < (int)(SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs))) {
+            spiffs_obj_id obj_id = obj_lu_buf[cur_entry - entry_offset];
+            if (obj_id == SPIFFS_OBJ_ID_FREE) {
+            } else if (obj_id == SPIFFS_OBJ_ID_DELETED) {
+                dele++;
+            } else {
+                allo++;
+            }
+            cur_entry++;
+        } // per entry
+        obj_lookup_page++;
+    } // per object lookup page
+    SPIFFS_GC_DBG("gc_check: wipe pallo:"_SPIPRIi" pdele:"_SPIPRIi"\n", allo, dele);
+    fs->stats_p_allocated -= allo;
+    fs->stats_p_deleted -= dele;
+    return res;
+}
+
+// Finds block candidates to erase
+s32_t spiffs_gc_find_candidate(
+    spiffs *fs,
+    spiffs_block_ix **block_candidates,
+    int *candidate_count,
+    char fs_crammed) {
+    s32_t res = SPIFFS_OK;
+    u32_t blocks = fs->block_count;
+    spiffs_block_ix cur_block = 0;
+    u32_t cur_block_addr = 0;
+    spiffs_obj_id *obj_lu_buf = (spiffs_obj_id *)fs->lu_work;
+    int cur_entry = 0;
+
+    // using fs->work area as sorted candidate memory, (spiffs_block_ix)cand_bix/(s32_t)score
+    int max_candidates = MIN(fs->block_count, (SPIFFS_CFG_LOG_PAGE_SZ(fs) - 8) / (sizeof(spiffs_block_ix) + sizeof(s32_t)));
+    *candidate_count = 0;
+    memset(fs->work, 0xff, SPIFFS_CFG_LOG_PAGE_SZ(fs));
+
+    // divide up work area into block indices and scores
+    spiffs_block_ix *cand_blocks = (spiffs_block_ix *)fs->work;
+    s32_t *cand_scores = (s32_t *)(fs->work + max_candidates * sizeof(spiffs_block_ix));
+
+    // align cand_scores on s32_t boundary
+    cand_scores = (s32_t *)(((intptr_t)cand_scores + sizeof(intptr_t) - 1) & ~(sizeof(intptr_t) - 1));
+
+    *block_candidates = cand_blocks;
+
+    int entries_per_page = (SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(spiffs_obj_id));
+
+    // check each block
+    while (res == SPIFFS_OK && blocks--) {
+        u16_t deleted_pages_in_block = 0;
+        u16_t used_pages_in_block = 0;
+
+        int obj_lookup_page = 0;
+        // check each object lookup page
+        while (res == SPIFFS_OK && obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs)) {
+            int entry_offset = obj_lookup_page * entries_per_page;
+            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                             0, cur_block_addr + SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+            // check each entry
+            while (res == SPIFFS_OK &&
+                    cur_entry - entry_offset < entries_per_page &&
+                    cur_entry < (int)(SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs))) {
+                spiffs_obj_id obj_id = obj_lu_buf[cur_entry - entry_offset];
+                if (obj_id == SPIFFS_OBJ_ID_FREE) {
+                    // when a free entry is encountered, scan logic ensures that all following entries are free also
+                    res = 1; // kill object lu loop
+                    break;
+                } else  if (obj_id == SPIFFS_OBJ_ID_DELETED) {
+                    deleted_pages_in_block++;
+                } else {
+                    used_pages_in_block++;
+                }
+                cur_entry++;
+            } // per entry
+            obj_lookup_page++;
+        } // per object lookup page
+        if (res == 1) res = SPIFFS_OK;
+
+        // calculate score and insert into candidate table
+        // stoneage sort, but probably not so many blocks
+        if (res == SPIFFS_OK /*&& deleted_pages_in_block > 0*/) {
+            // read erase count
+            spiffs_obj_id erase_count;
+            res = _spiffs_rd(fs, SPIFFS_OP_C_READ | SPIFFS_OP_T_OBJ_LU2, 0,
+                             SPIFFS_ERASE_COUNT_PADDR(fs, cur_block),
+                             sizeof(spiffs_obj_id), (u8_t *)&erase_count);
+            SPIFFS_CHECK_RES(res);
+
+            spiffs_obj_id erase_age;
+            if (fs->max_erase_count > erase_count) {
+                erase_age = fs->max_erase_count - erase_count;
+            } else {
+                erase_age = SPIFFS_OBJ_ID_FREE - (erase_count - fs->max_erase_count);
+            }
+
+            s32_t score =
+                deleted_pages_in_block * SPIFFS_GC_HEUR_W_DELET +
+                used_pages_in_block * SPIFFS_GC_HEUR_W_USED +
+                erase_age * (fs_crammed ? 0 : SPIFFS_GC_HEUR_W_ERASE_AGE);
+            int cand_ix = 0;
+            SPIFFS_GC_DBG("gc_check: bix:"_SPIPRIbl" del:"_SPIPRIi" use:"_SPIPRIi" score:"_SPIPRIi"\n", cur_block, deleted_pages_in_block, used_pages_in_block, score);
+            while (cand_ix < max_candidates) {
+                if (cand_blocks[cand_ix] == (spiffs_block_ix) - 1) {
+                    cand_blocks[cand_ix] = cur_block;
+                    cand_scores[cand_ix] = score;
+                    break;
+                } else if (cand_scores[cand_ix] < score) {
+                    int reorder_cand_ix = max_candidates - 2;
+                    while (reorder_cand_ix >= cand_ix) {
+                        cand_blocks[reorder_cand_ix + 1] = cand_blocks[reorder_cand_ix];
+                        cand_scores[reorder_cand_ix + 1] = cand_scores[reorder_cand_ix];
+                        reorder_cand_ix--;
+                    }
+                    cand_blocks[cand_ix] = cur_block;
+                    cand_scores[cand_ix] = score;
+                    break;
+                }
+                cand_ix++;
+            }
+            (*candidate_count)++;
+        }
+
+        cur_entry = 0;
+        cur_block++;
+        cur_block_addr += SPIFFS_CFG_LOG_BLOCK_SZ(fs);
+    } // per block
+
+    return res;
+}
+
+typedef enum {
+    FIND_OBJ_DATA,
+    MOVE_OBJ_DATA,
+    MOVE_OBJ_IX,
+    FINISHED
+} spiffs_gc_clean_state;
+
+typedef struct {
+    spiffs_gc_clean_state state;
+    spiffs_obj_id cur_obj_id;
+    spiffs_span_ix cur_objix_spix;
+    spiffs_page_ix cur_objix_pix;
+    spiffs_page_ix cur_data_pix;
+    int stored_scan_entry_index;
+    u8_t obj_id_found;
+} spiffs_gc;
+
+// Empties given block by moving all data into free pages of another block
+// Strategy:
+//   loop:
+//   scan object lookup for object data pages
+//   for first found id, check spix and load corresponding object index page to memory
+//   push object scan lookup entry index
+//     rescan object lookup, find data pages with same id and referenced by same object index
+//     move data page, update object index in memory
+//     when reached end of lookup, store updated object index
+//   pop object scan lookup entry index
+//   repeat loop until end of object lookup
+//   scan object lookup again for remaining object index pages, move to new page in other block
+//
+s32_t spiffs_gc_clean(spiffs *fs, spiffs_block_ix bix) {
+    s32_t res = SPIFFS_OK;
+    const int entries_per_page = (SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(spiffs_obj_id));
+    // this is the global localizer being pushed and popped
+    int cur_entry = 0;
+    spiffs_obj_id *obj_lu_buf = (spiffs_obj_id *)fs->lu_work;
+    spiffs_gc gc; // our stack frame/state
+    spiffs_page_ix cur_pix = 0;
+    spiffs_page_object_ix_header *objix_hdr = (spiffs_page_object_ix_header *)fs->work;
+    spiffs_page_object_ix *objix = (spiffs_page_object_ix *)fs->work;
+
+    SPIFFS_GC_DBG("gc_clean: cleaning block "_SPIPRIbl"\n", bix);
+
+    memset(&gc, 0, sizeof(spiffs_gc));
+    gc.state = FIND_OBJ_DATA;
+
+    if (fs->free_cursor_block_ix == bix) {
+        // move free cursor to next block, cannot use free pages from the block we want to clean
+        fs->free_cursor_block_ix = (bix + 1) % fs->block_count;
+        fs->free_cursor_obj_lu_entry = 0;
+        SPIFFS_GC_DBG("gc_clean: move free cursor to block "_SPIPRIbl"\n", fs->free_cursor_block_ix);
+    }
+
+    while (res == SPIFFS_OK && gc.state != FINISHED) {
+        SPIFFS_GC_DBG("gc_clean: state = "_SPIPRIi" entry:"_SPIPRIi"\n", gc.state, cur_entry);
+        gc.obj_id_found = 0; // reset (to no found data page)
+
+        // scan through lookup pages
+        int obj_lookup_page = cur_entry / entries_per_page;
+        u8_t scan = 1;
+        // check each object lookup page
+        while (scan && res == SPIFFS_OK && obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs)) {
+            int entry_offset = obj_lookup_page * entries_per_page;
+            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                             0, bix * SPIFFS_CFG_LOG_BLOCK_SZ(fs) + SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page),
+                             SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+            // check each object lookup entry
+            while (scan && res == SPIFFS_OK &&
+                    cur_entry - entry_offset < entries_per_page && cur_entry < (int)(SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs))) {
+                spiffs_obj_id obj_id = obj_lu_buf[cur_entry - entry_offset];
+                cur_pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, cur_entry);
+
+                // act upon object id depending on gc state
+                switch (gc.state) {
+                    case FIND_OBJ_DATA:
+                        // find a data page
+                        if (obj_id != SPIFFS_OBJ_ID_DELETED && obj_id != SPIFFS_OBJ_ID_FREE &&
+                                ((obj_id & SPIFFS_OBJ_ID_IX_FLAG) == 0)) {
+                            // found a data page, stop scanning and handle in switch case below
+                            SPIFFS_GC_DBG("gc_clean: FIND_DATA state:"_SPIPRIi" - found obj id "_SPIPRIid"\n", gc.state, obj_id);
+                            gc.obj_id_found = 1;
+                            gc.cur_obj_id = obj_id;
+                            gc.cur_data_pix = cur_pix;
+                            scan = 0;
+                        }
+                        break;
+                    case MOVE_OBJ_DATA:
+                        // evacuate found data pages for corresponding object index we have in memory,
+                        // update memory representation
+                        if (obj_id == gc.cur_obj_id) {
+                            spiffs_page_header p_hdr;
+                            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                                             0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix), sizeof(spiffs_page_header), (u8_t *)&p_hdr);
+                            SPIFFS_CHECK_RES(res);
+                            SPIFFS_GC_DBG("gc_clean: MOVE_DATA found data page "_SPIPRIid":"_SPIPRIsp" @ "_SPIPRIpg"\n", gc.cur_obj_id, p_hdr.span_ix, cur_pix);
+                            if (SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, p_hdr.span_ix) != gc.cur_objix_spix) {
+                                SPIFFS_GC_DBGF("gc_clean: MOVE_DATA no objix spix match, take in another run\n");
+                            } else {
+                                spiffs_page_ix new_data_pix;
+                                if (p_hdr.flags & SPIFFS_PH_FLAG_DELET) {
+                                    // move page
+                                    res = spiffs_page_move(fs, 0, 0, obj_id, &p_hdr, cur_pix, &new_data_pix);
+                                    SPIFFS_GC_DBG("gc_clean: MOVE_DATA move objix "_SPIPRIid":"_SPIPRIsp" page "_SPIPRIpg" to "_SPIPRIpg"\n", gc.cur_obj_id, p_hdr.span_ix, cur_pix, new_data_pix);
+                                    SPIFFS_CHECK_RES(res);
+                                    // move wipes obj_lu, reload it
+                                    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                                                     0, bix * SPIFFS_CFG_LOG_BLOCK_SZ(fs) + SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page),
+                                                     SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+                                    SPIFFS_CHECK_RES(res);
+                                } else {
+                                    // page is deleted but not deleted in lookup, scrap it -
+                                    // might seem unnecessary as we will erase this block, but
+                                    // we might get aborted
+                                    SPIFFS_GC_DBG("gc_clean: MOVE_DATA wipe objix "_SPIPRIid":"_SPIPRIsp" page "_SPIPRIpg"\n", obj_id, p_hdr.span_ix, cur_pix);
+                                    res = spiffs_page_delete(fs, cur_pix);
+                                    SPIFFS_CHECK_RES(res);
+                                    new_data_pix = SPIFFS_OBJ_ID_FREE;
+                                }
+                                // update memory representation of object index page with new data page
+                                if (gc.cur_objix_spix == 0) {
+                                    // update object index header page
+                                    ((spiffs_page_ix *)((u8_t *)objix_hdr + sizeof(spiffs_page_object_ix_header)))[p_hdr.span_ix] = new_data_pix;
+                                    SPIFFS_GC_DBG("gc_clean: MOVE_DATA wrote page "_SPIPRIpg" to objix_hdr entry "_SPIPRIsp" in mem\n", new_data_pix, (spiffs_span_ix)SPIFFS_OBJ_IX_ENTRY(fs, p_hdr.span_ix));
+                                } else {
+                                    // update object index page
+                                    ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, p_hdr.span_ix)] = new_data_pix;
+                                    SPIFFS_GC_DBG("gc_clean: MOVE_DATA wrote page "_SPIPRIpg" to objix entry "_SPIPRIsp" in mem\n", new_data_pix, (spiffs_span_ix)SPIFFS_OBJ_IX_ENTRY(fs, p_hdr.span_ix));
+                                }
+                            }
+                        }
+                        break;
+                    case MOVE_OBJ_IX:
+                        // find and evacuate object index pages
+                        if (obj_id != SPIFFS_OBJ_ID_DELETED && obj_id != SPIFFS_OBJ_ID_FREE &&
+                                (obj_id & SPIFFS_OBJ_ID_IX_FLAG)) {
+                            // found an index object id
+                            spiffs_page_header p_hdr;
+                            spiffs_page_ix new_pix;
+                            // load header
+                            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                                             0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix), sizeof(spiffs_page_header), (u8_t *)&p_hdr);
+                            SPIFFS_CHECK_RES(res);
+                            if (p_hdr.flags & SPIFFS_PH_FLAG_DELET) {
+                                // move page
+                                res = spiffs_page_move(fs, 0, 0, obj_id, &p_hdr, cur_pix, &new_pix);
+                                SPIFFS_GC_DBG("gc_clean: MOVE_OBJIX move objix "_SPIPRIid":"_SPIPRIsp" page "_SPIPRIpg" to "_SPIPRIpg"\n", obj_id, p_hdr.span_ix, cur_pix, new_pix);
+                                SPIFFS_CHECK_RES(res);
+                                spiffs_cb_object_event(fs, (spiffs_page_object_ix *)&p_hdr,
+                                                       SPIFFS_EV_IX_MOV, obj_id, p_hdr.span_ix, new_pix, 0);
+                                // move wipes obj_lu, reload it
+                                res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                                                 0, bix * SPIFFS_CFG_LOG_BLOCK_SZ(fs) + SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page),
+                                                 SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+                                SPIFFS_CHECK_RES(res);
+                            } else {
+                                // page is deleted but not deleted in lookup, scrap it -
+                                // might seem unnecessary as we will erase this block, but
+                                // we might get aborted
+                                SPIFFS_GC_DBG("gc_clean: MOVE_OBJIX wipe objix "_SPIPRIid":"_SPIPRIsp" page "_SPIPRIpg"\n", obj_id, p_hdr.span_ix, cur_pix);
+                                res = spiffs_page_delete(fs, cur_pix);
+                                if (res == SPIFFS_OK) {
+                                    spiffs_cb_object_event(fs, (spiffs_page_object_ix *)0,
+                                                           SPIFFS_EV_IX_DEL, obj_id, p_hdr.span_ix, cur_pix, 0);
+                                }
+                            }
+                            SPIFFS_CHECK_RES(res);
+                        }
+                        break;
+                    default:
+                        scan = 0;
+                        break;
+                } // switch gc state
+                cur_entry++;
+            } // per entry
+            obj_lookup_page++; // no need to check scan variable here, obj_lookup_page is set in start of loop
+        } // per object lookup page
+        if (res != SPIFFS_OK) break;
+
+        // state finalization and switch
+        switch (gc.state) {
+            case FIND_OBJ_DATA:
+                if (gc.obj_id_found) {
+                    // handle found data page -
+                    // find out corresponding obj ix page and load it to memory
+                    spiffs_page_header p_hdr;
+                    spiffs_page_ix objix_pix;
+                    gc.stored_scan_entry_index = cur_entry; // push cursor
+                    cur_entry = 0; // restart scan from start
+                    gc.state = MOVE_OBJ_DATA;
+                    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                                     0, SPIFFS_PAGE_TO_PADDR(fs, cur_pix), sizeof(spiffs_page_header), (u8_t *)&p_hdr);
+                    SPIFFS_CHECK_RES(res);
+                    gc.cur_objix_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, p_hdr.span_ix);
+                    SPIFFS_GC_DBG("gc_clean: FIND_DATA find objix span_ix:"_SPIPRIsp"\n", gc.cur_objix_spix);
+                    res = spiffs_obj_lu_find_id_and_span(fs, gc.cur_obj_id | SPIFFS_OBJ_ID_IX_FLAG, gc.cur_objix_spix, 0, &objix_pix);
+                    if (res == SPIFFS_ERR_NOT_FOUND) {
+                        // on borked systems we might get an ERR_NOT_FOUND here -
+                        // this is handled by simply deleting the page as it is not referenced
+                        // from anywhere
+                        SPIFFS_GC_DBG("gc_clean: FIND_OBJ_DATA objix not found! Wipe page "_SPIPRIpg"\n", gc.cur_data_pix);
+                        res = spiffs_page_delete(fs, gc.cur_data_pix);
+                        SPIFFS_CHECK_RES(res);
+                        // then we restore states and continue scanning for data pages
+                        cur_entry = gc.stored_scan_entry_index; // pop cursor
+                        gc.state = FIND_OBJ_DATA;
+                        break; // done
+                    }
+                    SPIFFS_CHECK_RES(res);
+                    SPIFFS_GC_DBG("gc_clean: FIND_DATA found object index at page "_SPIPRIpg"\n", objix_pix);
+                    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                                     0, SPIFFS_PAGE_TO_PADDR(fs, objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+                    SPIFFS_CHECK_RES(res);
+                    // cannot allow a gc if the presumed index in fact is no index, a
+                    // check must run or lot of data may be lost
+                    SPIFFS_VALIDATE_OBJIX(objix->p_hdr, gc.cur_obj_id | SPIFFS_OBJ_ID_IX_FLAG, gc.cur_objix_spix);
+                    gc.cur_objix_pix = objix_pix;
+                } else {
+                    // no more data pages found, passed thru all block, start evacuating object indices
+                    gc.state = MOVE_OBJ_IX;
+                    cur_entry = 0; // restart entry scan index
+                }
+                break;
+            case MOVE_OBJ_DATA: {
+                // store modified objix (hdr) page residing in memory now that all
+                // data pages belonging to this object index and residing in the block
+                // we want to evacuate
+                spiffs_page_ix new_objix_pix;
+                gc.state = FIND_OBJ_DATA;
+                cur_entry = gc.stored_scan_entry_index; // pop cursor
+                if (gc.cur_objix_spix == 0) {
+                    // store object index header page
+                    res = spiffs_object_update_index_hdr(fs, 0, gc.cur_obj_id | SPIFFS_OBJ_ID_IX_FLAG, gc.cur_objix_pix, fs->work, 0, 0, 0, &new_objix_pix);
+                    SPIFFS_GC_DBG("gc_clean: MOVE_DATA store modified objix_hdr page, "_SPIPRIpg":"_SPIPRIsp"\n", new_objix_pix, 0);
+                    SPIFFS_CHECK_RES(res);
+                } else {
+                    // store object index page
+                    res = spiffs_page_move(fs, 0, fs->work, gc.cur_obj_id | SPIFFS_OBJ_ID_IX_FLAG, 0, gc.cur_objix_pix, &new_objix_pix);
+                    SPIFFS_GC_DBG("gc_clean: MOVE_DATA store modified objix page, "_SPIPRIpg":"_SPIPRIsp"\n", new_objix_pix, objix->p_hdr.span_ix);
+                    SPIFFS_CHECK_RES(res);
+                    spiffs_cb_object_event(fs, (spiffs_page_object_ix *)fs->work,
+                                           SPIFFS_EV_IX_UPD, gc.cur_obj_id, objix->p_hdr.span_ix, new_objix_pix, 0);
+                }
+            }
+            break;
+            case MOVE_OBJ_IX:
+                // scanned thru all block, no more object indices found - our work here is done
+                gc.state = FINISHED;
+                break;
+            default:
+                cur_entry = 0;
+                break;
+        } // switch gc.state
+        SPIFFS_GC_DBG("gc_clean: state-> "_SPIPRIi"\n", gc.state);
+    } // while state != FINISHED
+
+
+    return res;
+}
+
+#endif // !SPIFFS_READ_ONLY
diff --git a/armsrc/spiffs_hydrogen.c b/armsrc/spiffs_hydrogen.c
new file mode 100644
index 000000000..dee72fb5f
--- /dev/null
+++ b/armsrc/spiffs_hydrogen.c
@@ -0,0 +1,1466 @@
+/*
+ * spiffs_hydrogen.c
+ *
+ *  Created on: Jun 16, 2013
+ *      Author: petera
+ */
+
+#include "spiffs.h"
+#include "spiffs_nucleus.h"
+#include "printf.h"
+
+#if SPIFFS_CACHE == 1
+static s32_t spiffs_fflush_cache(spiffs *fs, spiffs_file fh);
+#endif
+
+#if SPIFFS_BUFFER_HELP
+u32_t SPIFFS_buffer_bytes_for_filedescs(spiffs *fs, u32_t num_descs) {
+    return num_descs * sizeof(spiffs_fd);
+}
+#if SPIFFS_CACHE
+u32_t SPIFFS_buffer_bytes_for_cache(spiffs *fs, u32_t num_pages) {
+    return sizeof(spiffs_cache) + num_pages * (sizeof(spiffs_cache_page) + SPIFFS_CFG_LOG_PAGE_SZ(fs));
+}
+#endif
+#endif
+
+u8_t SPIFFS_mounted(spiffs *fs) {
+    return SPIFFS_CHECK_MOUNT(fs);
+}
+
+s32_t SPIFFS_format(spiffs *fs) {
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    SPIFFS_API_CHECK_CFG(fs);
+    if (SPIFFS_CHECK_MOUNT(fs)) {
+        fs->err_code = SPIFFS_ERR_MOUNTED;
+        return -1;
+    }
+
+    s32_t res;
+    SPIFFS_LOCK(fs);
+
+    spiffs_block_ix bix = 0;
+    while (bix < fs->block_count) {
+        fs->max_erase_count = 0;
+        res = spiffs_erase_block(fs, bix);
+        if (res != SPIFFS_OK) {
+            res = SPIFFS_ERR_ERASE_FAIL;
+        }
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+        bix++;
+    }
+
+    SPIFFS_UNLOCK(fs);
+
+    return 0;
+#endif // SPIFFS_READ_ONLY
+}
+
+#if SPIFFS_USE_MAGIC && SPIFFS_USE_MAGIC_LENGTH && SPIFFS_SINGLETON==0
+
+s32_t SPIFFS_probe_fs(spiffs_config *config) {
+    SPIFFS_API_DBG("%s\n", __func__);
+    s32_t res = spiffs_probe(config);
+    return res;
+}
+
+#endif // SPIFFS_USE_MAGIC && SPIFFS_USE_MAGIC_LENGTH && SPIFFS_SINGLETON==0
+
+s32_t SPIFFS_mount(spiffs *fs, spiffs_config *config, u8_t *work,
+                   u8_t *fd_space, u32_t fd_space_size,
+                   void *cache, u32_t cache_size,
+                   spiffs_check_callback check_cb_f) {
+    SPIFFS_API_DBG("%s "
+                   " sz:"_SPIPRIi " logpgsz:"_SPIPRIi " logblksz:"_SPIPRIi " perasz:"_SPIPRIi
+                   " addr:"_SPIPRIad
+                   " fdsz:"_SPIPRIi " cachesz:"_SPIPRIi
+                   "\n",
+                   __func__,
+                   SPIFFS_CFG_PHYS_SZ(fs),
+                   SPIFFS_CFG_LOG_PAGE_SZ(fs),
+                   SPIFFS_CFG_LOG_BLOCK_SZ(fs),
+                   SPIFFS_CFG_PHYS_ERASE_SZ(fs),
+                   SPIFFS_CFG_PHYS_ADDR(fs),
+                   fd_space_size, cache_size);
+    void *user_data;
+    SPIFFS_LOCK(fs);
+    user_data = fs->user_data;
+    memset(fs, 0, sizeof(spiffs));
+    _SPIFFS_MEMCPY(&fs->cfg, config, sizeof(spiffs_config));
+    fs->user_data = user_data;
+    fs->block_count = SPIFFS_CFG_PHYS_SZ(fs) / SPIFFS_CFG_LOG_BLOCK_SZ(fs);
+    fs->work = &work[0];
+    fs->lu_work = &work[SPIFFS_CFG_LOG_PAGE_SZ(fs)];
+    memset(fd_space, 0, fd_space_size);
+    // align fd_space pointer to pointer size byte boundary
+    u8_t ptr_size = sizeof(void *);
+    u8_t addr_lsb = ((u8_t)(intptr_t)fd_space) & (ptr_size - 1);
+    if (addr_lsb) {
+        fd_space += (ptr_size - addr_lsb);
+        fd_space_size -= (ptr_size - addr_lsb);
+    }
+    fs->fd_space = fd_space;
+    fs->fd_count = (fd_space_size / sizeof(spiffs_fd));
+
+    // align cache pointer to 4 byte boundary
+    addr_lsb = ((u8_t)(intptr_t)cache) & (ptr_size - 1);
+    if (addr_lsb) {
+        u8_t *cache_8 = (u8_t *)cache;
+        cache_8 += (ptr_size - addr_lsb);
+        cache = cache_8;
+        cache_size -= (ptr_size - addr_lsb);
+    }
+    if (cache_size & (ptr_size - 1)) {
+        cache_size -= (cache_size & (ptr_size - 1));
+    }
+
+#if SPIFFS_CACHE
+    fs->cache = cache;
+    fs->cache_size = (cache_size > (SPIFFS_CFG_LOG_PAGE_SZ(fs) * 32)) ? SPIFFS_CFG_LOG_PAGE_SZ(fs) * 32 : cache_size;
+    spiffs_cache_init(fs);
+#endif
+
+    s32_t res;
+
+#if SPIFFS_USE_MAGIC
+    res = SPIFFS_CHECK_MAGIC_POSSIBLE(fs) ? SPIFFS_OK : SPIFFS_ERR_MAGIC_NOT_POSSIBLE;
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+#endif
+
+    fs->config_magic = SPIFFS_CONFIG_MAGIC;
+
+    res = spiffs_obj_lu_scan(fs);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    SPIFFS_DBG("page index byte len:         "_SPIPRIi"\n", (u32_t)SPIFFS_CFG_LOG_PAGE_SZ(fs));
+    SPIFFS_DBG("object lookup pages:         "_SPIPRIi"\n", (u32_t)SPIFFS_OBJ_LOOKUP_PAGES(fs));
+    SPIFFS_DBG("page pages per block:        "_SPIPRIi"\n", (u32_t)SPIFFS_PAGES_PER_BLOCK(fs));
+    SPIFFS_DBG("page header length:          "_SPIPRIi"\n", (u32_t)sizeof(spiffs_page_header));
+    SPIFFS_DBG("object header index entries: "_SPIPRIi"\n", (u32_t)SPIFFS_OBJ_HDR_IX_LEN(fs));
+    SPIFFS_DBG("object index entries:        "_SPIPRIi"\n", (u32_t)SPIFFS_OBJ_IX_LEN(fs));
+    SPIFFS_DBG("available file descriptors:  "_SPIPRIi"\n", (u32_t)fs->fd_count);
+    SPIFFS_DBG("free blocks:                 "_SPIPRIi"\n", (u32_t)fs->free_blocks);
+
+    fs->check_cb_f = check_cb_f;
+
+    fs->mounted = 1;
+
+    SPIFFS_UNLOCK(fs);
+
+    return 0;
+}
+
+void SPIFFS_unmount(spiffs *fs) {
+    SPIFFS_API_DBG("%s\n", __func__);
+    if (!SPIFFS_CHECK_CFG(fs) || !SPIFFS_CHECK_MOUNT(fs)) return;
+    SPIFFS_LOCK(fs);
+    u32_t i;
+    spiffs_fd *fds = (spiffs_fd *)fs->fd_space;
+    for (i = 0; i < fs->fd_count; i++) {
+        spiffs_fd *cur_fd = &fds[i];
+        if (cur_fd->file_nbr != 0) {
+#if SPIFFS_CACHE
+            (void)spiffs_fflush_cache(fs, cur_fd->file_nbr);
+#endif
+            spiffs_fd_return(fs, cur_fd->file_nbr);
+        }
+    }
+    fs->mounted = 0;
+
+    SPIFFS_UNLOCK(fs);
+}
+
+s32_t SPIFFS_errno(spiffs *fs) {
+    return fs->err_code;
+}
+
+void SPIFFS_clearerr(spiffs *fs) {
+    SPIFFS_API_DBG("%s\n", __func__);
+    fs->err_code = SPIFFS_OK;
+}
+
+s32_t SPIFFS_creat(spiffs *fs, const char *path, spiffs_mode mode) {
+    SPIFFS_API_DBG("%s '%s'\n", __func__, path);
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    (void)path;
+    (void)mode;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    (void)mode;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    if (strlen(path) > SPIFFS_OBJ_NAME_LEN - 1) {
+        SPIFFS_API_CHECK_RES(fs, SPIFFS_ERR_NAME_TOO_LONG);
+    }
+    SPIFFS_LOCK(fs);
+    spiffs_obj_id obj_id;
+    s32_t res;
+
+    res = spiffs_obj_lu_find_free_obj_id(fs, &obj_id, (const u8_t *)path);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    res = spiffs_object_create(fs, obj_id, (const u8_t *)path, 0, SPIFFS_TYPE_FILE, 0);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    SPIFFS_UNLOCK(fs);
+    return 0;
+#endif // SPIFFS_READ_ONLY
+}
+
+spiffs_file SPIFFS_open(spiffs *fs, const char *path, spiffs_flags flags, spiffs_mode mode) {
+    SPIFFS_API_DBG("%s '%s' "_SPIPRIfl "\n", __func__, path, flags);
+    (void)mode;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    if (strlen(path) > SPIFFS_OBJ_NAME_LEN - 1) {
+        SPIFFS_API_CHECK_RES(fs, SPIFFS_ERR_NAME_TOO_LONG);
+    }
+    SPIFFS_LOCK(fs);
+
+    spiffs_fd *fd;
+    spiffs_page_ix pix;
+
+#if SPIFFS_READ_ONLY
+    // not valid flags in read only mode
+    flags &= ~(SPIFFS_WRONLY | SPIFFS_CREAT | SPIFFS_TRUNC);
+#endif // SPIFFS_READ_ONLY
+
+    s32_t res = spiffs_fd_find_new(fs, &fd, path);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_find_object_index_header_by_name(fs, (const u8_t *)path, &pix);
+    if ((flags & SPIFFS_O_CREAT) == 0) {
+        if (res < SPIFFS_OK) {
+            spiffs_fd_return(fs, fd->file_nbr);
+        }
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+
+    if (res == SPIFFS_OK &&
+            (flags & (SPIFFS_O_CREAT | SPIFFS_O_EXCL)) == (SPIFFS_O_CREAT | SPIFFS_O_EXCL)) {
+        // creat and excl and file exists - fail
+        res = SPIFFS_ERR_FILE_EXISTS;
+        spiffs_fd_return(fs, fd->file_nbr);
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+
+    if ((flags & SPIFFS_O_CREAT) && res == SPIFFS_ERR_NOT_FOUND) {
+#if !SPIFFS_READ_ONLY
+        spiffs_obj_id obj_id;
+        // no need to enter conflicting name here, already looked for it above
+        res = spiffs_obj_lu_find_free_obj_id(fs, &obj_id, 0);
+        if (res < SPIFFS_OK) {
+            spiffs_fd_return(fs, fd->file_nbr);
+        }
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+        res = spiffs_object_create(fs, obj_id, (const u8_t *)path, 0, SPIFFS_TYPE_FILE, &pix);
+        if (res < SPIFFS_OK) {
+            spiffs_fd_return(fs, fd->file_nbr);
+        }
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+        flags &= ~SPIFFS_O_TRUNC;
+#endif // !SPIFFS_READ_ONLY
+    } else {
+        if (res < SPIFFS_OK) {
+            spiffs_fd_return(fs, fd->file_nbr);
+        }
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+    res = spiffs_object_open_by_page(fs, pix, fd, flags, mode);
+    if (res < SPIFFS_OK) {
+        spiffs_fd_return(fs, fd->file_nbr);
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+#if !SPIFFS_READ_ONLY
+    if (flags & SPIFFS_O_TRUNC) {
+        res = spiffs_object_truncate(fd, 0, 0);
+        if (res < SPIFFS_OK) {
+            spiffs_fd_return(fs, fd->file_nbr);
+        }
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+#endif // !SPIFFS_READ_ONLY
+
+    fd->fdoffset = 0;
+
+    SPIFFS_UNLOCK(fs);
+
+    return SPIFFS_FH_OFFS(fs, fd->file_nbr);
+}
+
+spiffs_file SPIFFS_open_by_dirent(spiffs *fs, struct spiffs_dirent *e, spiffs_flags flags, spiffs_mode mode) {
+    SPIFFS_API_DBG("%s '%s':"_SPIPRIid " "_SPIPRIfl "\n", __func__, e->name, e->obj_id, flags);
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    spiffs_fd *fd;
+
+    s32_t res = spiffs_fd_find_new(fs, &fd, 0);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_open_by_page(fs, e->pix, fd, flags, mode);
+    if (res < SPIFFS_OK) {
+        spiffs_fd_return(fs, fd->file_nbr);
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+#if !SPIFFS_READ_ONLY
+    if (flags & SPIFFS_O_TRUNC) {
+        res = spiffs_object_truncate(fd, 0, 0);
+        if (res < SPIFFS_OK) {
+            spiffs_fd_return(fs, fd->file_nbr);
+        }
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+#endif // !SPIFFS_READ_ONLY
+
+    fd->fdoffset = 0;
+
+    SPIFFS_UNLOCK(fs);
+
+    return SPIFFS_FH_OFFS(fs, fd->file_nbr);
+}
+
+spiffs_file SPIFFS_open_by_page(spiffs *fs, spiffs_page_ix page_ix, spiffs_flags flags, spiffs_mode mode) {
+    SPIFFS_API_DBG("%s "_SPIPRIpg " "_SPIPRIfl "\n", __func__, page_ix, flags);
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    spiffs_fd *fd;
+
+    s32_t res = spiffs_fd_find_new(fs, &fd, 0);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    if (SPIFFS_IS_LOOKUP_PAGE(fs, page_ix)) {
+        res = SPIFFS_ERR_NOT_A_FILE;
+        spiffs_fd_return(fs, fd->file_nbr);
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+
+    res = spiffs_object_open_by_page(fs, page_ix, fd, flags, mode);
+    if (res == SPIFFS_ERR_IS_FREE ||
+            res == SPIFFS_ERR_DELETED ||
+            res == SPIFFS_ERR_NOT_FINALIZED ||
+            res == SPIFFS_ERR_NOT_INDEX ||
+            res == SPIFFS_ERR_INDEX_SPAN_MISMATCH) {
+        res = SPIFFS_ERR_NOT_A_FILE;
+    }
+    if (res < SPIFFS_OK) {
+        spiffs_fd_return(fs, fd->file_nbr);
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+#if !SPIFFS_READ_ONLY
+    if (flags & SPIFFS_O_TRUNC) {
+        res = spiffs_object_truncate(fd, 0, 0);
+        if (res < SPIFFS_OK) {
+            spiffs_fd_return(fs, fd->file_nbr);
+        }
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+#endif // !SPIFFS_READ_ONLY
+
+    fd->fdoffset = 0;
+
+    SPIFFS_UNLOCK(fs);
+
+    return SPIFFS_FH_OFFS(fs, fd->file_nbr);
+}
+
+static s32_t spiffs_hydro_read(spiffs *fs, spiffs_file fh, void *buf, s32_t len) {
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    spiffs_fd *fd;
+    s32_t res;
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    if ((fd->flags & SPIFFS_O_RDONLY) == 0) {
+        res = SPIFFS_ERR_NOT_READABLE;
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+
+    if (fd->size == SPIFFS_UNDEFINED_LEN && len > 0) {
+        // special case for zero sized files
+        res = SPIFFS_ERR_END_OF_OBJECT;
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+
+#if SPIFFS_CACHE_WR
+    spiffs_fflush_cache(fs, fh);
+#endif
+
+    if (fd->fdoffset + len >= fd->size) {
+        // reading beyond file size
+        s32_t avail = fd->size - fd->fdoffset;
+        if (avail <= 0) {
+            SPIFFS_API_CHECK_RES_UNLOCK(fs, SPIFFS_ERR_END_OF_OBJECT);
+        }
+        res = spiffs_object_read(fd, fd->fdoffset, avail, (u8_t *)buf);
+        if (res == SPIFFS_ERR_END_OF_OBJECT) {
+            fd->fdoffset += avail;
+            SPIFFS_UNLOCK(fs);
+            return avail;
+        } else {
+            SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+            len = avail;
+        }
+    } else {
+        // reading within file size
+        res = spiffs_object_read(fd, fd->fdoffset, len, (u8_t *)buf);
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+    fd->fdoffset += len;
+
+    SPIFFS_UNLOCK(fs);
+
+    return len;
+}
+
+s32_t SPIFFS_read(spiffs *fs, spiffs_file fh, void *buf, s32_t len) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd " "_SPIPRIi "\n", __func__, fh, len);
+    s32_t res = spiffs_hydro_read(fs, fh, buf, len);
+    if (res == SPIFFS_ERR_END_OF_OBJECT) {
+        res = 0;
+    }
+    return res;
+}
+
+
+#if !SPIFFS_READ_ONLY
+static s32_t spiffs_hydro_write(spiffs *fs, spiffs_fd *fd, void *buf, u32_t offset, s32_t len) {
+    (void)fs;
+    s32_t res = SPIFFS_OK;
+    s32_t remaining = len;
+    if (fd->size != SPIFFS_UNDEFINED_LEN && offset < fd->size) {
+        s32_t m_len = MIN((s32_t)(fd->size - offset), len);
+        res = spiffs_object_modify(fd, offset, (u8_t *)buf, m_len);
+        SPIFFS_CHECK_RES(res);
+        remaining -= m_len;
+        u8_t *buf_8 = (u8_t *)buf;
+        buf_8 += m_len;
+        buf = buf_8;
+        offset += m_len;
+    }
+    if (remaining > 0) {
+        res = spiffs_object_append(fd, offset, (u8_t *)buf, remaining);
+        SPIFFS_CHECK_RES(res);
+    }
+    return len;
+
+}
+#endif // !SPIFFS_READ_ONLY
+
+s32_t SPIFFS_write(spiffs *fs, spiffs_file fh, void *buf, s32_t len) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd " "_SPIPRIi "\n", __func__, fh, len);
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    (void)fh;
+    (void)buf;
+    (void)len;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    spiffs_fd *fd;
+    s32_t res;
+    u32_t offset;
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    if ((fd->flags & SPIFFS_O_WRONLY) == 0) {
+        res = SPIFFS_ERR_NOT_WRITABLE;
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+
+    if ((fd->flags & SPIFFS_O_APPEND)) {
+        fd->fdoffset = fd->size == SPIFFS_UNDEFINED_LEN ? 0 : fd->size;
+    }
+    offset = fd->fdoffset;
+
+#if SPIFFS_CACHE_WR
+    if (fd->cache_page == 0) {
+        // see if object id is associated with cache already
+        fd->cache_page = spiffs_cache_page_get_by_fd(fs, fd);
+    }
+#endif
+    if (fd->flags & SPIFFS_O_APPEND) {
+        if (fd->size == SPIFFS_UNDEFINED_LEN) {
+            offset = 0;
+        } else {
+            offset = fd->size;
+        }
+#if SPIFFS_CACHE_WR
+        if (fd->cache_page) {
+            offset = MAX(offset, fd->cache_page->ucache.swrc.offset + fd->cache_page->ucache.swrc.size);
+        }
+#endif
+    }
+
+#if SPIFFS_CACHE_WR
+    if ((fd->flags & SPIFFS_O_DIRECT) == 0) {
+        if (len < (s32_t)SPIFFS_CFG_LOG_PAGE_SZ(fs)) {
+            // small write, try to cache it
+            u8_t alloc_cpage = 1;
+            if (fd->cache_page) {
+                // have a cached page for this fd already, check cache page boundaries
+                if (offset < fd->cache_page->ucache.swrc.offset || // writing before cache
+                        offset > fd->cache_page->ucache.swrc.offset + fd->cache_page->ucache.swrc.size || // writing after cache
+                        offset + len > fd->cache_page->ucache.swrc.offset + SPIFFS_CFG_LOG_PAGE_SZ(fs)) { // writing beyond cache page
+                    // boundary violation, write back cache first and allocate new
+                    SPIFFS_CACHE_DBG("CACHE_WR_DUMP: dumping cache page "_SPIPRIi" for fd "_SPIPRIfd":"_SPIPRIid", boundary viol, offs:"_SPIPRIi" size:"_SPIPRIi"\n",
+                                     fd->cache_page->ix, fd->file_nbr, fd->obj_id, fd->cache_page->ucache.swrc.offset, fd->cache_page->ucache.swrc.size);
+                    res = spiffs_hydro_write(fs, fd,
+                                             spiffs_get_cache_page(fs, spiffs_get_cache(fs), fd->cache_page->ix),
+                                             fd->cache_page->ucache.swrc.offset, fd->cache_page->ucache.swrc.size);
+                    spiffs_cache_fd_release(fs, fd->cache_page);
+                    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+                } else {
+                    // writing within cache
+                    alloc_cpage = 0;
+                }
+            }
+
+            if (alloc_cpage) {
+                fd->cache_page = spiffs_cache_page_allocate_by_fd(fs, fd);
+                if (fd->cache_page) {
+                    fd->cache_page->ucache.swrc.offset = offset;
+                    fd->cache_page->ucache.swrc.size = 0;
+                    SPIFFS_CACHE_DBG("CACHE_WR_ALLO: allocating cache page "_SPIPRIi" for fd "_SPIPRIfd":"_SPIPRIid"\n",
+                                     fd->cache_page->ix, fd->file_nbr, fd->obj_id);
+                }
+            }
+
+            if (fd->cache_page) {
+                u32_t offset_in_cpage = offset - fd->cache_page->ucache.swrc.offset;
+                SPIFFS_CACHE_DBG("CACHE_WR_WRITE: storing to cache page "_SPIPRIi" for fd "_SPIPRIfd":"_SPIPRIid", offs "_SPIPRIi":"_SPIPRIi" len "_SPIPRIi"\n",
+                                 fd->cache_page->ix, fd->file_nbr, fd->obj_id,
+                                 offset, offset_in_cpage, len);
+                spiffs_cache *cache = spiffs_get_cache(fs);
+                u8_t *cpage_data = spiffs_get_cache_page(fs, cache, fd->cache_page->ix);
+#ifdef _SPIFFS_TEST
+                {
+                    intptr_t __a1 = (u8_t *)&cpage_data[offset_in_cpage] - (u8_t *)cache;
+                    intptr_t __a2 = (u8_t *)&cpage_data[offset_in_cpage] + len - (u8_t *)cache;
+                    intptr_t __b = sizeof(spiffs_cache) + cache->cpage_count * (sizeof(spiffs_cache_page) + SPIFFS_CFG_LOG_PAGE_SZ(fs));
+                    if (__a1 > __b || __a2 > __b) {
+                        printf("FATAL OOB: CACHE_WR: memcpy to cache buffer ixs:%4ld..%4ld of %4ld\n", __a1, __a2, __b);
+                        ERREXIT();
+                    }
+                }
+#endif
+                _SPIFFS_MEMCPY(&cpage_data[offset_in_cpage], buf, len);
+                fd->cache_page->ucache.swrc.size = MAX(fd->cache_page->ucache.swrc.size, offset_in_cpage + len);
+                fd->fdoffset += len;
+                SPIFFS_UNLOCK(fs);
+                return len;
+            } else {
+                res = spiffs_hydro_write(fs, fd, buf, offset, len);
+                SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+                fd->fdoffset += len;
+                SPIFFS_UNLOCK(fs);
+                return res;
+            }
+        } else {
+            // big write, no need to cache it - but first check if there is a cached write already
+            if (fd->cache_page) {
+                // write back cache first
+                SPIFFS_CACHE_DBG("CACHE_WR_DUMP: dumping cache page "_SPIPRIi" for fd "_SPIPRIfd":"_SPIPRIid", big write, offs:"_SPIPRIi" size:"_SPIPRIi"\n",
+                                 fd->cache_page->ix, fd->file_nbr, fd->obj_id, fd->cache_page->ucache.swrc.offset, fd->cache_page->ucache.swrc.size);
+                res = spiffs_hydro_write(fs, fd,
+                                         spiffs_get_cache_page(fs, spiffs_get_cache(fs), fd->cache_page->ix),
+                                         fd->cache_page->ucache.swrc.offset, fd->cache_page->ucache.swrc.size);
+                spiffs_cache_fd_release(fs, fd->cache_page);
+                SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+                // data written below
+            }
+        }
+    }
+#endif
+
+    res = spiffs_hydro_write(fs, fd, buf, offset, len);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    fd->fdoffset += len;
+
+    SPIFFS_UNLOCK(fs);
+
+    return res;
+#endif // SPIFFS_READ_ONLY
+}
+
+s32_t SPIFFS_lseek(spiffs *fs, spiffs_file fh, s32_t offs, int whence) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd " "_SPIPRIi " %s\n", __func__, fh, offs, (const char *[]) {"SET", "CUR", "END", "???"}[MIN(whence, 3)]);
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    spiffs_fd *fd;
+    s32_t res;
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+#if SPIFFS_CACHE_WR
+    spiffs_fflush_cache(fs, fh);
+#endif
+
+    s32_t file_size = fd->size == SPIFFS_UNDEFINED_LEN ? 0 : fd->size;
+
+    switch (whence) {
+        case SPIFFS_SEEK_CUR:
+            offs = fd->fdoffset + offs;
+            break;
+        case SPIFFS_SEEK_END:
+            offs = file_size + offs;
+            break;
+    }
+    if (offs < 0) {
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, SPIFFS_ERR_SEEK_BOUNDS);
+    }
+    if (offs > file_size) {
+        fd->fdoffset = file_size;
+        res = SPIFFS_ERR_END_OF_OBJECT;
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    spiffs_span_ix data_spix = (offs > 0 ? (offs - 1) : 0) / SPIFFS_DATA_PAGE_SIZE(fs);
+    spiffs_span_ix objix_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, data_spix);
+    if (fd->cursor_objix_spix != objix_spix) {
+        spiffs_page_ix pix;
+        res = spiffs_obj_lu_find_id_and_span(
+                  fs, fd->obj_id | SPIFFS_OBJ_ID_IX_FLAG, objix_spix, 0, &pix);
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+        fd->cursor_objix_spix = objix_spix;
+        fd->cursor_objix_pix = pix;
+    }
+    fd->fdoffset = offs;
+
+    SPIFFS_UNLOCK(fs);
+
+    return offs;
+}
+
+s32_t SPIFFS_remove(spiffs *fs, const char *path) {
+    SPIFFS_API_DBG("%s '%s'\n", __func__, path);
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    (void)path;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    if (strlen(path) > SPIFFS_OBJ_NAME_LEN - 1) {
+        SPIFFS_API_CHECK_RES(fs, SPIFFS_ERR_NAME_TOO_LONG);
+    }
+    SPIFFS_LOCK(fs);
+
+    spiffs_fd *fd;
+    spiffs_page_ix pix;
+    s32_t res;
+
+    res = spiffs_fd_find_new(fs, &fd, 0);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_find_object_index_header_by_name(fs, (const u8_t *)path, &pix);
+    if (res != SPIFFS_OK) {
+        spiffs_fd_return(fs, fd->file_nbr);
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_open_by_page(fs, pix, fd, 0, 0);
+    if (res != SPIFFS_OK) {
+        spiffs_fd_return(fs, fd->file_nbr);
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_truncate(fd, 0, 1);
+    if (res != SPIFFS_OK) {
+        spiffs_fd_return(fs, fd->file_nbr);
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    SPIFFS_UNLOCK(fs);
+    return 0;
+#endif // SPIFFS_READ_ONLY
+}
+
+s32_t SPIFFS_fremove(spiffs *fs, spiffs_file fh) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd "\n", __func__, fh);
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    (void)fh;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    spiffs_fd *fd;
+    s32_t res;
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    if ((fd->flags & SPIFFS_O_WRONLY) == 0) {
+        res = SPIFFS_ERR_NOT_WRITABLE;
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+
+#if SPIFFS_CACHE_WR
+    spiffs_cache_fd_release(fs, fd->cache_page);
+#endif
+
+    res = spiffs_object_truncate(fd, 0, 1);
+
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    SPIFFS_UNLOCK(fs);
+
+    return 0;
+#endif // SPIFFS_READ_ONLY
+}
+
+static s32_t spiffs_stat_pix(spiffs *fs, spiffs_page_ix pix, spiffs_file fh, spiffs_stat *s) {
+    (void)fh;
+    spiffs_page_object_ix_header objix_hdr;
+    spiffs_obj_id obj_id;
+    s32_t res = _spiffs_rd(fs,  SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ, fh,
+                           SPIFFS_PAGE_TO_PADDR(fs, pix), sizeof(spiffs_page_object_ix_header), (u8_t *)&objix_hdr);
+    SPIFFS_API_CHECK_RES(fs, res);
+
+    u32_t obj_id_addr = SPIFFS_BLOCK_TO_PADDR(fs, SPIFFS_BLOCK_FOR_PAGE(fs, pix)) +
+                        SPIFFS_OBJ_LOOKUP_ENTRY_FOR_PAGE(fs, pix) * sizeof(spiffs_obj_id);
+    res = _spiffs_rd(fs,  SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ, fh,
+                     obj_id_addr, sizeof(spiffs_obj_id), (u8_t *)&obj_id);
+    SPIFFS_API_CHECK_RES(fs, res);
+
+    s->obj_id = obj_id & ~SPIFFS_OBJ_ID_IX_FLAG;
+    s->type = objix_hdr.type;
+    s->size = objix_hdr.size == SPIFFS_UNDEFINED_LEN ? 0 : objix_hdr.size;
+    s->pix = pix;
+    strncpy((char *)s->name, (char *)objix_hdr.name, SPIFFS_OBJ_NAME_LEN);
+#if SPIFFS_OBJ_META_LEN
+    _SPIFFS_MEMCPY(s->meta, objix_hdr.meta, SPIFFS_OBJ_META_LEN);
+#endif
+
+    return res;
+}
+
+s32_t SPIFFS_stat(spiffs *fs, const char *path, spiffs_stat *s) {
+    SPIFFS_API_DBG("%s '%s'\n", __func__, path);
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    if (strlen(path) > SPIFFS_OBJ_NAME_LEN - 1) {
+        SPIFFS_API_CHECK_RES(fs, SPIFFS_ERR_NAME_TOO_LONG);
+    }
+    SPIFFS_LOCK(fs);
+
+    s32_t res;
+    spiffs_page_ix pix;
+
+    res = spiffs_object_find_object_index_header_by_name(fs, (const u8_t *)path, &pix);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_stat_pix(fs, pix, 0, s);
+
+    SPIFFS_UNLOCK(fs);
+
+    return res;
+}
+
+s32_t SPIFFS_fstat(spiffs *fs, spiffs_file fh, spiffs_stat *s) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd "\n", __func__, fh);
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    spiffs_fd *fd;
+    s32_t res;
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+#if SPIFFS_CACHE_WR
+    spiffs_fflush_cache(fs, fh);
+#endif
+
+    res = spiffs_stat_pix(fs, fd->objix_hdr_pix, fh, s);
+
+    SPIFFS_UNLOCK(fs);
+
+    return res;
+}
+
+// Checks if there are any cached writes for the object id associated with
+// given filehandle. If so, these writes are flushed.
+#if SPIFFS_CACHE == 1
+static s32_t spiffs_fflush_cache(spiffs *fs, spiffs_file fh) {
+    (void)fs;
+    (void)fh;
+    s32_t res = SPIFFS_OK;
+#if !SPIFFS_READ_ONLY && SPIFFS_CACHE_WR
+
+    spiffs_fd *fd;
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES(fs, res);
+
+    if ((fd->flags & SPIFFS_O_DIRECT) == 0) {
+        if (fd->cache_page == 0) {
+            // see if object id is associated with cache already
+            fd->cache_page = spiffs_cache_page_get_by_fd(fs, fd);
+        }
+        if (fd->cache_page) {
+            SPIFFS_CACHE_DBG("CACHE_WR_DUMP: dumping cache page "_SPIPRIi" for fd "_SPIPRIfd":"_SPIPRIid", flush, offs:"_SPIPRIi" size:"_SPIPRIi"\n",
+                             fd->cache_page->ix, fd->file_nbr,  fd->obj_id, fd->cache_page->ucache.swrc.offset, fd->cache_page->ucache.swrc.size);
+            res = spiffs_hydro_write(fs, fd,
+                                     spiffs_get_cache_page(fs, spiffs_get_cache(fs), fd->cache_page->ix),
+                                     fd->cache_page->ucache.swrc.offset, fd->cache_page->ucache.swrc.size);
+            if (res < SPIFFS_OK) {
+                fs->err_code = res;
+            }
+            spiffs_cache_fd_release(fs, fd->cache_page);
+        }
+    }
+#endif
+
+    return res;
+}
+#endif
+
+s32_t SPIFFS_fflush(spiffs *fs, spiffs_file fh) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd "\n", __func__, fh);
+    (void)fh;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    s32_t res = SPIFFS_OK;
+#if !SPIFFS_READ_ONLY && SPIFFS_CACHE_WR
+    SPIFFS_LOCK(fs);
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+    res = spiffs_fflush_cache(fs, fh);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    SPIFFS_UNLOCK(fs);
+#endif
+
+    return res;
+}
+
+s32_t SPIFFS_close(spiffs *fs, spiffs_file fh) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd "\n", __func__, fh);
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+
+    s32_t res = SPIFFS_OK;
+    SPIFFS_LOCK(fs);
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+#if SPIFFS_CACHE
+    res = spiffs_fflush_cache(fs, fh);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+#endif
+    res = spiffs_fd_return(fs, fh);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    SPIFFS_UNLOCK(fs);
+
+    return res;
+}
+
+s32_t SPIFFS_rename(spiffs *fs, const char *old_path, const char *new_path) {
+    SPIFFS_API_DBG("%s %s %s\n", __func__, old_path, new_path);
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    (void)old_path;
+    (void)new_path;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    if (strlen(new_path) > SPIFFS_OBJ_NAME_LEN - 1 ||
+            strlen(old_path) > SPIFFS_OBJ_NAME_LEN - 1) {
+        SPIFFS_API_CHECK_RES(fs, SPIFFS_ERR_NAME_TOO_LONG);
+    }
+    SPIFFS_LOCK(fs);
+
+    spiffs_page_ix pix_old, pix_dummy;
+    spiffs_fd *fd;
+
+    s32_t res = spiffs_object_find_object_index_header_by_name(fs, (const u8_t *)old_path, &pix_old);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_find_object_index_header_by_name(fs, (const u8_t *)new_path, &pix_dummy);
+    if (res == SPIFFS_ERR_NOT_FOUND) {
+        res = SPIFFS_OK;
+    } else if (res == SPIFFS_OK) {
+        res = SPIFFS_ERR_CONFLICTING_NAME;
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_fd_find_new(fs, &fd, 0);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_open_by_page(fs, pix_old, fd, 0, 0);
+    if (res != SPIFFS_OK) {
+        spiffs_fd_return(fs, fd->file_nbr);
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id, fd->objix_hdr_pix, 0, (const u8_t *)new_path,
+                                         0, 0, &pix_dummy);
+#if SPIFFS_TEMPORAL_FD_CACHE
+    if (res == SPIFFS_OK) {
+        spiffs_fd_temporal_cache_rehash(fs, old_path, new_path);
+    }
+#endif
+
+    spiffs_fd_return(fs, fd->file_nbr);
+
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    SPIFFS_UNLOCK(fs);
+
+    return res;
+#endif // SPIFFS_READ_ONLY
+}
+
+#if SPIFFS_OBJ_META_LEN
+s32_t SPIFFS_update_meta(spiffs *fs, const char *name, const void *meta) {
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    (void)name;
+    (void)meta;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    spiffs_page_ix pix, pix_dummy;
+    spiffs_fd *fd;
+
+    s32_t res = spiffs_object_find_object_index_header_by_name(fs, (const u8_t *)name, &pix);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_fd_find_new(fs, &fd, 0);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_open_by_page(fs, pix, fd, 0, 0);
+    if (res != SPIFFS_OK) {
+        spiffs_fd_return(fs, fd->file_nbr);
+    }
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id, fd->objix_hdr_pix, 0, 0, meta,
+                                         0, &pix_dummy);
+
+    spiffs_fd_return(fs, fd->file_nbr);
+
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    SPIFFS_UNLOCK(fs);
+
+    return res;
+#endif // SPIFFS_READ_ONLY
+}
+
+s32_t SPIFFS_fupdate_meta(spiffs *fs, spiffs_file fh, const void *meta) {
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    (void)fh;
+    (void)meta;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    s32_t res;
+    spiffs_fd *fd;
+    spiffs_page_ix pix_dummy;
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    if ((fd->flags & SPIFFS_O_WRONLY) == 0) {
+        res = SPIFFS_ERR_NOT_WRITABLE;
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    }
+
+    res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id, fd->objix_hdr_pix, 0, 0, meta,
+                                         0, &pix_dummy);
+
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    SPIFFS_UNLOCK(fs);
+
+    return res;
+#endif // SPIFFS_READ_ONLY
+}
+#endif // SPIFFS_OBJ_META_LEN
+
+spiffs_DIR *SPIFFS_opendir(spiffs *fs, const char *name, spiffs_DIR *d) {
+    SPIFFS_API_DBG("%s\n", __func__);
+    (void)name;
+
+    if (!SPIFFS_CHECK_CFG((fs))) {
+        (fs)->err_code = SPIFFS_ERR_NOT_CONFIGURED;
+        return 0;
+    }
+
+    if (!SPIFFS_CHECK_MOUNT(fs)) {
+        fs->err_code = SPIFFS_ERR_NOT_MOUNTED;
+        return 0;
+    }
+
+    d->fs = fs;
+    d->block = 0;
+    d->entry = 0;
+    return d;
+}
+
+static s32_t spiffs_read_dir_v(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_block_ix bix,
+    int ix_entry,
+    const void *user_const_p,
+    void *user_var_p) {
+    (void)user_const_p;
+    s32_t res;
+    spiffs_page_object_ix_header objix_hdr;
+    if (obj_id == SPIFFS_OBJ_ID_FREE || obj_id == SPIFFS_OBJ_ID_DELETED ||
+            (obj_id & SPIFFS_OBJ_ID_IX_FLAG) == 0) {
+        return SPIFFS_VIS_COUNTINUE;
+    }
+
+    spiffs_page_ix pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, ix_entry);
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                     0, SPIFFS_PAGE_TO_PADDR(fs, pix), sizeof(spiffs_page_object_ix_header), (u8_t *)&objix_hdr);
+    if (res != SPIFFS_OK) return res;
+    if ((obj_id & SPIFFS_OBJ_ID_IX_FLAG) &&
+            objix_hdr.p_hdr.span_ix == 0 &&
+            (objix_hdr.p_hdr.flags & (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_IXDELE)) ==
+            (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_IXDELE)) {
+        struct spiffs_dirent *e = (struct spiffs_dirent *)user_var_p;
+        e->obj_id = obj_id;
+        strcpy((char *)e->name, (char *)objix_hdr.name);
+        e->type = objix_hdr.type;
+        e->size = objix_hdr.size == SPIFFS_UNDEFINED_LEN ? 0 : objix_hdr.size;
+        e->pix = pix;
+#if SPIFFS_OBJ_META_LEN
+        _SPIFFS_MEMCPY(e->meta, objix_hdr.meta, SPIFFS_OBJ_META_LEN);
+#endif
+        return SPIFFS_OK;
+    }
+    return SPIFFS_VIS_COUNTINUE;
+}
+
+struct spiffs_dirent *SPIFFS_readdir(spiffs_DIR *d, struct spiffs_dirent *e) {
+    SPIFFS_API_DBG("%s\n", __func__);
+    if (!SPIFFS_CHECK_MOUNT(d->fs)) {
+        d->fs->err_code = SPIFFS_ERR_NOT_MOUNTED;
+        return 0;
+    }
+    SPIFFS_LOCK(d->fs);
+
+    spiffs_block_ix bix;
+    int entry;
+    s32_t res;
+    struct spiffs_dirent *ret = 0;
+
+    res = spiffs_obj_lu_find_entry_visitor(d->fs,
+                                           d->block,
+                                           d->entry,
+                                           SPIFFS_VIS_NO_WRAP,
+                                           0,
+                                           spiffs_read_dir_v,
+                                           0,
+                                           e,
+                                           &bix,
+                                           &entry);
+    if (res == SPIFFS_OK) {
+        d->block = bix;
+        d->entry = entry + 1;
+        e->obj_id &= ~SPIFFS_OBJ_ID_IX_FLAG;
+        ret = e;
+    } else {
+        d->fs->err_code = res;
+    }
+    SPIFFS_UNLOCK(d->fs);
+    return ret;
+}
+
+s32_t SPIFFS_closedir(spiffs_DIR *d) {
+    SPIFFS_API_DBG("%s\n", __func__);
+    SPIFFS_API_CHECK_CFG(d->fs);
+    SPIFFS_API_CHECK_MOUNT(d->fs);
+    return 0;
+}
+
+s32_t SPIFFS_check(spiffs *fs) {
+    SPIFFS_API_DBG("%s\n", __func__);
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    s32_t res;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    res = spiffs_lookup_consistency_check(fs, 0);
+
+    res = spiffs_object_index_consistency_check(fs);
+
+    res = spiffs_page_consistency_check(fs);
+
+    res = spiffs_obj_lu_scan(fs);
+
+    SPIFFS_UNLOCK(fs);
+    return res;
+#endif // SPIFFS_READ_ONLY
+}
+
+s32_t SPIFFS_info(spiffs *fs, u32_t *total, u32_t *used) {
+    SPIFFS_API_DBG("%s\n", __func__);
+    s32_t res = SPIFFS_OK;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    u32_t pages_per_block = SPIFFS_PAGES_PER_BLOCK(fs);
+    u32_t blocks = fs->block_count;
+    u32_t obj_lu_pages = SPIFFS_OBJ_LOOKUP_PAGES(fs);
+    u32_t data_page_size = SPIFFS_DATA_PAGE_SIZE(fs);
+    u32_t total_data_pages = (blocks - 2) * (pages_per_block - obj_lu_pages) + 1; // -2 for spare blocks, +1 for emergency page
+
+    if (total) {
+        *total = total_data_pages * data_page_size;
+    }
+
+    if (used) {
+        *used = fs->stats_p_allocated * data_page_size;
+    }
+
+    SPIFFS_UNLOCK(fs);
+    return res;
+}
+
+s32_t SPIFFS_gc_quick(spiffs *fs, u16_t max_free_pages) {
+    SPIFFS_API_DBG("%s "_SPIPRIi "\n", __func__, max_free_pages);
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    (void)max_free_pages;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    s32_t res;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    res = spiffs_gc_quick(fs, max_free_pages);
+
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    SPIFFS_UNLOCK(fs);
+    return 0;
+#endif // SPIFFS_READ_ONLY
+}
+
+
+s32_t SPIFFS_gc(spiffs *fs, u32_t size) {
+    SPIFFS_API_DBG("%s "_SPIPRIi "\n", __func__, size);
+#if SPIFFS_READ_ONLY
+    (void)fs;
+    (void)size;
+    return SPIFFS_ERR_RO_NOT_IMPL;
+#else
+    s32_t res;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    res = spiffs_gc_check(fs, size);
+
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+    SPIFFS_UNLOCK(fs);
+    return 0;
+#endif // SPIFFS_READ_ONLY
+}
+
+s32_t SPIFFS_eof(spiffs *fs, spiffs_file fh) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd "\n", __func__, fh);
+    s32_t res;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+
+    spiffs_fd *fd;
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+#if SPIFFS_CACHE_WR
+    res = spiffs_fflush_cache(fs, fh);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+#endif
+
+    res = (fd->fdoffset >= (fd->size == SPIFFS_UNDEFINED_LEN ? 0 : fd->size));
+
+    SPIFFS_UNLOCK(fs);
+    return res;
+}
+
+s32_t SPIFFS_tell(spiffs *fs, spiffs_file fh) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd "\n", __func__, fh);
+    s32_t res;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+
+    spiffs_fd *fd;
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+#if SPIFFS_CACHE_WR
+    res = spiffs_fflush_cache(fs, fh);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+#endif
+
+    res = fd->fdoffset;
+
+    SPIFFS_UNLOCK(fs);
+    return res;
+}
+
+s32_t SPIFFS_set_file_callback_func(spiffs *fs, spiffs_file_callback cb_func) {
+    SPIFFS_API_DBG("%s\n", __func__);
+    SPIFFS_LOCK(fs);
+    fs->file_cb_f = cb_func;
+    SPIFFS_UNLOCK(fs);
+    return 0;
+}
+
+#if SPIFFS_IX_MAP
+
+s32_t SPIFFS_ix_map(spiffs *fs,  spiffs_file fh, spiffs_ix_map *map,
+                    u32_t offset, u32_t len, spiffs_page_ix *map_buf) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd " "_SPIPRIi " "_SPIPRIi "\n", __func__, fh, offset, len);
+    s32_t res;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+
+    spiffs_fd *fd;
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    if (fd->ix_map) {
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, SPIFFS_ERR_IX_MAP_MAPPED);
+    }
+
+    map->map_buf = map_buf;
+    map->offset = offset;
+    // nb: spix range includes last
+    map->start_spix = offset / SPIFFS_DATA_PAGE_SIZE(fs);
+    map->end_spix = (offset + len) / SPIFFS_DATA_PAGE_SIZE(fs);
+    memset(map_buf, 0, sizeof(spiffs_page_ix) * (map->end_spix - map->start_spix + 1));
+    fd->ix_map = map;
+
+    // scan for pixes
+    res = spiffs_populate_ix_map(fs, fd, 0, map->end_spix - map->start_spix + 1);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    SPIFFS_UNLOCK(fs);
+    return res;
+}
+
+s32_t SPIFFS_ix_unmap(spiffs *fs,  spiffs_file fh) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd "\n", __func__, fh);
+    s32_t res;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+
+    spiffs_fd *fd;
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    if (fd->ix_map == 0) {
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, SPIFFS_ERR_IX_MAP_UNMAPPED);
+    }
+
+    fd->ix_map = 0;
+
+    SPIFFS_UNLOCK(fs);
+    return res;
+}
+
+s32_t SPIFFS_ix_remap(spiffs *fs, spiffs_file fh, u32_t offset) {
+    SPIFFS_API_DBG("%s "_SPIPRIfd " "_SPIPRIi "\n", __func__, fh, offset);
+    s32_t res = SPIFFS_OK;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    fh = SPIFFS_FH_UNOFFS(fs, fh);
+
+    spiffs_fd *fd;
+    res = spiffs_fd_get(fs, fh, &fd);
+    SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+
+    if (fd->ix_map == 0) {
+        SPIFFS_API_CHECK_RES_UNLOCK(fs, SPIFFS_ERR_IX_MAP_UNMAPPED);
+    }
+
+    spiffs_ix_map *map = fd->ix_map;
+
+    s32_t spix_diff = offset / SPIFFS_DATA_PAGE_SIZE(fs) - map->start_spix;
+    map->offset = offset;
+
+    // move existing pixes if within map offs
+    if (spix_diff != 0) {
+        // move vector
+        int i;
+        const s32_t vec_len = map->end_spix - map->start_spix + 1; // spix range includes last
+        map->start_spix += spix_diff;
+        map->end_spix += spix_diff;
+        if (spix_diff >= vec_len) {
+            // moving beyond range
+            memset(&map->map_buf, 0, vec_len * sizeof(spiffs_page_ix));
+            // populate_ix_map is inclusive
+            res = spiffs_populate_ix_map(fs, fd, 0, vec_len - 1);
+            SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+        } else if (spix_diff > 0) {
+            // diff positive
+            for (i = 0; i < vec_len - spix_diff; i++) {
+                map->map_buf[i] = map->map_buf[i + spix_diff];
+            }
+            // memset is non-inclusive
+            memset(&map->map_buf[vec_len - spix_diff], 0, spix_diff * sizeof(spiffs_page_ix));
+            // populate_ix_map is inclusive
+            res = spiffs_populate_ix_map(fs, fd, vec_len - spix_diff, vec_len - 1);
+            SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+        } else {
+            // diff negative
+            for (i = vec_len - 1; i >= -spix_diff; i--) {
+                map->map_buf[i] = map->map_buf[i + spix_diff];
+            }
+            // memset is non-inclusive
+            memset(&map->map_buf[0], 0, -spix_diff * sizeof(spiffs_page_ix));
+            // populate_ix_map is inclusive
+            res = spiffs_populate_ix_map(fs, fd, 0, -spix_diff - 1);
+            SPIFFS_API_CHECK_RES_UNLOCK(fs, res);
+        }
+
+    }
+
+    SPIFFS_UNLOCK(fs);
+    return res;
+}
+
+s32_t SPIFFS_bytes_to_ix_map_entries(spiffs *fs, u32_t bytes) {
+    SPIFFS_API_CHECK_CFG(fs);
+    // always add one extra page, the offset might change to the middle of a page
+    return (bytes + SPIFFS_DATA_PAGE_SIZE(fs)) / SPIFFS_DATA_PAGE_SIZE(fs);
+}
+
+s32_t SPIFFS_ix_map_entries_to_bytes(spiffs *fs, u32_t map_page_ix_entries) {
+    SPIFFS_API_CHECK_CFG(fs);
+    return map_page_ix_entries * SPIFFS_DATA_PAGE_SIZE(fs);
+}
+
+#endif // SPIFFS_IX_MAP
+
+#if SPIFFS_TEST_VISUALISATION
+s32_t SPIFFS_vis(spiffs *fs) {
+    s32_t res = SPIFFS_OK;
+    SPIFFS_API_CHECK_CFG(fs);
+    SPIFFS_API_CHECK_MOUNT(fs);
+    SPIFFS_LOCK(fs);
+
+    int entries_per_page = (SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(spiffs_obj_id));
+    spiffs_obj_id *obj_lu_buf = (spiffs_obj_id *)fs->lu_work;
+    spiffs_block_ix bix = 0;
+
+    while (bix < fs->block_count) {
+        // check each object lookup page
+        int obj_lookup_page = 0;
+        int cur_entry = 0;
+
+        while (res == SPIFFS_OK && obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs)) {
+            int entry_offset = obj_lookup_page * entries_per_page;
+            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                             0, bix * SPIFFS_CFG_LOG_BLOCK_SZ(fs) + SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+            // check each entry
+            while (res == SPIFFS_OK &&
+                    cur_entry - entry_offset < entries_per_page && cur_entry < (int)(SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs))) {
+                spiffs_obj_id obj_id = obj_lu_buf[cur_entry - entry_offset];
+                if (cur_entry == 0) {
+                    spiffs_printf(_SPIPRIbl" ", bix);
+                } else if ((cur_entry & 0x3f) == 0) {
+                    spiffs_printf("     ");
+                }
+                if (obj_id == SPIFFS_OBJ_ID_FREE) {
+                    spiffs_printf(SPIFFS_TEST_VIS_FREE_STR);
+                } else if (obj_id == SPIFFS_OBJ_ID_DELETED) {
+                    spiffs_printf(SPIFFS_TEST_VIS_DELE_STR);
+                } else if (obj_id & SPIFFS_OBJ_ID_IX_FLAG) {
+                    spiffs_printf(SPIFFS_TEST_VIS_INDX_STR(obj_id));
+                } else {
+                    spiffs_printf(SPIFFS_TEST_VIS_DATA_STR(obj_id));
+                }
+                cur_entry++;
+                if ((cur_entry & 0x3f) == 0) {
+                    spiffs_printf("\n");
+                }
+            } // per entry
+            obj_lookup_page++;
+        } // per object lookup page
+
+        spiffs_obj_id erase_count;
+        res = _spiffs_rd(fs, SPIFFS_OP_C_READ | SPIFFS_OP_T_OBJ_LU2, 0,
+                         SPIFFS_ERASE_COUNT_PADDR(fs, bix),
+                         sizeof(spiffs_obj_id), (u8_t *)&erase_count);
+        SPIFFS_CHECK_RES(res);
+
+        if (erase_count != (spiffs_obj_id) - 1) {
+            spiffs_printf("\tera_cnt: "_SPIPRIi"\n", erase_count);
+        } else {
+            spiffs_printf("\tera_cnt: N/A\n");
+        }
+
+        bix++;
+    } // per block
+
+    spiffs_printf("era_cnt_max: "_SPIPRIi"\n", fs->max_erase_count);
+    spiffs_printf("last_errno:  "_SPIPRIi"\n", fs->err_code);
+    spiffs_printf("blocks:      "_SPIPRIi"\n", fs->block_count);
+    spiffs_printf("free_blocks: "_SPIPRIi"\n", fs->free_blocks);
+    spiffs_printf("page_alloc:  "_SPIPRIi"\n", fs->stats_p_allocated);
+    spiffs_printf("page_delet:  "_SPIPRIi"\n", fs->stats_p_deleted);
+    SPIFFS_UNLOCK(fs);
+    u32_t total, used;
+    SPIFFS_info(fs, &total, &used);
+    spiffs_printf("used:        "_SPIPRIi" of "_SPIPRIi"\n", used, total);
+    return res;
+}
+#endif
diff --git a/armsrc/spiffs_nucleus.c b/armsrc/spiffs_nucleus.c
new file mode 100644
index 000000000..7951a7952
--- /dev/null
+++ b/armsrc/spiffs_nucleus.c
@@ -0,0 +1,2364 @@
+#include "spiffs.h"
+#include "spiffs_nucleus.h"
+#include "printf.h"
+
+static s32_t spiffs_page_data_check(spiffs *fs, spiffs_fd *fd, spiffs_page_ix pix, spiffs_span_ix spix) {
+    s32_t res = SPIFFS_OK;
+    if (pix == (spiffs_page_ix) - 1) {
+        // referring to page 0xffff...., bad object index
+        return SPIFFS_ERR_INDEX_REF_FREE;
+    }
+    if (pix % SPIFFS_PAGES_PER_BLOCK(fs) < SPIFFS_OBJ_LOOKUP_PAGES(fs)) {
+        // referring to an object lookup page, bad object index
+        return SPIFFS_ERR_INDEX_REF_LU;
+    }
+    if (pix > SPIFFS_MAX_PAGES(fs)) {
+        // referring to a bad page
+        return SPIFFS_ERR_INDEX_REF_INVALID;
+    }
+#if SPIFFS_PAGE_CHECK
+    spiffs_page_header ph;
+    res = _spiffs_rd(
+              fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_READ,
+              fd->file_nbr,
+              SPIFFS_PAGE_TO_PADDR(fs, pix),
+              sizeof(spiffs_page_header),
+              (u8_t *)&ph);
+    SPIFFS_CHECK_RES(res);
+    SPIFFS_VALIDATE_DATA(ph, fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG, spix);
+#endif
+    return res;
+}
+
+#if !SPIFFS_READ_ONLY
+static s32_t spiffs_page_index_check(spiffs *fs, spiffs_fd *fd, spiffs_page_ix pix, spiffs_span_ix spix) {
+    s32_t res = SPIFFS_OK;
+    if (pix == (spiffs_page_ix) - 1) {
+        // referring to page 0xffff...., bad object index
+        return SPIFFS_ERR_INDEX_FREE;
+    }
+    if (pix % SPIFFS_PAGES_PER_BLOCK(fs) < SPIFFS_OBJ_LOOKUP_PAGES(fs)) {
+        // referring to an object lookup page, bad object index
+        return SPIFFS_ERR_INDEX_LU;
+    }
+    if (pix > SPIFFS_MAX_PAGES(fs)) {
+        // referring to a bad page
+        return SPIFFS_ERR_INDEX_INVALID;
+    }
+#if SPIFFS_PAGE_CHECK
+    spiffs_page_header ph;
+    res = _spiffs_rd(
+              fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ,
+              fd->file_nbr,
+              SPIFFS_PAGE_TO_PADDR(fs, pix),
+              sizeof(spiffs_page_header),
+              (u8_t *)&ph);
+    SPIFFS_CHECK_RES(res);
+    SPIFFS_VALIDATE_OBJIX(ph, fd->obj_id, spix);
+#endif
+    return res;
+}
+#endif // !SPIFFS_READ_ONLY
+
+#if !SPIFFS_CACHE
+
+s32_t spiffs_phys_rd(
+    spiffs *fs,
+    u32_t addr,
+    u32_t len,
+    u8_t *dst) {
+    return SPIFFS_HAL_READ(fs, addr, len, dst);
+}
+
+s32_t spiffs_phys_wr(
+    spiffs *fs,
+    u32_t addr,
+    u32_t len,
+    u8_t *src) {
+    return SPIFFS_HAL_WRITE(fs, addr, len, src);
+}
+
+#endif
+
+#if !SPIFFS_READ_ONLY
+s32_t spiffs_phys_cpy(
+    spiffs *fs,
+    spiffs_file fh,
+    u32_t dst,
+    u32_t src,
+    u32_t len) {
+    (void)fh;
+    s32_t res;
+    u8_t b[SPIFFS_COPY_BUFFER_STACK];
+    while (len > 0) {
+        u32_t chunk_size = MIN(SPIFFS_COPY_BUFFER_STACK, len);
+        res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_MOVS, fh, src, chunk_size, b);
+        SPIFFS_CHECK_RES(res);
+        res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_MOVD,  fh, dst, chunk_size, b);
+        SPIFFS_CHECK_RES(res);
+        len -= chunk_size;
+        src += chunk_size;
+        dst += chunk_size;
+    }
+    return SPIFFS_OK;
+}
+#endif // !SPIFFS_READ_ONLY
+
+// Find object lookup entry containing given id with visitor.
+// Iterate over object lookup pages in each block until a given object id entry is found.
+// When found, the visitor function is called with block index, entry index and user data.
+// If visitor returns SPIFFS_VIS_CONTINUE, the search goes on. Otherwise, the search will be
+// ended and visitor's return code is returned to caller.
+// If no visitor is given (0) the search returns on first entry with matching object id.
+// If no match is found in all look up, SPIFFS_VIS_END is returned.
+// @param fs                    the file system
+// @param starting_block        the starting block to start search in
+// @param starting_lu_entry     the look up index entry to start search in
+// @param flags                 ored combination of SPIFFS_VIS_CHECK_ID, SPIFFS_VIS_CHECK_PH,
+//                              SPIFFS_VIS_NO_WRAP
+// @param obj_id                argument object id
+// @param v                     visitor callback function
+// @param user_const_p          any const pointer, passed to the callback visitor function
+// @param user_var_p            any pointer, passed to the callback visitor function
+// @param block_ix              reported block index where match was found
+// @param lu_entry              reported look up index where match was found
+s32_t spiffs_obj_lu_find_entry_visitor(
+    spiffs *fs,
+    spiffs_block_ix starting_block,
+    int starting_lu_entry,
+    u8_t flags,
+    spiffs_obj_id obj_id,
+    spiffs_visitor_f v,
+    const void *user_const_p,
+    void *user_var_p,
+    spiffs_block_ix *block_ix,
+    int *lu_entry) {
+    s32_t res = SPIFFS_OK;
+    s32_t entry_count = fs->block_count * SPIFFS_OBJ_LOOKUP_MAX_ENTRIES(fs);
+    spiffs_block_ix cur_block = starting_block;
+    u32_t cur_block_addr = starting_block * SPIFFS_CFG_LOG_BLOCK_SZ(fs);
+
+    spiffs_obj_id *obj_lu_buf = (spiffs_obj_id *)fs->lu_work;
+    int cur_entry = starting_lu_entry;
+    int entries_per_page = (SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(spiffs_obj_id));
+
+    // wrap initial
+    if (cur_entry > (int)SPIFFS_OBJ_LOOKUP_MAX_ENTRIES(fs) - 1) {
+        cur_entry = 0;
+        cur_block++;
+        cur_block_addr = cur_block * SPIFFS_CFG_LOG_BLOCK_SZ(fs);
+        if (cur_block >= fs->block_count) {
+            if (flags & SPIFFS_VIS_NO_WRAP) {
+                return SPIFFS_VIS_END;
+            } else {
+                // block wrap
+                cur_block = 0;
+                cur_block_addr = 0;
+            }
+        }
+    }
+
+    // check each block
+    while (res == SPIFFS_OK && entry_count > 0) {
+        int obj_lookup_page = cur_entry / entries_per_page;
+        // check each object lookup page
+        while (res == SPIFFS_OK && obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs)) {
+            int entry_offset = obj_lookup_page * entries_per_page;
+            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                             0, cur_block_addr + SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+            // check each entry
+            while (res == SPIFFS_OK &&
+                    cur_entry - entry_offset < entries_per_page && // for non-last obj lookup pages
+                    cur_entry < (int)SPIFFS_OBJ_LOOKUP_MAX_ENTRIES(fs)) { // for last obj lookup page
+                if ((flags & SPIFFS_VIS_CHECK_ID) == 0 || obj_lu_buf[cur_entry - entry_offset] == obj_id) {
+                    if (block_ix) *block_ix = cur_block;
+                    if (lu_entry) *lu_entry = cur_entry;
+                    if (v) {
+                        res = v(
+                                  fs,
+                                  (flags & SPIFFS_VIS_CHECK_PH) ? obj_id : obj_lu_buf[cur_entry - entry_offset],
+                                  cur_block,
+                                  cur_entry,
+                                  user_const_p,
+                                  user_var_p);
+                        if (res == SPIFFS_VIS_COUNTINUE || res == SPIFFS_VIS_COUNTINUE_RELOAD) {
+                            if (res == SPIFFS_VIS_COUNTINUE_RELOAD) {
+                                res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                                                 0, cur_block_addr + SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->lu_work);
+                                SPIFFS_CHECK_RES(res);
+                            }
+                            res = SPIFFS_OK;
+                            cur_entry++;
+                            entry_count--;
+                            continue;
+                        } else {
+                            return res;
+                        }
+                    } else {
+                        return SPIFFS_OK;
+                    }
+                }
+                entry_count--;
+                cur_entry++;
+            } // per entry
+            obj_lookup_page++;
+        } // per object lookup page
+        cur_entry = 0;
+        cur_block++;
+        cur_block_addr += SPIFFS_CFG_LOG_BLOCK_SZ(fs);
+        if (cur_block >= fs->block_count) {
+            if (flags & SPIFFS_VIS_NO_WRAP) {
+                return SPIFFS_VIS_END;
+            } else {
+                // block wrap
+                cur_block = 0;
+                cur_block_addr = 0;
+            }
+        }
+    } // per block
+
+    SPIFFS_CHECK_RES(res);
+
+    return SPIFFS_VIS_END;
+}
+
+#if !SPIFFS_READ_ONLY
+s32_t spiffs_erase_block(
+    spiffs *fs,
+    spiffs_block_ix bix) {
+    s32_t res;
+    u32_t addr = SPIFFS_BLOCK_TO_PADDR(fs, bix);
+    s32_t size = SPIFFS_CFG_LOG_BLOCK_SZ(fs);
+
+    // here we ignore res, just try erasing the block
+    while (size > 0) {
+        SPIFFS_DBG("erase "_SPIPRIad":"_SPIPRIi"\n", addr,  SPIFFS_CFG_PHYS_ERASE_SZ(fs));
+        SPIFFS_HAL_ERASE(fs, addr, SPIFFS_CFG_PHYS_ERASE_SZ(fs));
+
+        addr += SPIFFS_CFG_PHYS_ERASE_SZ(fs);
+        size -= SPIFFS_CFG_PHYS_ERASE_SZ(fs);
+    }
+    fs->free_blocks++;
+
+    // register erase count for this block
+    res = _spiffs_wr(fs, SPIFFS_OP_C_WRTHRU | SPIFFS_OP_T_OBJ_LU2, 0,
+                     SPIFFS_ERASE_COUNT_PADDR(fs, bix),
+                     sizeof(spiffs_obj_id), (u8_t *)&fs->max_erase_count);
+    SPIFFS_CHECK_RES(res);
+
+#if SPIFFS_USE_MAGIC
+    // finally, write magic
+    spiffs_obj_id magic = SPIFFS_MAGIC(fs, bix);
+    res = _spiffs_wr(fs, SPIFFS_OP_C_WRTHRU | SPIFFS_OP_T_OBJ_LU2, 0,
+                     SPIFFS_MAGIC_PADDR(fs, bix),
+                     sizeof(spiffs_obj_id), (u8_t *)&magic);
+    SPIFFS_CHECK_RES(res);
+#endif
+
+    fs->max_erase_count++;
+    if (fs->max_erase_count == SPIFFS_OBJ_ID_IX_FLAG) {
+        fs->max_erase_count = 0;
+    }
+
+    return res;
+}
+#endif // !SPIFFS_READ_ONLY
+
+#if SPIFFS_USE_MAGIC && SPIFFS_USE_MAGIC_LENGTH && SPIFFS_SINGLETON==0
+s32_t spiffs_probe(
+    spiffs_config *cfg) {
+    s32_t res;
+    u32_t paddr;
+    spiffs dummy_fs; // create a dummy fs struct just to be able to use macros
+    _SPIFFS_MEMCPY(&dummy_fs.cfg, cfg, sizeof(spiffs_config));
+    dummy_fs.block_count = 0;
+
+    // Read three magics, as one block may be in an aborted erase state.
+    // At least two of these must contain magic and be in decreasing order.
+    spiffs_obj_id magic[3];
+    spiffs_obj_id bix_count[3];
+
+    spiffs_block_ix bix;
+    for (bix = 0; bix < 3; bix++) {
+        paddr = SPIFFS_MAGIC_PADDR(&dummy_fs, bix);
+#if SPIFFS_HAL_CALLBACK_EXTRA
+        // not any proper fs to report here, so callback with null
+        // (cross fingers that no-one gets angry)
+        res = cfg->hal_read_f((void *)0, paddr, sizeof(spiffs_obj_id), (u8_t *)&magic[bix]);
+#else
+        res = cfg->hal_read_f(paddr, sizeof(spiffs_obj_id), (u8_t *)&magic[bix]);
+#endif
+        bix_count[bix] = magic[bix] ^ SPIFFS_MAGIC(&dummy_fs, 0);
+        SPIFFS_CHECK_RES(res);
+    }
+
+    // check that we have sane number of blocks
+    if (bix_count[0] < 3) return SPIFFS_ERR_PROBE_TOO_FEW_BLOCKS;
+    // check that the order is correct, take aborted erases in calculation
+    // first block aborted erase
+    if (magic[0] == (spiffs_obj_id)(-1) && bix_count[1] - bix_count[2] == 1) {
+        return (bix_count[1] + 1) * cfg->log_block_size;
+    }
+    // second block aborted erase
+    if (magic[1] == (spiffs_obj_id)(-1) && bix_count[0] - bix_count[2] == 2) {
+        return bix_count[0] * cfg->log_block_size;
+    }
+    // third block aborted erase
+    if (magic[2] == (spiffs_obj_id)(-1) && bix_count[0] - bix_count[1] == 1) {
+        return bix_count[0] * cfg->log_block_size;
+    }
+    // no block has aborted erase
+    if (bix_count[0] - bix_count[1] == 1 && bix_count[1] - bix_count[2] == 1) {
+        return bix_count[0] * cfg->log_block_size;
+    }
+
+    return SPIFFS_ERR_PROBE_NOT_A_FS;
+}
+#endif // SPIFFS_USE_MAGIC && SPIFFS_USE_MAGIC_LENGTH && SPIFFS_SINGLETON==0
+
+
+static s32_t spiffs_obj_lu_scan_v(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_block_ix bix,
+    int ix_entry,
+    const void *user_const_p,
+    void *user_var_p) {
+    (void)bix;
+    (void)user_const_p;
+    (void)user_var_p;
+    if (obj_id == SPIFFS_OBJ_ID_FREE) {
+        if (ix_entry == 0) {
+            fs->free_blocks++;
+            // todo optimize further, return SPIFFS_NEXT_BLOCK
+        }
+    } else if (obj_id == SPIFFS_OBJ_ID_DELETED) {
+        fs->stats_p_deleted++;
+    } else {
+        fs->stats_p_allocated++;
+    }
+
+    return SPIFFS_VIS_COUNTINUE;
+}
+
+
+// Scans thru all obj lu and counts free, deleted and used pages
+// Find the maximum block erase count
+// Checks magic if enabled
+s32_t spiffs_obj_lu_scan(
+    spiffs *fs) {
+    s32_t res;
+    spiffs_block_ix bix;
+    int entry;
+#if SPIFFS_USE_MAGIC
+    spiffs_block_ix unerased_bix = (spiffs_block_ix) - 1;
+#endif
+
+    // find out erase count
+    // if enabled, check magic
+    bix = 0;
+    spiffs_obj_id erase_count_final;
+    spiffs_obj_id erase_count_min = SPIFFS_OBJ_ID_FREE;
+    spiffs_obj_id erase_count_max = 0;
+    while (bix < fs->block_count) {
+#if SPIFFS_USE_MAGIC
+        spiffs_obj_id magic;
+        res = _spiffs_rd(fs,
+                         SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                         0, SPIFFS_MAGIC_PADDR(fs, bix),
+                         sizeof(spiffs_obj_id), (u8_t *)&magic);
+
+        SPIFFS_CHECK_RES(res);
+        if (magic != SPIFFS_MAGIC(fs, bix)) {
+            if (unerased_bix == (spiffs_block_ix) - 1) {
+                // allow one unerased block as it might be powered down during an erase
+                unerased_bix = bix;
+            } else {
+                // more than one unerased block, bail out
+                SPIFFS_CHECK_RES(SPIFFS_ERR_NOT_A_FS);
+            }
+        }
+#endif
+        spiffs_obj_id erase_count;
+        res = _spiffs_rd(fs,
+                         SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                         0, SPIFFS_ERASE_COUNT_PADDR(fs, bix),
+                         sizeof(spiffs_obj_id), (u8_t *)&erase_count);
+        SPIFFS_CHECK_RES(res);
+        if (erase_count != SPIFFS_OBJ_ID_FREE) {
+            erase_count_min = MIN(erase_count_min, erase_count);
+            erase_count_max = MAX(erase_count_max, erase_count);
+        }
+        bix++;
+    }
+
+    if (erase_count_min == 0 && erase_count_max == SPIFFS_OBJ_ID_FREE) {
+        // clean system, set counter to zero
+        erase_count_final = 0;
+    } else if (erase_count_max - erase_count_min > (SPIFFS_OBJ_ID_FREE) / 2) {
+        // wrap, take min
+        erase_count_final = erase_count_min + 1;
+    } else {
+        erase_count_final = erase_count_max + 1;
+    }
+
+    fs->max_erase_count = erase_count_final;
+
+#if SPIFFS_USE_MAGIC
+    if (unerased_bix != (spiffs_block_ix) - 1) {
+        // found one unerased block, remedy
+        SPIFFS_DBG("mount: erase block "_SPIPRIbl"\n", bix);
+#if SPIFFS_READ_ONLY
+        res = SPIFFS_ERR_RO_ABORTED_OPERATION;
+#else
+        res = spiffs_erase_block(fs, unerased_bix);
+#endif // SPIFFS_READ_ONLY
+        SPIFFS_CHECK_RES(res);
+    }
+#endif
+
+    // count blocks
+
+    fs->free_blocks = 0;
+    fs->stats_p_allocated = 0;
+    fs->stats_p_deleted = 0;
+
+    res = spiffs_obj_lu_find_entry_visitor(fs,
+                                           0,
+                                           0,
+                                           0,
+                                           0,
+                                           spiffs_obj_lu_scan_v,
+                                           0,
+                                           0,
+                                           &bix,
+                                           &entry);
+
+    if (res == SPIFFS_VIS_END) {
+        res = SPIFFS_OK;
+    }
+
+    SPIFFS_CHECK_RES(res);
+
+    return res;
+}
+
+#if !SPIFFS_READ_ONLY
+// Find free object lookup entry
+// Iterate over object lookup pages in each block until a free object id entry is found
+s32_t spiffs_obj_lu_find_free(
+    spiffs *fs,
+    spiffs_block_ix starting_block,
+    int starting_lu_entry,
+    spiffs_block_ix *block_ix,
+    int *lu_entry) {
+    s32_t res;
+    if (!fs->cleaning && fs->free_blocks < 2) {
+        res = spiffs_gc_quick(fs, 0);
+        if (res == SPIFFS_ERR_NO_DELETED_BLOCKS) {
+            res = SPIFFS_OK;
+        }
+        SPIFFS_CHECK_RES(res);
+        if (fs->free_blocks < 2) {
+            return SPIFFS_ERR_FULL;
+        }
+    }
+    res = spiffs_obj_lu_find_id(fs, starting_block, starting_lu_entry,
+                                SPIFFS_OBJ_ID_FREE, block_ix, lu_entry);
+    if (res == SPIFFS_OK) {
+        fs->free_cursor_block_ix = *block_ix;
+        fs->free_cursor_obj_lu_entry = (*lu_entry) + 1;
+        if (*lu_entry == 0) {
+            fs->free_blocks--;
+        }
+    }
+    if (res == SPIFFS_ERR_FULL) {
+        SPIFFS_DBGF("fs full\n");
+    }
+
+    return res;
+}
+#endif // !SPIFFS_READ_ONLY
+
+// Find object lookup entry containing given id
+// Iterate over object lookup pages in each block until a given object id entry is found
+s32_t spiffs_obj_lu_find_id(
+    spiffs *fs,
+    spiffs_block_ix starting_block,
+    int starting_lu_entry,
+    spiffs_obj_id obj_id,
+    spiffs_block_ix *block_ix,
+    int *lu_entry) {
+    s32_t res = spiffs_obj_lu_find_entry_visitor(
+                    fs, starting_block, starting_lu_entry, SPIFFS_VIS_CHECK_ID, obj_id, 0, 0, 0, block_ix, lu_entry);
+    if (res == SPIFFS_VIS_END) {
+        res = SPIFFS_ERR_NOT_FOUND;
+    }
+    return res;
+}
+
+
+static s32_t spiffs_obj_lu_find_id_and_span_v(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_block_ix bix,
+    int ix_entry,
+    const void *user_const_p,
+    void *user_var_p) {
+    s32_t res;
+    spiffs_page_header ph;
+    spiffs_page_ix pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, ix_entry);
+    res = _spiffs_rd(fs, 0, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                     SPIFFS_PAGE_TO_PADDR(fs, pix), sizeof(spiffs_page_header), (u8_t *)&ph);
+    SPIFFS_CHECK_RES(res);
+    if (ph.obj_id == obj_id &&
+            ph.span_ix == *((spiffs_span_ix *)user_var_p) &&
+            (ph.flags & (SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_USED)) == SPIFFS_PH_FLAG_DELET &&
+            !((obj_id & SPIFFS_OBJ_ID_IX_FLAG) && (ph.flags & SPIFFS_PH_FLAG_IXDELE) == 0 && ph.span_ix == 0) &&
+            (user_const_p == 0 || *((const spiffs_page_ix *)user_const_p) != pix)) {
+        return SPIFFS_OK;
+    } else {
+        return SPIFFS_VIS_COUNTINUE;
+    }
+}
+
+// Find object lookup entry containing given id and span index
+// Iterate over object lookup pages in each block until a given object id entry is found
+s32_t spiffs_obj_lu_find_id_and_span(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_span_ix spix,
+    spiffs_page_ix exclusion_pix,
+    spiffs_page_ix *pix) {
+    s32_t res;
+    spiffs_block_ix bix;
+    int entry;
+
+    res = spiffs_obj_lu_find_entry_visitor(fs,
+                                           fs->cursor_block_ix,
+                                           fs->cursor_obj_lu_entry,
+                                           SPIFFS_VIS_CHECK_ID,
+                                           obj_id,
+                                           spiffs_obj_lu_find_id_and_span_v,
+                                           exclusion_pix ? &exclusion_pix : 0,
+                                           &spix,
+                                           &bix,
+                                           &entry);
+
+    if (res == SPIFFS_VIS_END) {
+        res = SPIFFS_ERR_NOT_FOUND;
+    }
+
+    SPIFFS_CHECK_RES(res);
+
+    if (pix) {
+        *pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, entry);
+    }
+
+    fs->cursor_block_ix = bix;
+    fs->cursor_obj_lu_entry = entry;
+
+    return res;
+}
+
+// Find object lookup entry containing given id and span index in page headers only
+// Iterate over object lookup pages in each block until a given object id entry is found
+s32_t spiffs_obj_lu_find_id_and_span_by_phdr(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_span_ix spix,
+    spiffs_page_ix exclusion_pix,
+    spiffs_page_ix *pix) {
+    s32_t res;
+    spiffs_block_ix bix;
+    int entry;
+
+    res = spiffs_obj_lu_find_entry_visitor(fs,
+                                           fs->cursor_block_ix,
+                                           fs->cursor_obj_lu_entry,
+                                           SPIFFS_VIS_CHECK_PH,
+                                           obj_id,
+                                           spiffs_obj_lu_find_id_and_span_v,
+                                           exclusion_pix ? &exclusion_pix : 0,
+                                           &spix,
+                                           &bix,
+                                           &entry);
+
+    if (res == SPIFFS_VIS_END) {
+        res = SPIFFS_ERR_NOT_FOUND;
+    }
+
+    SPIFFS_CHECK_RES(res);
+
+    if (pix) {
+        *pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, entry);
+    }
+
+    fs->cursor_block_ix = bix;
+    fs->cursor_obj_lu_entry = entry;
+
+    return res;
+}
+
+#if SPIFFS_IX_MAP
+
+// update index map of given fd with given object index data
+static void spiffs_update_ix_map(spiffs *fs,
+                                 spiffs_fd *fd, spiffs_span_ix objix_spix, spiffs_page_object_ix *objix) {
+#if SPIFFS_SINGLETON
+    (void)fs;
+#endif
+    spiffs_ix_map *map = fd->ix_map;
+    spiffs_span_ix map_objix_start_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, map->start_spix);
+    spiffs_span_ix map_objix_end_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, map->end_spix);
+
+    // check if updated ix is within map range
+    if (objix_spix < map_objix_start_spix || objix_spix > map_objix_end_spix) {
+        return;
+    }
+
+    // update memory mapped page index buffer to new pages
+
+    // get range of updated object index map data span indices
+    spiffs_span_ix objix_data_spix_start =
+        SPIFFS_DATA_SPAN_IX_FOR_OBJ_IX_SPAN_IX(fs, objix_spix);
+    spiffs_span_ix objix_data_spix_end = objix_data_spix_start +
+                                         (objix_spix == 0 ? SPIFFS_OBJ_HDR_IX_LEN(fs) : SPIFFS_OBJ_IX_LEN(fs));
+
+    // calc union of object index range and index map range array
+    spiffs_span_ix map_spix = MAX(map->start_spix, objix_data_spix_start);
+    spiffs_span_ix map_spix_end = MIN(map->end_spix + 1, objix_data_spix_end);
+
+    while (map_spix < map_spix_end) {
+        spiffs_page_ix objix_data_pix;
+        if (objix_spix == 0) {
+            // get data page from object index header page
+            objix_data_pix = ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix_header)))[map_spix];
+        } else {
+            // get data page from object index page
+            objix_data_pix = ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, map_spix)];
+        }
+
+        if (objix_data_pix == (spiffs_page_ix) - 1) {
+            // reached end of object, abort
+            break;
+        }
+
+        map->map_buf[map_spix - map->start_spix] = objix_data_pix;
+        SPIFFS_DBG("map "_SPIPRIid":"_SPIPRIsp" ("_SPIPRIsp"--"_SPIPRIsp") objix.spix:"_SPIPRIsp" to pix "_SPIPRIpg"\n",
+                   fd->obj_id, map_spix - map->start_spix,
+                   map->start_spix, map->end_spix,
+                   objix->p_hdr.span_ix,
+                   objix_data_pix);
+
+        map_spix++;
+    }
+}
+
+typedef struct {
+    spiffs_fd *fd;
+    u32_t remaining_objix_pages_to_visit;
+    spiffs_span_ix map_objix_start_spix;
+    spiffs_span_ix map_objix_end_spix;
+} spiffs_ix_map_populate_state;
+
+static s32_t spiffs_populate_ix_map_v(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_block_ix bix,
+    int ix_entry,
+    const void *user_const_p,
+    void *user_var_p) {
+    (void)user_const_p;
+    s32_t res;
+    spiffs_ix_map_populate_state *state = (spiffs_ix_map_populate_state *)user_var_p;
+    spiffs_page_ix pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, ix_entry);
+
+    // load header to check it
+    spiffs_page_object_ix *objix = (spiffs_page_object_ix *)fs->work;
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                     0, SPIFFS_PAGE_TO_PADDR(fs, pix), sizeof(spiffs_page_object_ix), (u8_t *)objix);
+    SPIFFS_CHECK_RES(res);
+    SPIFFS_VALIDATE_OBJIX(objix->p_hdr, obj_id, objix->p_hdr.span_ix);
+
+    // check if hdr is ok, and if objix range overlap with ix map range
+    if ((objix->p_hdr.flags & (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_IXDELE)) ==
+            (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_IXDELE) &&
+            objix->p_hdr.span_ix >= state->map_objix_start_spix &&
+            objix->p_hdr.span_ix <= state->map_objix_end_spix) {
+        // ok, load rest of object index
+        res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                         0, SPIFFS_PAGE_TO_PADDR(fs, pix) + sizeof(spiffs_page_object_ix),
+                         SPIFFS_CFG_LOG_PAGE_SZ(fs) - sizeof(spiffs_page_object_ix),
+                         (u8_t *)objix + sizeof(spiffs_page_object_ix));
+        SPIFFS_CHECK_RES(res);
+
+        spiffs_update_ix_map(fs, state->fd, objix->p_hdr.span_ix, objix);
+
+        state->remaining_objix_pages_to_visit--;
+        SPIFFS_DBG("map "_SPIPRIid" ("_SPIPRIsp"--"_SPIPRIsp") remaining objix pages "_SPIPRIi"\n",
+                   state->fd->obj_id,
+                   state->fd->ix_map->start_spix, state->fd->ix_map->end_spix,
+                   state->remaining_objix_pages_to_visit);
+    }
+
+    if (res == SPIFFS_OK) {
+        res = state->remaining_objix_pages_to_visit ? SPIFFS_VIS_COUNTINUE : SPIFFS_VIS_END;
+    }
+    return res;
+}
+
+// populates index map, from vector entry start to vector entry end, inclusive
+s32_t spiffs_populate_ix_map(spiffs *fs, spiffs_fd *fd, u32_t vec_entry_start, u32_t vec_entry_end) {
+    s32_t res;
+    spiffs_ix_map *map = fd->ix_map;
+    spiffs_ix_map_populate_state state;
+    vec_entry_start = MIN((u32_t)(map->end_spix - map->start_spix), vec_entry_start);
+    vec_entry_end = MAX((u32_t)(map->end_spix - map->start_spix), vec_entry_end);
+    if (vec_entry_start > vec_entry_end) {
+        return SPIFFS_ERR_IX_MAP_BAD_RANGE;
+    }
+    state.map_objix_start_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, map->start_spix + vec_entry_start);
+    state.map_objix_end_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, map->start_spix + vec_entry_end);
+    state.remaining_objix_pages_to_visit =
+        state.map_objix_end_spix - state.map_objix_start_spix + 1;
+    state.fd = fd;
+
+    res = spiffs_obj_lu_find_entry_visitor(
+              fs,
+              SPIFFS_BLOCK_FOR_PAGE(fs, fd->objix_hdr_pix),
+              SPIFFS_OBJ_LOOKUP_ENTRY_FOR_PAGE(fs, fd->objix_hdr_pix),
+              SPIFFS_VIS_CHECK_ID,
+              fd->obj_id | SPIFFS_OBJ_ID_IX_FLAG,
+              spiffs_populate_ix_map_v,
+              0,
+              &state,
+              0,
+              0);
+
+    if (res == SPIFFS_VIS_END) {
+        res = SPIFFS_OK;
+    }
+
+    return res;
+}
+
+#endif
+
+
+#if !SPIFFS_READ_ONLY
+// Allocates a free defined page with given obj_id
+// Occupies object lookup entry and page
+// data may be NULL; where only page header is stored, len and page_offs is ignored
+s32_t spiffs_page_allocate_data(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_page_header *ph,
+    u8_t *data,
+    u32_t len,
+    u32_t page_offs,
+    u8_t finalize,
+    spiffs_page_ix *pix) {
+    s32_t res = SPIFFS_OK;
+    spiffs_block_ix bix;
+    int entry;
+
+    // find free entry
+    res = spiffs_obj_lu_find_free(fs, fs->free_cursor_block_ix, fs->free_cursor_obj_lu_entry, &bix, &entry);
+    SPIFFS_CHECK_RES(res);
+
+    // occupy page in object lookup
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_UPDT,
+                     0, SPIFFS_BLOCK_TO_PADDR(fs, bix) + entry * sizeof(spiffs_obj_id), sizeof(spiffs_obj_id), (u8_t *)&obj_id);
+    SPIFFS_CHECK_RES(res);
+
+    fs->stats_p_allocated++;
+
+    // write page header
+    ph->flags &= ~SPIFFS_PH_FLAG_USED;
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                     0, SPIFFS_OBJ_LOOKUP_ENTRY_TO_PADDR(fs, bix, entry), sizeof(spiffs_page_header), (u8_t *)ph);
+    SPIFFS_CHECK_RES(res);
+
+    // write page data
+    if (data) {
+        res = _spiffs_wr(fs,  SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                         0, SPIFFS_OBJ_LOOKUP_ENTRY_TO_PADDR(fs, bix, entry) + sizeof(spiffs_page_header) + page_offs, len, data);
+        SPIFFS_CHECK_RES(res);
+    }
+
+    // finalize header if necessary
+    if (finalize && (ph->flags & SPIFFS_PH_FLAG_FINAL)) {
+        ph->flags &= ~SPIFFS_PH_FLAG_FINAL;
+        res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                         0, SPIFFS_OBJ_LOOKUP_ENTRY_TO_PADDR(fs, bix, entry) + offsetof(spiffs_page_header, flags),
+                         sizeof(u8_t),
+                         (u8_t *)&ph->flags);
+        SPIFFS_CHECK_RES(res);
+    }
+
+    // return written page
+    if (pix) {
+        *pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, entry);
+    }
+
+    return res;
+}
+#endif // !SPIFFS_READ_ONLY
+
+#if !SPIFFS_READ_ONLY
+// Moves a page from src to a free page and finalizes it. Updates page index. Page data is given in param page.
+// If page data is null, provided header is used for metainfo and page data is physically copied.
+s32_t spiffs_page_move(
+    spiffs *fs,
+    spiffs_file fh,
+    u8_t *page_data,
+    spiffs_obj_id obj_id,
+    spiffs_page_header *page_hdr,
+    spiffs_page_ix src_pix,
+    spiffs_page_ix *dst_pix) {
+    s32_t res;
+    u8_t was_final = 0;
+    spiffs_page_header *p_hdr;
+    spiffs_block_ix bix;
+    int entry;
+    spiffs_page_ix free_pix;
+
+    // find free entry
+    res = spiffs_obj_lu_find_free(fs, fs->free_cursor_block_ix, fs->free_cursor_obj_lu_entry, &bix, &entry);
+    SPIFFS_CHECK_RES(res);
+    free_pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, entry);
+
+    if (dst_pix) *dst_pix = free_pix;
+
+    p_hdr = page_data ? (spiffs_page_header *)page_data : page_hdr;
+    if (page_data) {
+        // got page data
+        was_final = (p_hdr->flags & SPIFFS_PH_FLAG_FINAL) == 0;
+        // write unfinalized page
+        p_hdr->flags |= SPIFFS_PH_FLAG_FINAL;
+        p_hdr->flags &= ~SPIFFS_PH_FLAG_USED;
+        res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                         0, SPIFFS_PAGE_TO_PADDR(fs, free_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), page_data);
+    } else {
+        // copy page data
+        res = spiffs_phys_cpy(fs, fh, SPIFFS_PAGE_TO_PADDR(fs, free_pix), SPIFFS_PAGE_TO_PADDR(fs, src_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs));
+    }
+    SPIFFS_CHECK_RES(res);
+
+    // mark entry in destination object lookup
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_UPDT,
+                     0, SPIFFS_BLOCK_TO_PADDR(fs, SPIFFS_BLOCK_FOR_PAGE(fs, free_pix)) + SPIFFS_OBJ_LOOKUP_ENTRY_FOR_PAGE(fs, free_pix) * sizeof(spiffs_page_ix),
+                     sizeof(spiffs_obj_id),
+                     (u8_t *)&obj_id);
+    SPIFFS_CHECK_RES(res);
+
+    fs->stats_p_allocated++;
+
+    if (was_final) {
+        // mark finalized in destination page
+        p_hdr->flags &= ~(SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_USED);
+        res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                         fh,
+                         SPIFFS_PAGE_TO_PADDR(fs, free_pix) + offsetof(spiffs_page_header, flags),
+                         sizeof(u8_t),
+                         (u8_t *)&p_hdr->flags);
+        SPIFFS_CHECK_RES(res);
+    }
+    // mark source deleted
+    res = spiffs_page_delete(fs, src_pix);
+    return res;
+}
+#endif // !SPIFFS_READ_ONLY
+
+#if !SPIFFS_READ_ONLY
+// Deletes a page and removes it from object lookup.
+s32_t spiffs_page_delete(
+    spiffs *fs,
+    spiffs_page_ix pix) {
+    s32_t res;
+    // mark deleted entry in source object lookup
+    spiffs_obj_id d_obj_id = SPIFFS_OBJ_ID_DELETED;
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_DELE,
+                     0,
+                     SPIFFS_BLOCK_TO_PADDR(fs, SPIFFS_BLOCK_FOR_PAGE(fs, pix)) + SPIFFS_OBJ_LOOKUP_ENTRY_FOR_PAGE(fs, pix) * sizeof(spiffs_page_ix),
+                     sizeof(spiffs_obj_id),
+                     (u8_t *)&d_obj_id);
+    SPIFFS_CHECK_RES(res);
+
+    fs->stats_p_deleted++;
+    fs->stats_p_allocated--;
+
+    // mark deleted in source page
+    u8_t flags = 0xff;
+#if SPIFFS_NO_BLIND_WRITES
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_READ,
+                     0, SPIFFS_PAGE_TO_PADDR(fs, pix) + offsetof(spiffs_page_header, flags),
+                     sizeof(flags), &flags);
+    SPIFFS_CHECK_RES(res);
+#endif
+    flags &= ~(SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_USED);
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_DELE,
+                     0,
+                     SPIFFS_PAGE_TO_PADDR(fs, pix) + offsetof(spiffs_page_header, flags),
+                     sizeof(flags), &flags);
+
+    return res;
+}
+#endif // !SPIFFS_READ_ONLY
+
+#if !SPIFFS_READ_ONLY
+// Create an object index header page with empty index and undefined length
+s32_t spiffs_object_create(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    const u8_t name[],
+    const u8_t meta[],
+    spiffs_obj_type type,
+    spiffs_page_ix *objix_hdr_pix) {
+    s32_t res = SPIFFS_OK;
+    spiffs_block_ix bix;
+    spiffs_page_object_ix_header oix_hdr;
+    int entry;
+
+    res = spiffs_gc_check(fs, SPIFFS_DATA_PAGE_SIZE(fs));
+    SPIFFS_CHECK_RES(res);
+
+    obj_id |= SPIFFS_OBJ_ID_IX_FLAG;
+
+    // find free entry
+    res = spiffs_obj_lu_find_free(fs, fs->free_cursor_block_ix, fs->free_cursor_obj_lu_entry, &bix, &entry);
+    SPIFFS_CHECK_RES(res);
+    SPIFFS_DBG("create: found free page @ "_SPIPRIpg" bix:"_SPIPRIbl" entry:"_SPIPRIsp"\n", (spiffs_page_ix)SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, entry), bix, entry);
+
+    // occupy page in object lookup
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_UPDT,
+                     0, SPIFFS_BLOCK_TO_PADDR(fs, bix) + entry * sizeof(spiffs_obj_id), sizeof(spiffs_obj_id), (u8_t *)&obj_id);
+    SPIFFS_CHECK_RES(res);
+
+    fs->stats_p_allocated++;
+
+    // write empty object index page
+    oix_hdr.p_hdr.obj_id = obj_id;
+    oix_hdr.p_hdr.span_ix = 0;
+    oix_hdr.p_hdr.flags = 0xff & ~(SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_USED);
+    oix_hdr.type = type;
+    oix_hdr.size = SPIFFS_UNDEFINED_LEN; // keep ones so we can update later without wasting this page
+    strncpy((char *)oix_hdr.name, (const char *)name, SPIFFS_OBJ_NAME_LEN);
+#if SPIFFS_OBJ_META_LEN
+    if (meta) {
+        _SPIFFS_MEMCPY(oix_hdr.meta, meta, SPIFFS_OBJ_META_LEN);
+    } else {
+        memset(oix_hdr.meta, 0xff, SPIFFS_OBJ_META_LEN);
+    }
+#else
+    (void) meta;
+#endif
+
+    // update page
+    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                     0, SPIFFS_OBJ_LOOKUP_ENTRY_TO_PADDR(fs, bix, entry), sizeof(spiffs_page_object_ix_header), (u8_t *)&oix_hdr);
+
+    SPIFFS_CHECK_RES(res);
+    spiffs_cb_object_event(fs, (spiffs_page_object_ix *)&oix_hdr,
+                           SPIFFS_EV_IX_NEW, obj_id, 0, SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, entry), SPIFFS_UNDEFINED_LEN);
+
+    if (objix_hdr_pix) {
+        *objix_hdr_pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, entry);
+    }
+
+    return res;
+}
+#endif // !SPIFFS_READ_ONLY
+
+#if !SPIFFS_READ_ONLY
+// update object index header with any combination of name/size/index
+// new_objix_hdr_data may be null, if so the object index header page is loaded
+// name may be null, if so name is not changed
+// size may be null, if so size is not changed
+s32_t spiffs_object_update_index_hdr(
+    spiffs *fs,
+    spiffs_fd *fd,
+    spiffs_obj_id obj_id,
+    spiffs_page_ix objix_hdr_pix,
+    u8_t *new_objix_hdr_data,
+    const u8_t name[],
+    const u8_t meta[],
+    u32_t size,
+    spiffs_page_ix *new_pix) {
+    s32_t res = SPIFFS_OK;
+    spiffs_page_object_ix_header *objix_hdr;
+    spiffs_page_ix new_objix_hdr_pix;
+
+    obj_id |=  SPIFFS_OBJ_ID_IX_FLAG;
+
+    if (new_objix_hdr_data) {
+        // object index header page already given to us, no need to load it
+        objix_hdr = (spiffs_page_object_ix_header *)new_objix_hdr_data;
+    } else {
+        // read object index header page
+        res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ,
+                         fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, objix_hdr_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+        SPIFFS_CHECK_RES(res);
+        objix_hdr = (spiffs_page_object_ix_header *)fs->work;
+    }
+
+    SPIFFS_VALIDATE_OBJIX(objix_hdr->p_hdr, obj_id, 0);
+
+    // change name
+    if (name) {
+        strncpy((char *)objix_hdr->name, (const char *)name, SPIFFS_OBJ_NAME_LEN);
+    }
+#if SPIFFS_OBJ_META_LEN
+    if (meta) {
+        _SPIFFS_MEMCPY(objix_hdr->meta, meta, SPIFFS_OBJ_META_LEN);
+    }
+#else
+    (void) meta;
+#endif
+    if (size) {
+        objix_hdr->size = size;
+    }
+
+    // move and update page
+    res = spiffs_page_move(fs, fd == 0 ? 0 : fd->file_nbr, (u8_t *)objix_hdr, obj_id, 0, objix_hdr_pix, &new_objix_hdr_pix);
+
+    if (res == SPIFFS_OK) {
+        if (new_pix) {
+            *new_pix = new_objix_hdr_pix;
+        }
+        // callback on object index update
+        spiffs_cb_object_event(fs, (spiffs_page_object_ix *)objix_hdr,
+                               new_objix_hdr_data ? SPIFFS_EV_IX_UPD : SPIFFS_EV_IX_UPD_HDR,
+                               obj_id, objix_hdr->p_hdr.span_ix, new_objix_hdr_pix, objix_hdr->size);
+        if (fd) fd->objix_hdr_pix = new_objix_hdr_pix; // if this is not in the registered cluster
+    }
+
+    return res;
+}
+#endif // !SPIFFS_READ_ONLY
+
+void spiffs_cb_object_event(
+    spiffs *fs,
+    spiffs_page_object_ix *objix,
+    int ev,
+    spiffs_obj_id obj_id_raw,
+    spiffs_span_ix spix,
+    spiffs_page_ix new_pix,
+    u32_t new_size) {
+#if SPIFFS_IX_MAP == 0
+    (void)objix;
+#endif
+    // update index caches in all file descriptors
+    spiffs_obj_id obj_id = obj_id_raw & ~SPIFFS_OBJ_ID_IX_FLAG;
+    u32_t i;
+    spiffs_fd *fds = (spiffs_fd *)fs->fd_space;
+    SPIFFS_DBG("       CALLBACK  %s obj_id:"_SPIPRIid" spix:"_SPIPRIsp" npix:"_SPIPRIpg" nsz:"_SPIPRIi"\n", (const char *[]) {"UPD", "NEW", "DEL", "MOV", "HUP", "???"}[MIN(ev, 5)],
+    obj_id_raw, spix, new_pix, new_size);
+    for (i = 0; i < fs->fd_count; i++) {
+        spiffs_fd *cur_fd = &fds[i];
+        if ((cur_fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG) != obj_id) continue; // fd not related to updated file
+#if !SPIFFS_TEMPORAL_FD_CACHE
+        if (cur_fd->file_nbr == 0) continue; // fd closed
+#endif
+        if (spix == 0) { // object index header update
+            if (ev != SPIFFS_EV_IX_DEL) {
+#if SPIFFS_TEMPORAL_FD_CACHE
+                if (cur_fd->score == 0) continue; // never used fd
+#endif
+                SPIFFS_DBG("       callback: setting fd "_SPIPRIfd":"_SPIPRIid"(fdoffs:"_SPIPRIi" offs:"_SPIPRIi") objix_hdr_pix to "_SPIPRIpg", size:"_SPIPRIi"\n",
+                           SPIFFS_FH_OFFS(fs, cur_fd->file_nbr), cur_fd->obj_id, cur_fd->fdoffset, cur_fd->offset, new_pix, new_size);
+                cur_fd->objix_hdr_pix = new_pix;
+                if (new_size != 0) {
+                    // update size and offsets for fds to this file
+                    cur_fd->size = new_size;
+                    u32_t act_new_size = new_size == SPIFFS_UNDEFINED_LEN ? 0 : new_size;
+#if SPIFFS_CACHE_WR
+                    if (act_new_size > 0 && cur_fd->cache_page) {
+                        act_new_size = MAX(act_new_size, cur_fd->cache_page->ucache.swrc.offset + cur_fd->cache_page->ucache.swrc.size);
+                    }
+#endif
+                    if (cur_fd->offset > act_new_size) {
+                        cur_fd->offset = act_new_size;
+                    }
+                    if (cur_fd->fdoffset > act_new_size) {
+                        cur_fd->fdoffset = act_new_size;
+                    }
+#if SPIFFS_CACHE_WR
+                    if (cur_fd->cache_page && cur_fd->cache_page->ucache.swrc.offset > act_new_size + 1) {
+                        SPIFFS_CACHE_DBG("CACHE_DROP: file trunced, dropping cache page "_SPIPRIi", no writeback\n", cur_fd->cache_page->ix);
+                        spiffs_cache_fd_release(fs, cur_fd->cache_page);
+                    }
+#endif
+                }
+            } else {
+                // removing file
+#if SPIFFS_CACHE_WR
+                if (cur_fd->file_nbr && cur_fd->cache_page) {
+                    SPIFFS_CACHE_DBG("CACHE_DROP: file deleted, dropping cache page "_SPIPRIi", no writeback\n", cur_fd->cache_page->ix);
+                    spiffs_cache_fd_release(fs, cur_fd->cache_page);
+                }
+#endif
+                SPIFFS_DBG("       callback: release fd "_SPIPRIfd":"_SPIPRIid" span:"_SPIPRIsp" objix_pix to "_SPIPRIpg"\n", SPIFFS_FH_OFFS(fs, cur_fd->file_nbr), cur_fd->obj_id, spix, new_pix);
+                cur_fd->file_nbr = 0;
+                cur_fd->obj_id = SPIFFS_OBJ_ID_DELETED;
+            }
+        } // object index header update
+        if (cur_fd->cursor_objix_spix == spix) {
+            if (ev != SPIFFS_EV_IX_DEL) {
+                SPIFFS_DBG("       callback: setting fd "_SPIPRIfd":"_SPIPRIid" span:"_SPIPRIsp" objix_pix to "_SPIPRIpg"\n", SPIFFS_FH_OFFS(fs, cur_fd->file_nbr), cur_fd->obj_id, spix, new_pix);
+                cur_fd->cursor_objix_pix = new_pix;
+            } else {
+                cur_fd->cursor_objix_pix = 0;
+            }
+        }
+    } // fd update loop
+
+#if SPIFFS_IX_MAP
+
+    // update index maps
+    if (ev == SPIFFS_EV_IX_UPD || ev == SPIFFS_EV_IX_NEW) {
+        for (i = 0; i < fs->fd_count; i++) {
+            spiffs_fd *cur_fd = &fds[i];
+            // check fd opened, having ix map, match obj id
+            if (cur_fd->file_nbr == 0 ||
+                    cur_fd->ix_map == 0 ||
+                    (cur_fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG) != obj_id) continue;
+            SPIFFS_DBG("       callback: map ix update fd "_SPIPRIfd":"_SPIPRIid" span:"_SPIPRIsp"\n", SPIFFS_FH_OFFS(fs, cur_fd->file_nbr), cur_fd->obj_id, spix);
+            spiffs_update_ix_map(fs, cur_fd, spix, objix);
+        }
+    }
+
+#endif
+
+    // callback to user if object index header
+    if (fs->file_cb_f && spix == 0 && (obj_id_raw & SPIFFS_OBJ_ID_IX_FLAG)) {
+        spiffs_fileop_type op;
+        if (ev == SPIFFS_EV_IX_NEW) {
+            op = SPIFFS_CB_CREATED;
+        } else if (ev == SPIFFS_EV_IX_UPD ||
+                   ev == SPIFFS_EV_IX_MOV ||
+                   ev == SPIFFS_EV_IX_UPD_HDR) {
+            op = SPIFFS_CB_UPDATED;
+        } else if (ev == SPIFFS_EV_IX_DEL) {
+            op = SPIFFS_CB_DELETED;
+        } else {
+            SPIFFS_DBG("       callback: WARNING unknown callback event "_SPIPRIi"\n", ev);
+            return; // bail out
+        }
+        fs->file_cb_f(fs, op, obj_id, new_pix);
+    }
+}
+
+// Open object by id
+s32_t spiffs_object_open_by_id(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_fd *fd,
+    spiffs_flags flags,
+    spiffs_mode mode) {
+    s32_t res = SPIFFS_OK;
+    spiffs_page_ix pix;
+
+    res = spiffs_obj_lu_find_id_and_span(fs, obj_id | SPIFFS_OBJ_ID_IX_FLAG, 0, 0, &pix);
+    SPIFFS_CHECK_RES(res);
+
+    res = spiffs_object_open_by_page(fs, pix, fd, flags, mode);
+
+    return res;
+}
+
+// Open object by page index
+s32_t spiffs_object_open_by_page(
+    spiffs *fs,
+    spiffs_page_ix pix,
+    spiffs_fd *fd,
+    spiffs_flags flags,
+    spiffs_mode mode) {
+    (void)mode;
+    s32_t res = SPIFFS_OK;
+    spiffs_page_object_ix_header oix_hdr;
+    spiffs_obj_id obj_id;
+
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ,
+                     fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, pix), sizeof(spiffs_page_object_ix_header), (u8_t *)&oix_hdr);
+    SPIFFS_CHECK_RES(res);
+
+    spiffs_block_ix bix = SPIFFS_BLOCK_FOR_PAGE(fs, pix);
+    int entry = SPIFFS_OBJ_LOOKUP_ENTRY_FOR_PAGE(fs, pix);
+
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
+                     0,  SPIFFS_BLOCK_TO_PADDR(fs, bix) + entry * sizeof(spiffs_obj_id), sizeof(spiffs_obj_id), (u8_t *)&obj_id);
+
+    fd->fs = fs;
+    fd->objix_hdr_pix = pix;
+    fd->size = oix_hdr.size;
+    fd->offset = 0;
+    fd->cursor_objix_pix = pix;
+    fd->cursor_objix_spix = 0;
+    fd->obj_id = obj_id;
+    fd->flags = flags;
+
+    SPIFFS_VALIDATE_OBJIX(oix_hdr.p_hdr, fd->obj_id, 0);
+
+    SPIFFS_DBG("open: fd "_SPIPRIfd" is obj id "_SPIPRIid"\n", SPIFFS_FH_OFFS(fs, fd->file_nbr), fd->obj_id);
+
+    return res;
+}
+
+#if !SPIFFS_READ_ONLY
+// Append to object
+// keep current object index (header) page in fs->work buffer
+s32_t spiffs_object_append(spiffs_fd *fd, u32_t offset, u8_t *data, u32_t len) {
+    spiffs *fs = fd->fs;
+    s32_t res = SPIFFS_OK;
+    u32_t written = 0;
+
+    SPIFFS_DBG("append: "_SPIPRIi" bytes @ offs "_SPIPRIi" of size "_SPIPRIi"\n", len, offset, fd->size);
+
+    if (offset > fd->size) {
+        SPIFFS_DBGF("append: offset reversed to size\n");
+        offset = fd->size;
+    }
+
+    res = spiffs_gc_check(fs, len + SPIFFS_DATA_PAGE_SIZE(fs)); // add an extra page of data worth for meta
+    if (res != SPIFFS_OK) {
+        SPIFFS_DBG("append: gc check fail "_SPIPRIi"\n", res);
+    }
+    SPIFFS_CHECK_RES(res);
+
+    spiffs_page_object_ix_header *objix_hdr = (spiffs_page_object_ix_header *)fs->work;
+    spiffs_page_object_ix *objix = (spiffs_page_object_ix *)fs->work;
+    spiffs_page_header p_hdr;
+
+    spiffs_span_ix cur_objix_spix = 0;
+    spiffs_span_ix prev_objix_spix = (spiffs_span_ix) - 1;
+    spiffs_page_ix cur_objix_pix = fd->objix_hdr_pix;
+    spiffs_page_ix new_objix_hdr_page;
+
+    spiffs_span_ix data_spix = offset / SPIFFS_DATA_PAGE_SIZE(fs);
+    spiffs_page_ix data_page;
+    u32_t page_offs = offset % SPIFFS_DATA_PAGE_SIZE(fs);
+
+    // write all data
+    while (res == SPIFFS_OK && written < len) {
+        // calculate object index page span index
+        cur_objix_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, data_spix);
+
+        // handle storing and loading of object indices
+        if (cur_objix_spix != prev_objix_spix) {
+            // new object index page
+            // within this clause we return directly if something fails, object index mess-up
+            if (written > 0) {
+                // store previous object index page, unless first pass
+                SPIFFS_DBG("append: "_SPIPRIid" store objix "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", fd->obj_id,
+                           cur_objix_pix, prev_objix_spix, written);
+                if (prev_objix_spix == 0) {
+                    // this is an update to object index header page
+                    objix_hdr->size = offset + written;
+                    if (offset == 0) {
+                        // was an empty object, update same page (size was 0xffffffff)
+                        res = spiffs_page_index_check(fs, fd, cur_objix_pix, 0);
+                        SPIFFS_CHECK_RES(res);
+                        res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_UPDT,
+                                         fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, cur_objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+                        SPIFFS_CHECK_RES(res);
+                    } else {
+                        // was a nonempty object, update to new page
+                        res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                                             fd->objix_hdr_pix, fs->work, 0, 0, offset + written, &new_objix_hdr_page);
+                        SPIFFS_CHECK_RES(res);
+                        SPIFFS_DBG("append: "_SPIPRIid" store new objix_hdr, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", fd->obj_id,
+                                   new_objix_hdr_page, 0, written);
+                    }
+                } else {
+                    // this is an update to an object index page
+                    res = spiffs_page_index_check(fs, fd, cur_objix_pix, prev_objix_spix);
+                    SPIFFS_CHECK_RES(res);
+
+                    res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_UPDT,
+                                     fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, cur_objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+                    SPIFFS_CHECK_RES(res);
+                    spiffs_cb_object_event(fs, (spiffs_page_object_ix *)fs->work,
+                                           SPIFFS_EV_IX_UPD, fd->obj_id, objix->p_hdr.span_ix, cur_objix_pix, 0);
+                    // update length in object index header page
+                    res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                                         fd->objix_hdr_pix, 0, 0, 0, offset + written, &new_objix_hdr_page);
+                    SPIFFS_CHECK_RES(res);
+                    SPIFFS_DBG("append: "_SPIPRIid" store new size I "_SPIPRIi" in objix_hdr, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", fd->obj_id,
+                               offset + written, new_objix_hdr_page, 0, written);
+                }
+                fd->size = offset + written;
+                fd->offset = offset + written;
+            }
+
+            // create or load new object index page
+            if (cur_objix_spix == 0) {
+                // load object index header page, must always exist
+                SPIFFS_DBG("append: "_SPIPRIid" load objixhdr page "_SPIPRIpg":"_SPIPRIsp"\n", fd->obj_id, cur_objix_pix, cur_objix_spix);
+                res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ,
+                                 fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, cur_objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+                SPIFFS_CHECK_RES(res);
+                SPIFFS_VALIDATE_OBJIX(objix_hdr->p_hdr, fd->obj_id, cur_objix_spix);
+            } else {
+                spiffs_span_ix len_objix_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, (fd->size - 1) / SPIFFS_DATA_PAGE_SIZE(fs));
+                // on subsequent passes, create a new object index page
+                if (written > 0 || cur_objix_spix > len_objix_spix) {
+                    p_hdr.obj_id = fd->obj_id | SPIFFS_OBJ_ID_IX_FLAG;
+                    p_hdr.span_ix = cur_objix_spix;
+                    p_hdr.flags = 0xff & ~(SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_INDEX);
+                    res = spiffs_page_allocate_data(fs, fd->obj_id | SPIFFS_OBJ_ID_IX_FLAG,
+                                                    &p_hdr, 0, 0, 0, 1, &cur_objix_pix);
+                    SPIFFS_CHECK_RES(res);
+                    // quick "load" of new object index page
+                    memset(fs->work, 0xff, SPIFFS_CFG_LOG_PAGE_SZ(fs));
+                    _SPIFFS_MEMCPY(fs->work, &p_hdr, sizeof(spiffs_page_header));
+                    spiffs_cb_object_event(fs, (spiffs_page_object_ix *)fs->work,
+                                           SPIFFS_EV_IX_NEW, fd->obj_id, cur_objix_spix, cur_objix_pix, 0);
+                    SPIFFS_DBG("append: "_SPIPRIid" create objix page, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", fd->obj_id
+                               , cur_objix_pix, cur_objix_spix, written);
+                } else {
+                    // on first pass, we load existing object index page
+                    spiffs_page_ix pix;
+                    SPIFFS_DBG("append: "_SPIPRIid" find objix span_ix:"_SPIPRIsp"\n", fd->obj_id, cur_objix_spix);
+                    if (fd->cursor_objix_spix == cur_objix_spix) {
+                        pix = fd->cursor_objix_pix;
+                    } else {
+                        res = spiffs_obj_lu_find_id_and_span(fs, fd->obj_id | SPIFFS_OBJ_ID_IX_FLAG, cur_objix_spix, 0, &pix);
+                        SPIFFS_CHECK_RES(res);
+                    }
+                    SPIFFS_DBG("append: "_SPIPRIid" found object index at page "_SPIPRIpg" [fd size "_SPIPRIi"]\n", fd->obj_id, pix, fd->size);
+                    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ,
+                                     fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+                    SPIFFS_CHECK_RES(res);
+                    SPIFFS_VALIDATE_OBJIX(objix_hdr->p_hdr, fd->obj_id, cur_objix_spix);
+                    cur_objix_pix = pix;
+                }
+                fd->cursor_objix_pix = cur_objix_pix;
+                fd->cursor_objix_spix = cur_objix_spix;
+                fd->offset = offset + written;
+                fd->size = offset + written;
+            }
+            prev_objix_spix = cur_objix_spix;
+        }
+
+        // write data
+        u32_t to_write = MIN(len - written, SPIFFS_DATA_PAGE_SIZE(fs) - page_offs);
+        if (page_offs == 0) {
+            // at beginning of a page, allocate and write a new page of data
+            p_hdr.obj_id = fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG;
+            p_hdr.span_ix = data_spix;
+            p_hdr.flags = 0xff & ~(SPIFFS_PH_FLAG_FINAL);  // finalize immediately
+            res = spiffs_page_allocate_data(fs, fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG,
+                                            &p_hdr, &data[written], to_write, page_offs, 1, &data_page);
+            SPIFFS_DBG("append: "_SPIPRIid" store new data page, "_SPIPRIpg":"_SPIPRIsp" offset:"_SPIPRIi", len "_SPIPRIi", written "_SPIPRIi"\n", fd->obj_id,
+                       data_page, data_spix, page_offs, to_write, written);
+        } else {
+            // append to existing page, fill out free data in existing page
+            if (cur_objix_spix == 0) {
+                // get data page from object index header page
+                data_page = ((spiffs_page_ix *)((u8_t *)objix_hdr + sizeof(spiffs_page_object_ix_header)))[data_spix];
+            } else {
+                // get data page from object index page
+                data_page = ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, data_spix)];
+            }
+
+            res = spiffs_page_data_check(fs, fd, data_page, data_spix);
+            SPIFFS_CHECK_RES(res);
+
+            res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                             fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, data_page) + sizeof(spiffs_page_header) + page_offs, to_write, &data[written]);
+            SPIFFS_DBG("append: "_SPIPRIid" store to existing data page, "_SPIPRIpg":"_SPIPRIsp" offset:"_SPIPRIi", len "_SPIPRIi", written "_SPIPRIi"\n", fd->obj_id
+                       , data_page, data_spix, page_offs, to_write, written);
+        }
+
+        if (res != SPIFFS_OK) break;
+
+        // update memory representation of object index page with new data page
+        if (cur_objix_spix == 0) {
+            // update object index header page
+            ((spiffs_page_ix *)((u8_t *)objix_hdr + sizeof(spiffs_page_object_ix_header)))[data_spix] = data_page;
+            SPIFFS_DBG("append: "_SPIPRIid" wrote page "_SPIPRIpg" to objix_hdr entry "_SPIPRIsp" in mem\n", fd->obj_id
+                       , data_page, data_spix);
+            objix_hdr->size = offset + written;
+        } else {
+            // update object index page
+            ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, data_spix)] = data_page;
+            SPIFFS_DBG("append: "_SPIPRIid" wrote page "_SPIPRIpg" to objix entry "_SPIPRIsp" in mem\n", fd->obj_id
+                       , data_page, (spiffs_span_ix)SPIFFS_OBJ_IX_ENTRY(fs, data_spix));
+        }
+
+        // update internals
+        page_offs = 0;
+        data_spix++;
+        written += to_write;
+    } // while all data
+
+    fd->size = offset + written;
+    fd->offset = offset + written;
+    fd->cursor_objix_pix = cur_objix_pix;
+    fd->cursor_objix_spix = cur_objix_spix;
+
+    // finalize updated object indices
+    s32_t res2 = SPIFFS_OK;
+    if (cur_objix_spix != 0) {
+        // wrote beyond object index header page
+        // write last modified object index page, unless object header index page
+        SPIFFS_DBG("append: "_SPIPRIid" store objix page, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", fd->obj_id,
+                   cur_objix_pix, cur_objix_spix, written);
+
+        res2 = spiffs_page_index_check(fs, fd, cur_objix_pix, cur_objix_spix);
+        SPIFFS_CHECK_RES(res2);
+
+        res2 = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_UPDT,
+                          fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, cur_objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+        SPIFFS_CHECK_RES(res2);
+        spiffs_cb_object_event(fs, (spiffs_page_object_ix *)fs->work,
+                               SPIFFS_EV_IX_UPD, fd->obj_id, objix->p_hdr.span_ix, cur_objix_pix, 0);
+
+        // update size in object header index page
+        res2 = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                              fd->objix_hdr_pix, 0, 0, 0, offset + written, &new_objix_hdr_page);
+        SPIFFS_DBG("append: "_SPIPRIid" store new size II "_SPIPRIi" in objix_hdr, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi", res "_SPIPRIi"\n", fd->obj_id
+                   , offset + written, new_objix_hdr_page, 0, written, res2);
+        SPIFFS_CHECK_RES(res2);
+    } else {
+        // wrote within object index header page
+        if (offset == 0) {
+            // wrote to empty object - simply update size and write whole page
+            objix_hdr->size = offset + written;
+            SPIFFS_DBG("append: "_SPIPRIid" store fresh objix_hdr page, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", fd->obj_id
+                       , cur_objix_pix, cur_objix_spix, written);
+
+            res2 = spiffs_page_index_check(fs, fd, cur_objix_pix, cur_objix_spix);
+            SPIFFS_CHECK_RES(res2);
+
+            res2 = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_UPDT,
+                              fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, cur_objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+            SPIFFS_CHECK_RES(res2);
+            // callback on object index update
+            spiffs_cb_object_event(fs, (spiffs_page_object_ix *)fs->work,
+                                   SPIFFS_EV_IX_UPD_HDR, fd->obj_id, objix_hdr->p_hdr.span_ix, cur_objix_pix, objix_hdr->size);
+        } else {
+            // modifying object index header page, update size and make new copy
+            res2 = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                                  fd->objix_hdr_pix, fs->work, 0, 0, offset + written, &new_objix_hdr_page);
+            SPIFFS_DBG("append: "_SPIPRIid" store modified objix_hdr page, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", fd->obj_id
+                       , new_objix_hdr_page, 0, written);
+            SPIFFS_CHECK_RES(res2);
+        }
+    }
+
+    return res;
+} // spiffs_object_append
+#endif // !SPIFFS_READ_ONLY
+
+#if !SPIFFS_READ_ONLY
+// Modify object
+// keep current object index (header) page in fs->work buffer
+s32_t spiffs_object_modify(spiffs_fd *fd, u32_t offset, u8_t *data, u32_t len) {
+    spiffs *fs = fd->fs;
+    s32_t res = SPIFFS_OK;
+    u32_t written = 0;
+
+    res = spiffs_gc_check(fs, len + SPIFFS_DATA_PAGE_SIZE(fs));
+    SPIFFS_CHECK_RES(res);
+
+    spiffs_page_object_ix_header *objix_hdr = (spiffs_page_object_ix_header *)fs->work;
+    spiffs_page_object_ix *objix = (spiffs_page_object_ix *)fs->work;
+    spiffs_page_header p_hdr;
+
+    spiffs_span_ix cur_objix_spix = 0;
+    spiffs_span_ix prev_objix_spix = (spiffs_span_ix) - 1;
+    spiffs_page_ix cur_objix_pix = fd->objix_hdr_pix;
+    spiffs_page_ix new_objix_hdr_pix;
+
+    spiffs_span_ix data_spix = offset / SPIFFS_DATA_PAGE_SIZE(fs);
+    spiffs_page_ix data_pix;
+    u32_t page_offs = offset % SPIFFS_DATA_PAGE_SIZE(fs);
+
+
+    // write all data
+    while (res == SPIFFS_OK && written < len) {
+        // calculate object index page span index
+        cur_objix_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, data_spix);
+
+        // handle storing and loading of object indices
+        if (cur_objix_spix != prev_objix_spix) {
+            // new object index page
+            // within this clause we return directly if something fails, object index mess-up
+            if (written > 0) {
+                // store previous object index (header) page, unless first pass
+                if (prev_objix_spix == 0) {
+                    // store previous object index header page
+                    res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                                         fd->objix_hdr_pix, fs->work, 0, 0, 0, &new_objix_hdr_pix);
+                    SPIFFS_DBG("modify: store modified objix_hdr page, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", new_objix_hdr_pix, 0, written);
+                    SPIFFS_CHECK_RES(res);
+                } else {
+                    // store new version of previous object index page
+                    spiffs_page_ix new_objix_pix;
+
+                    res = spiffs_page_index_check(fs, fd, cur_objix_pix, prev_objix_spix);
+                    SPIFFS_CHECK_RES(res);
+
+                    res = spiffs_page_move(fs, fd->file_nbr, (u8_t *)objix, fd->obj_id, 0, cur_objix_pix, &new_objix_pix);
+                    SPIFFS_DBG("modify: store previous modified objix page, "_SPIPRIid":"_SPIPRIsp", written "_SPIPRIi"\n", new_objix_pix, objix->p_hdr.span_ix, written);
+                    SPIFFS_CHECK_RES(res);
+                    spiffs_cb_object_event(fs, (spiffs_page_object_ix *)objix,
+                                           SPIFFS_EV_IX_UPD, fd->obj_id, objix->p_hdr.span_ix, new_objix_pix, 0);
+                }
+            }
+
+            // load next object index page
+            if (cur_objix_spix == 0) {
+                // load object index header page, must exist
+                SPIFFS_DBG("modify: load objixhdr page "_SPIPRIpg":"_SPIPRIsp"\n", cur_objix_pix, cur_objix_spix);
+                res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ,
+                                 fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, cur_objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+                SPIFFS_CHECK_RES(res);
+                SPIFFS_VALIDATE_OBJIX(objix_hdr->p_hdr, fd->obj_id, cur_objix_spix);
+            } else {
+                // load existing object index page on first pass
+                spiffs_page_ix pix;
+                SPIFFS_DBG("modify: find objix span_ix:"_SPIPRIsp"\n", cur_objix_spix);
+                if (fd->cursor_objix_spix == cur_objix_spix) {
+                    pix = fd->cursor_objix_pix;
+                } else {
+                    res = spiffs_obj_lu_find_id_and_span(fs, fd->obj_id | SPIFFS_OBJ_ID_IX_FLAG, cur_objix_spix, 0, &pix);
+                    SPIFFS_CHECK_RES(res);
+                }
+                SPIFFS_DBG("modify: found object index at page "_SPIPRIpg"\n", pix);
+                res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ,
+                                 fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+                SPIFFS_CHECK_RES(res);
+                SPIFFS_VALIDATE_OBJIX(objix_hdr->p_hdr, fd->obj_id, cur_objix_spix);
+                cur_objix_pix = pix;
+            }
+            fd->cursor_objix_pix = cur_objix_pix;
+            fd->cursor_objix_spix = cur_objix_spix;
+            fd->offset = offset + written;
+            prev_objix_spix = cur_objix_spix;
+        }
+
+        // write partial data
+        u32_t to_write = MIN(len - written, SPIFFS_DATA_PAGE_SIZE(fs) - page_offs);
+        spiffs_page_ix orig_data_pix;
+        if (cur_objix_spix == 0) {
+            // get data page from object index header page
+            orig_data_pix = ((spiffs_page_ix *)((u8_t *)objix_hdr + sizeof(spiffs_page_object_ix_header)))[data_spix];
+        } else {
+            // get data page from object index page
+            orig_data_pix = ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, data_spix)];
+        }
+
+        p_hdr.obj_id = fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG;
+        p_hdr.span_ix = data_spix;
+        p_hdr.flags = 0xff;
+        if (page_offs == 0 && to_write == SPIFFS_DATA_PAGE_SIZE(fs)) {
+            // a full page, allocate and write a new page of data
+            res = spiffs_page_allocate_data(fs, fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG,
+                                            &p_hdr, &data[written], to_write, page_offs, 1, &data_pix);
+            SPIFFS_DBG("modify: store new data page, "_SPIPRIpg":"_SPIPRIsp" offset:"_SPIPRIi", len "_SPIPRIi", written "_SPIPRIi"\n", data_pix, data_spix, page_offs, to_write, written);
+        } else {
+            // write to existing page, allocate new and copy unmodified data
+
+            res = spiffs_page_data_check(fs, fd, orig_data_pix, data_spix);
+            SPIFFS_CHECK_RES(res);
+
+            res = spiffs_page_allocate_data(fs, fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG,
+                                            &p_hdr, 0, 0, 0, 0, &data_pix);
+            if (res != SPIFFS_OK) break;
+
+            // copy unmodified data
+            if (page_offs > 0) {
+                // before modification
+                res = spiffs_phys_cpy(fs, fd->file_nbr,
+                                      SPIFFS_PAGE_TO_PADDR(fs, data_pix) + sizeof(spiffs_page_header),
+                                      SPIFFS_PAGE_TO_PADDR(fs, orig_data_pix) + sizeof(spiffs_page_header),
+                                      page_offs);
+                if (res != SPIFFS_OK) break;
+            }
+            if (page_offs + to_write < SPIFFS_DATA_PAGE_SIZE(fs)) {
+                // after modification
+                res = spiffs_phys_cpy(fs, fd->file_nbr,
+                                      SPIFFS_PAGE_TO_PADDR(fs, data_pix) + sizeof(spiffs_page_header) + page_offs + to_write,
+                                      SPIFFS_PAGE_TO_PADDR(fs, orig_data_pix) + sizeof(spiffs_page_header) + page_offs + to_write,
+                                      SPIFFS_DATA_PAGE_SIZE(fs) - (page_offs + to_write));
+                if (res != SPIFFS_OK) break;
+            }
+
+            res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                             fd->file_nbr,
+                             SPIFFS_PAGE_TO_PADDR(fs, data_pix) + sizeof(spiffs_page_header) + page_offs, to_write, &data[written]);
+            if (res != SPIFFS_OK) break;
+            p_hdr.flags &= ~SPIFFS_PH_FLAG_FINAL;
+            res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                             fd->file_nbr,
+                             SPIFFS_PAGE_TO_PADDR(fs, data_pix) + offsetof(spiffs_page_header, flags),
+                             sizeof(u8_t),
+                             (u8_t *)&p_hdr.flags);
+            if (res != SPIFFS_OK) break;
+
+            SPIFFS_DBG("modify: store to existing data page, src:"_SPIPRIpg", dst:"_SPIPRIpg":"_SPIPRIsp" offset:"_SPIPRIi", len "_SPIPRIi", written "_SPIPRIi"\n", orig_data_pix, data_pix, data_spix, page_offs, to_write, written);
+        }
+
+        // delete original data page
+        res = spiffs_page_delete(fs, orig_data_pix);
+        if (res != SPIFFS_OK) break;
+        // update memory representation of object index page with new data page
+        if (cur_objix_spix == 0) {
+            // update object index header page
+            ((spiffs_page_ix *)((u8_t *)objix_hdr + sizeof(spiffs_page_object_ix_header)))[data_spix] = data_pix;
+            SPIFFS_DBG("modify: wrote page "_SPIPRIpg" to objix_hdr entry "_SPIPRIsp" in mem\n", data_pix, data_spix);
+        } else {
+            // update object index page
+            ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, data_spix)] = data_pix;
+            SPIFFS_DBG("modify: wrote page "_SPIPRIpg" to objix entry "_SPIPRIsp" in mem\n", data_pix, (spiffs_span_ix)SPIFFS_OBJ_IX_ENTRY(fs, data_spix));
+        }
+
+        // update internals
+        page_offs = 0;
+        data_spix++;
+        written += to_write;
+    } // while all data
+
+    fd->offset = offset + written;
+    fd->cursor_objix_pix = cur_objix_pix;
+    fd->cursor_objix_spix = cur_objix_spix;
+
+    // finalize updated object indices
+    s32_t res2 = SPIFFS_OK;
+    if (cur_objix_spix != 0) {
+        // wrote beyond object index header page
+        // write last modified object index page
+        // move and update page
+        spiffs_page_ix new_objix_pix;
+
+        res2 = spiffs_page_index_check(fs, fd, cur_objix_pix, cur_objix_spix);
+        SPIFFS_CHECK_RES(res2);
+
+        res2 = spiffs_page_move(fs, fd->file_nbr, (u8_t *)objix, fd->obj_id, 0, cur_objix_pix, &new_objix_pix);
+        SPIFFS_DBG("modify: store modified objix page, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", new_objix_pix, cur_objix_spix, written);
+        fd->cursor_objix_pix = new_objix_pix;
+        fd->cursor_objix_spix = cur_objix_spix;
+        SPIFFS_CHECK_RES(res2);
+        spiffs_cb_object_event(fs, (spiffs_page_object_ix *)objix,
+                               SPIFFS_EV_IX_UPD, fd->obj_id, objix->p_hdr.span_ix, new_objix_pix, 0);
+
+    } else {
+        // wrote within object index header page
+        res2 = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                              fd->objix_hdr_pix, fs->work, 0, 0, 0, &new_objix_hdr_pix);
+        SPIFFS_DBG("modify: store modified objix_hdr page, "_SPIPRIpg":"_SPIPRIsp", written "_SPIPRIi"\n", new_objix_hdr_pix, 0, written);
+        SPIFFS_CHECK_RES(res2);
+    }
+
+    return res;
+} // spiffs_object_modify
+#endif // !SPIFFS_READ_ONLY
+
+static s32_t spiffs_object_find_object_index_header_by_name_v(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_block_ix bix,
+    int ix_entry,
+    const void *user_const_p,
+    void *user_var_p) {
+    (void)user_var_p;
+    s32_t res;
+    spiffs_page_object_ix_header objix_hdr;
+    spiffs_page_ix pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, ix_entry);
+    if (obj_id == SPIFFS_OBJ_ID_FREE || obj_id == SPIFFS_OBJ_ID_DELETED ||
+            (obj_id & SPIFFS_OBJ_ID_IX_FLAG) == 0) {
+        return SPIFFS_VIS_COUNTINUE;
+    }
+    res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                     0, SPIFFS_PAGE_TO_PADDR(fs, pix), sizeof(spiffs_page_object_ix_header), (u8_t *)&objix_hdr);
+    SPIFFS_CHECK_RES(res);
+    if (objix_hdr.p_hdr.span_ix == 0 &&
+            (objix_hdr.p_hdr.flags & (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_IXDELE)) ==
+            (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_IXDELE)) {
+        if (strcmp((const char *)user_const_p, (char *)objix_hdr.name) == 0) {
+            return SPIFFS_OK;
+        }
+    }
+
+    return SPIFFS_VIS_COUNTINUE;
+}
+
+// Finds object index header page by name
+s32_t spiffs_object_find_object_index_header_by_name(
+    spiffs *fs,
+    const u8_t name[SPIFFS_OBJ_NAME_LEN],
+    spiffs_page_ix *pix) {
+    s32_t res;
+    spiffs_block_ix bix;
+    int entry;
+
+    res = spiffs_obj_lu_find_entry_visitor(fs,
+                                           fs->cursor_block_ix,
+                                           fs->cursor_obj_lu_entry,
+                                           0,
+                                           0,
+                                           spiffs_object_find_object_index_header_by_name_v,
+                                           name,
+                                           0,
+                                           &bix,
+                                           &entry);
+
+    if (res == SPIFFS_VIS_END) {
+        res = SPIFFS_ERR_NOT_FOUND;
+    }
+    SPIFFS_CHECK_RES(res);
+
+    if (pix) {
+        *pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, entry);
+    }
+
+    fs->cursor_block_ix = bix;
+    fs->cursor_obj_lu_entry = entry;
+
+    return res;
+}
+
+#if !SPIFFS_READ_ONLY
+// Truncates object to new size. If new size is null, object may be removed totally
+s32_t spiffs_object_truncate(
+    spiffs_fd *fd,
+    u32_t new_size,
+    u8_t remove_full) {
+    s32_t res = SPIFFS_OK;
+    spiffs *fs = fd->fs;
+
+    if ((fd->size == SPIFFS_UNDEFINED_LEN || fd->size == 0) && !remove_full) {
+        // no op
+        return res;
+    }
+
+    // need 2 pages if not removing: object index page + possibly chopped data page
+    if (remove_full == 0) {
+        res = spiffs_gc_check(fs, SPIFFS_DATA_PAGE_SIZE(fs) * 2);
+        SPIFFS_CHECK_RES(res);
+    }
+
+    spiffs_page_ix objix_pix = fd->objix_hdr_pix;
+    spiffs_span_ix data_spix = (fd->size > 0 ? fd->size - 1 : 0) / SPIFFS_DATA_PAGE_SIZE(fs);
+    u32_t cur_size = fd->size == (u32_t)SPIFFS_UNDEFINED_LEN ? 0 : fd->size ;
+    spiffs_span_ix cur_objix_spix = 0;
+    spiffs_span_ix prev_objix_spix = (spiffs_span_ix) - 1;
+    spiffs_page_object_ix_header *objix_hdr = (spiffs_page_object_ix_header *)fs->work;
+    spiffs_page_object_ix *objix = (spiffs_page_object_ix *)fs->work;
+    spiffs_page_ix data_pix;
+    spiffs_page_ix new_objix_hdr_pix;
+
+    // before truncating, check if object is to be fully removed and mark this
+    if (remove_full && new_size == 0) {
+        u8_t flags = ~(SPIFFS_PH_FLAG_USED | SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_IXDELE);
+        res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_UPDT,
+                         fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, fd->objix_hdr_pix) + offsetof(spiffs_page_header, flags),
+                         sizeof(u8_t),
+                         (u8_t *)&flags);
+        SPIFFS_CHECK_RES(res);
+    }
+
+    // delete from end of object until desired len is reached
+    while (cur_size > new_size) {
+        cur_objix_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, data_spix);
+
+        // put object index for current data span index in work buffer
+        if (prev_objix_spix != cur_objix_spix) {
+            if (prev_objix_spix != (spiffs_span_ix) - 1) {
+                // remove previous object index page
+                SPIFFS_DBG("truncate: delete objix page "_SPIPRIpg":"_SPIPRIsp"\n", objix_pix, prev_objix_spix);
+
+                res = spiffs_page_index_check(fs, fd, objix_pix, prev_objix_spix);
+                SPIFFS_CHECK_RES(res);
+
+                res = spiffs_page_delete(fs, objix_pix);
+                SPIFFS_CHECK_RES(res);
+                spiffs_cb_object_event(fs, (spiffs_page_object_ix *)0,
+                                       SPIFFS_EV_IX_DEL, fd->obj_id, objix->p_hdr.span_ix, objix_pix, 0);
+                if (prev_objix_spix > 0) {
+                    // Update object index header page, unless we totally want to remove the file.
+                    // If fully removing, we're not keeping consistency as good as when storing the header between chunks,
+                    // would we be aborted. But when removing full files, a crammed system may otherwise
+                    // report ERR_FULL a la windows. We cannot have that.
+                    // Hence, take the risk - if aborted, a file check would free the lost pages and mend things
+                    // as the file is marked as fully deleted in the beginning.
+                    if (remove_full == 0) {
+                        SPIFFS_DBG("truncate: update objix hdr page "_SPIPRIpg":"_SPIPRIsp" to size "_SPIPRIi"\n", fd->objix_hdr_pix, prev_objix_spix, cur_size);
+                        res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                                             fd->objix_hdr_pix, 0, 0, 0, cur_size, &new_objix_hdr_pix);
+                        SPIFFS_CHECK_RES(res);
+                    }
+                    fd->size = cur_size;
+                }
+            }
+            // load current object index (header) page
+            if (cur_objix_spix == 0) {
+                objix_pix = fd->objix_hdr_pix;
+            } else {
+                res = spiffs_obj_lu_find_id_and_span(fs, fd->obj_id | SPIFFS_OBJ_ID_IX_FLAG, cur_objix_spix, 0, &objix_pix);
+                SPIFFS_CHECK_RES(res);
+            }
+
+            SPIFFS_DBG("truncate: load objix page "_SPIPRIpg":"_SPIPRIsp" for data spix:"_SPIPRIsp"\n", objix_pix, cur_objix_spix, data_spix);
+            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ,
+                             fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+            SPIFFS_CHECK_RES(res);
+            SPIFFS_VALIDATE_OBJIX(objix_hdr->p_hdr, fd->obj_id, cur_objix_spix);
+            fd->cursor_objix_pix = objix_pix;
+            fd->cursor_objix_spix = cur_objix_spix;
+            fd->offset = cur_size;
+
+            prev_objix_spix = cur_objix_spix;
+        }
+
+        if (cur_objix_spix == 0) {
+            // get data page from object index header page
+            data_pix = ((spiffs_page_ix *)((u8_t *)objix_hdr + sizeof(spiffs_page_object_ix_header)))[data_spix];
+            ((spiffs_page_ix *)((u8_t *)objix_hdr + sizeof(spiffs_page_object_ix_header)))[data_spix] = SPIFFS_OBJ_ID_FREE;
+        } else {
+            // get data page from object index page
+            data_pix = ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, data_spix)];
+            ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, data_spix)] = SPIFFS_OBJ_ID_FREE;
+        }
+
+        SPIFFS_DBG("truncate: got data pix "_SPIPRIpg"\n", data_pix);
+
+        if (new_size == 0 || remove_full || cur_size - new_size >= SPIFFS_DATA_PAGE_SIZE(fs)) {
+            // delete full data page
+            res = spiffs_page_data_check(fs, fd, data_pix, data_spix);
+            if (res != SPIFFS_ERR_DELETED && res != SPIFFS_OK && res != SPIFFS_ERR_INDEX_REF_FREE) {
+                SPIFFS_DBG("truncate: err validating data pix "_SPIPRIi"\n", res);
+                break;
+            }
+
+            if (res == SPIFFS_OK) {
+                res = spiffs_page_delete(fs, data_pix);
+                if (res != SPIFFS_OK) {
+                    SPIFFS_DBG("truncate: err deleting data pix "_SPIPRIi"\n", res);
+                    break;
+                }
+            } else if (res == SPIFFS_ERR_DELETED || res == SPIFFS_ERR_INDEX_REF_FREE) {
+                res = SPIFFS_OK;
+            }
+
+            // update current size
+            if (cur_size % SPIFFS_DATA_PAGE_SIZE(fs) == 0) {
+                cur_size -= SPIFFS_DATA_PAGE_SIZE(fs);
+            } else {
+                cur_size -= cur_size % SPIFFS_DATA_PAGE_SIZE(fs);
+            }
+            fd->size = cur_size;
+            fd->offset = cur_size;
+            SPIFFS_DBG("truncate: delete data page "_SPIPRIpg" for data spix:"_SPIPRIsp", cur_size:"_SPIPRIi"\n", data_pix, data_spix, cur_size);
+        } else {
+            // delete last page, partially
+            spiffs_page_header p_hdr;
+            spiffs_page_ix new_data_pix;
+            u32_t bytes_to_remove = SPIFFS_DATA_PAGE_SIZE(fs) - (new_size % SPIFFS_DATA_PAGE_SIZE(fs));
+            SPIFFS_DBG("truncate: delete "_SPIPRIi" bytes from data page "_SPIPRIpg" for data spix:"_SPIPRIsp", cur_size:"_SPIPRIi"\n", bytes_to_remove, data_pix, data_spix, cur_size);
+
+            res = spiffs_page_data_check(fs, fd, data_pix, data_spix);
+            if (res != SPIFFS_OK) break;
+
+            p_hdr.obj_id = fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG;
+            p_hdr.span_ix = data_spix;
+            p_hdr.flags = 0xff;
+            // allocate new page and copy unmodified data
+            res = spiffs_page_allocate_data(fs, fd->obj_id & ~SPIFFS_OBJ_ID_IX_FLAG,
+                                            &p_hdr, 0, 0, 0, 0, &new_data_pix);
+            if (res != SPIFFS_OK) break;
+            res = spiffs_phys_cpy(fs, 0,
+                                  SPIFFS_PAGE_TO_PADDR(fs, new_data_pix) + sizeof(spiffs_page_header),
+                                  SPIFFS_PAGE_TO_PADDR(fs, data_pix) + sizeof(spiffs_page_header),
+                                  SPIFFS_DATA_PAGE_SIZE(fs) - bytes_to_remove);
+            if (res != SPIFFS_OK) break;
+            // delete original data page
+            res = spiffs_page_delete(fs, data_pix);
+            if (res != SPIFFS_OK) break;
+            p_hdr.flags &= ~SPIFFS_PH_FLAG_FINAL;
+            res = _spiffs_wr(fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_UPDT,
+                             fd->file_nbr,
+                             SPIFFS_PAGE_TO_PADDR(fs, new_data_pix) + offsetof(spiffs_page_header, flags),
+                             sizeof(u8_t),
+                             (u8_t *)&p_hdr.flags);
+            if (res != SPIFFS_OK) break;
+
+            // update memory representation of object index page with new data page
+            if (cur_objix_spix == 0) {
+                // update object index header page
+                ((spiffs_page_ix *)((u8_t *)objix_hdr + sizeof(spiffs_page_object_ix_header)))[data_spix] = new_data_pix;
+                SPIFFS_DBG("truncate: wrote page "_SPIPRIpg" to objix_hdr entry "_SPIPRIsp" in mem\n", new_data_pix, (spiffs_span_ix)SPIFFS_OBJ_IX_ENTRY(fs, data_spix));
+            } else {
+                // update object index page
+                ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, data_spix)] = new_data_pix;
+                SPIFFS_DBG("truncate: wrote page "_SPIPRIpg" to objix entry "_SPIPRIsp" in mem\n", new_data_pix, (spiffs_span_ix)SPIFFS_OBJ_IX_ENTRY(fs, data_spix));
+            }
+            cur_size = new_size;
+            fd->size = new_size;
+            fd->offset = cur_size;
+            break;
+        }
+        data_spix--;
+    } // while all data
+
+    // update object indices
+    if (cur_objix_spix == 0) {
+        // update object index header page
+        if (cur_size == 0) {
+            if (remove_full) {
+                // remove object altogether
+                SPIFFS_DBG("truncate: remove object index header page "_SPIPRIpg"\n", objix_pix);
+
+                res = spiffs_page_index_check(fs, fd, objix_pix, 0);
+                SPIFFS_CHECK_RES(res);
+
+                res = spiffs_page_delete(fs, objix_pix);
+                SPIFFS_CHECK_RES(res);
+                spiffs_cb_object_event(fs, (spiffs_page_object_ix *)0,
+                                       SPIFFS_EV_IX_DEL, fd->obj_id, 0, objix_pix, 0);
+            } else {
+                // make uninitialized object
+                SPIFFS_DBG("truncate: reset objix_hdr page "_SPIPRIpg"\n", objix_pix);
+                memset(fs->work + sizeof(spiffs_page_object_ix_header), 0xff,
+                       SPIFFS_CFG_LOG_PAGE_SZ(fs) - sizeof(spiffs_page_object_ix_header));
+                res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                                     objix_pix, fs->work, 0, 0, SPIFFS_UNDEFINED_LEN, &new_objix_hdr_pix);
+                SPIFFS_CHECK_RES(res);
+            }
+        } else {
+            // update object index header page
+            SPIFFS_DBGF("truncate: update object index header page with indices and size\n");
+            res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                                 objix_pix, fs->work, 0, 0, cur_size, &new_objix_hdr_pix);
+            SPIFFS_CHECK_RES(res);
+        }
+    } else {
+        // update both current object index page and object index header page
+        spiffs_page_ix new_objix_pix;
+
+        res = spiffs_page_index_check(fs, fd, objix_pix, cur_objix_spix);
+        SPIFFS_CHECK_RES(res);
+
+        // move and update object index page
+        res = spiffs_page_move(fs, fd->file_nbr, (u8_t *)objix_hdr, fd->obj_id, 0, objix_pix, &new_objix_pix);
+        SPIFFS_CHECK_RES(res);
+        spiffs_cb_object_event(fs, (spiffs_page_object_ix *)objix_hdr,
+                               SPIFFS_EV_IX_UPD, fd->obj_id, objix->p_hdr.span_ix, new_objix_pix, 0);
+        SPIFFS_DBG("truncate: store modified objix page, "_SPIPRIpg":"_SPIPRIsp"\n", new_objix_pix, cur_objix_spix);
+        fd->cursor_objix_pix = new_objix_pix;
+        fd->cursor_objix_spix = cur_objix_spix;
+        fd->offset = cur_size;
+        // update object index header page with new size
+        res = spiffs_object_update_index_hdr(fs, fd, fd->obj_id,
+                                             fd->objix_hdr_pix, 0, 0, 0, cur_size, &new_objix_hdr_pix);
+        SPIFFS_CHECK_RES(res);
+    }
+    fd->size = cur_size;
+
+    return res;
+} // spiffs_object_truncate
+#endif // !SPIFFS_READ_ONLY
+
+s32_t spiffs_object_read(
+    spiffs_fd *fd,
+    u32_t offset,
+    u32_t len,
+    u8_t *dst) {
+    s32_t res = SPIFFS_OK;
+    spiffs *fs = fd->fs;
+    spiffs_page_ix objix_pix;
+    spiffs_page_ix data_pix;
+    spiffs_span_ix data_spix = offset / SPIFFS_DATA_PAGE_SIZE(fs);
+    u32_t cur_offset = offset;
+    spiffs_span_ix cur_objix_spix;
+    spiffs_span_ix prev_objix_spix = (spiffs_span_ix) - 1;
+    spiffs_page_object_ix_header *objix_hdr = (spiffs_page_object_ix_header *)fs->work;
+    spiffs_page_object_ix *objix = (spiffs_page_object_ix *)fs->work;
+
+    while (cur_offset < offset + len) {
+#if SPIFFS_IX_MAP
+        // check if we have a memory, index map and if so, if we're within index map's range
+        // and if so, if the entry is populated
+        if (fd->ix_map && data_spix >= fd->ix_map->start_spix && data_spix <= fd->ix_map->end_spix
+                && fd->ix_map->map_buf[data_spix - fd->ix_map->start_spix]) {
+            data_pix = fd->ix_map->map_buf[data_spix - fd->ix_map->start_spix];
+        } else {
+#endif
+            cur_objix_spix = SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, data_spix);
+            if (prev_objix_spix != cur_objix_spix) {
+                // load current object index (header) page
+                if (cur_objix_spix == 0) {
+                    objix_pix = fd->objix_hdr_pix;
+                } else {
+                    SPIFFS_DBG("read: find objix "_SPIPRIid":"_SPIPRIsp"\n", fd->obj_id, cur_objix_spix);
+                    if (fd->cursor_objix_spix == cur_objix_spix) {
+                        objix_pix = fd->cursor_objix_pix;
+                    } else {
+                        res = spiffs_obj_lu_find_id_and_span(fs, fd->obj_id | SPIFFS_OBJ_ID_IX_FLAG, cur_objix_spix, 0, &objix_pix);
+                        SPIFFS_CHECK_RES(res);
+                    }
+                }
+                SPIFFS_DBG("read: load objix page "_SPIPRIpg":"_SPIPRIsp" for data spix:"_SPIPRIsp"\n", objix_pix, cur_objix_spix, data_spix);
+                res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_IX | SPIFFS_OP_C_READ,
+                                 fd->file_nbr, SPIFFS_PAGE_TO_PADDR(fs, objix_pix), SPIFFS_CFG_LOG_PAGE_SZ(fs), fs->work);
+                SPIFFS_CHECK_RES(res);
+                SPIFFS_VALIDATE_OBJIX(objix->p_hdr, fd->obj_id, cur_objix_spix);
+
+                fd->offset = cur_offset;
+                fd->cursor_objix_pix = objix_pix;
+                fd->cursor_objix_spix = cur_objix_spix;
+
+                prev_objix_spix = cur_objix_spix;
+            }
+
+            if (cur_objix_spix == 0) {
+                // get data page from object index header page
+                data_pix = ((spiffs_page_ix *)((u8_t *)objix_hdr + sizeof(spiffs_page_object_ix_header)))[data_spix];
+            } else {
+                // get data page from object index page
+                data_pix = ((spiffs_page_ix *)((u8_t *)objix + sizeof(spiffs_page_object_ix)))[SPIFFS_OBJ_IX_ENTRY(fs, data_spix)];
+            }
+#if SPIFFS_IX_MAP
+        }
+#endif
+        // all remaining data
+        u32_t len_to_read = offset + len - cur_offset;
+        // remaining data in page
+        len_to_read = MIN(len_to_read, SPIFFS_DATA_PAGE_SIZE(fs) - (cur_offset % SPIFFS_DATA_PAGE_SIZE(fs)));
+        // remaining data in file
+        len_to_read = MIN(len_to_read, fd->size - cur_offset);
+        SPIFFS_DBG("read: offset:"_SPIPRIi" rd:"_SPIPRIi" data spix:"_SPIPRIsp" is data_pix:"_SPIPRIpg" addr:"_SPIPRIad"\n", cur_offset, len_to_read, data_spix, data_pix,
+                   (u32_t)(SPIFFS_PAGE_TO_PADDR(fs, data_pix) + sizeof(spiffs_page_header) + (cur_offset % SPIFFS_DATA_PAGE_SIZE(fs))));
+        if (len_to_read <= 0) {
+            res = SPIFFS_ERR_END_OF_OBJECT;
+            break;
+        }
+        res = spiffs_page_data_check(fs, fd, data_pix, data_spix);
+        SPIFFS_CHECK_RES(res);
+        res = _spiffs_rd(
+                  fs, SPIFFS_OP_T_OBJ_DA | SPIFFS_OP_C_READ,
+                  fd->file_nbr,
+                  SPIFFS_PAGE_TO_PADDR(fs, data_pix) + sizeof(spiffs_page_header) + (cur_offset % SPIFFS_DATA_PAGE_SIZE(fs)),
+                  len_to_read,
+                  dst);
+        SPIFFS_CHECK_RES(res);
+        dst += len_to_read;
+        cur_offset += len_to_read;
+        fd->offset = cur_offset;
+        data_spix++;
+    }
+
+    return res;
+}
+
+#if !SPIFFS_READ_ONLY
+typedef struct {
+    spiffs_obj_id min_obj_id;
+    spiffs_obj_id max_obj_id;
+    u32_t compaction;
+    const u8_t *conflicting_name;
+} spiffs_free_obj_id_state;
+
+static s32_t spiffs_obj_lu_find_free_obj_id_bitmap_v(spiffs *fs, spiffs_obj_id id, spiffs_block_ix bix, int ix_entry,
+                                                     const void *user_const_p, void *user_var_p) {
+    if (id != SPIFFS_OBJ_ID_FREE && id != SPIFFS_OBJ_ID_DELETED) {
+        spiffs_obj_id min_obj_id = *((spiffs_obj_id *)user_var_p);
+        const u8_t *conflicting_name = (const u8_t *)user_const_p;
+
+        // if conflicting name parameter is given, also check if this name is found in object index hdrs
+        if (conflicting_name && (id & SPIFFS_OBJ_ID_IX_FLAG)) {
+            spiffs_page_ix pix = SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, bix, ix_entry);
+            int res;
+            spiffs_page_object_ix_header objix_hdr;
+            res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                             0, SPIFFS_PAGE_TO_PADDR(fs, pix), sizeof(spiffs_page_object_ix_header), (u8_t *)&objix_hdr);
+            SPIFFS_CHECK_RES(res);
+            if (objix_hdr.p_hdr.span_ix == 0 &&
+                    (objix_hdr.p_hdr.flags & (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_IXDELE)) ==
+                    (SPIFFS_PH_FLAG_DELET | SPIFFS_PH_FLAG_IXDELE)) {
+                if (strcmp((const char *)user_const_p, (char *)objix_hdr.name) == 0) {
+                    return SPIFFS_ERR_CONFLICTING_NAME;
+                }
+            }
+        }
+
+        id &= ~SPIFFS_OBJ_ID_IX_FLAG;
+        u32_t bit_ix = (id - min_obj_id) & 7;
+        int byte_ix = (id - min_obj_id) >> 3;
+        if (byte_ix >= 0 && (u32_t)byte_ix < SPIFFS_CFG_LOG_PAGE_SZ(fs)) {
+            fs->work[byte_ix] |= (1 << bit_ix);
+        }
+    }
+    return SPIFFS_VIS_COUNTINUE;
+}
+
+static s32_t spiffs_obj_lu_find_free_obj_id_compact_v(spiffs *fs, spiffs_obj_id id, spiffs_block_ix bix, int ix_entry,
+                                                      const void *user_const_p, void *user_var_p) {
+    (void)user_var_p;
+    if (id != SPIFFS_OBJ_ID_FREE && id != SPIFFS_OBJ_ID_DELETED && (id & SPIFFS_OBJ_ID_IX_FLAG)) {
+        s32_t res;
+        const spiffs_free_obj_id_state *state = (const spiffs_free_obj_id_state *)user_const_p;
+        spiffs_page_object_ix_header objix_hdr;
+
+        res = _spiffs_rd(fs, SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
+                         0, SPIFFS_OBJ_LOOKUP_ENTRY_TO_PADDR(fs, bix, ix_entry), sizeof(spiffs_page_object_ix_header), (u8_t *)&objix_hdr);
+        if (res == SPIFFS_OK && objix_hdr.p_hdr.span_ix == 0 &&
+                ((objix_hdr.p_hdr.flags & (SPIFFS_PH_FLAG_INDEX | SPIFFS_PH_FLAG_FINAL | SPIFFS_PH_FLAG_DELET)) ==
+                 (SPIFFS_PH_FLAG_DELET))) {
+            // ok object look up entry
+            if (state->conflicting_name && strcmp((const char *)state->conflicting_name, (char *)objix_hdr.name) == 0) {
+                return SPIFFS_ERR_CONFLICTING_NAME;
+            }
+
+            id &= ~SPIFFS_OBJ_ID_IX_FLAG;
+            if (id >= state->min_obj_id && id <= state->max_obj_id) {
+                u8_t *map = (u8_t *)fs->work;
+                int ix = (id - state->min_obj_id) / state->compaction;
+                //SPIFFS_DBG("free_obj_id: add ix "_SPIPRIi" for id "_SPIPRIid" min"_SPIPRIid" max"_SPIPRIid" comp:"_SPIPRIi"\n", ix, id, state->min_obj_id, state->max_obj_id, state->compaction);
+                map[ix]++;
+            }
+        }
+    }
+    return SPIFFS_VIS_COUNTINUE;
+}
+
+// Scans thru all object lookup for object index header pages. If total possible number of
+// object ids cannot fit into a work buffer, these are grouped. When a group containing free
+// object ids is found, the object lu is again scanned for object ids within group and bitmasked.
+// Finally, the bitmask is searched for a free id
+s32_t spiffs_obj_lu_find_free_obj_id(spiffs *fs, spiffs_obj_id *obj_id, const u8_t *conflicting_name) {
+    s32_t res = SPIFFS_OK;
+    u32_t max_objects = (fs->block_count * SPIFFS_OBJ_LOOKUP_MAX_ENTRIES(fs)) / 2;
+    spiffs_free_obj_id_state state;
+    spiffs_obj_id free_obj_id = SPIFFS_OBJ_ID_FREE;
+    state.min_obj_id = 1;
+    state.max_obj_id = max_objects + 1;
+    if (state.max_obj_id & SPIFFS_OBJ_ID_IX_FLAG) {
+        state.max_obj_id = ((spiffs_obj_id) - 1) & ~SPIFFS_OBJ_ID_IX_FLAG;
+    }
+    state.compaction = 0;
+    state.conflicting_name = conflicting_name;
+    while (res == SPIFFS_OK && free_obj_id == SPIFFS_OBJ_ID_FREE) {
+        if (state.max_obj_id - state.min_obj_id <= (spiffs_obj_id)SPIFFS_CFG_LOG_PAGE_SZ(fs) * 8) {
+            // possible to represent in bitmap
+            u32_t i, j;
+            SPIFFS_DBG("free_obj_id: BITM min:"_SPIPRIid" max:"_SPIPRIid"\n", state.min_obj_id, state.max_obj_id);
+
+            memset(fs->work, 0, SPIFFS_CFG_LOG_PAGE_SZ(fs));
+            res = spiffs_obj_lu_find_entry_visitor(fs, 0, 0, 0, 0, spiffs_obj_lu_find_free_obj_id_bitmap_v,
+                                                   conflicting_name, &state.min_obj_id, 0, 0);
+            if (res == SPIFFS_VIS_END) res = SPIFFS_OK;
+            SPIFFS_CHECK_RES(res);
+            // traverse bitmask until found free obj_id
+            for (i = 0; i < SPIFFS_CFG_LOG_PAGE_SZ(fs); i++) {
+                u8_t mask = fs->work[i];
+                if (mask == 0xff) {
+                    continue;
+                }
+                for (j = 0; j < 8; j++) {
+                    if ((mask & (1 << j)) == 0) {
+                        *obj_id = (i << 3) + j + state.min_obj_id;
+                        return SPIFFS_OK;
+                    }
+                }
+            }
+            return SPIFFS_ERR_FULL;
+        } else {
+            // not possible to represent all ids in range in a bitmap, compact and count
+            if (state.compaction != 0) {
+                // select element in compacted table, decrease range and recompact
+                u32_t i, min_i = 0;
+                u8_t *map = (u8_t *)fs->work;
+                u8_t min_count = 0xff;
+
+                for (i = 0; i < SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(u8_t); i++) {
+                    if (map[i] < min_count) {
+                        min_count = map[i];
+                        min_i = i;
+                        if (min_count == 0) {
+                            break;
+                        }
+                    }
+                }
+
+                if (min_count == state.compaction) {
+                    // there are no free objids!
+                    SPIFFS_DBGF("free_obj_id: compacted table is full\n");
+                    return SPIFFS_ERR_FULL;
+                }
+
+                SPIFFS_DBG("free_obj_id: COMP select index:"_SPIPRIi" min_count:"_SPIPRIi" min:"_SPIPRIid" max:"_SPIPRIid" compact:"_SPIPRIi"\n", min_i, min_count, state.min_obj_id, state.max_obj_id, state.compaction);
+
+                if (min_count == 0) {
+                    // no id in this range, skip compacting and use directly
+                    *obj_id = min_i * state.compaction + state.min_obj_id;
+                    return SPIFFS_OK;
+                } else {
+                    SPIFFS_DBG("free_obj_id: COMP SEL chunk:"_SPIPRIi" min:"_SPIPRIid" -> "_SPIPRIid"\n", state.compaction, state.min_obj_id, state.min_obj_id + min_i *  state.compaction);
+                    state.min_obj_id += min_i *  state.compaction;
+                    state.max_obj_id = state.min_obj_id + state.compaction;
+                    // decrease compaction
+                }
+                if ((state.max_obj_id - state.min_obj_id <= (spiffs_obj_id)SPIFFS_CFG_LOG_PAGE_SZ(fs) * 8)) {
+                    // no need for compacting, use bitmap
+                    continue;
+                }
+            }
+            // in a work memory of log_page_size bytes, we may fit in log_page_size ids
+            // todo what if compaction is > 255 - then we cannot fit it in a byte
+            state.compaction = (state.max_obj_id - state.min_obj_id) / ((SPIFFS_CFG_LOG_PAGE_SZ(fs) / sizeof(u8_t)));
+            SPIFFS_DBG("free_obj_id: COMP min:"_SPIPRIid" max:"_SPIPRIid" compact:"_SPIPRIi"\n", state.min_obj_id, state.max_obj_id, state.compaction);
+
+            memset(fs->work, 0, SPIFFS_CFG_LOG_PAGE_SZ(fs));
+            res = spiffs_obj_lu_find_entry_visitor(fs, 0, 0, 0, 0, spiffs_obj_lu_find_free_obj_id_compact_v, &state, 0, 0, 0);
+            if (res == SPIFFS_VIS_END) res = SPIFFS_OK;
+            SPIFFS_CHECK_RES(res);
+            state.conflicting_name = 0; // searched for conflicting name once, no need to do it again
+        }
+    }
+
+    return res;
+}
+#endif // !SPIFFS_READ_ONLY
+
+#if SPIFFS_TEMPORAL_FD_CACHE
+// djb2 hash
+static u32_t spiffs_hash(spiffs *fs, const u8_t *name) {
+    (void)fs;
+    u32_t hash = 5381;
+    u8_t c;
+    int i = 0;
+    while ((c = name[i++]) && i < SPIFFS_OBJ_NAME_LEN) {
+        hash = (hash * 33) ^ c;
+    }
+    return hash;
+}
+#endif
+
+s32_t spiffs_fd_find_new(spiffs *fs, spiffs_fd **fd, const char *name) {
+#if SPIFFS_TEMPORAL_FD_CACHE
+    u32_t i;
+    u16_t min_score = 0xffff;
+    u32_t cand_ix = (u32_t) -1;
+    u32_t name_hash = name ? spiffs_hash(fs, (const u8_t *)name) : 0;
+    spiffs_fd *fds = (spiffs_fd *)fs->fd_space;
+
+    if (name) {
+        // first, decrease score of all closed descriptors
+        for (i = 0; i < fs->fd_count; i++) {
+            spiffs_fd *cur_fd = &fds[i];
+            if (cur_fd->file_nbr == 0) {
+                if (cur_fd->score > 1) { // score == 0 indicates never used fd
+                    cur_fd->score--;
+                }
+            }
+        }
+    }
+
+    // find the free fd with least score or name match
+    for (i = 0; i < fs->fd_count; i++) {
+        spiffs_fd *cur_fd = &fds[i];
+        if (cur_fd->file_nbr == 0) {
+            if (name && cur_fd->name_hash == name_hash) {
+                cand_ix = i;
+                break;
+            }
+            if (cur_fd->score < min_score) {
+                min_score = cur_fd->score;
+                cand_ix = i;
+            }
+        }
+    }
+
+    if (cand_ix != (u32_t) -1) {
+        spiffs_fd *cur_fd = &fds[cand_ix];
+        if (name) {
+            if (cur_fd->name_hash == name_hash && cur_fd->score > 0) {
+                // opened an fd with same name hash, assume same file
+                // set search point to saved obj index page and hope we have a correct match directly
+                // when start searching - if not, we will just keep searching until it is found
+                fs->cursor_block_ix = SPIFFS_BLOCK_FOR_PAGE(fs, cur_fd->objix_hdr_pix);
+                fs->cursor_obj_lu_entry = SPIFFS_OBJ_LOOKUP_ENTRY_FOR_PAGE(fs, cur_fd->objix_hdr_pix);
+                // update score
+                if (cur_fd->score < 0xffff - SPIFFS_TEMPORAL_CACHE_HIT_SCORE) {
+                    cur_fd->score += SPIFFS_TEMPORAL_CACHE_HIT_SCORE;
+                } else {
+                    cur_fd->score = 0xffff;
+                }
+            } else {
+                // no hash hit, restore this fd to initial state
+                cur_fd->score = SPIFFS_TEMPORAL_CACHE_HIT_SCORE;
+                cur_fd->name_hash = name_hash;
+            }
+        }
+        cur_fd->file_nbr = cand_ix + 1;
+        *fd = cur_fd;
+        return SPIFFS_OK;
+    } else {
+        return SPIFFS_ERR_OUT_OF_FILE_DESCS;
+    }
+#else
+    (void)name;
+    u32_t i;
+    spiffs_fd *fds = (spiffs_fd *)fs->fd_space;
+    for (i = 0; i < fs->fd_count; i++) {
+        spiffs_fd *cur_fd = &fds[i];
+        if (cur_fd->file_nbr == 0) {
+            cur_fd->file_nbr = i + 1;
+            *fd = cur_fd;
+            return SPIFFS_OK;
+        }
+    }
+    return SPIFFS_ERR_OUT_OF_FILE_DESCS;
+#endif
+}
+
+s32_t spiffs_fd_return(spiffs *fs, spiffs_file f) {
+    if (f <= 0 || f > (s16_t)fs->fd_count) {
+        return SPIFFS_ERR_BAD_DESCRIPTOR;
+    }
+    spiffs_fd *fds = (spiffs_fd *)fs->fd_space;
+    spiffs_fd *fd = &fds[f - 1];
+    if (fd->file_nbr == 0) {
+        return SPIFFS_ERR_FILE_CLOSED;
+    }
+    fd->file_nbr = 0;
+#if SPIFFS_IX_MAP
+    fd->ix_map = 0;
+#endif
+    return SPIFFS_OK;
+}
+
+s32_t spiffs_fd_get(spiffs *fs, spiffs_file f, spiffs_fd **fd) {
+    if (f <= 0 || f > (s16_t)fs->fd_count) {
+        return SPIFFS_ERR_BAD_DESCRIPTOR;
+    }
+    spiffs_fd *fds = (spiffs_fd *)fs->fd_space;
+    *fd = &fds[f - 1];
+    if ((*fd)->file_nbr == 0) {
+        return SPIFFS_ERR_FILE_CLOSED;
+    }
+    return SPIFFS_OK;
+}
+
+#if SPIFFS_TEMPORAL_FD_CACHE
+void spiffs_fd_temporal_cache_rehash(
+    spiffs *fs,
+    const char *old_path,
+    const char *new_path) {
+    u32_t i;
+    u32_t old_hash = spiffs_hash(fs, (const u8_t *)old_path);
+    u32_t new_hash = spiffs_hash(fs, (const u8_t *)new_path);
+    spiffs_fd *fds = (spiffs_fd *)fs->fd_space;
+    for (i = 0; i < fs->fd_count; i++) {
+        spiffs_fd *cur_fd = &fds[i];
+        if (cur_fd->score > 0 && cur_fd->name_hash == old_hash) {
+            cur_fd->name_hash = new_hash;
+        }
+    }
+}
+#endif
diff --git a/armsrc/spiffs_nucleus.h b/armsrc/spiffs_nucleus.h
new file mode 100644
index 000000000..c4a360384
--- /dev/null
+++ b/armsrc/spiffs_nucleus.h
@@ -0,0 +1,843 @@
+/*
+ * spiffs_nucleus.h
+ *
+ *  Created on: Jun 15, 2013
+ *      Author: petera
+ */
+
+/* SPIFFS layout
+ *
+ * spiffs is designed for following spi flash characteristics:
+ *   - only big areas of data (blocks) can be erased
+ *   - erasing resets all bits in a block to ones
+ *   - writing pulls ones to zeroes
+ *   - zeroes cannot be pulled to ones, without erase
+ *   - wear leveling
+ *
+ * spiffs is also meant to be run on embedded, memory constraint devices.
+ *
+ * Entire area is divided in blocks. Entire area is also divided in pages.
+ * Each block contains same number of pages. A page cannot be erased, but a
+ * block can be erased.
+ *
+ * Entire area must be block_size * x
+ * page_size must be block_size / (2^y) where y > 2
+ *
+ * ex: area = 1024*1024 bytes, block size = 65536 bytes, page size = 256 bytes
+ *
+ * BLOCK 0  PAGE 0       object lookup 1
+ *          PAGE 1       object lookup 2
+ *          ...
+ *          PAGE n-1     object lookup n
+ *          PAGE n       object data 1
+ *          PAGE n+1     object data 2
+ *          ...
+ *          PAGE n+m-1   object data m
+ *
+ * BLOCK 1  PAGE n+m     object lookup 1
+ *          PAGE n+m+1   object lookup 2
+ *          ...
+ *          PAGE 2n+m-1  object lookup n
+ *          PAGE 2n+m    object data 1
+ *          PAGE 2n+m    object data 2
+ *          ...
+ *          PAGE 2n+2m-1 object data m
+ * ...
+ *
+ * n is number of object lookup pages, which is number of pages needed to index all pages
+ * in a block by object id
+ *   : block_size / page_size * sizeof(obj_id) / page_size
+ * m is number data pages, which is number of pages in block minus number of lookup pages
+ *   : block_size / page_size - block_size / page_size * sizeof(obj_id) / page_size
+ * thus, n+m is total number of pages in a block
+ *   : block_size / page_size
+ *
+ * ex: n = 65536/256*2/256 = 2, m = 65536/256 - 2 = 254 => n+m = 65536/256 = 256
+ *
+ * Object lookup pages contain object id entries. Each entry represent the corresponding
+ * data page.
+ * Assuming a 16 bit object id, an object id being 0xffff represents a free page.
+ * An object id being 0x0000 represents a deleted page.
+ *
+ * ex: page 0 : lookup : 0008 0001 0aaa ffff ffff ffff ffff ffff ..
+ *     page 1 : lookup : ffff ffff ffff ffff ffff ffff ffff ffff ..
+ *     page 2 : data   : data for object id 0008
+ *     page 3 : data   : data for object id 0001
+ *     page 4 : data   : data for object id 0aaa
+ *     ...
+ *
+ *
+ * Object data pages can be either object index pages or object content.
+ * All object data pages contains a data page header, containing object id and span index.
+ * The span index denotes the object page ordering amongst data pages with same object id.
+ * This applies to both object index pages (when index spans more than one page of entries),
+ * and object data pages.
+ * An object index page contains page entries pointing to object content page. The entry index
+ * in a object index page correlates to the span index in the actual object data page.
+ * The first object index page (span index 0) is called object index header page, and also
+ * contains object flags (directory/file), size, object name etc.
+ *
+ * ex:
+ *  BLOCK 1
+ *    PAGE 256: objectl lookup page 1
+ *      [*123] [ 123] [ 123] [ 123]
+ *      [ 123] [*123] [ 123] [ 123]
+ *      [free] [free] [free] [free] ...
+ *    PAGE 257: objectl lookup page 2
+ *      [free] [free] [free] [free] ...
+ *    PAGE 258: object index page (header)
+ *      obj.id:0123 span.ix:0000 flags:INDEX
+ *      size:1600 name:ex.txt type:file
+ *      [259] [260] [261] [262]
+ *    PAGE 259: object data page
+ *      obj.id:0123 span.ix:0000 flags:DATA
+ *    PAGE 260: object data page
+ *      obj.id:0123 span.ix:0001 flags:DATA
+ *    PAGE 261: object data page
+ *      obj.id:0123 span.ix:0002 flags:DATA
+ *    PAGE 262: object data page
+ *      obj.id:0123 span.ix:0003 flags:DATA
+ *    PAGE 263: object index page
+ *      obj.id:0123 span.ix:0001 flags:INDEX
+ *      [264] [265] [fre] [fre]
+ *      [fre] [fre] [fre] [fre]
+ *    PAGE 264: object data page
+ *      obj.id:0123 span.ix:0004 flags:DATA
+ *    PAGE 265: object data page
+ *      obj.id:0123 span.ix:0005 flags:DATA
+ *
+ */
+#ifndef SPIFFS_NUCLEUS_H_
+#define SPIFFS_NUCLEUS_H_
+
+#define _SPIFFS_ERR_CHECK_FIRST         (SPIFFS_ERR_INTERNAL - 1)
+#define SPIFFS_ERR_CHECK_OBJ_ID_MISM    (SPIFFS_ERR_INTERNAL - 1)
+#define SPIFFS_ERR_CHECK_SPIX_MISM      (SPIFFS_ERR_INTERNAL - 2)
+#define SPIFFS_ERR_CHECK_FLAGS_BAD      (SPIFFS_ERR_INTERNAL - 3)
+#define _SPIFFS_ERR_CHECK_LAST          (SPIFFS_ERR_INTERNAL - 4)
+
+// visitor result, continue searching
+#define SPIFFS_VIS_COUNTINUE            (SPIFFS_ERR_INTERNAL - 20)
+// visitor result, continue searching after reloading lu buffer
+#define SPIFFS_VIS_COUNTINUE_RELOAD     (SPIFFS_ERR_INTERNAL - 21)
+// visitor result, stop searching
+#define SPIFFS_VIS_END                  (SPIFFS_ERR_INTERNAL - 22)
+
+// updating an object index contents
+#define SPIFFS_EV_IX_UPD                (0)
+// creating a new object index
+#define SPIFFS_EV_IX_NEW                (1)
+// deleting an object index
+#define SPIFFS_EV_IX_DEL                (2)
+// moving an object index without updating contents
+#define SPIFFS_EV_IX_MOV                (3)
+// updating an object index header data only, not the table itself
+#define SPIFFS_EV_IX_UPD_HDR            (4)
+
+#define SPIFFS_OBJ_ID_IX_FLAG           ((spiffs_obj_id)(1<<(8*sizeof(spiffs_obj_id)-1)))
+
+#define SPIFFS_UNDEFINED_LEN            (u32_t)(-1)
+
+#define SPIFFS_OBJ_ID_DELETED           ((spiffs_obj_id)0)
+#define SPIFFS_OBJ_ID_FREE              ((spiffs_obj_id)-1)
+
+
+
+#if defined(__GNUC__) || defined(__clang__) || defined(__TI_COMPILER_VERSION__)
+/* For GCC, clang and TI compilers */
+#define SPIFFS_PACKED __attribute__((packed))
+#elif defined(__ICCARM__) || defined(__CC_ARM)
+/* For IAR ARM and Keil MDK-ARM compilers */
+#define SPIFFS_PACKED
+
+#else
+/* Unknown compiler */
+#define SPIFFS_PACKED
+#endif
+
+
+
+#if SPIFFS_USE_MAGIC
+#if !SPIFFS_USE_MAGIC_LENGTH
+#define SPIFFS_MAGIC(fs, bix)           \
+  ((spiffs_obj_id)(0x20140529 ^ SPIFFS_CFG_LOG_PAGE_SZ(fs)))
+#else // SPIFFS_USE_MAGIC_LENGTH
+#define SPIFFS_MAGIC(fs, bix)           \
+  ((spiffs_obj_id)(0x20140529 ^ SPIFFS_CFG_LOG_PAGE_SZ(fs) ^ ((fs)->block_count - (bix))))
+#endif // SPIFFS_USE_MAGIC_LENGTH
+#endif // SPIFFS_USE_MAGIC
+
+#define SPIFFS_CONFIG_MAGIC             (0x20090315)
+
+#if SPIFFS_SINGLETON == 0
+#define SPIFFS_CFG_LOG_PAGE_SZ(fs) \
+  ((fs)->cfg.log_page_size)
+#define SPIFFS_CFG_LOG_BLOCK_SZ(fs) \
+  ((fs)->cfg.log_block_size)
+#define SPIFFS_CFG_PHYS_SZ(fs) \
+  ((fs)->cfg.phys_size)
+#define SPIFFS_CFG_PHYS_ERASE_SZ(fs) \
+  ((fs)->cfg.phys_erase_block)
+#define SPIFFS_CFG_PHYS_ADDR(fs) \
+  ((fs)->cfg.phys_addr)
+#endif
+
+// total number of pages
+#define SPIFFS_MAX_PAGES(fs) \
+  ( SPIFFS_CFG_PHYS_SZ(fs)/SPIFFS_CFG_LOG_PAGE_SZ(fs) )
+// total number of pages per block, including object lookup pages
+#define SPIFFS_PAGES_PER_BLOCK(fs) \
+  ( SPIFFS_CFG_LOG_BLOCK_SZ(fs)/SPIFFS_CFG_LOG_PAGE_SZ(fs) )
+// number of object lookup pages per block
+#define SPIFFS_OBJ_LOOKUP_PAGES(fs)     \
+  (MAX(1, (SPIFFS_PAGES_PER_BLOCK(fs) * sizeof(spiffs_obj_id)) / SPIFFS_CFG_LOG_PAGE_SZ(fs)) )
+// checks if page index belongs to object lookup
+#define SPIFFS_IS_LOOKUP_PAGE(fs,pix)     \
+  (((pix) % SPIFFS_PAGES_PER_BLOCK(fs)) < SPIFFS_OBJ_LOOKUP_PAGES(fs))
+// number of object lookup entries in all object lookup pages
+#define SPIFFS_OBJ_LOOKUP_MAX_ENTRIES(fs) \
+  (SPIFFS_PAGES_PER_BLOCK(fs)-SPIFFS_OBJ_LOOKUP_PAGES(fs))
+// converts a block to physical address
+#define SPIFFS_BLOCK_TO_PADDR(fs, block) \
+  ( SPIFFS_CFG_PHYS_ADDR(fs) + (block)* SPIFFS_CFG_LOG_BLOCK_SZ(fs) )
+// converts a object lookup entry to page index
+#define SPIFFS_OBJ_LOOKUP_ENTRY_TO_PIX(fs, block, entry) \
+  ((block)*SPIFFS_PAGES_PER_BLOCK(fs) + (SPIFFS_OBJ_LOOKUP_PAGES(fs) + entry))
+// converts a object lookup entry to physical address of corresponding page
+#define SPIFFS_OBJ_LOOKUP_ENTRY_TO_PADDR(fs, block, entry) \
+  (SPIFFS_BLOCK_TO_PADDR(fs, block) + (SPIFFS_OBJ_LOOKUP_PAGES(fs) + entry) * SPIFFS_CFG_LOG_PAGE_SZ(fs) )
+// converts a page to physical address
+#define SPIFFS_PAGE_TO_PADDR(fs, page) \
+  ( SPIFFS_CFG_PHYS_ADDR(fs) + (page) * SPIFFS_CFG_LOG_PAGE_SZ(fs) )
+// converts a physical address to page
+#define SPIFFS_PADDR_TO_PAGE(fs, addr) \
+  ( ((addr) -  SPIFFS_CFG_PHYS_ADDR(fs)) / SPIFFS_CFG_LOG_PAGE_SZ(fs) )
+// gives index in page for a physical address
+#define SPIFFS_PADDR_TO_PAGE_OFFSET(fs, addr) \
+  ( ((addr) - SPIFFS_CFG_PHYS_ADDR(fs)) % SPIFFS_CFG_LOG_PAGE_SZ(fs) )
+// returns containing block for given page
+#define SPIFFS_BLOCK_FOR_PAGE(fs, page) \
+  ( (page) / SPIFFS_PAGES_PER_BLOCK(fs) )
+// returns starting page for block
+#define SPIFFS_PAGE_FOR_BLOCK(fs, block) \
+  ( (block) * SPIFFS_PAGES_PER_BLOCK(fs) )
+// converts page to entry in object lookup page
+#define SPIFFS_OBJ_LOOKUP_ENTRY_FOR_PAGE(fs, page) \
+  ( (page) % SPIFFS_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs) )
+// returns data size in a data page
+#define SPIFFS_DATA_PAGE_SIZE(fs) \
+    ( SPIFFS_CFG_LOG_PAGE_SZ(fs) - sizeof(spiffs_page_header) )
+// returns physical address for block's erase count,
+// always in the physical last entry of the last object lookup page
+#define SPIFFS_ERASE_COUNT_PADDR(fs, bix) \
+  ( SPIFFS_BLOCK_TO_PADDR(fs, bix) + SPIFFS_OBJ_LOOKUP_PAGES(fs) * SPIFFS_CFG_LOG_PAGE_SZ(fs) - sizeof(spiffs_obj_id) )
+// returns physical address for block's magic,
+// always in the physical second last entry of the last object lookup page
+#define SPIFFS_MAGIC_PADDR(fs, bix) \
+  ( SPIFFS_BLOCK_TO_PADDR(fs, bix) + SPIFFS_OBJ_LOOKUP_PAGES(fs) * SPIFFS_CFG_LOG_PAGE_SZ(fs) - sizeof(spiffs_obj_id)*2 )
+// checks if there is any room for magic in the object luts
+#define SPIFFS_CHECK_MAGIC_POSSIBLE(fs) \
+  ( (SPIFFS_OBJ_LOOKUP_MAX_ENTRIES(fs) % (SPIFFS_CFG_LOG_PAGE_SZ(fs)/sizeof(spiffs_obj_id))) * sizeof(spiffs_obj_id) \
+    <= (SPIFFS_CFG_LOG_PAGE_SZ(fs)-sizeof(spiffs_obj_id)*2) )
+
+// define helpers object
+
+// entries in an object header page index
+#define SPIFFS_OBJ_HDR_IX_LEN(fs) \
+  ((SPIFFS_CFG_LOG_PAGE_SZ(fs) - sizeof(spiffs_page_object_ix_header))/sizeof(spiffs_page_ix))
+// entries in an object page index
+#define SPIFFS_OBJ_IX_LEN(fs) \
+  ((SPIFFS_CFG_LOG_PAGE_SZ(fs) - sizeof(spiffs_page_object_ix))/sizeof(spiffs_page_ix))
+// object index entry for given data span index
+#define SPIFFS_OBJ_IX_ENTRY(fs, spix) \
+  ((spix) < SPIFFS_OBJ_HDR_IX_LEN(fs) ? (spix) : (((spix)-SPIFFS_OBJ_HDR_IX_LEN(fs))%SPIFFS_OBJ_IX_LEN(fs)))
+// object index span index number for given data span index or entry
+#define SPIFFS_OBJ_IX_ENTRY_SPAN_IX(fs, spix) \
+  ((spix) < SPIFFS_OBJ_HDR_IX_LEN(fs) ? 0 : (1+((spix)-SPIFFS_OBJ_HDR_IX_LEN(fs))/SPIFFS_OBJ_IX_LEN(fs)))
+// get data span index for object index span index
+#define SPIFFS_DATA_SPAN_IX_FOR_OBJ_IX_SPAN_IX(fs, spix) \
+  ( (spix) == 0 ? 0 : (SPIFFS_OBJ_HDR_IX_LEN(fs) + (((spix)-1) * SPIFFS_OBJ_IX_LEN(fs))) )
+
+#if SPIFFS_FILEHDL_OFFSET
+#define SPIFFS_FH_OFFS(fs, fh)   ((fh) != 0 ? ((fh) + (fs)->cfg.fh_ix_offset) : 0)
+#define SPIFFS_FH_UNOFFS(fs, fh) ((fh) != 0 ? ((fh) - (fs)->cfg.fh_ix_offset) : 0)
+#else
+#define SPIFFS_FH_OFFS(fs, fh)   (fh)
+#define SPIFFS_FH_UNOFFS(fs, fh) (fh)
+#endif
+
+
+#define SPIFFS_OP_T_OBJ_LU    (0<<0)
+#define SPIFFS_OP_T_OBJ_LU2   (1<<0)
+#define SPIFFS_OP_T_OBJ_IX    (2<<0)
+#define SPIFFS_OP_T_OBJ_DA    (3<<0)
+#define SPIFFS_OP_C_DELE      (0<<2)
+#define SPIFFS_OP_C_UPDT      (1<<2)
+#define SPIFFS_OP_C_MOVS      (2<<2)
+#define SPIFFS_OP_C_MOVD      (3<<2)
+#define SPIFFS_OP_C_FLSH      (4<<2)
+#define SPIFFS_OP_C_READ      (5<<2)
+#define SPIFFS_OP_C_WRTHRU    (6<<2)
+
+#define SPIFFS_OP_TYPE_MASK (3<<0)
+#define SPIFFS_OP_COM_MASK  (7<<2)
+
+
+// if 0, this page is written to, else clean
+#define SPIFFS_PH_FLAG_USED   (1<<0)
+// if 0, writing is finalized, else under modification
+#define SPIFFS_PH_FLAG_FINAL  (1<<1)
+// if 0, this is an index page, else a data page
+#define SPIFFS_PH_FLAG_INDEX  (1<<2)
+// if 0, page is deleted, else valid
+#define SPIFFS_PH_FLAG_DELET  (1<<7)
+// if 0, this index header is being deleted
+#define SPIFFS_PH_FLAG_IXDELE (1<<6)
+
+
+#define SPIFFS_CHECK_MOUNT(fs) \
+  ((fs)->mounted != 0)
+
+#define SPIFFS_CHECK_CFG(fs) \
+  ((fs)->config_magic == SPIFFS_CONFIG_MAGIC)
+
+#define SPIFFS_CHECK_RES(res) \
+  do { \
+    if ((res) < SPIFFS_OK) return (res); \
+  } while (0);
+
+#define SPIFFS_API_CHECK_MOUNT(fs) \
+  if (!SPIFFS_CHECK_MOUNT((fs))) { \
+    (fs)->err_code = SPIFFS_ERR_NOT_MOUNTED; \
+    return SPIFFS_ERR_NOT_MOUNTED; \
+  }
+
+#define SPIFFS_API_CHECK_CFG(fs) \
+  if (!SPIFFS_CHECK_CFG((fs))) { \
+    (fs)->err_code = SPIFFS_ERR_NOT_CONFIGURED; \
+    return SPIFFS_ERR_NOT_CONFIGURED; \
+  }
+
+#define SPIFFS_API_CHECK_RES(fs, res) \
+  if ((res) < SPIFFS_OK) { \
+    (fs)->err_code = (res); \
+    return (res); \
+  }
+
+#define SPIFFS_API_CHECK_RES_UNLOCK(fs, res) \
+  if ((res) < SPIFFS_OK) { \
+    (fs)->err_code = (res); \
+    SPIFFS_UNLOCK(fs); \
+    return (res); \
+  }
+
+#define SPIFFS_VALIDATE_OBJIX(ph, objid, spix) \
+    if (((ph).flags & SPIFFS_PH_FLAG_USED) != 0) return SPIFFS_ERR_IS_FREE; \
+    if (((ph).flags & SPIFFS_PH_FLAG_DELET) == 0) return SPIFFS_ERR_DELETED; \
+    if (((ph).flags & SPIFFS_PH_FLAG_FINAL) != 0) return SPIFFS_ERR_NOT_FINALIZED; \
+    if (((ph).flags & SPIFFS_PH_FLAG_INDEX) != 0) return SPIFFS_ERR_NOT_INDEX; \
+    if (((objid) & SPIFFS_OBJ_ID_IX_FLAG) == 0) return SPIFFS_ERR_NOT_INDEX; \
+    if ((ph).span_ix != (spix)) return SPIFFS_ERR_INDEX_SPAN_MISMATCH;
+//if ((spix) == 0 && ((ph).flags & SPIFFS_PH_FLAG_IXDELE) == 0) return SPIFFS_ERR_DELETED;
+
+#define SPIFFS_VALIDATE_DATA(ph, objid, spix) \
+    if (((ph).flags & SPIFFS_PH_FLAG_USED) != 0) return SPIFFS_ERR_IS_FREE; \
+    if (((ph).flags & SPIFFS_PH_FLAG_DELET) == 0) return SPIFFS_ERR_DELETED; \
+    if (((ph).flags & SPIFFS_PH_FLAG_FINAL) != 0) return SPIFFS_ERR_NOT_FINALIZED; \
+    if (((ph).flags & SPIFFS_PH_FLAG_INDEX) == 0) return SPIFFS_ERR_IS_INDEX; \
+    if ((objid) & SPIFFS_OBJ_ID_IX_FLAG) return SPIFFS_ERR_IS_INDEX; \
+    if ((ph).span_ix != (spix)) return SPIFFS_ERR_DATA_SPAN_MISMATCH;
+
+
+// check id, only visit matching objec ids
+#define SPIFFS_VIS_CHECK_ID     (1<<0)
+// report argument object id to visitor - else object lookup id is reported
+#define SPIFFS_VIS_CHECK_PH     (1<<1)
+// stop searching at end of all look up pages
+#define SPIFFS_VIS_NO_WRAP      (1<<2)
+
+#if SPIFFS_HAL_CALLBACK_EXTRA
+
+#define SPIFFS_HAL_WRITE(_fs, _paddr, _len, _src) \
+  (_fs)->cfg.hal_write_f((_fs), (_paddr), (_len), (_src))
+#define SPIFFS_HAL_READ(_fs, _paddr, _len, _dst) \
+  (_fs)->cfg.hal_read_f((_fs), (_paddr), (_len), (_dst))
+#define SPIFFS_HAL_ERASE(_fs, _paddr, _len) \
+  (_fs)->cfg.hal_erase_f((_fs), (_paddr), (_len))
+
+#else // SPIFFS_HAL_CALLBACK_EXTRA
+
+#define SPIFFS_HAL_WRITE(_fs, _paddr, _len, _src) \
+  (_fs)->cfg.hal_write_f((_paddr), (_len), (_src))
+#define SPIFFS_HAL_READ(_fs, _paddr, _len, _dst) \
+  (_fs)->cfg.hal_read_f((_paddr), (_len), (_dst))
+#define SPIFFS_HAL_ERASE(_fs, _paddr, _len) \
+  (_fs)->cfg.hal_erase_f((_paddr), (_len))
+
+#endif // SPIFFS_HAL_CALLBACK_EXTRA
+
+#if SPIFFS_CACHE
+
+#define SPIFFS_CACHE_FLAG_DIRTY       (1<<0)
+#define SPIFFS_CACHE_FLAG_WRTHRU      (1<<1)
+#define SPIFFS_CACHE_FLAG_OBJLU       (1<<2)
+#define SPIFFS_CACHE_FLAG_OBJIX       (1<<3)
+#define SPIFFS_CACHE_FLAG_DATA        (1<<4)
+#define SPIFFS_CACHE_FLAG_TYPE_WR     (1<<7)
+
+#define SPIFFS_CACHE_PAGE_SIZE(fs) \
+  (sizeof(spiffs_cache_page) + SPIFFS_CFG_LOG_PAGE_SZ(fs))
+
+#define spiffs_get_cache(fs) \
+  ((spiffs_cache *)((fs)->cache))
+
+#define spiffs_get_cache_page_hdr(fs, c, ix) \
+  ((spiffs_cache_page *)(&((c)->cpages[(ix) * SPIFFS_CACHE_PAGE_SIZE(fs)])))
+
+#define spiffs_get_cache_page(fs, c, ix) \
+  ((u8_t *)(&((c)->cpages[(ix) * SPIFFS_CACHE_PAGE_SIZE(fs)])) + sizeof(spiffs_cache_page))
+
+// cache page struct
+typedef struct {
+    // cache flags
+    u8_t flags;
+    // cache page index
+    u8_t ix;
+    // last access of this cache page
+    u32_t last_access;
+    union {
+        // type read cache
+        struct spix {
+            // read cache page index
+            spiffs_page_ix pix;
+        } spix;
+#if SPIFFS_CACHE_WR
+        // type write cache
+        struct swrc {
+            // write cache
+            spiffs_obj_id obj_id;
+            // offset in cache page
+            u32_t offset;
+            // size of cache page
+            u16_t size;
+        } swrc;
+#endif
+    } ucache;
+} spiffs_cache_page;
+
+// cache struct
+typedef struct {
+    u8_t cpage_count;
+    u32_t last_access;
+    u32_t cpage_use_map;
+    u32_t cpage_use_mask;
+    u8_t *cpages;
+} spiffs_cache;
+
+#endif
+
+
+// spiffs nucleus file descriptor
+typedef struct {
+    // the filesystem of this descriptor
+    spiffs *fs;
+    // number of file descriptor - if 0, the file descriptor is closed
+    spiffs_file file_nbr;
+    // object id - if SPIFFS_OBJ_ID_ERASED, the file was deleted
+    spiffs_obj_id obj_id;
+    // size of the file
+    u32_t size;
+    // cached object index header page index
+    spiffs_page_ix objix_hdr_pix;
+    // cached offset object index page index
+    spiffs_page_ix cursor_objix_pix;
+    // cached offset object index span index
+    spiffs_span_ix cursor_objix_spix;
+    // current absolute offset
+    u32_t offset;
+    // current file descriptor offset (cached)
+    u32_t fdoffset;
+    // fd flags
+    spiffs_flags flags;
+#if SPIFFS_CACHE_WR
+    spiffs_cache_page *cache_page;
+#endif
+#if SPIFFS_TEMPORAL_FD_CACHE
+    // djb2 hash of filename
+    u32_t name_hash;
+    // hit score (score == 0 indicates never used fd)
+    u16_t score;
+#endif
+#if SPIFFS_IX_MAP
+    // spiffs index map, if 0 it means unmapped
+    spiffs_ix_map *ix_map;
+#endif
+} spiffs_fd;
+
+
+// object structs
+
+// page header, part of each page except object lookup pages
+// NB: this is always aligned when the data page is an object index,
+// as in this case struct spiffs_page_object_ix is used
+typedef struct SPIFFS_PACKED {
+    // object id
+    spiffs_obj_id obj_id;
+    // object span index
+    spiffs_span_ix span_ix;
+    // flags
+    u8_t flags;
+} spiffs_page_header;
+
+// object index header page header
+typedef struct SPIFFS_PACKED
+#if SPIFFS_ALIGNED_OBJECT_INDEX_TABLES
+__attribute((aligned(sizeof(spiffs_page_ix))))
+#endif
+{
+    // common page header
+    spiffs_page_header p_hdr;
+    // alignment
+    u8_t _align[4 - ((sizeof(spiffs_page_header) & 3) == 0 ? 4 : (sizeof(spiffs_page_header) & 3))];
+    // size of object
+    u32_t size;
+    // type of object
+    spiffs_obj_type type;
+    // name of object
+    u8_t name[SPIFFS_OBJ_NAME_LEN];
+#if SPIFFS_OBJ_META_LEN
+    // metadata. not interpreted by SPIFFS in any way.
+    u8_t meta[SPIFFS_OBJ_META_LEN];
+#endif
+}
+spiffs_page_object_ix_header;
+
+// object index page header
+typedef struct SPIFFS_PACKED {
+    spiffs_page_header p_hdr;
+    u8_t _align[4 - ((sizeof(spiffs_page_header) & 3) == 0 ? 4 : (sizeof(spiffs_page_header) & 3))];
+} spiffs_page_object_ix;
+
+// callback func for object lookup visitor
+typedef s32_t (*spiffs_visitor_f)(spiffs *fs, spiffs_obj_id id, spiffs_block_ix bix, int ix_entry,
+                                  const void *user_const_p, void *user_var_p);
+
+
+#if SPIFFS_CACHE
+#define _spiffs_rd(fs, op, fh, addr, len, dst) \
+    spiffs_phys_rd((fs), (op), (fh), (addr), (len), (dst))
+#define _spiffs_wr(fs, op, fh, addr, len, src) \
+    spiffs_phys_wr((fs), (op), (fh), (addr), (len), (src))
+#else
+#define _spiffs_rd(fs, op, fh, addr, len, dst) \
+    spiffs_phys_rd((fs), (addr), (len), (dst))
+#define _spiffs_wr(fs, op, fh, addr, len, src) \
+    spiffs_phys_wr((fs), (addr), (len), (src))
+#endif
+
+#ifndef MIN
+#define MIN(a,b) ((a) < (b) ? (a) : (b))
+#endif
+#ifndef MAX
+#define MAX(a,b) ((a) > (b) ? (a) : (b))
+#endif
+
+// ---------------
+
+s32_t spiffs_phys_rd(
+    spiffs *fs,
+#if SPIFFS_CACHE
+    u8_t op,
+    spiffs_file fh,
+#endif
+    u32_t addr,
+    u32_t len,
+    u8_t *dst);
+
+s32_t spiffs_phys_wr(
+    spiffs *fs,
+#if SPIFFS_CACHE
+    u8_t op,
+    spiffs_file fh,
+#endif
+    u32_t addr,
+    u32_t len,
+    u8_t *src);
+
+s32_t spiffs_phys_cpy(
+    spiffs *fs,
+    spiffs_file fh,
+    u32_t dst,
+    u32_t src,
+    u32_t len);
+
+s32_t spiffs_phys_count_free_blocks(
+    spiffs *fs);
+
+s32_t spiffs_obj_lu_find_entry_visitor(
+    spiffs *fs,
+    spiffs_block_ix starting_block,
+    int starting_lu_entry,
+    u8_t flags,
+    spiffs_obj_id obj_id,
+    spiffs_visitor_f v,
+    const void *user_const_p,
+    void *user_var_p,
+    spiffs_block_ix *block_ix,
+    int *lu_entry);
+
+s32_t spiffs_erase_block(
+    spiffs *fs,
+    spiffs_block_ix bix);
+
+#if SPIFFS_USE_MAGIC && SPIFFS_USE_MAGIC_LENGTH
+s32_t spiffs_probe(
+    spiffs_config *cfg);
+#endif // SPIFFS_USE_MAGIC && SPIFFS_USE_MAGIC_LENGTH
+
+// ---------------
+
+s32_t spiffs_obj_lu_scan(
+    spiffs *fs);
+
+s32_t spiffs_obj_lu_find_free_obj_id(
+    spiffs *fs,
+    spiffs_obj_id *obj_id,
+    const u8_t *conflicting_name);
+
+s32_t spiffs_obj_lu_find_free(
+    spiffs *fs,
+    spiffs_block_ix starting_block,
+    int starting_lu_entry,
+    spiffs_block_ix *block_ix,
+    int *lu_entry);
+
+s32_t spiffs_obj_lu_find_id(
+    spiffs *fs,
+    spiffs_block_ix starting_block,
+    int starting_lu_entry,
+    spiffs_obj_id obj_id,
+    spiffs_block_ix *block_ix,
+    int *lu_entry);
+
+s32_t spiffs_obj_lu_find_id_and_span(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_span_ix spix,
+    spiffs_page_ix exclusion_pix,
+    spiffs_page_ix *pix);
+
+s32_t spiffs_obj_lu_find_id_and_span_by_phdr(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_span_ix spix,
+    spiffs_page_ix exclusion_pix,
+    spiffs_page_ix *pix);
+
+// ---------------
+
+s32_t spiffs_page_allocate_data(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_page_header *ph,
+    u8_t *data,
+    u32_t len,
+    u32_t page_offs,
+    u8_t finalize,
+    spiffs_page_ix *pix);
+
+s32_t spiffs_page_move(
+    spiffs *fs,
+    spiffs_file fh,
+    u8_t *page_data,
+    spiffs_obj_id obj_id,
+    spiffs_page_header *page_hdr,
+    spiffs_page_ix src_pix,
+    spiffs_page_ix *dst_pix);
+
+s32_t spiffs_page_delete(
+    spiffs *fs,
+    spiffs_page_ix pix);
+
+// ---------------
+
+s32_t spiffs_object_create(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    const u8_t name[],
+    const u8_t meta[],
+    spiffs_obj_type type,
+    spiffs_page_ix *objix_hdr_pix);
+
+s32_t spiffs_object_update_index_hdr(
+    spiffs *fs,
+    spiffs_fd *fd,
+    spiffs_obj_id obj_id,
+    spiffs_page_ix objix_hdr_pix,
+    u8_t *new_objix_hdr_data,
+    const u8_t name[],
+    const u8_t meta[],
+    u32_t size,
+    spiffs_page_ix *new_pix);
+
+#if SPIFFS_IX_MAP
+
+s32_t spiffs_populate_ix_map(
+    spiffs *fs,
+    spiffs_fd *fd,
+    u32_t vec_entry_start,
+    u32_t vec_entry_end);
+
+#endif
+
+void spiffs_cb_object_event(
+    spiffs *fs,
+    spiffs_page_object_ix *objix,
+    int ev,
+    spiffs_obj_id obj_id,
+    spiffs_span_ix spix,
+    spiffs_page_ix new_pix,
+    u32_t new_size);
+
+s32_t spiffs_object_open_by_id(
+    spiffs *fs,
+    spiffs_obj_id obj_id,
+    spiffs_fd *f,
+    spiffs_flags flags,
+    spiffs_mode mode);
+
+s32_t spiffs_object_open_by_page(
+    spiffs *fs,
+    spiffs_page_ix pix,
+    spiffs_fd *f,
+    spiffs_flags flags,
+    spiffs_mode mode);
+
+s32_t spiffs_object_append(
+    spiffs_fd *fd,
+    u32_t offset,
+    u8_t *data,
+    u32_t len);
+
+s32_t spiffs_object_modify(
+    spiffs_fd *fd,
+    u32_t offset,
+    u8_t *data,
+    u32_t len);
+
+s32_t spiffs_object_read(
+    spiffs_fd *fd,
+    u32_t offset,
+    u32_t len,
+    u8_t *dst);
+
+s32_t spiffs_object_truncate(
+    spiffs_fd *fd,
+    u32_t new_len,
+    u8_t remove_object);
+
+s32_t spiffs_object_find_object_index_header_by_name(
+    spiffs *fs,
+    const u8_t name[SPIFFS_OBJ_NAME_LEN],
+    spiffs_page_ix *pix);
+
+// ---------------
+
+s32_t spiffs_gc_check(
+    spiffs *fs,
+    u32_t len);
+
+s32_t spiffs_gc_erase_page_stats(
+    spiffs *fs,
+    spiffs_block_ix bix);
+
+s32_t spiffs_gc_find_candidate(
+    spiffs *fs,
+    spiffs_block_ix **block_candidate,
+    int *candidate_count,
+    char fs_crammed);
+
+s32_t spiffs_gc_clean(
+    spiffs *fs,
+    spiffs_block_ix bix);
+
+s32_t spiffs_gc_quick(
+    spiffs *fs, u16_t max_free_pages);
+
+// ---------------
+
+s32_t spiffs_fd_find_new(
+    spiffs *fs,
+    spiffs_fd **fd,
+    const char *name);
+
+s32_t spiffs_fd_return(
+    spiffs *fs,
+    spiffs_file f);
+
+s32_t spiffs_fd_get(
+    spiffs *fs,
+    spiffs_file f,
+    spiffs_fd **fd);
+
+#if SPIFFS_TEMPORAL_FD_CACHE
+void spiffs_fd_temporal_cache_rehash(
+    spiffs *fs,
+    const char *old_path,
+    const char *new_path);
+#endif
+
+#if SPIFFS_CACHE
+void spiffs_cache_init(
+    spiffs *fs);
+
+void spiffs_cache_drop_page(
+    spiffs *fs,
+    spiffs_page_ix pix);
+
+#if SPIFFS_CACHE_WR
+spiffs_cache_page *spiffs_cache_page_allocate_by_fd(
+    spiffs *fs,
+    spiffs_fd *fd);
+
+void spiffs_cache_fd_release(
+    spiffs *fs,
+    spiffs_cache_page *cp);
+
+spiffs_cache_page *spiffs_cache_page_get_by_fd(
+    spiffs *fs,
+    spiffs_fd *fd);
+#endif
+#endif
+
+s32_t spiffs_lookup_consistency_check(
+    spiffs *fs,
+    u8_t check_all_objects);
+
+s32_t spiffs_page_consistency_check(
+    spiffs *fs);
+
+s32_t spiffs_object_index_consistency_check(
+    spiffs *fs);
+
+// memcpy macro,
+// checked in test builds, otherwise plain memcpy (unless already defined)
+#ifdef _SPIFFS_TEST
+#define _SPIFFS_MEMCPY(__d, __s, __l) do { \
+    intptr_t __a1 = (intptr_t)((u8_t*)(__s)); \
+    intptr_t __a2 = (intptr_t)((u8_t*)(__s)+(__l)); \
+    intptr_t __b1 = (intptr_t)((u8_t*)(__d)); \
+    intptr_t __b2 = (intptr_t)((u8_t*)(__d)+(__l)); \
+    if (__a1 <= __b2 && __b1 <= __a2) { \
+      printf("FATAL OVERLAP: memcpy from %lx..%lx to %lx..%lx\n", __a1, __a2, __b1, __b2); \
+      ERREXIT(); \
+    } \
+    memcpy((__d),(__s),(__l)); \
+} while (0)
+#else
+#ifndef _SPIFFS_MEMCPY
+#define _SPIFFS_MEMCPY(__d, __s, __l) do{memcpy((__d),(__s),(__l));}while(0)
+#endif
+#endif //_SPIFFS_TEST
+
+#endif /* SPIFFS_NUCLEUS_H_ */
diff --git a/armsrc/string.c b/armsrc/string.c
index 5c154643c..ea113680e 100644
--- a/armsrc/string.c
+++ b/armsrc/string.c
@@ -106,3 +106,92 @@ void itoa(int n, char s[]) {
 }
 
 //////////////////////////////////////// END 'itoa' CODE
+
+
+
+char *strcpy(char *dst, const char *src) {
+    char *save = dst;
+
+    for (; (*dst = *src) != '\0'; ++src, ++dst);
+    return save;
+}
+
+char *strncpy(char *dst, const char *src, size_t n) {
+    if (n != 0) {
+        char *d = dst;
+        const char *s = src;
+
+        do {
+            if ((*d++ = *s++) == 0) {
+                /* NUL pad the remaining n-1 bytes */
+                while (--n) {
+                    *d++ = 0;
+                }
+                break;
+            }
+        } while (--n);
+    }
+    return dst;
+}
+
+int strcmp(const char *s1, const char *s2) {
+    while (*s1 == *s2++) {
+        if (*s1++ == 0) {
+            return (0);
+        }
+    }
+    return (*(unsigned char *) s1 - * (unsigned char *) --s2);
+}
+
+char *__strtok_r(char *, const char *, char **);
+
+char *__strtok_r(char *s, const char *delim, char **last) {
+    char *spanp, *tok;
+    int c, sc;
+
+    if (s == NULL && (s = *last) == NULL)
+        return (NULL);
+
+    /*
+     * Skip (span) leading delimiters (s += strspn(s, delim), sort of).
+     */
+cont:
+    c = *s++;
+    for (spanp = (char *)delim; (sc = *spanp++) != 0;) {
+        if (c == sc)
+            goto cont;
+    }
+
+    if (c == 0) {
+        /* no non-delimiter characters */
+        *last = NULL;
+        return (NULL);
+    }
+    tok = s - 1;
+
+    /*
+     * Scan token (scan for delimiters: s += strcspn(s, delim), sort of).
+     * Note that delim must have one NUL; we stop if we see that, too.
+     */
+    for (;;) {
+        c = *s++;
+        spanp = (char *)delim;
+        do {
+            if ((sc = *spanp++) == c) {
+                if (c == 0)
+                    s = NULL;
+                else
+                    s[-1] = '\0';
+                *last = s;
+                return (tok);
+            }
+        } while (sc != 0);
+    }
+    /* NOTREACHED */
+}
+
+char *strtok(char *s, const char *delim) {
+    static char *last;
+
+    return (__strtok_r(s, delim, &last));
+}
diff --git a/armsrc/string.h b/armsrc/string.h
index f6f094c48..c30293a2a 100644
--- a/armsrc/string.h
+++ b/armsrc/string.h
@@ -23,5 +23,9 @@ char *strncat(char *dest, const char *src, unsigned int n);
 char *strcat(char *dest, const char *src);
 void strreverse(char s[]);
 void itoa(int n, char s[]);
+char *strcpy(char *dst, const char *src);
+char *strncpy(char *destination, const char *source, size_t num);
+int strcmp(const char *s1, const char *s2);
+char *strtok(char *s, const char *delim);
 
 #endif /* __STRING_H */
diff --git a/bootrom/bootrom.c b/bootrom/bootrom.c
index e889bb7da..1a3dafc12 100644
--- a/bootrom/bootrom.c
+++ b/bootrom/bootrom.c
@@ -122,10 +122,11 @@ void UsbPacketReceived(uint8_t *packet, int len) {
     switch (c->cmd) {
         case CMD_DEVICE_INFO: {
             dont_ack = 1;
-            arg0 = DEVICE_INFO_FLAG_BOOTROM_PRESENT | 
+            arg0 = DEVICE_INFO_FLAG_BOOTROM_PRESENT |
                    DEVICE_INFO_FLAG_CURRENT_MODE_BOOTROM |
                    DEVICE_INFO_FLAG_UNDERSTANDS_START_FLASH |
-                   DEVICE_INFO_FLAG_UNDERSTANDS_CHIP_INFO;
+                   DEVICE_INFO_FLAG_UNDERSTANDS_CHIP_INFO |
+                   DEVICE_INFO_FLAG_UNDERSTANDS_VERSION;
             if (common_area.flags.osimage_present)
                 arg0 |= DEVICE_INFO_FLAG_OSIMAGE_PRESENT;
 
@@ -140,6 +141,13 @@ void UsbPacketReceived(uint8_t *packet, int len) {
         }
         break;
 
+        case CMD_BL_VERSION: {
+            dont_ack = 1;
+            arg0 = BL_VERSION_1_0_0;
+            reply_old(CMD_BL_VERSION, arg0, 0, 0, 0, 0);
+        }
+        break;
+
         case CMD_SETUP_WRITE: {
             /* The temporary write buffer of the embedded flash controller is mapped to the
             * whole memory region, only the last 8 bits are decoded.
@@ -153,11 +161,19 @@ void UsbPacketReceived(uint8_t *packet, int len) {
         case CMD_FINISH_WRITE: {
             uint32_t *flash_mem = (uint32_t *)(&_flash_start);
             for (int j = 0; j < 2; j++) {
-                for (i = 0 + (64 * j); i < 64 + (64 * j); i++) {
-                    flash_mem[i] = c->d.asDwords[i];
-                }
-
                 uint32_t flash_address = arg0 + (0x100 * j);
+                AT91PS_EFC efc_bank = AT91C_BASE_EFC0;
+                int offset = 0;
+                uint32_t page_n = (flash_address - ((uint32_t)flash_mem)) / AT91C_IFLASH_PAGE_SIZE;
+                if (page_n >= AT91C_IFLASH_NB_OF_PAGES / 2) {
+                    page_n -= AT91C_IFLASH_NB_OF_PAGES / 2;
+                    efc_bank = AT91C_BASE_EFC1;
+                    // We need to offset the writes or it will not fill the correct bank write buffer.
+                    offset = (AT91C_IFLASH_NB_OF_PAGES / 2) * AT91C_IFLASH_PAGE_SIZE / sizeof(uint32_t);
+                }
+                for (i = 0 + (64 * j); i < 64 + (64 * j); i++) {
+                    flash_mem[offset + i] = c->d.asDwords[i];
+                }
 
                 /* Check that the address that we are supposed to write to is within our allowed region */
                 if (((flash_address + AT91C_IFLASH_PAGE_SIZE - 1) >= end_addr) || (flash_address < start_addr)) {
@@ -165,16 +181,15 @@ void UsbPacketReceived(uint8_t *packet, int len) {
                     dont_ack = 1;
                     reply_old(CMD_NACK, 0, 0, 0, 0, 0);
                 } else {
-                    uint32_t page_n = (flash_address - ((uint32_t)flash_mem)) / AT91C_IFLASH_PAGE_SIZE;
-                    /* Translate address to flash page and do flash, update here for the 512k part */
-                    AT91C_BASE_EFC0->EFC_FCR = MC_FLASH_COMMAND_KEY |
-                                               MC_FLASH_COMMAND_PAGEN(page_n) |
-                                               AT91C_MC_FCMD_START_PROG;
+
+                    efc_bank->EFC_FCR = MC_FLASH_COMMAND_KEY |
+                                        MC_FLASH_COMMAND_PAGEN(page_n) |
+                                        AT91C_MC_FCMD_START_PROG;
                 }
 
                 // Wait until flashing of page finishes
                 uint32_t sr;
-                while (!((sr = AT91C_BASE_EFC0->EFC_FSR) & AT91C_MC_FRDY));
+                while (!((sr = efc_bank->EFC_FSR) & AT91C_MC_FRDY));
                 if (sr & (AT91C_MC_LOCKE | AT91C_MC_PROGE)) {
                     dont_ack = 1;
                     reply_old(CMD_NACK, sr, 0, 0, 0, 0);
diff --git a/client/Makefile b/client/Makefile
index a1109a497..440f2b888 100644
--- a/client/Makefile
+++ b/client/Makefile
@@ -227,6 +227,7 @@ CMDSRCS =    crapto1/crapto1.c \
             cmdlfvisa2000.c \
             cmdtrace.c \
             cmdflashmem.c \
+            cmdflashmemspiffs.c \
             cmdsmartcard.c \
             cmdusart.c \
             cmdparser.c \
diff --git a/client/cmdanalyse.c b/client/cmdanalyse.c
index 1eee47c78..0821cc4d1 100644
--- a/client/cmdanalyse.c
+++ b/client/cmdanalyse.c
@@ -623,8 +623,7 @@ static int CmdAnalyseA(const char *Cmd) {
         for (uint8_t k=0; k<4; k = (k+1) % 4 ) {
             PrintAndLogEx(NORMAL, "\e[s%c\e[u", star[k]);
             fflush(stdout);
-            if (ukbhit()) {
-                int gc = getchar(); (void)gc;
+            if (kbd_enter_pressed()) {
                 break;
             }
         }
diff --git a/client/cmdcrc.c b/client/cmdcrc.c
index 58cab45e5..af4baae45 100644
--- a/client/cmdcrc.c
+++ b/client/cmdcrc.c
@@ -248,7 +248,7 @@ int RunModel(char *inModel, char *inHexStr, bool reverse, char endian, char *res
     //set model
     c = mbynam(&model, inModel);
     if (!c) {
-        PrintAndLogEx(WARNING, "error: preset model '%s' not found.  Use reveng -D to list presets. [%d]", inModel, c);
+        PrintAndLogEx(ERR, "error: preset model '%s' not found.  Use reveng -D to list presets. [%d]", inModel, c);
         return 0;
     }
     if (c < 0) {
diff --git a/client/cmddata.c b/client/cmddata.c
index d6a4e8e73..7ade24586 100644
--- a/client/cmddata.c
+++ b/client/cmddata.c
@@ -602,7 +602,7 @@ static int Cmdmandecoderaw(const char *Cmd) {
     }
 
     if (high > 7 || low < 0) {
-        PrintAndLogEx(WARNING, "Error: please raw demod the wave first then manchester raw decode");
+        PrintAndLogEx(ERR, "Error: please raw demod the wave first then manchester raw decode");
         return PM3_ESOFT;
     }
 
@@ -611,7 +611,7 @@ static int Cmdmandecoderaw(const char *Cmd) {
     uint8_t alignPos = 0;
     errCnt = manrawdecode(bits, &size, invert, &alignPos);
     if (errCnt >= maxErr) {
-        PrintAndLogEx(WARNING, "Too many errors: %u", errCnt);
+        PrintAndLogEx(ERR, "Too many errors: %u", errCnt);
         return PM3_ESOFT;
     }
 
@@ -658,11 +658,11 @@ static int CmdBiphaseDecodeRaw(const char *Cmd) {
 
     errCnt = BiphaseRawDecode(bits, &size, &offset, invert);
     if (errCnt < 0) {
-        PrintAndLogEx(WARNING, "Error during decode:%d", errCnt);
+        PrintAndLogEx(ERR, "Error during decode:%d", errCnt);
         return PM3_ESOFT;
     }
     if (errCnt > maxErr) {
-        PrintAndLogEx(WARNING, "Too many errors attempting to decode: %d", errCnt);
+        PrintAndLogEx(ERR, "Too many errors attempting to decode: %d", errCnt);
         return PM3_ESOFT;
     }
 
@@ -865,7 +865,7 @@ static int CmdBitsamples(const char *Cmd) {
     int cnt = 0;
     uint8_t got[12288];
 
-    if (!GetFromDevice(BIG_BUF, got, sizeof(got), 0, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF, got, sizeof(got), 0, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "command execution time out");
         return PM3_ETIMEOUT;
     }
@@ -1290,7 +1290,7 @@ int CmdPSK1rawDemod(const char *Cmd) {
     int ans = PSKDemod(Cmd, true);
     //output
     if (ans != PM3_SUCCESS) {
-        if (g_debugMode) PrintAndLogEx(WARNING, "Error demoding: %d", ans);
+        if (g_debugMode) PrintAndLogEx(ERR, "Error demoding: %d", ans);
         return PM3_ESOFT;
     }
     PrintAndLogEx(NORMAL, "PSK1 demoded bitstream:");
@@ -1307,7 +1307,7 @@ static int CmdPSK2rawDemod(const char *Cmd) {
 
     int ans = PSKDemod(Cmd, true);
     if (ans != PM3_SUCCESS) {
-        if (g_debugMode) PrintAndLogEx(WARNING, "Error demoding: %d", ans);
+        if (g_debugMode) PrintAndLogEx(ERR, "Error demoding: %d", ans);
         return PM3_ESOFT;
     }
     psk1TOpsk2(DemodBuffer, DemodBufferLen);
@@ -1397,7 +1397,7 @@ static int CmdHexsamples(const char *Cmd) {
         return PM3_EINVARG;
     }
 
-    if (!GetFromDevice(BIG_BUF, got, requested, offset, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF, got, requested, offset, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "command execution time out");
         return PM3_ESOFT;
     }
@@ -1470,7 +1470,7 @@ int getSamples(uint32_t n, bool silent) {
     if (!silent) PrintAndLogEx(NORMAL, "Reading %d bytes from device memory\n", n);
 
     PacketResponseNG response;
-    if (!GetFromDevice(BIG_BUF, got, n, 0, &response, 10000, true)) {
+    if (!GetFromDevice(BIG_BUF, got, n, 0, NULL, 0, &response, 10000, true)) {
         PrintAndLogEx(WARNING, "timeout while waiting for reply.");
         return PM3_ETIMEOUT;
     }
diff --git a/client/cmdflashmem.c b/client/cmdflashmem.c
index dafda3ae6..cfe122c50 100644
--- a/client/cmdflashmem.c
+++ b/client/cmdflashmem.c
@@ -37,17 +37,6 @@ static int usage_flashmem_spibaud(void) {
     return PM3_SUCCESS;
 }
 
-static int usage_flashmem_read(void) {
-    PrintAndLogEx(NORMAL, "Read flash memory on device");
-    PrintAndLogEx(NORMAL, "Usage:  mem read o <offset> l <len>");
-    PrintAndLogEx(NORMAL, "  o <offset>    :      offset in memory");
-    PrintAndLogEx(NORMAL, "  l <len>       :      length");
-    PrintAndLogEx(NORMAL, "");
-    PrintAndLogEx(NORMAL, "Examples:");
-    PrintAndLogEx(NORMAL, "        mem read o 0 l 32");    // read 32 bytes starting at offset 0
-    PrintAndLogEx(NORMAL, "        mem read o 1024 l 10"); // read 10 bytes starting at offset 1024
-    return PM3_SUCCESS;
-}
 static int usage_flashmem_load(void) {
     PrintAndLogEx(NORMAL, "Loads binary file into flash memory on device");
     PrintAndLogEx(NORMAL, "Usage:  mem load [o <offset>] f <file name> [m|t|i]");
@@ -67,17 +56,19 @@ static int usage_flashmem_load(void) {
     PrintAndLogEx(NORMAL, "        mem load f default_iclass_keys i");
     return PM3_SUCCESS;
 }
-static int usage_flashmem_save(void) {
-    PrintAndLogEx(NORMAL, "Saves flash memory on device into the file");
-    PrintAndLogEx(NORMAL, " Usage:  mem save [o <offset>] [l <length>] f <file name>");
+static int usage_flashmem_dump(void) {
+    PrintAndLogEx(NORMAL, "Dumps flash memory on device into a file or in console");
+    PrintAndLogEx(NORMAL, " Usage:  mem dump [o <offset>] [l <length>] [f <file name>] [p]");
     PrintAndLogEx(NORMAL, "  o <offset>    :      offset in memory");
     PrintAndLogEx(NORMAL, "  l <length>    :      length");
     PrintAndLogEx(NORMAL, "  f <filename>  :      file name");
+    PrintAndLogEx(NORMAL, "  p             :      print dump in console");
+    PrintAndLogEx(NORMAL, " You must specify at lease option f or option p, both if you wish");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
-    PrintAndLogEx(NORMAL, "        mem save f myfile");                 // download whole flashmem to file myfile
-    PrintAndLogEx(NORMAL, "        mem save f myfile l 4096");          // download 4096 bytes from default offset 0 to file myfile
-    PrintAndLogEx(NORMAL, "        mem save f myfile o 1024 l 4096");   // downlowd 4096 bytes from offset 1024 to file myfile
+    PrintAndLogEx(NORMAL, "        mem dump f myfile");                 // download whole flashmem to file myfile
+    PrintAndLogEx(NORMAL, "        mem dump p o 262015 l 128");         // display 128 bytes from offset 262015 (RSA sig)
+    PrintAndLogEx(NORMAL, "        mem dump p f myfile o 241664 l 58"); // download and display 58 bytes from offset 241664 to file myfile
     return PM3_SUCCESS;
 }
 static int usage_flashmem_wipe(void) {
@@ -105,46 +96,6 @@ static int usage_flashmem_info(void) {
     return PM3_SUCCESS;
 }
 
-static int CmdFlashMemRead(const char *Cmd) {
-
-    uint8_t cmdp = 0;
-    bool errors = false;
-    uint32_t start_index = 0, len  = 0;
-
-    while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
-        switch (tolower(param_getchar(Cmd, cmdp))) {
-            case 'o':
-                start_index = param_get32ex(Cmd, cmdp + 1, 0, 10);
-                cmdp += 2;
-                break;
-            case 'l':
-                len = param_get32ex(Cmd, cmdp + 1, 0, 10);
-                cmdp += 2;
-                break;
-            case 'h':
-                return usage_flashmem_read();
-            default:
-                PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
-                errors = true;
-                break;
-        }
-    }
-
-    //Validations
-    if (errors || cmdp == 0) {
-        usage_flashmem_read();
-        return PM3_EINVARG;
-    }
-    if (start_index + len > FLASH_MEM_MAX_SIZE) {
-        PrintAndLogDevice(WARNING, "error, start_index + length is larger than available memory");
-        return PM3_EOVFLOW;
-    }
-
-    clearCommandBuffer();
-    SendCommandMIX(CMD_FLASHMEM_READ, start_index, len, 0, NULL, 0);
-    return PM3_SUCCESS;
-}
-
 static int CmdFlashmemSpiBaudrate(const char *Cmd) {
 
     char ctmp = tolower(param_getchar(Cmd, 0));
@@ -258,7 +209,7 @@ static int CmdFlashMemLoad(const char *Cmd) {
             }
 
             if (datalen > FLASH_MEM_MAX_SIZE) {
-                PrintAndLogDevice(WARNING, "error, filesize is larger than available memory");
+                PrintAndLogDevice(ERR, "error, filesize is larger than available memory");
                 free(data);
                 return PM3_EOVFLOW;
             }
@@ -311,17 +262,18 @@ static int CmdFlashMemLoad(const char *Cmd) {
     PrintAndLogEx(SUCCESS, "Wrote "_GREEN_("%u")"bytes to offset "_GREEN_("%u"), datalen, start_index);
     return PM3_SUCCESS;
 }
-static int CmdFlashMemSave(const char *Cmd) {
+static int CmdFlashMemDump(const char *Cmd) {
 
     char filename[FILE_PATH_SIZE] = {0};
     uint8_t cmdp = 0;
     bool errors = false;
+    bool print = false;
     uint32_t start_index = 0, len = FLASH_MEM_MAX_SIZE;
 
     while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
         switch (tolower(param_getchar(Cmd, cmdp))) {
             case 'h':
-                return usage_flashmem_save();
+                return usage_flashmem_dump();
             case 'l':
                 len = param_get32ex(Cmd, cmdp + 1, FLASH_MEM_MAX_SIZE, 10);
                 cmdp += 2;
@@ -330,6 +282,10 @@ static int CmdFlashMemSave(const char *Cmd) {
                 start_index = param_get32ex(Cmd, cmdp + 1, 0, 10);
                 cmdp += 2;
                 break;
+            case 'p':
+                print = true;
+                cmdp += 1;
+                break;
             case 'f':
                 //File handling
                 if (param_getstr(Cmd, cmdp + 1, filename, FILE_PATH_SIZE) >= FILE_PATH_SIZE) {
@@ -348,25 +304,32 @@ static int CmdFlashMemSave(const char *Cmd) {
 
     //Validations
     if (errors || cmdp == 0) {
-        usage_flashmem_save();
+        usage_flashmem_dump();
         return PM3_EINVARG;
     }
 
     uint8_t *dump = calloc(len, sizeof(uint8_t));
     if (!dump) {
-        PrintAndLogDevice(WARNING, "error, cannot allocate memory ");
+        PrintAndLogDevice(ERR, "error, cannot allocate memory ");
         return PM3_EMALLOC;
     }
 
     PrintAndLogEx(INFO, "downloading "_YELLOW_("%u")"bytes from flashmem", len);
-    if (!GetFromDevice(FLASH_MEM, dump, len, start_index, NULL, -1, true)) {
+    if (!GetFromDevice(FLASH_MEM, dump, len, start_index, NULL, 0, NULL, -1, true)) {
         PrintAndLogEx(FAILED, "ERROR; downloading from flashmemory");
         free(dump);
         return PM3_EFLASH;
     }
 
-    saveFile(filename, ".bin", dump, len);
-    saveFileEML(filename, dump, len, 16);
+    if (print) {
+        print_hex_break(dump, len, 32);
+    }
+
+    if (filename[0] != '\0') {
+        saveFile(filename, ".bin", dump, len);
+        saveFileEML(filename, dump, len, 16);
+    }
+
     free(dump);
     return PM3_SUCCESS;
 }
@@ -611,11 +574,11 @@ static int CmdFlashMemInfo(const char *Cmd) {
 
 static command_t CommandTable[] = {
     {"help",    CmdHelp,            AlwaysAvailable, "This help"},
+    {"spiffs",  CmdFlashMemSpiFFS,  IfPm3Flash,      "High level SPI FileSystem Flash manipulation [rdv40]"},
     {"spibaud", CmdFlashmemSpiBaudrate, IfPm3Flash,  "Set Flash memory Spi baudrate [rdv40]"},
-    {"read",    CmdFlashMemRead,    IfPm3Flash,      "Read Flash memory [rdv40]"},
     {"info",    CmdFlashMemInfo,    IfPm3Flash,      "Flash memory information [rdv40]"},
     {"load",    CmdFlashMemLoad,    IfPm3Flash,      "Load data into flash memory [rdv40]"},
-    {"save",    CmdFlashMemSave,    IfPm3Flash,      "Save data from flash memory [rdv40]"},
+    {"dump",    CmdFlashMemDump,    IfPm3Flash,      "Dump data from flash memory [rdv40]"},
     {"wipe",    CmdFlashMemWipe,    IfPm3Flash,      "Wipe data from flash memory [rdv40]"},
     {NULL, NULL, NULL, NULL}
 };
diff --git a/client/cmdflashmem.h b/client/cmdflashmem.h
index 3c37239b7..b08164837 100644
--- a/client/cmdflashmem.h
+++ b/client/cmdflashmem.h
@@ -15,14 +15,16 @@
 #include <stdlib.h>
 #include <string.h>
 #include <ctype.h>
+#include "pmflash.h"
+#include "common.h"
 #include "proxmark3.h"
 #include "ui.h"
 #include "cmdparser.h"
-#include "common.h"
 #include "util.h"
 #include "util_posix.h"         // msclock
 #include "loclass/fileutils.h"  //saveFile
 #include "comms.h"              //getfromdevice
+#include "cmdflashmemspiffs.h" // spiffs commands
 
 typedef enum {
     DICTIONARY_NONE = 0,
diff --git a/client/cmdflashmemspiffs.c b/client/cmdflashmemspiffs.c
new file mode 100644
index 000000000..bc7a408bd
--- /dev/null
+++ b/client/cmdflashmemspiffs.c
@@ -0,0 +1,460 @@
+//-----------------------------------------------------------------------------
+// Copyright (C) 2018 iceman
+//
+// This code is licensed to you under the terms of the GNU GPL, version 2 or,
+// at your option, any later version. See the LICENSE.txt file for the text of
+// the license.
+//-----------------------------------------------------------------------------
+// Proxmark3 RDV40 Flash memory commands
+//-----------------------------------------------------------------------------
+#include "cmdflashmemspiffs.h"
+
+#include "mbedtls/base64.h"
+#include "mbedtls/rsa.h"
+#include "mbedtls/sha1.h"
+
+static int CmdHelp(const char *Cmd);
+
+static int CmdFlashMemSpiFFSMount(const char *Cmd) {
+    (void)Cmd; // Cmd is not used so far
+    clearCommandBuffer();
+    SendCommandNG(CMD_SPIFFS_MOUNT, NULL, 0);
+    return PM3_SUCCESS;
+}
+
+static int CmdFlashMemSpiFFSUnmount(const char *Cmd) {
+    (void)Cmd; // Cmd is not used so far
+    clearCommandBuffer();
+    SendCommandNG(CMD_SPIFFS_UNMOUNT, NULL, 0);
+    return PM3_SUCCESS;
+}
+
+static int CmdFlashMemSpiFFSTest(const char *Cmd) {
+    (void)Cmd; // Cmd is not used so far
+    clearCommandBuffer();
+    SendCommandNG(CMD_SPIFFS_TEST, NULL, 0);
+    return PM3_SUCCESS;
+}
+
+static int CmdFlashMemSpiFFSTree(const char *Cmd) {
+    (void)Cmd; // Cmd is not used so far
+    clearCommandBuffer();
+    SendCommandNG(CMD_SPIFFS_PRINT_TREE, NULL, 0);
+    return PM3_SUCCESS;
+}
+
+static int CmdFlashMemSpiFFSInfo(const char *Cmd) {
+    (void)Cmd; // Cmd is not used so far
+    clearCommandBuffer();
+    SendCommandNG(CMD_SPIFFS_PRINT_FSINFO, NULL, 0);
+    return PM3_SUCCESS;
+}
+
+static int usage_flashmemspiffs_remove(void) {
+    PrintAndLogEx(NORMAL, "Remove a file from spiffs filesystem");
+    PrintAndLogEx(NORMAL, " Usage:  mem spiffs remove <filename>");
+    return PM3_SUCCESS;
+}
+
+static int usage_flashmemspiffs_rename(void) {
+    PrintAndLogEx(NORMAL, "Rename/move a file in spiffs filesystem");
+    PrintAndLogEx(NORMAL, " Usage:  mem spiffs rename <source> <destination>");
+    return PM3_SUCCESS;
+}
+
+static int usage_flashmemspiffs_copy(void) {
+    PrintAndLogEx(NORMAL, "Copy a file to another (destructively) in spiffs filesystem");
+    PrintAndLogEx(NORMAL, " Usage:  mem spiffs copy <source> <destination>");
+    return PM3_SUCCESS;
+}
+
+static int usage_flashmemspiffs_dump(void) {
+    PrintAndLogEx(NORMAL, "Dumps flash memory on device into a file or in console");
+    PrintAndLogEx(NORMAL, "Size is handled by first sending a STAT command against file existence");
+    PrintAndLogEx(NORMAL, " Usage:  mem spiffs dump o <filename> [f <file name> [e]]  [p]");
+    PrintAndLogEx(NORMAL, "  o <filename>    :      filename in SPIFFS");
+    PrintAndLogEx(NORMAL, "  f <filename>    :      file name to save to");
+    PrintAndLogEx(NORMAL, "  p               :      print dump in console");
+    PrintAndLogEx(NORMAL, "  e               :      also save in EML format (good for tags save and dictonnary files)");
+    PrintAndLogEx(NORMAL, " You must specify at lease option f or option p, both if you wish");
+    PrintAndLogEx(NORMAL, "");
+    PrintAndLogEx(NORMAL, "Examples:");
+    PrintAndLogEx(NORMAL, "        mem spiffs dump o hf_colin/lasttag f lasttag e");
+    PrintAndLogEx(NORMAL, "        mem spiffs dump o hf_colin/lasttag p");
+    return PM3_SUCCESS;
+}
+
+static int usage_flashmemspiffs_load(void) {
+    PrintAndLogEx(NORMAL, "Uploads binary-wise file into device filesystem");
+    PrintAndLogEx(NORMAL, "Usage:  mem spiffs load o <filename> f <filename>");
+    PrintAndLogEx(NORMAL, "Warning: mem area to be written must have been wiped first");
+    PrintAndLogEx(NORMAL, "(this is already taken care when loading dictionaries)");
+    PrintAndLogEx(NORMAL, "  o <filename>  :  destination filename");
+    PrintAndLogEx(NORMAL, "  f <filename>  :  local filename");
+    PrintAndLogEx(NORMAL, "");
+    PrintAndLogEx(NORMAL, "Examples:");
+    PrintAndLogEx(NORMAL, "        mem spiffs load f myfile o myapp.conf");
+    return PM3_SUCCESS;
+}
+
+static int CmdFlashMemSpiFFSRemove(const char *Cmd) {
+
+    char filename[32] = {0};
+    bool errors = false;
+
+    /*char ctmp = tolower(param_getchar(Cmd, 0));
+    if (strlen(Cmd) < 1 || ctmp == 'h') {
+        return usage_flashmemspiffs_remove();
+    }*/
+
+    if (param_getstr(Cmd, 0, filename, 32) >= 32) {
+        PrintAndLogEx(FAILED, "Filename too long");
+        errors = true;
+    }
+
+    // check null filename ?
+    if (errors) {
+        usage_flashmemspiffs_remove();
+        return PM3_EINVARG;
+    }
+
+    SendCommandMIX(CMD_SPIFFS_REMOVE, 0, 0, 0, (uint8_t *)filename, 32);
+    return PM3_SUCCESS;
+}
+
+static int CmdFlashMemSpiFFSRename(const char *Cmd) {
+
+    char srcfilename[32] = {0};
+    char destfilename[32] = {0};
+    bool errors = false;
+
+    /*char ctmp = tolower(param_getchar(Cmd, 0));
+    if (strlen(Cmd) < 1 || ctmp == 'h') {
+        return usage_flashmemspiffs_rename();
+    }*/
+
+    if (param_getstr(Cmd, 0, srcfilename, 32) >= 32) {
+        PrintAndLogEx(FAILED, "Source Filename too long");
+        errors = true;
+    }
+    if (srcfilename[0] == '\0') {
+        PrintAndLogEx(FAILED, "Source Filename missing or invalid");
+        errors = true;
+    }
+
+    if (param_getstr(Cmd, 1, destfilename, 32) >= 32) {
+        PrintAndLogEx(FAILED, "Source Filename too long");
+        errors = true;
+    }
+    if (destfilename[0] == '\0') {
+        PrintAndLogEx(FAILED, "Source Filename missing or invalid");
+        errors = true;
+    }
+
+    // check null filename ?
+    if (errors) {
+        usage_flashmemspiffs_rename();
+        return PM3_EINVARG;
+    }
+
+    char data[65];
+    sprintf(data, "%s,%s", srcfilename, destfilename);
+    SendCommandMIX(CMD_SPIFFS_RENAME, 0, 0, 0, (uint8_t *)data, 65);
+    return PM3_SUCCESS;
+}
+
+static int CmdFlashMemSpiFFSCopy(const char *Cmd) {
+
+    char srcfilename[32] = {0};
+    char destfilename[32] = {0};
+    bool errors = false;
+
+    /*char ctmp = tolower(param_getchar(Cmd, 0));
+    if (strlen(Cmd) < 1 || ctmp == 'h') {
+        return usage_flashmemspiffs_copy();
+    }*/
+
+    if (param_getstr(Cmd, 0, srcfilename, 32) >= 32) {
+        PrintAndLogEx(FAILED, "Source Filename too long");
+        errors = true;
+    }
+    if (srcfilename[0] == '\0') {
+        PrintAndLogEx(FAILED, "Source Filename missing or invalid");
+        errors = true;
+    }
+
+    if (param_getstr(Cmd, 1, destfilename, 32) >= 32) {
+        PrintAndLogEx(FAILED, "Source Filename too long");
+        errors = true;
+    }
+    if (destfilename[0] == '\0') {
+        PrintAndLogEx(FAILED, "Source Filename missing or invalid");
+        errors = true;
+    }
+
+    // check null filename ?
+    if (errors) {
+        usage_flashmemspiffs_copy();
+        return PM3_EINVARG;
+    }
+
+    char data[65];
+    sprintf(data, "%s,%s", srcfilename, destfilename);
+    SendCommandMIX(CMD_SPIFFS_COPY, 0, 0, 0, (uint8_t *)data, 65);
+    return PM3_SUCCESS;
+}
+
+static int CmdFlashMemSpiFFSDump(const char *Cmd) {
+
+    char filename[FILE_PATH_SIZE] = {0};
+    uint8_t cmdp = 0;
+    bool errors = false;
+    bool print = false;
+    uint32_t start_index = 0, len = FLASH_MEM_MAX_SIZE;
+    char destfilename[32] = {0};
+    bool eml = false;
+
+    while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
+        switch (tolower(param_getchar(Cmd, cmdp))) {
+            case 'h':
+                return usage_flashmemspiffs_dump();
+            /*case 'l':
+                len = param_get32ex(Cmd, cmdp + 1, FLASH_MEM_MAX_SIZE, 10);
+                cmdp += 2;
+                break;*/
+            case 'o':
+                param_getstr(Cmd, cmdp + 1, destfilename, 32);
+                cmdp += 2;
+                break;
+            case 'p':
+                print = true;
+                cmdp += 1;
+                break;
+            case 'e':
+                eml = true;
+                cmdp += 1;
+                break;
+            case 'f':
+                // File handling
+                if (param_getstr(Cmd, cmdp + 1, filename, FILE_PATH_SIZE) >= FILE_PATH_SIZE) {
+                    PrintAndLogEx(FAILED, "Filename too long");
+                    errors = true;
+                    break;
+                }
+                cmdp += 2;
+                break;
+            default:
+                PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
+                errors = true;
+                break;
+        }
+    }
+
+    if ((filename[0] == '\0') && (!print)) {
+        PrintAndLogEx(FAILED, "No print asked and Local dump Filename missing or invalid");
+        errors = true;
+    }
+
+    if (destfilename[0] == '\0') {
+        PrintAndLogEx(FAILED, "SPIFFS Filename missing or invalid");
+        errors = true;
+    }
+
+    // Validations
+    if (errors || cmdp == 0) {
+        usage_flashmemspiffs_dump();
+        return PM3_EINVARG;
+    }
+
+    // get size from spiffs itself !
+    SendCommandMIX(CMD_SPIFFS_STAT, 0, 0, 0, (uint8_t *)destfilename, 32);
+    PacketResponseNG resp;
+    if (!WaitForResponseTimeout(CMD_ACK, &resp, 2000)) {
+        PrintAndLogEx(WARNING, "timeout while waiting for reply.");
+        return PM3_ETIMEOUT;
+    }
+
+    len = resp.oldarg[0];
+
+    uint8_t *dump = calloc(len, sizeof(uint8_t));
+    if (!dump) {
+        PrintAndLogDevice(ERR, "error, cannot allocate memory ");
+        return PM3_EMALLOC;
+    }
+
+    PrintAndLogEx(INFO, "downloading "_YELLOW_("%u") "bytes from spiffs (flashmem)", len);
+    if (!GetFromDevice(SPIFFS, dump, len, start_index, (uint8_t *)destfilename, 32, NULL, -1, true)) {
+        PrintAndLogEx(FAILED, "ERROR; downloading from spiffs(flashmemory)");
+        free(dump);
+        return PM3_EFLASH;
+    }
+
+    if (print) {
+        print_hex_break(dump, len, 32);
+    }
+
+    if (filename[0] != '\0') {
+        saveFile(filename, "", dump, len);
+        if (eml) {
+            saveFileEML(filename, dump, len, 16);
+        }
+    }
+
+    free(dump);
+    return PM3_SUCCESS;
+}
+
+static int CmdFlashMemSpiFFSLoad(const char *Cmd) {
+
+    uint32_t append = 0;
+    char filename[FILE_PATH_SIZE] = {0};
+    char destfilename[32] = {0};
+    bool errors = false;
+    uint8_t cmdp = 0;
+
+    while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
+        switch (tolower(param_getchar(Cmd, cmdp))) {
+            case 'h':
+                return usage_flashmemspiffs_load();
+            case 'f':
+                if (param_getstr(Cmd, cmdp + 1, filename, FILE_PATH_SIZE) >= FILE_PATH_SIZE) {
+                    PrintAndLogEx(FAILED, "Filename too long");
+                    errors = true;
+                    break;
+                }
+                cmdp += 2;
+                break;
+            case 'o':
+                param_getstr(Cmd, cmdp + 1, destfilename, 32);
+                cmdp += 2;
+                break;
+            default:
+                PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
+                errors = true;
+                break;
+        }
+    }
+
+    if (destfilename[0] == '\0') {
+        PrintAndLogEx(FAILED, "Destination Filename missing or invalid");
+        errors = true;
+    }
+
+    // Validations
+    if (errors || cmdp == 0) {
+        usage_flashmemspiffs_load();
+        return PM3_EINVARG;
+    }
+
+    size_t datalen = 0;
+    int res = 0;
+    uint8_t *data = calloc(FLASH_MEM_MAX_SIZE, sizeof(uint8_t));
+
+    res = loadFile(filename, "", data, FLASH_MEM_MAX_SIZE, &datalen);
+    // int res = loadFileEML( filename, data, &datalen);
+    if (res) {
+        free(data);
+        return PM3_EFILE;
+    }
+
+    if (datalen > FLASH_MEM_MAX_SIZE) {
+        PrintAndLogDevice(ERR, "error, filesize is larger than available memory");
+        free(data);
+        return PM3_EOVFLOW;
+    }
+
+    uint8_t *newdata = realloc(data, datalen);
+    if (newdata == NULL) {
+        free(data);
+        return PM3_EMALLOC;
+    } else {
+        data = newdata;
+    }
+
+    // We want to mount before multiple operation so the lazy writes/append will not
+    // trigger a mount + umount each loop iteration (lazy ops device side)
+    SendCommandNG(CMD_SPIFFS_MOUNT, NULL, 0);
+
+    // Send to device
+    uint32_t bytes_sent = 0;
+    uint32_t bytes_remaining = datalen;
+
+    // fast push mode
+    conn.block_after_ACK = true;
+
+    // SendCommandMIX(CMD_SPIFFS_COPY, 0, 0, 0, (uint8_t *)data, 65);
+
+    while (bytes_remaining > 0) {
+        uint32_t bytes_in_packet = MIN(FLASH_MEM_BLOCK_SIZE, bytes_remaining);
+
+        clearCommandBuffer();
+
+        char fdata[32 + bytes_in_packet];
+        memset(fdata, 0, sizeof(fdata));
+        memcpy(fdata, destfilename, 32);
+        memcpy(fdata + 32, data + bytes_sent, bytes_in_packet);
+        // sprintf(fdata, "%s%s", destfilename, data + bytes_sent);
+
+        if (bytes_sent > 0)
+            append = 1;
+
+        SendCommandOLD(CMD_SPIFFS_WRITE, append, bytes_in_packet, 0, fdata, 32 + bytes_in_packet);
+
+        bytes_remaining -= bytes_in_packet;
+        bytes_sent += bytes_in_packet;
+
+        PacketResponseNG resp;
+        if (!WaitForResponseTimeout(CMD_ACK, &resp, 2000)) {
+            PrintAndLogEx(WARNING, "timeout while waiting for reply.");
+            conn.block_after_ACK = false;
+            free(data);
+            return PM3_ETIMEOUT;
+        }
+
+        uint8_t isok = resp.oldarg[0] & 0xFF;
+        if (!isok) {
+            conn.block_after_ACK = false;
+            PrintAndLogEx(FAILED, "Flash write fail [offset %u]", bytes_sent);
+            return PM3_EFLASH;
+        }
+    }
+
+    conn.block_after_ACK = false;
+    free(data);
+    PrintAndLogEx(SUCCESS, "Wrote "_GREEN_("%u") "bytes to file "_GREEN_("%s"), datalen, destfilename);
+
+    // We want to unmount after these to set things back to normal but more than this
+    // unmouting ensure that SPIFFS CACHES are all flushed so our file is actually written on memory
+    SendCommandNG(CMD_SPIFFS_UNMOUNT, NULL, 0);
+    return PM3_SUCCESS;
+}
+
+static command_t CommandTable[] = {
+
+    {"help", CmdHelp, AlwaysAvailable, "This help"},
+    {
+        "copy", CmdFlashMemSpiFFSCopy, IfPm3Flash,
+        "Copy a file to another (destructively) in SPIFFS FileSystem in FlashMEM (spiffs)"
+    },
+    {"dump", CmdFlashMemSpiFFSDump, IfPm3Flash, "Dump a file from SPIFFS FileSystem in FlashMEM (spiffs)"},
+    {"info", CmdFlashMemSpiFFSInfo, IfPm3Flash, "Print filesystem info and usage statistics (spiffs)"},
+    {"load", CmdFlashMemSpiFFSLoad, IfPm3Flash, "Upload file into SPIFFS Filesystem (spiffs)"},
+    {"mount", CmdFlashMemSpiFFSMount, IfPm3Flash, "Mount the SPIFFS Filesystem if not already mounted (spiffs)"},
+    {"remove", CmdFlashMemSpiFFSRemove, IfPm3Flash, "Remove a file from SPIFFS FileSystem in FlashMEM (spiffs)"},
+    {"rename", CmdFlashMemSpiFFSRename, IfPm3Flash, "Rename/move a file in SPIFFS FileSystem in FlashMEM (spiffs)"},
+    {"test", CmdFlashMemSpiFFSTest, IfPm3Flash, "Test SPIFFS Operations (require wiping pages 0 and 1)"},
+    {"tree", CmdFlashMemSpiFFSTree, IfPm3Flash, "Print the Flash Memory FileSystem Tree (spiffs)"},
+    {"unmount", CmdFlashMemSpiFFSUnmount, IfPm3Flash, "Un-mount the SPIFFS Filesystem if not already mounted (spiffs)"},
+    {NULL, NULL, NULL, NULL}
+};
+
+static int CmdHelp(const char *Cmd) {
+    (void)Cmd; // Cmd is not used so far
+    CmdsHelp(CommandTable);
+    return PM3_SUCCESS;
+}
+
+int CmdFlashMemSpiFFS(const char *Cmd) {
+    clearCommandBuffer();
+    return CmdsParse(CommandTable, Cmd);
+}
diff --git a/client/cmdflashmemspiffs.h b/client/cmdflashmemspiffs.h
new file mode 100644
index 000000000..b3589ce86
--- /dev/null
+++ b/client/cmdflashmemspiffs.h
@@ -0,0 +1,30 @@
+//-----------------------------------------------------------------------------
+// Copyright (C) 2018 iceman
+//
+// This code is licensed to you under the terms of the GNU GPL, version 2 or,
+// at your option, any later version. See the LICENSE.txt file for the text of
+// the license.
+//-----------------------------------------------------------------------------
+// Proxmark3 RDV40 Flash memory commands
+//-----------------------------------------------------------------------------
+
+#ifndef CMDFLASHMEMSPIFFS_H__
+#define CMDFLASHMEMSPIFFS_H__
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <ctype.h>
+#include "pmflash.h"
+#include "common.h"
+#include "proxmark3.h"
+#include "ui.h"
+#include "cmdparser.h"
+#include "util.h"
+#include "util_posix.h"         // msclock
+#include "loclass/fileutils.h"  //saveFile
+#include "comms.h"              //getfromdevice
+
+int CmdFlashMemSpiFFS(const char *Cmd);
+
+#endif
diff --git a/client/cmdhf.c b/client/cmdhf.c
index 7e42202a9..03a5c1681 100644
--- a/client/cmdhf.c
+++ b/client/cmdhf.c
@@ -40,7 +40,7 @@ static int usage_hf_sniff() {
 static int usage_hf_tune() {
     PrintAndLogEx(NORMAL, "Usage: hf tune [<iter>]");
     PrintAndLogEx(NORMAL, "Continuously measure HF antenna tuning.");
-    PrintAndLogEx(NORMAL, "Press button or keyboard to interrupt.");
+    PrintAndLogEx(NORMAL, "Press button or Enter to interrupt.");
     PrintAndLogEx(NORMAL, "Options:");
     PrintAndLogEx(NORMAL, "       <iter>               - number of iterations (default: infinite)");
     PrintAndLogEx(NORMAL, "");
@@ -98,7 +98,7 @@ int CmdHFTune(const char *Cmd) {
     int iter =  param_get32ex(Cmd, 0, 0, 10);
 
     PacketResponseNG resp;
-    PrintAndLogEx(SUCCESS, "Measuring HF antenna, click button or press a key to exit");
+    PrintAndLogEx(SUCCESS, "Measuring HF antenna, click button or press Enter to exit");
     clearCommandBuffer();
     uint8_t mode[] = {1};
     SendCommandNG(CMD_MEASURE_ANTENNA_TUNING_HF, mode, sizeof(mode));
@@ -109,9 +109,7 @@ int CmdHFTune(const char *Cmd) {
     mode[0] = 2;
     // loop forever (till button pressed) if iter = 0 (default)
     for (uint8_t i = 0; iter == 0 || i < iter; i++) {
-        if (ukbhit()) { // abort by keyboard press
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) { // abort by keyboard press
             break;
         }
         SendCommandNG(CMD_MEASURE_ANTENNA_TUNING_HF, mode, sizeof(mode));
diff --git a/client/cmdhf14a.c b/client/cmdhf14a.c
index 018431a28..f92a35370 100644
--- a/client/cmdhf14a.c
+++ b/client/cmdhf14a.c
@@ -375,9 +375,7 @@ static int CmdHF14ACUIDs(const char *Cmd) {
     // repeat n times
     for (int i = 0; i < n; i++) {
 
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(WARNING, "\n[!] aborted via keyboard!\n");
             break;
         }
@@ -489,7 +487,7 @@ int CmdHF14ASim(const char *Cmd) {
 
     PrintAndLogEx(SUCCESS, "press pm3-button to abort simulation");
 
-    while (!ukbhit()) {
+    while (!kbd_enter_pressed()) {
         if (WaitForResponseTimeout(CMD_SIMULATE_MIFARE_CARD, &resp, 1500) == 0) continue;
         if (resp.status != PM3_SUCCESS) break;
 
@@ -800,7 +798,7 @@ int ExchangeAPDU14a(uint8_t *datain, int datainlen, bool activateField, bool lea
             vActivateField = false;
             if (*dataoutlen) {
                 if (clen != datainlen)
-                    PrintAndLogEx(WARNING, "APDU: I-block/R-block sequence error. Data len=%d, Sent=%d, Last packet len=%d", datainlen, clen, *dataoutlen);
+                    PrintAndLogEx(ERR, "APDU: I-block/R-block sequence error. Data len=%d, Sent=%d, Last packet len=%d", datainlen, clen, *dataoutlen);
                 break;
             }
         } while (clen < datainlen);
@@ -836,20 +834,33 @@ int ExchangeAPDU14a(uint8_t *datain, int datainlen, bool activateField, bool lea
 static int CmdHF14AAPDU(const char *Cmd) {
     uint8_t data[PM3_CMD_DATA_SIZE];
     int datalen = 0;
+    uint8_t header[PM3_CMD_DATA_SIZE];
+    int headerlen = 0;
     bool activateField = false;
     bool leaveSignalON = false;
     bool decodeTLV = false;
+    bool decodeAPDU = false;
+    bool makeAPDU = false;
+    bool extendedAPDU = false;
+    int le = 0;
 
     CLIParserInit("hf 14a apdu",
-                  "Sends an ISO 7816-4 APDU via ISO 14443-4 block transmission protocol (T=CL)",
-                  "Sample:\n\thf 14a apdu -st 00A404000E325041592E5359532E444446303100\n");
+                  "Sends an ISO 7816-4 APDU via ISO 14443-4 block transmission protocol (T=CL). works with all apdu types from ISO 7816-4:2013",
+                  "Sample:\n\thf 14a apdu -st 00A404000E325041592E5359532E444446303100\n"
+                  "\thf 14a apdu -sd 00A404000E325041592E5359532E444446303100 - decode apdu\n"
+                  "\thf 14a apdu -sm 00A40400 325041592E5359532E4444463031 -l 256 - encode standard apdu\n"
+                  "\thf 14a apdu -sm 00A40400 325041592E5359532E4444463031 -el 65536 - encode extended apdu\n");
 
     void *argtable[] = {
         arg_param_begin,
-        arg_lit0("sS",  "select",  "activate field and select card"),
-        arg_lit0("kK",  "keep",    "leave the signal field ON after receive response"),
-        arg_lit0("tT",  "tlv",     "executes TLV decoder if it possible"),
-        arg_strx1(NULL, NULL,      "<APDU (hex)>", NULL),
+        arg_lit0("sS",  "select",   "activate field and select card"),
+        arg_lit0("kK",  "keep",     "leave the signal field ON after receive response"),
+        arg_lit0("tT",  "tlv",      "executes TLV decoder if it possible"),
+        arg_lit0("dD",  "decapdu",  "decode apdu request if it possible"),
+        arg_str0("mM",  "make",     "<head (CLA INS P1 P2) hex>", "make apdu with head from this field and data from data field. Must be 4 bytes length: <CLA INS P1 P2>"),
+        arg_lit0("eE",  "extended", "make extended length apdu if `m` parameter included"),
+        arg_int0("lL",  "le",       "<Le (int)>", "Le apdu parameter if `m` parameter included"),
+        arg_strx1(NULL, NULL,       "<APDU (hex) | data (hex)>", "data if `m` parameter included"),
         arg_param_end
     };
     CLIExecWithReturn(Cmd, argtable, false);
@@ -857,12 +868,66 @@ static int CmdHF14AAPDU(const char *Cmd) {
     activateField = arg_get_lit(1);
     leaveSignalON = arg_get_lit(2);
     decodeTLV = arg_get_lit(3);
-    // len = data + PCB(1b) + CRC(2b)
-    CLIGetHexBLessWithReturn(4, data, &datalen, 1 + 2);
+    decodeAPDU = arg_get_lit(4);
+
+    CLIGetHexWithReturn(5, header, &headerlen);
+    makeAPDU = headerlen > 0;
+    if (makeAPDU && headerlen != 4) {
+        PrintAndLogEx(ERR, "header length must be 4 bytes instead of %d", headerlen);
+        return 1;
+    }
+    extendedAPDU = arg_get_lit(6);
+    le = arg_get_int_def(7, 0);
+
+    if (makeAPDU) {
+        uint8_t apdudata[PM3_CMD_DATA_SIZE] = {0};
+        int apdudatalen = 0;
+
+        CLIGetHexBLessWithReturn(8, apdudata, &apdudatalen, 1 + 2);
+
+        APDUStruct apdu;
+        apdu.cla = header[0];
+        apdu.ins = header[1];
+        apdu.p1 = header[2];
+        apdu.p2 = header[3];
+
+        apdu.lc = apdudatalen;
+        apdu.data = apdudata;
+
+        apdu.extended_apdu = extendedAPDU;
+        apdu.le = le;
+
+        if (APDUEncode(&apdu, data, &datalen)) {
+            PrintAndLogEx(ERR, "can't make apdu with provided parameters.");
+            return 2;
+        }
+
+    } else {
+        if (extendedAPDU) {
+            PrintAndLogEx(ERR, "make mode not set but here `e` option.");
+            return 3;
+        }
+        if (le > 0) {
+            PrintAndLogEx(ERR, "make mode not set but here `l` option.");
+            return 3;
+        }
+
+        // len = data + PCB(1b) + CRC(2b)
+        CLIGetHexBLessWithReturn(8, data, &datalen, 1 + 2);
+    }
 
     CLIParserFree();
     PrintAndLogEx(NORMAL, ">>>>[%s%s%s] %s", activateField ? "sel " : "", leaveSignalON ? "keep " : "", decodeTLV ? "TLV" : "", sprint_hex(data, datalen));
 
+    if (decodeAPDU) {
+        APDUStruct apdu;
+
+        if (APDUDecode(data, datalen, &apdu) == 0)
+            APDUPrint(apdu);
+        else
+            PrintAndLogEx(WARNING, "can't decode APDU.");
+    }
+
     int res = ExchangeAPDU14a(data, datalen, activateField, leaveSignalON, data, PM3_CMD_DATA_SIZE, &datalen);
 
     if (res)
diff --git a/client/cmdhf15.c b/client/cmdhf15.c
index 9a40d03cd..33c3957fe 100644
--- a/client/cmdhf15.c
+++ b/client/cmdhf15.c
@@ -564,7 +564,7 @@ static int CmdHF15Demod(const char *Cmd) {
     }
 
     if (mask != 0x01) {
-        PrintAndLogEx(WARNING, "Error, uneven octet! (discard extra bits!)");
+        PrintAndLogEx(WARNING, "Warning, uneven octet! (discard extra bits!)");
         PrintAndLogEx(NORMAL, "   mask = %02x", mask);
     }
     PrintAndLogEx(NORMAL, "%d octets", k);
@@ -634,7 +634,7 @@ static int CmdHF15Info(const char *Cmd) {
     recv = resp.data.asBytes;
 
     if (recv[0] & ISO15_RES_ERROR) {
-        PrintAndLogEx(WARNING, "iso15693 card returned error %i: %s", recv[0], TagErrorStr(recv[0]));
+        PrintAndLogEx(ERR, "iso15693 card returned error %i: %s", recv[0], TagErrorStr(recv[0]));
         return 3;
     }
 
@@ -1069,7 +1069,7 @@ static int CmdHF15Read(const char *Cmd) {
     }
 
     if (recv[0] & ISO15_RES_ERROR) {
-        PrintAndLogEx(WARNING, "iso15693 card returned error %i: %s", recv[0], TagErrorStr(recv[0]));
+        PrintAndLogEx(ERR, "iso15693 card returned error %i: %s", recv[0], TagErrorStr(recv[0]));
         return 3;
     }
 
@@ -1246,7 +1246,7 @@ static int CmdHF15Restore(const char *Cmd) {
             fclose(f);
             return 0;
         } else if (bytes_read != blocksize) {
-            PrintAndLogEx(WARNING, "File reading error (%s), %u bytes read instead of %u bytes.", filename, bytes_read, blocksize);
+            PrintAndLogEx(ERR, "File reading error (%s), %u bytes read instead of %u bytes.", filename, bytes_read, blocksize);
             fclose(f);
             return 2;
         }
@@ -1348,7 +1348,7 @@ static int CmdHF15CSetUID(const char *Cmd) {
     data[3][5] = uid[1];
     data[3][6] = uid[0];
 
-    for (int i=0; i<4; i++) {
+    for (int i = 0; i < 4; i++) {
         AddCrc15(data[i], 7);
 
         clearCommandBuffer();
diff --git a/client/cmdhffelica.c b/client/cmdhffelica.c
index 6f3e28737..5a6f1d33a 100644
--- a/client/cmdhffelica.c
+++ b/client/cmdhffelica.c
@@ -135,7 +135,7 @@ static int CmdHFFelicaSim(const char *Cmd) {
     if (verbose)
         PrintAndLogEx(NORMAL, "Press pm3-button to abort simulation");
 
-    while (!ukbhit()) {
+    while (!kbd_enter_pressed()) {
         if (!WaitForResponseTimeout(CMD_ACK, &resp, 1500)) continue;
     }
     return 0;
@@ -357,9 +357,7 @@ static int CmdHFFelicaDumpLite(const char *Cmd) {
         timeout++;
         printf(".");
         fflush(stdout);
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(WARNING, "\n[!] aborted via keyboard!\n");
             DropField();
             return 1;
@@ -385,7 +383,7 @@ static int CmdHFFelicaDumpLite(const char *Cmd) {
         return 1;
     }
 
-    if (!GetFromDevice(BIG_BUF, trace, tracelen, 0, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF, trace, tracelen, 0, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "command execution time out");
         free(trace);
         return 0;
diff --git a/client/cmdhficlass.c b/client/cmdhficlass.c
index 2bdb91fa1..91757bbd4 100644
--- a/client/cmdhficlass.c
+++ b/client/cmdhficlass.c
@@ -176,7 +176,7 @@ static int usage_hf_iclass_managekeys(void) {
     return 0;
 }
 static int usage_hf_iclass_reader(void) {
-    PrintAndLogEx(NORMAL, "Act as a Iclass reader.  Look for iClass tags until a key or the pm3 button is pressed\n");
+    PrintAndLogEx(NORMAL, "Act as a Iclass reader.  Look for iClass tags until Enter or the pm3 button is pressed\n");
     PrintAndLogEx(NORMAL, "Usage:  hf iclass reader [h] [1]\n");
     PrintAndLogEx(NORMAL, "Options:");
     PrintAndLogEx(NORMAL, "    h   This help text");
@@ -376,16 +376,14 @@ static int CmdHFiClassSim(const char *Cmd) {
 
         case 2: {
             PrintAndLogEx(INFO, "Starting iCLASS sim 2 attack (elite mode)");
-            PrintAndLogEx(INFO, "press keyboard to cancel");
+            PrintAndLogEx(INFO, "press Enter to cancel");
             PacketResponseNG resp;
             clearCommandBuffer();
             SendCommandOLD(CMD_SIMULATE_TAG_ICLASS, simType, NUM_CSNS, 0, csns, 8 * NUM_CSNS);
 
             while (!WaitForResponseTimeout(CMD_ACK, &resp, 2000)) {
                 tries++;
-                if (ukbhit()) {
-                    int gc = getchar();
-                    (void)gc;
+                if (kbd_enter_pressed()) {
                     PrintAndLogEx(WARNING, "\naborted via keyboard.");
                     return 0;
                 }
@@ -427,16 +425,14 @@ static int CmdHFiClassSim(const char *Cmd) {
         case 4: {
             // reader in key roll mode,  when it has two keys it alternates when trying to verify.
             PrintAndLogEx(INFO, "Starting iCLASS sim 4 attack (elite mode, reader in key roll mode)");
-            PrintAndLogEx(INFO, "press keyboard to cancel");
+            PrintAndLogEx(INFO, "press Enter to cancel");
             PacketResponseNG resp;
             clearCommandBuffer();
             SendCommandOLD(CMD_SIMULATE_TAG_ICLASS, simType, NUM_CSNS, 0, csns, 8 * NUM_CSNS);
 
             while (!WaitForResponseTimeout(CMD_ACK, &resp, 2000)) {
                 tries++;
-                if (ukbhit()) {
-                    int gc = getchar();
-                    (void)gc;
+                if (kbd_enter_pressed()) {
                     PrintAndLogEx(WARNING, "\naborted via keyboard.");
                     return 0;
                 }
@@ -554,14 +550,14 @@ static int CmdHFiClassELoad(const char *Cmd) {
     fseek(f, 0, SEEK_SET);
 
     if (fsize <= 0) {
-        PrintAndLogDevice(WARNING, "error, when getting filesize");
+        PrintAndLogDevice(ERR, "error, when getting filesize");
         fclose(f);
         return 1;
     }
 
     uint8_t *dump = calloc(fsize, sizeof(uint8_t));
     if (!dump) {
-        PrintAndLogDevice(WARNING, "error, cannot allocate memory ");
+        PrintAndLogDevice(ERR, "error, cannot allocate memory ");
         fclose(f);
         return 1;
     }
@@ -573,7 +569,7 @@ static int CmdHFiClassELoad(const char *Cmd) {
     //Validate
 
     if (bytes_read < fsize) {
-        PrintAndLogDevice(WARNING, "error, could only read %d bytes (should be %d)", bytes_read, fsize);
+        PrintAndLogDevice(ERR, "error, could only read %d bytes (should be %d)", bytes_read, fsize);
         free(dump);
         return 1;
     }
@@ -654,7 +650,7 @@ static int CmdHFiClassDecrypt(const char *Cmd) {
     fseek(f, 0, SEEK_SET);
 
     if (fsize <= 0) {
-        PrintAndLogEx(WARNING, "error, when getting filesize");
+        PrintAndLogEx(ERR, "error, when getting filesize");
         fclose(f);
         return 2;
     }
@@ -669,7 +665,7 @@ static int CmdHFiClassDecrypt(const char *Cmd) {
     size_t bytes_read = fread(decrypted, 1, fsize, f);
     fclose(f);
     if (bytes_read == 0) {
-        PrintAndLogEx(WARNING, "file reading error");
+        PrintAndLogEx(ERR, "file reading error");
         free(decrypted);
         return 3;
     }
@@ -969,9 +965,7 @@ static int CmdHFiClassReader_Dump(const char *Cmd) {
     while (true) {
         printf(".");
         fflush(stdout);
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(WARNING, "\n[!] aborted via keyboard!\n");
             DropField();
             return 0;
@@ -996,7 +990,7 @@ static int CmdHFiClassReader_Dump(const char *Cmd) {
     }
 
     // response ok - now get bigbuf content of the dump
-    if (!GetFromDevice(BIG_BUF, tag_data + (blockno * 8), blocksRead * 8, startindex, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF, tag_data + (blockno * 8), blocksRead * 8, startindex, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "command execution time out");
         return 0;
     }
@@ -1039,7 +1033,7 @@ static int CmdHFiClassReader_Dump(const char *Cmd) {
                 blocksRead = (sizeof(tag_data) - gotBytes) / 8;
             }
             // get dumped data from bigbuf
-            if (!GetFromDevice(BIG_BUF, tag_data + gotBytes, blocksRead * 8, startindex, NULL, 2500, false)) {
+            if (!GetFromDevice(BIG_BUF, tag_data + gotBytes, blocksRead * 8, startindex, NULL, 0, NULL, 2500, false)) {
                 PrintAndLogEx(WARNING, "command execution time out");
                 return 0;
             }
@@ -1287,7 +1281,7 @@ static int CmdHFiClassCloneTag(const char *Cmd) {
     fseek(f, startblock * 8, SEEK_SET);
     size_t bytes_read = fread(tag_data, sizeof(iclass_block_t), endblock - startblock + 1, f);
     if (bytes_read == 0) {
-        PrintAndLogEx(WARNING, "file reading error.");
+        PrintAndLogEx(ERR, "file reading error.");
         fclose(f);
         return 2;
     }
@@ -1454,7 +1448,7 @@ static int CmdHFiClass_loclass(const char *Cmd) {
         errors += testMAC();
         errors += doKeyTests(0);
         errors += testElite();
-        if (errors) PrintAndLogDevice(WARNING, "There were errors!!!");
+        if (errors) PrintAndLogDevice(ERR, "There were errors!!!");
         return errors;
     }
     return PM3_SUCCESS;
@@ -1524,7 +1518,7 @@ static int CmdHFiClassReadTagFile(const char *Cmd) {
     fseek(f, 0, SEEK_SET);
 
     if (fsize <= 0) {
-        PrintAndLogEx(WARNING, "Error, when getting filesize");
+        PrintAndLogEx(ERR, "Error, when getting filesize");
         fclose(f);
         return 1;
     }
@@ -1682,7 +1676,7 @@ static int loadKeys(char *filename) {
     fseek(f, 0, SEEK_SET);
 
     if (fsize <= 0) {
-        PrintAndLogEx(WARNING, "Error, when getting filesize");
+        PrintAndLogEx(ERR, "Error, when getting filesize");
         fclose(f);
         return 1;
     }
@@ -1945,9 +1939,7 @@ static int CmdHFiClassCheckKeys(const char *Cmd) {
         uint64_t t2 = msclock();
         uint8_t timeout = 0;
 
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(WARNING, "\n[!] Aborted via keyboard!\n");
             goto out;
         }
@@ -2446,7 +2438,7 @@ int readIclass(bool loop, bool verbose) {
 
     // loop in client not device - else on windows have a communication error
     PacketResponseNG resp;
-    while (!ukbhit()) {
+    while (!kbd_enter_pressed()) {
 
         clearCommandBuffer();
         SendCommandMIX(CMD_READER_ICLASS, flags, 0, 0, NULL, 0);
diff --git a/client/cmdhflegic.c b/client/cmdhflegic.c
index 8bd976030..e8089c3d8 100644
--- a/client/cmdhflegic.c
+++ b/client/cmdhflegic.c
@@ -781,7 +781,7 @@ int legic_read_mem(uint32_t offset, uint32_t len, uint32_t iv, uint8_t *out, uin
         PrintAndLogEx(WARNING, "Fail, only managed to read %u bytes", *outlen);
 
     // copy data from device
-    if (!GetFromDevice(BIG_BUF_EML, out, *outlen, 0, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF_EML, out, *outlen, 0, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "Fail, transfer from device time-out");
         return 4;
     }
@@ -930,7 +930,7 @@ static int CmdLegicDump(const char *Cmd) {
         PrintAndLogEx(WARNING, "Fail, only managed to read 0x%02X bytes of 0x%02X", readlen, dumplen);
 
     // copy data from device
-    if (!GetFromDevice(BIG_BUF_EML, data, readlen, 0, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF_EML, data, readlen, 0, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "Fail, transfer from device time-out");
         free(data);
         return 4;
@@ -1037,7 +1037,7 @@ static int CmdLegicRestore(const char *Cmd) {
     fclose(f);
 
     if (bytes_read == 0) {
-        PrintAndLogEx(WARNING, "File reading error");
+        PrintAndLogEx(ERR, "File reading error");
         free(data);
         return 2;
     }
@@ -1139,7 +1139,7 @@ static int CmdLegicELoad(const char *Cmd) {
     // load file
     size_t bytes_read = fread(data, 1, numofbytes, f);
     if (bytes_read == 0) {
-        PrintAndLogEx(WARNING, "File reading error");
+        PrintAndLogEx(ERR, "File reading error");
         free(data);
         fclose(f);
         f = NULL;
@@ -1200,7 +1200,7 @@ static int CmdLegicESave(const char *Cmd) {
 
     // download emulator memory
     PrintAndLogEx(SUCCESS, "Reading emulator memory...");
-    if (!GetFromDevice(BIG_BUF_EML, data, numofbytes, 0, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF_EML, data, numofbytes, 0, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "Fail, transfer from device time-out");
         free(data);
         return 4;
diff --git a/client/cmdhfmf.c b/client/cmdhfmf.c
index 4fa3d8dc0..6e2ca62f5 100644
--- a/client/cmdhfmf.c
+++ b/client/cmdhfmf.c
@@ -765,7 +765,7 @@ static int CmdHF14AMfDump(const char *Cmd) {
     for (sectorNo = 0; sectorNo < numSectors; sectorNo++) {
         bytes_read = fread(keyA[sectorNo], 1, 6, f);
         if (bytes_read != 6) {
-            PrintAndLogEx(WARNING, "File reading error.");
+            PrintAndLogEx(ERR, "File reading error.");
             fclose(f);
             return PM3_EFILE;
         }
@@ -775,7 +775,7 @@ static int CmdHF14AMfDump(const char *Cmd) {
     for (sectorNo = 0; sectorNo < numSectors; sectorNo++) {
         bytes_read = fread(keyB[sectorNo], 1, 6, f);
         if (bytes_read != 6) {
-            PrintAndLogEx(WARNING, "File reading error.");
+            PrintAndLogEx(ERR, "File reading error.");
             fclose(f);
             return PM3_EFILE;
         }
@@ -984,7 +984,7 @@ static int CmdHF14AMfRestore(const char *Cmd) {
     for (sectorNo = 0; sectorNo < numSectors; sectorNo++) {
         bytes_read = fread(keyA[sectorNo], 1, 6, fkeys);
         if (bytes_read != 6) {
-            PrintAndLogEx(WARNING, "File reading error  " _YELLOW_("%s"), keyFilename);
+            PrintAndLogEx(ERR, "File reading error  " _YELLOW_("%s"), keyFilename);
             fclose(fkeys);
             return 2;
         }
@@ -993,7 +993,7 @@ static int CmdHF14AMfRestore(const char *Cmd) {
     for (sectorNo = 0; sectorNo < numSectors; sectorNo++) {
         bytes_read = fread(keyB[sectorNo], 1, 6, fkeys);
         if (bytes_read != 6) {
-            PrintAndLogEx(WARNING, "File reading error " _YELLOW_("%s"), keyFilename);
+            PrintAndLogEx(ERR, "File reading error " _YELLOW_("%s"), keyFilename);
             fclose(fkeys);
             return 2;
         }
@@ -1021,7 +1021,7 @@ static int CmdHF14AMfRestore(const char *Cmd) {
             memcpy(data, key, 6);
             bytes_read = fread(bldata, 1, 16, fdump);
             if (bytes_read != 16) {
-                PrintAndLogEx(WARNING, "File reading error " _YELLOW_("%s"), dataFilename);
+                PrintAndLogEx(ERR, "File reading error " _YELLOW_("%s"), dataFilename);
                 fclose(fdump);
                 fdump = NULL;
                 return 2;
@@ -1129,7 +1129,7 @@ static int CmdHF14AMfNested(const char *Cmd) {
         int16_t isOK = mfnested(blockNo, keyType, key, trgBlockNo, trgKeyType, keyBlock, true);
         switch (isOK) {
             case -1 :
-                PrintAndLogEx(WARNING, "Error: No response from Proxmark3.\n");
+                PrintAndLogEx(ERR, "Error: No response from Proxmark3.\n");
                 break;
             case -2 :
                 PrintAndLogEx(WARNING, "Button pressed. Aborted.\n");
@@ -1162,7 +1162,7 @@ static int CmdHF14AMfNested(const char *Cmd) {
                 }
                 return PM3_SUCCESS;
             default :
-                PrintAndLogEx(WARNING, "Unknown Error.\n");
+                PrintAndLogEx(ERR, "Unknown Error.\n");
         }
         return PM3_SUCCESS;
     } else { // ------------------------------------  multiple sectors working
@@ -1199,7 +1199,7 @@ static int CmdHF14AMfNested(const char *Cmd) {
                     int16_t isOK = mfnested(blockNo, keyType, key, FirstBlockOfSector(sectorNo), trgKeyType, keyBlock, calibrate);
                     switch (isOK) {
                         case -1 :
-                            PrintAndLogEx(WARNING, "error: No response from Proxmark3.\n");
+                            PrintAndLogEx(ERR, "error: No response from Proxmark3.\n");
                             break;
                         case -2 :
                             PrintAndLogEx(WARNING, "button pressed. Aborted.\n");
@@ -1221,7 +1221,7 @@ static int CmdHF14AMfNested(const char *Cmd) {
                             continue;
 
                         default :
-                            PrintAndLogEx(WARNING, "unknown Error.\n");
+                            PrintAndLogEx(ERR, "unknown Error.\n");
                     }
                     free(e_sector);
                     return PM3_ESOFT;
@@ -1502,7 +1502,7 @@ static int CmdHF14AMfNestedHard(const char *Cmd) {
     if (isOK) {
         switch (isOK) {
             case 1 :
-                PrintAndLogEx(WARNING, "Error: No response from Proxmark3.\n");
+                PrintAndLogEx(ERR, "Error: No response from Proxmark3.\n");
                 break;
             case 2 :
                 PrintAndLogEx(NORMAL, "Button pressed. Aborted.\n");
@@ -1692,9 +1692,7 @@ static int CmdHF14AMfChk_fast(const char *Cmd) {
             // main keychunk loop
             for (i = 0; i < keycnt; i += chunksize) {
 
-                if (ukbhit()) {
-                    int gc = getchar();
-                    (void)gc;
+                if (kbd_enter_pressed()) {
                     PrintAndLogEx(WARNING, "\naborted via keyboard!\n");
                     goto out;
                 }
@@ -1961,6 +1959,8 @@ static int CmdHF14AMfChk(const char *Cmd) {
     // fast push mode
     conn.block_after_ACK = true;
 
+    // clear trace log by first check keys call only
+    bool clearLog = true;
     // check keys.
     for (trgKeyType = (keyType == 2) ? 0 : keyType; trgKeyType < 2; (keyType == 2) ? (++trgKeyType) : (trgKeyType = 2)) {
 
@@ -1974,20 +1974,20 @@ static int CmdHF14AMfChk(const char *Cmd) {
 
                 printf(".");
                 fflush(stdout);
-                if (ukbhit()) {
-                    int gc = getchar();
-                    (void)gc;
+                if (kbd_enter_pressed()) {
                     PrintAndLogEx(INFO, "\naborted via keyboard!\n");
                     goto out;
                 }
 
                 uint16_t size = keycnt - c > max_keys ? max_keys : keycnt - c;
 
-                if (mfCheckKeys(b, trgKeyType, true, size, &keyBlock[6 * c], &key64) == PM3_SUCCESS) {
+                if (mfCheckKeys(b, trgKeyType, clearLog, size, &keyBlock[6 * c], &key64) == PM3_SUCCESS) {
                     e_sector[i].Key[trgKeyType] = key64;
                     e_sector[i].foundKey[trgKeyType] = true;
+                    clearLog = false;
                     break;
                 }
+                clearLog = false;
             }
             b < 127 ? (b += 4) : (b += 16);
         }
@@ -2281,7 +2281,7 @@ static int CmdHF14AMfSim(const char *Cmd) {
     if (flags & FLAG_INTERACTIVE) {
         PrintAndLogEx(INFO, "Press pm3-button or send another cmd to abort simulation");
 
-        while (!ukbhit()) {
+        while (!kbd_enter_pressed()) {
             if (!WaitForResponseTimeout(CMD_ACK, &resp, 1500)) continue;
             if (!(flags & FLAG_NR_AR_ATTACK)) break;
             if ((resp.oldarg[0] & 0xffff) != CMD_SIMULATE_MIFARE_CARD) break;
@@ -2328,8 +2328,8 @@ static int CmdHF14AMfSniff(const char *Cmd) {
 
     PrintAndLogEx(NORMAL, "-------------------------------------------------------------------------\n");
     PrintAndLogEx(NORMAL, "Executing mifare sniffing command. \n");
-    PrintAndLogEx(NORMAL, "Press the key on the Proxmark3 device to abort both Proxmark3 and client.\n");
-    PrintAndLogEx(NORMAL, "Press the key on pc keyboard to abort the client.\n");
+    PrintAndLogEx(NORMAL, "Press the button on the Proxmark3 device to abort both Proxmark3 and client.\n");
+    PrintAndLogEx(NORMAL, "Press Enter to abort the client.\n");
     PrintAndLogEx(NORMAL, "-------------------------------------------------------------------------\n");
 
     clearCommandBuffer();
@@ -2341,9 +2341,7 @@ static int CmdHF14AMfSniff(const char *Cmd) {
     while (true) {
         printf(".");
         fflush(stdout);
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(INFO, "\naborted via keyboard!\n");
             break;
         }
@@ -2695,7 +2693,7 @@ static int CmdHF14AMfESave(const char *Cmd) {
     memset(dump, 0, bytes);
 
     PrintAndLogEx(INFO, "downloading from emulator memory");
-    if (!GetFromDevice(BIG_BUF_EML, dump, bytes, 0, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF_EML, dump, bytes, 0, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "Fail, transfer from device time-out");
         free(dump);
         return PM3_ETIMEOUT;
@@ -2830,7 +2828,7 @@ static int CmdHF14AMfCSetUID(const char *Cmd) {
 
     res = mfCSetUID(uid, (atqaPresent) ? atqa : NULL, (atqaPresent) ? sak : NULL, oldUid, wipeCard);
     if (res) {
-        PrintAndLogEx(WARNING, "Can't set UID. error=%d", res);
+        PrintAndLogEx(ERR, "Can't set UID. error=%d", res);
         return PM3_ESOFT;
     }
 
@@ -2860,7 +2858,7 @@ static int CmdHF14AMfCSetBlk(const char *Cmd) {
 
     res = mfCSetBlock(blockNo, block, NULL, params);
     if (res) {
-        PrintAndLogEx(WARNING, "Can't write block. error=%d", res);
+        PrintAndLogEx(ERR, "Can't write block. error=%d", res);
         return PM3_ESOFT;
     }
     return PM3_SUCCESS;
@@ -2935,7 +2933,7 @@ static int CmdHF14AMfCLoad(const char *Cmd) {
 
     // 64 or 256blocks.
     if (datalen != 1024 && datalen != 4096) {
-        PrintAndLogEx(WARNING, "File content error. ");
+        PrintAndLogEx(ERR, "File content error. ");
         free(data);
         return PM3_EFILE;
     }
@@ -2974,7 +2972,7 @@ static int CmdHF14AMfCLoad(const char *Cmd) {
 
     // 64 or 256blocks.
     if (blockNum != 16 * 4 && blockNum != 32 * 4 + 8 * 16) {
-        PrintAndLogEx(WARNING, "File content error. There must be 64 blocks");
+        PrintAndLogEx(ERR, "File content error. There must be 64 blocks");
         free(data);
         return PM3_EFILE;
     }
@@ -2996,7 +2994,7 @@ static int CmdHF14AMfCGetBlk(const char *Cmd) {
 
     int res = mfCGetBlock(blockNo, data, MAGIC_SINGLE);
     if (res) {
-        PrintAndLogEx(WARNING, "Can't read block. error=%d", res);
+        PrintAndLogEx(ERR, "Can't read block. error=%d", res);
         return PM3_ESOFT;
     }
 
@@ -3047,7 +3045,7 @@ static int CmdHF14AMfCGetSc(const char *Cmd) {
 
         int res = mfCGetBlock(start + i, data, flags);
         if (res) {
-            PrintAndLogEx(WARNING, "Can't read block. %d error=%d", start + i, res);
+            PrintAndLogEx(ERR, "Can't read block. %d error=%d", start + i, res);
             return PM3_ESOFT;
         }
         PrintAndLogEx(NORMAL, "%3d | %s", start + i, sprint_hex(data, 16));
@@ -3242,7 +3240,7 @@ static int CmdHf14AMfNack(const char *Cmd) {
     bool verbose = (ctmp == 'v');
 
     if (verbose)
-        PrintAndLogEx(INFO, "Started testing card for NACK bug. Press key to abort");
+        PrintAndLogEx(INFO, "Started testing card for NACK bug. Press Enter to abort");
 
     detect_classic_nackbug(verbose);
     return PM3_SUCCESS;
@@ -3307,9 +3305,7 @@ static int CmdHF14AMfice(const char *Cmd) {
     uint64_t t1 = msclock();
 
     do {
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(INFO, "\naborted via keyboard!\n");
             break;
         }
@@ -3595,7 +3591,7 @@ static command_t CommandTable[] = {
     {"list",        CmdHF14AMfList,         AlwaysAvailable,  "List Mifare history"},
     {"darkside",    CmdHF14AMfDarkside,     IfPm3Iso14443a,  "Darkside attack. read parity error messages."},
     {"nested",      CmdHF14AMfNested,       IfPm3Iso14443a,  "Nested attack. Test nested authentication"},
-    {"hardnested",  CmdHF14AMfNestedHard,   IfPm3Iso14443a,  "Nested attack for hardened Mifare cards"},
+    {"hardnested",  CmdHF14AMfNestedHard,   AlwaysAvailable, "Nested attack for hardened Mifare cards"},
     {"keybrute",    CmdHF14AMfKeyBrute,     IfPm3Iso14443a,  "J_Run's 2nd phase of multiple sector nested authentication key recovery"},
     {"nack",        CmdHf14AMfNack,         IfPm3Iso14443a,  "Test for Mifare NACK bug"},
     {"chk",         CmdHF14AMfChk,          IfPm3Iso14443a,  "Check keys"},
diff --git a/client/cmdhfmfhard.c b/client/cmdhfmfhard.c
index 2b96d8d77..4de46d36a 100644
--- a/client/cmdhfmfhard.c
+++ b/client/cmdhfmfhard.c
@@ -258,7 +258,7 @@ static void init_bitflip_bitarrays(void) {
                 fseek(statesfile, 0, SEEK_END);
                 int fsize = ftell(statesfile);
                 if (fsize == -1) {
-                    PrintAndLogEx(WARNING, "File read error with %s. Aborting...\n", state_file_name);
+                    PrintAndLogEx(ERR, "File read error with %s. Aborting...\n", state_file_name);
                     fclose(statesfile);
                     exit(5);
                 }
@@ -267,7 +267,7 @@ static void init_bitflip_bitarrays(void) {
                 uint8_t input_buffer[filesize];
                 size_t bytesread = fread(input_buffer, 1, filesize, statesfile);
                 if (bytesread != filesize) {
-                    PrintAndLogEx(WARNING, "File read error with %s. Aborting...\n", state_file_name);
+                    PrintAndLogEx(ERR, "File read error with %s. Aborting...\n", state_file_name);
                     fclose(statesfile);
                     //inflateEnd(&compressed_stream);
                     exit(5);
@@ -279,7 +279,7 @@ static void init_bitflip_bitarrays(void) {
                 if ((float)count / (1 << 24) < IGNORE_BITFLIP_THRESHOLD) {
                     uint32_t *bitset = (uint32_t *)malloc_bitarray(sizeof(uint32_t) * (1 << 19));
                     if (bitset == NULL) {
-                        PrintAndLogEx(WARNING, "Out of memory error in init_bitflip_statelists(). Aborting...\n");
+                        PrintAndLogEx(ERR, "Out of memory error in init_bitflip_statelists(). Aborting...\n");
                         inflateEnd(&compressed_stream);
                         exit(4);
                     }
@@ -390,7 +390,7 @@ static void init_part_sum_bitarrays(void) {
         for (uint16_t part_sum_a0 = 0; part_sum_a0 < NUM_PART_SUMS; part_sum_a0++) {
             part_sum_a0_bitarrays[odd_even][part_sum_a0] = (uint32_t *)malloc_bitarray(sizeof(uint32_t) * (1 << 19));
             if (part_sum_a0_bitarrays[odd_even][part_sum_a0] == NULL) {
-                PrintAndLogEx(WARNING, "Out of memory error in init_part_suma0_statelists(). Aborting...\n");
+                PrintAndLogEx(ERR, "Out of memory error in init_part_suma0_statelists(). Aborting...\n");
                 exit(4);
             }
             clear_bitarray24(part_sum_a0_bitarrays[odd_even][part_sum_a0]);
@@ -410,7 +410,7 @@ static void init_part_sum_bitarrays(void) {
         for (uint16_t part_sum_a8 = 0; part_sum_a8 < NUM_PART_SUMS; part_sum_a8++) {
             part_sum_a8_bitarrays[odd_even][part_sum_a8] = (uint32_t *)malloc_bitarray(sizeof(uint32_t) * (1 << 19));
             if (part_sum_a8_bitarrays[odd_even][part_sum_a8] == NULL) {
-                PrintAndLogEx(WARNING, "Out of memory error in init_part_suma8_statelists(). Aborting...\n");
+                PrintAndLogEx(ERR, "Out of memory error in init_part_suma8_statelists(). Aborting...\n");
                 exit(4);
             }
             clear_bitarray24(part_sum_a8_bitarrays[odd_even][part_sum_a8]);
@@ -449,7 +449,7 @@ static void init_sum_bitarrays(void) {
         for (odd_even_t odd_even = EVEN_STATE; odd_even <= ODD_STATE; odd_even++) {
             sum_a0_bitarrays[odd_even][sum_a0] = (uint32_t *)malloc_bitarray(sizeof(uint32_t) * (1 << 19));
             if (sum_a0_bitarrays[odd_even][sum_a0] == NULL) {
-                PrintAndLogEx(WARNING, "Out of memory error in init_sum_bitarrays(). Aborting...\n");
+                PrintAndLogEx(ERR, "Out of memory error in init_sum_bitarrays(). Aborting...\n");
                 exit(4);
             }
             clear_bitarray24(sum_a0_bitarrays[odd_even][sum_a0]);
@@ -564,14 +564,14 @@ static void init_nonce_memory(void) {
         }
         nonces[i].states_bitarray[EVEN_STATE] = (uint32_t *)malloc_bitarray(sizeof(uint32_t) * (1 << 19));
         if (nonces[i].states_bitarray[EVEN_STATE] == NULL) {
-            PrintAndLogEx(WARNING, "Out of memory error in init_nonce_memory(). Aborting...\n");
+            PrintAndLogEx(ERR, "Out of memory error in init_nonce_memory(). Aborting...\n");
             exit(4);
         }
         set_bitarray24(nonces[i].states_bitarray[EVEN_STATE]);
         nonces[i].num_states_bitarray[EVEN_STATE] = 1 << 24;
         nonces[i].states_bitarray[ODD_STATE] = (uint32_t *)malloc_bitarray(sizeof(uint32_t) * (1 << 19));
         if (nonces[i].states_bitarray[ODD_STATE] == NULL) {
-            PrintAndLogEx(WARNING, "Out of memory error in init_nonce_memory(). Aborting...\n");
+            PrintAndLogEx(ERR, "Out of memory error in init_nonce_memory(). Aborting...\n");
             exit(4);
         }
         set_bitarray24(nonces[i].states_bitarray[ODD_STATE]);
@@ -1048,7 +1048,7 @@ static int read_nonce_file(char *filename) {
     hardnested_print_progress(0, progress_text, (float)(1LL << 47), 0);
     size_t bytes_read = fread(read_buf, 1, 6, fnonces);
     if (bytes_read != 6) {
-        PrintAndLogEx(WARNING, "File reading error.");
+        PrintAndLogEx(ERR, "File reading error.");
         fclose(fnonces);
         return 1;
     }
@@ -1711,12 +1711,12 @@ static void add_matching_states(statelist_t *candidates, uint8_t part_sum_a0, ui
     const uint32_t worstcase_size = 1 << 20;
     candidates->states[odd_even] = (uint32_t *)malloc(sizeof(uint32_t) * worstcase_size);
     if (candidates->states[odd_even] == NULL) {
-        PrintAndLogEx(WARNING, "Out of memory error in add_matching_states() - statelist.\n");
+        PrintAndLogEx(ERR, "Out of memory error in add_matching_states() - statelist.\n");
         exit(4);
     }
     uint32_t *candidates_bitarray = (uint32_t *)malloc_bitarray(sizeof(uint32_t) * worstcase_size);
     if (candidates_bitarray == NULL) {
-        PrintAndLogEx(WARNING, "Out of memory error in add_matching_states() - bitarray.\n");
+        PrintAndLogEx(ERR, "Out of memory error in add_matching_states() - bitarray.\n");
         free(candidates->states[odd_even]);
         exit(4);
     }
@@ -1772,7 +1772,7 @@ static void add_bitflip_candidates(uint8_t byte) {
         uint32_t worstcase_size = nonces[byte].num_states_bitarray[odd_even] + 1;
         candidates1->states[odd_even] = (uint32_t *)malloc(sizeof(uint32_t) * worstcase_size);
         if (candidates1->states[odd_even] == NULL) {
-            PrintAndLogEx(WARNING, "Out of memory error in add_bitflip_candidates().\n");
+            PrintAndLogEx(ERR, "Out of memory error in add_bitflip_candidates().\n");
             exit(4);
         }
 
diff --git a/client/cmdhfmfu.c b/client/cmdhfmfu.c
index eb7d26c4f..df760f187 100644
--- a/client/cmdhfmfu.c
+++ b/client/cmdhfmfu.c
@@ -1232,7 +1232,7 @@ static int CmdHF14AMfUInfo(const char *Cmd) {
     status = ul_read(0, data, sizeof(data));
     if (status == -1) {
         DropField();
-        PrintAndLogEx(WARNING, "Error: tag didn't answer to READ");
+        PrintAndLogEx(ERR, "Error: tag didn't answer to READ");
         return status;
     } else if (status == 16) {
         ul_print_default(data);
@@ -1248,7 +1248,7 @@ static int CmdHF14AMfUInfo(const char *Cmd) {
         uint8_t ulc_conf[16] = {0x00};
         status = ul_read(0x28, ulc_conf, sizeof(ulc_conf));
         if (status == -1) {
-            PrintAndLogEx(WARNING, "Error: tag didn't answer to READ UL-C");
+            PrintAndLogEx(ERR, "Error: tag didn't answer to READ UL-C");
             DropField();
             return status;
         }
@@ -1263,7 +1263,7 @@ static int CmdHF14AMfUInfo(const char *Cmd) {
             status = ul_read(0x2C, ulc_deskey, sizeof(ulc_deskey));
             if (status == -1) {
                 DropField();
-                PrintAndLogEx(WARNING, "Error: tag didn't answer to READ magic");
+                PrintAndLogEx(ERR, "Error: tag didn't answer to READ magic");
                 return status;
             }
             if (status == 16) ulc_print_3deskey(ulc_deskey);
@@ -1306,7 +1306,7 @@ static int CmdHF14AMfUInfo(const char *Cmd) {
         uint8_t ulev1_signature[32] = {0x00};
         status = ulev1_readSignature(ulev1_signature, sizeof(ulev1_signature));
         if (status == -1) {
-            PrintAndLogEx(WARNING, "Error: tag didn't answer to READ SIGNATURE");
+            PrintAndLogEx(ERR, "Error: tag didn't answer to READ SIGNATURE");
             DropField();
             return status;
         }
@@ -1322,7 +1322,7 @@ static int CmdHF14AMfUInfo(const char *Cmd) {
         uint8_t version[10] = {0x00};
         status  = ulev1_getVersion(version, sizeof(version));
         if (status == -1) {
-            PrintAndLogEx(WARNING, "Error: tag didn't answer to GETVERSION");
+            PrintAndLogEx(ERR, "Error: tag didn't answer to GETVERSION");
             DropField();
             return status;
         } else if (status == 10) {
@@ -1346,7 +1346,7 @@ static int CmdHF14AMfUInfo(const char *Cmd) {
         if (startconfigblock) { // if we know where the config block is...
             status = ul_read(startconfigblock, ulev1_conf, sizeof(ulev1_conf));
             if (status == -1) {
-                PrintAndLogEx(WARNING, "Error: tag didn't answer to READ EV1");
+                PrintAndLogEx(ERR, "Error: tag didn't answer to READ EV1");
                 DropField();
                 return status;
             } else if (status == 16) {
@@ -1931,7 +1931,7 @@ static int CmdHF14AMfUDump(const char *Cmd) {
         bufferSize = sizeof(data);
     }
 
-    if (!GetFromDevice(BIG_BUF, data, bufferSize, startindex, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF, data, bufferSize, startindex, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "command execution time out");
         return 1;
     }
@@ -2134,7 +2134,7 @@ static int CmdHF14AMfURestore(const char *Cmd) {
     long fsize = ftell(f);
     fseek(f, 0, SEEK_SET);
     if (fsize <= 0) {
-        PrintAndLogEx(WARNING, "Error, when getting filesize");
+        PrintAndLogEx(ERR, "Error, when getting filesize");
         fclose(f);
         return 1;
     }
@@ -2150,7 +2150,7 @@ static int CmdHF14AMfURestore(const char *Cmd) {
     size_t bytes_read = fread(dump, 1, fsize, f);
     fclose(f);
     if (bytes_read < MFU_DUMP_PREFIX_LENGTH) {
-        PrintAndLogEx(WARNING, "Error, dump file is too small");
+        PrintAndLogEx(ERR, "Error, dump file is too small");
         free(dump);
         return 1;
     }
@@ -2167,7 +2167,7 @@ static int CmdHF14AMfURestore(const char *Cmd) {
     uint8_t pages = (bytes_read - MFU_DUMP_PREFIX_LENGTH) / 4;
 
     if (pages - 1 != mem->pages) {
-        PrintAndLogEx(WARNING, "Error, invalid dump, wrong page count");
+        PrintAndLogEx(ERR, "Error, invalid dump, wrong page count");
         free(dump);
         return 1;
     }
diff --git a/client/cmdhftopaz.c b/client/cmdhftopaz.c
index 2b743a574..d604d4baf 100644
--- a/client/cmdhftopaz.c
+++ b/client/cmdhftopaz.c
@@ -321,7 +321,7 @@ static void topaz_print_control_TLVs(uint8_t *memory) {
 static int topaz_read_dynamic_data(void) {
     // first read the remaining block of segment 0
     if (topaz_read_block(topaz_tag.uid, 0x0f, &topaz_tag.dynamic_memory[0]) == -1) {
-        PrintAndLogEx(WARNING, "Error while reading dynamic memory block %02x. Aborting...", 0x0f);
+        PrintAndLogEx(ERR, "Error while reading dynamic memory block %02x. Aborting...", 0x0f);
         return -1;
     }
 
@@ -329,7 +329,7 @@ static int topaz_read_dynamic_data(void) {
     uint8_t max_segment = topaz_tag.size / 128 - 1;
     for (uint8_t segment = 1; segment <= max_segment; segment++) {
         if (topaz_read_segment(topaz_tag.uid, segment, &topaz_tag.dynamic_memory[(segment - 1) * 128 + 8]) == -1) {
-            PrintAndLogEx(WARNING, "Error while reading dynamic memory block %02x. Aborting...", 0x0f);
+            PrintAndLogEx(ERR, "Error while reading dynamic memory block %02x. Aborting...", 0x0f);
             return -1;
         }
     }
@@ -381,7 +381,7 @@ static int CmdHFTopazReader(const char *Cmd) {
     status = topaz_select(atqa, rid_response);
 
     if (status == -1) {
-        if (verbose) PrintAndLogEx(WARNING, "Error: couldn't receive ATQA");
+        if (verbose) PrintAndLogEx(ERR, "Error: couldn't receive ATQA");
         return -1;
     }
 
@@ -393,7 +393,7 @@ static int CmdHFTopazReader(const char *Cmd) {
     }
 
     if (status == -2) {
-        PrintAndLogEx(WARNING, "Error: tag didn't answer to RID");
+        PrintAndLogEx(ERR, "Error: tag didn't answer to RID");
         topaz_switch_off_field();
         return -1;
     }
@@ -411,7 +411,7 @@ static int CmdHFTopazReader(const char *Cmd) {
     status = topaz_rall(uid_echo, rall_response);
 
     if (status == -1) {
-        PrintAndLogEx(WARNING, "Error: tag didn't answer to RALL");
+        PrintAndLogEx(ERR, "Error: tag didn't answer to RALL");
         topaz_switch_off_field();
         return -1;
     }
diff --git a/client/cmdhw.c b/client/cmdhw.c
index 7fb241667..1348f16b4 100644
--- a/client/cmdhw.c
+++ b/client/cmdhw.c
@@ -479,6 +479,13 @@ static int CmdSetMux(const char *Cmd) {
     return PM3_SUCCESS;
 }
 
+static int CmdStandalone(const char *Cmd) {
+    (void)Cmd; // Cmd is not used so far
+    clearCommandBuffer();
+    SendCommandNG(CMD_STANDALONE, NULL, 0);
+    return PM3_SUCCESS;
+}
+
 static int CmdTune(const char *Cmd) {
     return CmdTuneSamples(Cmd);
 }
@@ -591,6 +598,7 @@ static command_t CommandTable[] = {
     {"reset",         CmdReset,       IfPm3Present,    "Reset the Proxmark3"},
     {"setlfdivisor",  CmdSetDivisor,  IfPm3Present,    "<19 - 255> -- Drive LF antenna at 12Mhz/(divisor+1)"},
     {"setmux",        CmdSetMux,      IfPm3Present,    "Set the ADC mux to a specific value"},
+    {"standalone",    CmdStandalone,  IfPm3Present,    "Jump to the standalone mode"},
     {"status",        CmdStatus,      IfPm3Present,    "Show runtime status information about the connected Proxmark3"},
     {"tune",          CmdTune,        IfPm3Present,    "Measure antenna tuning"},
     {"version",       CmdVersion,     IfPm3Present,    "Show version information about the connected Proxmark3"},
diff --git a/client/cmdlf.c b/client/cmdlf.c
index 8459eca3a..42b26be0d 100644
--- a/client/cmdlf.c
+++ b/client/cmdlf.c
@@ -573,7 +573,7 @@ int CmdLFfskSim(const char *Cmd) {
                     dataLen = hextobinarray((char *)data, hexData);
 
                 if (dataLen == 0) errors = true;
-                if (errors) PrintAndLogEx(WARNING, "Error getting hex data");
+                if (errors) PrintAndLogEx(ERR, "Error getting hex data");
                 cmdp += 2;
                 break;
             default:
@@ -685,7 +685,7 @@ int CmdLFaskSim(const char *Cmd) {
                     dataLen = hextobinarray((char *)data, hexData);
 
                 if (dataLen == 0) errors = true;
-                if (errors) PrintAndLogEx(WARNING, "Error getting hex data, datalen: %d", dataLen);
+                if (errors) PrintAndLogEx(ERR, "Error getting hex data, datalen: %d", dataLen);
                 cmdp += 2;
                 break;
             default:
@@ -788,7 +788,7 @@ int CmdLFpskSim(const char *Cmd) {
                     dataLen = hextobinarray((char *)data, hexData);
 
                 if (dataLen == 0) errors = true;
-                if (errors) PrintAndLogEx(WARNING, "Error getting hex data");
+                if (errors) PrintAndLogEx(ERR, "Error getting hex data");
                 cmdp += 2;
                 break;
             default:
@@ -957,7 +957,7 @@ static bool CheckChipType(bool getDeviceData) {
         retval = true;
         goto out;
     }
-        
+
     //check for t55xx chip...
     if (tryDetectP1(true)) {
         PrintAndLogEx(SUCCESS, "\nChipset detection : " _GREEN_("T55xx") "found");
@@ -1084,8 +1084,8 @@ int CmdLFfind(const char *Cmd) {
     }
 out:
     // identify chipset
-    if ( CheckChipType(isOnline) == false ) {
-        PrintAndLogEx(DEBUG, "Automatic chip type detection " _RED_("failed") );
+    if (CheckChipType(isOnline) == false) {
+        PrintAndLogEx(DEBUG, "Automatic chip type detection " _RED_("failed"));
     }
     return PM3_SUCCESS;
 }
diff --git a/client/cmdlfawid.c b/client/cmdlfawid.c
index 7cb24d9c2..386d14e5e 100644
--- a/client/cmdlfawid.c
+++ b/client/cmdlfawid.c
@@ -103,7 +103,7 @@ static int sendTry(uint8_t fmtlen, uint32_t fc, uint32_t cn, uint32_t delay, uin
         PrintAndLogEx(INFO, "Trying FC: %u; CN: %u", fc, cn);
 
     if (getAWIDBits(fmtlen, fc, cn, bits) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
@@ -333,7 +333,7 @@ static int CmdAWIDSim(const char *Cmd) {
     verify_values(&fmtlen, &fc, &cn);
 
     if (getAWIDBits(fmtlen, fc, cn, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
@@ -388,7 +388,7 @@ static int CmdAWIDClone(const char *Cmd) {
     verify_values(&fmtlen, &fc, &cn);
 
     if (getAWIDBits(fmtlen, fc, cn, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
@@ -419,7 +419,7 @@ static int CmdAWIDClone(const char *Cmd) {
 
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
@@ -491,7 +491,7 @@ static int CmdAWIDBrute(const char *Cmd) {
     }
 
     PrintAndLogEx(SUCCESS, "Bruteforceing AWID %d Reader", fmtlen);
-    PrintAndLogEx(SUCCESS, "Press pm3-button to abort simulation or press key");
+    PrintAndLogEx(SUCCESS, "Press pm3-button to abort simulation or press Enter");
 
     uint16_t up = cn;
     uint16_t down = cn;
@@ -503,9 +503,7 @@ static int CmdAWIDBrute(const char *Cmd) {
             PrintAndLogEx(WARNING, "Device offline\n");
             return PM3_ENODATA;
         }
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(INFO, "aborted via keyboard!");
             return sendPing();
         }
diff --git a/client/cmdlfcotag.c b/client/cmdlfcotag.c
index 0f5e7a835..134d55c68 100644
--- a/client/cmdlfcotag.c
+++ b/client/cmdlfcotag.c
@@ -91,7 +91,7 @@ static int CmdCOTAGRead(const char *Cmd) {
         }
         case 1: {
 
-            if (!GetFromDevice(BIG_BUF, DemodBuffer, COTAG_BITS, 0, NULL, 1000, false)) {
+            if (!GetFromDevice(BIG_BUF, DemodBuffer, COTAG_BITS, 0, NULL, 0, NULL, 1000, false)) {
                 PrintAndLogEx(WARNING, "timeout while waiting for reply.");
                 return PM3_ETIMEOUT;
             }
diff --git a/client/cmdlfem4x.c b/client/cmdlfem4x.c
index 6a5fae76f..165f22001 100644
--- a/client/cmdlfem4x.c
+++ b/client/cmdlfem4x.c
@@ -243,7 +243,7 @@ void printEM410x(uint32_t hi, uint64_t id) {
 
     if (!id && !hi) return;
 
-    PrintAndLogEx(SUCCESS, "EM410x%s pattern found", (hi) ? " XL	" : "");
+    PrintAndLogEx(SUCCESS, "EM410x%s pattern found", (hi) ? " XL" : "");
 
     uint64_t n = 1;
     uint64_t id2lo = 0;
@@ -482,12 +482,12 @@ static int CmdEM410xBrute(const char *Cmd) {
 
     int filelen = param_getstr(Cmd, 0, filename, FILE_PATH_SIZE);
     if (filelen == 0) {
-        PrintAndLogEx(WARNING, "Error: Please specify a filename");
+        PrintAndLogEx(ERR, "Error: Please specify a filename");
         return PM3_EINVARG;
     }
 
     if ((f = fopen(filename, "r")) == NULL) {
-        PrintAndLogEx(WARNING, "Error: Could not open UIDs file ["_YELLOW_("%s")"]", filename);
+        PrintAndLogEx(ERR, "Error: Could not open UIDs file ["_YELLOW_("%s")"]", filename);
         return PM3_EFILE;
     }
 
@@ -544,9 +544,7 @@ static int CmdEM410xBrute(const char *Cmd) {
         char testuid[11];
         testuid[10] = 0;
 
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(WARNING, "\nAborted via keyboard!\n");
             free(uidBlock);
             return PM3_EOPABORTED;
@@ -580,9 +578,7 @@ static int CmdEM410xBrute(const char *Cmd) {
 static int CmdEM410xWatch(const char *Cmd) {
     (void)Cmd; // Cmd is not used so far
     do {
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(WARNING, "\naborted via keyboard!\n");
             break;
         }
@@ -617,21 +613,21 @@ static int CmdEM410xWrite(const char *Cmd) {
 
     // Check ID
     if (id == 0xFFFFFFFFFFFFFFFF) {
-        PrintAndLogEx(WARNING, "Error! ID is required.\n");
+        PrintAndLogEx(ERR, "Error! ID is required.\n");
         return PM3_EINVARG;
     }
     if (id >= 0x10000000000) {
-        PrintAndLogEx(WARNING, "Error! Given EM410x ID is longer than 40 bits.\n");
+        PrintAndLogEx(ERR, "Error! Given EM410x ID is longer than 40 bits.\n");
         return PM3_EINVARG;
     }
 
     // Check Card
     if (card == 0xFF) {
-        PrintAndLogEx(WARNING, "Error! Card type required.\n");
+        PrintAndLogEx(ERR, "Error! Card type required.\n");
         return PM3_EINVARG;
     }
     if (card < 0) {
-        PrintAndLogEx(WARNING, "Error! Bad card type selected.\n");
+        PrintAndLogEx(ERR, "Error! Bad card type selected.\n");
         return PM3_EINVARG;
     }
 
@@ -641,7 +637,7 @@ static int CmdEM410xWrite(const char *Cmd) {
 
     // Allowed clock rates: 16, 32, 40 and 64
     if ((clock1 != 16) && (clock1 != 32) && (clock1 != 64) && (clock1 != 40)) {
-        PrintAndLogEx(WARNING, "Error! Clock rate" _YELLOW_("%d")" not valid. Supported clock rates are 16, 32, 40 and 64.\n", clock1);
+        PrintAndLogEx(ERR, "Error! Clock rate" _YELLOW_("%d")" not valid. Supported clock rates are 16, 32, 40 and 64.\n", clock1);
         return PM3_EINVARG;
     }
 
@@ -841,7 +837,7 @@ int EM4x50Read(const char *Cmd, bool verbose) {
             }
         }
         if (!clk) {
-            if (verbose || g_debugMode) PrintAndLogEx(WARNING, "Error: EM4x50 - didn't find a clock");
+            if (verbose || g_debugMode) PrintAndLogEx(ERR, "Error: EM4x50 - didn't find a clock");
             return PM3_ESOFT;
         }
     } else tol = clk / 8;
@@ -990,7 +986,7 @@ static bool downloadSamplesEM() {
 
     // 8 bit preamble + 32 bit word response (max clock (128) * 40bits = 5120 samples)
     uint8_t got[6000];
-    if (!GetFromDevice(BIG_BUF, got, sizeof(got), 0, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF, got, sizeof(got), 0, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "command execution time out");
         return false;
     }
@@ -1266,7 +1262,7 @@ static int CmdEM4x05Write(const char *Cmd) {
     SendCommandNG(CMD_EM4X_WRITE_WORD, (uint8_t *)&payload, sizeof(payload));
     PacketResponseNG resp;
     if (!WaitForResponseTimeout(CMD_EM4X_WRITE_WORD, &resp, 2000)) {
-        PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+        PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
         return PM3_ETIMEOUT;
     }
 
@@ -1377,10 +1373,10 @@ static void printEM4x05config(uint32_t wordData) {
     PrintAndLogEx(NORMAL, "    PSK CF:   %u | %s", PSKcf, cf);
     PrintAndLogEx(NORMAL, "     Delay:   %u | %s", delay, cdelay);
     PrintAndLogEx(NORMAL, " LastWordR:  %02u | Address of last word for default read - meaning %u blocks are output", LWR, numblks);
-    PrintAndLogEx(NORMAL, " ReadLogin:   %u | Read login is %s", readLogin, readLogin ? _YELLOW_("required") :  _GREEN_("not required") );
-    PrintAndLogEx(NORMAL, "   ReadHKL:   %u | Read housekeeping words login is %s", readHKL, readHKL ? _YELLOW_("required") : _GREEN_("not required") );
-    PrintAndLogEx(NORMAL, "WriteLogin:   %u | Write login is %s", writeLogin, writeLogin ? _YELLOW_("required") :  _GREEN_("not required") );
-    PrintAndLogEx(NORMAL, "  WriteHKL:   %u | Write housekeeping words login is %s", writeHKL, writeHKL ? _YELLOW_("required") :  _GREEN_("not Required") );
+    PrintAndLogEx(NORMAL, " ReadLogin:   %u | Read login is %s", readLogin, readLogin ? _YELLOW_("required") :  _GREEN_("not required"));
+    PrintAndLogEx(NORMAL, "   ReadHKL:   %u | Read housekeeping words login is %s", readHKL, readHKL ? _YELLOW_("required") : _GREEN_("not required"));
+    PrintAndLogEx(NORMAL, "WriteLogin:   %u | Write login is %s", writeLogin, writeLogin ? _YELLOW_("required") :  _GREEN_("not required"));
+    PrintAndLogEx(NORMAL, "  WriteHKL:   %u | Write housekeeping words login is %s", writeHKL, writeHKL ? _YELLOW_("required") :  _GREEN_("not Required"));
     PrintAndLogEx(NORMAL, "    R.A.W.:   %u | Read after write is %s", raw, raw ? "on" : "off");
     PrintAndLogEx(NORMAL, "   Disable:   %u | Disable command is %s", disable, disable ? "accepted" : "not accepted");
     PrintAndLogEx(NORMAL, "    R.T.F.:   %u | Reader talk first is %s", rtf, rtf ? _YELLOW_("enabled") : "disabled");
@@ -1400,7 +1396,7 @@ static void printEM4x05info(uint32_t block0, uint32_t serial) {
             snprintf(ctstr + strlen(ctstr), sizeof(ctstr) - strlen(ctstr), _YELLOW_("%s"), "EM4305");
             break;
         case 8:
-            snprintf(ctstr + strlen(ctstr), sizeof(ctstr) - strlen(ctstr), _YELLOW_("%s"), "EM4205");        
+            snprintf(ctstr + strlen(ctstr), sizeof(ctstr) - strlen(ctstr), _YELLOW_("%s"), "EM4205");
             break;
         case 4:
             snprintf(ctstr + strlen(ctstr), sizeof(ctstr) - strlen(ctstr), _YELLOW_("%s"), "Unknown");
diff --git a/client/cmdlffdx.c b/client/cmdlffdx.c
index 8cf93b12f..35f3216ad 100644
--- a/client/cmdlffdx.c
+++ b/client/cmdlffdx.c
@@ -264,7 +264,7 @@ static int CmdFdxClone(const char *Cmd) {
 
     // getFDXBits(uint64_t national_id, uint16_t country, uint8_t isanimal, uint8_t isextended, uint32_t extended, uint8_t *bits)
     if (getFDXBits(animalid, countryid, 1, 0, 0, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
@@ -300,7 +300,7 @@ static int CmdFdxClone(const char *Cmd) {
 
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
diff --git a/client/cmdlfguard.c b/client/cmdlfguard.c
index 54073fc02..1b09eb18d 100644
--- a/client/cmdlfguard.c
+++ b/client/cmdlfguard.c
@@ -155,7 +155,7 @@ static int CmdGuardClone(const char *Cmd) {
     cardnumber = (cn & 0x0000FFFF);
 
     if (getGuardBits(fmtlen, facilitycode, cardnumber, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
@@ -189,7 +189,7 @@ static int CmdGuardClone(const char *Cmd) {
 
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
@@ -213,7 +213,7 @@ static int CmdGuardSim(const char *Cmd) {
     cardnumber = (cn & 0x0000FFFF);
 
     if (getGuardBits(fmtlen, facilitycode, cardnumber, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
diff --git a/client/cmdlfhid.c b/client/cmdlfhid.c
index b300858d2..cc395c8c5 100644
--- a/client/cmdlfhid.c
+++ b/client/cmdlfhid.c
@@ -580,9 +580,7 @@ static int CmdHIDBrute(const char *Cmd) {
             return PM3_ENODATA;
         }
 
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             PrintAndLogEx(INFO, "aborted via keyboard!");
             return sendPing();
         }
diff --git a/client/cmdlfhitag.c b/client/cmdlfhitag.c
index 47e08eb2e..0fc4b053c 100644
--- a/client/cmdlfhitag.c
+++ b/client/cmdlfhitag.c
@@ -137,7 +137,7 @@ static int CmdLFHitagList(const char *Cmd) {
 
     // Query for the actual size of the trace
     PacketResponseNG response;
-    if (!GetFromDevice(BIG_BUF, got, PM3_CMD_DATA_SIZE, 0, &response, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF, got, PM3_CMD_DATA_SIZE, 0, NULL, 0, &response, 2500, false)) {
         PrintAndLogEx(WARNING, "command execution time out");
         free(got);
         return 2;
@@ -152,7 +152,7 @@ static int CmdLFHitagList(const char *Cmd) {
             return 2;
         }
         got = p;
-        if (!GetFromDevice(BIG_BUF, got, traceLen, 0, NULL, 2500, false)) {
+        if (!GetFromDevice(BIG_BUF, got, traceLen, 0, NULL, 0, NULL, 2500, false)) {
             PrintAndLogEx(WARNING, "command execution time out");
             free(got);
             return 2;
@@ -177,7 +177,7 @@ static int CmdLFHitagList(const char *Cmd) {
     if (strlen(filename) > 0) {
         f = fopen(filename, "wb");
         if (!f) {
-            PrintAndLogEx(WARNING, "Error: Could not open file [%s]", filename);
+            PrintAndLogEx(ERR, "Error: Could not open file [%s]", filename);
             return PM3_EFILE;
         }
     }
diff --git a/client/cmdlfio.c b/client/cmdlfio.c
index ad37b9fd7..d187daaab 100644
--- a/client/cmdlfio.c
+++ b/client/cmdlfio.c
@@ -199,7 +199,7 @@ static int CmdIOProxSim(const char *Cmd) {
     PrintAndLogEx(SUCCESS, "Press pm3-button to abort simulation or run another command");
 
     if (getIOProxBits(version, fc, cn, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
     // IOProx uses: fcHigh: 10, fcLow: 8, clk: 64, invert: 1
@@ -249,7 +249,7 @@ static int CmdIOProxClone(const char *Cmd) {
     }
 
     if (getIOProxBits(version, fc, cn, bits) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
diff --git a/client/cmdlfjablotron.c b/client/cmdlfjablotron.c
index a5b9166ac..c2478705c 100644
--- a/client/cmdlfjablotron.c
+++ b/client/cmdlfjablotron.c
@@ -145,7 +145,7 @@ static int CmdJablotronClone(const char *Cmd) {
     }
 
     if (getJablotronBits(fullcode, bits) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
@@ -174,7 +174,7 @@ static int CmdJablotronClone(const char *Cmd) {
 
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
diff --git a/client/cmdlfkeri.c b/client/cmdlfkeri.c
index 82d1b9f84..b3761055f 100644
--- a/client/cmdlfkeri.c
+++ b/client/cmdlfkeri.c
@@ -164,7 +164,7 @@ static int CmdKeriClone(const char *Cmd) {
 
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
diff --git a/client/cmdlfnedap.c b/client/cmdlfnedap.c
index d081ff56c..0b998d716 100644
--- a/client/cmdlfnedap.c
+++ b/client/cmdlfnedap.c
@@ -182,7 +182,7 @@ static int CmdLFNedapClone(const char *Cmd) {
     cardnumber = (cn & 0x00FFFFFF);
 
     if ( getNedapBits(cardnumber, bits) == PM3_SUCCESS ) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
@@ -220,7 +220,7 @@ static int CmdLFNedapClone(const char *Cmd) {
 
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
@@ -243,7 +243,7 @@ static int CmdLFNedapSim(const char *Cmd) {
     memset(bs, 0x00, sizeof(bs));
 
     if (getNedapBits(cardnumber, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
diff --git a/client/cmdlfnoralsy.c b/client/cmdlfnoralsy.c
index 9e3413405..34ad1023f 100644
--- a/client/cmdlfnoralsy.c
+++ b/client/cmdlfnoralsy.c
@@ -142,7 +142,7 @@ static int CmdNoralsyClone(const char *Cmd) {
         blocks[0] = T5555_MODULATION_MANCHESTER | T5555_SET_BITRATE(32) | T5555_ST_TERMINATOR | 3 << T5555_MAXBLOCK_SHIFT;
 
     if (getnoralsyBits(id, year, bits) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
@@ -172,7 +172,7 @@ static int CmdNoralsyClone(const char *Cmd) {
 
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
@@ -195,7 +195,7 @@ static int CmdNoralsySim(const char *Cmd) {
     year = param_get32ex(Cmd, 1, 2000, 10);
 
     if (getnoralsyBits(id, year, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
diff --git a/client/cmdlfparadox.c b/client/cmdlfparadox.c
index 684730ef3..c5c022d9b 100644
--- a/client/cmdlfparadox.c
+++ b/client/cmdlfparadox.c
@@ -128,7 +128,7 @@ static int CmdParadoxSim(const char *Cmd) {
     cardnumber = (cn & 0x0000FFFF);
 
     // if ( GetParadoxBits(facilitycode, cardnumber, bs) != PM3_SUCCESS) {
-    // PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+    // PrintAndLogEx(ERR, "Error with tag bitstream generation.");
     // return 1;
     // }
 
diff --git a/client/cmdlfpresco.c b/client/cmdlfpresco.c
index a51155e78..3f1ce71f3 100644
--- a/client/cmdlfpresco.c
+++ b/client/cmdlfpresco.c
@@ -138,7 +138,7 @@ static int CmdPrescoClone(const char *Cmd) {
 
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
diff --git a/client/cmdlfpyramid.c b/client/cmdlfpyramid.c
index 91b895bcb..4e5588ec6 100644
--- a/client/cmdlfpyramid.c
+++ b/client/cmdlfpyramid.c
@@ -212,7 +212,7 @@ static int CmdPyramidClone(const char *Cmd) {
     cardnumber = (cn & 0x0000FFFF);
 
     if (getPyramidBits(facilitycode, cardnumber, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
@@ -249,7 +249,7 @@ static int CmdPyramidClone(const char *Cmd) {
 
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
@@ -272,7 +272,7 @@ static int CmdPyramidSim(const char *Cmd) {
     cardnumber = (cn & 0x0000FFFF);
 
     if (getPyramidBits(facilitycode, cardnumber, bs) != PM3_SUCCESS) {
-        PrintAndLogEx(WARNING, "Error with tag bitstream generation.");
+        PrintAndLogEx(ERR, "Error with tag bitstream generation.");
         return PM3_ESOFT;
     }
 
diff --git a/client/cmdlft55xx.c b/client/cmdlft55xx.c
index 923e50110..3d0871893 100644
--- a/client/cmdlft55xx.c
+++ b/client/cmdlft55xx.c
@@ -14,6 +14,13 @@
 
 #include "cmdlft55xx.h"
 
+// Some defines for readability
+#define T55xx_DLMode_Fixed         0 // Default Mode
+#define T55xx_DLMode_LLR           1 // Long Leading Reference
+#define T55xx_DLMode_Leading0      2 // Leading Zero
+#define T55xx_DLMode_1of4          3 // 1 of 4
+#define T55xx_LongLeadingReference 4 // Value to tell Write Bit to send long reference
+
 // Default configuration
 t55xx_conf_block_t config = { .modulation = DEMOD_ASK, .inverted = false, .offset = 0x00, .block0 = 0x00, .Q5 = false };
 
@@ -43,12 +50,14 @@ static int usage_t55xx_config() {
     return PM3_SUCCESS;
 }
 static int usage_t55xx_read() {
-    PrintAndLogEx(NORMAL, "Usage:  lf t55xx read [b <block>] [p <password>] <override_safety> <page1>");
+    PrintAndLogEx(NORMAL, "Usage:  lf t55xx read [r <mode>] b <block> [p <password>] <override_safety> <page1>");
     PrintAndLogEx(NORMAL, "Options:");
     PrintAndLogEx(NORMAL, "     b <block>    - block number to read. Between 0-7");
     PrintAndLogEx(NORMAL, "     p <password> - OPTIONAL password (8 hex characters)");
     PrintAndLogEx(NORMAL, "     o            - OPTIONAL override safety check");
     PrintAndLogEx(NORMAL, "     1            - OPTIONAL read Page 1 instead of Page 0");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default), '1' long leading ref.");
+    PrintAndLogEx(NORMAL, "                                      '2' leading zero,               '3' 1 of 4 coding ref.");
     PrintAndLogEx(NORMAL, "     ****WARNING****");
     PrintAndLogEx(NORMAL, "     Use of read with password on a tag not configured for a pwd");
     PrintAndLogEx(NORMAL, "     can damage the tag");
@@ -61,13 +70,15 @@ static int usage_t55xx_read() {
     return PM3_SUCCESS;
 }
 static int usage_t55xx_write() {
-    PrintAndLogEx(NORMAL, "Usage:  lf t55xx write [b <block>] [d <data>] [p <password>] [1] [t]");
+    PrintAndLogEx(NORMAL, "Usage:  lf t55xx write [r <mode>] b <block> d <data> [p <password>] [1] [t]");
     PrintAndLogEx(NORMAL, "Options:");
     PrintAndLogEx(NORMAL, "     b <block>    - block number to write. Between 0-7");
     PrintAndLogEx(NORMAL, "     d <data>     - 4 bytes of data to write (8 hex characters)");
     PrintAndLogEx(NORMAL, "     p <password> - OPTIONAL password 4bytes (8 hex characters)");
     PrintAndLogEx(NORMAL, "     1            - OPTIONAL write Page 1 instead of Page 0");
     PrintAndLogEx(NORMAL, "     t            - OPTIONAL test mode write - ****DANGER****");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default), '1' long leading ref.");
+    PrintAndLogEx(NORMAL, "                                      '2' leading zero,               '3' 1 of 4 coding ref.");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "      lf t55xx write b 3 d 11223344            - write 11223344 to block 3");
@@ -76,24 +87,29 @@ static int usage_t55xx_write() {
     return PM3_SUCCESS;
 }
 static int usage_t55xx_trace() {
-    PrintAndLogEx(NORMAL, "Usage:  lf t55xx trace [1]");
+    PrintAndLogEx(NORMAL, "Usage:  lf t55xx trace [r mode]");
     PrintAndLogEx(NORMAL, "Options:");
-    PrintAndLogEx(NORMAL, "     1            - if set, use Graphbuffer otherwise read data from tag.");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default), '1' long leading ref.");
+    PrintAndLogEx(NORMAL, "                                               '2' leading zero,               '3' 1 of 4 coding ref.");
+    // Command did not seem to support the 1 option (yet) so have removed the help lines
+    // PrintAndLogEx(NORMAL, "     1            - if set, use Graphbuffer otherwise read data from tag.");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "      lf t55xx trace");
-    PrintAndLogEx(NORMAL, "      lf t55xx trace 1");
+    // PrintAndLogEx(NORMAL, "      lf t55xx trace 1");
     PrintAndLogEx(NORMAL, "");
     return PM3_SUCCESS;
 }
 static int usage_t55xx_info() {
-    PrintAndLogEx(NORMAL, "Usage:  lf t55xx info [1] [d <data> [q]]");
+    PrintAndLogEx(NORMAL, "Usage:  lf t55xx info [1] [r <mode>] [d <data> [q]]");
     PrintAndLogEx(NORMAL, "Options:");
     PrintAndLogEx(NORMAL, "     (default)    - read data from tag.");
     PrintAndLogEx(NORMAL, "     1            - if set, use Graphbuffer instead of reading tag.");
     PrintAndLogEx(NORMAL, "     d <data>     - 4 bytes of data (8 hex characters)");
     PrintAndLogEx(NORMAL, "                    if set, use these data instead of reading tag.");
     PrintAndLogEx(NORMAL, "     q            - if set, provided data are interpreted as Q5 config.");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default), '1' long leading ref.");
+    PrintAndLogEx(NORMAL, "                                      '2' leading zero,               '3' 1 of 4 coding ref.");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "      lf t55xx info");
@@ -104,10 +120,12 @@ static int usage_t55xx_info() {
     return PM3_SUCCESS;
 }
 static int usage_t55xx_dump() {
-    PrintAndLogEx(NORMAL, "Usage:  lf t55xx dump <password> [o]");
+    PrintAndLogEx(NORMAL, "Usage:  lf t55xx dump [r <mode>] [<password> [o]]");
     PrintAndLogEx(NORMAL, "Options:");
     PrintAndLogEx(NORMAL, "     <password>   - OPTIONAL password 4bytes (8 hex symbols)");
     PrintAndLogEx(NORMAL, "     o            - OPTIONAL override, force pwd read despite danger to card");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default), '1' long leading ref.");
+    PrintAndLogEx(NORMAL, "                                      '2' leading zero,               '3' 1 of 4 coding ref.");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "      lf t55xx dump");
@@ -116,10 +134,13 @@ static int usage_t55xx_dump() {
     return PM3_SUCCESS;
 }
 static int usage_t55xx_detect() {
-    PrintAndLogEx(NORMAL, "Usage:  lf t55xx detect [1] [p <password>]");
+    PrintAndLogEx(NORMAL, "Usage:  lf t55xx detect [1] [r <mode>] [p <password>]");
     PrintAndLogEx(NORMAL, "Options:");
     PrintAndLogEx(NORMAL, "     1            - if set, use Graphbuffer otherwise read data from tag.");
-    PrintAndLogEx(NORMAL, "     p <password> - OPTIONAL password (8 hex characters)");
+    PrintAndLogEx(NORMAL, "     p <password  - OPTIONAL password (8 hex characters)");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default)");
+    PrintAndLogEx(NORMAL, "                      '1' long leading ref.,  '2' leading zero ");
+    PrintAndLogEx(NORMAL, "                      '3' 1 of 4 coding ref., '4' try all modes");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "      lf t55xx detect");
@@ -130,10 +151,13 @@ static int usage_t55xx_detect() {
 }
 static int usage_t55xx_detectP1() {
     PrintAndLogEx(NORMAL, "Command: Detect Page 1 of a t55xx chip");
-    PrintAndLogEx(NORMAL, "Usage:  lf t55xx p1detect [1] [p <password>]");
+    PrintAndLogEx(NORMAL, "Usage:  lf t55xx p1detect [1] [r <mode>] [p <password>]");
     PrintAndLogEx(NORMAL, "Options:");
     PrintAndLogEx(NORMAL, "     1            - if set, use Graphbuffer otherwise read data from tag.");
     PrintAndLogEx(NORMAL, "     p <password> - OPTIONAL password (8 hex characters)");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default)");
+    PrintAndLogEx(NORMAL, "                      '1' long leading ref.,  '2' leading zero ");
+    PrintAndLogEx(NORMAL, "                      '3' 1 of 4 coding ref., '4' try all modes");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "      lf t55xx p1detect");
@@ -143,11 +167,13 @@ static int usage_t55xx_detectP1() {
     return PM3_SUCCESS;
 }
 static int usage_t55xx_wakup() {
-    PrintAndLogEx(NORMAL, "Usage:  lf t55xx wakeup [h] p <password>");
-    PrintAndLogEx(NORMAL, "This commands send the Answer-On-Request command and leaves the readerfield ON afterwards.");
+    PrintAndLogEx(NORMAL, "Usage:  lf t55xx wakeup [h] [r <mode>] p <password>");
+    PrintAndLogEx(NORMAL, "This commands sends the Answer-On-Request command and leaves the readerfield ON afterwards.");
     PrintAndLogEx(NORMAL, "Options:");
     PrintAndLogEx(NORMAL, "     h            - this help");
     PrintAndLogEx(NORMAL, "     p <password> - password 4bytes (8 hex symbols)");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default), '1' long leading ref.");
+    PrintAndLogEx(NORMAL, "                                      '2' leading zero,               '3' 1 of 4 coding ref.");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "      lf t55xx wakeup p 11223344  - send wakeup password");
@@ -158,11 +184,14 @@ static int usage_t55xx_chk() {
     PrintAndLogEx(NORMAL, "press " _YELLOW_("'enter'") " to cancel the command");
     PrintAndLogEx(NORMAL, "WARNING: this may brick non-password protected chips!");
     PrintAndLogEx(NORMAL, "Try to reading block 7 before\n");
-    PrintAndLogEx(NORMAL, "Usage: lf t55xx chk [h] <m> [i <*.dic>]");
+    PrintAndLogEx(NORMAL, "Usage: lf t55xx chk [h] [m] [r <mode>] [i <*.dic>]");
     PrintAndLogEx(NORMAL, "Options:");
-    PrintAndLogEx(NORMAL, "     h           - this help");
-    PrintAndLogEx(NORMAL, "     m           - use dictionary from flashmemory\n");
-    PrintAndLogEx(NORMAL, "     i <*.dic>   - loads a default keys dictionary file <*.dic>");
+    PrintAndLogEx(NORMAL, "     h            - this help");
+    PrintAndLogEx(NORMAL, "     m            - use dictionary from flashmemory\n");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default)");
+    PrintAndLogEx(NORMAL, "                      '1' long leading ref.,  '2' leading zero ");
+    PrintAndLogEx(NORMAL, "                      '3' 1 of 4 coding ref., '4' try all modes");
+    PrintAndLogEx(NORMAL, "     i <*.dic>    - loads a default keys dictionary file <*.dic>");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "       lf t55xx chk m");
@@ -174,16 +203,19 @@ static int usage_t55xx_bruteforce() {
     PrintAndLogEx(NORMAL, "This command uses bruteforce to scan a number range");
     PrintAndLogEx(NORMAL, "press " _YELLOW_("'enter'") " to cancel the command");
     PrintAndLogEx(NORMAL, "WARNING: this may brick non-password protected chips!");
-    PrintAndLogEx(NORMAL, "Try to reading block 7 before\n");
-    PrintAndLogEx(NORMAL, "Usage: lf t55xx bruteforce [h] <start password> <end password>");
+    PrintAndLogEx(NORMAL, "Try reading block 7 before\n");
+    PrintAndLogEx(NORMAL, "Usage: lf t55xx bruteforce [h] [r <mode>] <start password> <end password>");
     PrintAndLogEx(NORMAL, "       password must be 4 bytes (8 hex symbols)");
     PrintAndLogEx(NORMAL, "Options:");
-    PrintAndLogEx(NORMAL, "     h           - this help");
-    PrintAndLogEx(NORMAL, "     <start_pwd> - 4 byte hex value to start pwd search at");
-    PrintAndLogEx(NORMAL, "     <end_pwd>   - 4 byte hex value to end pwd search at");
+    PrintAndLogEx(NORMAL, "     h            - this help");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default)");
+    PrintAndLogEx(NORMAL, "                      '1' long leading ref.,  '2' leading zero ");
+    PrintAndLogEx(NORMAL, "                      '3' 1 of 4 coding ref., '4' try all modes");
+    PrintAndLogEx(NORMAL, "     <start_pwd>  - 4 byte hex value to start pwd search at");
+    PrintAndLogEx(NORMAL, "     <end_pwd>    - 4 byte hex value to end pwd search at");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
-    PrintAndLogEx(NORMAL, "       lf t55xx bruteforce aaaaaa77 aaaaaa99");
+    PrintAndLogEx(NORMAL, "       lf t55xx bruteforce r 2 aaaaaa77 aaaaaa99");
     PrintAndLogEx(NORMAL, "");
     return PM3_SUCCESS;
 }
@@ -191,17 +223,20 @@ static int usage_t55xx_recoverpw() {
     PrintAndLogEx(NORMAL, "This command uses a few tricks to try to recover mangled password");
     PrintAndLogEx(NORMAL, "press " _YELLOW_("'enter'") " to cancel the command");
     PrintAndLogEx(NORMAL, "WARNING: this may brick non-password protected chips!");
-    PrintAndLogEx(NORMAL, "Try to reading block 7 before\n");
-    PrintAndLogEx(NORMAL, "Usage: lf t55xx recoverpw [password]");
+    PrintAndLogEx(NORMAL, "Try reading block 7 before\n");
+    PrintAndLogEx(NORMAL, "Usage: lf t55xx recoverpw [r <mode>] [password]");
     PrintAndLogEx(NORMAL, "       password must be 4 bytes (8 hex symbols)");
     PrintAndLogEx(NORMAL, "       default password is 51243648, used by many cloners");
     PrintAndLogEx(NORMAL, "Options:");
-    PrintAndLogEx(NORMAL, "     h           - this help");
-    PrintAndLogEx(NORMAL, "     [password]  - 4 byte hex value of password written by cloner");
+    PrintAndLogEx(NORMAL, "     h            - this help");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default)");
+    PrintAndLogEx(NORMAL, "                       '1' long leading ref.,  '2' leading zero ");
+    PrintAndLogEx(NORMAL, "                       '3' 1 of 4 coding ref., '4' try all modes");
+    PrintAndLogEx(NORMAL, "     [password]   - 4 byte hex value of password written by cloner");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "       lf t55xx recoverpw");
-    PrintAndLogEx(NORMAL, "       lf t55xx recoverpw 51243648");
+    PrintAndLogEx(NORMAL, "       lf t55xx r 3 recoverpw 51243648");
     PrintAndLogEx(NORMAL, "");
     return PM3_SUCCESS;
 }
@@ -218,16 +253,21 @@ static int usage_t55xx_wipe() {
     return PM3_SUCCESS;
 }
 static int usage_lf_deviceconfig() {
-    PrintAndLogEx(NORMAL, "Sets t55x7 timings for direkt commands. The timings are set here in Field Clocks (FC), \nwhich is converted to (US) on device");
-    PrintAndLogEx(NORMAL, "Usage: lf t55xx deviceconfig a <gap> b <gap> c <gap> d <gap> e <gap> p");
+    PrintAndLogEx(NORMAL, "Sets t55x7 timings for direct commands. The timings are set here in Field Clocks (FC), \nwhich is converted to (US) on device");
+    PrintAndLogEx(NORMAL, "Usage: lf t55xx deviceconfig [r <mode>] a <gap> b <gap> c <gap> d <gap> e <gap> f <gap> g <gap> [p]");
     PrintAndLogEx(NORMAL, "Options:");
-    PrintAndLogEx(NORMAL, "       h              - This help");
-    PrintAndLogEx(NORMAL, "       a <8..255>     - Set start gap");
-    PrintAndLogEx(NORMAL, "       b <8..255>     - Set write gap");
-    PrintAndLogEx(NORMAL, "       c <8..255>     - Set write ZERO gap");
-    PrintAndLogEx(NORMAL, "       d <8..255>     - Set write ONE gap");
-    PrintAndLogEx(NORMAL, "       e <8..255>     - Set read gap");
-    PrintAndLogEx(NORMAL, "       p              - persist to flashmemory");
+    PrintAndLogEx(NORMAL, "     h            - This help");
+    PrintAndLogEx(NORMAL, "     a <8..255>   - Set start gap");
+    PrintAndLogEx(NORMAL, "     b <8..255>   - Set write gap");
+    PrintAndLogEx(NORMAL, "     c <8..255>   - Set write ZERO gap");
+    PrintAndLogEx(NORMAL, "     d <8..255>   - Set write ONE gap");
+    PrintAndLogEx(NORMAL, "     e <8..255>   - Set read gap");
+    PrintAndLogEx(NORMAL, "     f <8..255>   - Set write TWO gap (1 of 4 only)");
+    PrintAndLogEx(NORMAL, "     g <8..255>   - Set write THREE gap (1 of 4 only)");
+    PrintAndLogEx(NORMAL, "     p            - persist to flashmemory");
+    PrintAndLogEx(NORMAL, "     r <mode>     - downlink encoding '0' fixed bit length (default), '1' long leading ref.");
+    PrintAndLogEx(NORMAL, "                                      '2' leading zero,               '3' 1 of 4 coding ref.");
+    PrintAndLogEx(NORMAL, "     z            - Set default t55x7 timings (use p to save if required)");
     PrintAndLogEx(NORMAL, "");
     PrintAndLogEx(NORMAL, "Examples:");
     PrintAndLogEx(NORMAL, "      lf t55xx deviceconfig a 29 b 17 c 15 d 47 e 15   - default T55XX");
@@ -359,13 +399,12 @@ static int CmdT55xxSetConfig(const char *Cmd) {
     return printConfiguration(config);
 }
 
-int T55xxReadBlock(uint8_t block, bool page1, bool usepwd, bool override, uint32_t password) {
+int T55xxReadBlock(uint8_t block, bool page1, bool usepwd, uint8_t override, uint32_t password, uint8_t downlink_mode) {
     //Password mode
     if (usepwd) {
         // try reading the config block and verify that PWD bit is set before doing this!
         if (!override) {
-
-            if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, false, 0)) return PM3_ESOFT;
+            if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, false, 0, downlink_mode)) return PM3_ESOFT;
 
             if (!tryDetectModulation()) {
                 PrintAndLogEx(NORMAL, "Safety Check: Could not detect if PWD bit is set in config block. Exits.");
@@ -376,11 +415,14 @@ int T55xxReadBlock(uint8_t block, bool page1, bool usepwd, bool override, uint32
                 page1 = false;
             }
         } else {
-            PrintAndLogEx(NORMAL, "Safety Check Overriden - proceeding despite risk");
+            // Show only if first for command i.e. override = 1 (override and display) override = 2 (override and dont display)
+            if ((override & 2) != 2)
+                PrintAndLogEx(NORMAL, "Safety Check Overriden - proceeding despite risk");
         }
     }
 
-    if (!AquireData(page1, block, usepwd, password)) return PM3_ESOFT;
+
+    if (!AquireData(page1, block, usepwd, password, downlink_mode)) return PM3_ESOFT;
     if (!DecodeT55xxBlock()) return PM3_ESOFT;
 
     printT55xxBlock(block);
@@ -388,13 +430,15 @@ int T55xxReadBlock(uint8_t block, bool page1, bool usepwd, bool override, uint32
 }
 
 static int CmdT55xxReadBlock(const char *Cmd) {
-    uint8_t block = REGULAR_READ_MODE_BLOCK;
-    uint32_t password = 0; //default to blank Block 7
-    bool usepwd = false;
-    bool override = false;
-    bool page1 = false;
-    bool errors = false;
-    uint8_t cmdp = 0;
+    uint8_t  block         = REGULAR_READ_MODE_BLOCK;
+    uint32_t password      = 0; //default to blank Block 7
+    bool     usepwd        = false;
+    bool     override      = false;
+    bool     page1         = false;
+    bool     errors        = false;
+    uint8_t  cmdp          = 0;
+    uint8_t  downlink_mode = 0;
+
     while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
         switch (tolower(param_getchar(Cmd, cmdp))) {
             case 'h':
@@ -416,6 +460,13 @@ static int CmdT55xxReadBlock(const char *Cmd) {
                 page1 = true;
                 cmdp++;
                 break;
+            case 'r':
+            case 'R':
+                downlink_mode = param_getchar(Cmd, cmdp + 1) - '0';
+                if (downlink_mode > 3) downlink_mode = 0;
+                cmdp += 2;
+                break;
+
             default:
                 PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
                 errors = true;
@@ -430,7 +481,7 @@ static int CmdT55xxReadBlock(const char *Cmd) {
     }
 
     printT5xxHeader(page1);
-    return T55xxReadBlock(block, page1, usepwd, override, password);
+    return T55xxReadBlock(block, page1, usepwd, override, password, downlink_mode);
 }
 
 bool DecodeT55xxBlock(void) {
@@ -512,11 +563,42 @@ static int SanityOfflineCheck(bool useGraphBuffer) {
     return PM3_SUCCESS;
 }
 
+void T55xx_Print_DownlinkMode(uint8_t downlink_mode) {
+    char Msg[80];
+    sprintf(Msg, "Downlink Mode used : ");
+
+    switch (downlink_mode) {
+        case  0 :
+            strcat(Msg, "default/fixed bit length");
+            break;
+        case  1 :
+            strcat(Msg, "long leading reference (r 1)");
+            break;
+        case  2 :
+            strcat(Msg, "leading zero reference (r 2)");
+            break;
+        case  3 :
+            strcat(Msg, "1 of 4 coding reference (r 3)");
+            break;
+        default :
+            strcat(Msg, "default/fixed bit length");
+            break;
+    }
+
+    PrintAndLogEx(NORMAL, Msg);
+}
+//
 static int CmdT55xxDetect(const char *Cmd) {
-    bool errors = false;
-    bool useGB = false, usepwd = false;
-    uint32_t password = 0;
-    uint8_t cmdp = 0;
+
+    bool     errors           = false;
+    bool     useGB            = false;
+    bool     usepwd           = false;
+    bool     try_all_dl_modes = false;
+    bool     found            = false;
+    uint32_t password         = 0;
+    uint8_t  cmdp             = 0;
+    uint8_t  downlink_mode    = 0;
+    uint8_t  dl_mode          = 0;
 
     while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
         switch (tolower(param_getchar(Cmd, cmdp))) {
@@ -532,6 +614,12 @@ static int CmdT55xxDetect(const char *Cmd) {
                 useGB = true;
                 cmdp++;
                 break;
+            case 'r':
+                downlink_mode = param_getchar(Cmd, cmdp + 1) - '0';
+                if (downlink_mode == 4) try_all_dl_modes = true;
+                if (downlink_mode > 3) downlink_mode = 0;
+                cmdp += 2;
+                break;
             default:
                 PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
                 errors = true;
@@ -544,16 +632,47 @@ static int CmdT55xxDetect(const char *Cmd) {
     if (SanityOfflineCheck(useGB) != PM3_SUCCESS) return PM3_ENODATA;
 
     if (!useGB) {
-        if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, usepwd, password))
-            return PM3_ENODATA;
+        for (dl_mode = downlink_mode; dl_mode < 4; dl_mode++) {
+            found = AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, usepwd, password, dl_mode);
+
+            // found = false if password is supplied but wrong d/l mode
+            // so keep trying other modes (if requested)
+            /*
+            if (!found) {
+                printf ("Aquire not found");
+                return PM3_ENODATA;
+            }
+            */
+            if (tryDetectModulation()) {
+                T55xx_Print_DownlinkMode(dl_mode);
+                dl_mode = 4;
+                found = true;
+            } else found = false;
+
+            if (!try_all_dl_modes) dl_mode = 4;
+        }
     }
 
+
+    if (useGB) found = tryDetectModulation();
+
+    if (!found)
+        PrintAndLogEx(WARNING, "Could not detect modulation automatically. Try setting it manually with " _YELLOW_("\'lf t55xx config\'"));
+
+
+    /*
+    if (!useGB) {
+        if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, usepwd, password,downlink_mode))
+            return PM3_ENODATA;
+    }
     if (!tryDetectModulation())
         PrintAndLogEx(WARNING, "Could not detect modulation automatically. Try setting it manually with " _YELLOW_("\'lf t55xx config\'"));
+    else
+        T55xx_Print_DownlinkMode (downlink_mode);
+    */
 
     return PM3_SUCCESS;
 }
-
 // detect configuration?
 bool tryDetectModulation(void) {
 
@@ -997,9 +1116,13 @@ int printConfiguration(t55xx_conf_block_t b) {
 }
 
 static int CmdT55xxWakeUp(const char *Cmd) {
-    uint32_t password = 0;
-    uint8_t cmdp = 0;
-    bool errors = false;
+
+    uint32_t password      = 0;
+    uint8_t  cmdp          = 0;
+    bool     errors        = false;
+    uint8_t  downlink_mode = 0;
+    uint8_t  flags         = 0;
+
     while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
         switch (tolower(param_getchar(Cmd, cmdp))) {
             case 'h':
@@ -1009,30 +1132,40 @@ static int CmdT55xxWakeUp(const char *Cmd) {
                 cmdp += 2;
                 errors = false;
                 break;
+            case 'r':
+                downlink_mode = param_getchar(Cmd, cmdp + 1) - '0';
+                if (downlink_mode > 3) downlink_mode = 0;
+                cmdp += 2;
+                break;
             default:
                 PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
                 errors = true;
                 break;
         }
     }
+
     if (errors) return usage_t55xx_wakup();
 
+    flags = (downlink_mode & 3) << 3;
     clearCommandBuffer();
-    SendCommandMIX(CMD_T55XX_WAKEUP, password, 0, 0, NULL, 0);
+    SendCommandMIX(CMD_T55XX_WAKEUP, password, flags, 0, NULL, 0);
     PrintAndLogEx(SUCCESS, "Wake up command sent. Try read now");
+
     return PM3_SUCCESS;
 }
 
 static int CmdT55xxWriteBlock(const char *Cmd) {
-    uint8_t block = 0xFF; //default to invalid block
-    uint32_t data = 0; //default to blank Block
-    uint32_t password = 0; //default to blank Block 7
-    bool usepwd = false;
-    bool page1 = false;
-    bool gotdata = false;
-    bool testMode = false;
-    bool errors = false;
-    uint8_t cmdp = 0;
+    uint8_t  block         = 0xFF; //default to invalid block
+    uint32_t data          = 0;    //default to blank Block
+    uint32_t password      = 0;    //default to blank Block 7
+    bool     usepwd        = false;
+    bool     page1         = false;
+    bool     gotdata       = false;
+    bool     testMode      = false;
+    bool     errors        = false;
+    uint8_t  cmdp          = 0;
+    uint32_t downlink_mode = 0;
+
     while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
         switch (tolower(param_getchar(Cmd, cmdp))) {
             case 'h':
@@ -1064,6 +1197,11 @@ static int CmdT55xxWriteBlock(const char *Cmd) {
                 page1 = true;
                 cmdp++;
                 break;
+            case 'r':
+                downlink_mode = param_getchar(Cmd, cmdp + 1) - '0';
+                if (downlink_mode > 3) downlink_mode = 0;
+                cmdp += 2;
+                break;
             default:
                 PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
                 errors = true;
@@ -1074,9 +1212,10 @@ static int CmdT55xxWriteBlock(const char *Cmd) {
 
     PacketResponseNG resp;
     uint8_t flags;
-    flags = (usepwd) ? 0x1 : 0;
-    flags |= (page1) ? 0x2 : 0;
+    flags  = (usepwd)   ? 0x1 : 0;
+    flags |= (page1)    ? 0x2 : 0;
     flags |= (testMode) ? 0x4 : 0;
+    flags |= (downlink_mode << 3);
 
     char pwdStr[16] = {0};
     snprintf(pwdStr, sizeof(pwdStr), "pwd: 0x%08X", password);
@@ -1096,30 +1235,40 @@ static int CmdT55xxWriteBlock(const char *Cmd) {
        uses struct in pm3_cmd.h
     */
     t55xx_write_block_t ng;
-    ng.data = data;
-    ng.pwd = password;
+    ng.data    = data;
+    ng.pwd     = password;
     ng.blockno = block;
-    ng.flags = flags;
+    ng.flags   = flags;
 
     SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
     if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, 2000)) {
-        PrintAndLogEx(WARNING, "Error occurred, device did not ACK write operation. (May be due to old firmware)");
+        PrintAndLogEx(ERR, "Error occurred, device did not ACK write operation. (May be due to old firmware)");
         return PM3_ETIMEOUT;
     }
     return PM3_SUCCESS;
 }
 
 static int CmdT55xxReadTrace(const char *Cmd) {
-    char cmdp = tolower(param_getchar(Cmd, 0));
-    if (strlen(Cmd) > 1 || cmdp == 'h') return usage_t55xx_trace();
+    uint8_t cmd_len       = 0;
+    uint8_t downlink_mode = 0;
 
-    if (strlen(Cmd) == 0) {
+    char cmdp = tolower(param_getchar(Cmd, 0));
+    if (cmdp == 'r') {
+        downlink_mode = param_getchar(Cmd, 1) - '0';
+        if (downlink_mode > 3) downlink_mode = 0;
+        cmd_len = 3;
+    }
+    if ((strlen(Cmd) != cmd_len) || (cmdp == 'h')) return usage_t55xx_trace();
+
+    if (strlen(Cmd) == cmd_len) {
         // sanity check.
         if (SanityOfflineCheck(false) != PM3_SUCCESS) return PM3_ENODATA;
 
         bool pwdmode = false;
         uint32_t password = 0;
-        if (!AquireData(T55x7_PAGE1, T55x7_TRACE_BLOCK1, pwdmode, password))
+//        REGULAR_READ_MODE_BLOCK - yeilds correct Page 1 Block 2 data i.e. + 32 bit offset.
+//        if (!AquireData(T55x7_PAGE1, T55x7_TRACE_BLOCK1, pwdmode, password,downlink_mode))
+        if (!AquireData(T55x7_PAGE1, REGULAR_READ_MODE_BLOCK, pwdmode, password, downlink_mode))
             return PM3_ENODATA;
     }
 
@@ -1361,9 +1510,10 @@ static int CmdT55xxInfo(const char *Cmd) {
         Normal mode
         Extended mode
     */
-    bool frombuff = false, gotdata = false, dataasq5 = false;
-    uint8_t cmdp = 0;
-    uint32_t block0 = 0;
+    bool     frombuff      = false, gotdata = false, dataasq5 = false;
+    uint8_t  cmdp          = 0;
+    uint8_t  downlink_mode = 0;
+    uint32_t block0        = 0;
 
     while (param_getchar(Cmd, cmdp) != 0x00) {
         switch (tolower(param_getchar(Cmd, cmdp))) {
@@ -1382,6 +1532,11 @@ static int CmdT55xxInfo(const char *Cmd) {
                 dataasq5 = true;
                 cmdp += 2;
                 break;
+            case 'r':
+                downlink_mode = param_getchar(Cmd, cmdp + 1) - '0';
+                if (downlink_mode > 3) downlink_mode = 0;
+                cmdp += 2;
+                break;
             default:
                 PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
                 return usage_t55xx_info();
@@ -1400,7 +1555,7 @@ static int CmdT55xxInfo(const char *Cmd) {
 
         bool pwdmode = false;
         uint32_t password = 0;
-        if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, pwdmode, password))
+        if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, pwdmode, password, downlink_mode))
             return PM3_ENODATA;
     }
     if (!gotdata) {
@@ -1495,30 +1650,43 @@ static int CmdT55xxInfo(const char *Cmd) {
 
 static int CmdT55xxDump(const char *Cmd) {
 
-    uint32_t password = 0;
-    bool override = false;
-    char cmdp = tolower(param_getchar(Cmd, 0));
-    if (cmdp == 'h') return usage_t55xx_dump();
+    uint32_t password      = 0;
+    uint8_t  override      = false;
+    uint8_t  cmd_opt_idx   = 0;
+    uint8_t  downlink_mode = 0;
+    uint8_t  pwd_offset    = 0;
+    char     cmdp = tolower(param_getchar(Cmd, 0));
 
-    bool usepwd = (strlen(Cmd) > 0);
+
+    if (cmdp == 'h') return usage_t55xx_dump();
+    if (cmdp == 'r') {
+        cmd_opt_idx++;
+        downlink_mode = param_getchar(Cmd, cmd_opt_idx++) - '0';
+        if (downlink_mode > 3) downlink_mode = 0;
+        pwd_offset = 3;
+    }
+    bool usepwd = (strlen(Cmd) > pwd_offset);
     if (usepwd) {
-        password = param_get32ex(Cmd, 0, 0, 16);
-        if (param_getchar(Cmd, 1) == 'o')
+        password = param_get32ex(Cmd, cmd_opt_idx++, 0, 16);
+        if (param_getchar(Cmd, cmd_opt_idx++) == 'o')
             override = true;
     }
 
     printT5xxHeader(0);
-    for (uint8_t i = 0; i < 8; ++i)
-        T55xxReadBlock(i, 0, usepwd, override, password);
-
+    for (uint8_t i = 0; i < 8; ++i) {
+        T55xxReadBlock(i, 0, usepwd, override, password, downlink_mode);
+        // idea for better user experience and display.
+        // only show override warning on the first block read
+        if (override) override |= 2; // flag not to show safty for 2nd and on.
+    }
     printT5xxHeader(1);
     for (uint8_t i = 0; i < 4; i++)
-        T55xxReadBlock(i, 1, usepwd, override, password);
+        T55xxReadBlock(i, 1, usepwd, override, password, downlink_mode);
 
     return PM3_SUCCESS;
 }
 
-bool AquireData(uint8_t page, uint8_t block, bool pwdmode, uint32_t password) {
+bool AquireData(uint8_t page, uint8_t block, bool pwdmode, uint32_t password, uint8_t downlink_mode) {
     // arg0 bitmodes:
     //  b0 = pwdmode
     //  b1 = page to read from
@@ -1527,15 +1695,17 @@ bool AquireData(uint8_t page, uint8_t block, bool pwdmode, uint32_t password) {
     // arg2: password
     struct p {
         uint32_t password;
-        uint8_t blockno;
-        uint8_t page;
-        bool pwdmode;
+        uint8_t  blockno;
+        uint8_t  page;
+        bool     pwdmode;
+        uint8_t  downlink_mode;
     } PACKED;
     struct p payload;
-    payload.password = password;
-    payload.blockno = block;
-    payload.page = page & 0x1;
-    payload.pwdmode = pwdmode;
+    payload.password      = password;
+    payload.blockno       = block;
+    payload.page          = page & 0x1;
+    payload.pwdmode       = pwdmode;
+    payload.downlink_mode = downlink_mode;
 
     clearCommandBuffer();
     SendCommandNG(CMD_T55XX_READ_BLOCK, (uint8_t *)&payload, sizeof(payload));
@@ -1795,16 +1965,27 @@ static void t55x7_create_config_block(int tagtype) {
 */
 
 static int CmdResetRead(const char *Cmd) {
-    (void)Cmd; // Cmd is not used so far
+
+    uint8_t downlink_mode = 0;
+    uint8_t flags         = 0;
+
+
+    if (strlen(Cmd) == 3)
+        downlink_mode = param_getchar(Cmd, 1) - '0';
+
+    if (downlink_mode > 3) downlink_mode = 0;
+
+    printf("DL : %d\n", downlink_mode);
+    flags = downlink_mode << 3;
     clearCommandBuffer();
-    SendCommandNG(CMD_T55XX_RESET_READ, NULL, 0);
+    SendCommandNG(CMD_T55XX_RESET_READ, &flags, sizeof(flags));
     if (!WaitForResponseTimeout(CMD_ACK, NULL, 2500)) {
         PrintAndLogEx(WARNING, "command execution time out");
         return PM3_ETIMEOUT;
     }
 
     uint8_t got[BIGBUF_SIZE - 1];
-    if (!GetFromDevice(BIG_BUF, got, sizeof(got), 0, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF, got, sizeof(got), 0, NULL, 0, NULL, 2500, false)) {
         PrintAndLogEx(WARNING, "command execution time out");
         return PM3_ETIMEOUT;
     }
@@ -1829,13 +2010,13 @@ static int CmdT55xxWipe(const char *Cmd) {
     else
         snprintf(ptrData, sizeof(writeData), "b 0 d 000880E0 p 0");
 
-    if (CmdT55xxWriteBlock(ptrData) != PM3_SUCCESS) PrintAndLogEx(WARNING, "Error writing blk 0");
+    if (CmdT55xxWriteBlock(ptrData) != PM3_SUCCESS) PrintAndLogEx(WARNING, "Warning: error writing blk 0");
 
     for (uint8_t blk = 1; blk < 8; blk++) {
 
         snprintf(ptrData, sizeof(writeData), "b %d d 0", blk);
 
-        if (CmdT55xxWriteBlock(ptrData) != PM3_SUCCESS) PrintAndLogEx(WARNING, "Error writing blk %d", blk);
+        if (CmdT55xxWriteBlock(ptrData) != PM3_SUCCESS) PrintAndLogEx(WARNING, "Warning: error writing blk %d", blk);
 
         memset(writeData, 0x00, sizeof(writeData));
     }
@@ -1843,9 +2024,7 @@ static int CmdT55xxWipe(const char *Cmd) {
 }
 
 static bool IsCancelled(void) {
-    if (ukbhit()) {
-        int gc = getchar();
-        (void)gc;
+    if (kbd_enter_pressed()) {
         PrintAndLogEx(WARNING, "\naborted via keyboard!\n");
         return true;
     }
@@ -1855,13 +2034,44 @@ static bool IsCancelled(void) {
 // load a default pwd file.
 static int CmdT55xxChkPwds(const char *Cmd) {
 
-    char filename[FILE_PATH_SIZE] = {0};
-    bool found = false;
+    char    filename[FILE_PATH_SIZE] = {0};
+    bool    found = false;
     uint8_t timeout = 0;
     uint8_t *keyBlock = NULL;
+    bool    from_flash = false;
+    bool    try_all_dl_modes = false;
+    uint8_t downlink_mode = 0;
+    int     len;
+    char    cmdp;
+    bool    use_pwd_file = false;
+    int     dl_mode; // to try each downlink mode for each password
+
+
+    cmdp = tolower(param_getchar(Cmd, 0));
+
+    if (cmdp == 'h') return usage_t55xx_chk();
+    if (cmdp == 'm') {
+        from_flash = true;
+        Cmd += 2;
+        cmdp = tolower(param_getchar(Cmd, 0));
+    }
+    if (cmdp == 'r') {
+        Cmd += 2;
+        downlink_mode = param_getchar(Cmd, 0) - '0'; // get 2nd option, as this is fixed order.
+        if (downlink_mode == 4) try_all_dl_modes = true;
+        if (downlink_mode > 3) downlink_mode = 0;
+        Cmd += 2;
+        cmdp = param_getchar(Cmd, 0);
+    }
+    if (cmdp == 'i') {
+        Cmd += 2;
+        len = strlen(Cmd);
+        if (len > FILE_PATH_SIZE) len = FILE_PATH_SIZE;
+        memcpy(filename, Cmd, len);
+        use_pwd_file = true;
+    }
+
 
-    char cmdp = tolower(param_getchar(Cmd, 0));
-    if (strlen(Cmd) == 0 || cmdp == 'h') return usage_t55xx_chk();
 
     /*
     // block 7,  page1 = false, usepwd = false, override = false, pwd = 00000000
@@ -1874,10 +2084,11 @@ static int CmdT55xxChkPwds(const char *Cmd) {
     */
 
     uint64_t t1 = msclock();
+    uint8_t flags = downlink_mode << 3;
 
-    if (cmdp == 'm') {
+    if (from_flash) {
         clearCommandBuffer();
-        SendCommandNG(CMD_T55XX_CHKPWDS, NULL, 0);
+        SendCommandNG(CMD_T55XX_CHKPWDS, &flags, sizeof(flags));
         PacketResponseNG resp;
 
         while (!WaitForResponseTimeout(CMD_ACK, &resp, 2000)) {
@@ -1893,10 +2104,12 @@ static int CmdT55xxChkPwds(const char *Cmd) {
         if (resp.oldarg[0]) {
             PrintAndLogEx(SUCCESS, "\nFound a candidate [ " _YELLOW_("%08X") " ]. Trying to validate", resp.oldarg[1]);
 
-            if (AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, true, resp.oldarg[1])) {
+            if (AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, true, resp.oldarg[1], downlink_mode)) {
                 found = tryDetectModulation();
                 if (found) {
-                    PrintAndLogEx(SUCCESS, "Found valid password: [ " _GREEN_("%08") " ]", resp.oldarg[1]);
+                    PrintAndLogEx(SUCCESS, "Found valid password: [ " _GREEN_("%08X") " ]", resp.oldarg[1]);
+                    T55xx_Print_DownlinkMode(downlink_mode);
+
                 } else {
                     PrintAndLogEx(WARNING, "Check pwd failed");
                 }
@@ -1909,19 +2122,14 @@ static int CmdT55xxChkPwds(const char *Cmd) {
         goto out;
     }
 
-    if (cmdp == 'i') {
-
-        int len = strlen(Cmd + 2);
-        if (len > FILE_PATH_SIZE) len = FILE_PATH_SIZE;
-        memcpy(filename, Cmd + 2, len);
-
+    if (use_pwd_file) {
         uint16_t keycount = 0;
         size_t datalen = 0;
 
         // TODO, a way of reallocating memory if file was larger
         keyBlock = calloc(4 * 200, sizeof(uint8_t));
         if (keyBlock == NULL) {
-            PrintAndLogDevice(WARNING, "error, cannot allocate memory ");
+            PrintAndLogDevice(ERR, "error, cannot allocate memory ");
             return PM3_ESOFT;
         }
 
@@ -1950,19 +2158,25 @@ static int CmdT55xxChkPwds(const char *Cmd) {
             curr_password = bytes_to_num(keyBlock + 4 * c, 4);
 
             PrintAndLogEx(INFO, "Testing %08X", curr_password);
+            for (dl_mode = downlink_mode; dl_mode <= 3; dl_mode++) {
 
-            if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, true, curr_password)) {
-                continue;
+                if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, true, curr_password, dl_mode)) {
+                    continue;
+                }
+
+                found = tryDetectModulation();
+                if (found) {
+                    PrintAndLogEx(SUCCESS, "Found valid password: [ " _GREEN_("%08X") " ]", curr_password);
+                    T55xx_Print_DownlinkMode(dl_mode);
+                    dl_mode = 4; // Exit other downlink mode checks
+                    c = keycount; // Exit loop
+                }
+
+                if (!try_all_dl_modes) // Exit loop if not trying all downlink modes
+                    dl_mode = 4;
             }
-
-            found = tryDetectModulation();
-            if (found)
-                break;
         }
-        if (found)
-            PrintAndLogEx(SUCCESS, "Found valid password: [ " _GREEN_("%08X") " ]", curr_password);
-        else
-            PrintAndLogEx(WARNING, "Check pwd failed");
+        if (!found) PrintAndLogEx(WARNING, "Check pwd failed");
     }
 
     free(keyBlock);
@@ -1978,16 +2192,25 @@ static int CmdT55xxBruteForce(const char *Cmd) {
 
     uint32_t start_password = 0x00000000; //start password
     uint32_t end_password   = 0xFFFFFFFF; //end   password
-    uint32_t curr = 0;
-    bool found = false;
+    uint32_t curr           = 0;
+    uint8_t  downlink_mode  = 0;
+    uint8_t  cmd_opt_idx    = 0;
+    uint8_t  found          = 0; // > 0 if found xx1 xx downlink needed, 1 found
+
+    char cmdp = tolower(param_getchar(Cmd, cmd_opt_idx));
 
-    char cmdp = tolower(param_getchar(Cmd, 0));
     if (cmdp == 'h') return usage_t55xx_bruteforce();
+    if (cmdp == 'r') { // downlink mode supplied
+        cmd_opt_idx++;   // skip over 'r'
+        downlink_mode = param_getchar(Cmd, cmd_opt_idx++) - '0';
+        if (downlink_mode > 4) downlink_mode = 0;
+    }
+
 
     uint64_t t1 = msclock();
 
-    start_password = param_get32ex(Cmd, 0, 0, 16);
-    end_password = param_get32ex(Cmd, 1, 0, 16);
+    start_password = param_get32ex(Cmd, cmd_opt_idx++, 0, 16);
+    end_password   = param_get32ex(Cmd, cmd_opt_idx++, 0, 16);
 
     curr = start_password;
 
@@ -1997,7 +2220,7 @@ static int CmdT55xxBruteForce(const char *Cmd) {
 
     PrintAndLogEx(INFO, "Search password range [%08X -> %08X]", start_password, end_password);
 
-    while (!found) {
+    while (found == 0) {
 
         printf(".");
         fflush(stdout);
@@ -2006,7 +2229,7 @@ static int CmdT55xxBruteForce(const char *Cmd) {
             return PM3_EOPABORTED;
         }
 
-        found = tryOnePassword(curr);
+        found = tryOnePassword(curr, downlink_mode);
 
         if (curr == end_password)
             break;
@@ -2016,47 +2239,70 @@ static int CmdT55xxBruteForce(const char *Cmd) {
 
     PrintAndLogEx(NORMAL, "");
 
-    if (found)
-        PrintAndLogEx(SUCCESS, "Found valid password: [ " _GREEN_("%08X") " ]", curr);
-    else
-        PrintAndLogEx(WARNING, "Bruteforce failed, last tried: [ " _YELLOW_("%08X") " ]", --curr);
+    if (found) {
+        PrintAndLogEx(SUCCESS, "Found valid password: [ " _GREEN_("%08X") "]", curr - 1);
+        T55xx_Print_DownlinkMode((found >> 1) & 3);
+    } else
+        PrintAndLogEx(WARNING, "Bruteforce failed, last tried: [ " _YELLOW_("%08X") " ]", curr);
 
     t1 = msclock() - t1;
     PrintAndLogEx(SUCCESS, "\nTime in bruteforce: %.0f seconds\n", (float)t1 / 1000.0);
     return PM3_SUCCESS;
 }
 
-int tryOnePassword(uint32_t password) {
+uint8_t tryOnePassword(uint32_t password, uint8_t downlink_mode) {
+
+    bool  try_all_dl_modes = false;
+    uint8_t dl_mode          = 0;
+
     PrintAndLogEx(INFO, "Trying password %08X", password);
 
-    AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, true, password);
+    if (downlink_mode == 4) try_all_dl_modes = true;
 
-    if (getSignalProperties()->isnoise == false)
-        return 0;
+    downlink_mode = (downlink_mode & 3); // ensure 0-3
 
-    if (tryDetectModulation())
-        return 1;
-    else
-        return 0;
+    // check if dl mode 4 and loop if needed
+    for (dl_mode = downlink_mode; dl_mode < 4; dl_mode++) {
+
+        AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, true, password, dl_mode);
+
+        //  if (getSignalProperties()->isnoise == false) {
+        //  } else {
+        if (tryDetectModulation()) {
+            return 1 + (dl_mode << 1);
+        }
+        //  }
+        if (!try_all_dl_modes) dl_mode = 4;
+    }
+    return 0;
 }
 
 static int CmdT55xxRecoverPW(const char *Cmd) {
-    int bit = 0;
+    int      bit           = 0;
     uint32_t orig_password = 0x0;
     uint32_t curr_password = 0x0;
     uint32_t prev_password = 0xffffffff;
-    uint32_t mask = 0x0;
-    int found = 0;
-    char cmdp = tolower(param_getchar(Cmd, 0));
-    if (cmdp == 'h') return usage_t55xx_recoverpw();
+    uint32_t mask          = 0x0;
+    uint8_t  downlink_mode = 0;
+    uint8_t  found         = 0;
+    uint8_t  cmd_opt_idx   = 0;
 
-    orig_password = param_get32ex(Cmd, 0, 0x51243648, 16); //password used by handheld cloners
+    char     cmdp = tolower(param_getchar(Cmd, cmd_opt_idx));
+
+    if (cmdp == 'h') return usage_t55xx_recoverpw();
+    if (cmdp == 'r') { // downlink mode supplied
+        cmd_opt_idx++; // skip over 'r'
+        downlink_mode = param_getchar(Cmd, cmd_opt_idx++) - '0';
+        if (downlink_mode > 4) downlink_mode = 0;
+    }
+
+    orig_password = param_get32ex(Cmd, cmd_opt_idx++, 0x51243648, 16); //password used by handheld cloners
 
     // first try fliping each bit in the expected password
     while (bit < 32) {
         curr_password = orig_password ^ (1u << bit);
-        found = tryOnePassword(curr_password);
-        if (found == 1)
+        found = tryOnePassword(curr_password, downlink_mode);
+        if (found > 0) // xx1 for found xx = dl mode used
             goto out;
 
         bit++;
@@ -2080,8 +2326,8 @@ static int CmdT55xxRecoverPW(const char *Cmd) {
             continue;
         }
 
-        found = tryOnePassword(curr_password);
-        if (found == 1)
+        found = tryOnePassword(curr_password, downlink_mode);
+        if (found > 0)
             goto out;
 
         bit++;
@@ -2102,8 +2348,8 @@ static int CmdT55xxRecoverPW(const char *Cmd) {
             bit++;
             continue;
         }
-        found = tryOnePassword(curr_password);
-        if (found == 1)
+        found = tryOnePassword(curr_password, downlink_mode);
+        if (found > 0)
             goto out;
 
         bit++;
@@ -2117,9 +2363,10 @@ out:
 
     PrintAndLogEx(NORMAL, "");
 
-    if (found == 1)
-        PrintAndLogEx(SUCCESS, "Found valid password: [ " _GREEN_("%08X") " ]", curr_password);
-    else
+    if (found > 0) {
+        PrintAndLogEx(SUCCESS, "Found valid password: [ " _GREEN_("%08X") "]", curr_password);
+        T55xx_Print_DownlinkMode((found >> 1) & 3);
+    } else
         PrintAndLogEx(WARNING, "Recover pwd failed");
 
     return PM3_SUCCESS;
@@ -2129,14 +2376,14 @@ out:
 // some return all page 1 (64 bits) and others return just that block (32 bits)
 // unfortunately the 64 bits makes this more likely to get a false positive...
 bool tryDetectP1(bool getData) {
-    uint8_t preamble[] = {1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 1};
-    size_t startIdx = 0;
-    uint8_t fc1 = 0, fc2 = 0, ans = 0;
-    int clk = 0, firstClockEdge = 0;
-    bool st = true;
+    uint8_t  preamble[] = {1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 1};
+    size_t   startIdx   = 0;
+    uint8_t  fc1        = 0, fc2 = 0, ans = 0;
+    int      clk        = 0, firstClockEdge = 0;
+    bool     st         = true;
 
     if (getData) {
-        if (!AquireData(T55x7_PAGE1, T55x7_TRACE_BLOCK1, false, 0))
+        if (!AquireData(T55x7_PAGE1, T55x7_TRACE_BLOCK1, false, 0, 0))
             return false;
     }
 
@@ -2155,7 +2402,7 @@ bool tryDetectP1(bool getData) {
         }
         return false;
     }
-    
+
     // try ask clock detect.  it could be another type even if successful.
     clk = GetAskClock("", false);
     if (clk > 0) {
@@ -2164,14 +2411,14 @@ bool tryDetectP1(bool getData) {
                 (DemodBufferLen == 32 || DemodBufferLen == 64)) {
             return true;
         }
-       
+
         st = true;
         if ((ASKDemod_ext("0 1 1", false, false, 1, &st) == PM3_SUCCESS) &&
                 preambleSearchEx(DemodBuffer, preamble, sizeof(preamble), &DemodBufferLen, &startIdx, false) &&
                 (DemodBufferLen == 32 || DemodBufferLen == 64)) {
             return true;
         }
-        
+
         if ((ASKbiphaseDemod("0 0 0 2", false) == PM3_SUCCESS) &&
                 preambleSearchEx(DemodBuffer, preamble, sizeof(preamble), &DemodBufferLen, &startIdx, false) &&
                 (DemodBufferLen == 32 || DemodBufferLen == 64)) {
@@ -2184,7 +2431,7 @@ bool tryDetectP1(bool getData) {
             return true;
         }
     }
-    
+
     // try NRZ clock detect.  it could be another type even if successful.
     clk = GetNrzClock("", false); //has the most false positives :(
     if (clk > 0) {
@@ -2239,11 +2486,15 @@ bool tryDetectP1(bool getData) {
 }
 //  does this need to be a callable command?
 static int CmdT55xxDetectPage1(const char *Cmd) {
-    bool errors = false;
-    bool useGB = false;
-    bool usepwd = false;
-    uint32_t password = 0;
-    uint8_t cmdp = 0;
+    bool     errors           = false;
+    bool     useGB            = false;
+    bool     usepwd           = false;
+    bool     try_all_dl_modes = false;
+    uint8_t  found            = 0;
+    uint32_t password         = 0;
+    uint8_t  cmdp             = 0;
+    uint8_t  downlink_mode    = 0;
+    uint8_t  dl_mode          = 0;
 
     while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
         switch (tolower(param_getchar(Cmd, cmdp))) {
@@ -2259,6 +2510,12 @@ static int CmdT55xxDetectPage1(const char *Cmd) {
                 useGB = true;
                 cmdp++;
                 break;
+            case 'r':
+                downlink_mode = param_getchar(Cmd, cmdp + 1) - '0';
+                if (downlink_mode == 4) try_all_dl_modes = true;
+                if (downlink_mode > 3) downlink_mode = 0;
+                cmdp += 2;
+                break;
             default:
                 PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
                 errors = true;
@@ -2268,21 +2525,39 @@ static int CmdT55xxDetectPage1(const char *Cmd) {
     if (errors) return usage_t55xx_detectP1();
 
     if (!useGB) {
-        if (!AquireData(T55x7_PAGE1, T55x7_TRACE_BLOCK1, usepwd, password))
-            return PM3_ENODATA;
+        for (dl_mode = downlink_mode; dl_mode < 4; dl_mode++) {
+            found = AquireData(T55x7_PAGE1, T55x7_TRACE_BLOCK1, usepwd, password, dl_mode);
+            //return PM3_ENODATA;
+            if (tryDetectP1(false)) { //tryDetectModulation())
+                found = dl_mode;
+                dl_mode = 4;
+            } else found = false;
+
+            if (!try_all_dl_modes) dl_mode = 4;
+        }
+
     }
 
-    if (tryDetectP1(false))
+    if (useGB) found = tryDetectP1(false);
+
+    if (found) {
         PrintAndLogEx(SUCCESS, "T55xx chip found!");
+        T55xx_Print_DownlinkMode(found);
+    } else
+        PrintAndLogEx(WARNING, "Could not detect modulation automatically. Try setting it manually with " _YELLOW_("\'lf t55xx config\'"));
+
 
     return PM3_SUCCESS;
 }
 
 static int CmdT55xxSetDeviceConfig(const char *Cmd) {
-    uint8_t startgap = 0, writegap = 0;
-    uint8_t write0 = 0, write1 = 0, readgap = 0;
-    bool errors = false, shall_persist = false;
-    uint8_t cmdp = 0;
+    uint8_t startgap      = 0, writegap = 0, readgap = 0;
+    uint8_t write0        = 0, write1   = 0, write2  = 0, write3 = 0;
+    bool    errors        = false, shall_persist = false;
+    uint8_t cmdp          = 0;
+    uint8_t downlink_mode = 0;
+    bool    set_defaults  = false;
+
     while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
         switch (tolower(param_getchar(Cmd, cmdp))) {
             case 'h':
@@ -2307,10 +2582,27 @@ static int CmdT55xxSetDeviceConfig(const char *Cmd) {
                 errors |= param_getdec(Cmd, cmdp + 1, &readgap);
                 cmdp += 2;
                 break;
+            case 'f':
+                errors |= param_getdec(Cmd, cmdp + 1, &write2);
+                cmdp += 2;
+                break;
+            case 'g':
+                errors |= param_getdec(Cmd, cmdp + 1, &write3);
+                cmdp += 2;
+                break;
+            case 'r':
+                downlink_mode = param_getchar(Cmd, cmdp + 1) - '0';
+                if (downlink_mode > 3) downlink_mode = 0;
+                cmdp += 2;
+                break;
             case 'p':
                 shall_persist = true;
                 cmdp++;
                 break;
+            case 'z':
+                set_defaults = true;
+                cmdp++;
+                break;
             default:
                 PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
                 errors = 1;
@@ -2321,8 +2613,61 @@ static int CmdT55xxSetDeviceConfig(const char *Cmd) {
     //Validations
     if (errors || cmdp == 0) return usage_lf_deviceconfig();
 
-    t55xx_config conf = { startgap * 8, writegap * 8, write0 * 8, write1 * 8, readgap * 8 };
+    t55xx_config conf = {0};
+/* 	 if (erase) {
+		memset (&conf,0xff, sizeof(conf));
+		printf ("Conf.m[0] %x\n",conf.m[0].start_gap);
+		*/
+		//
+    if (set_defaults){
+		// fixed bit length
+		conf.m[T55xx_DLMode_Fixed].start_gap  = 29 * 8;
+		conf.m[T55xx_DLMode_Fixed].write_gap  = 17 * 8;
+		conf.m[T55xx_DLMode_Fixed].write_0    = 15 * 8;
+		conf.m[T55xx_DLMode_Fixed].write_1    = 47 * 8;
+		conf.m[T55xx_DLMode_Fixed].read_gap   = 15 * 8;
+		conf.m[T55xx_DLMode_Fixed].write_2    = 0;
+		conf.m[T55xx_DLMode_Fixed].write_3    = 0;
+	
+	   // long leading reference
+		conf.m[T55xx_DLMode_LLR].start_gap  = 31 * 8;
+		conf.m[T55xx_DLMode_LLR].write_gap  = 20 * 8;
+		conf.m[T55xx_DLMode_LLR].write_0    = 18 * 8;
+		conf.m[T55xx_DLMode_LLR].write_1    = 50 * 8;
+		conf.m[T55xx_DLMode_LLR].read_gap   = 15 * 8;
+		conf.m[T55xx_DLMode_LLR].write_2    = 0;
+		conf.m[T55xx_DLMode_LLR].write_3    = 0;
+   
+	   // leading zero
+		conf.m[T55xx_DLMode_Leading0].start_gap  = 31 * 8;
+		conf.m[T55xx_DLMode_Leading0].write_gap  = 20 * 8;
+		conf.m[T55xx_DLMode_Leading0].write_0    = 18 * 8;
+		conf.m[T55xx_DLMode_Leading0].write_1    = 40 * 8;
+		conf.m[T55xx_DLMode_Leading0].read_gap   = 15 * 8;
+		conf.m[T55xx_DLMode_Leading0].write_2    = 0;
+		conf.m[T55xx_DLMode_Leading0].write_3    = 0;
 
+		// 1 of 4 coding reference
+		conf.m[T55xx_DLMode_1of4].start_gap  = 29 * 8;
+		conf.m[T55xx_DLMode_1of4].write_gap  = 17 * 8;
+		conf.m[T55xx_DLMode_1of4].write_0    = 15 * 8;
+		conf.m[T55xx_DLMode_1of4].write_1    = 31 * 8;
+		conf.m[T55xx_DLMode_1of4].read_gap   = 15 * 8;
+		conf.m[T55xx_DLMode_1of4].write_2    = 47 * 8;
+		conf.m[T55xx_DLMode_1of4].write_3    = 63 * 8;
+
+	 }
+	 else {
+	 
+		conf.m[downlink_mode].start_gap  = startgap * 8;
+		conf.m[downlink_mode].write_gap  = writegap * 8;
+		conf.m[downlink_mode].write_0    = write0   * 8;
+		conf.m[downlink_mode].write_1    = write1   * 8;
+		conf.m[downlink_mode].read_gap   = readgap  * 8;
+		conf.m[downlink_mode].write_2    = write2   * 8;
+		conf.m[downlink_mode].write_3    = write3   * 8;
+	 }
+	 
     clearCommandBuffer();
     SendCommandOLD(CMD_SET_LF_T55XX_CONFIG, shall_persist, 0, 0, &conf, sizeof(t55xx_config));
     return PM3_SUCCESS;
@@ -2330,9 +2675,9 @@ static int CmdT55xxSetDeviceConfig(const char *Cmd) {
 
 static command_t CommandTable[] = {
     {"help",         CmdHelp,                 AlwaysAvailable, "This help"},
-    {"bruteforce",   CmdT55xxBruteForce,      IfPm3Lf,         "<start password> <end password> [i <*.dic>] Simple bruteforce attack to find password"},
+    {"bruteforce",   CmdT55xxBruteForce,      IfPm3Lf,         "<start password> <end password> Simple bruteforce attack to find password"},
     {"config",       CmdT55xxSetConfig,       AlwaysAvailable, "Set/Get T55XX configuration (modulation, inverted, offset, rate)"},
-    {"chk",          CmdT55xxChkPwds,         IfPm3Lf,         "Check passwords"},
+    {"chk",          CmdT55xxChkPwds,         IfPm3Lf,         "Check passwords from dictionary/flash"},
     {"detect",       CmdT55xxDetect,          AlwaysAvailable, "[1] Try detecting the tag modulation from reading the configuration block."},
     {"deviceconfig", CmdT55xxSetDeviceConfig, IfPm3Lf,         "Set/Get T55XX device configuration (startgap, writegap, write0, write1, readgap"},
     {"p1detect",     CmdT55xxDetectPage1,     IfPm3Lf,         "[1] Try detecting if this is a t55xx tag by reading page 1"},
diff --git a/client/cmdlft55xx.h b/client/cmdlft55xx.h
index 56fc942fd..c61ef137d 100644
--- a/client/cmdlft55xx.h
+++ b/client/cmdlft55xx.h
@@ -148,9 +148,9 @@ char *GetModelStrFromCID(uint32_t cid);
 char *GetSelectedModulationStr(uint8_t id);
 void printT5xxHeader(uint8_t page);
 void printT55xxBlock(uint8_t blockNum);
-int printConfiguration(t55xx_conf_block_t b);
+int  printConfiguration(t55xx_conf_block_t b);
 
-int T55xxReadBlock(uint8_t block, bool page1, bool usepwd, bool override, uint32_t password);
+int  T55xxReadBlock(uint8_t block, bool page1, bool usepwd, uint8_t override, uint32_t password, uint8_t downlink_mode);
 bool GetT55xxBlockData(uint32_t *blockdata);
 bool DecodeT55xxBlock(void);
 bool tryDetectModulation(void);
@@ -158,9 +158,9 @@ bool testKnownConfigBlock(uint32_t block0);
 
 bool tryDetectP1(bool getData);
 bool test(uint8_t mode, uint8_t *offset, int *fndBitRate, uint8_t clk, bool *Q5);
-int special(const char *Cmd);
-bool AquireData(uint8_t page, uint8_t block, bool pwdmode, uint32_t password);
-int tryOnePassword(uint32_t password);
+int  special(const char *Cmd);
+bool AquireData(uint8_t page, uint8_t block, bool pwdmode, uint32_t password, uint8_t downlink_mode);
+uint8_t  tryOnePassword(uint32_t password, uint8_t downlink_mode);
 
 void printT55x7Trace(t55x7_tracedata_t data, uint8_t repeat);
 void printT5555Trace(t5555_tracedata_t data, uint8_t repeat);
diff --git a/client/cmdlfti.c b/client/cmdlfti.c
index 3baa8c2e6..5236550b2 100644
--- a/client/cmdlfti.c
+++ b/client/cmdlfti.c
@@ -233,7 +233,7 @@ static int CmdTIDemod(const char *Cmd) {
 
         // only 15 bits compare, last bit of ident is not valid
         if ((shift3 ^ shift0) & 0x7FFF) {
-            PrintAndLogEx(WARNING, "Error: Ident mismatch!");
+            PrintAndLogEx(WARNING, "Warning: Ident mismatch!");
         }
         // WARNING the order of the bytes in which we calc crc below needs checking
         // i'm 99% sure the crc algorithm is correct, but it may need to eat the
@@ -257,7 +257,7 @@ static int CmdTIDemod(const char *Cmd) {
         PrintAndLogEx(INFO, "Tag data = %08X%08X  [Crc %04X %s]", shift1, shift0, crc, crcStr);
 
         if (crc != (shift2 & 0xFFFF))
-            PrintAndLogEx(WARNING, "Error: CRC mismatch, calculated %04X, got %04X", crc, shift2 & 0xFFFF);
+            PrintAndLogEx(WARNING, "Warning: CRC mismatch, calculated %04X, got %04X", crc, shift2 & 0xFFFF);
 
         retval = PM3_SUCCESS;
         goto out;
diff --git a/client/cmdlfviking.c b/client/cmdlfviking.c
index 750070a34..41193ea12 100644
--- a/client/cmdlfviking.c
+++ b/client/cmdlfviking.c
@@ -91,7 +91,7 @@ static int CmdVikingClone(const char *Cmd) {
     SendCommandMIX(CMD_VIKING_CLONE_TAG, rawID >> 32, rawID & 0xFFFFFFFF, Q5, NULL, 0);
     PacketResponseNG resp;
     if (!WaitForResponseTimeout(CMD_ACK, &resp, T55XX_WRITE_TIMEOUT)) {
-        PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+        PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
         return PM3_ETIMEOUT;
     }
     return PM3_SUCCESS;
diff --git a/client/cmdlfvisa2000.c b/client/cmdlfvisa2000.c
index b717f65dd..12d4ba1c6 100644
--- a/client/cmdlfvisa2000.c
+++ b/client/cmdlfvisa2000.c
@@ -186,7 +186,7 @@ static int CmdVisa2kClone(const char *Cmd) {
         SendCommandNG(CMD_T55XX_WRITE_BLOCK, (uint8_t *)&ng, sizeof(ng));
         if (!WaitForResponseTimeout(CMD_T55XX_WRITE_BLOCK, &resp, T55XX_WRITE_TIMEOUT)) {
 
-            PrintAndLogEx(WARNING, "Error occurred, device did not respond during write operation.");
+            PrintAndLogEx(ERR, "Error occurred, device did not respond during write operation.");
             return PM3_ETIMEOUT;
         }
     }
diff --git a/client/cmdmain.c b/client/cmdmain.c
index 311e2d6de..6748f81f0 100644
--- a/client/cmdmain.c
+++ b/client/cmdmain.c
@@ -31,6 +31,32 @@ static int CmdRem(const char *Cmd) {
     return PM3_SUCCESS;
 }
 
+static int usage_msleep(void) {
+    PrintAndLogEx(NORMAL, "Sleep for given amount of milliseconds");
+    PrintAndLogEx(NORMAL, "");
+    PrintAndLogEx(NORMAL, "Usage:  msleep <ms>");
+    PrintAndLogEx(NORMAL, "Options:");
+    PrintAndLogEx(NORMAL, "       h          This help");
+    PrintAndLogEx(NORMAL, "       <ms>       time in milliseconds");
+    PrintAndLogEx(NORMAL, "");
+    PrintAndLogEx(NORMAL, "Examples:");
+    PrintAndLogEx(NORMAL, "       msleep 100");
+    return PM3_SUCCESS;
+}
+
+static int CmdMsleep(const char *Cmd) {
+    uint32_t ms = 0;
+    char ctmp = tolower(param_getchar(Cmd, 0));
+    if (strlen(Cmd) < 1 || ctmp == 'h') return usage_msleep();
+    if (param_getchar(Cmd, 0) != 0x00) {
+        ms = param_get32ex(Cmd, 0, 0, 10);
+        if (ms == 0)
+            return usage_msleep();
+    }
+    msleep(ms);
+    return PM3_SUCCESS;
+}
+
 static int CmdQuit(const char *Cmd) {
     (void)Cmd; // Cmd is not used so far
     return PM3_EFATAL;
@@ -50,8 +76,9 @@ static command_t CommandTable[] = {
     {"hw",      CmdHW,        AlwaysAvailable,         "{ Hardware commands... }"},
     {"lf",      CmdLF,        AlwaysAvailable,         "{ Low Frequency commands... }"},
     {"mem",     CmdFlashMem,  IfPm3Flash,              "{ Flash Memory manipulation... }"},
-    {"rem",     CmdRem,       AlwaysAvailable,         "{ Add text to row in log file }"},
-    {"reveng",  CmdRev,       AlwaysAvailable,         "{ Crc calculations from the RevEng software... }"},
+    {"msleep",  CmdMsleep,    AlwaysAvailable,         "Add a pause in milliseconds"},
+    {"rem",     CmdRem,       AlwaysAvailable,         "Add text to row in log file"},
+    {"reveng",  CmdRev,       AlwaysAvailable,         "{ Crc calculations from the RevEng software }"},
     {"sc",      CmdSmartcard, IfPm3Smartcard,          "{ Smart card ISO7816 commands... }"},
     {"script",  CmdScript,    AlwaysAvailable,         "{ Scripting commands }"},
     {"trace",   CmdTrace,     AlwaysAvailable,         "{ Trace manipulation... }"},
diff --git a/client/cmdsmartcard.c b/client/cmdsmartcard.c
index c8529fa23..c49f0d0e7 100644
--- a/client/cmdsmartcard.c
+++ b/client/cmdsmartcard.c
@@ -284,7 +284,7 @@ static int PrintATR(uint8_t *atr, size_t atrlen) {
             vxor ^= atr[i];
 
         if (vxor)
-            PrintAndLogEx(WARNING, "Check sum error. Must be 0 got 0x%02X", vxor);
+            PrintAndLogEx(WARNING, "Invalid check sum. Must be 0 got 0x%02X", vxor);
         else
             PrintAndLogEx(INFO, "Check sum OK.");
     }
@@ -295,7 +295,7 @@ static int PrintATR(uint8_t *atr, size_t atrlen) {
     uint8_t calen = 2 + T1len + TD1len + TDilen + K;
 
     if (atrlen != calen && atrlen != calen + 1)  // may be CRC
-        PrintAndLogEx(ERR, "ATR length error. len: %d, T1len: %d, TD1len: %d, TDilen: %d, K: %d", atrlen, T1len, TD1len, TDilen, K);
+        PrintAndLogEx(WARNING, "Invalid ATR length. len: %d, T1len: %d, TD1len: %d, TDilen: %d, K: %d", atrlen, T1len, TD1len, TDilen, K);
 
     if (K > 0)
         PrintAndLogEx(INFO, "\nHistorical bytes | len 0x%02d | format %02x", K, atr[2 + T1len + TD1len + TDilen]);
@@ -568,14 +568,14 @@ static int CmdSmartUpgrade(const char *Cmd) {
     fseek(f, 0, SEEK_SET);
 
     if (fsize <= 0) {
-        PrintAndLogEx(WARNING, "error, when getting filesize");
+        PrintAndLogEx(ERR, "error, when getting filesize");
         fclose(f);
         return 1;
     }
 
     uint8_t *dump = calloc(fsize, sizeof(uint8_t));
     if (!dump) {
-        PrintAndLogEx(WARNING, "error, cannot allocate memory ");
+        PrintAndLogEx(ERR, "error, cannot allocate memory ");
         fclose(f);
         return 1;
     }
@@ -914,9 +914,7 @@ static int smart_brute_sfi(bool decodeTLV) {
 
         for (uint16_t rec = 1; rec <= 255; rec++) {
 
-            if (ukbhit()) {
-                int gc = getchar();
-                (void)gc;
+            if (kbd_enter_pressed()) {
                 PrintAndLogEx(WARNING, "\naborted via keyboard!\n");
                 free(buf);
                 return 1;
diff --git a/client/cmdsmartcard.h b/client/cmdsmartcard.h
index 9032ab265..6b960f311 100644
--- a/client/cmdsmartcard.h
+++ b/client/cmdsmartcard.h
@@ -25,7 +25,7 @@
 #include "emv/emvcore.h"        // decodeTVL
 #include "emv/apduinfo.h"       // APDUcode description
 #include "emv/dump.h"           // dump_buffer
-#include "crypto/libpcrypto.h"	// sha512hash
+#include "crypto/libpcrypto.h"  // sha512hash
 
 int CmdSmartcard(const char *Cmd);
 
diff --git a/client/cmdtrace.c b/client/cmdtrace.c
index 914bf06d6..3a2ea9aef 100644
--- a/client/cmdtrace.c
+++ b/client/cmdtrace.c
@@ -759,7 +759,7 @@ int CmdTraceList(const char *Cmd) {
     if (isOnline) {
         // Query for the size of the trace,  downloading PM3_CMD_DATA_SIZE
         PacketResponseNG response;
-        if (!GetFromDevice(BIG_BUF, trace, PM3_CMD_DATA_SIZE, 0, &response, 4000, true)) {
+        if (!GetFromDevice(BIG_BUF, trace, PM3_CMD_DATA_SIZE, 0, NULL, 0, &response, 4000, true)) {
             PrintAndLogEx(WARNING, "timeout while waiting for reply.");
             return 1;
         }
@@ -773,7 +773,7 @@ int CmdTraceList(const char *Cmd) {
                 return 2;
             }
             trace = p;
-            if (!GetFromDevice(BIG_BUF, trace, traceLen, 0, NULL, 2500, false)) {
+            if (!GetFromDevice(BIG_BUF, trace, traceLen, 0, NULL, 0, NULL, 2500, false)) {
                 PrintAndLogEx(WARNING, "command execution time out");
                 free(trace);
                 return 3;
diff --git a/client/cmdusart.c b/client/cmdusart.c
index e57834545..60b977430 100644
--- a/client/cmdusart.c
+++ b/client/cmdusart.c
@@ -312,11 +312,9 @@ static int CmdUsartBtFactory(const char *Cmd) {
     PrintAndLogEx(WARNING, _RED_("WARNING: process only if strictly needed!"));
     PrintAndLogEx(WARNING, "This requires BT turned ON and NOT connected!");
     PrintAndLogEx(WARNING, "Is the add-on blue light blinking? (Say 'n' if you want to abort) [y/n]");
-    while (!ukbhit()) {
-        msleep(200);
-    }
 
-    if (tolower(getchar()) != 'y') {
+    char input[3];
+    if ((fgets(input, sizeof(input), stdin) == NULL) || (strncmp(input, "y\n", sizeof(input)) != 0)) {
         PrintAndLogEx(NORMAL, "");
         PrintAndLogEx(FAILED, "Aborting.");
         return PM3_EOPABORTED;
@@ -448,11 +446,10 @@ static int CmdUsartBtFactory(const char *Cmd) {
     }
 
     if ((baudrate != USART_BAUD_RATE) || (parity != USART_PARITY)) {
-        PrintAndLogEx(WARNING, "Add-on uart settings changed, please turn BT add-on OFF and ON again, then press any key.");
-        while (!ukbhit()) {
+        PrintAndLogEx(WARNING, "Add-on uart settings changed, please turn BT add-on OFF and ON again, then press Enter.");
+        while (!kbd_enter_pressed()) {
             msleep(200);
         }
-        getchar();
         PrintAndLogEx(NORMAL, "");
         PrintAndLogEx(INFO, "Trying to connect add-on with the new settings.");
         found = usart_bt_testcomm(USART_BAUD_RATE, USART_PARITY) == PM3_SUCCESS;
diff --git a/client/comms.c b/client/comms.c
index 5aa704728..ef00ebc44 100644
--- a/client/comms.c
+++ b/client/comms.c
@@ -136,7 +136,7 @@ static void SendCommandNG_internal(uint16_t cmd, uint8_t *data, size_t len, bool
     txBufferNG.pre.ng = ng;
     txBufferNG.pre.length = len;
     txBufferNG.pre.cmd = cmd;
-    if ( len > 0 && data ) 
+    if (len > 0 && data)
         memcpy(&txBufferNG.data, data, len);
 
     if ((conn.send_via_fpc_usart && conn.send_with_crc_on_fpc) || ((!conn.send_via_fpc_usart) && conn.send_with_crc_on_usb)) {
@@ -359,7 +359,7 @@ __attribute__((force_align_arg_pointer))
 
                     res = uart_receive(sp, (uint8_t *)&rx_raw.data, length, &rxlen);
                     if ((res != PM3_SUCCESS) || (rxlen != length)) {
-                        PrintAndLogEx(WARNING, "Received packet frame error variable part too short? %d/%d", rxlen, length);
+                        PrintAndLogEx(WARNING, "Received packet frame with variable part too short? %d/%d", rxlen, length);
                         error = true;
                     } else {
 
@@ -392,7 +392,7 @@ __attribute__((force_align_arg_pointer))
                 if (!error) {                        // Get the postamble
                     res = uart_receive(sp, (uint8_t *)&rx_raw.foopost, sizeof(PacketResponseNGPostamble), &rxlen);
                     if ((res != PM3_SUCCESS) || (rxlen != sizeof(PacketResponseNGPostamble))) {
-                        PrintAndLogEx(WARNING, "Received packet frame error fetching postamble");
+                        PrintAndLogEx(WARNING, "Received packet frame without postamble");
                         error = true;
                     }
                 }
@@ -402,7 +402,7 @@ __attribute__((force_align_arg_pointer))
                         uint8_t first, second;
                         compute_crc(CRC_14443_A, (uint8_t *)&rx_raw, sizeof(PacketResponseNGPreamble) + length, &first, &second);
                         if ((first << 8) + second != rx.crc) {
-                            PrintAndLogEx(WARNING, "Received packet frame CRC error %02X%02X <> %04X", first, second, rx.crc);
+                            PrintAndLogEx(WARNING, "Received packet frame with invalid CRC %02X%02X <> %04X", first, second, rx.crc);
                             error = true;
                         }
                     }
@@ -424,7 +424,7 @@ __attribute__((force_align_arg_pointer))
 
                 res = uart_receive(sp, ((uint8_t *)&rx_old) + sizeof(PacketResponseNGPreamble), sizeof(PacketResponseOLD) - sizeof(PacketResponseNGPreamble), &rxlen);
                 if ((res != PM3_SUCCESS) || (rxlen != sizeof(PacketResponseOLD) - sizeof(PacketResponseNGPreamble))) {
-                    PrintAndLogEx(WARNING, "Received packet OLD frame payload error too short? %d/%d", rxlen, sizeof(PacketResponseOLD) - sizeof(PacketResponseNGPreamble));
+                    PrintAndLogEx(WARNING, "Received packet OLD frame with payload too short? %d/%d", rxlen, sizeof(PacketResponseOLD) - sizeof(PacketResponseNGPreamble));
                     error = true;
                 }
                 if (!error) {
@@ -734,12 +734,14 @@ bool WaitForResponse(uint32_t cmd, PacketResponseNG *response) {
 * @param dest Destination address for transfer
 * @param bytes number of bytes to be transferred
 * @param start_index offset into Proxmark3 BigBuf[]
+* @param data used by SPIFFS to provide filename
+* @param datalen used by SPIFFS to provide filename length
 * @param response struct to copy last command (CMD_ACK) into
 * @param ms_timeout timeout in milliseconds
 * @param show_warning display message after 2 seconds
 * @return true if command was returned, otherwise false
 */
-bool GetFromDevice(DeviceMemType_t memtype, uint8_t *dest, uint32_t bytes, uint32_t start_index, PacketResponseNG *response, size_t ms_timeout, bool show_warning) {
+bool GetFromDevice(DeviceMemType_t memtype, uint8_t *dest, uint32_t bytes, uint32_t start_index, uint8_t *data, uint32_t datalen, PacketResponseNG *response, size_t ms_timeout, bool show_warning) {
 
     if (dest == NULL) return false;
     if (bytes == 0) return true;
@@ -760,6 +762,10 @@ bool GetFromDevice(DeviceMemType_t memtype, uint8_t *dest, uint32_t bytes, uint3
             SendCommandMIX(CMD_DOWNLOAD_EML_BIGBUF, start_index, bytes, 0, NULL, 0);
             return dl_it(dest, bytes, start_index, response, ms_timeout, show_warning, CMD_DOWNLOADED_EML_BIGBUF);
         }
+        case SPIFFS: {
+            SendCommandMIX(CMD_SPIFFS_DOWNLOAD, start_index, bytes, 0, data, datalen);
+            return dl_it(dest, bytes, start_index, response, ms_timeout, show_warning, CMD_SPIFFS_DOWNLOADED);
+        }
         case FLASH_MEM: {
             SendCommandMIX(CMD_FLASHMEM_DOWNLOAD, start_index, bytes, 0, NULL, 0);
             return dl_it(dest, bytes, start_index, response, ms_timeout, show_warning, CMD_FLASHMEM_DOWNLOADED);
diff --git a/client/comms.h b/client/comms.h
index 6ec14331d..bc036f008 100644
--- a/client/comms.h
+++ b/client/comms.h
@@ -37,6 +37,7 @@ typedef enum {
     BIG_BUF_EML,
     FLASH_MEM,
     SIM_MEM,
+    SPIFFS
 } DeviceMemType_t;
 
 typedef struct {
@@ -72,7 +73,8 @@ bool WaitForResponseTimeoutW(uint32_t cmd, PacketResponseNG *response, size_t ms
 bool WaitForResponseTimeout(uint32_t cmd, PacketResponseNG *response, size_t ms_timeout);
 bool WaitForResponse(uint32_t cmd, PacketResponseNG *response);
 
-bool GetFromDevice(DeviceMemType_t memtype, uint8_t *dest, uint32_t bytes, uint32_t start_index, PacketResponseNG *response, size_t ms_timeout, bool show_warning);
+//bool GetFromDevice(DeviceMemType_t memtype, uint8_t *dest, uint32_t bytes, uint32_t start_index, PacketResponseNG *response, size_t ms_timeout, bool show_warning);
+bool GetFromDevice(DeviceMemType_t memtype, uint8_t *dest, uint32_t bytes, uint32_t start_index, uint8_t *data, uint32_t datalen, PacketResponseNG *response, size_t ms_timeout, bool show_warning);
 
 #endif
 
diff --git a/client/default_keys.dic b/client/default_keys.dic
index 4810e56da..4df09cb1c 100644
--- a/client/default_keys.dic
+++ b/client/default_keys.dic
@@ -944,3 +944,9 @@ A23456789123
 A00003000084
 675A32413770
 395244733978
+A0004A000036
+2C9F3D45BA13
+4243414F5250
+DFE73BE48AC6
+#
+B069D0D03D17
diff --git a/client/default_keys_dic2lua.awk b/client/default_keys_dic2lua.awk
index 432ace8bf..47a4d5439 100644
--- a/client/default_keys_dic2lua.awk
+++ b/client/default_keys_dic2lua.awk
@@ -1,24 +1,23 @@
 BEGIN {
-	print "--[["
-	print "These are default_keys dictionary"
-	print "This file is automatically generated from default_keys.h - DON'T EDIT MANUALLY."
-	print "--]]"
-	print "local _keys = {"
+    print "--[["
+    print "These are default_keys dictionary"
+    print "This file is automatically generated from default_keys.h - DON'T EDIT MANUALLY."
+    print "--]]"
+    print "local _keys = {"
 }
 
 $1 ~ /^[A-Fa-f0-9]+/ { sub(/\r/, ""); print "   '"substr($1,1,12)"'," }
 
 END {
-	print "}"
-	
-	print "---"
-	print "--    The keys above have just been pasted in, for completeness sake. They contain duplicates. "
-	print "--    We need to weed the duplicates out before we expose the list to someone who actually wants to use them"
-	print "--    @param list a list to do 'uniq' on"
-	print ""
-	print "local function uniq(list)"
+    print "}"
+    print "---"
+    print "--    The keys above have just been pasted in, for completeness sake. They contain duplicates. "
+    print "--    We need to weed the duplicates out before we expose the list to someone who actually wants to use them"
+    print "--    @param list a list to do 'uniq' on"
     print ""
-	print "    local foobar = {}"
+    print "local function uniq(list)"
+    print ""
+    print "    local foobar = {}"
     print "    for _, value in pairs(list) do"
     print "        value = value:lower()"
     print "        if not foobar[value] then"
@@ -29,4 +28,4 @@ END {
     print "    return foobar"
     print "end"
     print "return uniq(_keys)"
-}
\ No newline at end of file
+}
diff --git a/client/deprecated-hid-flasher/flasher/elf.h b/client/deprecated-hid-flasher/flasher/elf.h
index 0551cbd76..eef2863d8 100644
--- a/client/deprecated-hid-flasher/flasher/elf.h
+++ b/client/deprecated-hid-flasher/flasher/elf.h
@@ -9,6 +9,8 @@
 #ifndef __ELF_H__
 #define __ELF_H__
 
+#include "proxmark3.h"
+
 typedef struct {
     uint32_t p_type;
     uint32_t p_offset;
@@ -18,7 +20,7 @@ typedef struct {
     uint32_t p_memsz;
     uint32_t p_flags;
     uint32_t p_align;
-} __attribute__((__packed__)) Elf32_Phdr;
+} PACKED Elf32_Phdr;
 
 #define EI_NIDENT 16
 
@@ -37,7 +39,7 @@ typedef struct {
     uint16_t e_shentsize;
     uint16_t e_shnum;
     uint16_t e_shtrndx;
-} __attribute__((__packed__)) Elf32_Ehdr;
+} PACKED Elf32_Ehdr;
 
 #define PT_NULL      0
 #define PT_LOAD      1
diff --git a/client/deprecated-hid-flasher/flasher/proxmark3.h b/client/deprecated-hid-flasher/flasher/proxmark3.h
index f5dd2b7fa..c9e42efd8 100644
--- a/client/deprecated-hid-flasher/flasher/proxmark3.h
+++ b/client/deprecated-hid-flasher/flasher/proxmark3.h
@@ -14,4 +14,16 @@
 
 #define PROXPROMPT "proxmark3> "
 
+#ifdef _MSC_VER
+typedef DWORD uint32_t;
+typedef BYTE uint8_t;
+#define PACKED
+// stuff
+#else
+#include <stdint.h>
+#include <stdbool.h>
+#define PACKED __attribute__((packed))
+#endif
+
+
 #endif
diff --git a/client/deprecated-hid-flasher/flasher/sleep.h b/client/deprecated-hid-flasher/flasher/sleep.h
index f3aac0c8f..ffb5486aa 100644
--- a/client/deprecated-hid-flasher/flasher/sleep.h
+++ b/client/deprecated-hid-flasher/flasher/sleep.h
@@ -6,18 +6,18 @@
 #define SLEEP_H__
 
 #ifdef _WIN32
-    #include <windows.h>
-    #define msleep(n) Sleep(n)
+#include <windows.h>
+#define msleep(n) Sleep(n)
 #else
-    #include <time.h>
-    #include <errno.h>
-    static void nsleep(uint64_t n) {
-        struct timespec timeout;
-        timeout.tv_sec = n / 1000000000;
-        timeout.tv_nsec = n % 1000000000;
-        while (nanosleep(&timeout, &timeout) && errno == EINTR);
-    }
-    #define msleep(n) nsleep(1000000 * (uint64_t)n)
+#include <time.h>
+#include <errno.h>
+static void nsleep(uint64_t n) {
+    struct timespec timeout;
+    timeout.tv_sec = n / 1000000000;
+    timeout.tv_nsec = n % 1000000000;
+    while (nanosleep(&timeout, &timeout) && errno == EINTR);
+}
+#define msleep(n) nsleep(1000000 * (uint64_t)n)
 #endif
 
 #endif
diff --git a/client/deprecated-hid-flasher/flasher/usb_cmd.h b/client/deprecated-hid-flasher/flasher/usb_cmd.h
index 2b383faf8..60c359e5a 100644
--- a/client/deprecated-hid-flasher/flasher/usb_cmd.h
+++ b/client/deprecated-hid-flasher/flasher/usb_cmd.h
@@ -12,16 +12,8 @@
 
 #ifndef __USB_CMD_H
 #define __USB_CMD_H
-#ifdef _MSC_VER
-typedef DWORD uint32_t;
-typedef BYTE uint8_t;
-#define PACKED
-// stuff
-#else
-#include <stdint.h>
-#include <stdbool.h>
-#define PACKED __attribute__((packed))
-#endif
+
+#include "proxmark3.h"
 
 typedef struct {
     uint32_t    cmd;
diff --git a/client/deprecated-hid-flasher/unbind-proxmark b/client/deprecated-hid-flasher/unbind-proxmark
index 986c0015d..325607cb6 100755
--- a/client/deprecated-hid-flasher/unbind-proxmark
+++ b/client/deprecated-hid-flasher/unbind-proxmark
@@ -1,16 +1,16 @@
 #!/bin/sh
 
 for i in /sys/bus/usb/devices/*; do
-	if grep "9ac4" "${i}/idVendor" >/dev/null 2>&1; then
-		echo "Found Proxmark..."
-		dev=`basename "${i}"`
+    if grep "9ac4" "${i}/idVendor" >/dev/null 2>&1; then
+        echo "Found Proxmark..."
+        dev=`basename "${i}"`
 
-		for j in /sys/bus/usb/drivers/usbhid/*; do
-			if basename "${j}"|grep "^${dev}" >/dev/null; then
-				bound="`basename "${j}"`"
-				echo "Unbinding ${bound}..."
-				echo -n "${bound}" >/sys/bus/usb/drivers/usbhid/unbind
-			fi
-		done
-	fi
+        for j in /sys/bus/usb/drivers/usbhid/*; do
+            if basename "${j}"|grep "^${dev}" >/dev/null; then
+                bound="`basename "${j}"`"
+                echo "Unbinding ${bound}..."
+                echo -n "${bound}" >/sys/bus/usb/drivers/usbhid/unbind
+            fi
+        done
+    fi
 done
diff --git a/client/elf.h b/client/elf.h
index 0551cbd76..50533224f 100644
--- a/client/elf.h
+++ b/client/elf.h
@@ -9,6 +9,8 @@
 #ifndef __ELF_H__
 #define __ELF_H__
 
+#include "common.h"
+
 typedef struct {
     uint32_t p_type;
     uint32_t p_offset;
@@ -18,7 +20,7 @@ typedef struct {
     uint32_t p_memsz;
     uint32_t p_flags;
     uint32_t p_align;
-} __attribute__((__packed__)) Elf32_Phdr;
+} PACKED Elf32_Phdr;
 
 #define EI_NIDENT 16
 
@@ -37,7 +39,7 @@ typedef struct {
     uint16_t e_shentsize;
     uint16_t e_shnum;
     uint16_t e_shtrndx;
-} __attribute__((__packed__)) Elf32_Ehdr;
+} PACKED Elf32_Ehdr;
 
 #define PT_NULL      0
 #define PT_LOAD      1
diff --git a/client/emv/apduinfo.c b/client/emv/apduinfo.c
index df44ef961..48e97622b 100644
--- a/client/emv/apduinfo.c
+++ b/client/emv/apduinfo.c
@@ -314,3 +314,190 @@ const char *GetAPDUCodeDescription(uint8_t sw1, uint8_t sw2) {
     else
         return APDUCodeTable[0].Description; //empty string
 }
+
+int APDUDecode(uint8_t *data, int len, APDUStruct *apdu) {
+    ExtAPDUHeader *hapdu = (ExtAPDUHeader *)data;
+
+    apdu->cla = hapdu->cla;
+    apdu->ins = hapdu->ins;
+    apdu->p1 = hapdu->p1;
+    apdu->p2 = hapdu->p2;
+
+    apdu->lc = 0;
+    apdu->data = NULL;
+    apdu->le = 0;
+    apdu->extended_apdu = false;
+    apdu->case_type = 0x00;
+
+    uint8_t b0 = hapdu->lc[0];
+
+    // case 1
+    if (len == 4) {
+        apdu->case_type = 0x01;
+    }
+
+    // case 2S (Le)
+    if (len == 5) {
+        apdu->case_type = 0x02;
+        apdu->le = b0;
+        if (!apdu->le)
+            apdu->le = 0x100;
+    }
+
+    // case 3S (Lc + data)
+    if (len == 5U + b0 && b0 != 0) {
+        apdu->case_type = 0x03;
+        apdu->lc = b0;
+    }
+
+    // case 4S (Lc + data + Le)
+    if (len == 5U + b0 + 1U && b0 != 0) {
+        apdu->case_type = 0x04;
+        apdu->lc = b0;
+        apdu->le = data[len - 1];
+        if (!apdu->le)
+            apdu->le = 0x100;
+    }
+
+    // extended length apdu
+    if (len >= 7 && b0 == 0) {
+        uint16_t extlen = (hapdu->lc[1] << 8) + hapdu->lc[2];
+
+        // case 2E (Le) - extended
+        if (len == 7) {
+            apdu->case_type = 0x12;
+            apdu->extended_apdu = true;
+            apdu->le = extlen;
+            if (!apdu->le)
+                apdu->le = 0x10000;
+        }
+
+        // case 3E (Lc + data) - extended
+        if (len == 7U + extlen) {
+            apdu->case_type = 0x13;
+            apdu->extended_apdu = true;
+            apdu->lc = extlen;
+        }
+
+        // case 4E (Lc + data + Le) - extended 2-byte Le
+        if (len == 7U + extlen + 2U) {
+            apdu->case_type = 0x14;
+            apdu->extended_apdu = true;
+            apdu->lc = extlen;
+            apdu->le = (data[len - 2] << 8) + data[len - 1];
+            if (!apdu->le)
+                apdu->le = 0x10000;
+        }
+
+        // case 4E (Lc + data + Le) - extended 3-byte Le
+        if (len == 7U + extlen + 3U && data[len - 3] == 0) {
+            apdu->case_type = 0x24;
+            apdu->extended_apdu = true;
+            apdu->lc = extlen;
+            apdu->le = (data[len - 2] << 8) + data[len - 1];
+            if (!apdu->le)
+                apdu->le = 0x10000;
+        }
+    }
+
+    if (!apdu->case_type)
+        return 1;
+
+    if (apdu->lc) {
+        if (apdu->extended_apdu) {
+            apdu->data = data + 7;
+        } else {
+            apdu->data = data + 5;
+        }
+
+    }
+
+    return 0;
+}
+
+int APDUEncode(APDUStruct *apdu, uint8_t *data, int *len) {
+    if (len)
+        *len = 0;
+
+    if (apdu->le > 0x10000 || apdu->lc > 0xffff)
+        return 1;
+
+    size_t dptr = 0;
+    data[dptr++] = apdu->cla;
+    data[dptr++] = apdu->ins;
+    data[dptr++] = apdu->p1;
+    data[dptr++] = apdu->p2;
+
+    if (apdu->lc) {
+        if (apdu->extended_apdu || apdu->lc > 0xff || apdu->le > 0x100) {
+            data[dptr++] = 0x00;
+            data[dptr++] = (apdu->lc >> 8) & 0xff;
+            data[dptr++] = (apdu->lc) & 0xff;
+            memmove(&data[dptr], apdu->data, apdu->lc);
+            dptr += apdu->lc;
+            apdu->extended_apdu = true;
+        } else {
+            data[dptr++] = apdu->lc;
+            memmove(&data[dptr], apdu->data, apdu->lc);
+            dptr += apdu->lc;
+        }
+    }
+
+    if (apdu->le) {
+        if (apdu->extended_apdu) {
+            if (apdu->le != 0x10000) {
+                data[dptr++] = 0x00;
+                data[dptr++] = (apdu->le >> 8) & 0xff;
+                data[dptr++] = (apdu->le) & 0xff;
+            } else {
+                data[dptr++] = 0x00;
+                data[dptr++] = 0x00;
+                data[dptr++] = 0x00;
+            }
+        } else {
+            if (apdu->le != 0x100)
+                data[dptr++] = apdu->le;
+            else
+                data[dptr++] = 0x00;
+        }
+    }
+
+    if (len)
+        *len = dptr;
+    return 0;
+}
+
+int APDUEncodeS(sAPDU *sapdu, bool extended, uint16_t le, uint8_t *data, int *len) {
+    if (extended && le > 0x100)
+        return 10;
+
+    APDUStruct apdu;
+
+    apdu.cla = sapdu->CLA;
+    apdu.ins = sapdu->INS;
+    apdu.p1 = sapdu->P1;
+    apdu.p2 = sapdu->P2;
+
+    apdu.lc = sapdu->Lc;
+    if (sapdu->Lc)
+        apdu.data = sapdu->data;
+    else
+        apdu.data = NULL;
+    apdu.le = le;
+
+    apdu.extended_apdu = extended;
+    apdu.case_type = 0x00;
+
+    return APDUEncode(&apdu, data, len);
+}
+
+void APDUPrint(APDUStruct apdu) {
+    APDUPrintEx(apdu, 0);
+}
+
+void APDUPrintEx(APDUStruct apdu, size_t maxdatalen) {
+    PrintAndLogEx(INFO, "APDU: %scase=0x%02x cla=0x%02x ins=0x%02x p1=0x%02x p2=0x%02x Lc=0x%02x(%d) Le=0x%02x(%d)",
+                  apdu.extended_apdu ? "[e]" : "", apdu.case_type, apdu.cla, apdu.ins, apdu.p1, apdu.p2, apdu.lc, apdu.lc, apdu.le, apdu.le);
+    if (maxdatalen > 0)
+        PrintAndLogEx(INFO, "data: %s%s", sprint_hex(apdu.data, MIN(apdu.lc, maxdatalen)), apdu.lc > maxdatalen ? "..." : "");
+}
diff --git a/client/emv/apduinfo.h b/client/emv/apduinfo.h
index 317d661ff..25d062dcf 100644
--- a/client/emv/apduinfo.h
+++ b/client/emv/apduinfo.h
@@ -14,8 +14,11 @@
 #include <stdio.h>
 #include <stdint.h>
 #include <stdlib.h>
+#include <stdbool.h>
 #include <inttypes.h>
 
+#include "util.h"
+
 #define APDUCODE_TYPE_NONE     0
 #define APDUCODE_TYPE_INFO     1
 #define APDUCODE_TYPE_WARNING  2
@@ -31,4 +34,39 @@ typedef struct {
 const APDUCode *GetAPDUCode(uint8_t sw1, uint8_t sw2);
 const char *GetAPDUCodeDescription(uint8_t sw1, uint8_t sw2);
 
+typedef struct {
+    uint8_t CLA;
+    uint8_t INS;
+    uint8_t P1;
+    uint8_t P2;
+    uint8_t Lc;
+    uint8_t *data;
+} PACKED sAPDU;
+
+typedef struct {
+    uint8_t cla;
+    uint8_t ins;
+    uint8_t p1;
+    uint8_t p2;
+    uint8_t lc[3];
+} PACKED ExtAPDUHeader;
+
+typedef struct {
+    uint8_t cla;
+    uint8_t ins;
+    uint8_t p1;
+    uint8_t p2;
+    uint16_t lc;
+    uint8_t *data;
+    uint32_t le;
+    bool extended_apdu;
+    uint8_t case_type;
+} PACKED APDUStruct;
+
+extern int APDUDecode(uint8_t *data, int len, APDUStruct *apdu);
+extern int APDUEncode(APDUStruct *apdu, uint8_t *data, int *len);
+extern int APDUEncodeS(sAPDU *apdu, bool extended, uint16_t le, uint8_t *data, int *len);
+extern void APDUPrint(APDUStruct apdu);
+extern void APDUPrintEx(APDUStruct apdu, size_t maxdatalen);
+
 #endif
diff --git a/client/emv/cmdemv.c b/client/emv/cmdemv.c
index 22fdea153..0cf177620 100644
--- a/client/emv/cmdemv.c
+++ b/client/emv/cmdemv.c
@@ -830,6 +830,12 @@ static int CmdEMVExec(const char *Cmd) {
         SetAPDULogging(showAPDU);
         res = EMVSearchPSE(channel, activateField, true, psenum, decodeTLV, tlvSelect);
 
+        // check PPSE instead of PSE and vice versa
+        if (res) {
+            PrintAndLogEx(NORMAL, "Check PPSE instead of PSE and vice versa...");
+            res = EMVSearchPSE(channel, false, true, psenum == 1 ? 2 : 1, decodeTLV, tlvSelect);
+        }
+
         // check PPSE and select application id
         if (!res) {
             TLVPrintAIDlistFromSelectTLV(tlvSelect);
@@ -931,7 +937,7 @@ static int CmdEMVExec(const char *Cmd) {
 
     while (AFL && AFL->len) {
         if (AFL->len % 4) {
-            PrintAndLogEx(WARNING, "Error: Wrong AFL length: %d", AFL->len);
+            PrintAndLogEx(WARNING, "Warning: Wrong AFL length: %d", AFL->len);
             break;
         }
 
@@ -1001,7 +1007,7 @@ static int CmdEMVExec(const char *Cmd) {
         AIP = AIPtlv->value[0] + AIPtlv->value[1] * 0x100;
         PrintAndLogEx(NORMAL, "* * AIP=%04x", AIP);
     } else {
-        PrintAndLogEx(ERR, "Can't found AIP.");
+        PrintAndLogEx(ERR, "Can't find AIP.");
     }
 
     // SDA
@@ -1047,11 +1053,11 @@ static int CmdEMVExec(const char *Cmd) {
                         TLVPrintFromTLVLev(cvr, 1);
                     }
                 } else {
-                    PrintAndLogEx(NORMAL, "WARNING: IAD not found.");
+                    PrintAndLogEx(WARNING, "WARNING: IAD not found.");
                 }
 
             } else {
-                PrintAndLogEx(WARNING, "Error AC: Application Transaction Counter (ATC) not found.");
+                PrintAndLogEx(WARNING, "Warning AC: Application Transaction Counter (ATC) not found.");
             }
         }
     }
@@ -1131,14 +1137,14 @@ static int CmdEMVExec(const char *Cmd) {
                             PrintAndLogEx(NORMAL, "Transaction approved ONLINE.");
                             break;
                         default:
-                            PrintAndLogEx(WARNING, "Error: CID transaction code error %2x", CID->value[0] & EMVAC_AC_MASK);
+                            PrintAndLogEx(WARNING, "Warning: CID transaction code error %2x", CID->value[0] & EMVAC_AC_MASK);
                             break;
                     }
                 } else {
-                    PrintAndLogEx(WARNING, "Error: Wrong CID length %d", CID->len);
+                    PrintAndLogEx(WARNING, "Warning: Wrong CID length %d", CID->len);
                 }
             } else {
-                PrintAndLogEx(WARNING, "Error: CID(9F27) not found.");
+                PrintAndLogEx(WARNING, "Warning: CID(9F27) not found.");
             }
 
         }
diff --git a/client/emv/emv_pk.c b/client/emv/emv_pk.c
index ae23c8470..b03567440 100644
--- a/client/emv/emv_pk.c
+++ b/client/emv/emv_pk.c
@@ -24,7 +24,7 @@
 #include "emv_pk.h"
 #include "crypto.h"
 #include "proxmark3.h"
-
+#include "util.h"
 #include <stdbool.h>
 #include <string.h>
 #include <stdlib.h>
@@ -398,7 +398,7 @@ static struct emv_pk *emv_pk_get_ca_pk_from_file(const char *fname,
 
     FILE *f = fopen(fname, "r");
     if (!f) {
-        perror("fopen");
+        PrintAndLogEx(ERR, "Error: can't open file %s.", fname);
         return NULL;
     }
 
diff --git a/client/emv/emvcore.c b/client/emv/emvcore.c
index e9e725448..c814eb11e 100644
--- a/client/emv/emvcore.c
+++ b/client/emv/emvcore.c
@@ -277,24 +277,25 @@ static int EMVExchangeEx(EMVCommandChannel channel, bool ActivateField, bool Lea
     }
 
     // COMPUTE APDU
-    memcpy(data, &apdu, 5);
-    if (apdu.data)
-        memcpy(&data[5], apdu.data, apdu.Lc);
+    int datalen = 0;
+    if (APDUEncodeS(&apdu, false, IncludeLe ? 0x100 : 0x00, data, &datalen)) {
+        PrintAndLogEx(ERR, "APDU encoding error.");
+        return 201;
+    }
 
     if (APDULogging)
-        PrintAndLogEx(SUCCESS, ">>>> %s", sprint_hex(data, (IncludeLe ? 6 : 5) + apdu.Lc));
+        PrintAndLogEx(SUCCESS, ">>>> %s", sprint_hex(data, datalen));
 
     switch (channel) {
         case ECC_CONTACTLESS:
-            // 6 byes + data = INS + CLA + P1 + P2 + Lc + <data = Nc> + Le(?IncludeLe)
-            res = ExchangeAPDU14a(data, (IncludeLe ? 6 : 5) + apdu.Lc, ActivateField, LeaveFieldON, Result, (int)MaxResultLen, (int *)ResultLen);
+            res = ExchangeAPDU14a(data, datalen, ActivateField, LeaveFieldON, Result, (int)MaxResultLen, (int *)ResultLen);
             if (res) {
                 return res;
             }
             break;
         case ECC_CONTACT:
             if (IfPm3Smartcard())
-                res = ExchangeAPDUSC(data, (IncludeLe ? 6 : 5) + apdu.Lc, ActivateField, LeaveFieldON, Result, (int)MaxResultLen, (int *)ResultLen);
+                res = ExchangeAPDUSC(data, datalen, ActivateField, LeaveFieldON, Result, (int)MaxResultLen, (int *)ResultLen);
             else
                 res = 1;
             if (res) {
@@ -336,7 +337,7 @@ static int EMVExchangeEx(EMVCommandChannel channel, bool ActivateField, bool Lea
 }
 
 int EMVExchange(EMVCommandChannel channel, bool LeaveFieldON, sAPDU apdu, uint8_t *Result, size_t MaxResultLen, size_t *ResultLen, uint16_t *sw, struct tlvdb *tlv) {
-    return EMVExchangeEx(channel, false, LeaveFieldON, apdu, (channel == ECC_CONTACTLESS), Result, MaxResultLen, ResultLen, sw, tlv);
+    return EMVExchangeEx(channel, false, LeaveFieldON, apdu, false, Result, MaxResultLen, ResultLen, sw, tlv);
 }
 
 int EMVSelect(EMVCommandChannel channel, bool ActivateField, bool LeaveFieldON, uint8_t *AID, size_t AIDLen, uint8_t *Result, size_t MaxResultLen, size_t *ResultLen, uint16_t *sw, struct tlvdb *tlv) {
@@ -511,7 +512,7 @@ int EMVSearchPSE(EMVCommandChannel channel, bool ActivateField, bool LeaveFieldO
             PrintAndLogEx(WARNING, "%s ERROR: Can't get TLV from response.", PSE_or_PPSE);
         }
     } else {
-        PrintAndLogEx(WARNING, "%s ERROR: Can't select PPSE AID. Error: %d", PSE_or_PPSE, res);
+        PrintAndLogEx(ERR, "%s ERROR: Can't select PPSE AID. Error: %d", PSE_or_PPSE, res);
     }
 
     if (!LeaveFieldON)
@@ -531,14 +532,17 @@ int EMVSearch(EMVCommandChannel channel, bool ActivateField, bool LeaveFieldON,
     int retrycnt = 0;
     for (int i = 0; i < AIDlistLen; i ++) {
         param_gethex_to_eol(AIDlist[i].aid, 0, aidbuf, sizeof(aidbuf), &aidlen);
-        res = EMVSelect(channel, (i == 0) ? ActivateField : false, (i == AIDlistLen - 1) ? LeaveFieldON : true, aidbuf, aidlen, data, sizeof(data), &datalen, &sw, tlv);
+        res = EMVSelect(channel, (i == 0) ? ActivateField : false, true, aidbuf, aidlen, data, sizeof(data), &datalen, &sw, tlv);
         // retry if error and not returned sw error
         if (res && res != 5) {
             if (++retrycnt < 3) {
                 i--;
             } else {
-                // (1) - card select error, proxmark error OR (200) - result length = 0
-                if (res == 1 || res == 200) {
+                // (1) - card select error, (4) reply timeout, (200) - result length = 0
+                if (res == 1 || res == 4 || res == 200) {
+                    if (!LeaveFieldON)
+                        DropFieldEx(channel);
+
                     PrintAndLogEx(WARNING, "Exit...");
                     return 1;
                 }
@@ -562,6 +566,9 @@ int EMVSearch(EMVCommandChannel channel, bool ActivateField, bool LeaveFieldON,
         }
     }
 
+    if (!LeaveFieldON)
+        DropFieldEx(channel);
+
     return 0;
 }
 
@@ -609,9 +616,9 @@ int EMVGPO(EMVCommandChannel channel, bool LeaveFieldON, uint8_t *PDOL, size_t P
 }
 
 int EMVReadRecord(EMVCommandChannel channel, bool LeaveFieldON, uint8_t SFI, uint8_t SFIrec, uint8_t *Result, size_t MaxResultLen, size_t *ResultLen, uint16_t *sw, struct tlvdb *tlv) {
-    int res = EMVExchange(channel, LeaveFieldON, (sAPDU) {0x00, 0xb2, SFIrec, (SFI << 3) | 0x04, 0, NULL}, Result, MaxResultLen, ResultLen, sw, tlv);
-    if (*sw == 0x6700) {
-        PrintAndLogEx(INFO, ">>> trying to reissue command withouth Le...");
+    int res = EMVExchangeEx(channel, false, LeaveFieldON, (sAPDU) {0x00, 0xb2, SFIrec, (SFI << 3) | 0x04, 0, NULL}, true, Result, MaxResultLen, ResultLen, sw, tlv);
+    if (*sw == 0x6700 || *sw == 0x6f00) {
+        PrintAndLogEx(INFO, ">>> trying to reissue command without Le...");
         res = EMVExchangeEx(channel, false, LeaveFieldON, (sAPDU) {0x00, 0xb2, SFIrec, (SFI << 3) | 0x04, 0, NULL}, false, Result, MaxResultLen, ResultLen, sw, tlv);
     }
     return res;
@@ -622,9 +629,9 @@ int EMVAC(EMVCommandChannel channel, bool LeaveFieldON, uint8_t RefControl, uint
 }
 
 int EMVGenerateChallenge(EMVCommandChannel channel, bool LeaveFieldON, uint8_t *Result, size_t MaxResultLen, size_t *ResultLen, uint16_t *sw, struct tlvdb *tlv) {
-    int res = EMVExchange(channel, LeaveFieldON, (sAPDU) {0x00, 0x84, 0x00, 0x00, 0x00, NULL}, Result, MaxResultLen, ResultLen, sw, tlv);
-    if (*sw == 0x6700) {
-        PrintAndLogEx(INFO, ">>> trying to reissue command withouth Le...");
+    int res = EMVExchangeEx(channel, false, LeaveFieldON, (sAPDU) {0x00, 0x84, 0x00, 0x00, 0x00, NULL}, true, Result, MaxResultLen, ResultLen, sw, tlv);
+    if (*sw == 0x6700 || *sw == 0x6f00) {
+        PrintAndLogEx(INFO, ">>> trying to reissue command without Le...");
         res = EMVExchangeEx(channel, false, LeaveFieldON, (sAPDU) {0x00, 0x84, 0x00, 0x00, 0x00, NULL}, false, Result, MaxResultLen, ResultLen, sw, tlv);
     }
     return res;
@@ -654,14 +661,14 @@ int trSDA(struct tlvdb *tlv) {
 
     struct emv_pk *pk = get_ca_pk(tlv);
     if (!pk) {
-        PrintAndLogEx(WARNING, "Error: Key not found. Exit.");
+        PrintAndLogEx(ERR, "Error: Key not found. Exit.");
         return 2;
     }
 
     struct emv_pk *issuer_pk = emv_pki_recover_issuer_cert(pk, tlv);
     if (!issuer_pk) {
         emv_pk_free(pk);
-        PrintAndLogEx(WARNING, "Error: Issuer certificate not found. Exit.");
+        PrintAndLogEx(ERR, "Error: Issuer certificate not found. Exit.");
         return 2;
     }
 
@@ -693,7 +700,7 @@ int trSDA(struct tlvdb *tlv) {
     } else {
         emv_pk_free(issuer_pk);
         emv_pk_free(pk);
-        PrintAndLogEx(WARNING, "SSAD verify error");
+        PrintAndLogEx(ERR, "SSAD verify error");
         return 4;
     }
 
@@ -712,21 +719,21 @@ int trDDA(EMVCommandChannel channel, bool decodeTLV, struct tlvdb *tlv) {
 
     struct emv_pk *pk = get_ca_pk(tlv);
     if (!pk) {
-        PrintAndLogEx(WARNING, "Error: Key not found. Exit.");
+        PrintAndLogEx(ERR, "Error: Key not found. Exit.");
         return 2;
     }
 
     const struct tlv *sda_tlv = tlvdb_get(tlv, 0x21, NULL);
     /* if (!sda_tlv || sda_tlv->len < 1) { it may be 0!!!!
             emv_pk_free(pk);
-            PrintAndLogEx(WARNING, "Error: Can't find input list for Offline Data Authentication. Exit.");
+            PrintAndLogEx(ERR, "Error: Can't find input list for Offline Data Authentication. Exit.");
             return 3;
         }
     */
     struct emv_pk *issuer_pk = emv_pki_recover_issuer_cert(pk, tlv);
     if (!issuer_pk) {
         emv_pk_free(pk);
-        PrintAndLogEx(WARNING, "Error: Issuer certificate not found. Exit.");
+        PrintAndLogEx(ERR, "Error: Issuer certificate not found. Exit.");
         return 2;
     }
     PrintAndLogEx(SUCCESS, "Issuer PK recovered. RID %02hhx:%02hhx:%02hhx:%02hhx:%02hhx IDX %02hhx CSN %02hhx:%02hhx:%02hhx\n",
@@ -745,7 +752,7 @@ int trDDA(EMVCommandChannel channel, bool decodeTLV, struct tlvdb *tlv) {
     if (!icc_pk) {
         emv_pk_free(pk);
         emv_pk_free(issuer_pk);
-        PrintAndLogEx(WARNING, "Error: ICC certificate not found. Exit.");
+        PrintAndLogEx(ERR, "Error: ICC certificate not found. Exit.");
         return 2;
     }
     PrintAndLogEx(SUCCESS, "ICC PK recovered. RID %02hhx:%02hhx:%02hhx:%02hhx:%02hhx IDX %02hhx CSN %02hhx:%02hhx:%02hhx\n",
@@ -790,7 +797,7 @@ int trDDA(EMVCommandChannel channel, bool decodeTLV, struct tlvdb *tlv) {
 
         const struct tlvdb *atc_db = emv_pki_recover_atc_ex(icc_pk, tlv, true);
         if (!atc_db) {
-            PrintAndLogEx(WARNING, "Error: Can't recover IDN (ICC Dynamic Number)");
+            PrintAndLogEx(ERR, "Error: Can't recover IDN (ICC Dynamic Number)");
             emv_pk_free(pk);
             emv_pk_free(issuer_pk);
             emv_pk_free(icc_pk);
@@ -823,7 +830,7 @@ int trDDA(EMVCommandChannel channel, bool decodeTLV, struct tlvdb *tlv) {
             PrintAndLogEx(NORMAL, "SDAD verified OK. (Data Authentication Code: %02hhx:%02hhx)\n", dac_tlv->value[0], dac_tlv->value[1]);
             tlvdb_add(tlv, dac_db);
         } else {
-            PrintAndLogEx(WARNING, "Error: SSAD verify error");
+            PrintAndLogEx(ERR, "Error: SSAD verify error");
             emv_pk_free(pk);
             emv_pk_free(issuer_pk);
             emv_pk_free(icc_pk);
@@ -839,7 +846,7 @@ int trDDA(EMVCommandChannel channel, bool decodeTLV, struct tlvdb *tlv) {
 
         struct tlv *ddol_data_tlv = dol_process(ddol_tlv, tlv, 0);
         if (!ddol_data_tlv) {
-            PrintAndLogEx(WARNING, "Error: Can't create DDOL TLV");
+            PrintAndLogEx(ERR, "Error: Can't create DDOL TLV");
             emv_pk_free(pk);
             emv_pk_free(issuer_pk);
             emv_pk_free(icc_pk);
@@ -851,7 +858,7 @@ int trDDA(EMVCommandChannel channel, bool decodeTLV, struct tlvdb *tlv) {
         PrintAndLogEx(NORMAL, "\n* Internal Authenticate");
         int res = EMVInternalAuthenticate(channel, true, (uint8_t *)ddol_data_tlv->value, ddol_data_tlv->len, buf, sizeof(buf), &len, &sw, NULL);
         if (res) {
-            PrintAndLogEx(WARNING, "Internal Authenticate error(%d): %4x. Exit...", res, sw);
+            PrintAndLogEx(ERR, "Internal Authenticate error(%d): %4x. Exit...", res, sw);
             free(ddol_data_tlv);
             emv_pk_free(pk);
             emv_pk_free(issuer_pk);
@@ -862,7 +869,7 @@ int trDDA(EMVCommandChannel channel, bool decodeTLV, struct tlvdb *tlv) {
         struct tlvdb *dda_db = NULL;
         if (buf[0] == 0x80) {
             if (len < 3) {
-                PrintAndLogEx(WARNING, "Error: Internal Authenticate format1 parsing error. length=%d", len);
+                PrintAndLogEx(WARNING, "Warning: Internal Authenticate format1 parsing error. length=%d", len);
             } else {
                 // parse response 0x80
                 struct tlvdb *t80 = tlvdb_parse_multi(buf, len);
@@ -882,7 +889,7 @@ int trDDA(EMVCommandChannel channel, bool decodeTLV, struct tlvdb *tlv) {
         } else {
             dda_db = tlvdb_parse_multi(buf, len);
             if (!dda_db) {
-                PrintAndLogEx(WARNING, "Error: Can't parse Internal Authenticate result as TLV");
+                PrintAndLogEx(ERR, "Error: Can't parse Internal Authenticate result as TLV");
                 free(ddol_data_tlv);
                 emv_pk_free(pk);
                 emv_pk_free(issuer_pk);
@@ -898,7 +905,7 @@ int trDDA(EMVCommandChannel channel, bool decodeTLV, struct tlvdb *tlv) {
         struct tlvdb *idn_db = emv_pki_recover_idn_ex(icc_pk, dda_db, ddol_data_tlv, true);
         free(ddol_data_tlv);
         if (!idn_db) {
-            PrintAndLogEx(WARNING, "Error: Can't recover IDN (ICC Dynamic Number)");
+            PrintAndLogEx(ERR, "Error: Can't recover IDN (ICC Dynamic Number)");
             tlvdb_free(dda_db);
             emv_pk_free(pk);
             emv_pk_free(issuer_pk);
@@ -935,20 +942,20 @@ int trCDA(struct tlvdb *tlv, struct tlvdb *ac_tlv, struct tlv *pdol_data_tlv, st
 
     struct emv_pk *pk = get_ca_pk(tlv);
     if (!pk) {
-        PrintAndLogEx(WARNING, "Error: Key not found. Exit.");
+        PrintAndLogEx(ERR, "Error: Key not found. Exit.");
         return 2;
     }
 
     const struct tlv *sda_tlv = tlvdb_get(tlv, 0x21, NULL);
     if (!sda_tlv || sda_tlv->len < 1) {
-        PrintAndLogEx(WARNING, "Error: Can't find input list for Offline Data Authentication. Exit.");
+        PrintAndLogEx(ERR, "Error: Can't find input list for Offline Data Authentication. Exit.");
         emv_pk_free(pk);
         return 3;
     }
 
     struct emv_pk *issuer_pk = emv_pki_recover_issuer_cert(pk, tlv);
     if (!issuer_pk) {
-        PrintAndLogEx(WARNING, "Error: Issuer certificate not found. Exit.");
+        PrintAndLogEx(ERR, "Error: Issuer certificate not found. Exit.");
         emv_pk_free(pk);
         return 2;
     }
@@ -966,7 +973,7 @@ int trCDA(struct tlvdb *tlv, struct tlvdb *ac_tlv, struct tlv *pdol_data_tlv, st
 
     struct emv_pk *icc_pk = emv_pki_recover_icc_cert(issuer_pk, tlv, sda_tlv);
     if (!icc_pk) {
-        PrintAndLogEx(WARNING, "Error: ICC certificate not found. Exit.");
+        PrintAndLogEx(ERR, "Error: ICC certificate not found. Exit.");
         emv_pk_free(pk);
         emv_pk_free(issuer_pk);
         return 2;
@@ -989,7 +996,7 @@ int trCDA(struct tlvdb *tlv, struct tlvdb *ac_tlv, struct tlv *pdol_data_tlv, st
         PrintAndLogEx(NORMAL, "SSAD verified OK. (%02hhx:%02hhx)", dac_tlv->value[0], dac_tlv->value[1]);
         tlvdb_add(tlv, dac_db);
     } else {
-        PrintAndLogEx(WARNING, "Error: SSAD verify error");
+        PrintAndLogEx(ERR, "Error: SSAD verify error");
         emv_pk_free(pk);
         emv_pk_free(issuer_pk);
         emv_pk_free(icc_pk);
diff --git a/client/emv/emvcore.h b/client/emv/emvcore.h
index 219c3f9b5..3faae283d 100644
--- a/client/emv/emvcore.h
+++ b/client/emv/emvcore.h
@@ -45,15 +45,6 @@ enum TransactionType {
 };
 extern const char *TransactionTypeStr[];
 
-typedef struct {
-    uint8_t CLA;
-    uint8_t INS;
-    uint8_t P1;
-    uint8_t P2;
-    uint8_t Lc;
-    uint8_t *data;
-} sAPDU;
-
 enum CardPSVendor {
     CV_NA,
     CV_VISA,
diff --git a/client/flash.c b/client/flash.c
index 3a1350c7c..f2f0b526a 100644
--- a/client/flash.c
+++ b/client/flash.c
@@ -12,18 +12,13 @@
 
 #define FLASH_START            0x100000
 
-#ifdef HAS_512_FLASH
-# define FLASH_SIZE             (512*1024)
-#else
-# define FLASH_SIZE             (256*1024)
-#endif
-
-#define FLASH_END              (FLASH_START + FLASH_SIZE)
 #define BOOTLOADER_SIZE        0x2000
 #define BOOTLOADER_END         (FLASH_START + BOOTLOADER_SIZE)
 
 #define BLOCK_SIZE             0x200
 
+#define FLASHER_VERSION        BL_VERSION_1_0_0
+
 static const uint8_t elf_ident[] = {
     0x7f, 'E', 'L', 'F',
     ELFCLASS32,
@@ -31,9 +26,45 @@ static const uint8_t elf_ident[] = {
     EV_CURRENT
 };
 
+static int chipid_to_mem_avail(uint32_t iChipID) {
+    int mem_avail = 0;
+    switch ((iChipID & 0xF00) >> 8) {
+        case 0:
+            mem_avail = 0;
+            break;
+        case 1:
+            mem_avail = 8;
+            break;
+        case 2:
+            mem_avail = 16;
+            break;
+        case 3:
+            mem_avail = 32;
+            break;
+        case 5:
+            mem_avail = 64;
+            break;
+        case 7:
+            mem_avail = 128;
+            break;
+        case 9:
+            mem_avail = 256;
+            break;
+        case 10:
+            mem_avail = 512;
+            break;
+        case 12:
+            mem_avail = 1024;
+            break;
+        case 14:
+            mem_avail = 2048;
+    }
+    return mem_avail;
+}
+
 // Turn PHDRs into flasher segments, checking for PHDR sanity and merging adjacent
 // unaligned segments if needed
-static int build_segs_from_phdrs(flash_file_t *ctx, FILE *fd, Elf32_Phdr *phdrs, uint16_t num_phdrs) {
+static int build_segs_from_phdrs(flash_file_t *ctx, FILE *fd, Elf32_Phdr *phdrs, uint16_t num_phdrs, uint32_t flash_end) {
     Elf32_Phdr *phdr = phdrs;
     flash_seg_t *seg;
     uint32_t last_end = 0;
@@ -77,11 +108,11 @@ static int build_segs_from_phdrs(flash_file_t *ctx, FILE *fd, Elf32_Phdr *phdrs,
             PrintAndLogEx(ERR, "Error: PHDRs not sorted or overlap");
             return -1;
         }
-        if (paddr < FLASH_START || (paddr + filesz) > FLASH_END) {
+        if (paddr < FLASH_START || (paddr + filesz) > flash_end) {
             PrintAndLogEx(ERR, "Error: PHDR is not contained in Flash");
             return -1;
         }
-        if (vaddr >= FLASH_START && vaddr < FLASH_END && (flags & PF_W)) {
+        if (vaddr >= FLASH_START && vaddr < flash_end && (flags & PF_W)) {
             PrintAndLogEx(ERR, "Error: Flash VMA segment is writable");
             return -1;
         }
@@ -153,7 +184,7 @@ static int build_segs_from_phdrs(flash_file_t *ctx, FILE *fd, Elf32_Phdr *phdrs,
 }
 
 // Sanity check segments and check for bootloader writes
-static int check_segs(flash_file_t *ctx, int can_write_bl) {
+static int check_segs(flash_file_t *ctx, int can_write_bl, uint32_t flash_end) {
     for (int i = 0; i < ctx->num_segs; i++) {
         flash_seg_t *seg = &ctx->segments[i];
 
@@ -165,7 +196,7 @@ static int check_segs(flash_file_t *ctx, int can_write_bl) {
             PrintAndLogEx(ERR, "Error: Segment is outside of flash bounds");
             return -1;
         }
-        if (seg->start + seg->length > FLASH_END) {
+        if (seg->start + seg->length > flash_end) {
             PrintAndLogEx(ERR, "Error: Segment is outside of flash bounds");
             return -1;
         }
@@ -182,17 +213,17 @@ static int check_segs(flash_file_t *ctx, int can_write_bl) {
 }
 
 // Load an ELF file and prepare it for flashing
-int flash_load(flash_file_t *ctx, const char *name, int can_write_bl) {
+int flash_load(flash_file_t *ctx, const char *name, int can_write_bl, int flash_size) {
     FILE *fd;
     Elf32_Ehdr ehdr;
     Elf32_Phdr *phdrs = NULL;
     uint16_t num_phdrs;
+    uint32_t flash_end  = FLASH_START + flash_size;
     int res;
 
     fd = fopen(name, "rb");
     if (!fd) {
         PrintAndLogEx(ERR, _RED_("Could not open file") "%s  >>> ", name);
-        perror(NULL);
         goto fail;
     }
 
@@ -240,10 +271,10 @@ int flash_load(flash_file_t *ctx, const char *name, int can_write_bl) {
         goto fail;
     }
 
-    res = build_segs_from_phdrs(ctx, fd, phdrs, num_phdrs);
+    res = build_segs_from_phdrs(ctx, fd, phdrs, num_phdrs, flash_end);
     if (res < 0)
         goto fail;
-    res = check_segs(ctx, can_write_bl);
+    res = check_segs(ctx, can_write_bl, flash_end);
     if (res < 0)
         goto fail;
 
@@ -346,9 +377,19 @@ static int wait_for_ack(PacketResponseNG *ack) {
     return 0;
 }
 
+static void flash_suggest_update_bootloader(void) {
+    PrintAndLogEx(ERR, _RED_("It is recommended that you first update your bootloader alone,"));
+    PrintAndLogEx(ERR, _RED_("reboot the Proxmark3 then only update the main firmware") "\n");
+}
+
+static void flash_suggest_update_flasher(void) {
+    PrintAndLogEx(ERR, _RED_("It is recommended that you first update your flasher"));
+}
+
 // Go into flashing mode
-int flash_start_flashing(int enable_bl_writes, char *serial_port_name, uint32_t * chipinfo) {
+int flash_start_flashing(int enable_bl_writes, char *serial_port_name, uint32_t *max_allowed) {
     uint32_t state;
+    uint32_t chipinfo = 0;
 
     if (enter_bootloader(serial_port_name) < 0)
         return -1;
@@ -360,23 +401,71 @@ int flash_start_flashing(int enable_bl_writes, char *serial_port_name, uint32_t
         SendCommandBL(CMD_CHIP_INFO, 0, 0, 0, NULL, 0);
         PacketResponseNG resp;
         WaitForResponse(CMD_CHIP_INFO, &resp);
-        *chipinfo = resp.oldarg[0];
+        chipinfo = resp.oldarg[0];
     }
 
+    int version = BL_VERSION_INVALID;
+    if (state & DEVICE_INFO_FLAG_UNDERSTANDS_VERSION) {
+        SendCommandBL(CMD_BL_VERSION, 0, 0, 0, NULL, 0);
+        PacketResponseNG resp;
+        WaitForResponse(CMD_BL_VERSION, &resp);
+        version = resp.oldarg[0];
+        if ((BL_VERSION_MAJOR(version) < BL_VERSION_FIRST_MAJOR) || (BL_VERSION_MAJOR(version) > BL_VERSION_LAST_MAJOR)) {
+            // version info seems fishy
+            version = BL_VERSION_INVALID;
+            PrintAndLogEx(ERR, _RED_("Note: Your bootloader reported an invalid version number"));
+            flash_suggest_update_bootloader();
+            //
+        } else if (BL_VERSION_MAJOR(version) < BL_VERSION_MAJOR(FLASHER_VERSION)) {
+            PrintAndLogEx(ERR, _RED_("Note: Your bootloader reported a version older than this flasher"));
+            flash_suggest_update_bootloader();
+        } else if (BL_VERSION_MAJOR(version) > BL_VERSION_MAJOR(FLASHER_VERSION)) {
+            PrintAndLogEx(ERR, _RED_("Note: Your bootloader is more recent than this flasher"));
+            flash_suggest_update_flasher();
+        }
+    } else {
+        PrintAndLogEx(ERR, _RED_("Note: Your bootloader does not understand the new CMD_BL_VERSION command"));
+        flash_suggest_update_bootloader();
+    }
+
+    uint32_t flash_end = FLASH_START + AT91C_IFLASH_PAGE_SIZE * AT91C_IFLASH_NB_OF_PAGES / 2;
+    *max_allowed = 256;
+
+    int mem_avail = chipid_to_mem_avail(chipinfo);
+    if (mem_avail != 0) {
+        PrintAndLogEx(NORMAL, "Available memory on this board: %uK bytes\n", mem_avail);
+        if (mem_avail > 256) {
+            if (BL_VERSION_MAJOR(version) < BL_VERSION_MAJOR(BL_VERSION_1_0_0)) {
+                PrintAndLogEx(ERR, _RED_("Your bootloader does not support writing above 256k"));
+                flash_suggest_update_bootloader();
+            } else {
+                flash_end = FLASH_START + AT91C_IFLASH_PAGE_SIZE * AT91C_IFLASH_NB_OF_PAGES;
+                *max_allowed = mem_avail;
+            }
+        }
+    } else {
+        PrintAndLogEx(NORMAL, "Available memory on this board: "_RED_("UNKNOWN")"\n");
+        PrintAndLogEx(ERR, _RED_("Note: Your bootloader does not understand the new CHIP_INFO command"));
+        flash_suggest_update_bootloader();
+    }
+
+    if (enable_bl_writes) {
+        PrintAndLogEx(INFO, "Permitted flash range: 0x%08x-0x%08x", FLASH_START, flash_end);
+    } else {
+        PrintAndLogEx(INFO, "Permitted flash range: 0x%08x-0x%08x", BOOTLOADER_END, flash_end);
+    }
     if (state & DEVICE_INFO_FLAG_UNDERSTANDS_START_FLASH) {
-        // This command is stupid. Why the heck does it care which area we're
-        // flashing, as long as it's not the bootloader area? The mind boggles.
         PacketResponseNG resp;
 
         if (enable_bl_writes) {
-            SendCommandBL(CMD_START_FLASH, FLASH_START, FLASH_END, START_FLASH_MAGIC, NULL, 0);
+            SendCommandBL(CMD_START_FLASH, FLASH_START, flash_end, START_FLASH_MAGIC, NULL, 0);
         } else {
-            SendCommandBL(CMD_START_FLASH, BOOTLOADER_END, FLASH_END, 0, NULL, 0);
+            SendCommandBL(CMD_START_FLASH, BOOTLOADER_END, flash_end, 0, NULL, 0);
         }
         return wait_for_ack(&resp);
     } else {
         PrintAndLogEx(ERR, _RED_("Note: Your bootloader does not understand the new START_FLASH command"));
-        PrintAndLogEx(ERR, _RED_("It is recommended that you update your bootloader") "\n");
+        flash_suggest_update_bootloader();
     }
     return 0;
 }
diff --git a/client/flash.h b/client/flash.h
index 7140de673..4f05029dc 100644
--- a/client/flash.h
+++ b/client/flash.h
@@ -37,11 +37,10 @@ typedef struct {
     flash_seg_t *segments;
 } flash_file_t;
 
-int flash_load(flash_file_t *ctx, const char *name, int can_write_bl);
-int flash_start_flashing(int enable_bl_writes, char *serial_port_name, uint32_t *chipid);
+int flash_load(flash_file_t *ctx, const char *name, int can_write_bl, int flash_size);
+int flash_start_flashing(int enable_bl_writes, char *serial_port_name, uint32_t *max_allowed);
 int flash_write(flash_file_t *ctx);
 void flash_free(flash_file_t *ctx);
 int flash_stop_flashing(void);
-
 #endif
 
diff --git a/client/flasher.c b/client/flasher.c
index 94e2ef6ba..97e3d7a1c 100644
--- a/client/flasher.c
+++ b/client/flasher.c
@@ -21,62 +21,34 @@
 #include "ui.h"
 
 #define MAX_FILES 4
+#define ONE_KB 1024
 
 static void usage(char *argv0) {
-    PrintAndLogEx(NORMAL, "Usage:   %s <port> [-b] image.elf [image.elf...]\n", argv0);
-    PrintAndLogEx(NORMAL, "\t-b\tEnable flashing of bootloader area (DANGEROUS)\n");
-    PrintAndLogEx(NORMAL, "\nExample:\n\n\t %s "SERIAL_PORT_EXAMPLE_H" armsrc/obj/fullimage.elf", argv0);
+    PrintAndLogEx(NORMAL, "Usage:   %s <port> [-b] image.elf [image.elf...]", argv0);
+    PrintAndLogEx(NORMAL, "         %s <port> -i\n", argv0);
+    PrintAndLogEx(NORMAL, "\t-b\tEnable flashing of bootloader area (DANGEROUS)");
+    PrintAndLogEx(NORMAL, "\t-i\tProbe the connected Proxmark3 to retrieve its memory size");
+    PrintAndLogEx(NORMAL, "\nExamples:\n\t %s "SERIAL_PORT_EXAMPLE_H" -i", argv0);
+    PrintAndLogEx(NORMAL, "\t %s "SERIAL_PORT_EXAMPLE_H" armsrc/obj/fullimage.elf", argv0);
 #ifdef __linux__
-    PrintAndLogEx(NORMAL, "\nNote (Linux): if the flasher gets stuck in 'Waiting for Proxmark3 to reappear on <DEVICE>',");
-    PrintAndLogEx(NORMAL, "              you need to blacklist Proxmark3 for modem-manager - see wiki for more details:\n");
-    PrintAndLogEx(NORMAL, "              https://github.com/Proxmark/proxmark3/wiki/Gentoo Linux\n");
-    PrintAndLogEx(NORMAL, "              https://github.com/Proxmark/proxmark3/wiki/Ubuntu Linux\n");
-    PrintAndLogEx(NORMAL, "              https://github.com/Proxmark/proxmark3/wiki/OSX\n");
+    PrintAndLogEx(NORMAL, "\nNote (Linux):\nif the flasher gets stuck in 'Waiting for Proxmark3 to reappear on <DEVICE>',");
+    PrintAndLogEx(NORMAL, "you need to blacklist Proxmark3 for modem-manager - see documentation for more details:");
+    PrintAndLogEx(NORMAL, "* https://github.com/RfidResearchGroup/proxmark3/blob/master/doc/md/Installation_Instructions/ModemManager-Must-Be-Discarded.md");
+    PrintAndLogEx(NORMAL, "\nMore info on flashing procedure from the official Proxmark3 wiki:");
+    PrintAndLogEx(NORMAL, "* https://github.com/Proxmark/proxmark3/wiki/Gentoo%%20Linux");
+    PrintAndLogEx(NORMAL, "* https://github.com/Proxmark/proxmark3/wiki/Ubuntu%%20Linux");
+    PrintAndLogEx(NORMAL, "* https://github.com/Proxmark/proxmark3/wiki/OSX\n");
 #endif
 }
 
-int chipid_to_mem_avail(uint32_t iChipID) {
-    int mem_avail = 0;
-    switch ((iChipID & 0xF00) >> 8) {
-        case 0:
-            mem_avail = 0;
-            break;
-        case 1:
-            mem_avail = 8;
-            break;
-        case 2:
-            mem_avail = 16;
-            break;
-        case 3:
-            mem_avail = 32;
-            break;
-        case 5:
-            mem_avail = 64;
-            break;
-        case 7:
-            mem_avail = 128;
-            break;
-        case 9:
-            mem_avail = 256;
-            break;
-        case 10:
-            mem_avail = 512;
-            break;
-        case 12:
-            mem_avail = 1024;
-            break;
-        case 14:
-            mem_avail = 2048;
-    }
-    return mem_avail;
-}
-
 int main(int argc, char **argv) {
     int can_write_bl = 0;
     int num_files = 0;
     int res;
+    int ret = 0;
     flash_file_t files[MAX_FILES];
-
+    char *filenames[MAX_FILES];
+    bool info = false;
     memset(files, 0, sizeof(files));
 
     session.supports_colors = false;
@@ -97,16 +69,14 @@ int main(int argc, char **argv) {
         if (argv[i][0] == '-') {
             if (!strcmp(argv[i], "-b")) {
                 can_write_bl = 1;
+            } else if (!strcmp(argv[i], "-i")) {
+                info = true;
             } else {
                 usage(argv[0]);
                 return -1;
             }
         } else {
-            res = flash_load(&files[num_files], argv[i], can_write_bl);
-            if (res < 0)
-                return -1;
-
-            PrintAndLogEx(NORMAL, "");
+            filenames[num_files] = argv[i];
             num_files++;
         }
     }
@@ -120,36 +90,48 @@ int main(int argc, char **argv) {
         return -1;
     }
 
-    uint32_t chipid = 0;
-    res = flash_start_flashing(can_write_bl, serial_port_name, &chipid);
-    if (res < 0)
-        return -1;
-
-    int mem_avail = chipid_to_mem_avail(chipid);
-    if (mem_avail != 0) {
-        PrintAndLogEx(NORMAL, "Available memory on this board: %uK bytes\n", mem_avail);
-    } else {
-        PrintAndLogEx(NORMAL, "Available memory on this board: "_RED_("UNKNOWN")"\n");
-        PrintAndLogEx(ERR, _RED_("Note: Your bootloader does not understand the new CHIP_INFO command"));
-        PrintAndLogEx(ERR, _RED_("It is recommended that you update your bootloader") "\n");
+    uint32_t max_allowed = 0;
+    res = flash_start_flashing(can_write_bl, serial_port_name, &max_allowed);
+    if (res < 0) {
+        ret = -1;
+        goto finish;
     }
+
+    if (info)
+        goto finish;
+
+    for (int i = 0 ; i < num_files; ++i) {
+        res = flash_load(&files[i], filenames[i], can_write_bl, max_allowed * ONE_KB);
+        if (res < 0) {
+            ret = -1;
+            goto finish;
+        }
+        PrintAndLogEx(NORMAL, "");
+    }
+
     PrintAndLogEx(SUCCESS, "\n" _BLUE_("Flashing..."));
-// TODO check if enough space on Pm3 mem to write the given files
+
     for (int i = 0; i < num_files; i++) {
         res = flash_write(&files[i]);
-        if (res < 0)
-            return -1;
+        if (res < 0) {
+            ret = -1;
+            goto finish;
+        }
         flash_free(&files[i]);
         PrintAndLogEx(NORMAL, "\n");
     }
 
+finish:
     res = flash_stop_flashing();
     if (res < 0)
-        return -1;
+        ret = -1;
 
     CloseProxmark();
 
-    PrintAndLogEx(SUCCESS, _BLUE_("All done."));
-    PrintAndLogEx(SUCCESS, "\nHave a nice day!");
-    return 0;
+    if (ret == 0)
+        PrintAndLogEx(SUCCESS, _BLUE_("All done."));
+    else
+        PrintAndLogEx(ERR, "Aborted on error.");
+    PrintAndLogEx(NORMAL, "\nHave a nice day!");
+    return ret;
 }
diff --git a/client/fpga_compress.c b/client/fpga_compress.c
index 7cd4652e1..a030afe81 100644
--- a/client/fpga_compress.c
+++ b/client/fpga_compress.c
@@ -375,7 +375,7 @@ static void print_version_info_preamble(FILE *outfile, int num_infiles) {
     fprintf(outfile, "\n");
     fprintf(outfile, "\n");
     fprintf(outfile, "const int fpga_bitstream_num = %d;\n", num_infiles);
-    fprintf(outfile, "const char* const fpga_version_information[%d] = {\n", num_infiles);
+    fprintf(outfile, "const char *const fpga_version_information[%d] = {\n", num_infiles);
 }
 
 static int generate_fpga_version_info(FILE *infile[], char *infile_names[], int num_infiles, FILE *outfile) {
@@ -386,7 +386,7 @@ static int generate_fpga_version_info(FILE *infile[], char *infile_names[], int
 
     for (int i = 0; i < num_infiles; i++) {
         FpgaGatherVersion(infile[i], infile_names[i], version_string, sizeof(version_string));
-        fprintf(outfile, "\t\" %s\"", version_string);
+        fprintf(outfile, "    \" %s\"", version_string);
         if (i != num_infiles - 1) {
             fprintf(outfile, ",");
         }
diff --git a/client/loclass/elite_crack.c b/client/loclass/elite_crack.c
index a461557b1..5b019d8e0 100644
--- a/client/loclass/elite_crack.c
+++ b/client/loclass/elite_crack.c
@@ -517,7 +517,7 @@ int bruteforceDump(uint8_t dump[], size_t dumpsize, uint16_t keytable[]) {
         first16bytes[i] = keytable[i] & 0xFF;
 
         if (!(keytable[i] & CRACKED))
-            PrintAndLogDevice(WARNING, "error, we are missing byte %d, custom key calculation will fail...", i);
+            PrintAndLogDevice(WARNING, "Warning: we are missing byte %d, custom key calculation will fail...", i);
     }
     errors += calculateMasterKey(first16bytes, NULL);
     return errors;
@@ -541,7 +541,7 @@ int bruteforceFile(const char *filename, uint16_t keytable[]) {
     fseek(f, 0, SEEK_SET);
 
     if (fsize <= 0) {
-        PrintAndLogDevice(WARNING, "Error, when getting filesize");
+        PrintAndLogDevice(ERR, "Error, when getting filesize");
         fclose(f);
         return 1;
     }
@@ -557,7 +557,7 @@ int bruteforceFile(const char *filename, uint16_t keytable[]) {
     fclose(f);
 
     if (bytes_read < fsize) {
-        PrintAndLogDevice(WARNING, "Error, could only read %d bytes (should be %d)", bytes_read, fsize);
+        PrintAndLogDevice(WARNING, "Warning: could only read %d bytes (should be %d)", bytes_read, fsize);
     }
 
     uint8_t res = bruteforceDump(dump, fsize, keytable);
@@ -612,7 +612,7 @@ static int _testBruteforce() {
         } else if (fileExists("client/loclass/iclass_dump.bin")) {
             errors |= bruteforceFile("client/loclass/iclass_dump.bin", keytable);
         } else {
-            PrintAndLogDevice(WARNING, "Error: The file iclass_dump.bin was not found!");
+            PrintAndLogDevice(ERR, "Error: The file iclass_dump.bin was not found!");
         }
     }
     return errors;
@@ -627,14 +627,14 @@ static int _test_iclass_key_permutation() {
     permutekey_rev(testcase_output, testcase_output_rev);
 
     if (memcmp(testcase_output, testcase_output_correct, 8) != 0) {
-        PrintAndLogDevice(WARNING, "Error with iclass key permute!");
+        PrintAndLogDevice(ERR, "Error with iclass key permute!");
         printarr("testcase_output", testcase_output, 8);
         printarr("testcase_output_correct", testcase_output_correct, 8);
         return 1;
 
     }
     if (memcmp(testcase, testcase_output_rev, 8) != 0) {
-        PrintAndLogDevice(WARNING, "Error with reverse iclass key permute");
+        PrintAndLogDevice(ERR, "Error with reverse iclass key permute");
         printarr("testcase", testcase, 8);
         printarr("testcase_output_rev", testcase_output_rev, 8);
         return 1;
@@ -651,7 +651,7 @@ static int _testHash1() {
     hash1(csn, k);
 
     if (memcmp(k, expected, 8) != 0) {
-        PrintAndLogDevice(WARNING, "Error with hash1!");
+        PrintAndLogDevice(ERR, "Error with hash1!");
         printarr("calculated", k, 8);
         printarr("expected", expected, 8);
         return 1;
diff --git a/client/loclass/loclass information.txt b/client/loclass/loclass_information.txt
similarity index 93%
rename from client/loclass/loclass information.txt
rename to client/loclass/loclass_information.txt
index b01e447b0..307d28203 100644
--- a/client/loclass/loclass information.txt	
+++ b/client/loclass/loclass_information.txt
@@ -4,7 +4,7 @@ iclass_dump.bin
 ===============
 The file iclass_dump.bin contains CSN's mac results from 128 CSNs.
 Hence when running the test mode, 
-	'hf iclass loclass t'
+    'hf iclass loclass t'
 it shows a long output from the bruteforce test.
 
 iclass_key.bin
diff --git a/client/lualibs/hf_reader.lua b/client/lualibs/hf_reader.lua
index b99b521da..ba7e3a17c 100644
--- a/client/lualibs/hf_reader.lua
+++ b/client/lualibs/hf_reader.lua
@@ -15,10 +15,10 @@ local reader15693 = require('read15')
 -- @return if successfull: an table containing card info
 -- @return if unsuccessfull : nil, error
 local function waitForTag()
-    print("Waiting for card... press any key to quit")
+    print("Waiting for card... press Enter to quit")
     local readers = {reader14443A, reader14443B, reader15693}
     local i = 0;
-    while not core.ukbhit() do
+    while not core.kbd_enter_pressed() do
         i = (i % 3) +1
         r = readers[i]
         print("Reading with ",i)
diff --git a/client/lualibs/read14a.lua b/client/lualibs/read14a.lua
index 5cdf5a2e4..3bf243235 100644
--- a/client/lualibs/read14a.lua
+++ b/client/lualibs/read14a.lua
@@ -60,7 +60,7 @@ local function parse14443a(data)
         uint8_t sak;
         uint8_t ats_len;
         uint8_t ats[256];
-    } __attribute__((__packed__)) iso14a_card_select_t;
+    } PACKED iso14a_card_select_t;
     --]]
 
     local count, uid, uidlen, atqa, sak, ats_len, ats = bin.unpack('H10CH2CCH', data)
@@ -121,8 +121,8 @@ end
 -- @return if successfull: an table containing card info
 -- @return if unsuccessfull : nil, error
 local function waitFor14443a()
-    print('Waiting for card... press any key to quit')
-    while not core.ukbhit() do
+    print('Waiting for card... press Enter to quit')
+    while not core.kbd_enter_pressed() do
         res, err = read14443a()
         if res then return res end
         -- err means that there was no response from card
diff --git a/client/lualibs/read14b.lua b/client/lualibs/read14b.lua
index 0421bee5b..67d2bd469 100644
--- a/client/lualibs/read14b.lua
+++ b/client/lualibs/read14b.lua
@@ -39,7 +39,7 @@ local function parse1443b(data)
         uint8_t atqb[7];
         uint8_t chipid;
         uint8_t cid;
-    } __attribute__((__packed__)) iso14b_card_select_t;
+    } PACKED iso14b_card_select_t;
 
     --]]
 
@@ -99,8 +99,8 @@ end
 -- @return if successfull: an table containing card info
 -- @return if unsuccessfull : nil, error
 local function waitFor14443b()
-    print('Waiting for card... press any key to quit')
-    while not core.ukbhit() do
+    print('Waiting for card... press Enter to quit')
+    while not core.kbd_enter_pressed() do
         res, err = read14443b(false)
         if res then return res end
         -- err means that there was no response from card
diff --git a/client/lualibs/read15.lua b/client/lualibs/read15.lua
index 0b9a6476e..a70641186 100644
--- a/client/lualibs/read15.lua
+++ b/client/lualibs/read15.lua
@@ -133,8 +133,8 @@ end
 -- @return if successfull: an table containing card info
 -- @return if unsuccessfull : nil, error
 local function waitFor15693()
-    print('Waiting for card... press any key to quit')
-    while not core.ukbhit() do
+    print('Waiting for card... press Enter to quit')
+    while not core.kbd_enter_pressed() do
         res, err = read15693()
         if res then return res end
         -- err means that there was no response from card
diff --git a/client/mifare/mifarehost.c b/client/mifare/mifarehost.c
index 893664e86..100b595ce 100644
--- a/client/mifare/mifarehost.c
+++ b/client/mifare/mifarehost.c
@@ -28,9 +28,7 @@ int mfDarkside(uint8_t blockno, uint8_t key_type, uint64_t *key) {
         SendCommandMIX(CMD_READER_MIFARE, arg0, blockno, key_type, NULL, 0);
 
         //flush queue
-        while (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        while (kbd_enter_pressed()) {
             return PM3_EOPABORTED;
         }
 
@@ -38,9 +36,7 @@ int mfDarkside(uint8_t blockno, uint8_t key_type, uint64_t *key) {
         while (true) {
             printf(".");
             fflush(stdout);
-            if (ukbhit()) {
-                int gc = getchar();
-                (void)gc;
+            if (kbd_enter_pressed()) {
                 return PM3_EOPABORTED;
             }
 
@@ -918,7 +914,7 @@ int detect_classic_prng(void) {
 
     // if select tag failed.
     if (resp.oldarg[0] == 0) {
-        PrintAndLogEx(WARNING, "error:  selecting tag failed,  can't detect prng\n");
+        PrintAndLogEx(ERR, "error:  selecting tag failed,  can't detect prng\n");
         return PM3_ERFTRANS;
     }
     if (!WaitForResponseTimeout(CMD_ACK, &respA, 2500)) {
@@ -928,7 +924,7 @@ int detect_classic_prng(void) {
 
     // check respA
     if (respA.oldarg[0] != 4) {
-        PrintAndLogEx(WARNING, "PRNG data error: Wrong length: %d", respA.oldarg[0]);
+        PrintAndLogEx(ERR, "PRNG data error: Wrong length: %d", respA.oldarg[0]);
         return PM3_ESOFT;
     }
 
@@ -955,9 +951,7 @@ int detect_classic_nackbug(bool verbose) {
     while (true) {
         printf(".");
         fflush(stdout);
-        if (ukbhit()) {
-            int gc = getchar();
-            (void)gc;
+        if (kbd_enter_pressed()) {
             return PM3_EOPABORTED;
         }
 
@@ -1002,7 +996,7 @@ int detect_classic_nackbug(bool verbose) {
                     PrintAndLogEx(SUCCESS, "No NACK bug detected");
                     return PM3_SUCCESS;
                 default :
-                    PrintAndLogEx(WARNING, "errorcode from device [%i]", ok);
+                    PrintAndLogEx(ERR, "errorcode from device [%i]", ok);
                     return PM3_EUNDEF;
             }
             break;
diff --git a/client/proxmark3.c b/client/proxmark3.c
index 480f39fa3..1d3a0e8b5 100644
--- a/client/proxmark3.c
+++ b/client/proxmark3.c
@@ -503,6 +503,9 @@ int main(int argc, char *argv[]) {
         CloseProxmark();
     }
 
+    if ((port != NULL) && (!session.pm3_present))
+        exit(EXIT_FAILURE);
+
     if (!session.pm3_present)
         PrintAndLogEx(INFO, "Running in " _YELLOW_("OFFLINE") "mode. Check \"%s -h\" if it's not what you want.\n", exec_name);
 
@@ -531,5 +534,5 @@ int main(int argc, char *argv[]) {
         CloseProxmark();
     }
 
-    exit(0);
+    exit(EXIT_SUCCESS);
 }
diff --git a/client/scripting.c b/client/scripting.c
index cfa361256..4a2ff28d6 100644
--- a/client/scripting.c
+++ b/client/scripting.c
@@ -223,7 +223,7 @@ static int l_GetFromBigBuf(lua_State *L) {
         return returnToLuaWithError(L, "Allocating memory failed");
     }
 
-    if (!GetFromDevice(BIG_BUF, data, len, startindex, NULL, 2500, false)) {
+    if (!GetFromDevice(BIG_BUF, data, len, startindex, NULL, 0, NULL, 2500, false)) {
         free(data);
         return returnToLuaWithError(L, "command execution time out");
     }
@@ -263,7 +263,7 @@ static int l_GetFromFlashMem(lua_State *L) {
         if (!data)
             return returnToLuaWithError(L, "Allocating memory failed");
 
-        if (!GetFromDevice(FLASH_MEM, data, len, startindex, NULL, -1, false)) {
+        if (!GetFromDevice(FLASH_MEM, data, len, startindex, NULL, 0, NULL, -1, false)) {
             free(data);
             return returnToLuaWithError(L, "command execution time out");
         }
@@ -411,8 +411,8 @@ static int l_foobar(lua_State *L) {
  * @param L
  * @return boolean, true if kbhit, false otherwise.
  */
-static int l_ukbhit(lua_State *L) {
-    lua_pushboolean(L, ukbhit() ? true : false);
+static int l_kbd_enter_pressed(lua_State *L) {
+    lua_pushboolean(L, kbd_enter_pressed() ? true : false);
     return 1;
 }
 
@@ -903,7 +903,7 @@ static int l_T55xx_readblock(lua_State *L) {
         // try reading the config block and verify that PWD bit is set before doing this!
         if (!override) {
 
-            if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, false, 0)) {
+            if (!AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, false, 0, 0)) {
                 return returnToLuaWithError(L, "Failed to read config block");
             }
 
@@ -920,7 +920,7 @@ static int l_T55xx_readblock(lua_State *L) {
         }
     }
 
-    if (!AquireData(usepage1, block, usepwd, password)) {
+    if (!AquireData(usepage1, block, usepwd, password, 0)) {
         return returnToLuaWithError(L, "Failed to aquire data from card");
     }
 
@@ -977,7 +977,7 @@ static int l_T55xx_detect(lua_State *L) {
 
     if (!useGB) {
 
-        isok = AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, usepwd, password);
+        isok = AquireData(T55x7_PAGE0, T55x7_CONFIGURATION_BLOCK, usepwd, password, 0);
         if (isok == false) {
             return returnToLuaWithError(L, "Failed to aquire LF signal data");
         }
@@ -1064,7 +1064,7 @@ int set_pm3_libraries(lua_State *L) {
         {"WaitForResponseTimeout",      l_WaitForResponseTimeout},
         {"mfDarkside",                  l_mfDarkside},
         {"foobar",                      l_foobar},
-        {"ukbhit",                      l_ukbhit},
+        {"kbd_enter_pressed",               l_kbd_enter_pressed},
         {"clearCommandBuffer",          l_clearCommandBuffer},
         {"console",                     l_CmdConsole},
         {"iso15693_crc",                l_iso15693_crc},
diff --git a/client/scripts/didump.lua b/client/scripts/didump.lua
index b0cc964ee..d5ac6f873 100644
--- a/client/scripts/didump.lua
+++ b/client/scripts/didump.lua
@@ -488,7 +488,7 @@ local function readtag(mfkey, aeskey )
 
     for blockNo = 0, numBlocks-1 do
 
-        if core.ukbhit() then
+        if core.kbd_enter_pressed() then
             print("[fail] aborted by user")
             return nil
         end
diff --git a/client/scripts/formatMifare.lua b/client/scripts/formatMifare.lua
index e200d92cb..7c6fd92c0 100644
--- a/client/scripts/formatMifare.lua
+++ b/client/scripts/formatMifare.lua
@@ -206,7 +206,7 @@ local function main(args)
             if x then core.console(cmd) end
         end
 
-        if core.ukbhit() then
+        if core.kbd_enter_pressed() then
             print('aborted by user')
             break
         end
diff --git a/client/scripts/legic.lua b/client/scripts/legic.lua
index 9637ef6c8..7e0f8d3be 100644
--- a/client/scripts/legic.lua
+++ b/client/scripts/legic.lua
@@ -2654,7 +2654,7 @@ function modifyMode()
                 local stamp=getSegmentStamp(inTAG.SEG[x])
                 print("Stamp : "..stamp)
                 stamp=str2bytes(stamp)
-                print("lenght: "..#stamp)
+                print("length: "..#stamp)
               end,
     ---
     -- calculate crc16
diff --git a/client/scripts/lf_bulk.lua b/client/scripts/lf_bulk.lua
index 5e8562ae7..54c60fb89 100644
--- a/client/scripts/lf_bulk.lua
+++ b/client/scripts/lf_bulk.lua
@@ -186,7 +186,7 @@ local function main(args)
     for cardnum = baseid, endid do
         local card = cardHex(cardnum, facility)
         print('Press enter to program card '..cardnum..':'..facility..' (hex: '..card..')')
-        --This would be better with 'press any key', but we'll take what we can get.
+        --This would be better with 'press Enter', but we'll take what we can get.
         io.read()
         core.console( ('lf hid clone %s'):format(card) )
     end
diff --git a/client/scripts/mfkeys.lua b/client/scripts/mfkeys.lua
index 7d4a726d6..65f50eb9c 100644
--- a/client/scripts/mfkeys.lua
+++ b/client/scripts/mfkeys.lua
@@ -208,7 +208,7 @@ local function perform_check(numsectors)
 
     for sector = 0, #keys do
         -- Check if user aborted
-        if core.ukbhit() then
+        if core.kbd_enter_pressed() then
             print('Aborted by user')
             break
         end
diff --git a/client/scripts/mifare_autopwn.lua b/client/scripts/mifare_autopwn.lua
index a9859ac25..7a04cd641 100644
--- a/client/scripts/mifare_autopwn.lua
+++ b/client/scripts/mifare_autopwn.lua
@@ -70,7 +70,7 @@ end
 -- @return if successfull: an table containing card info
 -- @return if unsuccessfull : nil, error
 local function wait_for_mifare()
-    while not core.ukbhit() do
+    while not core.kbd_enter_pressed() do
         res, err = lib14a.read()
         if res then return res end
         -- err means that there was no response from card
@@ -161,7 +161,7 @@ local function main(args)
 
     while not _exit do
         if print_message then
-            print('Waiting for card or press any key to stop')
+            print('Waiting for card or press Enter to stop')
             print_message = false
         end
         res, err = wait_for_mifare()
diff --git a/client/scripts/ndef_dump.lua b/client/scripts/ndef_dump.lua
index d162e35cf..856053207 100644
--- a/client/scripts/ndef_dump.lua
+++ b/client/scripts/ndef_dump.lua
@@ -239,9 +239,9 @@ local function main( args)
     for k,v in ipairs(blockData) do
 
 --        print(string.format('Block %02x: %02x %02x %02x %02x', k-1, string.byte(v, 1,4)))
-	print(string.format(' %02x | %s', k-1, v) )
+        print(string.format(' %02x | %s', k-1, v) )
     end
-   print('|---|-------------------|')
+    print('|---|-------------------|')
 
     local filename, err = utils.WriteDumpFile(info.uid, blockData)
     if err then return oops(err) end
diff --git a/client/scripts/read_pwd_mem.lua b/client/scripts/read_pwd_mem.lua
index 4aa2e6299..4a097d26d 100644
--- a/client/scripts/read_pwd_mem.lua
+++ b/client/scripts/read_pwd_mem.lua
@@ -81,18 +81,19 @@ local function main(args)
         if o == 'o' then offset = tonumber(a) end
 
         -- num of bytes to read
-        if o == 'l' then length = tonumber(a) end
+        if o == 'l' then
+            length = tonumber(a)
+            if length < 0 or length > 256 then
+                return oops('Error: Length is not valid. Must be less than 256')
+            end
+        end
 
         -- keylength
         if o == 'k' then keylength = tonumber(a); usedkey = true end
 
-        if o == 'm' then keylength =6; usedkey = true; offset = 0x3F000-0x6000; end
-        if o == 't' then keylength =4; usedkey = true; offset = 0x3F000-0x3000; end
-        if o == 'i' then keylength =8; usedkey = true; offset = 0x3F000-0x4000; end
-    end
-
-    if length < 0 or length > 256 then
-        return oops('Error: Length is not valid. Must be less than 256')
+        if o == 'm' then keylength = 6; usedkey = true; length = 8192; offset = 0x3F000-0x6000; end
+        if o == 't' then keylength = 4; usedkey = true; length = 4096; offset = 0x3F000-0x3000; end
+        if o == 'i' then keylength = 8; usedkey = true; length = 4096; offset = 0x3F000-0x4000; end
     end
 
     if (offset < 0) or (offset % 4 ~= 0) then
@@ -102,9 +103,7 @@ local function main(args)
     print('Memory offset', offset)
     print('Length       ', length)
     print('Key length   ', keylength)
-    print( string.rep('--',20) )
-
-    if usedkey then length = 4096 end
+    print( string.rep('--', 20) )
 
     data, err = core.GetFromFlashMem(offset, length)
     if err then return oops(err) end
@@ -116,7 +115,6 @@ local function main(args)
 
         local kl = keylength * 2
         for i = 1, keys do
-
             key  = string.sub(s, (i - 1) * kl + 1, i * kl )
             print(string.format('[%02d] %s',i, key))
         end
diff --git a/client/scripts/tnp3dump.lua b/client/scripts/tnp3dump.lua
index 00070eacc..0faadecaf 100644
--- a/client/scripts/tnp3dump.lua
+++ b/client/scripts/tnp3dump.lua
@@ -198,7 +198,7 @@ local function main(args)
 
         io.flush()
 
-        if core.ukbhit() then
+        if core.kbd_enter_pressed() then
             print("aborted by user")
             break
         end
diff --git a/client/scripts/tracetest.lua b/client/scripts/tracetest.lua
index e5943fd6a..44cd66b58 100644
--- a/client/scripts/tracetest.lua
+++ b/client/scripts/tracetest.lua
@@ -118,7 +118,7 @@ local function main(args)
 
         core.clearCommandBuffer()
 
-        if core.ukbhit() then
+        if core.kbd_enter_pressed() then
             print('aborted by user')
             break
         end
diff --git a/client/util.c b/client/util.c
index 568a022f1..f6f89e0ff 100644
--- a/client/util.c
+++ b/client/util.c
@@ -26,38 +26,45 @@ uint8_t g_debugMode = 0;
 #define MAX_BIN_BREAK_LENGTH   (3072+384+1)
 
 #ifndef _WIN32
-#include <termios.h>
-#include <sys/ioctl.h>
 #include <unistd.h>
-#include <stdarg.h>
+#include <fcntl.h>
 
-int ukbhit(void) {
-    int cnt = 0;
-    int error;
-    static struct termios Otty, Ntty;
-
-    if (tcgetattr(STDIN_FILENO, &Otty) == -1) return -1;
-
-    Ntty = Otty;
-
-    Ntty.c_iflag          = 0x0000;   // input mode
-    Ntty.c_oflag          = 0x0000;   // output mode
-    Ntty.c_lflag          &= ~ICANON; // control mode = raw
-    Ntty.c_cc[VMIN]       = 1;        // return if at least 1 character is in the queue
-    Ntty.c_cc[VTIME]      = 0;        // no timeout. Wait forever
-
-    if (0 == (error = tcsetattr(STDIN_FILENO, TCSANOW, &Ntty))) {  // set new attributes
-        error += ioctl(STDIN_FILENO, FIONREAD, &cnt);              // get number of characters available
-        error += tcsetattr(STDIN_FILENO, TCSANOW, &Otty);          // reset attributes
+int kbd_enter_pressed(void) {
+    int flags;
+    if ((flags = fcntl(STDIN_FILENO, F_GETFL, 0)) < 0) {
+        PrintAndLogEx(ERR, "fcntl failed in kbd_enter_pressed");
+        return -1;
     }
-    return (error == 0 ? cnt : -1);
+    //non-blocking
+    flags |= O_NONBLOCK;
+    if (fcntl(STDIN_FILENO, F_SETFL, flags) < 0) {
+        PrintAndLogEx(ERR, "fcntl failed in kbd_enter_pressed");
+        return -1;
+    }
+    int c;
+    int ret = 0;
+    do { //get all available chars
+        c = getchar();
+        ret |= c == '\n';
+    } while (c != EOF);
+    //blocking
+    flags &= ~O_NONBLOCK;
+    if (fcntl(STDIN_FILENO, F_SETFL, flags) < 0) {
+        PrintAndLogEx(ERR, "fcntl failed in kbd_enter_pressed");
+        return -1;
+    }
+    return ret;
 }
 
 #else
 
 #include <conio.h>
-int ukbhit(void) {
-    return kbhit();
+int kbd_enter_pressed(void) {
+    int ret = 0;
+    while (kbhit()) {
+        ret |= getch() == '\r';
+    }
+    return ret;
 }
 #endif
 
@@ -170,8 +177,8 @@ bool CheckStringIsHEXValue(const char *value) {
 void hex_to_buffer(const uint8_t *buf, const uint8_t *hex_data, const size_t hex_len, const size_t hex_max_len,
                    const size_t min_str_len, const size_t spaces_between, bool uppercase) {
 
-    if (buf == NULL ) return;
-    
+    if (buf == NULL) return;
+
     char *tmp = (char *)buf;
     size_t i;
     memset(tmp, 0x00, hex_max_len);
@@ -197,16 +204,16 @@ void hex_to_buffer(const uint8_t *buf, const uint8_t *hex_data, const size_t hex
 
 // printing and converting functions
 void print_hex(const uint8_t *data, const size_t len) {
-    if (data == NULL || len == 0 ) return;
-    
+    if (data == NULL || len == 0) return;
+
     for (size_t i = 0; i < len; i++)
         printf("%02x ", data[i]);
     printf("\n");
 }
 
 void print_hex_break(const uint8_t *data, const size_t len, uint8_t breaks) {
-    if (data == NULL || len == 0 ) return;
-    
+    if (data == NULL || len == 0) return;
+
     int rownum = 0;
     printf("[%02d] | ", rownum);
     for (size_t i = 0; i < len; ++i) {
diff --git a/client/util.h b/client/util.h
index e194bb5be..3f677df7a 100644
--- a/client/util.h
+++ b/client/util.h
@@ -20,94 +20,12 @@
 #include <time.h>
 #include "ui.h"     // PrintAndLog
 #include "commonutil.h"
+#include "common.h"
 
 #ifdef ANDROID
 #include <endian.h>
 #endif
 
-#ifndef ROTR
-# define ROTR(x,n) (((uintmax_t)(x) >> (n)) | ((uintmax_t)(x) << ((sizeof(x) * 8) - (n))))
-#endif
-#ifndef ROTL
-# define ROTL(x,n) (((uintmax_t)(x) << (n)) | ((uintmax_t)(x) >> ((sizeof(x) * 8) - (n))))
-#endif
-
-#ifndef MIN
-# define MIN(a, b) (((a) < (b)) ? (a) : (b))
-#endif
-#ifndef MAX
-# define MAX(a, b) (((a) > (b)) ? (a) : (b))
-#endif
-
-// endian change for 64bit
-#ifdef __GNUC__
-#ifndef BSWAP_64
-#define BSWAP_64(x) __builtin_bswap64(x)
-#endif
-#else
-#ifdef _MSC_VER
-#ifndef BSWAP_64
-#define BSWAP_64(x) _byteswap_uint64(x)
-#endif
-#else
-#ifndef BSWAP_64
-#define BSWAP_64(x) \
-    (((uint64_t)(x) << 56) | \
-     (((uint64_t)(x) << 40) & 0xff000000000000ULL) | \
-     (((uint64_t)(x) << 24) & 0xff0000000000ULL) | \
-     (((uint64_t)(x) << 8)  & 0xff00000000ULL) | \
-     (((uint64_t)(x) >> 8)  & 0xff000000ULL) | \
-     (((uint64_t)(x) >> 24) & 0xff0000ULL) | \
-     (((uint64_t)(x) >> 40) & 0xff00ULL) | \
-     ((uint64_t)(x)  >> 56))
-#endif
-#endif
-#endif
-
-// endian change for 32bit
-#ifdef __GNUC__
-#ifndef BSWAP_32
-#define BSWAP_32(x) __builtin_bswap32(x)
-#endif
-#else
-#ifdef _MSC_VER
-#ifndef BSWAP_32
-#define BSWAP_32(x) _byteswap_ulong(x)
-#endif
-#else
-#ifndef BSWAP_32
-# define BSWAP_32(x) \
-    ((((x) & 0xff000000) >> 24) | (((x) & 0x00ff0000) >>  8) | \
-     (((x) & 0x0000ff00) <<  8) | (((x) & 0x000000ff) << 24))
-#endif
-#endif
-#endif
-
-#define EVEN                        0
-#define ODD                         1
-
-// Nibble logic
-#ifndef NIBBLE_HIGH
-# define NIBBLE_HIGH(b) ( (b & 0xF0) >> 4 )
-#endif
-#ifndef NIBBLE_LOW
-# define NIBBLE_LOW(b)  ( b & 0x0F )
-#endif
-#ifndef CRUMB
-# define CRUMB(b,p)    (((b & (0x3 << p) ) >> p ) & 0xF)
-#endif
-#ifndef SWAP_NIBBLE
-# define SWAP_NIBBLE(b)  ( (NIBBLE_LOW(b)<< 4) | NIBBLE_HIGH(b))
-#endif
-
-// Binary Encoded Digit
-#ifndef BCD2DEC
-# define BCD2DEC(bcd) HornerScheme(bcd, 0x10, 10)
-#endif
-#ifndef DEC2BCD
-# define DEC2BCD(dec) HornerScheme(dec, 10, 0x10)
-#endif
-
 // used for save/load files
 #ifndef FILE_PATH_SIZE
 # define FILE_PATH_SIZE 1000
@@ -129,7 +47,7 @@
 
 uint8_t g_debugMode;
 
-int ukbhit(void);
+int kbd_enter_pressed(void);
 void AddLogLine(const char *fn, const char *data, const char *c);
 void AddLogHex(const char *fn, const char *extData, const uint8_t *data, const size_t len);
 void AddLogUint64(const char *fn, const char *data, const uint64_t value);
diff --git a/common/Makefile.common b/common/Makefile.common
index 2ab434562..282f242d9 100644
--- a/common/Makefile.common
+++ b/common/Makefile.common
@@ -63,7 +63,7 @@ DETECTED_OS=Windows
 endif
 
 # Also search prerequisites in the common directory (for usb.c), the fpga directory (for fpga.bit), and the zlib directory
-VPATH = . ../common ../common/crapto1 ../common/polarssl ../fpga ../zlib ../armsrc/Standalone ../uart
+VPATH = . ../common ../common/crapto1 ../common/mbedtls ../fpga ../zlib ../armsrc/Standalone ../uart
 
 INCLUDES = ../include/proxmark3.h ../include/at91sam7s512.h ../include/config_gpio.h ../include/pm3_cmd.h $(APP_INCLUDES)
 
diff --git a/common/Makefile.hal b/common/Makefile.hal
index ff3f83a4c..46154d244 100644
--- a/common/Makefile.hal
+++ b/common/Makefile.hal
@@ -1,10 +1,14 @@
 # Default platform if no platform specified
 PLATFORM?=PM3RDV4
-# Default standalone if no standalone specified
-# (you can set explicitly STANDALONE= to disable standalone modes)
-STANDALONE?=LF_SAMYRUN
 
-define KNOWN_DEFINITIONS
+# Standalone Mode info (path depends if make is called at top or from armsrc)
+-include armsrc/Standalone/Makefile.hal
+-include Standalone/Makefile.hal
+ifndef DEFAULT_STANDALONE
+$(error Could not find armsrc/Standalone/Makefile.hal)
+endif
+
+define KNOWN_PLATFORM_DEFINITIONS
 
 Known definitions:
 
@@ -30,39 +34,12 @@ Known definitions:
 | BTADDON         | Proxmark3 rdv4 BT add-on               |
 +----------------------------------------------------------+
 
-+==========================================================+
-| STANDALONE      | DESCRIPTION                            |
-+==========================================================+
-|                 | No standalone mode                     |
-+----------------------------------------------------------+
-| LF_SAMYRUN (def)| HID26 read/clone/sim                   |
-|                 | - Samy Kamkar                          |
-+----------------------------------------------------------+
-| LF_ICERUN       | standalone mode skeleton               |
-|                 | - iceman                               |
-+----------------------------------------------------------+
-| LF_PROXBRUTE    | HID ProxII bruteforce                  |
-|                 | - Brad Antoniewicz                     |
-+----------------------------------------------------------+
-| LF_HIDBRUTE     | HID corporate 1000 bruteforce          |
-|                 | - Federico dotta & Maurizio Agazzini   |
-+----------------------------------------------------------+
-| HF_YOUNG        | Mifare sniff/simulation                |
-|                 | - Craig Young                          |
-+----------------------------------------------------------+
-| HF_MATTYRUN     | Mifare sniff/clone                     |
-|                 | - Matías A. Ré Medina                  |
-+----------------------------------------------------------+
-| HF_COLIN        | Mifare ultra fast sniff/sim/clone      |
-|                 | - Colin Brigato                        |
-+----------------------------------------------------------+
-| HF_BOG          | 14a sniff with ULC/ULEV1/NTAG auth     |
-|                 | storing in flashmem - Bogito           |
-+----------------------------------------------------------+
+endef
 
+define HELP_DEFINITIONS
 Options to define platform, platform extras and/or standalone mode:
 (1) Run make with PLATFORM, PLATFORM_EXTRAS and/or STANDALONE as follows:
-make PLATFORM=PM3EASY STANDALONE=HF_COLIN
+make PLATFORM=PM3EASY STANDALONE=$(HELP_EXAMPLE_STANDALONE)
 
 (2) Save a file called Makefile.platform with contents:
 PLATFORM=PM3EASY
@@ -71,11 +48,17 @@ or if you have a Proxmark 3 RDV4 with the BT add-on:
 PLATFORM=PM3RDV4
 PLATFORM_EXTRAS=BTADDON
 
-Default standalone mode is LF_SAMYRUN.
+Default standalone mode is $(DEFAULT_STANDALONE).
 To disable standalone modes, set explicitly an empty STANDALONE:
 STANDALONE=
 endef
 
+define KNOWN_DEFINITIONS
+$(KNOWN_PLATFORM_DEFINITIONS)
+$(KNOWN_STANDALONE_DEFINITIONS)
+$(HELP_DEFINITIONS)
+endef
+
 PLTNAME = Unknown Platform
 
 ifeq ($(PLATFORM),PM3RDV4)
@@ -131,13 +114,10 @@ PLATFORM_DEFS += \
     -DWITH_HFSNIFF
 
 # Standalone mode
-STANDALONE_MODES := LF_SAMYRUN LF_ICERUN LF_PROXBRUTE LF_HIDBRUTE
-STANDALONE_MODES += HF_YOUNG HF_MATTYRUN HF_COLIN HF_BOG
-ifneq ($(filter $(STANDALONE),$(STANDALONE_MODES)),)
-    PLATFORM_DEFS += -DWITH_STANDALONE_$(STANDALONE)
-else ifneq ($(STANDALONE),)
-    $(error Invalid STANDALONE: $(STANDALONE). $(KNOWN_DEFINITIONS))
+ifneq ($(strip $(filter $(PLATFORM_DEFS),$(STANDALONE_REQ_DEFS))),$(strip $(STANDALONE_REQ_DEFS)))
+    $(error Chosen Standalone mode $(STANDALONE) requires $(strip $(STANDALONE_REQ_DEFS)), unsupported by $(PLTNAME))
 endif
+PLATFORM_DEFS+=$(STANDALONE_PLATFORM_DEFS)
 
 $(info $(findstring WITH_STANDALONE_*,$(PLATFORM_DEFS)))
 
@@ -171,15 +151,6 @@ ifeq (,$(PLATFORM_DEFS_INFO_STANDALONE))
     PLATFORM_DEFS_INFO_STANDALONE = No standalone mode selected
 endif
 
-export PLATFORM
-export PLATFORM_EXTRAS
-export PLATFORM_EXTRAS_INFO
-export PLTNAME
-export MCU
-export PLATFORM_DEFS
-export PLATFORM_DEFS_INFO
-export PLATFORM_DEFS_INFO_STANDALONE
-
 PLATFORM_CHANGED=false
 ifneq ($(PLATFORM), $(CACHED_PLATFORM))
     PLATFORM_CHANGED=true
@@ -189,6 +160,16 @@ else ifneq ($(PLATFORM_DEFS), $(CACHED_PLATFORM_DEFS))
     PLATFORM_CHANGED=true
 endif
 
+export PLATFORM
+export PLATFORM_EXTRAS
+export PLATFORM_EXTRAS_INFO
+export PLTNAME
+export MCU
+export PLATFORM_DEFS
+export PLATFORM_DEFS_INFO
+export PLATFORM_DEFS_INFO_STANDALONE
+export PLATFORM_CHANGED
+
 $(info ===================================================================)
 $(info Platform name:     $(PLTNAME))
 $(info PLATFORM:          $(PLATFORM))
diff --git a/common/desfire.h b/common/desfire.h
index 3cffd2967..41e5b6e12 100644
--- a/common/desfire.h
+++ b/common/desfire.h
@@ -3,7 +3,7 @@
 
 #include <string.h>
 #include <stdarg.h>
-#include "aes.h"
+#include "mbedtls/aes.h"
 #include "mifare.h"
 
 #define MAX_CRYPTO_BLOCK_SIZE 16
@@ -71,10 +71,6 @@ enum DESFIRE_CRYPTOALGO {
 struct desfire_key {
     enum DESFIRE_CRYPTOALGO type;
     uint8_t data[24];
-    // DES_key_schedule ks1;
-    // DES_key_schedule ks2;
-    // DES_key_schedule ks3;
-    AesCtx aes_ks;
     uint8_t cmac_sk1[24];
     uint8_t cmac_sk2[24];
     uint8_t aes_version;
diff --git a/common/lfdemod.c b/common/lfdemod.c
index ef18b59ef..f8daec079 100644
--- a/common/lfdemod.c
+++ b/common/lfdemod.c
@@ -103,7 +103,7 @@ void computeSignalProperties(uint8_t *samples, uint32_t size) {
     // we can detect noise
     signalprop.isnoise =  signalprop.amplitude < NOISE_AMPLITUDE_THRESHOLD;
 
-    if (g_debugMode) 
+    if (g_debugMode)
         printSignal();
 }
 
@@ -1365,8 +1365,8 @@ static int millerRawDecode(uint8_t *bits, size_t *size, int invert) {
 int BiphaseRawDecode(uint8_t *bits, size_t *size, int *offset, int invert) {
     //sanity check
     if (*size < 51) return -1;
- 
-    if ( *offset < 0 ) *offset = 0;
+
+    if (*offset < 0) *offset = 0;
 
     uint16_t bitnum = 0;
     uint16_t errCnt = 0;
diff --git a/common/mbedtls/config.h b/common/mbedtls/config.h
index dfda7ca40..9f4b19c5e 100644
--- a/common/mbedtls/config.h
+++ b/common/mbedtls/config.h
@@ -489,7 +489,7 @@
  * This option is independent of \c MBEDTLS_AES_FEWER_TABLES.
  *
  */
-//#define MBEDTLS_AES_ROM_TABLES
+#define MBEDTLS_AES_ROM_TABLES
 
 /**
  * \def MBEDTLS_AES_FEWER_TABLES
@@ -511,7 +511,7 @@
  * This option is independent of \c MBEDTLS_AES_ROM_TABLES.
  *
  */
-//#define MBEDTLS_AES_FEWER_TABLES
+#define MBEDTLS_AES_FEWER_TABLES
 
 /**
  * \def MBEDTLS_CAMELLIA_SMALL_MEMORY
diff --git a/common/mbedtls/platform_util.c b/common/mbedtls/platform_util.c
index a3634e246..c30a52c08 100644
--- a/common/mbedtls/platform_util.c
+++ b/common/mbedtls/platform_util.c
@@ -60,9 +60,9 @@
  * mbedtls_platform_zeroize() to use a suitable implementation for their
  * platform and needs.
  */
-static void *(* const volatile memset_func)(void *, int, size_t) = memset;
+//static void *(* const volatile memset_func)(void *, int, size_t) = memset;
 
 void mbedtls_platform_zeroize(void *buf, size_t len) {
-    memset_func(buf, 0, len);
+    memset(buf, 0, len);
 }
 #endif /* MBEDTLS_PLATFORM_ZEROIZE_ALT */
diff --git a/doc/bt_manual_v10.md b/doc/bt_manual_v10.md
index 032680cfa..1cc8d3690 100644
--- a/doc/bt_manual_v10.md
+++ b/doc/bt_manual_v10.md
@@ -2,14 +2,14 @@
 ### Bluetooth / Battery add-on 'Blue Shark'
 _rev. v1.0 (draft)_
 
-### 1.	FEATURES
+### 1. FEATURES
 
-*	Built-in Bluetooth 2.0 with EDR Bluetooth module, default baud rate 115200.
-*	Built-in 400 mAh polymer lithium-ion battery, typical standby time up to 3.5 hours.
-*	Additional heat dissipating fins can significantly reduce the temperature when the HF antenna is in operation for a long time.
-*	Complete lithium charging management system, seamless switching power supply. Full overcharge and overdischarge protection.
-*	Bluetooth has an independent power switch that can be turned on or off.
-*	It's compact and easy to carry. The clamp structure is easy to install and replace.
+* Built-in Bluetooth 2.0 with EDR Bluetooth module, default baud rate 115200.
+* Built-in 400 mAh polymer lithium-ion battery, typical standby time up to 3.5 hours.
+* Additional heat dissipating fins can significantly reduce the temperature when the HF antenna is in operation for a long time.
+* Complete lithium charging management system, seamless switching power supply. Full overcharge and overdischarge protection.
+* Bluetooth has an independent power switch that can be turned on or off.
+* It's compact and easy to carry. The clamp structure is easy to install and replace.
 
 
 It can easily connect to Bluetooth mobile phone, portable computer, etc. Without USB cable, complicated permissions or driver settings.
@@ -17,35 +17,35 @@ It can easily connect to Bluetooth mobile phone, portable computer, etc. Without
 Built-in battery can support standalone mode, off-line sniffing, off-line reading & simulation, etc. The temperature of the device is stable.
 
 
-### 2.	PARAMETERS
+### 2. PARAMETERS
 
-*	Battery capacity:	400 mAh
-*	Standby time:		3.5h @ StandBy; 2.9h @ LF-On; 50min @ HF-On;
-*	Charging Current:	200mA (Plug in USB Default Charging)
-*	Charging time:		2.5h
-*	Num of charges:         400 -> 70% capacity (standard LIPO)
-*	Bluetooth power:	4dBm, -85 dBm @ 2Mbps
-*	Bluetooth distance:	6m (depending on the environment and device orientation)
-*	Size and weight:	54.4mm * 29.4mm * 13.5mm 24g
+* Battery capacity:   400 mAh
+* Standby time:       3.5h @ StandBy; 2.9h @ LF-On; 50min @ HF-On;
+* Charging Current:   200mA (Plug in USB Default Charging)
+* Charging time:      2.5h
+* Num of charges:     400 -> 70% capacity (standard LIPO)
+* Bluetooth power:    4dBm, -85 dBm @ 2Mbps
+* Bluetooth distance: 6m (depending on the environment and device orientation)
+* Size and weight:    54.4mm * 29.4mm * 13.5mm 24g
 
 
-### 3.	ASSEMBLY STEPS
+### 3. ASSEMBLY STEPS
 
-*	Unplug your Proxmark3 RDV4.0 device from any usb cable.
-*	Remove the plastic upper case of Proxmark3 RDV4.0 with opener.
-*	Remove temporarily the antenna with a H5 (Hex/Allen) screwdriver to expose the FPC interface.<p>
-	<img src="https://sneaktechnology.com/wp-content/uploads/2019/06/FPC-Interface.png" alt="Image of blue shark add-on fpc interface" width="300"></p>
-*	Turn off all power switches, insert the FPC wire into the FPC connector, and lock the FPC connector.<p>
-	<img src="https://sneaktechnology.com/wp-content/uploads/2019/06/FPC-Connected.png" alt="Image of blue shark add-on fpc wire" width="300"></p>
-*	Tear off the blue film of heat conductive double-sided tape. Align the add-on to the hole positions and gently insert it into the case.<p>
-	<img src="https://sneaktechnology.com/wp-content/uploads/2019/06/Blue-Film.png" alt="Image of blue shark add-on blue film location" width="300"></p>
-*	Assembly finished!
+* Unplug your Proxmark3 RDV4.0 device from any usb cable.
+* Remove the plastic upper case of Proxmark3 RDV4.0 with opener.
+* Remove temporarily the antenna with a H5 (Hex/Allen) screwdriver to expose the FPC interface.<p>
+  <img src="https://sneaktechnology.com/wp-content/uploads/2019/06/FPC-Interface.png" alt="Image of blue shark add-on fpc interface" width="300"></p>
+* Turn off all power switches, insert the FPC wire into the FPC connector, and lock the FPC connector.<p>
+  <img src="https://sneaktechnology.com/wp-content/uploads/2019/06/FPC-Connected.png" alt="Image of blue shark add-on fpc wire" width="300"></p>
+* Tear off the blue film of heat conductive double-sided tape. Align the add-on to the hole positions and gently insert it into the case.<p>
+  <img src="https://sneaktechnology.com/wp-content/uploads/2019/06/Blue-Film.png" alt="Image of blue shark add-on blue film location" width="300"></p>
+* Assembly finished!
 
 <p align='center'>
 <img src="http://www.icedev.se/proxmark3/blueshark/addon_open_1.jpg" alt="Image of blue shark add-on open fit" width="300"><img src="http://www.icedev.se/proxmark3/blueshark/addon_fitted_1.jpg" alt="Image of blue shark add-on fitted" width="300">
 </p>
 
-### 4.	COMPILATION / FLASHING
+### 4. COMPILATION / FLASHING
 
 #### From Source
 
@@ -69,7 +69,7 @@ From the [homebrew-proxmark3 readme](https://github.com/RfidResearchGroup/homebr
 1. `brew tap rfidresearchgroup/proxmark3`
 2. `brew install --with-blueshark proxmark3`
 
-### 5.	CONNECT WITH BLUETOOTH
+### 5. CONNECT WITH BLUETOOTH
 
 You can have both USB cable connect and BT active at the same time and connect to either serial port.  
 You can also switch serial port from inside the proxmark3 client using the new command `hw connect`.
@@ -82,30 +82,30 @@ You can also switch serial port from inside the proxmark3 client using the new c
 
 #### Windows
 
-#### (1)	Connecting your RDV4.0 with Bluetooth on mobile phone or computer
-*	Open Bluetooth and search for a device named PM3_RDV4.0.
-*	Enter the paired password 1234 and establish the connection.
-*	The blue state LED on the add-on will keep blinking after the connection is established. Only when the mobile phone or computer opens the correct COM port, the blue LED turns on solid, indicating that the connection is successful.
+#### (1) Connecting your RDV4.0 with Bluetooth on mobile phone or computer
+* Open Bluetooth and search for a device named PM3_RDV4.0.
+* Enter the paired password 1234 and establish the connection.
+* The blue state LED on the add-on will keep blinking after the connection is established. Only when the mobile phone or computer opens the correct COM port, the blue LED turns on solid, indicating that the connection is successful.
 
-#### (2)	Fast connection using dedicated USB Bluetooth adapter under Windows
+#### (2) Fast connection using dedicated USB Bluetooth adapter under Windows
 
 <p align='center'>
 <img src="http://www.icedev.se/proxmark3/blueshark/addon_hc06_dongle_1.jpg" alt="Image of blue shark add-on HC-06 white dongle" width="300"></p>
 
-*	Install driver:
+* Install driver:
 http://www.silabs.com/products/development-tools/software/usb-to-uart-bridge-vcp-drivers  
-*	Insert the adapter into the USB port. The adapter will search automatically and establish the connection. The adapter will remember the device that was first connected and after that the same device will be connected.
-*	The adapter button can be used to delete memory so that other add-on can be searched and connected.<p align='center'>
-	<img src="https://sneaktechnology.com/wp-content/uploads/2019/06/Button.png" alt="Image of blue shark add-on HC-06 white dongle button" width="200">
-*	After the connection is established, the blue state LED on add-on will turn on solid.
-*	look for _CP2104 USB tp UART bridge controller_ under devices in order to get the assigned serial port
+* Insert the adapter into the USB port. The adapter will search automatically and establish the connection. The adapter will remember the device that was first connected and after that the same device will be connected.
+* The adapter button can be used to delete memory so that other add-on can be searched and connected.<p align='center'>
+    <img src="https://sneaktechnology.com/wp-content/uploads/2019/06/Button.png" alt="Image of blue shark add-on HC-06 white dongle button" width="200">
+* After the connection is established, the blue state LED on add-on will turn on solid.
+* look for _CP2104 USB tp UART bridge controller_ under devices in order to get the assigned serial port
 
 #### Linux
 
 #### (1) Connecting rdv4.0 with Bluetooth on Linux computer
 
 1. Find the MAC address of the Bluetooth add-on, named PM3_RDV4.0.
-	
+
 ```sh
 sudo hcitool scan
 Scanning ...
@@ -161,8 +161,8 @@ be connected.
 
   2. The adapter button can be used to delete memory so that other add-on
 can be searched and connected.<p align='center'>
-	<img src="https://sneaktechnology.com/wp-content/uploads/2019/06/Button.png" alt="Image of blue shark add-on HC-06 white dongle button" width="200">
-	</p>
+     <img src="https://sneaktechnology.com/wp-content/uploads/2019/06/Button.png" alt="Image of blue shark add-on HC-06 white dongle button" width="200">
+</p>
 
   3. After the connection is established, the blue state LED on add-on will
 turn on solid.
@@ -195,22 +195,22 @@ After reboot you can go ahead to pairing your Proxmark3 RDV4 Blue Shark:
 ```
 
 
-### 6.	OTHER NOTES
+### 6. OTHER NOTES
 
-#### (1)	UART and LED behavior
+#### (1) UART and LED behavior
 Bluetooth is connected to Proxmark3 RDV4.0 via UART. The USB and UART interfaces of RDV4.0 can coexist without conflict, and no special switching is required. 
 
 The following link has helpful notes on UART usage and baud rates:
 https://github.com/RfidResearchGroup/proxmark3/blob/master/doc/uart_notes.md
 
-#### (2)	Disassembly
+#### (2) Disassembly
 There is a heat conductive double-sided tape inside the add-on, which has strong adhesive force. Therefore, if add-on needs to be removed, it needs to be pulled out from the heat sink end with greater efforts. Each disassembly will reduce the viscidity of double-sided tape. When double-sided tape is well protected, it will not affect the second use. Thermal conductivity will be slightly worse and will therefore have a direct impact on the thermal performance of the heat sink.
 
-#### (3)	Battery charging
+#### (3) Battery charging
 The battery charging circuit is turned on by default. Any time a USB cable is inserted, the battery will be automatically charged. The red LED will remain bright when charging. 
 The red LED will be extinguished when charging is completed.
 
-#### (4)	Get better signals
-For the better heat dissipation, we have used a cast metal enclosure for the add-on. As a result Bluetooth wireless signals are sacrificed. For example, if the back of add-on is facing the Bluetooth host, the signal is very bad and the distance will be reduced. The best signal strength can be obtained when the front glass faces the Bluetooth host.
-	If the Proxmark3 is not responding, it may be due to a poor Bluetooth connection. To improve performance, try repositioning the Proxmark3 so the glass face is directed toward the host.
+#### (4) Get better signals
+For the better heat dissipation, we have used a cast metal enclosure for the add-on. As a result Bluetooth wireless signals are sacrificed. For example, if the back of add-on is facing the Bluetooth host, the signal is very bad and the distance will be reduced. The best signal strength can be obtained when the front glass faces the Bluetooth host.  
+If the Proxmark3 is not responding, it may be due to a poor Bluetooth connection. To improve performance, try repositioning the Proxmark3 so the glass face is directed toward the host.
 
diff --git a/doc/ext_flash_notes.md b/doc/ext_flash_notes.md
new file mode 100644
index 000000000..46059d9ca
--- /dev/null
+++ b/doc/ext_flash_notes.md
@@ -0,0 +1,95 @@
+# External flash
+
+External 256kbytes flash is a unique feature of the RDV4 edition.
+
+## Addresses
+
+Flash memory is
+
+* 256kb (0x40000= 262144)
+* divided into 4 pages of 64kb (0x10000 = 65536)
+* 4 pages divided into 16 sectors of 4kb (0x1000 = 4096), so last sector is at 0x3F000
+
+Therefore a flash address can be interpreted as such:
+```
+0xPSxxx       e.g. 0x3FF7F
+  ^ page             ^ page 3
+   ^ sector           ^ sector 0xF
+    ^^^ offset         ^^^ offset 0xF7F
+```
+
+## Layout
+
+Page 0:
+* available for user data
+* to dump it: `mem dump f page0_dump o 0 l 65536`
+* to erase it: `mem wipe p 0`
+
+Page 1:
+* available for user data
+* to dump it: `mem dump f page1_dump o 65536 l 65536`
+* to erase it: `mem wipe p 1`
+
+Page 2:
+* available for user data
+* to dump it: `mem dump f page2_dump o 131072 l 65536`
+* to erase it: `mem wipe p 2`
+
+Page 3:
+* used by Proxmark3 RDV4 specific functions: flash signature and keys dictionaries, see below for details
+* to dump it: `mem dump f page3_dump o 196608 l 65536`
+* to erase it:
+  * **Beware** it will erase your flash signature (see below) so better to back it up first as you won't be able to regenerate it by yourself!
+  * It's possible to erase completely page 3 by erase the entire flash memory with the voluntarily undocumented command `mem wipe i`.
+  * Updating keys dictionaries doesn't require to erase page 3.
+
+## Page3 Layout
+
+Page3 is used as follows by the Proxmark3 RDV4 firmware:
+
+* **MF_KEYS**
+  * offset: page 3 sector  9 (0x9) @ 3*0x10000+9*0x1000=0x39000
+  * length: 2 sectors
+
+* **ICLASS_KEYS**
+  * offset: page 3 sector 11 (0xB) @ 3*0x10000+11*0x1000=0x3B000
+  * length: 1 sector
+
+* **T55XX_KEYS**
+  * offset: page 3 sector 12 (0xC) @ 3*0x10000+12*0x1000=0x3C000
+  * length: 1 sector
+
+* **T55XX_CONFIG**
+  * offset: page 3 sector 13 (0xD) @ 3*0x10000+13*0x1000=0x3D000
+  * length: 1 sector (actually only a few bytes are used to store `t55xx_config` structure)
+
+* **RSA SIGNATURE**, see below for details
+  * offset: page 3 sector 15 (0xF) offset 0xF7F @ 3*0x10000+15*0x1000+0xF7F=0x3FF7F
+  * length: 128 bytes
+  * offset should have been 0x3FF80 but historically it's one byte off and therefore the last byte of the flash is unused
+
+## RSA signature
+
+To ensure your Proxmark3 RDV4 is not a counterfeit product, its external flash contains a RSA signature of the flash unique ID.
+You can verify it with: `mem info`
+
+```
+[usb] pm3 --> mem info
+          
+[=] --- Flash memory Information ---------
+          
+[=] -------------------------------------------------------------          
+[=] ID            | xx xx xx xx xx xx xx xx           
+[=] SHA1          | xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx           
+[=] RSA SIGNATURE |          
+[00] | xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx 
+[01] | xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx 
+[02] | xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx 
+[03] | xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx xx 
+[=] KEY length   | 128          
+[+] RSA key validation ok          
+[+] RSA Verification ok          
+```
+
+For a backup of the signature: `mem dump p f flash_signature_dump o 262015 l 128`
+
diff --git a/doc/md/Installation_Instructions/Linux-Installation-Instructions.md b/doc/md/Installation_Instructions/Linux-Installation-Instructions.md
index 8954a38f7..fe23db001 100644
--- a/doc/md/Installation_Instructions/Linux-Installation-Instructions.md
+++ b/doc/md/Installation_Instructions/Linux-Installation-Instructions.md
@@ -25,7 +25,7 @@ Install the requirements
 
 ```sh
 sudo apt-get install p7zip git ca-certificates build-essential libreadline5 libreadline-dev \
-libusb-0.1-4 libusb-dev perl pkg-config wget libncurses5-dev gcc-arm-none-eabi libstdc++-arm-none-eabi-newlib libqt4-dev
+libusb-0.1-4 libusb-dev perl pkg-config wget libncurses5-dev gcc-arm-none-eabi libnewlib-dev libqt4-dev
 ```
 
 If you don't need the graphical components of the Proxmark3 client, you can skip the installation of `libqt4-dev`.
@@ -41,8 +41,10 @@ Additional AUR packages:
 ```sh
 yaourt -S termcap
 ```
-
-Note that with only these requirements, you will not get the graphical components of the Proxmark3 client. (Untested: how to get it? `yaourt -S qt4` ?)
+If you want graphical output (such as in `hw tune`):
+```sh
+sudo pacman -Su qt5-base
+```
 
 # Clone the RRG/Iceman repository
 
diff --git a/doc/md/Installation_Instructions/Windows-Installation-Instructions.md b/doc/md/Installation_Instructions/Windows-Installation-Instructions.md
index ac88906da..db9a9084f 100644
--- a/doc/md/Installation_Instructions/Windows-Installation-Instructions.md
+++ b/doc/md/Installation_Instructions/Windows-Installation-Instructions.md
@@ -1,60 +1,169 @@
-# Building on Windows
-You will need to use the Gator96100 Proxspace package to assist in your windows installation.
-This can be downloaded from https://github.com/Gator96100/ProxSpace/
+# Installing on Windows
+
+There are two ways to install, build and use Proxmark3 on Windows:
+
+* Using Gator96100 **ProxSpace**, a package to assist in your Windows installation of MinGW
+* Using native **WSL**, if you're running a Windows 10 version recent enough (FCU 1709 or later)
 
 ---
-# Video Installation guide
+
+# Installing on Windows with ProxSpace
+
+## Video Installation guide
 [![Windows Installation tutorial](https://github.com/5w0rdfish/Proxmark3-RDV4-ParrotOS/blob/master/screenshot-www.youtube.com-2019.03.17-20-44-33.png)](https://youtu.be/zzF0NCMJnYU "Windows Installation Tutorial")
 
-## Manual Installation
+## Driver Installation
 
-### Driver Installation
+Install required drivers for your Windows installation. You may need admin privileges to do this.  
+Step by step guides are online such as [RiscCorps](https://store.ryscc.com/blogs/news/how-to-install-a-proxmark3-driver-on-windows-10).
 
-Install required drivers for your windows installation. You will may need admin privileges to do this. 
-(This is covered in the video) Step by step guides are online such as [RiscCorps](https://store.ryscc.com/blogs/news/how-to-install-a-proxmark3-driver-on-windows-10)
+## Download / clone ProxSpace repo
 
-### Install Github
+Download the Gator96100 ProxSpace package from https://github.com/Gator96100/ProxSpace/
 
-Install Github for Windows https://desktop.github.com/
+If you prefer, you can clone it, provided that you installed Github for Windows https://desktop.github.com/.
 
-### Download / clone Proxspace repo
-
-Download the required proxspace repo. https://github.com/Gator96100/ProxSpace/
-
-Extract 'ProxSpace' to a location on drive without spaces. 
+Extract 'ProxSpace' to a location path without spaces.  
 For example D:\OneDrive\Documents\GitHub is ok whereas C:\My Documents\My Projects\proxspace is not.
 
-### Clone the RRG/Iceman repository
+If you're running Windows in a Virtualbox guest, make sure not to install ProxSpace on a vbox shared drive. (It's ok later to move the `/pm3` subfolder to a shared drive and edit the `*.bat`)
 
-```sh
-git clone https://github.com/RfidResearchGroup/proxmark3.git
-```
-
-### Copy files to Proxspace
-
-Copy all the contents from the `proxmark3` folder into the proxspace `pm3` folder
-
-### Run the .bat
+## Launch ProxSpace
 
 Run `runme.bat` or `runme64.bat` depending on your Windows architecture.
 
-Please note you will need to use `/` as you are using BASH.
+You'll get a Bash prompt and your home directory should become the ProxSpace `pm3` sub-directory.
 
-### Compile and use the project
+Please note you will need to use `/` in paths as you are using Bash.
+
+## Clone the RRG/Iceman repository
+
+```sh
+cd
+git clone https://github.com/RfidResearchGroup/proxmark3.git
+cd proxmark3
+```
+
+If you're a contributing developer, you probably want to be able to use `make style`. If so, you've to install astyle:
+
+```sh
+pacman -S mingw-w64-x86_64-astyle
+```
+
+## Compile and use the project
 
 Now you're ready to follow the [compilation instructions](/doc/md/Use_of_Proxmark/0_Compilation-Instructions.md).
 
-The only differences are that executables end with `.exe` (e.g. `client/flasher.exe`) and that the Proxmark3 port is one of your `comX` ports where "X" is the com port number assigned to proxmark3 under Windows.
+To use the compiled client and flasher, the only differences are that executables end with `.exe` (e.g. `client/flasher.exe`) and that the Proxmark3 port is one of your `comX` ports where "X" is the com port number assigned to proxmark3 under Windows.
 
-So flashing will resemble
+To flash: In principle, the helper script `flash-all.sh` should auto-detect your COM port, so you can just try:
+
+```sh
+./flash-all.sh
+```
+
+If COM port detection failed, you'll have to call the flasher manually and specify the correct port:
 
 ```sh
 client/flasher.exe comX -b bootrom/obj/bootrom.elf armsrc/obj/fullimage.elf
 ```
 
-And running the client will resemble
+Similarly, to run the client, you may try:
 
 ```sh
-cd client
-./proxmark3.exe comX
+./proxmark3.sh
 ```
+
+Or, by specifying the COM port manually:
+
+```sh
+client/proxmark3.exe comX
+```
+
+# Installing on Windows with WSL
+
+It requires to run a Windows 10 version 1709 or above. Previous versions didn't have support for COM ports.
+
+Install WSL with e.g. the standard Ubuntu.
+
+For WSL configuration, see [Manage and configure Windows Subsystem for Linux](https://docs.microsoft.com/en-us/windows/wsl/wsl-config).
+
+## X Server Installation
+
+If you want to run the graphical components of the Proxmark3 client, you need to install a X Server such as [VcXsrv](https://sourceforge.net/projects/vcxsrv/) or [Xming](https://sourceforge.net/projects/xming/) and launch it, e.g. by executing XLaunch.
+
+## Dependencies
+
+Enter WSL prompt (`wsl`) and from there, follow the [Linux Installation Instructions](/doc/md/Installation_Instructions/Linux-Installation-Instructions.md) for Ubuntu, summarized here below:
+
+```sh
+sudo apt-get update
+sudo apt-get install p7zip git ca-certificates build-essential libreadline5 libreadline-dev libusb-0.1-4 \
+libusb-dev perl pkg-config wget libncurses5-dev gcc-arm-none-eabi libstdc++-arm-none-eabi-newlib \
+libqt4-dev
+```
+
+If you don't need the graphical components of the Proxmark3 client, you can skip the installation of `libqt4-dev`.
+
+## Clone the RRG/Iceman repository
+
+```sh
+git clone https://github.com/RfidResearchGroup/proxmark3.git
+```
+
+## Compile and use the project
+
+Now you're ready to follow the [compilation instructions](/doc/md/Use_of_Proxmark/0_Compilation-Instructions.md).
+
+To use the compiled client and flasher, the only difference is that the Proxmark3 port is translated from your `comX` port where "X" is the com port number assigned to proxmark3 under Windows, to a `/dev/ttySX`.
+
+Depending on the Windows version, you might need to give permission to the current user to access `/dev/ttySX`: (change X to your port number)
+
+```sh
+ls -al /dev/ttySX
+groups|grep dialout
+```
+
+If group ownership is `dialout` and your user is member of `dialout` group, all is fine. Else you'll have to provide access to `/dev/ttySX`: (Unfortunately the access rights of the port won't survive and will have to be fixed again next time.)
+
+```sh
+sudo chmod 666 /dev/ttySX
+```
+
+If you installed a X Server and compiled the Proxmark3 with QT4 support, you've to export the `DISPLAY` environment variable:
+
+```sh
+export DISPLAY=:0
+```
+
+and add it to your Bash profile for the next times:
+
+```sh
+echo "export DISPLAY=:0" >> ~/.bashrc
+```
+
+To flash: In principle, the helper script `flash-all.sh` should auto-detect your COMX==/dev/ttySX port, so you can just try:
+
+```sh
+./flash-all.sh
+```
+
+If port detection failed, you'll have to call the flasher manually and specify the correct port:
+
+```sh
+client/flasher /dev/ttySX -b bootrom/obj/bootrom.elf armsrc/obj/fullimage.elf
+```
+
+Similarly, to run the client, you may try:
+
+```sh
+./proxmark3.sh
+```
+
+Or, by specifying the COM port manually:
+
+```sh
+client/proxmark3 /dev/ttySX
+```
+
+Note that it may take a quite long time for a freshly plugged Proxmark3 to be visible on a WSL /dev/ttySX port.
diff --git a/doc/md/Use_of_Proxmark/0_Compilation-Instructions.md b/doc/md/Use_of_Proxmark/0_Compilation-Instructions.md
index bb3699a72..7493fba1a 100644
--- a/doc/md/Use_of_Proxmark/0_Compilation-Instructions.md
+++ b/doc/md/Use_of_Proxmark/0_Compilation-Instructions.md
@@ -1,5 +1,16 @@
 # Compilation instructions
 
+## Tuning compilation parameters
+
+The client and the Proxmark3 firmware should always be in sync.
+Nevertheless, the firmware can be tuned depending on the Proxmark3 platform and options.
+
+Indeed, the RRG/Iceman fork can be used on other Proxmark3 hardware platforms as well.
+
+Via some definitions, you can adjust the firmware for a given platform, but also to add features like the support of the Blue Shark add-on or to select which standalone mode to embed.
+
+To learn how to adjust the firmware, please read [Advanced compilation parameters](/doc/md/Use_of_Proxmark/4_Advanced-compilation-parameters.md).
+
 ## Get the latest commits
 
 ```sh
@@ -15,48 +26,23 @@ make clean && make all
 
 ## Flash the BOOTROM & FULLIMAGE
 
+In most cases, you can run the script `flash-all.sh` which try to auto-detect the port to use, on several OS.
+
+For the other cases, specify the port by yourself. For example, for a Proxmark3 connected via USB under Linux:
+
 ```sh
 client/flasher /dev/ttyACM0 -b bootrom/obj/bootrom.elf armsrc/obj/fullimage.elf
 ```
 
 ## Run the client
 
-```sh
-cd client
-./proxmark3 /dev/ttyACM0
-```
+In most cases, you can run the script `proxmark3.sh` which try to auto-detect the port to use, on several OS.
 
-## Compiling for other boards
+For the other cases, specify the port by yourself. For example, for a Proxmark3 connected via USB under Linux:
 
-Available boards
-
-| BOARD           | PLATFORM                               |
-|:---------------:|:---------------------------------------|
-| `PM3RDV4` (def) | Proxmark3 rdv4      with AT91SAM7S512  |
-| `PM3EVO`        | Proxmark3 EVO       with AT91SAM7S512  |
-| `PM3EASY`       | Proxmark3 rdv3 Easy with AT91SAM7S256  |
-| `PM3RDV2`       | Proxmark3 rdv2      with AT91SAM7S512  |
-| `PM3OLD256`     | Proxmark3 V1        with AT91SAM7S256  |
-| `PM3OLD512`     | Proxmark3 V1        with AT91SAM7S512  |
-
-Create a file named `Makefile.platform` in the root directory of the repository:
+Here, for example, for a Proxmark3 connected via USB under Linux:
 
 ```sh
-# PLATFORM=${BOARD}
-# Following example is to compile sources for Proxmark3 rdv3 Easy
-PLATFORM=PM3EASY
-```
-
-From this point:
-
-```sh
-# Clean and compile
-make clean && make all
-
-# Flash the BOOTROM & FULLIMAGE
-client/flasher /dev/ttyACM0 -b bootrom/obj/bootrom.elf armsrc/obj/fullimage.elf
-
-# Run the client
 cd client
 ./proxmark3 /dev/ttyACM0
 ```
diff --git a/doc/md/Use_of_Proxmark/2_Configuration-and-Verification.md b/doc/md/Use_of_Proxmark/2_Configuration-and-Verification.md
index 736ef70a5..0d3665f37 100644
--- a/doc/md/Use_of_Proxmark/2_Configuration-and-Verification.md
+++ b/doc/md/Use_of_Proxmark/2_Configuration-and-Verification.md
@@ -5,6 +5,12 @@ pm3 --> mem load f default_keys m
 pm3 --> mem load f default_pwd t
 pm3 --> mem load f default_iclass_keys i
 pm3 --> lf t55xx deviceconfig a 29 b 17 c 15 d 47 e 15 p
+pm3 --> lf t55xx deviceconfig r 1 a 31 b 20 c 18 d 50 e 15 p
+pm3 --> lf t55xx deviceconfig r 2 a 31 b 20 c 18 d 40 e 15 p
+pm3 --> lf t55xx deviceconfig r 3 a 29 b 17 c 15 d 31 e 15 f 47 g 63 p
+
+Set all t55xx settings to defaults (will set all 4 at once)
+pm3 --> lf t55xx deviceconfig z p
 ```
 
 ### Verify sim module firmware version
diff --git a/doc/md/Use_of_Proxmark/4_Advanced-compilation-parameters.md b/doc/md/Use_of_Proxmark/4_Advanced-compilation-parameters.md
new file mode 100644
index 000000000..aea5b5e30
--- /dev/null
+++ b/doc/md/Use_of_Proxmark/4_Advanced-compilation-parameters.md
@@ -0,0 +1,82 @@
+# Advanced compilation parameters
+
+The client and the Proxmark3 firmware should always be in sync.
+Nevertheless, the firmware can be tuned depending on the Proxmark3 platform and options.
+
+Indeed, the RRG/Iceman fork can be used on other Proxmark3 hardware platforms as well.
+
+Via some definitions, you can adjust the firmware for a given platform, but also to add features like the support of the Blue Shark add-on or to select which standalone mode to embed.
+
+## Client
+
+The client doesn't depend on the capabilities of the Proxmark3 it's connected to.
+So you can use the same client for different Proxmark3 platforms, given that everything is running the same version.
+
+## Firmware
+
+By default, the firmware is of course tuned for the Proxmark3 Rdv4.0 device, which has built-in support for 256kb onboard flash SPI memory, Sim module (smart card support), FPC connector.
+These features make it very different from all other devices, there is non other like this one.
+
+**Recommendation**: if you don't have a RDV4, we strongly recommend your device to have at least a 512kb arm chip, since this repo is on the very edge of 256kb limit.
+
+A firmware built for the RDV4 can still run on the other platforms as it will auto-detect during boot that external SPI and Sim are not present, still it will boot faster if it's tuned to the platform.
+
+If you need to tune things and save the configuration, create a file `Makefile.platform` in the root directory of the repository, see `Makefile.platform.sample`.
+For an up-to-date exhaustive list of options, you can run `make PLATFORM=`.
+
+## PLATFORM
+
+Here are the supported values you can assign to `PLATFORM` in `Makefile.platform`:
+
+| PLATFORM        | DESCRIPTION                            |
+|-----------------|----------------------------------------|
+| PM3RDV4 (def)   | Proxmark3 rdv4      with AT91SAM7S512  |
+| PM3EVO          | Proxmark3 EVO       with AT91SAM7S512  |
+| PM3EASY         | Proxmark3 rdv3 Easy with AT91SAM7S256  |
+| PM3RDV2         | Proxmark3 rdv2      with AT91SAM7S512  |
+| PM3OLD256       | Proxmark3 V1        with AT91SAM7S256  |
+| PM3OLD512       | Proxmark3 V1        with AT91SAM7S512  |
+
+By default `PLATFORM=PM3RDV4`.
+
+Known issues:
+
+* 256kb Arm chip devices: The compiled firmware image from this repo may/will be too large for your device. 
+* PM3 Evo: it has a different led/button pin assignment.  It tends to be messed up.
+
+## PLATFORM_EXTRAS
+
+Here are the supported values you can assign to `PLATFORM_EXTRAS` in `Makefile.platform`:
+
+| PLATFORM_EXTRAS | DESCRIPTION                            |
+|-----------------|----------------------------------------|
+| BTADDON         | Proxmark3 rdv4 BT add-on               |
+
+By default `PLATFORM_EXTRAS=`.
+
+If you have installed a Blue Shark add-on on your RDV4, define `PLATFORM_EXTRAS=BTADDON` in your `Makefile.platform`.
+
+
+## STANDALONE
+
+The RRG/Iceman fork gives you to easily choose which standalone mode to embed in the firmware.
+
+Here are the supported values you can assign to `STANDALONE` in `Makefile.platform`:
+
+| STANDALONE      | DESCRIPTION                            |
+|-----------------|----------------------------------------|
+|                 | No standalone mode
+| LF_SAMYRUN (def)| HID26 read/clone/sim - Samy Kamkar
+| LF_ICERUN       | standalone mode skeleton - iceman
+| LF_PROXBRUTE    | HID ProxII bruteforce - Brad Antoniewicz
+| LF_HIDBRUTE     | HID corporate 1000 bruteforce - Federico dotta & Maurizio Agazzini
+| HF_YOUNG        | Mifare sniff/simulation - Craig Young
+| HF_MATTYRUN     | Mifare sniff/clone - Matías A. Ré Medina
+| HF_COLIN        | Mifare ultra fast sniff/sim/clone - Colin Brigato
+| HF_BOG          | 14a sniff with ULC/ULEV1/NTAG auth storing in flashmem - Bogito
+
+By default `STANDALONE=LF_SAMYRUN`.
+
+## Next step
+
+See [Compilation instructions](/doc/md/Use_of_Proxmark/0_Compilation-Instructions.md)
diff --git a/doc/new_frame_format.md b/doc/new_frame_format.md
index 76424f8b6..17c94120a 100644
--- a/doc/new_frame_format.md
+++ b/doc/new_frame_format.md
@@ -373,9 +373,9 @@ It was needed to tune pm3 RX usart `maxtry` :
     time client/proxmark3 -p /dev/ttyUSB0 -b 115200 -c "lf read"
     6.28s
 
-    time client/proxmark3 -p /dev/ttyACM0 -c "mem save f foo_usb"
+    time client/proxmark3 -p /dev/ttyACM0 -c "mem dump f foo_usb"
     1.48s
-    time client/proxmark3 -p /dev/ttyUSB0 -b 115200 -c "mem save f foo_fpc"
+    time client/proxmark3 -p /dev/ttyUSB0 -b 115200 -c "mem dump f foo_fpc"
     25.34s
 
 
diff --git a/doc/uart_notes.md b/doc/uart_notes.md
index d074bddc8..88b10a109 100644
--- a/doc/uart_notes.md
+++ b/doc/uart_notes.md
@@ -27,10 +27,10 @@ Proxmark3 RDV4 has a FPC connector outputting on 2 pins a USART from the ARM:
 USART support is in `common/usart.c`.
 
 There are mainly two ways to use this USART:
-* connect the host client to the Proxmark3 via this USART instead of USB-CDC, this is the `FPC_USART_HOST`. The most used way is through the BT add-on (blue shark) that we will cover later. Instead of BT add-on, we can also use e.g. a FTDI cable (mostly for internal development, it's much slower than USB-CDC anyway) or in the future other ways to connect the host such as a USART-to-Wi-Fi bridge.
+* connect the host client to the Proxmark3 via this USART instead of USB-CDC, this is the `FPC_USART_HOST` option you can add to `PLATFORM_EXTRAS` in `Makefile.platform`. The most used way is through the BT add-on (blue shark) that we will cover later. Instead of BT add-on, we can also use e.g. a FTDI cable (mostly for internal development, it's much slower than USB-CDC anyway) or in the future other ways to connect the host such as a USART-to-Wi-Fi bridge.
 * connect "slave" devices to the Proxmark3 to add functionnalities. In such case, the host client will use USB-CDC and the USART will be use to, e.g. connect the Proxmark3 to various daughterboards. These is no such example of daughterboard as of today, except when we're talking to the BT add-on in its AT configuration mode.
 
-This USART can be reached from the host client (if connected via USB-CDC) through the following commands, available in `FPC_USART_DEV` build:
+This USART can be reached from the host client (if connected via USB-CDC) through the following commands, available when you add `FPC_USART_DEV` to `PLATFORM_EXTRAS` in `Makefile.platform`:
 * `usart config`, to configure the baudrate and the parity of the Proxmark3 USART
 * `usart txrx/tx/rx/txhex/rxhex` to transmit and receive bytes
 
@@ -45,7 +45,7 @@ Internally, the desired baudrate is converted to UART settings: a BRGR and a FP.
 
 When the BT add-on is turned on but no actively connected to a host, it's in a configuration mode where it accepts "AT" commands and its blue LED is blinking at about 1Hz.
 
-Some specific commands are available in `BTADDON` build, only to configure specific features of the BT add-on:
+Some specific commands are available when you add `BTADDON` to `PLATFORM_EXTRAS` in `Makefile.platform` (it will automatically enable `FPC_USART_HOST` as well), to configure specific features of the BT add-on:
 * `usart btpin`, to change the BT add-on PIN
 * `usart btfactory`, to guess the current BT add-on UART settings and to reset its configuration.
 
diff --git a/flash-all.sh b/flash-all.sh
index 27e1fb526..d4fcdc975 100755
--- a/flash-all.sh
+++ b/flash-all.sh
@@ -1,4 +1,5 @@
 #!/bin/bash
 
-cd $(dirname "$0")
+PM3PATH=$(dirname "$0")
+cd "$PM3PATH" || exit 1
 . proxmark3.sh
diff --git a/flash-bootrom.sh b/flash-bootrom.sh
index 27e1fb526..d4fcdc975 100755
--- a/flash-bootrom.sh
+++ b/flash-bootrom.sh
@@ -1,4 +1,5 @@
 #!/bin/bash
 
-cd $(dirname "$0")
+PM3PATH=$(dirname "$0")
+cd "$PM3PATH" || exit 1
 . proxmark3.sh
diff --git a/flash-fullimage.sh b/flash-fullimage.sh
index 27e1fb526..d4fcdc975 100755
--- a/flash-fullimage.sh
+++ b/flash-fullimage.sh
@@ -1,4 +1,5 @@
 #!/bin/bash
 
-cd $(dirname "$0")
+PM3PATH=$(dirname "$0")
+cd "$PM3PATH" || exit 1
 . proxmark3.sh
diff --git a/include/at91sam7s512.h b/include/at91sam7s512.h
index db27893e2..bca65bdf3 100644
--- a/include/at91sam7s512.h
+++ b/include/at91sam7s512.h
@@ -71,126 +71,126 @@ typedef volatile unsigned int AT91_REG;// Hardware register definition
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_SYS {
-    AT91_REG	 AIC_SMR[32]; 	// Source Mode Register
-    AT91_REG	 AIC_SVR[32]; 	// Source Vector Register
-    AT91_REG	 AIC_IVR; 	// IRQ Vector Register
-    AT91_REG	 AIC_FVR; 	// FIQ Vector Register
-    AT91_REG	 AIC_ISR; 	// Interrupt Status Register
-    AT91_REG	 AIC_IPR; 	// Interrupt Pending Register
-    AT91_REG	 AIC_IMR; 	// Interrupt Mask Register
-    AT91_REG	 AIC_CISR; 	// Core Interrupt Status Register
-    AT91_REG	 Reserved0[2]; 	//
-    AT91_REG	 AIC_IECR; 	// Interrupt Enable Command Register
-    AT91_REG	 AIC_IDCR; 	// Interrupt Disable Command Register
-    AT91_REG	 AIC_ICCR; 	// Interrupt Clear Command Register
-    AT91_REG	 AIC_ISCR; 	// Interrupt Set Command Register
-    AT91_REG	 AIC_EOICR; 	// End of Interrupt Command Register
-    AT91_REG	 AIC_SPU; 	// Spurious Vector Register
-    AT91_REG	 AIC_DCR; 	// Debug Control Register (Protect)
-    AT91_REG	 Reserved1[1]; 	//
-    AT91_REG	 AIC_FFER; 	// Fast Forcing Enable Register
-    AT91_REG	 AIC_FFDR; 	// Fast Forcing Disable Register
-    AT91_REG	 AIC_FFSR; 	// Fast Forcing Status Register
-    AT91_REG	 Reserved2[45]; 	//
-    AT91_REG	 DBGU_CR; 	// Control Register
-    AT91_REG	 DBGU_MR; 	// Mode Register
-    AT91_REG	 DBGU_IER; 	// Interrupt Enable Register
-    AT91_REG	 DBGU_IDR; 	// Interrupt Disable Register
-    AT91_REG	 DBGU_IMR; 	// Interrupt Mask Register
-    AT91_REG	 DBGU_CSR; 	// Channel Status Register
-    AT91_REG	 DBGU_RHR; 	// Receiver Holding Register
-    AT91_REG	 DBGU_THR; 	// Transmitter Holding Register
-    AT91_REG	 DBGU_BRGR; 	// Baud Rate Generator Register
-    AT91_REG	 Reserved3[7]; 	//
-    AT91_REG	 DBGU_CIDR; 	// Chip ID Register
-    AT91_REG	 DBGU_EXID; 	// Chip ID Extension Register
-    AT91_REG	 DBGU_FNTR; 	// Force NTRST Register
-    AT91_REG	 Reserved4[45]; 	//
-    AT91_REG	 DBGU_RPR; 	// Receive Pointer Register
-    AT91_REG	 DBGU_RCR; 	// Receive Counter Register
-    AT91_REG	 DBGU_TPR; 	// Transmit Pointer Register
-    AT91_REG	 DBGU_TCR; 	// Transmit Counter Register
-    AT91_REG	 DBGU_RNPR; 	// Receive Next Pointer Register
-    AT91_REG	 DBGU_RNCR; 	// Receive Next Counter Register
-    AT91_REG	 DBGU_TNPR; 	// Transmit Next Pointer Register
-    AT91_REG	 DBGU_TNCR; 	// Transmit Next Counter Register
-    AT91_REG	 DBGU_PTCR; 	// PDC Transfer Control Register
-    AT91_REG	 DBGU_PTSR; 	// PDC Transfer Status Register
-    AT91_REG	 Reserved5[54]; 	//
-    AT91_REG	 PIOA_PER; 	// PIO Enable Register
-    AT91_REG	 PIOA_PDR; 	// PIO Disable Register
-    AT91_REG	 PIOA_PSR; 	// PIO Status Register
-    AT91_REG	 Reserved6[1]; 	//
-    AT91_REG	 PIOA_OER; 	// Output Enable Register
-    AT91_REG	 PIOA_ODR; 	// Output Disable Registerr
-    AT91_REG	 PIOA_OSR; 	// Output Status Register
-    AT91_REG	 Reserved7[1]; 	//
-    AT91_REG	 PIOA_IFER; 	// Input Filter Enable Register
-    AT91_REG	 PIOA_IFDR; 	// Input Filter Disable Register
-    AT91_REG	 PIOA_IFSR; 	// Input Filter Status Register
-    AT91_REG	 Reserved8[1]; 	//
-    AT91_REG	 PIOA_SODR; 	// Set Output Data Register
-    AT91_REG	 PIOA_CODR; 	// Clear Output Data Register
-    AT91_REG	 PIOA_ODSR; 	// Output Data Status Register
-    AT91_REG	 PIOA_PDSR; 	// Pin Data Status Register
-    AT91_REG	 PIOA_IER; 	// Interrupt Enable Register
-    AT91_REG	 PIOA_IDR; 	// Interrupt Disable Register
-    AT91_REG	 PIOA_IMR; 	// Interrupt Mask Register
-    AT91_REG	 PIOA_ISR; 	// Interrupt Status Register
-    AT91_REG	 PIOA_MDER; 	// Multi-driver Enable Register
-    AT91_REG	 PIOA_MDDR; 	// Multi-driver Disable Register
-    AT91_REG	 PIOA_MDSR; 	// Multi-driver Status Register
-    AT91_REG	 Reserved9[1]; 	//
-    AT91_REG	 PIOA_PPUDR; 	// Pull-up Disable Register
-    AT91_REG	 PIOA_PPUER; 	// Pull-up Enable Register
-    AT91_REG	 PIOA_PPUSR; 	// Pull-up Status Register
-    AT91_REG	 Reserved10[1]; 	//
-    AT91_REG	 PIOA_ASR; 	// Select A Register
-    AT91_REG	 PIOA_BSR; 	// Select B Register
-    AT91_REG	 PIOA_ABSR; 	// AB Select Status Register
-    AT91_REG	 Reserved11[9]; 	//
-    AT91_REG	 PIOA_OWER; 	// Output Write Enable Register
-    AT91_REG	 PIOA_OWDR; 	// Output Write Disable Register
-    AT91_REG	 PIOA_OWSR; 	// Output Write Status Register
-    AT91_REG	 Reserved12[469]; 	//
-    AT91_REG	 PMC_SCER; 	// System Clock Enable Register
-    AT91_REG	 PMC_SCDR; 	// System Clock Disable Register
-    AT91_REG	 PMC_SCSR; 	// System Clock Status Register
-    AT91_REG	 Reserved13[1]; 	//
-    AT91_REG	 PMC_PCER; 	// Peripheral Clock Enable Register
-    AT91_REG	 PMC_PCDR; 	// Peripheral Clock Disable Register
-    AT91_REG	 PMC_PCSR; 	// Peripheral Clock Status Register
-    AT91_REG	 Reserved14[1]; 	//
-    AT91_REG	 PMC_MOR; 	// Main Oscillator Register
-    AT91_REG	 PMC_MCFR; 	// Main Clock  Frequency Register
-    AT91_REG	 Reserved15[1]; 	//
-    AT91_REG	 PMC_PLLR; 	// PLL Register
-    AT91_REG	 PMC_MCKR; 	// Master Clock Register
-    AT91_REG	 Reserved16[3]; 	//
-    AT91_REG	 PMC_PCKR[3]; 	// Programmable Clock Register
-    AT91_REG	 Reserved17[5]; 	//
-    AT91_REG	 PMC_IER; 	// Interrupt Enable Register
-    AT91_REG	 PMC_IDR; 	// Interrupt Disable Register
-    AT91_REG	 PMC_SR; 	// Status Register
-    AT91_REG	 PMC_IMR; 	// Interrupt Mask Register
-    AT91_REG	 Reserved18[36]; 	//
-    AT91_REG	 RSTC_RCR; 	// Reset Control Register
-    AT91_REG	 RSTC_RSR; 	// Reset Status Register
-    AT91_REG	 RSTC_RMR; 	// Reset Mode Register
-    AT91_REG	 Reserved19[5]; 	//
-    AT91_REG	 RTTC_RTMR; 	// Real-time Mode Register
-    AT91_REG	 RTTC_RTAR; 	// Real-time Alarm Register
-    AT91_REG	 RTTC_RTVR; 	// Real-time Value Register
-    AT91_REG	 RTTC_RTSR; 	// Real-time Status Register
-    AT91_REG	 PITC_PIMR; 	// Period Interval Mode Register
-    AT91_REG	 PITC_PISR; 	// Period Interval Status Register
-    AT91_REG	 PITC_PIVR; 	// Period Interval Value Register
-    AT91_REG	 PITC_PIIR; 	// Period Interval Image Register
-    AT91_REG	 WDTC_WDCR; 	// Watchdog Control Register
-    AT91_REG	 WDTC_WDMR; 	// Watchdog Mode Register
-    AT91_REG	 WDTC_WDSR; 	// Watchdog Status Register
-    AT91_REG	 Reserved20[5]; 	//
-    AT91_REG	 VREG_MR; 	// Voltage Regulator Mode Register
+    AT91_REG     AIC_SMR[32];   // Source Mode Register
+    AT91_REG     AIC_SVR[32];   // Source Vector Register
+    AT91_REG     AIC_IVR;   // IRQ Vector Register
+    AT91_REG     AIC_FVR;   // FIQ Vector Register
+    AT91_REG     AIC_ISR;   // Interrupt Status Register
+    AT91_REG     AIC_IPR;   // Interrupt Pending Register
+    AT91_REG     AIC_IMR;   // Interrupt Mask Register
+    AT91_REG     AIC_CISR;  // Core Interrupt Status Register
+    AT91_REG     Reserved0[2];  //
+    AT91_REG     AIC_IECR;  // Interrupt Enable Command Register
+    AT91_REG     AIC_IDCR;  // Interrupt Disable Command Register
+    AT91_REG     AIC_ICCR;  // Interrupt Clear Command Register
+    AT91_REG     AIC_ISCR;  // Interrupt Set Command Register
+    AT91_REG     AIC_EOICR;     // End of Interrupt Command Register
+    AT91_REG     AIC_SPU;   // Spurious Vector Register
+    AT91_REG     AIC_DCR;   // Debug Control Register (Protect)
+    AT91_REG     Reserved1[1];  //
+    AT91_REG     AIC_FFER;  // Fast Forcing Enable Register
+    AT91_REG     AIC_FFDR;  // Fast Forcing Disable Register
+    AT91_REG     AIC_FFSR;  // Fast Forcing Status Register
+    AT91_REG     Reserved2[45];     //
+    AT91_REG     DBGU_CR;   // Control Register
+    AT91_REG     DBGU_MR;   // Mode Register
+    AT91_REG     DBGU_IER;  // Interrupt Enable Register
+    AT91_REG     DBGU_IDR;  // Interrupt Disable Register
+    AT91_REG     DBGU_IMR;  // Interrupt Mask Register
+    AT91_REG     DBGU_CSR;  // Channel Status Register
+    AT91_REG     DBGU_RHR;  // Receiver Holding Register
+    AT91_REG     DBGU_THR;  // Transmitter Holding Register
+    AT91_REG     DBGU_BRGR;     // Baud Rate Generator Register
+    AT91_REG     Reserved3[7];  //
+    AT91_REG     DBGU_CIDR;     // Chip ID Register
+    AT91_REG     DBGU_EXID;     // Chip ID Extension Register
+    AT91_REG     DBGU_FNTR;     // Force NTRST Register
+    AT91_REG     Reserved4[45];     //
+    AT91_REG     DBGU_RPR;  // Receive Pointer Register
+    AT91_REG     DBGU_RCR;  // Receive Counter Register
+    AT91_REG     DBGU_TPR;  // Transmit Pointer Register
+    AT91_REG     DBGU_TCR;  // Transmit Counter Register
+    AT91_REG     DBGU_RNPR;     // Receive Next Pointer Register
+    AT91_REG     DBGU_RNCR;     // Receive Next Counter Register
+    AT91_REG     DBGU_TNPR;     // Transmit Next Pointer Register
+    AT91_REG     DBGU_TNCR;     // Transmit Next Counter Register
+    AT91_REG     DBGU_PTCR;     // PDC Transfer Control Register
+    AT91_REG     DBGU_PTSR;     // PDC Transfer Status Register
+    AT91_REG     Reserved5[54];     //
+    AT91_REG     PIOA_PER;  // PIO Enable Register
+    AT91_REG     PIOA_PDR;  // PIO Disable Register
+    AT91_REG     PIOA_PSR;  // PIO Status Register
+    AT91_REG     Reserved6[1];  //
+    AT91_REG     PIOA_OER;  // Output Enable Register
+    AT91_REG     PIOA_ODR;  // Output Disable Registerr
+    AT91_REG     PIOA_OSR;  // Output Status Register
+    AT91_REG     Reserved7[1];  //
+    AT91_REG     PIOA_IFER;     // Input Filter Enable Register
+    AT91_REG     PIOA_IFDR;     // Input Filter Disable Register
+    AT91_REG     PIOA_IFSR;     // Input Filter Status Register
+    AT91_REG     Reserved8[1];  //
+    AT91_REG     PIOA_SODR;     // Set Output Data Register
+    AT91_REG     PIOA_CODR;     // Clear Output Data Register
+    AT91_REG     PIOA_ODSR;     // Output Data Status Register
+    AT91_REG     PIOA_PDSR;     // Pin Data Status Register
+    AT91_REG     PIOA_IER;  // Interrupt Enable Register
+    AT91_REG     PIOA_IDR;  // Interrupt Disable Register
+    AT91_REG     PIOA_IMR;  // Interrupt Mask Register
+    AT91_REG     PIOA_ISR;  // Interrupt Status Register
+    AT91_REG     PIOA_MDER;     // Multi-driver Enable Register
+    AT91_REG     PIOA_MDDR;     // Multi-driver Disable Register
+    AT91_REG     PIOA_MDSR;     // Multi-driver Status Register
+    AT91_REG     Reserved9[1];  //
+    AT91_REG     PIOA_PPUDR;    // Pull-up Disable Register
+    AT91_REG     PIOA_PPUER;    // Pull-up Enable Register
+    AT91_REG     PIOA_PPUSR;    // Pull-up Status Register
+    AT91_REG     Reserved10[1];     //
+    AT91_REG     PIOA_ASR;  // Select A Register
+    AT91_REG     PIOA_BSR;  // Select B Register
+    AT91_REG     PIOA_ABSR;     // AB Select Status Register
+    AT91_REG     Reserved11[9];     //
+    AT91_REG     PIOA_OWER;     // Output Write Enable Register
+    AT91_REG     PIOA_OWDR;     // Output Write Disable Register
+    AT91_REG     PIOA_OWSR;     // Output Write Status Register
+    AT91_REG     Reserved12[469];   //
+    AT91_REG     PMC_SCER;  // System Clock Enable Register
+    AT91_REG     PMC_SCDR;  // System Clock Disable Register
+    AT91_REG     PMC_SCSR;  // System Clock Status Register
+    AT91_REG     Reserved13[1];     //
+    AT91_REG     PMC_PCER;  // Peripheral Clock Enable Register
+    AT91_REG     PMC_PCDR;  // Peripheral Clock Disable Register
+    AT91_REG     PMC_PCSR;  // Peripheral Clock Status Register
+    AT91_REG     Reserved14[1];     //
+    AT91_REG     PMC_MOR;   // Main Oscillator Register
+    AT91_REG     PMC_MCFR;  // Main Clock  Frequency Register
+    AT91_REG     Reserved15[1];     //
+    AT91_REG     PMC_PLLR;  // PLL Register
+    AT91_REG     PMC_MCKR;  // Master Clock Register
+    AT91_REG     Reserved16[3];     //
+    AT91_REG     PMC_PCKR[3];   // Programmable Clock Register
+    AT91_REG     Reserved17[5];     //
+    AT91_REG     PMC_IER;   // Interrupt Enable Register
+    AT91_REG     PMC_IDR;   // Interrupt Disable Register
+    AT91_REG     PMC_SR;    // Status Register
+    AT91_REG     PMC_IMR;   // Interrupt Mask Register
+    AT91_REG     Reserved18[36];    //
+    AT91_REG     RSTC_RCR;  // Reset Control Register
+    AT91_REG     RSTC_RSR;  // Reset Status Register
+    AT91_REG     RSTC_RMR;  // Reset Mode Register
+    AT91_REG     Reserved19[5];     //
+    AT91_REG     RTTC_RTMR;     // Real-time Mode Register
+    AT91_REG     RTTC_RTAR;     // Real-time Alarm Register
+    AT91_REG     RTTC_RTVR;     // Real-time Value Register
+    AT91_REG     RTTC_RTSR;     // Real-time Status Register
+    AT91_REG     PITC_PIMR;     // Period Interval Mode Register
+    AT91_REG     PITC_PISR;     // Period Interval Status Register
+    AT91_REG     PITC_PIVR;     // Period Interval Value Register
+    AT91_REG     PITC_PIIR;     // Period Interval Image Register
+    AT91_REG     WDTC_WDCR;     // Watchdog Control Register
+    AT91_REG     WDTC_WDMR;     // Watchdog Mode Register
+    AT91_REG     WDTC_WDSR;     // Watchdog Status Register
+    AT91_REG     Reserved20[5];     //
+    AT91_REG     VREG_MR;   // Voltage Regulator Mode Register
 } AT91S_SYS, *AT91PS_SYS;
 #else
 
@@ -201,59 +201,59 @@ typedef struct _AT91S_SYS {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_AIC {
-    AT91_REG	 AIC_SMR[32]; 	// Source Mode Register
-    AT91_REG	 AIC_SVR[32]; 	// Source Vector Register
-    AT91_REG	 AIC_IVR; 	// IRQ Vector Register
-    AT91_REG	 AIC_FVR; 	// FIQ Vector Register
-    AT91_REG	 AIC_ISR; 	// Interrupt Status Register
-    AT91_REG	 AIC_IPR; 	// Interrupt Pending Register
-    AT91_REG	 AIC_IMR; 	// Interrupt Mask Register
-    AT91_REG	 AIC_CISR; 	// Core Interrupt Status Register
-    AT91_REG	 Reserved0[2]; 	//
-    AT91_REG	 AIC_IECR; 	// Interrupt Enable Command Register
-    AT91_REG	 AIC_IDCR; 	// Interrupt Disable Command Register
-    AT91_REG	 AIC_ICCR; 	// Interrupt Clear Command Register
-    AT91_REG	 AIC_ISCR; 	// Interrupt Set Command Register
-    AT91_REG	 AIC_EOICR; 	// End of Interrupt Command Register
-    AT91_REG	 AIC_SPU; 	// Spurious Vector Register
-    AT91_REG	 AIC_DCR; 	// Debug Control Register (Protect)
-    AT91_REG	 Reserved1[1]; 	//
-    AT91_REG	 AIC_FFER; 	// Fast Forcing Enable Register
-    AT91_REG	 AIC_FFDR; 	// Fast Forcing Disable Register
-    AT91_REG	 AIC_FFSR; 	// Fast Forcing Status Register
+    AT91_REG     AIC_SMR[32];   // Source Mode Register
+    AT91_REG     AIC_SVR[32];   // Source Vector Register
+    AT91_REG     AIC_IVR;   // IRQ Vector Register
+    AT91_REG     AIC_FVR;   // FIQ Vector Register
+    AT91_REG     AIC_ISR;   // Interrupt Status Register
+    AT91_REG     AIC_IPR;   // Interrupt Pending Register
+    AT91_REG     AIC_IMR;   // Interrupt Mask Register
+    AT91_REG     AIC_CISR;  // Core Interrupt Status Register
+    AT91_REG     Reserved0[2];  //
+    AT91_REG     AIC_IECR;  // Interrupt Enable Command Register
+    AT91_REG     AIC_IDCR;  // Interrupt Disable Command Register
+    AT91_REG     AIC_ICCR;  // Interrupt Clear Command Register
+    AT91_REG     AIC_ISCR;  // Interrupt Set Command Register
+    AT91_REG     AIC_EOICR;     // End of Interrupt Command Register
+    AT91_REG     AIC_SPU;   // Spurious Vector Register
+    AT91_REG     AIC_DCR;   // Debug Control Register (Protect)
+    AT91_REG     Reserved1[1];  //
+    AT91_REG     AIC_FFER;  // Fast Forcing Enable Register
+    AT91_REG     AIC_FFDR;  // Fast Forcing Disable Register
+    AT91_REG     AIC_FFSR;  // Fast Forcing Status Register
 } AT91S_AIC, *AT91PS_AIC;
 #else
-#define AIC_SMR         (AT91_CAST(AT91_REG *) 	0x00000000) // (AIC_SMR) Source Mode Register
-#define AIC_SVR         (AT91_CAST(AT91_REG *) 	0x00000080) // (AIC_SVR) Source Vector Register
-#define AIC_IVR         (AT91_CAST(AT91_REG *) 	0x00000100) // (AIC_IVR) IRQ Vector Register
-#define AIC_FVR         (AT91_CAST(AT91_REG *) 	0x00000104) // (AIC_FVR) FIQ Vector Register
-#define AIC_ISR         (AT91_CAST(AT91_REG *) 	0x00000108) // (AIC_ISR) Interrupt Status Register
-#define AIC_IPR         (AT91_CAST(AT91_REG *) 	0x0000010C) // (AIC_IPR) Interrupt Pending Register
-#define AIC_IMR         (AT91_CAST(AT91_REG *) 	0x00000110) // (AIC_IMR) Interrupt Mask Register
-#define AIC_CISR        (AT91_CAST(AT91_REG *) 	0x00000114) // (AIC_CISR) Core Interrupt Status Register
-#define AIC_IECR        (AT91_CAST(AT91_REG *) 	0x00000120) // (AIC_IECR) Interrupt Enable Command Register
-#define AIC_IDCR        (AT91_CAST(AT91_REG *) 	0x00000124) // (AIC_IDCR) Interrupt Disable Command Register
-#define AIC_ICCR        (AT91_CAST(AT91_REG *) 	0x00000128) // (AIC_ICCR) Interrupt Clear Command Register
-#define AIC_ISCR        (AT91_CAST(AT91_REG *) 	0x0000012C) // (AIC_ISCR) Interrupt Set Command Register
-#define AIC_EOICR       (AT91_CAST(AT91_REG *) 	0x00000130) // (AIC_EOICR) End of Interrupt Command Register
-#define AIC_SPU         (AT91_CAST(AT91_REG *) 	0x00000134) // (AIC_SPU) Spurious Vector Register
-#define AIC_DCR         (AT91_CAST(AT91_REG *) 	0x00000138) // (AIC_DCR) Debug Control Register (Protect)
-#define AIC_FFER        (AT91_CAST(AT91_REG *) 	0x00000140) // (AIC_FFER) Fast Forcing Enable Register
-#define AIC_FFDR        (AT91_CAST(AT91_REG *) 	0x00000144) // (AIC_FFDR) Fast Forcing Disable Register
-#define AIC_FFSR        (AT91_CAST(AT91_REG *) 	0x00000148) // (AIC_FFSR) Fast Forcing Status Register
+#define AIC_SMR         (AT91_CAST(AT91_REG *)  0x00000000) // (AIC_SMR) Source Mode Register
+#define AIC_SVR         (AT91_CAST(AT91_REG *)  0x00000080) // (AIC_SVR) Source Vector Register
+#define AIC_IVR         (AT91_CAST(AT91_REG *)  0x00000100) // (AIC_IVR) IRQ Vector Register
+#define AIC_FVR         (AT91_CAST(AT91_REG *)  0x00000104) // (AIC_FVR) FIQ Vector Register
+#define AIC_ISR         (AT91_CAST(AT91_REG *)  0x00000108) // (AIC_ISR) Interrupt Status Register
+#define AIC_IPR         (AT91_CAST(AT91_REG *)  0x0000010C) // (AIC_IPR) Interrupt Pending Register
+#define AIC_IMR         (AT91_CAST(AT91_REG *)  0x00000110) // (AIC_IMR) Interrupt Mask Register
+#define AIC_CISR        (AT91_CAST(AT91_REG *)  0x00000114) // (AIC_CISR) Core Interrupt Status Register
+#define AIC_IECR        (AT91_CAST(AT91_REG *)  0x00000120) // (AIC_IECR) Interrupt Enable Command Register
+#define AIC_IDCR        (AT91_CAST(AT91_REG *)  0x00000124) // (AIC_IDCR) Interrupt Disable Command Register
+#define AIC_ICCR        (AT91_CAST(AT91_REG *)  0x00000128) // (AIC_ICCR) Interrupt Clear Command Register
+#define AIC_ISCR        (AT91_CAST(AT91_REG *)  0x0000012C) // (AIC_ISCR) Interrupt Set Command Register
+#define AIC_EOICR       (AT91_CAST(AT91_REG *)  0x00000130) // (AIC_EOICR) End of Interrupt Command Register
+#define AIC_SPU         (AT91_CAST(AT91_REG *)  0x00000134) // (AIC_SPU) Spurious Vector Register
+#define AIC_DCR         (AT91_CAST(AT91_REG *)  0x00000138) // (AIC_DCR) Debug Control Register (Protect)
+#define AIC_FFER        (AT91_CAST(AT91_REG *)  0x00000140) // (AIC_FFER) Fast Forcing Enable Register
+#define AIC_FFDR        (AT91_CAST(AT91_REG *)  0x00000144) // (AIC_FFDR) Fast Forcing Disable Register
+#define AIC_FFSR        (AT91_CAST(AT91_REG *)  0x00000148) // (AIC_FFSR) Fast Forcing Status Register
 
 #endif
 // -------- AIC_SMR : (AIC Offset: 0x0) Control Register --------
 #define AT91C_AIC_PRIOR       (0x7 <<  0) // (AIC) Priority Level
-#define 	AT91C_AIC_PRIOR_LOWEST               (0x0) // (AIC) Lowest priority level
-#define 	AT91C_AIC_PRIOR_HIGHEST              (0x7) // (AIC) Highest priority level
+#define     AT91C_AIC_PRIOR_LOWEST               (0x0) // (AIC) Lowest priority level
+#define     AT91C_AIC_PRIOR_HIGHEST              (0x7) // (AIC) Highest priority level
 #define AT91C_AIC_SRCTYPE     (0x3 <<  5) // (AIC) Interrupt Source Type
-#define 	AT91C_AIC_SRCTYPE_INT_HIGH_LEVEL       (0x0 <<  5) // (AIC) Internal Sources Code Label High-level Sensitive
-#define 	AT91C_AIC_SRCTYPE_EXT_LOW_LEVEL        (0x0 <<  5) // (AIC) External Sources Code Label Low-level Sensitive
-#define 	AT91C_AIC_SRCTYPE_INT_POSITIVE_EDGE    (0x1 <<  5) // (AIC) Internal Sources Code Label Positive Edge triggered
-#define 	AT91C_AIC_SRCTYPE_EXT_NEGATIVE_EDGE    (0x1 <<  5) // (AIC) External Sources Code Label Negative Edge triggered
-#define 	AT91C_AIC_SRCTYPE_HIGH_LEVEL           (0x2 <<  5) // (AIC) Internal Or External Sources Code Label High-level Sensitive
-#define 	AT91C_AIC_SRCTYPE_POSITIVE_EDGE        (0x3 <<  5) // (AIC) Internal Or External Sources Code Label Positive Edge triggered
+#define     AT91C_AIC_SRCTYPE_INT_HIGH_LEVEL       (0x0 <<  5) // (AIC) Internal Sources Code Label High-level Sensitive
+#define     AT91C_AIC_SRCTYPE_EXT_LOW_LEVEL        (0x0 <<  5) // (AIC) External Sources Code Label Low-level Sensitive
+#define     AT91C_AIC_SRCTYPE_INT_POSITIVE_EDGE    (0x1 <<  5) // (AIC) Internal Sources Code Label Positive Edge triggered
+#define     AT91C_AIC_SRCTYPE_EXT_NEGATIVE_EDGE    (0x1 <<  5) // (AIC) External Sources Code Label Negative Edge triggered
+#define     AT91C_AIC_SRCTYPE_HIGH_LEVEL           (0x2 <<  5) // (AIC) Internal Or External Sources Code Label High-level Sensitive
+#define     AT91C_AIC_SRCTYPE_POSITIVE_EDGE        (0x3 <<  5) // (AIC) Internal Or External Sources Code Label Positive Edge triggered
 // -------- AIC_CISR : (AIC Offset: 0x114) AIC Core Interrupt Status Register --------
 #define AT91C_AIC_NFIQ        (0x1 <<  0) // (AIC) NFIQ Status
 #define AT91C_AIC_NIRQ        (0x1 <<  1) // (AIC) NIRQ Status
@@ -266,28 +266,28 @@ typedef struct _AT91S_AIC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_PDC {
-    AT91_REG	 PDC_RPR; 	// Receive Pointer Register
-    AT91_REG	 PDC_RCR; 	// Receive Counter Register
-    AT91_REG	 PDC_TPR; 	// Transmit Pointer Register
-    AT91_REG	 PDC_TCR; 	// Transmit Counter Register
-    AT91_REG	 PDC_RNPR; 	// Receive Next Pointer Register
-    AT91_REG	 PDC_RNCR; 	// Receive Next Counter Register
-    AT91_REG	 PDC_TNPR; 	// Transmit Next Pointer Register
-    AT91_REG	 PDC_TNCR; 	// Transmit Next Counter Register
-    AT91_REG	 PDC_PTCR; 	// PDC Transfer Control Register
-    AT91_REG	 PDC_PTSR; 	// PDC Transfer Status Register
+    AT91_REG     PDC_RPR;   // Receive Pointer Register
+    AT91_REG     PDC_RCR;   // Receive Counter Register
+    AT91_REG     PDC_TPR;   // Transmit Pointer Register
+    AT91_REG     PDC_TCR;   // Transmit Counter Register
+    AT91_REG     PDC_RNPR;  // Receive Next Pointer Register
+    AT91_REG     PDC_RNCR;  // Receive Next Counter Register
+    AT91_REG     PDC_TNPR;  // Transmit Next Pointer Register
+    AT91_REG     PDC_TNCR;  // Transmit Next Counter Register
+    AT91_REG     PDC_PTCR;  // PDC Transfer Control Register
+    AT91_REG     PDC_PTSR;  // PDC Transfer Status Register
 } AT91S_PDC, *AT91PS_PDC;
 #else
-#define PDC_RPR         (AT91_CAST(AT91_REG *) 	0x00000000) // (PDC_RPR) Receive Pointer Register
-#define PDC_RCR         (AT91_CAST(AT91_REG *) 	0x00000004) // (PDC_RCR) Receive Counter Register
-#define PDC_TPR         (AT91_CAST(AT91_REG *) 	0x00000008) // (PDC_TPR) Transmit Pointer Register
-#define PDC_TCR         (AT91_CAST(AT91_REG *) 	0x0000000C) // (PDC_TCR) Transmit Counter Register
-#define PDC_RNPR        (AT91_CAST(AT91_REG *) 	0x00000010) // (PDC_RNPR) Receive Next Pointer Register
-#define PDC_RNCR        (AT91_CAST(AT91_REG *) 	0x00000014) // (PDC_RNCR) Receive Next Counter Register
-#define PDC_TNPR        (AT91_CAST(AT91_REG *) 	0x00000018) // (PDC_TNPR) Transmit Next Pointer Register
-#define PDC_TNCR        (AT91_CAST(AT91_REG *) 	0x0000001C) // (PDC_TNCR) Transmit Next Counter Register
-#define PDC_PTCR        (AT91_CAST(AT91_REG *) 	0x00000020) // (PDC_PTCR) PDC Transfer Control Register
-#define PDC_PTSR        (AT91_CAST(AT91_REG *) 	0x00000024) // (PDC_PTSR) PDC Transfer Status Register
+#define PDC_RPR         (AT91_CAST(AT91_REG *)  0x00000000) // (PDC_RPR) Receive Pointer Register
+#define PDC_RCR         (AT91_CAST(AT91_REG *)  0x00000004) // (PDC_RCR) Receive Counter Register
+#define PDC_TPR         (AT91_CAST(AT91_REG *)  0x00000008) // (PDC_TPR) Transmit Pointer Register
+#define PDC_TCR         (AT91_CAST(AT91_REG *)  0x0000000C) // (PDC_TCR) Transmit Counter Register
+#define PDC_RNPR        (AT91_CAST(AT91_REG *)  0x00000010) // (PDC_RNPR) Receive Next Pointer Register
+#define PDC_RNCR        (AT91_CAST(AT91_REG *)  0x00000014) // (PDC_RNCR) Receive Next Counter Register
+#define PDC_TNPR        (AT91_CAST(AT91_REG *)  0x00000018) // (PDC_TNPR) Transmit Next Pointer Register
+#define PDC_TNCR        (AT91_CAST(AT91_REG *)  0x0000001C) // (PDC_TNCR) Transmit Next Counter Register
+#define PDC_PTCR        (AT91_CAST(AT91_REG *)  0x00000020) // (PDC_PTCR) PDC Transfer Control Register
+#define PDC_PTSR        (AT91_CAST(AT91_REG *)  0x00000024) // (PDC_PTSR) PDC Transfer Status Register
 
 #endif
 // -------- PDC_PTCR : (PDC Offset: 0x20) PDC Transfer Control Register --------
@@ -302,44 +302,44 @@ typedef struct _AT91S_PDC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_DBGU {
-    AT91_REG	 DBGU_CR; 	// Control Register
-    AT91_REG	 DBGU_MR; 	// Mode Register
-    AT91_REG	 DBGU_IER; 	// Interrupt Enable Register
-    AT91_REG	 DBGU_IDR; 	// Interrupt Disable Register
-    AT91_REG	 DBGU_IMR; 	// Interrupt Mask Register
-    AT91_REG	 DBGU_CSR; 	// Channel Status Register
-    AT91_REG	 DBGU_RHR; 	// Receiver Holding Register
-    AT91_REG	 DBGU_THR; 	// Transmitter Holding Register
-    AT91_REG	 DBGU_BRGR; 	// Baud Rate Generator Register
-    AT91_REG	 Reserved0[7]; 	//
-    AT91_REG	 DBGU_CIDR; 	// Chip ID Register
-    AT91_REG	 DBGU_EXID; 	// Chip ID Extension Register
-    AT91_REG	 DBGU_FNTR; 	// Force NTRST Register
-    AT91_REG	 Reserved1[45]; 	//
-    AT91_REG	 DBGU_RPR; 	// Receive Pointer Register
-    AT91_REG	 DBGU_RCR; 	// Receive Counter Register
-    AT91_REG	 DBGU_TPR; 	// Transmit Pointer Register
-    AT91_REG	 DBGU_TCR; 	// Transmit Counter Register
-    AT91_REG	 DBGU_RNPR; 	// Receive Next Pointer Register
-    AT91_REG	 DBGU_RNCR; 	// Receive Next Counter Register
-    AT91_REG	 DBGU_TNPR; 	// Transmit Next Pointer Register
-    AT91_REG	 DBGU_TNCR; 	// Transmit Next Counter Register
-    AT91_REG	 DBGU_PTCR; 	// PDC Transfer Control Register
-    AT91_REG	 DBGU_PTSR; 	// PDC Transfer Status Register
+    AT91_REG     DBGU_CR;   // Control Register
+    AT91_REG     DBGU_MR;   // Mode Register
+    AT91_REG     DBGU_IER;  // Interrupt Enable Register
+    AT91_REG     DBGU_IDR;  // Interrupt Disable Register
+    AT91_REG     DBGU_IMR;  // Interrupt Mask Register
+    AT91_REG     DBGU_CSR;  // Channel Status Register
+    AT91_REG     DBGU_RHR;  // Receiver Holding Register
+    AT91_REG     DBGU_THR;  // Transmitter Holding Register
+    AT91_REG     DBGU_BRGR;     // Baud Rate Generator Register
+    AT91_REG     Reserved0[7];  //
+    AT91_REG     DBGU_CIDR;     // Chip ID Register
+    AT91_REG     DBGU_EXID;     // Chip ID Extension Register
+    AT91_REG     DBGU_FNTR;     // Force NTRST Register
+    AT91_REG     Reserved1[45];     //
+    AT91_REG     DBGU_RPR;  // Receive Pointer Register
+    AT91_REG     DBGU_RCR;  // Receive Counter Register
+    AT91_REG     DBGU_TPR;  // Transmit Pointer Register
+    AT91_REG     DBGU_TCR;  // Transmit Counter Register
+    AT91_REG     DBGU_RNPR;     // Receive Next Pointer Register
+    AT91_REG     DBGU_RNCR;     // Receive Next Counter Register
+    AT91_REG     DBGU_TNPR;     // Transmit Next Pointer Register
+    AT91_REG     DBGU_TNCR;     // Transmit Next Counter Register
+    AT91_REG     DBGU_PTCR;     // PDC Transfer Control Register
+    AT91_REG     DBGU_PTSR;     // PDC Transfer Status Register
 } AT91S_DBGU, *AT91PS_DBGU;
 #else
-#define DBGU_CR         (AT91_CAST(AT91_REG *) 	0x00000000) // (DBGU_CR) Control Register
-#define DBGU_MR         (AT91_CAST(AT91_REG *) 	0x00000004) // (DBGU_MR) Mode Register
-#define DBGU_IER        (AT91_CAST(AT91_REG *) 	0x00000008) // (DBGU_IER) Interrupt Enable Register
-#define DBGU_IDR        (AT91_CAST(AT91_REG *) 	0x0000000C) // (DBGU_IDR) Interrupt Disable Register
-#define DBGU_IMR        (AT91_CAST(AT91_REG *) 	0x00000010) // (DBGU_IMR) Interrupt Mask Register
-#define DBGU_CSR        (AT91_CAST(AT91_REG *) 	0x00000014) // (DBGU_CSR) Channel Status Register
-#define DBGU_RHR        (AT91_CAST(AT91_REG *) 	0x00000018) // (DBGU_RHR) Receiver Holding Register
-#define DBGU_THR        (AT91_CAST(AT91_REG *) 	0x0000001C) // (DBGU_THR) Transmitter Holding Register
-#define DBGU_BRGR       (AT91_CAST(AT91_REG *) 	0x00000020) // (DBGU_BRGR) Baud Rate Generator Register
-//#define DBGU_CIDR       (AT91_CAST(AT91_REG *) 	0x00000040) // (DBGU_CIDR) Chip ID Register
-#define DBGU_EXID       (AT91_CAST(AT91_REG *) 	0x00000044) // (DBGU_EXID) Chip ID Extension Register
-#define DBGU_FNTR       (AT91_CAST(AT91_REG *) 	0x00000048) // (DBGU_FNTR) Force NTRST Register
+#define DBGU_CR         (AT91_CAST(AT91_REG *)  0x00000000) // (DBGU_CR) Control Register
+#define DBGU_MR         (AT91_CAST(AT91_REG *)  0x00000004) // (DBGU_MR) Mode Register
+#define DBGU_IER        (AT91_CAST(AT91_REG *)  0x00000008) // (DBGU_IER) Interrupt Enable Register
+#define DBGU_IDR        (AT91_CAST(AT91_REG *)  0x0000000C) // (DBGU_IDR) Interrupt Disable Register
+#define DBGU_IMR        (AT91_CAST(AT91_REG *)  0x00000010) // (DBGU_IMR) Interrupt Mask Register
+#define DBGU_CSR        (AT91_CAST(AT91_REG *)  0x00000014) // (DBGU_CSR) Channel Status Register
+#define DBGU_RHR        (AT91_CAST(AT91_REG *)  0x00000018) // (DBGU_RHR) Receiver Holding Register
+#define DBGU_THR        (AT91_CAST(AT91_REG *)  0x0000001C) // (DBGU_THR) Transmitter Holding Register
+#define DBGU_BRGR       (AT91_CAST(AT91_REG *)  0x00000020) // (DBGU_BRGR) Baud Rate Generator Register
+//#define DBGU_CIDR       (AT91_CAST(AT91_REG *)    0x00000040) // (DBGU_CIDR) Chip ID Register
+#define DBGU_EXID       (AT91_CAST(AT91_REG *)  0x00000044) // (DBGU_EXID) Chip ID Extension Register
+#define DBGU_FNTR       (AT91_CAST(AT91_REG *)  0x00000048) // (DBGU_FNTR) Force NTRST Register
 
 #endif
 // -------- DBGU_CR : (DBGU Offset: 0x0) Debug Unit Control Register --------
@@ -352,17 +352,17 @@ typedef struct _AT91S_DBGU {
 #define AT91C_US_RSTSTA       (0x1 <<  8) // (DBGU) Reset Status Bits
 // -------- DBGU_MR : (DBGU Offset: 0x4) Debug Unit Mode Register --------
 #define AT91C_US_PAR          (0x7 <<  9) // (DBGU) Parity type
-#define 	AT91C_US_PAR_EVEN                 (0x0 <<  9) // (DBGU) Even Parity
-#define 	AT91C_US_PAR_ODD                  (0x1 <<  9) // (DBGU) Odd Parity
-#define 	AT91C_US_PAR_SPACE                (0x2 <<  9) // (DBGU) Parity forced to 0 (Space)
-#define 	AT91C_US_PAR_MARK                 (0x3 <<  9) // (DBGU) Parity forced to 1 (Mark)
-#define 	AT91C_US_PAR_NONE                 (0x4 <<  9) // (DBGU) No Parity
-#define 	AT91C_US_PAR_MULTI_DROP           (0x6 <<  9) // (DBGU) Multi-drop mode
+#define     AT91C_US_PAR_EVEN                 (0x0 <<  9) // (DBGU) Even Parity
+#define     AT91C_US_PAR_ODD                  (0x1 <<  9) // (DBGU) Odd Parity
+#define     AT91C_US_PAR_SPACE                (0x2 <<  9) // (DBGU) Parity forced to 0 (Space)
+#define     AT91C_US_PAR_MARK                 (0x3 <<  9) // (DBGU) Parity forced to 1 (Mark)
+#define     AT91C_US_PAR_NONE                 (0x4 <<  9) // (DBGU) No Parity
+#define     AT91C_US_PAR_MULTI_DROP           (0x6 <<  9) // (DBGU) Multi-drop mode
 #define AT91C_US_CHMODE       (0x3 << 14) // (DBGU) Channel Mode
-#define 	AT91C_US_CHMODE_NORMAL               (0x0 << 14) // (DBGU) Normal Mode: The USART channel operates as an RX/TX USART.
-#define 	AT91C_US_CHMODE_AUTO                 (0x1 << 14) // (DBGU) Automatic Echo: Receiver Data Input is connected to the TXD pin.
-#define 	AT91C_US_CHMODE_LOCAL                (0x2 << 14) // (DBGU) Local Loopback: Transmitter Output Signal is connected to Receiver Input Signal.
-#define 	AT91C_US_CHMODE_REMOTE               (0x3 << 14) // (DBGU) Remote Loopback: RXD pin is internally connected to TXD pin.
+#define     AT91C_US_CHMODE_NORMAL               (0x0 << 14) // (DBGU) Normal Mode: The USART channel operates as an RX/TX USART.
+#define     AT91C_US_CHMODE_AUTO                 (0x1 << 14) // (DBGU) Automatic Echo: Receiver Data Input is connected to the TXD pin.
+#define     AT91C_US_CHMODE_LOCAL                (0x2 << 14) // (DBGU) Local Loopback: Transmitter Output Signal is connected to Receiver Input Signal.
+#define     AT91C_US_CHMODE_REMOTE               (0x3 << 14) // (DBGU) Remote Loopback: RXD pin is internally connected to TXD pin.
 // -------- DBGU_IER : (DBGU Offset: 0x8) Debug Unit Interrupt Enable Register --------
 #define AT91C_US_RXRDY        (0x1 <<  0) // (DBGU) RXRDY Interrupt
 #define AT91C_US_TXRDY        (0x1 <<  1) // (DBGU) TXRDY Interrupt
@@ -387,72 +387,72 @@ typedef struct _AT91S_DBGU {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_PIO {
-    AT91_REG	 PIO_PER; 	// PIO Enable Register
-    AT91_REG	 PIO_PDR; 	// PIO Disable Register
-    AT91_REG	 PIO_PSR; 	// PIO Status Register
-    AT91_REG	 Reserved0[1]; 	//
-    AT91_REG	 PIO_OER; 	// Output Enable Register
-    AT91_REG	 PIO_ODR; 	// Output Disable Registerr
-    AT91_REG	 PIO_OSR; 	// Output Status Register
-    AT91_REG	 Reserved1[1]; 	//
-    AT91_REG	 PIO_IFER; 	// Input Filter Enable Register
-    AT91_REG	 PIO_IFDR; 	// Input Filter Disable Register
-    AT91_REG	 PIO_IFSR; 	// Input Filter Status Register
-    AT91_REG	 Reserved2[1]; 	//
-    AT91_REG	 PIO_SODR; 	// Set Output Data Register
-    AT91_REG	 PIO_CODR; 	// Clear Output Data Register
-    AT91_REG	 PIO_ODSR; 	// Output Data Status Register
-    AT91_REG	 PIO_PDSR; 	// Pin Data Status Register
-    AT91_REG	 PIO_IER; 	// Interrupt Enable Register
-    AT91_REG	 PIO_IDR; 	// Interrupt Disable Register
-    AT91_REG	 PIO_IMR; 	// Interrupt Mask Register
-    AT91_REG	 PIO_ISR; 	// Interrupt Status Register
-    AT91_REG	 PIO_MDER; 	// Multi-driver Enable Register
-    AT91_REG	 PIO_MDDR; 	// Multi-driver Disable Register
-    AT91_REG	 PIO_MDSR; 	// Multi-driver Status Register
-    AT91_REG	 Reserved3[1]; 	//
-    AT91_REG	 PIO_PPUDR; 	// Pull-up Disable Register
-    AT91_REG	 PIO_PPUER; 	// Pull-up Enable Register
-    AT91_REG	 PIO_PPUSR; 	// Pull-up Status Register
-    AT91_REG	 Reserved4[1]; 	//
-    AT91_REG	 PIO_ASR; 	// Select A Register
-    AT91_REG	 PIO_BSR; 	// Select B Register
-    AT91_REG	 PIO_ABSR; 	// AB Select Status Register
-    AT91_REG	 Reserved5[9]; 	//
-    AT91_REG	 PIO_OWER; 	// Output Write Enable Register
-    AT91_REG	 PIO_OWDR; 	// Output Write Disable Register
-    AT91_REG	 PIO_OWSR; 	// Output Write Status Register
+    AT91_REG     PIO_PER;   // PIO Enable Register
+    AT91_REG     PIO_PDR;   // PIO Disable Register
+    AT91_REG     PIO_PSR;   // PIO Status Register
+    AT91_REG     Reserved0[1];  //
+    AT91_REG     PIO_OER;   // Output Enable Register
+    AT91_REG     PIO_ODR;   // Output Disable Registerr
+    AT91_REG     PIO_OSR;   // Output Status Register
+    AT91_REG     Reserved1[1];  //
+    AT91_REG     PIO_IFER;  // Input Filter Enable Register
+    AT91_REG     PIO_IFDR;  // Input Filter Disable Register
+    AT91_REG     PIO_IFSR;  // Input Filter Status Register
+    AT91_REG     Reserved2[1];  //
+    AT91_REG     PIO_SODR;  // Set Output Data Register
+    AT91_REG     PIO_CODR;  // Clear Output Data Register
+    AT91_REG     PIO_ODSR;  // Output Data Status Register
+    AT91_REG     PIO_PDSR;  // Pin Data Status Register
+    AT91_REG     PIO_IER;   // Interrupt Enable Register
+    AT91_REG     PIO_IDR;   // Interrupt Disable Register
+    AT91_REG     PIO_IMR;   // Interrupt Mask Register
+    AT91_REG     PIO_ISR;   // Interrupt Status Register
+    AT91_REG     PIO_MDER;  // Multi-driver Enable Register
+    AT91_REG     PIO_MDDR;  // Multi-driver Disable Register
+    AT91_REG     PIO_MDSR;  // Multi-driver Status Register
+    AT91_REG     Reserved3[1];  //
+    AT91_REG     PIO_PPUDR;     // Pull-up Disable Register
+    AT91_REG     PIO_PPUER;     // Pull-up Enable Register
+    AT91_REG     PIO_PPUSR;     // Pull-up Status Register
+    AT91_REG     Reserved4[1];  //
+    AT91_REG     PIO_ASR;   // Select A Register
+    AT91_REG     PIO_BSR;   // Select B Register
+    AT91_REG     PIO_ABSR;  // AB Select Status Register
+    AT91_REG     Reserved5[9];  //
+    AT91_REG     PIO_OWER;  // Output Write Enable Register
+    AT91_REG     PIO_OWDR;  // Output Write Disable Register
+    AT91_REG     PIO_OWSR;  // Output Write Status Register
 } AT91S_PIO, *AT91PS_PIO;
 #else
-#define PIO_PER         (AT91_CAST(AT91_REG *) 	0x00000000) // (PIO_PER) PIO Enable Register
-#define PIO_PDR         (AT91_CAST(AT91_REG *) 	0x00000004) // (PIO_PDR) PIO Disable Register
-#define PIO_PSR         (AT91_CAST(AT91_REG *) 	0x00000008) // (PIO_PSR) PIO Status Register
-#define PIO_OER         (AT91_CAST(AT91_REG *) 	0x00000010) // (PIO_OER) Output Enable Register
-#define PIO_ODR         (AT91_CAST(AT91_REG *) 	0x00000014) // (PIO_ODR) Output Disable Register
-#define PIO_OSR         (AT91_CAST(AT91_REG *) 	0x00000018) // (PIO_OSR) Output Status Register
-#define PIO_IFER        (AT91_CAST(AT91_REG *) 	0x00000020) // (PIO_IFER) Input Filter Enable Register
-#define PIO_IFDR        (AT91_CAST(AT91_REG *) 	0x00000024) // (PIO_IFDR) Input Filter Disable Register
-#define PIO_IFSR        (AT91_CAST(AT91_REG *) 	0x00000028) // (PIO_IFSR) Input Filter Status Register
-#define PIO_SODR        (AT91_CAST(AT91_REG *) 	0x00000030) // (PIO_SODR) Set Output Data Register
-#define PIO_CODR        (AT91_CAST(AT91_REG *) 	0x00000034) // (PIO_CODR) Clear Output Data Register
-#define PIO_ODSR        (AT91_CAST(AT91_REG *) 	0x00000038) // (PIO_ODSR) Output Data Status Register
-#define PIO_PDSR        (AT91_CAST(AT91_REG *) 	0x0000003C) // (PIO_PDSR) Pin Data Status Register
-#define PIO_IER         (AT91_CAST(AT91_REG *) 	0x00000040) // (PIO_IER) Interrupt Enable Register
-#define PIO_IDR         (AT91_CAST(AT91_REG *) 	0x00000044) // (PIO_IDR) Interrupt Disable Register
-#define PIO_IMR         (AT91_CAST(AT91_REG *) 	0x00000048) // (PIO_IMR) Interrupt Mask Register
-#define PIO_ISR         (AT91_CAST(AT91_REG *) 	0x0000004C) // (PIO_ISR) Interrupt Status Register
-#define PIO_MDER        (AT91_CAST(AT91_REG *) 	0x00000050) // (PIO_MDER) Multi-driver Enable Register
-#define PIO_MDDR        (AT91_CAST(AT91_REG *) 	0x00000054) // (PIO_MDDR) Multi-driver Disable Register
-#define PIO_MDSR        (AT91_CAST(AT91_REG *) 	0x00000058) // (PIO_MDSR) Multi-driver Status Register
-#define PIO_PPUDR       (AT91_CAST(AT91_REG *) 	0x00000060) // (PIO_PPUDR) Pull-up Disable Register
-#define PIO_PPUER       (AT91_CAST(AT91_REG *) 	0x00000064) // (PIO_PPUER) Pull-up Enable Register
-#define PIO_PPUSR       (AT91_CAST(AT91_REG *) 	0x00000068) // (PIO_PPUSR) Pull-up Status Register
-#define PIO_ASR         (AT91_CAST(AT91_REG *) 	0x00000070) // (PIO_ASR) Select A Register
-#define PIO_BSR         (AT91_CAST(AT91_REG *) 	0x00000074) // (PIO_BSR) Select B Register
-#define PIO_ABSR        (AT91_CAST(AT91_REG *) 	0x00000078) // (PIO_ABSR) AB Select Status Register
-#define PIO_OWER        (AT91_CAST(AT91_REG *) 	0x000000A0) // (PIO_OWER) Output Write Enable Register
-#define PIO_OWDR        (AT91_CAST(AT91_REG *) 	0x000000A4) // (PIO_OWDR) Output Write Disable Register
-#define PIO_OWSR        (AT91_CAST(AT91_REG *) 	0x000000A8) // (PIO_OWSR) Output Write Status Register
+#define PIO_PER         (AT91_CAST(AT91_REG *)  0x00000000) // (PIO_PER) PIO Enable Register
+#define PIO_PDR         (AT91_CAST(AT91_REG *)  0x00000004) // (PIO_PDR) PIO Disable Register
+#define PIO_PSR         (AT91_CAST(AT91_REG *)  0x00000008) // (PIO_PSR) PIO Status Register
+#define PIO_OER         (AT91_CAST(AT91_REG *)  0x00000010) // (PIO_OER) Output Enable Register
+#define PIO_ODR         (AT91_CAST(AT91_REG *)  0x00000014) // (PIO_ODR) Output Disable Register
+#define PIO_OSR         (AT91_CAST(AT91_REG *)  0x00000018) // (PIO_OSR) Output Status Register
+#define PIO_IFER        (AT91_CAST(AT91_REG *)  0x00000020) // (PIO_IFER) Input Filter Enable Register
+#define PIO_IFDR        (AT91_CAST(AT91_REG *)  0x00000024) // (PIO_IFDR) Input Filter Disable Register
+#define PIO_IFSR        (AT91_CAST(AT91_REG *)  0x00000028) // (PIO_IFSR) Input Filter Status Register
+#define PIO_SODR        (AT91_CAST(AT91_REG *)  0x00000030) // (PIO_SODR) Set Output Data Register
+#define PIO_CODR        (AT91_CAST(AT91_REG *)  0x00000034) // (PIO_CODR) Clear Output Data Register
+#define PIO_ODSR        (AT91_CAST(AT91_REG *)  0x00000038) // (PIO_ODSR) Output Data Status Register
+#define PIO_PDSR        (AT91_CAST(AT91_REG *)  0x0000003C) // (PIO_PDSR) Pin Data Status Register
+#define PIO_IER         (AT91_CAST(AT91_REG *)  0x00000040) // (PIO_IER) Interrupt Enable Register
+#define PIO_IDR         (AT91_CAST(AT91_REG *)  0x00000044) // (PIO_IDR) Interrupt Disable Register
+#define PIO_IMR         (AT91_CAST(AT91_REG *)  0x00000048) // (PIO_IMR) Interrupt Mask Register
+#define PIO_ISR         (AT91_CAST(AT91_REG *)  0x0000004C) // (PIO_ISR) Interrupt Status Register
+#define PIO_MDER        (AT91_CAST(AT91_REG *)  0x00000050) // (PIO_MDER) Multi-driver Enable Register
+#define PIO_MDDR        (AT91_CAST(AT91_REG *)  0x00000054) // (PIO_MDDR) Multi-driver Disable Register
+#define PIO_MDSR        (AT91_CAST(AT91_REG *)  0x00000058) // (PIO_MDSR) Multi-driver Status Register
+#define PIO_PPUDR       (AT91_CAST(AT91_REG *)  0x00000060) // (PIO_PPUDR) Pull-up Disable Register
+#define PIO_PPUER       (AT91_CAST(AT91_REG *)  0x00000064) // (PIO_PPUER) Pull-up Enable Register
+#define PIO_PPUSR       (AT91_CAST(AT91_REG *)  0x00000068) // (PIO_PPUSR) Pull-up Status Register
+#define PIO_ASR         (AT91_CAST(AT91_REG *)  0x00000070) // (PIO_ASR) Select A Register
+#define PIO_BSR         (AT91_CAST(AT91_REG *)  0x00000074) // (PIO_BSR) Select B Register
+#define PIO_ABSR        (AT91_CAST(AT91_REG *)  0x00000078) // (PIO_ABSR) AB Select Status Register
+#define PIO_OWER        (AT91_CAST(AT91_REG *)  0x000000A0) // (PIO_OWER) Output Write Enable Register
+#define PIO_OWDR        (AT91_CAST(AT91_REG *)  0x000000A4) // (PIO_OWDR) Output Write Disable Register
+#define PIO_OWSR        (AT91_CAST(AT91_REG *)  0x000000A8) // (PIO_OWSR) Output Write Status Register
 
 #endif
 
@@ -461,15 +461,15 @@ typedef struct _AT91S_PIO {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_CKGR {
-    AT91_REG	 CKGR_MOR; 	// Main Oscillator Register
-    AT91_REG	 CKGR_MCFR; 	// Main Clock  Frequency Register
-    AT91_REG	 Reserved0[1]; 	//
-    AT91_REG	 CKGR_PLLR; 	// PLL Register
+    AT91_REG     CKGR_MOR;  // Main Oscillator Register
+    AT91_REG     CKGR_MCFR;     // Main Clock  Frequency Register
+    AT91_REG     Reserved0[1];  //
+    AT91_REG     CKGR_PLLR;     // PLL Register
 } AT91S_CKGR, *AT91PS_CKGR;
 #else
-#define CKGR_MOR        (AT91_CAST(AT91_REG *) 	0x00000000) // (CKGR_MOR) Main Oscillator Register
-#define CKGR_MCFR       (AT91_CAST(AT91_REG *) 	0x00000004) // (CKGR_MCFR) Main Clock  Frequency Register
-#define CKGR_PLLR       (AT91_CAST(AT91_REG *) 	0x0000000C) // (CKGR_PLLR) PLL Register
+#define CKGR_MOR        (AT91_CAST(AT91_REG *)  0x00000000) // (CKGR_MOR) Main Oscillator Register
+#define CKGR_MCFR       (AT91_CAST(AT91_REG *)  0x00000004) // (CKGR_MCFR) Main Clock  Frequency Register
+#define CKGR_PLLR       (AT91_CAST(AT91_REG *)  0x0000000C) // (CKGR_PLLR) PLL Register
 
 #endif
 // -------- CKGR_MOR : (CKGR Offset: 0x0) Main Oscillator Register --------
@@ -481,59 +481,59 @@ typedef struct _AT91S_CKGR {
 #define AT91C_CKGR_MAINRDY    (0x1 << 16) // (CKGR) Main Clock Ready
 // -------- CKGR_PLLR : (CKGR Offset: 0xc) PLL B Register --------
 #define AT91C_CKGR_DIV        (0xFF <<  0) // (CKGR) Divider Selected
-#define 	AT91C_CKGR_DIV_0                    (0x0) // (CKGR) Divider output is 0
-#define 	AT91C_CKGR_DIV_BYPASS               (0x1) // (CKGR) Divider is bypassed
+#define     AT91C_CKGR_DIV_0                    (0x0) // (CKGR) Divider output is 0
+#define     AT91C_CKGR_DIV_BYPASS               (0x1) // (CKGR) Divider is bypassed
 #define AT91C_CKGR_PLLCOUNT   (0x3F <<  8) // (CKGR) PLL Counter
 #define AT91C_CKGR_OUT        (0x3 << 14) // (CKGR) PLL Output Frequency Range
-#define 	AT91C_CKGR_OUT_0                    (0x0 << 14) // (CKGR) Please refer to the PLL datasheet
-#define 	AT91C_CKGR_OUT_1                    (0x1 << 14) // (CKGR) Please refer to the PLL datasheet
-#define 	AT91C_CKGR_OUT_2                    (0x2 << 14) // (CKGR) Please refer to the PLL datasheet
-#define 	AT91C_CKGR_OUT_3                    (0x3 << 14) // (CKGR) Please refer to the PLL datasheet
+#define     AT91C_CKGR_OUT_0                    (0x0 << 14) // (CKGR) Please refer to the PLL datasheet
+#define     AT91C_CKGR_OUT_1                    (0x1 << 14) // (CKGR) Please refer to the PLL datasheet
+#define     AT91C_CKGR_OUT_2                    (0x2 << 14) // (CKGR) Please refer to the PLL datasheet
+#define     AT91C_CKGR_OUT_3                    (0x3 << 14) // (CKGR) Please refer to the PLL datasheet
 #define AT91C_CKGR_MUL        (0x7FF << 16) // (CKGR) PLL Multiplier
 #define AT91C_CKGR_USBDIV     (0x3 << 28) // (CKGR) Divider for USB Clocks
-#define 	AT91C_CKGR_USBDIV_0                    (0x0 << 28) // (CKGR) Divider output is PLL clock output
-#define 	AT91C_CKGR_USBDIV_1                    (0x1 << 28) // (CKGR) Divider output is PLL clock output divided by 2
-#define 	AT91C_CKGR_USBDIV_2                    (0x2 << 28) // (CKGR) Divider output is PLL clock output divided by 4
+#define     AT91C_CKGR_USBDIV_0                    (0x0 << 28) // (CKGR) Divider output is PLL clock output
+#define     AT91C_CKGR_USBDIV_1                    (0x1 << 28) // (CKGR) Divider output is PLL clock output divided by 2
+#define     AT91C_CKGR_USBDIV_2                    (0x2 << 28) // (CKGR) Divider output is PLL clock output divided by 4
 
 // *****************************************************************************
 //              SOFTWARE API DEFINITION  FOR Power Management Controler
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_PMC {
-    AT91_REG	 PMC_SCER; 	// System Clock Enable Register
-    AT91_REG	 PMC_SCDR; 	// System Clock Disable Register
-    AT91_REG	 PMC_SCSR; 	// System Clock Status Register
-    AT91_REG	 Reserved0[1]; 	//
-    AT91_REG	 PMC_PCER; 	// Peripheral Clock Enable Register
-    AT91_REG	 PMC_PCDR; 	// Peripheral Clock Disable Register
-    AT91_REG	 PMC_PCSR; 	// Peripheral Clock Status Register
-    AT91_REG	 Reserved1[1]; 	//
-    AT91_REG	 PMC_MOR; 	// Main Oscillator Register
-    AT91_REG	 PMC_MCFR; 	// Main Clock  Frequency Register
-    AT91_REG	 Reserved2[1]; 	//
-    AT91_REG	 PMC_PLLR; 	// PLL Register
-    AT91_REG	 PMC_MCKR; 	// Master Clock Register
-    AT91_REG	 Reserved3[3]; 	//
-    AT91_REG	 PMC_PCKR[3]; 	// Programmable Clock Register
-    AT91_REG	 Reserved4[5]; 	//
-    AT91_REG	 PMC_IER; 	// Interrupt Enable Register
-    AT91_REG	 PMC_IDR; 	// Interrupt Disable Register
-    AT91_REG	 PMC_SR; 	// Status Register
-    AT91_REG	 PMC_IMR; 	// Interrupt Mask Register
+    AT91_REG     PMC_SCER;  // System Clock Enable Register
+    AT91_REG     PMC_SCDR;  // System Clock Disable Register
+    AT91_REG     PMC_SCSR;  // System Clock Status Register
+    AT91_REG     Reserved0[1];  //
+    AT91_REG     PMC_PCER;  // Peripheral Clock Enable Register
+    AT91_REG     PMC_PCDR;  // Peripheral Clock Disable Register
+    AT91_REG     PMC_PCSR;  // Peripheral Clock Status Register
+    AT91_REG     Reserved1[1];  //
+    AT91_REG     PMC_MOR;   // Main Oscillator Register
+    AT91_REG     PMC_MCFR;  // Main Clock  Frequency Register
+    AT91_REG     Reserved2[1];  //
+    AT91_REG     PMC_PLLR;  // PLL Register
+    AT91_REG     PMC_MCKR;  // Master Clock Register
+    AT91_REG     Reserved3[3];  //
+    AT91_REG     PMC_PCKR[3];   // Programmable Clock Register
+    AT91_REG     Reserved4[5];  //
+    AT91_REG     PMC_IER;   // Interrupt Enable Register
+    AT91_REG     PMC_IDR;   // Interrupt Disable Register
+    AT91_REG     PMC_SR;    // Status Register
+    AT91_REG     PMC_IMR;   // Interrupt Mask Register
 } AT91S_PMC, *AT91PS_PMC;
 #else
-#define PMC_SCER        (AT91_CAST(AT91_REG *) 	0x00000000) // (PMC_SCER) System Clock Enable Register
-#define PMC_SCDR        (AT91_CAST(AT91_REG *) 	0x00000004) // (PMC_SCDR) System Clock Disable Register
-#define PMC_SCSR        (AT91_CAST(AT91_REG *) 	0x00000008) // (PMC_SCSR) System Clock Status Register
-#define PMC_PCER        (AT91_CAST(AT91_REG *) 	0x00000010) // (PMC_PCER) Peripheral Clock Enable Register
-#define PMC_PCDR        (AT91_CAST(AT91_REG *) 	0x00000014) // (PMC_PCDR) Peripheral Clock Disable Register
-#define PMC_PCSR        (AT91_CAST(AT91_REG *) 	0x00000018) // (PMC_PCSR) Peripheral Clock Status Register
-#define PMC_MCKR        (AT91_CAST(AT91_REG *) 	0x00000030) // (PMC_MCKR) Master Clock Register
-#define PMC_PCKR        (AT91_CAST(AT91_REG *) 	0x00000040) // (PMC_PCKR) Programmable Clock Register
-#define PMC_IER         (AT91_CAST(AT91_REG *) 	0x00000060) // (PMC_IER) Interrupt Enable Register
-#define PMC_IDR         (AT91_CAST(AT91_REG *) 	0x00000064) // (PMC_IDR) Interrupt Disable Register
-#define PMC_SR          (AT91_CAST(AT91_REG *) 	0x00000068) // (PMC_SR) Status Register
-#define PMC_IMR         (AT91_CAST(AT91_REG *) 	0x0000006C) // (PMC_IMR) Interrupt Mask Register
+#define PMC_SCER        (AT91_CAST(AT91_REG *)  0x00000000) // (PMC_SCER) System Clock Enable Register
+#define PMC_SCDR        (AT91_CAST(AT91_REG *)  0x00000004) // (PMC_SCDR) System Clock Disable Register
+#define PMC_SCSR        (AT91_CAST(AT91_REG *)  0x00000008) // (PMC_SCSR) System Clock Status Register
+#define PMC_PCER        (AT91_CAST(AT91_REG *)  0x00000010) // (PMC_PCER) Peripheral Clock Enable Register
+#define PMC_PCDR        (AT91_CAST(AT91_REG *)  0x00000014) // (PMC_PCDR) Peripheral Clock Disable Register
+#define PMC_PCSR        (AT91_CAST(AT91_REG *)  0x00000018) // (PMC_PCSR) Peripheral Clock Status Register
+#define PMC_MCKR        (AT91_CAST(AT91_REG *)  0x00000030) // (PMC_MCKR) Master Clock Register
+#define PMC_PCKR        (AT91_CAST(AT91_REG *)  0x00000040) // (PMC_PCKR) Programmable Clock Register
+#define PMC_IER         (AT91_CAST(AT91_REG *)  0x00000060) // (PMC_IER) Interrupt Enable Register
+#define PMC_IDR         (AT91_CAST(AT91_REG *)  0x00000064) // (PMC_IDR) Interrupt Disable Register
+#define PMC_SR          (AT91_CAST(AT91_REG *)  0x00000068) // (PMC_SR) Status Register
+#define PMC_IMR         (AT91_CAST(AT91_REG *)  0x0000006C) // (PMC_IMR) Interrupt Mask Register
 
 #endif
 // -------- PMC_SCER : (PMC Offset: 0x0) System Clock Enable Register --------
@@ -549,17 +549,17 @@ typedef struct _AT91S_PMC {
 // -------- CKGR_PLLR : (PMC Offset: 0x2c) PLL B Register --------
 // -------- PMC_MCKR : (PMC Offset: 0x30) Master Clock Register --------
 #define AT91C_PMC_CSS         (0x3 <<  0) // (PMC) Programmable Clock Selection
-#define 	AT91C_PMC_CSS_SLOW_CLK             (0x0) // (PMC) Slow Clock is selected
-#define 	AT91C_PMC_CSS_MAIN_CLK             (0x1) // (PMC) Main Clock is selected
-#define 	AT91C_PMC_CSS_PLL_CLK              (0x3) // (PMC) Clock from PLL is selected
+#define     AT91C_PMC_CSS_SLOW_CLK             (0x0) // (PMC) Slow Clock is selected
+#define     AT91C_PMC_CSS_MAIN_CLK             (0x1) // (PMC) Main Clock is selected
+#define     AT91C_PMC_CSS_PLL_CLK              (0x3) // (PMC) Clock from PLL is selected
 #define AT91C_PMC_PRES        (0x7 <<  2) // (PMC) Programmable Clock Prescaler
-#define 	AT91C_PMC_PRES_CLK                  (0x0 <<  2) // (PMC) Selected clock
-#define 	AT91C_PMC_PRES_CLK_2                (0x1 <<  2) // (PMC) Selected clock divided by 2
-#define 	AT91C_PMC_PRES_CLK_4                (0x2 <<  2) // (PMC) Selected clock divided by 4
-#define 	AT91C_PMC_PRES_CLK_8                (0x3 <<  2) // (PMC) Selected clock divided by 8
-#define 	AT91C_PMC_PRES_CLK_16               (0x4 <<  2) // (PMC) Selected clock divided by 16
-#define 	AT91C_PMC_PRES_CLK_32               (0x5 <<  2) // (PMC) Selected clock divided by 32
-#define 	AT91C_PMC_PRES_CLK_64               (0x6 <<  2) // (PMC) Selected clock divided by 64
+#define     AT91C_PMC_PRES_CLK                  (0x0 <<  2) // (PMC) Selected clock
+#define     AT91C_PMC_PRES_CLK_2                (0x1 <<  2) // (PMC) Selected clock divided by 2
+#define     AT91C_PMC_PRES_CLK_4                (0x2 <<  2) // (PMC) Selected clock divided by 4
+#define     AT91C_PMC_PRES_CLK_8                (0x3 <<  2) // (PMC) Selected clock divided by 8
+#define     AT91C_PMC_PRES_CLK_16               (0x4 <<  2) // (PMC) Selected clock divided by 16
+#define     AT91C_PMC_PRES_CLK_32               (0x5 <<  2) // (PMC) Selected clock divided by 32
+#define     AT91C_PMC_PRES_CLK_64               (0x6 <<  2) // (PMC) Selected clock divided by 64
 // -------- PMC_PCKR : (PMC Offset: 0x40) Programmable Clock Register --------
 // -------- PMC_IER : (PMC Offset: 0x60) PMC Interrupt Enable Register --------
 #define AT91C_PMC_MOSCS       (0x1 <<  0) // (PMC) MOSC Status/Enable/Disable/Mask
@@ -577,14 +577,14 @@ typedef struct _AT91S_PMC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_RSTC {
-    AT91_REG	 RSTC_RCR; 	// Reset Control Register
-    AT91_REG	 RSTC_RSR; 	// Reset Status Register
-    AT91_REG	 RSTC_RMR; 	// Reset Mode Register
+    AT91_REG     RSTC_RCR;  // Reset Control Register
+    AT91_REG     RSTC_RSR;  // Reset Status Register
+    AT91_REG     RSTC_RMR;  // Reset Mode Register
 } AT91S_RSTC, *AT91PS_RSTC;
 #else
-#define RSTC_RCR        (AT91_CAST(AT91_REG *) 	0x00000000) // (RSTC_RCR) Reset Control Register
-#define RSTC_RSR        (AT91_CAST(AT91_REG *) 	0x00000004) // (RSTC_RSR) Reset Status Register
-#define RSTC_RMR        (AT91_CAST(AT91_REG *) 	0x00000008) // (RSTC_RMR) Reset Mode Register
+#define RSTC_RCR        (AT91_CAST(AT91_REG *)  0x00000000) // (RSTC_RCR) Reset Control Register
+#define RSTC_RSR        (AT91_CAST(AT91_REG *)  0x00000004) // (RSTC_RSR) Reset Status Register
+#define RSTC_RMR        (AT91_CAST(AT91_REG *)  0x00000008) // (RSTC_RMR) Reset Mode Register
 
 #endif
 // -------- RSTC_RCR : (RSTC Offset: 0x0) Reset Control Register --------
@@ -596,12 +596,12 @@ typedef struct _AT91S_RSTC {
 #define AT91C_RSTC_URSTS      (0x1 <<  0) // (RSTC) User Reset Status
 #define AT91C_RSTC_BODSTS     (0x1 <<  1) // (RSTC) Brownout Detection Status
 #define AT91C_RSTC_RSTTYP     (0x7 <<  8) // (RSTC) Reset Type
-#define 	AT91C_RSTC_RSTTYP_POWERUP              (0x0 <<  8) // (RSTC) Power-up Reset. VDDCORE rising.
-#define 	AT91C_RSTC_RSTTYP_WAKEUP               (0x1 <<  8) // (RSTC) WakeUp Reset. VDDCORE rising.
-#define 	AT91C_RSTC_RSTTYP_WATCHDOG             (0x2 <<  8) // (RSTC) Watchdog Reset. Watchdog overflow occured.
-#define 	AT91C_RSTC_RSTTYP_SOFTWARE             (0x3 <<  8) // (RSTC) Software Reset. Processor reset required by the software.
-#define 	AT91C_RSTC_RSTTYP_USER                 (0x4 <<  8) // (RSTC) User Reset. NRST pin detected low.
-#define 	AT91C_RSTC_RSTTYP_BROWNOUT             (0x5 <<  8) // (RSTC) Brownout Reset occured.
+#define     AT91C_RSTC_RSTTYP_POWERUP              (0x0 <<  8) // (RSTC) Power-up Reset. VDDCORE rising.
+#define     AT91C_RSTC_RSTTYP_WAKEUP               (0x1 <<  8) // (RSTC) WakeUp Reset. VDDCORE rising.
+#define     AT91C_RSTC_RSTTYP_WATCHDOG             (0x2 <<  8) // (RSTC) Watchdog Reset. Watchdog overflow occured.
+#define     AT91C_RSTC_RSTTYP_SOFTWARE             (0x3 <<  8) // (RSTC) Software Reset. Processor reset required by the software.
+#define     AT91C_RSTC_RSTTYP_USER                 (0x4 <<  8) // (RSTC) User Reset. NRST pin detected low.
+#define     AT91C_RSTC_RSTTYP_BROWNOUT             (0x5 <<  8) // (RSTC) Brownout Reset occured.
 #define AT91C_RSTC_NRSTL      (0x1 << 16) // (RSTC) NRST pin level
 #define AT91C_RSTC_SRCMP      (0x1 << 17) // (RSTC) Software Reset Command in Progress.
 // -------- RSTC_RMR : (RSTC Offset: 0x8) Reset Mode Register --------
@@ -615,16 +615,16 @@ typedef struct _AT91S_RSTC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_RTTC {
-    AT91_REG	 RTTC_RTMR; 	// Real-time Mode Register
-    AT91_REG	 RTTC_RTAR; 	// Real-time Alarm Register
-    AT91_REG	 RTTC_RTVR; 	// Real-time Value Register
-    AT91_REG	 RTTC_RTSR; 	// Real-time Status Register
+    AT91_REG     RTTC_RTMR;     // Real-time Mode Register
+    AT91_REG     RTTC_RTAR;     // Real-time Alarm Register
+    AT91_REG     RTTC_RTVR;     // Real-time Value Register
+    AT91_REG     RTTC_RTSR;     // Real-time Status Register
 } AT91S_RTTC, *AT91PS_RTTC;
 #else
-#define RTTC_RTMR       (AT91_CAST(AT91_REG *) 	0x00000000) // (RTTC_RTMR) Real-time Mode Register
-#define RTTC_RTAR       (AT91_CAST(AT91_REG *) 	0x00000004) // (RTTC_RTAR) Real-time Alarm Register
-#define RTTC_RTVR       (AT91_CAST(AT91_REG *) 	0x00000008) // (RTTC_RTVR) Real-time Value Register
-#define RTTC_RTSR       (AT91_CAST(AT91_REG *) 	0x0000000C) // (RTTC_RTSR) Real-time Status Register
+#define RTTC_RTMR       (AT91_CAST(AT91_REG *)  0x00000000) // (RTTC_RTMR) Real-time Mode Register
+#define RTTC_RTAR       (AT91_CAST(AT91_REG *)  0x00000004) // (RTTC_RTAR) Real-time Alarm Register
+#define RTTC_RTVR       (AT91_CAST(AT91_REG *)  0x00000008) // (RTTC_RTVR) Real-time Value Register
+#define RTTC_RTSR       (AT91_CAST(AT91_REG *)  0x0000000C) // (RTTC_RTSR) Real-time Status Register
 
 #endif
 // -------- RTTC_RTMR : (RTTC Offset: 0x0) Real-time Mode Register --------
@@ -645,16 +645,16 @@ typedef struct _AT91S_RTTC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_PITC {
-    AT91_REG	 PITC_PIMR; 	// Period Interval Mode Register
-    AT91_REG	 PITC_PISR; 	// Period Interval Status Register
-    AT91_REG	 PITC_PIVR; 	// Period Interval Value Register
-    AT91_REG	 PITC_PIIR; 	// Period Interval Image Register
+    AT91_REG     PITC_PIMR;     // Period Interval Mode Register
+    AT91_REG     PITC_PISR;     // Period Interval Status Register
+    AT91_REG     PITC_PIVR;     // Period Interval Value Register
+    AT91_REG     PITC_PIIR;     // Period Interval Image Register
 } AT91S_PITC, *AT91PS_PITC;
 #else
-#define PITC_PIMR       (AT91_CAST(AT91_REG *) 	0x00000000) // (PITC_PIMR) Period Interval Mode Register
-#define PITC_PISR       (AT91_CAST(AT91_REG *) 	0x00000004) // (PITC_PISR) Period Interval Status Register
-#define PITC_PIVR       (AT91_CAST(AT91_REG *) 	0x00000008) // (PITC_PIVR) Period Interval Value Register
-#define PITC_PIIR       (AT91_CAST(AT91_REG *) 	0x0000000C) // (PITC_PIIR) Period Interval Image Register
+#define PITC_PIMR       (AT91_CAST(AT91_REG *)  0x00000000) // (PITC_PIMR) Period Interval Mode Register
+#define PITC_PISR       (AT91_CAST(AT91_REG *)  0x00000004) // (PITC_PISR) Period Interval Status Register
+#define PITC_PIVR       (AT91_CAST(AT91_REG *)  0x00000008) // (PITC_PIVR) Period Interval Value Register
+#define PITC_PIIR       (AT91_CAST(AT91_REG *)  0x0000000C) // (PITC_PIIR) Period Interval Image Register
 
 #endif
 // -------- PITC_PIMR : (PITC Offset: 0x0) Periodic Interval Mode Register --------
@@ -673,14 +673,14 @@ typedef struct _AT91S_PITC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_WDTC {
-    AT91_REG	 WDTC_WDCR; 	// Watchdog Control Register
-    AT91_REG	 WDTC_WDMR; 	// Watchdog Mode Register
-    AT91_REG	 WDTC_WDSR; 	// Watchdog Status Register
+    AT91_REG     WDTC_WDCR;     // Watchdog Control Register
+    AT91_REG     WDTC_WDMR;     // Watchdog Mode Register
+    AT91_REG     WDTC_WDSR;     // Watchdog Status Register
 } AT91S_WDTC, *AT91PS_WDTC;
 #else
-#define WDTC_WDCR       (AT91_CAST(AT91_REG *) 	0x00000000) // (WDTC_WDCR) Watchdog Control Register
-#define WDTC_WDMR       (AT91_CAST(AT91_REG *) 	0x00000004) // (WDTC_WDMR) Watchdog Mode Register
-#define WDTC_WDSR       (AT91_CAST(AT91_REG *) 	0x00000008) // (WDTC_WDSR) Watchdog Status Register
+#define WDTC_WDCR       (AT91_CAST(AT91_REG *)  0x00000000) // (WDTC_WDCR) Watchdog Control Register
+#define WDTC_WDMR       (AT91_CAST(AT91_REG *)  0x00000004) // (WDTC_WDMR) Watchdog Mode Register
+#define WDTC_WDSR       (AT91_CAST(AT91_REG *)  0x00000008) // (WDTC_WDSR) Watchdog Status Register
 
 #endif
 // -------- WDTC_WDCR : (WDTC Offset: 0x0) Periodic Interval Image Register --------
@@ -704,10 +704,10 @@ typedef struct _AT91S_WDTC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_VREG {
-    AT91_REG	 VREG_MR; 	// Voltage Regulator Mode Register
+    AT91_REG     VREG_MR;   // Voltage Regulator Mode Register
 } AT91S_VREG, *AT91PS_VREG;
 #else
-#define VREG_MR         (AT91_CAST(AT91_REG *) 	0x00000000) // (VREG_MR) Voltage Regulator Mode Register
+#define VREG_MR         (AT91_CAST(AT91_REG *)  0x00000000) // (VREG_MR) Voltage Regulator Mode Register
 
 #endif
 // -------- VREG_MR : (VREG Offset: 0x0) Voltage Regulator Mode Register --------
@@ -718,16 +718,16 @@ typedef struct _AT91S_VREG {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_EFC {
-    AT91_REG	 EFC_FMR; 	// MC Flash Mode Register
-    AT91_REG	 EFC_FCR; 	// MC Flash Command Register
-    AT91_REG	 EFC_FSR; 	// MC Flash Status Register
-    AT91_REG	 EFC_VR; 	// MC Flash Version Register
+    AT91_REG     EFC_FMR;   // MC Flash Mode Register
+    AT91_REG     EFC_FCR;   // MC Flash Command Register
+    AT91_REG     EFC_FSR;   // MC Flash Status Register
+    AT91_REG     EFC_VR;    // MC Flash Version Register
 } AT91S_EFC, *AT91PS_EFC;
 #else
-#define MC_FMR          (AT91_CAST(AT91_REG *) 	0x00000000) // (MC_FMR) MC Flash Mode Register
-#define MC_FCR          (AT91_CAST(AT91_REG *) 	0x00000004) // (MC_FCR) MC Flash Command Register
-#define MC_FSR          (AT91_CAST(AT91_REG *) 	0x00000008) // (MC_FSR) MC Flash Status Register
-#define MC_VR           (AT91_CAST(AT91_REG *) 	0x0000000C) // (MC_VR) MC Flash Version Register
+#define MC_FMR          (AT91_CAST(AT91_REG *)  0x00000000) // (MC_FMR) MC Flash Mode Register
+#define MC_FCR          (AT91_CAST(AT91_REG *)  0x00000004) // (MC_FCR) MC Flash Command Register
+#define MC_FSR          (AT91_CAST(AT91_REG *)  0x00000008) // (MC_FSR) MC Flash Status Register
+#define MC_VR           (AT91_CAST(AT91_REG *)  0x0000000C) // (MC_VR) MC Flash Version Register
 
 #endif
 // -------- MC_FMR : (EFC Offset: 0x0) MC Flash Mode Register --------
@@ -736,21 +736,21 @@ typedef struct _AT91S_EFC {
 #define AT91C_MC_PROGE        (0x1 <<  3) // (EFC) Programming Error
 #define AT91C_MC_NEBP         (0x1 <<  7) // (EFC) No Erase Before Programming
 #define AT91C_MC_FWS          (0x3 <<  8) // (EFC) Flash Wait State
-#define 	AT91C_MC_FWS_0FWS                 (0x0 <<  8) // (EFC) 1 cycle for Read, 2 for Write operations
-#define 	AT91C_MC_FWS_1FWS                 (0x1 <<  8) // (EFC) 2 cycles for Read, 3 for Write operations
-#define 	AT91C_MC_FWS_2FWS                 (0x2 <<  8) // (EFC) 3 cycles for Read, 4 for Write operations
-#define 	AT91C_MC_FWS_3FWS                 (0x3 <<  8) // (EFC) 4 cycles for Read, 4 for Write operations
+#define     AT91C_MC_FWS_0FWS                 (0x0 <<  8) // (EFC) 1 cycle for Read, 2 for Write operations
+#define     AT91C_MC_FWS_1FWS                 (0x1 <<  8) // (EFC) 2 cycles for Read, 3 for Write operations
+#define     AT91C_MC_FWS_2FWS                 (0x2 <<  8) // (EFC) 3 cycles for Read, 4 for Write operations
+#define     AT91C_MC_FWS_3FWS                 (0x3 <<  8) // (EFC) 4 cycles for Read, 4 for Write operations
 #define AT91C_MC_FMCN         (0xFF << 16) // (EFC) Flash Microsecond Cycle Number
 // -------- MC_FCR : (EFC Offset: 0x4) MC Flash Command Register --------
 #define AT91C_MC_FCMD         (0xF <<  0) // (EFC) Flash Command
-#define 	AT91C_MC_FCMD_START_PROG           (0x1) // (EFC) Starts the programming of th epage specified by PAGEN.
-#define 	AT91C_MC_FCMD_LOCK                 (0x2) // (EFC) Starts a lock sequence of the sector defined by the bits 4 to 7 of the field PAGEN.
-#define 	AT91C_MC_FCMD_PROG_AND_LOCK        (0x3) // (EFC) The lock sequence automatically happens after the programming sequence is completed.
-#define 	AT91C_MC_FCMD_UNLOCK               (0x4) // (EFC) Starts an unlock sequence of the sector defined by the bits 4 to 7 of the field PAGEN.
-#define 	AT91C_MC_FCMD_ERASE_ALL            (0x8) // (EFC) Starts the erase of the entire flash.If at least a page is locked, the command is cancelled.
-#define 	AT91C_MC_FCMD_SET_GP_NVM           (0xB) // (EFC) Set General Purpose NVM bits.
-#define 	AT91C_MC_FCMD_CLR_GP_NVM           (0xD) // (EFC) Clear General Purpose NVM bits.
-#define 	AT91C_MC_FCMD_SET_SECURITY         (0xF) // (EFC) Set Security Bit.
+#define     AT91C_MC_FCMD_START_PROG           (0x1) // (EFC) Starts the programming of th epage specified by PAGEN.
+#define     AT91C_MC_FCMD_LOCK                 (0x2) // (EFC) Starts a lock sequence of the sector defined by the bits 4 to 7 of the field PAGEN.
+#define     AT91C_MC_FCMD_PROG_AND_LOCK        (0x3) // (EFC) The lock sequence automatically happens after the programming sequence is completed.
+#define     AT91C_MC_FCMD_UNLOCK               (0x4) // (EFC) Starts an unlock sequence of the sector defined by the bits 4 to 7 of the field PAGEN.
+#define     AT91C_MC_FCMD_ERASE_ALL            (0x8) // (EFC) Starts the erase of the entire flash.If at least a page is locked, the command is cancelled.
+#define     AT91C_MC_FCMD_SET_GP_NVM           (0xB) // (EFC) Set General Purpose NVM bits.
+#define     AT91C_MC_FCMD_CLR_GP_NVM           (0xD) // (EFC) Clear General Purpose NVM bits.
+#define     AT91C_MC_FCMD_SET_SECURITY         (0xF) // (EFC) Set Security Bit.
 #define AT91C_MC_PAGEN        (0x3FF <<  8) // (EFC) Page Number
 #define AT91C_MC_KEY          (0xFF << 24) // (EFC) Writing Protect Key
 // -------- MC_FSR : (EFC Offset: 0x8) MC Flash Command Register --------
@@ -788,30 +788,30 @@ typedef struct _AT91S_EFC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_MC {
-    AT91_REG	 MC_RCR; 	// MC Remap Control Register
-    AT91_REG	 MC_ASR; 	// MC Abort Status Register
-    AT91_REG	 MC_AASR; 	// MC Abort Address Status Register
-    AT91_REG	 Reserved0[1]; 	//
-    AT91_REG	 MC_PUIA[16]; 	// MC Protection Unit Area
-    AT91_REG	 MC_PUP; 	// MC Protection Unit Peripherals
-    AT91_REG	 MC_PUER; 	// MC Protection Unit Enable Register
-    AT91_REG	 Reserved1[2]; 	//
-    AT91_REG	 MC0_FMR; 	// MC Flash Mode Register
-    AT91_REG	 MC0_FCR; 	// MC Flash Command Register
-    AT91_REG	 MC0_FSR; 	// MC Flash Status Register
-    AT91_REG	 MC0_VR; 	// MC Flash Version Register
-    AT91_REG	 MC1_FMR; 	// MC Flash Mode Register
-    AT91_REG	 MC1_FCR; 	// MC Flash Command Register
-    AT91_REG	 MC1_FSR; 	// MC Flash Status Register
-    AT91_REG	 MC1_VR; 	// MC Flash Version Register
+    AT91_REG     MC_RCR;    // MC Remap Control Register
+    AT91_REG     MC_ASR;    // MC Abort Status Register
+    AT91_REG     MC_AASR;   // MC Abort Address Status Register
+    AT91_REG     Reserved0[1];  //
+    AT91_REG     MC_PUIA[16];   // MC Protection Unit Area
+    AT91_REG     MC_PUP;    // MC Protection Unit Peripherals
+    AT91_REG     MC_PUER;   // MC Protection Unit Enable Register
+    AT91_REG     Reserved1[2];  //
+    AT91_REG     MC0_FMR;   // MC Flash Mode Register
+    AT91_REG     MC0_FCR;   // MC Flash Command Register
+    AT91_REG     MC0_FSR;   // MC Flash Status Register
+    AT91_REG     MC0_VR;    // MC Flash Version Register
+    AT91_REG     MC1_FMR;   // MC Flash Mode Register
+    AT91_REG     MC1_FCR;   // MC Flash Command Register
+    AT91_REG     MC1_FSR;   // MC Flash Status Register
+    AT91_REG     MC1_VR;    // MC Flash Version Register
 } AT91S_MC, *AT91PS_MC;
 #else
-#define MC_RCR          (AT91_CAST(AT91_REG *) 	0x00000000) // (MC_RCR) MC Remap Control Register
-#define MC_ASR          (AT91_CAST(AT91_REG *) 	0x00000004) // (MC_ASR) MC Abort Status Register
-#define MC_AASR         (AT91_CAST(AT91_REG *) 	0x00000008) // (MC_AASR) MC Abort Address Status Register
-#define MC_PUIA         (AT91_CAST(AT91_REG *) 	0x00000010) // (MC_PUIA) MC Protection Unit Area
-#define MC_PUP          (AT91_CAST(AT91_REG *) 	0x00000050) // (MC_PUP) MC Protection Unit Peripherals
-#define MC_PUER         (AT91_CAST(AT91_REG *) 	0x00000054) // (MC_PUER) MC Protection Unit Enable Register
+#define MC_RCR          (AT91_CAST(AT91_REG *)  0x00000000) // (MC_RCR) MC Remap Control Register
+#define MC_ASR          (AT91_CAST(AT91_REG *)  0x00000004) // (MC_ASR) MC Abort Status Register
+#define MC_AASR         (AT91_CAST(AT91_REG *)  0x00000008) // (MC_AASR) MC Abort Address Status Register
+#define MC_PUIA         (AT91_CAST(AT91_REG *)  0x00000010) // (MC_PUIA) MC Protection Unit Area
+#define MC_PUP          (AT91_CAST(AT91_REG *)  0x00000050) // (MC_PUP) MC Protection Unit Peripherals
+#define MC_PUER         (AT91_CAST(AT91_REG *)  0x00000054) // (MC_PUER) MC Protection Unit Enable Register
 
 #endif
 // -------- MC_RCR : (MC Offset: 0x0) MC Remap Control Register --------
@@ -821,40 +821,40 @@ typedef struct _AT91S_MC {
 #define AT91C_MC_MISADD       (0x1 <<  1) // (MC) Misaligned Addess Abort Status
 #define AT91C_MC_MPU          (0x1 <<  2) // (MC) Memory protection Unit Abort Status
 #define AT91C_MC_ABTSZ        (0x3 <<  8) // (MC) Abort Size Status
-#define 	AT91C_MC_ABTSZ_BYTE                 (0x0 <<  8) // (MC) Byte
-#define 	AT91C_MC_ABTSZ_HWORD                (0x1 <<  8) // (MC) Half-word
-#define 	AT91C_MC_ABTSZ_WORD                 (0x2 <<  8) // (MC) Word
+#define     AT91C_MC_ABTSZ_BYTE                 (0x0 <<  8) // (MC) Byte
+#define     AT91C_MC_ABTSZ_HWORD                (0x1 <<  8) // (MC) Half-word
+#define     AT91C_MC_ABTSZ_WORD                 (0x2 <<  8) // (MC) Word
 #define AT91C_MC_ABTTYP       (0x3 << 10) // (MC) Abort Type Status
-#define 	AT91C_MC_ABTTYP_DATAR                (0x0 << 10) // (MC) Data Read
-#define 	AT91C_MC_ABTTYP_DATAW                (0x1 << 10) // (MC) Data Write
-#define 	AT91C_MC_ABTTYP_FETCH                (0x2 << 10) // (MC) Code Fetch
+#define     AT91C_MC_ABTTYP_DATAR                (0x0 << 10) // (MC) Data Read
+#define     AT91C_MC_ABTTYP_DATAW                (0x1 << 10) // (MC) Data Write
+#define     AT91C_MC_ABTTYP_FETCH                (0x2 << 10) // (MC) Code Fetch
 #define AT91C_MC_MST0         (0x1 << 16) // (MC) Master 0 Abort Source
 #define AT91C_MC_MST1         (0x1 << 17) // (MC) Master 1 Abort Source
 #define AT91C_MC_SVMST0       (0x1 << 24) // (MC) Saved Master 0 Abort Source
 #define AT91C_MC_SVMST1       (0x1 << 25) // (MC) Saved Master 1 Abort Source
 // -------- MC_PUIA : (MC Offset: 0x10) MC Protection Unit Area --------
 #define AT91C_MC_PROT         (0x3 <<  0) // (MC) Protection
-#define 	AT91C_MC_PROT_PNAUNA               (0x0) // (MC) Privilege: No Access, User: No Access
-#define 	AT91C_MC_PROT_PRWUNA               (0x1) // (MC) Privilege: Read/Write, User: No Access
-#define 	AT91C_MC_PROT_PRWURO               (0x2) // (MC) Privilege: Read/Write, User: Read Only
-#define 	AT91C_MC_PROT_PRWURW               (0x3) // (MC) Privilege: Read/Write, User: Read/Write
+#define     AT91C_MC_PROT_PNAUNA               (0x0) // (MC) Privilege: No Access, User: No Access
+#define     AT91C_MC_PROT_PRWUNA               (0x1) // (MC) Privilege: Read/Write, User: No Access
+#define     AT91C_MC_PROT_PRWURO               (0x2) // (MC) Privilege: Read/Write, User: Read Only
+#define     AT91C_MC_PROT_PRWURW               (0x3) // (MC) Privilege: Read/Write, User: Read/Write
 #define AT91C_MC_SIZE         (0xF <<  4) // (MC) Internal Area Size
-#define 	AT91C_MC_SIZE_1KB                  (0x0 <<  4) // (MC) Area size 1KByte
-#define 	AT91C_MC_SIZE_2KB                  (0x1 <<  4) // (MC) Area size 2KByte
-#define 	AT91C_MC_SIZE_4KB                  (0x2 <<  4) // (MC) Area size 4KByte
-#define 	AT91C_MC_SIZE_8KB                  (0x3 <<  4) // (MC) Area size 8KByte
-#define 	AT91C_MC_SIZE_16KB                 (0x4 <<  4) // (MC) Area size 16KByte
-#define 	AT91C_MC_SIZE_32KB                 (0x5 <<  4) // (MC) Area size 32KByte
-#define 	AT91C_MC_SIZE_64KB                 (0x6 <<  4) // (MC) Area size 64KByte
-#define 	AT91C_MC_SIZE_128KB                (0x7 <<  4) // (MC) Area size 128KByte
-#define 	AT91C_MC_SIZE_256KB                (0x8 <<  4) // (MC) Area size 256KByte
-#define 	AT91C_MC_SIZE_512KB                (0x9 <<  4) // (MC) Area size 512KByte
-#define 	AT91C_MC_SIZE_1MB                  (0xA <<  4) // (MC) Area size 1MByte
-#define 	AT91C_MC_SIZE_2MB                  (0xB <<  4) // (MC) Area size 2MByte
-#define 	AT91C_MC_SIZE_4MB                  (0xC <<  4) // (MC) Area size 4MByte
-#define 	AT91C_MC_SIZE_8MB                  (0xD <<  4) // (MC) Area size 8MByte
-#define 	AT91C_MC_SIZE_16MB                 (0xE <<  4) // (MC) Area size 16MByte
-#define 	AT91C_MC_SIZE_64MB                 (0xF <<  4) // (MC) Area size 64MByte
+#define     AT91C_MC_SIZE_1KB                  (0x0 <<  4) // (MC) Area size 1KByte
+#define     AT91C_MC_SIZE_2KB                  (0x1 <<  4) // (MC) Area size 2KByte
+#define     AT91C_MC_SIZE_4KB                  (0x2 <<  4) // (MC) Area size 4KByte
+#define     AT91C_MC_SIZE_8KB                  (0x3 <<  4) // (MC) Area size 8KByte
+#define     AT91C_MC_SIZE_16KB                 (0x4 <<  4) // (MC) Area size 16KByte
+#define     AT91C_MC_SIZE_32KB                 (0x5 <<  4) // (MC) Area size 32KByte
+#define     AT91C_MC_SIZE_64KB                 (0x6 <<  4) // (MC) Area size 64KByte
+#define     AT91C_MC_SIZE_128KB                (0x7 <<  4) // (MC) Area size 128KByte
+#define     AT91C_MC_SIZE_256KB                (0x8 <<  4) // (MC) Area size 256KByte
+#define     AT91C_MC_SIZE_512KB                (0x9 <<  4) // (MC) Area size 512KByte
+#define     AT91C_MC_SIZE_1MB                  (0xA <<  4) // (MC) Area size 1MByte
+#define     AT91C_MC_SIZE_2MB                  (0xB <<  4) // (MC) Area size 2MByte
+#define     AT91C_MC_SIZE_4MB                  (0xC <<  4) // (MC) Area size 4MByte
+#define     AT91C_MC_SIZE_8MB                  (0xD <<  4) // (MC) Area size 8MByte
+#define     AT91C_MC_SIZE_16MB                 (0xE <<  4) // (MC) Area size 16MByte
+#define     AT91C_MC_SIZE_64MB                 (0xF <<  4) // (MC) Area size 64MByte
 #define AT91C_MC_BA           (0x3FFFF << 10) // (MC) Internal Area Base Address
 // -------- MC_PUP : (MC Offset: 0x50) MC Protection Unit Peripheral --------
 // -------- MC_PUER : (MC Offset: 0x54) MC Protection Unit Area --------
@@ -865,38 +865,38 @@ typedef struct _AT91S_MC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_SPI {
-    AT91_REG	 SPI_CR; 	// Control Register
-    AT91_REG	 SPI_MR; 	// Mode Register
-    AT91_REG	 SPI_RDR; 	// Receive Data Register
-    AT91_REG	 SPI_TDR; 	// Transmit Data Register
-    AT91_REG	 SPI_SR; 	// Status Register
-    AT91_REG	 SPI_IER; 	// Interrupt Enable Register
-    AT91_REG	 SPI_IDR; 	// Interrupt Disable Register
-    AT91_REG	 SPI_IMR; 	// Interrupt Mask Register
-    AT91_REG	 Reserved0[4]; 	//
-    AT91_REG	 SPI_CSR[4]; 	// Chip Select Register
-    AT91_REG	 Reserved1[48]; 	//
-    AT91_REG	 SPI_RPR; 	// Receive Pointer Register
-    AT91_REG	 SPI_RCR; 	// Receive Counter Register
-    AT91_REG	 SPI_TPR; 	// Transmit Pointer Register
-    AT91_REG	 SPI_TCR; 	// Transmit Counter Register
-    AT91_REG	 SPI_RNPR; 	// Receive Next Pointer Register
-    AT91_REG	 SPI_RNCR; 	// Receive Next Counter Register
-    AT91_REG	 SPI_TNPR; 	// Transmit Next Pointer Register
-    AT91_REG	 SPI_TNCR; 	// Transmit Next Counter Register
-    AT91_REG	 SPI_PTCR; 	// PDC Transfer Control Register
-    AT91_REG	 SPI_PTSR; 	// PDC Transfer Status Register
+    AT91_REG     SPI_CR;    // Control Register
+    AT91_REG     SPI_MR;    // Mode Register
+    AT91_REG     SPI_RDR;   // Receive Data Register
+    AT91_REG     SPI_TDR;   // Transmit Data Register
+    AT91_REG     SPI_SR;    // Status Register
+    AT91_REG     SPI_IER;   // Interrupt Enable Register
+    AT91_REG     SPI_IDR;   // Interrupt Disable Register
+    AT91_REG     SPI_IMR;   // Interrupt Mask Register
+    AT91_REG     Reserved0[4];  //
+    AT91_REG     SPI_CSR[4];    // Chip Select Register
+    AT91_REG     Reserved1[48];     //
+    AT91_REG     SPI_RPR;   // Receive Pointer Register
+    AT91_REG     SPI_RCR;   // Receive Counter Register
+    AT91_REG     SPI_TPR;   // Transmit Pointer Register
+    AT91_REG     SPI_TCR;   // Transmit Counter Register
+    AT91_REG     SPI_RNPR;  // Receive Next Pointer Register
+    AT91_REG     SPI_RNCR;  // Receive Next Counter Register
+    AT91_REG     SPI_TNPR;  // Transmit Next Pointer Register
+    AT91_REG     SPI_TNCR;  // Transmit Next Counter Register
+    AT91_REG     SPI_PTCR;  // PDC Transfer Control Register
+    AT91_REG     SPI_PTSR;  // PDC Transfer Status Register
 } AT91S_SPI, *AT91PS_SPI;
 #else
-#define SPI_CR          (AT91_CAST(AT91_REG *) 	0x00000000) // (SPI_CR) Control Register
-#define SPI_MR          (AT91_CAST(AT91_REG *) 	0x00000004) // (SPI_MR) Mode Register
-#define SPI_RDR         (AT91_CAST(AT91_REG *) 	0x00000008) // (SPI_RDR) Receive Data Register
-#define SPI_TDR         (AT91_CAST(AT91_REG *) 	0x0000000C) // (SPI_TDR) Transmit Data Register
-#define SPI_SR          (AT91_CAST(AT91_REG *) 	0x00000010) // (SPI_SR) Status Register
-#define SPI_IER         (AT91_CAST(AT91_REG *) 	0x00000014) // (SPI_IER) Interrupt Enable Register
-#define SPI_IDR         (AT91_CAST(AT91_REG *) 	0x00000018) // (SPI_IDR) Interrupt Disable Register
-#define SPI_IMR         (AT91_CAST(AT91_REG *) 	0x0000001C) // (SPI_IMR) Interrupt Mask Register
-#define SPI_CSR         (AT91_CAST(AT91_REG *) 	0x00000030) // (SPI_CSR) Chip Select Register
+#define SPI_CR          (AT91_CAST(AT91_REG *)  0x00000000) // (SPI_CR) Control Register
+#define SPI_MR          (AT91_CAST(AT91_REG *)  0x00000004) // (SPI_MR) Mode Register
+#define SPI_RDR         (AT91_CAST(AT91_REG *)  0x00000008) // (SPI_RDR) Receive Data Register
+#define SPI_TDR         (AT91_CAST(AT91_REG *)  0x0000000C) // (SPI_TDR) Transmit Data Register
+#define SPI_SR          (AT91_CAST(AT91_REG *)  0x00000010) // (SPI_SR) Status Register
+#define SPI_IER         (AT91_CAST(AT91_REG *)  0x00000014) // (SPI_IER) Interrupt Enable Register
+#define SPI_IDR         (AT91_CAST(AT91_REG *)  0x00000018) // (SPI_IDR) Interrupt Disable Register
+#define SPI_IMR         (AT91_CAST(AT91_REG *)  0x0000001C) // (SPI_IMR) Interrupt Mask Register
+#define SPI_CSR         (AT91_CAST(AT91_REG *)  0x00000030) // (SPI_CSR) Chip Select Register
 
 #endif
 // -------- SPI_CR : (SPI Offset: 0x0) SPI Control Register --------
@@ -907,8 +907,8 @@ typedef struct _AT91S_SPI {
 // -------- SPI_MR : (SPI Offset: 0x4) SPI Mode Register --------
 #define AT91C_SPI_MSTR        (0x1 <<  0) // (SPI) Master/Slave Mode
 #define AT91C_SPI_PS          (0x1 <<  1) // (SPI) Peripheral Select
-#define 	AT91C_SPI_PS_FIXED                (0x0 <<  1) // (SPI) Fixed Peripheral Select
-#define 	AT91C_SPI_PS_VARIABLE             (0x1 <<  1) // (SPI) Variable Peripheral Select
+#define     AT91C_SPI_PS_FIXED                (0x0 <<  1) // (SPI) Fixed Peripheral Select
+#define     AT91C_SPI_PS_VARIABLE             (0x1 <<  1) // (SPI) Variable Peripheral Select
 #define AT91C_SPI_PCSDEC      (0x1 <<  2) // (SPI) Chip Select Decode
 #define AT91C_SPI_FDIV        (0x1 <<  3) // (SPI) Clock Selection
 #define AT91C_SPI_MODFDIS     (0x1 <<  4) // (SPI) Mode Fault Detection
@@ -941,15 +941,15 @@ typedef struct _AT91S_SPI {
 #define AT91C_SPI_NCPHA       (0x1 <<  1) // (SPI) Clock Phase
 #define AT91C_SPI_CSAAT       (0x1 <<  3) // (SPI) Chip Select Active After Transfer
 #define AT91C_SPI_BITS        (0xF <<  4) // (SPI) Bits Per Transfer
-#define 	AT91C_SPI_BITS_8                    (0x0 <<  4) // (SPI) 8 Bits Per transfer
-#define 	AT91C_SPI_BITS_9                    (0x1 <<  4) // (SPI) 9 Bits Per transfer
-#define 	AT91C_SPI_BITS_10                   (0x2 <<  4) // (SPI) 10 Bits Per transfer
-#define 	AT91C_SPI_BITS_11                   (0x3 <<  4) // (SPI) 11 Bits Per transfer
-#define 	AT91C_SPI_BITS_12                   (0x4 <<  4) // (SPI) 12 Bits Per transfer
-#define 	AT91C_SPI_BITS_13                   (0x5 <<  4) // (SPI) 13 Bits Per transfer
-#define 	AT91C_SPI_BITS_14                   (0x6 <<  4) // (SPI) 14 Bits Per transfer
-#define 	AT91C_SPI_BITS_15                   (0x7 <<  4) // (SPI) 15 Bits Per transfer
-#define 	AT91C_SPI_BITS_16                   (0x8 <<  4) // (SPI) 16 Bits Per transfer
+#define     AT91C_SPI_BITS_8                    (0x0 <<  4) // (SPI) 8 Bits Per transfer
+#define     AT91C_SPI_BITS_9                    (0x1 <<  4) // (SPI) 9 Bits Per transfer
+#define     AT91C_SPI_BITS_10                   (0x2 <<  4) // (SPI) 10 Bits Per transfer
+#define     AT91C_SPI_BITS_11                   (0x3 <<  4) // (SPI) 11 Bits Per transfer
+#define     AT91C_SPI_BITS_12                   (0x4 <<  4) // (SPI) 12 Bits Per transfer
+#define     AT91C_SPI_BITS_13                   (0x5 <<  4) // (SPI) 13 Bits Per transfer
+#define     AT91C_SPI_BITS_14                   (0x6 <<  4) // (SPI) 14 Bits Per transfer
+#define     AT91C_SPI_BITS_15                   (0x7 <<  4) // (SPI) 15 Bits Per transfer
+#define     AT91C_SPI_BITS_16                   (0x8 <<  4) // (SPI) 16 Bits Per transfer
 #define AT91C_SPI_SCBR        (0xFF <<  8) // (SPI) Serial Clock Baud Rate
 #define AT91C_SPI_DLYBS       (0xFF << 16) // (SPI) Delay Before SPCK
 #define AT91C_SPI_DLYBCT      (0xFF << 24) // (SPI) Delay Between Consecutive Transfers
@@ -959,49 +959,49 @@ typedef struct _AT91S_SPI {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_ADC {
-    AT91_REG	 ADC_CR; 	// ADC Control Register
-    AT91_REG	 ADC_MR; 	// ADC Mode Register
-    AT91_REG	 Reserved0[2]; 	//
-    AT91_REG	 ADC_CHER; 	// ADC Channel Enable Register
-    AT91_REG	 ADC_CHDR; 	// ADC Channel Disable Register
-    AT91_REG	 ADC_CHSR; 	// ADC Channel Status Register
-    AT91_REG	 ADC_SR; 	// ADC Status Register
-    AT91_REG	 ADC_LCDR; 	// ADC Last Converted Data Register
-    AT91_REG	 ADC_IER; 	// ADC Interrupt Enable Register
-    AT91_REG	 ADC_IDR; 	// ADC Interrupt Disable Register
-    AT91_REG	 ADC_IMR; 	// ADC Interrupt Mask Register
-    AT91_REG	 ADC_CDR[8]; // ADC Channel Data Register
-    AT91_REG	 Reserved1[44]; 	//
-    AT91_REG	 ADC_RPR; 	// Receive Pointer Register
-    AT91_REG	 ADC_RCR; 	// Receive Counter Register
-    AT91_REG	 ADC_TPR; 	// Transmit Pointer Register
-    AT91_REG	 ADC_TCR; 	// Transmit Counter Register
-    AT91_REG	 ADC_RNPR; 	// Receive Next Pointer Register
-    AT91_REG	 ADC_RNCR; 	// Receive Next Counter Register
-    AT91_REG	 ADC_TNPR; 	// Transmit Next Pointer Register
-    AT91_REG	 ADC_TNCR; 	// Transmit Next Counter Register
-    AT91_REG	 ADC_PTCR; 	// PDC Transfer Control Register
-    AT91_REG	 ADC_PTSR; 	// PDC Transfer Status Register
+    AT91_REG     ADC_CR;    // ADC Control Register
+    AT91_REG     ADC_MR;    // ADC Mode Register
+    AT91_REG     Reserved0[2];  //
+    AT91_REG     ADC_CHER;  // ADC Channel Enable Register
+    AT91_REG     ADC_CHDR;  // ADC Channel Disable Register
+    AT91_REG     ADC_CHSR;  // ADC Channel Status Register
+    AT91_REG     ADC_SR;    // ADC Status Register
+    AT91_REG     ADC_LCDR;  // ADC Last Converted Data Register
+    AT91_REG     ADC_IER;   // ADC Interrupt Enable Register
+    AT91_REG     ADC_IDR;   // ADC Interrupt Disable Register
+    AT91_REG     ADC_IMR;   // ADC Interrupt Mask Register
+    AT91_REG     ADC_CDR[8]; // ADC Channel Data Register
+    AT91_REG     Reserved1[44];     //
+    AT91_REG     ADC_RPR;   // Receive Pointer Register
+    AT91_REG     ADC_RCR;   // Receive Counter Register
+    AT91_REG     ADC_TPR;   // Transmit Pointer Register
+    AT91_REG     ADC_TCR;   // Transmit Counter Register
+    AT91_REG     ADC_RNPR;  // Receive Next Pointer Register
+    AT91_REG     ADC_RNCR;  // Receive Next Counter Register
+    AT91_REG     ADC_TNPR;  // Transmit Next Pointer Register
+    AT91_REG     ADC_TNCR;  // Transmit Next Counter Register
+    AT91_REG     ADC_PTCR;  // PDC Transfer Control Register
+    AT91_REG     ADC_PTSR;  // PDC Transfer Status Register
 } AT91S_ADC, *AT91PS_ADC;
 #else
-#define ADC_CR          (AT91_CAST(AT91_REG *) 	0x00000000) // (ADC_CR) ADC Control Register
-#define ADC_MR          (AT91_CAST(AT91_REG *) 	0x00000004) // (ADC_MR) ADC Mode Register
-#define ADC_CHER        (AT91_CAST(AT91_REG *) 	0x00000010) // (ADC_CHER) ADC Channel Enable Register
-#define ADC_CHDR        (AT91_CAST(AT91_REG *) 	0x00000014) // (ADC_CHDR) ADC Channel Disable Register
-#define ADC_CHSR        (AT91_CAST(AT91_REG *) 	0x00000018) // (ADC_CHSR) ADC Channel Status Register
-#define ADC_SR          (AT91_CAST(AT91_REG *) 	0x0000001C) // (ADC_SR) ADC Status Register
-#define ADC_LCDR        (AT91_CAST(AT91_REG *) 	0x00000020) // (ADC_LCDR) ADC Last Converted Data Register
-#define ADC_IER         (AT91_CAST(AT91_REG *) 	0x00000024) // (ADC_IER) ADC Interrupt Enable Register
-#define ADC_IDR         (AT91_CAST(AT91_REG *) 	0x00000028) // (ADC_IDR) ADC Interrupt Disable Register
-#define ADC_IMR         (AT91_CAST(AT91_REG *) 	0x0000002C) // (ADC_IMR) ADC Interrupt Mask Register
-#define ADC_CDR0        (AT91_CAST(AT91_REG *) 	0x00000030) // (ADC_CDR0) ADC Channel Data Register 0
-#define ADC_CDR1        (AT91_CAST(AT91_REG *) 	0x00000034) // (ADC_CDR1) ADC Channel Data Register 1
-#define ADC_CDR2        (AT91_CAST(AT91_REG *) 	0x00000038) // (ADC_CDR2) ADC Channel Data Register 2
-#define ADC_CDR3        (AT91_CAST(AT91_REG *) 	0x0000003C) // (ADC_CDR3) ADC Channel Data Register 3
-#define ADC_CDR4        (AT91_CAST(AT91_REG *) 	0x00000040) // (ADC_CDR4) ADC Channel Data Register 4
-#define ADC_CDR5        (AT91_CAST(AT91_REG *) 	0x00000044) // (ADC_CDR5) ADC Channel Data Register 5
-#define ADC_CDR6        (AT91_CAST(AT91_REG *) 	0x00000048) // (ADC_CDR6) ADC Channel Data Register 6
-#define ADC_CDR7        (AT91_CAST(AT91_REG *) 	0x0000004C) // (ADC_CDR7) ADC Channel Data Register 7
+#define ADC_CR          (AT91_CAST(AT91_REG *)  0x00000000) // (ADC_CR) ADC Control Register
+#define ADC_MR          (AT91_CAST(AT91_REG *)  0x00000004) // (ADC_MR) ADC Mode Register
+#define ADC_CHER        (AT91_CAST(AT91_REG *)  0x00000010) // (ADC_CHER) ADC Channel Enable Register
+#define ADC_CHDR        (AT91_CAST(AT91_REG *)  0x00000014) // (ADC_CHDR) ADC Channel Disable Register
+#define ADC_CHSR        (AT91_CAST(AT91_REG *)  0x00000018) // (ADC_CHSR) ADC Channel Status Register
+#define ADC_SR          (AT91_CAST(AT91_REG *)  0x0000001C) // (ADC_SR) ADC Status Register
+#define ADC_LCDR        (AT91_CAST(AT91_REG *)  0x00000020) // (ADC_LCDR) ADC Last Converted Data Register
+#define ADC_IER         (AT91_CAST(AT91_REG *)  0x00000024) // (ADC_IER) ADC Interrupt Enable Register
+#define ADC_IDR         (AT91_CAST(AT91_REG *)  0x00000028) // (ADC_IDR) ADC Interrupt Disable Register
+#define ADC_IMR         (AT91_CAST(AT91_REG *)  0x0000002C) // (ADC_IMR) ADC Interrupt Mask Register
+#define ADC_CDR0        (AT91_CAST(AT91_REG *)  0x00000030) // (ADC_CDR0) ADC Channel Data Register 0
+#define ADC_CDR1        (AT91_CAST(AT91_REG *)  0x00000034) // (ADC_CDR1) ADC Channel Data Register 1
+#define ADC_CDR2        (AT91_CAST(AT91_REG *)  0x00000038) // (ADC_CDR2) ADC Channel Data Register 2
+#define ADC_CDR3        (AT91_CAST(AT91_REG *)  0x0000003C) // (ADC_CDR3) ADC Channel Data Register 3
+#define ADC_CDR4        (AT91_CAST(AT91_REG *)  0x00000040) // (ADC_CDR4) ADC Channel Data Register 4
+#define ADC_CDR5        (AT91_CAST(AT91_REG *)  0x00000044) // (ADC_CDR5) ADC Channel Data Register 5
+#define ADC_CDR6        (AT91_CAST(AT91_REG *)  0x00000048) // (ADC_CDR6) ADC Channel Data Register 6
+#define ADC_CDR7        (AT91_CAST(AT91_REG *)  0x0000004C) // (ADC_CDR7) ADC Channel Data Register 7
 
 #endif
 // -------- ADC_CR : (ADC Offset: 0x0) ADC Control Register --------
@@ -1009,26 +1009,26 @@ typedef struct _AT91S_ADC {
 #define AT91C_ADC_START       (0x1 <<  1) // (ADC) Start Conversion
 // -------- ADC_MR : (ADC Offset: 0x4) ADC Mode Register --------
 #define AT91C_ADC_TRGEN       (0x1 <<  0) // (ADC) Trigger Enable
-#define 	AT91C_ADC_TRGEN_DIS                  (0x0) // (ADC) Hradware triggers are disabled. Starting a conversion is only possible by software
-#define 	AT91C_ADC_TRGEN_EN                   (0x1) // (ADC) Hardware trigger selected by TRGSEL field is enabled.
+#define     AT91C_ADC_TRGEN_DIS                  (0x0) // (ADC) Hradware triggers are disabled. Starting a conversion is only possible by software
+#define     AT91C_ADC_TRGEN_EN                   (0x1) // (ADC) Hardware trigger selected by TRGSEL field is enabled.
 #define AT91C_ADC_TRGSEL      (0x7 <<  1) // (ADC) Trigger Selection
-#define 	AT91C_ADC_TRGSEL_TIOA0                (0x0 <<  1) // (ADC) Selected TRGSEL = TIAO0
-#define 	AT91C_ADC_TRGSEL_TIOA1                (0x1 <<  1) // (ADC) Selected TRGSEL = TIAO1
-#define 	AT91C_ADC_TRGSEL_TIOA2                (0x2 <<  1) // (ADC) Selected TRGSEL = TIAO2
-#define 	AT91C_ADC_TRGSEL_TIOA3                (0x3 <<  1) // (ADC) Selected TRGSEL = TIAO3
-#define 	AT91C_ADC_TRGSEL_TIOA4                (0x4 <<  1) // (ADC) Selected TRGSEL = TIAO4
-#define 	AT91C_ADC_TRGSEL_TIOA5                (0x5 <<  1) // (ADC) Selected TRGSEL = TIAO5
-#define 	AT91C_ADC_TRGSEL_EXT                  (0x6 <<  1) // (ADC) Selected TRGSEL = External Trigger
+#define     AT91C_ADC_TRGSEL_TIOA0                (0x0 <<  1) // (ADC) Selected TRGSEL = TIAO0
+#define     AT91C_ADC_TRGSEL_TIOA1                (0x1 <<  1) // (ADC) Selected TRGSEL = TIAO1
+#define     AT91C_ADC_TRGSEL_TIOA2                (0x2 <<  1) // (ADC) Selected TRGSEL = TIAO2
+#define     AT91C_ADC_TRGSEL_TIOA3                (0x3 <<  1) // (ADC) Selected TRGSEL = TIAO3
+#define     AT91C_ADC_TRGSEL_TIOA4                (0x4 <<  1) // (ADC) Selected TRGSEL = TIAO4
+#define     AT91C_ADC_TRGSEL_TIOA5                (0x5 <<  1) // (ADC) Selected TRGSEL = TIAO5
+#define     AT91C_ADC_TRGSEL_EXT                  (0x6 <<  1) // (ADC) Selected TRGSEL = External Trigger
 #define AT91C_ADC_LOWRES      (0x1 <<  4) // (ADC) Resolution.
-#define 	AT91C_ADC_LOWRES_10_BIT               (0x0 <<  4) // (ADC) 10-bit resolution
-#define 	AT91C_ADC_LOWRES_8_BIT                (0x1 <<  4) // (ADC) 8-bit resolution
+#define     AT91C_ADC_LOWRES_10_BIT               (0x0 <<  4) // (ADC) 10-bit resolution
+#define     AT91C_ADC_LOWRES_8_BIT                (0x1 <<  4) // (ADC) 8-bit resolution
 #define AT91C_ADC_SLEEP       (0x1 <<  5) // (ADC) Sleep Mode
-#define 	AT91C_ADC_SLEEP_NORMAL_MODE          (0x0 <<  5) // (ADC) Normal Mode
-#define 	AT91C_ADC_SLEEP_MODE                 (0x1 <<  5) // (ADC) Sleep Mode
+#define     AT91C_ADC_SLEEP_NORMAL_MODE          (0x0 <<  5) // (ADC) Normal Mode
+#define     AT91C_ADC_SLEEP_MODE                 (0x1 <<  5) // (ADC) Sleep Mode
 #define AT91C_ADC_PRESCAL     (0x3F <<  8) // (ADC) Prescaler rate selection
 #define AT91C_ADC_STARTUP     (0x1F << 16) // (ADC) Startup Time
 #define AT91C_ADC_SHTIM       (0xF << 24) // (ADC) Sample & Hold Time
-// -------- 	ADC_CHER : (ADC Offset: 0x10) ADC Channel Enable Register --------
+// --------     ADC_CHER : (ADC Offset: 0x10) ADC Channel Enable Register --------
 #define AT91C_ADC_CH0         (0x1 <<  0) // (ADC) Channel 0
 #define AT91C_ADC_CH1         (0x1 <<  1) // (ADC) Channel 1
 #define AT91C_ADC_CH2         (0x1 <<  2) // (ADC) Channel 2
@@ -1037,8 +1037,8 @@ typedef struct _AT91S_ADC {
 #define AT91C_ADC_CH5         (0x1 <<  5) // (ADC) Channel 5
 #define AT91C_ADC_CH6         (0x1 <<  6) // (ADC) Channel 6
 #define AT91C_ADC_CH7         (0x1 <<  7) // (ADC) Channel 7
-// -------- 	ADC_CHDR : (ADC Offset: 0x14) ADC Channel Disable Register --------
-// -------- 	ADC_CHSR : (ADC Offset: 0x18) ADC Channel Status Register --------
+// --------     ADC_CHDR : (ADC Offset: 0x14) ADC Channel Disable Register --------
+// --------     ADC_CHSR : (ADC Offset: 0x18) ADC Channel Status Register --------
 // -------- ADC_SR : (ADC Offset: 0x1c) ADC Status Register --------
 #define AT91C_ADC_EOC0        (0x1 <<  0) // (ADC) End of Conversion
 #define AT91C_ADC_EOC1        (0x1 <<  1) // (ADC) End of Conversion
@@ -1080,50 +1080,50 @@ typedef struct _AT91S_ADC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_SSC {
-    AT91_REG	 SSC_CR; 	// Control Register
-    AT91_REG	 SSC_CMR; 	// Clock Mode Register
-    AT91_REG	 Reserved0[2]; 	//
-    AT91_REG	 SSC_RCMR; 	// Receive Clock ModeRegister
-    AT91_REG	 SSC_RFMR; 	// Receive Frame Mode Register
-    AT91_REG	 SSC_TCMR; 	// Transmit Clock Mode Register
-    AT91_REG	 SSC_TFMR; 	// Transmit Frame Mode Register
-    AT91_REG	 SSC_RHR; 	// Receive Holding Register
-    AT91_REG	 SSC_THR; 	// Transmit Holding Register
-    AT91_REG	 Reserved1[2]; 	//
-    AT91_REG	 SSC_RSHR; 	// Receive Sync Holding Register
-    AT91_REG	 SSC_TSHR; 	// Transmit Sync Holding Register
-    AT91_REG	 Reserved2[2]; 	//
-    AT91_REG	 SSC_SR; 	// Status Register
-    AT91_REG	 SSC_IER; 	// Interrupt Enable Register
-    AT91_REG	 SSC_IDR; 	// Interrupt Disable Register
-    AT91_REG	 SSC_IMR; 	// Interrupt Mask Register
-    AT91_REG	 Reserved3[44]; 	//
-    AT91_REG	 SSC_RPR; 	// Receive Pointer Register
-    AT91_REG	 SSC_RCR; 	// Receive Counter Register
-    AT91_REG	 SSC_TPR; 	// Transmit Pointer Register
-    AT91_REG	 SSC_TCR; 	// Transmit Counter Register
-    AT91_REG	 SSC_RNPR; 	// Receive Next Pointer Register
-    AT91_REG	 SSC_RNCR; 	// Receive Next Counter Register
-    AT91_REG	 SSC_TNPR; 	// Transmit Next Pointer Register
-    AT91_REG	 SSC_TNCR; 	// Transmit Next Counter Register
-    AT91_REG	 SSC_PTCR; 	// PDC Transfer Control Register
-    AT91_REG	 SSC_PTSR; 	// PDC Transfer Status Register
+    AT91_REG     SSC_CR;    // Control Register
+    AT91_REG     SSC_CMR;   // Clock Mode Register
+    AT91_REG     Reserved0[2];  //
+    AT91_REG     SSC_RCMR;  // Receive Clock ModeRegister
+    AT91_REG     SSC_RFMR;  // Receive Frame Mode Register
+    AT91_REG     SSC_TCMR;  // Transmit Clock Mode Register
+    AT91_REG     SSC_TFMR;  // Transmit Frame Mode Register
+    AT91_REG     SSC_RHR;   // Receive Holding Register
+    AT91_REG     SSC_THR;   // Transmit Holding Register
+    AT91_REG     Reserved1[2];  //
+    AT91_REG     SSC_RSHR;  // Receive Sync Holding Register
+    AT91_REG     SSC_TSHR;  // Transmit Sync Holding Register
+    AT91_REG     Reserved2[2];  //
+    AT91_REG     SSC_SR;    // Status Register
+    AT91_REG     SSC_IER;   // Interrupt Enable Register
+    AT91_REG     SSC_IDR;   // Interrupt Disable Register
+    AT91_REG     SSC_IMR;   // Interrupt Mask Register
+    AT91_REG     Reserved3[44];     //
+    AT91_REG     SSC_RPR;   // Receive Pointer Register
+    AT91_REG     SSC_RCR;   // Receive Counter Register
+    AT91_REG     SSC_TPR;   // Transmit Pointer Register
+    AT91_REG     SSC_TCR;   // Transmit Counter Register
+    AT91_REG     SSC_RNPR;  // Receive Next Pointer Register
+    AT91_REG     SSC_RNCR;  // Receive Next Counter Register
+    AT91_REG     SSC_TNPR;  // Transmit Next Pointer Register
+    AT91_REG     SSC_TNCR;  // Transmit Next Counter Register
+    AT91_REG     SSC_PTCR;  // PDC Transfer Control Register
+    AT91_REG     SSC_PTSR;  // PDC Transfer Status Register
 } AT91S_SSC, *AT91PS_SSC;
 #else
-#define SSC_CR          (AT91_CAST(AT91_REG *) 	0x00000000) // (SSC_CR) Control Register
-#define SSC_CMR         (AT91_CAST(AT91_REG *) 	0x00000004) // (SSC_CMR) Clock Mode Register
-#define SSC_RCMR        (AT91_CAST(AT91_REG *) 	0x00000010) // (SSC_RCMR) Receive Clock ModeRegister
-#define SSC_RFMR        (AT91_CAST(AT91_REG *) 	0x00000014) // (SSC_RFMR) Receive Frame Mode Register
-#define SSC_TCMR        (AT91_CAST(AT91_REG *) 	0x00000018) // (SSC_TCMR) Transmit Clock Mode Register
-#define SSC_TFMR        (AT91_CAST(AT91_REG *) 	0x0000001C) // (SSC_TFMR) Transmit Frame Mode Register
-#define SSC_RHR         (AT91_CAST(AT91_REG *) 	0x00000020) // (SSC_RHR) Receive Holding Register
-#define SSC_THR         (AT91_CAST(AT91_REG *) 	0x00000024) // (SSC_THR) Transmit Holding Register
-#define SSC_RSHR        (AT91_CAST(AT91_REG *) 	0x00000030) // (SSC_RSHR) Receive Sync Holding Register
-#define SSC_TSHR        (AT91_CAST(AT91_REG *) 	0x00000034) // (SSC_TSHR) Transmit Sync Holding Register
-#define SSC_SR          (AT91_CAST(AT91_REG *) 	0x00000040) // (SSC_SR) Status Register
-#define SSC_IER         (AT91_CAST(AT91_REG *) 	0x00000044) // (SSC_IER) Interrupt Enable Register
-#define SSC_IDR         (AT91_CAST(AT91_REG *) 	0x00000048) // (SSC_IDR) Interrupt Disable Register
-#define SSC_IMR         (AT91_CAST(AT91_REG *) 	0x0000004C) // (SSC_IMR) Interrupt Mask Register
+#define SSC_CR          (AT91_CAST(AT91_REG *)  0x00000000) // (SSC_CR) Control Register
+#define SSC_CMR         (AT91_CAST(AT91_REG *)  0x00000004) // (SSC_CMR) Clock Mode Register
+#define SSC_RCMR        (AT91_CAST(AT91_REG *)  0x00000010) // (SSC_RCMR) Receive Clock ModeRegister
+#define SSC_RFMR        (AT91_CAST(AT91_REG *)  0x00000014) // (SSC_RFMR) Receive Frame Mode Register
+#define SSC_TCMR        (AT91_CAST(AT91_REG *)  0x00000018) // (SSC_TCMR) Transmit Clock Mode Register
+#define SSC_TFMR        (AT91_CAST(AT91_REG *)  0x0000001C) // (SSC_TFMR) Transmit Frame Mode Register
+#define SSC_RHR         (AT91_CAST(AT91_REG *)  0x00000020) // (SSC_RHR) Receive Holding Register
+#define SSC_THR         (AT91_CAST(AT91_REG *)  0x00000024) // (SSC_THR) Transmit Holding Register
+#define SSC_RSHR        (AT91_CAST(AT91_REG *)  0x00000030) // (SSC_RSHR) Receive Sync Holding Register
+#define SSC_TSHR        (AT91_CAST(AT91_REG *)  0x00000034) // (SSC_TSHR) Transmit Sync Holding Register
+#define SSC_SR          (AT91_CAST(AT91_REG *)  0x00000040) // (SSC_SR) Status Register
+#define SSC_IER         (AT91_CAST(AT91_REG *)  0x00000044) // (SSC_IER) Interrupt Enable Register
+#define SSC_IDR         (AT91_CAST(AT91_REG *)  0x00000048) // (SSC_IDR) Interrupt Disable Register
+#define SSC_IMR         (AT91_CAST(AT91_REG *)  0x0000004C) // (SSC_IMR) Interrupt Mask Register
 
 #endif
 // -------- SSC_CR : (SSC Offset: 0x0) SSC Control Register --------
@@ -1134,24 +1134,24 @@ typedef struct _AT91S_SSC {
 #define AT91C_SSC_SWRST       (0x1 << 15) // (SSC) Software Reset
 // -------- SSC_RCMR : (SSC Offset: 0x10) SSC Receive Clock Mode Register --------
 #define AT91C_SSC_CKS         (0x3 <<  0) // (SSC) Receive/Transmit Clock Selection
-#define 	AT91C_SSC_CKS_DIV                  (0x0) // (SSC) Divided Clock
-#define 	AT91C_SSC_CKS_TK                   (0x1) // (SSC) TK Clock signal
-#define 	AT91C_SSC_CKS_RK                   (0x2) // (SSC) RK pin
+#define     AT91C_SSC_CKS_DIV                  (0x0) // (SSC) Divided Clock
+#define     AT91C_SSC_CKS_TK                   (0x1) // (SSC) TK Clock signal
+#define     AT91C_SSC_CKS_RK                   (0x2) // (SSC) RK pin
 #define AT91C_SSC_CKO         (0x7 <<  2) // (SSC) Receive/Transmit Clock Output Mode Selection
-#define 	AT91C_SSC_CKO_NONE                 (0x0 <<  2) // (SSC) Receive/Transmit Clock Output Mode: None RK pin: Input-only
-#define 	AT91C_SSC_CKO_CONTINOUS            (0x1 <<  2) // (SSC) Continuous Receive/Transmit Clock RK pin: Output
-#define 	AT91C_SSC_CKO_DATA_TX              (0x2 <<  2) // (SSC) Receive/Transmit Clock only during data transfers RK pin: Output
+#define     AT91C_SSC_CKO_NONE                 (0x0 <<  2) // (SSC) Receive/Transmit Clock Output Mode: None RK pin: Input-only
+#define     AT91C_SSC_CKO_CONTINOUS            (0x1 <<  2) // (SSC) Continuous Receive/Transmit Clock RK pin: Output
+#define     AT91C_SSC_CKO_DATA_TX              (0x2 <<  2) // (SSC) Receive/Transmit Clock only during data transfers RK pin: Output
 #define AT91C_SSC_CKI         (0x1 <<  5) // (SSC) Receive/Transmit Clock Inversion
 #define AT91C_SSC_START       (0xF <<  8) // (SSC) Receive/Transmit Start Selection
-#define 	AT91C_SSC_START_CONTINOUS            (0x0 <<  8) // (SSC) Continuous, as soon as the receiver is enabled, and immediately after the end of transfer of the previous data.
-#define 	AT91C_SSC_START_TX                   (0x1 <<  8) // (SSC) Transmit/Receive start
-#define 	AT91C_SSC_START_LOW_RF               (0x2 <<  8) // (SSC) Detection of a low level on RF input
-#define 	AT91C_SSC_START_HIGH_RF              (0x3 <<  8) // (SSC) Detection of a high level on RF input
-#define 	AT91C_SSC_START_FALL_RF              (0x4 <<  8) // (SSC) Detection of a falling edge on RF input
-#define 	AT91C_SSC_START_RISE_RF              (0x5 <<  8) // (SSC) Detection of a rising edge on RF input
-#define 	AT91C_SSC_START_LEVEL_RF             (0x6 <<  8) // (SSC) Detection of any level change on RF input
-#define 	AT91C_SSC_START_EDGE_RF              (0x7 <<  8) // (SSC) Detection of any edge on RF input
-#define 	AT91C_SSC_START_0                    (0x8 <<  8) // (SSC) Compare 0
+#define     AT91C_SSC_START_CONTINOUS            (0x0 <<  8) // (SSC) Continuous, as soon as the receiver is enabled, and immediately after the end of transfer of the previous data.
+#define     AT91C_SSC_START_TX                   (0x1 <<  8) // (SSC) Transmit/Receive start
+#define     AT91C_SSC_START_LOW_RF               (0x2 <<  8) // (SSC) Detection of a low level on RF input
+#define     AT91C_SSC_START_HIGH_RF              (0x3 <<  8) // (SSC) Detection of a high level on RF input
+#define     AT91C_SSC_START_FALL_RF              (0x4 <<  8) // (SSC) Detection of a falling edge on RF input
+#define     AT91C_SSC_START_RISE_RF              (0x5 <<  8) // (SSC) Detection of a rising edge on RF input
+#define     AT91C_SSC_START_LEVEL_RF             (0x6 <<  8) // (SSC) Detection of any level change on RF input
+#define     AT91C_SSC_START_EDGE_RF              (0x7 <<  8) // (SSC) Detection of any edge on RF input
+#define     AT91C_SSC_START_0                    (0x8 <<  8) // (SSC) Compare 0
 #define AT91C_SSC_STTDLY      (0xFF << 16) // (SSC) Receive/Transmit Start Delay
 #define AT91C_SSC_PERIOD      (0xFF << 24) // (SSC) Receive/Transmit Period Divider Selection
 // -------- SSC_RFMR : (SSC Offset: 0x14) SSC Receive Frame Mode Register --------
@@ -1161,12 +1161,12 @@ typedef struct _AT91S_SSC {
 #define AT91C_SSC_DATNB       (0xF <<  8) // (SSC) Data Number per Frame
 #define AT91C_SSC_FSLEN       (0xF << 16) // (SSC) Receive/Transmit Frame Sync length
 #define AT91C_SSC_FSOS        (0x7 << 20) // (SSC) Receive/Transmit Frame Sync Output Selection
-#define 	AT91C_SSC_FSOS_NONE                 (0x0 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: None RK pin Input-only
-#define 	AT91C_SSC_FSOS_NEGATIVE             (0x1 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Negative Pulse
-#define 	AT91C_SSC_FSOS_POSITIVE             (0x2 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Positive Pulse
-#define 	AT91C_SSC_FSOS_LOW                  (0x3 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Driver Low during data transfer
-#define 	AT91C_SSC_FSOS_HIGH                 (0x4 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Driver High during data transfer
-#define 	AT91C_SSC_FSOS_TOGGLE               (0x5 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Toggling at each start of data transfer
+#define     AT91C_SSC_FSOS_NONE                 (0x0 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: None RK pin Input-only
+#define     AT91C_SSC_FSOS_NEGATIVE             (0x1 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Negative Pulse
+#define     AT91C_SSC_FSOS_POSITIVE             (0x2 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Positive Pulse
+#define     AT91C_SSC_FSOS_LOW                  (0x3 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Driver Low during data transfer
+#define     AT91C_SSC_FSOS_HIGH                 (0x4 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Driver High during data transfer
+#define     AT91C_SSC_FSOS_TOGGLE               (0x5 << 20) // (SSC) Selected Receive/Transmit Frame Sync Signal: Toggling at each start of data transfer
 #define AT91C_SSC_FSEDGE      (0x1 << 24) // (SSC) Frame Sync Edge Detection
 // -------- SSC_TCMR : (SSC Offset: 0x18) SSC Transmit Clock Mode Register --------
 // -------- SSC_TFMR : (SSC Offset: 0x1c) SSC Transmit Frame Mode Register --------
@@ -1194,49 +1194,49 @@ typedef struct _AT91S_SSC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_USART {
-    AT91_REG	 US_CR; 	// Control Register
-    AT91_REG	 US_MR; 	// Mode Register
-    AT91_REG	 US_IER; 	// Interrupt Enable Register
-    AT91_REG	 US_IDR; 	// Interrupt Disable Register
-    AT91_REG	 US_IMR; 	// Interrupt Mask Register
-    AT91_REG	 US_CSR; 	// Channel Status Register
-    AT91_REG	 US_RHR; 	// Receiver Holding Register
-    AT91_REG	 US_THR; 	// Transmitter Holding Register
-    AT91_REG	 US_BRGR; 	// Baud Rate Generator Register
-    AT91_REG	 US_RTOR; 	// Receiver Time-out Register
-    AT91_REG	 US_TTGR; 	// Transmitter Time-guard Register
-    AT91_REG	 Reserved0[5]; 	//
-    AT91_REG	 US_FIDI; 	// FI_DI_Ratio Register
-    AT91_REG	 US_NER; 	// Nb Errors Register
-    AT91_REG	 Reserved1[1]; 	//
-    AT91_REG	 US_IF; 	// IRDA_FILTER Register
-    AT91_REG	 Reserved2[44]; 	//
-    AT91_REG	 US_RPR; 	// Receive Pointer Register
-    AT91_REG	 US_RCR; 	// Receive Counter Register
-    AT91_REG	 US_TPR; 	// Transmit Pointer Register
-    AT91_REG	 US_TCR; 	// Transmit Counter Register
-    AT91_REG	 US_RNPR; 	// Receive Next Pointer Register
-    AT91_REG	 US_RNCR; 	// Receive Next Counter Register
-    AT91_REG	 US_TNPR; 	// Transmit Next Pointer Register
-    AT91_REG	 US_TNCR; 	// Transmit Next Counter Register
-    AT91_REG	 US_PTCR; 	// PDC Transfer Control Register
-    AT91_REG	 US_PTSR; 	// PDC Transfer Status Register
+    AT91_REG     US_CR;     // Control Register
+    AT91_REG     US_MR;     // Mode Register
+    AT91_REG     US_IER;    // Interrupt Enable Register
+    AT91_REG     US_IDR;    // Interrupt Disable Register
+    AT91_REG     US_IMR;    // Interrupt Mask Register
+    AT91_REG     US_CSR;    // Channel Status Register
+    AT91_REG     US_RHR;    // Receiver Holding Register
+    AT91_REG     US_THR;    // Transmitter Holding Register
+    AT91_REG     US_BRGR;   // Baud Rate Generator Register
+    AT91_REG     US_RTOR;   // Receiver Time-out Register
+    AT91_REG     US_TTGR;   // Transmitter Time-guard Register
+    AT91_REG     Reserved0[5];  //
+    AT91_REG     US_FIDI;   // FI_DI_Ratio Register
+    AT91_REG     US_NER;    // Nb Errors Register
+    AT91_REG     Reserved1[1];  //
+    AT91_REG     US_IF;     // IRDA_FILTER Register
+    AT91_REG     Reserved2[44];     //
+    AT91_REG     US_RPR;    // Receive Pointer Register
+    AT91_REG     US_RCR;    // Receive Counter Register
+    AT91_REG     US_TPR;    // Transmit Pointer Register
+    AT91_REG     US_TCR;    // Transmit Counter Register
+    AT91_REG     US_RNPR;   // Receive Next Pointer Register
+    AT91_REG     US_RNCR;   // Receive Next Counter Register
+    AT91_REG     US_TNPR;   // Transmit Next Pointer Register
+    AT91_REG     US_TNCR;   // Transmit Next Counter Register
+    AT91_REG     US_PTCR;   // PDC Transfer Control Register
+    AT91_REG     US_PTSR;   // PDC Transfer Status Register
 } AT91S_USART, *AT91PS_USART;
 #else
-#define US_CR           (AT91_CAST(AT91_REG *) 	0x00000000) // (US_CR) Control Register
-#define US_MR           (AT91_CAST(AT91_REG *) 	0x00000004) // (US_MR) Mode Register
-#define US_IER          (AT91_CAST(AT91_REG *) 	0x00000008) // (US_IER) Interrupt Enable Register
-#define US_IDR          (AT91_CAST(AT91_REG *) 	0x0000000C) // (US_IDR) Interrupt Disable Register
-#define US_IMR          (AT91_CAST(AT91_REG *) 	0x00000010) // (US_IMR) Interrupt Mask Register
-#define US_CSR          (AT91_CAST(AT91_REG *) 	0x00000014) // (US_CSR) Channel Status Register
-#define US_RHR          (AT91_CAST(AT91_REG *) 	0x00000018) // (US_RHR) Receiver Holding Register
-#define US_THR          (AT91_CAST(AT91_REG *) 	0x0000001C) // (US_THR) Transmitter Holding Register
-#define US_BRGR         (AT91_CAST(AT91_REG *) 	0x00000020) // (US_BRGR) Baud Rate Generator Register
-#define US_RTOR         (AT91_CAST(AT91_REG *) 	0x00000024) // (US_RTOR) Receiver Time-out Register
-#define US_TTGR         (AT91_CAST(AT91_REG *) 	0x00000028) // (US_TTGR) Transmitter Time-guard Register
-#define US_FIDI         (AT91_CAST(AT91_REG *) 	0x00000040) // (US_FIDI) FI_DI_Ratio Register
-#define US_NER          (AT91_CAST(AT91_REG *) 	0x00000044) // (US_NER) Nb Errors Register
-#define US_IF           (AT91_CAST(AT91_REG *) 	0x0000004C) // (US_IF) IRDA_FILTER Register
+#define US_CR           (AT91_CAST(AT91_REG *)  0x00000000) // (US_CR) Control Register
+#define US_MR           (AT91_CAST(AT91_REG *)  0x00000004) // (US_MR) Mode Register
+#define US_IER          (AT91_CAST(AT91_REG *)  0x00000008) // (US_IER) Interrupt Enable Register
+#define US_IDR          (AT91_CAST(AT91_REG *)  0x0000000C) // (US_IDR) Interrupt Disable Register
+#define US_IMR          (AT91_CAST(AT91_REG *)  0x00000010) // (US_IMR) Interrupt Mask Register
+#define US_CSR          (AT91_CAST(AT91_REG *)  0x00000014) // (US_CSR) Channel Status Register
+#define US_RHR          (AT91_CAST(AT91_REG *)  0x00000018) // (US_RHR) Receiver Holding Register
+#define US_THR          (AT91_CAST(AT91_REG *)  0x0000001C) // (US_THR) Transmitter Holding Register
+#define US_BRGR         (AT91_CAST(AT91_REG *)  0x00000020) // (US_BRGR) Baud Rate Generator Register
+#define US_RTOR         (AT91_CAST(AT91_REG *)  0x00000024) // (US_RTOR) Receiver Time-out Register
+#define US_TTGR         (AT91_CAST(AT91_REG *)  0x00000028) // (US_TTGR) Transmitter Time-guard Register
+#define US_FIDI         (AT91_CAST(AT91_REG *)  0x00000040) // (US_FIDI) FI_DI_Ratio Register
+#define US_NER          (AT91_CAST(AT91_REG *)  0x00000044) // (US_NER) Nb Errors Register
+#define US_IF           (AT91_CAST(AT91_REG *)  0x0000004C) // (US_IF) IRDA_FILTER Register
 
 #endif
 // -------- US_CR : (USART Offset: 0x0) Debug Unit Control Register --------
@@ -1253,29 +1253,29 @@ typedef struct _AT91S_USART {
 #define AT91C_US_RTSDIS       (0x1 << 19) // (USART) Request to Send Disable
 // -------- US_MR : (USART Offset: 0x4) Debug Unit Mode Register --------
 #define AT91C_US_USMODE       (0xF <<  0) // (USART) Usart mode
-#define 	AT91C_US_USMODE_NORMAL               (0x0) // (USART) Normal
-#define 	AT91C_US_USMODE_RS485                (0x1) // (USART) RS485
-#define 	AT91C_US_USMODE_HWHSH                (0x2) // (USART) Hardware Handshaking
-#define 	AT91C_US_USMODE_MODEM                (0x3) // (USART) Modem
-#define 	AT91C_US_USMODE_ISO7816_0            (0x4) // (USART) ISO7816 protocol: T = 0
-#define 	AT91C_US_USMODE_ISO7816_1            (0x6) // (USART) ISO7816 protocol: T = 1
-#define 	AT91C_US_USMODE_IRDA                 (0x8) // (USART) IrDA
-#define 	AT91C_US_USMODE_SWHSH                (0xC) // (USART) Software Handshaking
+#define     AT91C_US_USMODE_NORMAL               (0x0) // (USART) Normal
+#define     AT91C_US_USMODE_RS485                (0x1) // (USART) RS485
+#define     AT91C_US_USMODE_HWHSH                (0x2) // (USART) Hardware Handshaking
+#define     AT91C_US_USMODE_MODEM                (0x3) // (USART) Modem
+#define     AT91C_US_USMODE_ISO7816_0            (0x4) // (USART) ISO7816 protocol: T = 0
+#define     AT91C_US_USMODE_ISO7816_1            (0x6) // (USART) ISO7816 protocol: T = 1
+#define     AT91C_US_USMODE_IRDA                 (0x8) // (USART) IrDA
+#define     AT91C_US_USMODE_SWHSH                (0xC) // (USART) Software Handshaking
 #define AT91C_US_CLKS         (0x3 <<  4) // (USART) Clock Selection (Baud Rate generator Input Clock
-#define 	AT91C_US_CLKS_CLOCK                (0x0 <<  4) // (USART) Clock
-#define 	AT91C_US_CLKS_FDIV1                (0x1 <<  4) // (USART) fdiv1
-#define 	AT91C_US_CLKS_SLOW                 (0x2 <<  4) // (USART) slow_clock (ARM)
-#define 	AT91C_US_CLKS_EXT                  (0x3 <<  4) // (USART) External (SCK)
+#define     AT91C_US_CLKS_CLOCK                (0x0 <<  4) // (USART) Clock
+#define     AT91C_US_CLKS_FDIV1                (0x1 <<  4) // (USART) fdiv1
+#define     AT91C_US_CLKS_SLOW                 (0x2 <<  4) // (USART) slow_clock (ARM)
+#define     AT91C_US_CLKS_EXT                  (0x3 <<  4) // (USART) External (SCK)
 #define AT91C_US_CHRL         (0x3 <<  6) // (USART) Clock Selection (Baud Rate generator Input Clock
-#define 	AT91C_US_CHRL_5_BITS               (0x0 <<  6) // (USART) Character Length: 5 bits
-#define 	AT91C_US_CHRL_6_BITS               (0x1 <<  6) // (USART) Character Length: 6 bits
-#define 	AT91C_US_CHRL_7_BITS               (0x2 <<  6) // (USART) Character Length: 7 bits
-#define 	AT91C_US_CHRL_8_BITS               (0x3 <<  6) // (USART) Character Length: 8 bits
+#define     AT91C_US_CHRL_5_BITS               (0x0 <<  6) // (USART) Character Length: 5 bits
+#define     AT91C_US_CHRL_6_BITS               (0x1 <<  6) // (USART) Character Length: 6 bits
+#define     AT91C_US_CHRL_7_BITS               (0x2 <<  6) // (USART) Character Length: 7 bits
+#define     AT91C_US_CHRL_8_BITS               (0x3 <<  6) // (USART) Character Length: 8 bits
 #define AT91C_US_SYNC         (0x1 <<  8) // (USART) Synchronous Mode Select
 #define AT91C_US_NBSTOP       (0x3 << 12) // (USART) Number of Stop bits
-#define 	AT91C_US_NBSTOP_1_BIT                (0x0 << 12) // (USART) 1 stop bit
-#define 	AT91C_US_NBSTOP_15_BIT               (0x1 << 12) // (USART) Asynchronous (SYNC=0) 2 stop bits Synchronous (SYNC=1) 2 stop bits
-#define 	AT91C_US_NBSTOP_2_BIT                (0x2 << 12) // (USART) 2 stop bits
+#define     AT91C_US_NBSTOP_1_BIT                (0x0 << 12) // (USART) 1 stop bit
+#define     AT91C_US_NBSTOP_15_BIT               (0x1 << 12) // (USART) Asynchronous (SYNC=0) 2 stop bits Synchronous (SYNC=1) 2 stop bits
+#define     AT91C_US_NBSTOP_2_BIT                (0x2 << 12) // (USART) 2 stop bits
 #define AT91C_US_MSBF         (0x1 << 16) // (USART) Bit Order
 #define AT91C_US_MODE9        (0x1 << 17) // (USART) 9-bit Character length
 #define AT91C_US_CKLO         (0x1 << 18) // (USART) Clock Output Select
@@ -1306,41 +1306,41 @@ typedef struct _AT91S_USART {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_TWI {
-    AT91_REG	 TWI_CR; 	// Control Register
-    AT91_REG	 TWI_MMR; 	// Master Mode Register
-    AT91_REG	 Reserved0[1]; 	//
-    AT91_REG	 TWI_IADR; 	// Internal Address Register
-    AT91_REG	 TWI_CWGR; 	// Clock Waveform Generator Register
-    AT91_REG	 Reserved1[3]; 	//
-    AT91_REG	 TWI_SR; 	// Status Register
-    AT91_REG	 TWI_IER; 	// Interrupt Enable Register
-    AT91_REG	 TWI_IDR; 	// Interrupt Disable Register
-    AT91_REG	 TWI_IMR; 	// Interrupt Mask Register
-    AT91_REG	 TWI_RHR; 	// Receive Holding Register
-    AT91_REG	 TWI_THR; 	// Transmit Holding Register
-    AT91_REG	 Reserved2[50]; 	//
-    AT91_REG	 TWI_RPR; 	// Receive Pointer Register
-    AT91_REG	 TWI_RCR; 	// Receive Counter Register
-    AT91_REG	 TWI_TPR; 	// Transmit Pointer Register
-    AT91_REG	 TWI_TCR; 	// Transmit Counter Register
-    AT91_REG	 TWI_RNPR; 	// Receive Next Pointer Register
-    AT91_REG	 TWI_RNCR; 	// Receive Next Counter Register
-    AT91_REG	 TWI_TNPR; 	// Transmit Next Pointer Register
-    AT91_REG	 TWI_TNCR; 	// Transmit Next Counter Register
-    AT91_REG	 TWI_PTCR; 	// PDC Transfer Control Register
-    AT91_REG	 TWI_PTSR; 	// PDC Transfer Status Register
+    AT91_REG     TWI_CR;    // Control Register
+    AT91_REG     TWI_MMR;   // Master Mode Register
+    AT91_REG     Reserved0[1];  //
+    AT91_REG     TWI_IADR;  // Internal Address Register
+    AT91_REG     TWI_CWGR;  // Clock Waveform Generator Register
+    AT91_REG     Reserved1[3];  //
+    AT91_REG     TWI_SR;    // Status Register
+    AT91_REG     TWI_IER;   // Interrupt Enable Register
+    AT91_REG     TWI_IDR;   // Interrupt Disable Register
+    AT91_REG     TWI_IMR;   // Interrupt Mask Register
+    AT91_REG     TWI_RHR;   // Receive Holding Register
+    AT91_REG     TWI_THR;   // Transmit Holding Register
+    AT91_REG     Reserved2[50];     //
+    AT91_REG     TWI_RPR;   // Receive Pointer Register
+    AT91_REG     TWI_RCR;   // Receive Counter Register
+    AT91_REG     TWI_TPR;   // Transmit Pointer Register
+    AT91_REG     TWI_TCR;   // Transmit Counter Register
+    AT91_REG     TWI_RNPR;  // Receive Next Pointer Register
+    AT91_REG     TWI_RNCR;  // Receive Next Counter Register
+    AT91_REG     TWI_TNPR;  // Transmit Next Pointer Register
+    AT91_REG     TWI_TNCR;  // Transmit Next Counter Register
+    AT91_REG     TWI_PTCR;  // PDC Transfer Control Register
+    AT91_REG     TWI_PTSR;  // PDC Transfer Status Register
 } AT91S_TWI, *AT91PS_TWI;
 #else
-#define TWI_CR          (AT91_CAST(AT91_REG *) 	0x00000000) // (TWI_CR) Control Register
-#define TWI_MMR         (AT91_CAST(AT91_REG *) 	0x00000004) // (TWI_MMR) Master Mode Register
-#define TWI_IADR        (AT91_CAST(AT91_REG *) 	0x0000000C) // (TWI_IADR) Internal Address Register
-#define TWI_CWGR        (AT91_CAST(AT91_REG *) 	0x00000010) // (TWI_CWGR) Clock Waveform Generator Register
-#define TWI_SR          (AT91_CAST(AT91_REG *) 	0x00000020) // (TWI_SR) Status Register
-#define TWI_IER         (AT91_CAST(AT91_REG *) 	0x00000024) // (TWI_IER) Interrupt Enable Register
-#define TWI_IDR         (AT91_CAST(AT91_REG *) 	0x00000028) // (TWI_IDR) Interrupt Disable Register
-#define TWI_IMR         (AT91_CAST(AT91_REG *) 	0x0000002C) // (TWI_IMR) Interrupt Mask Register
-#define TWI_RHR         (AT91_CAST(AT91_REG *) 	0x00000030) // (TWI_RHR) Receive Holding Register
-#define TWI_THR         (AT91_CAST(AT91_REG *) 	0x00000034) // (TWI_THR) Transmit Holding Register
+#define TWI_CR          (AT91_CAST(AT91_REG *)  0x00000000) // (TWI_CR) Control Register
+#define TWI_MMR         (AT91_CAST(AT91_REG *)  0x00000004) // (TWI_MMR) Master Mode Register
+#define TWI_IADR        (AT91_CAST(AT91_REG *)  0x0000000C) // (TWI_IADR) Internal Address Register
+#define TWI_CWGR        (AT91_CAST(AT91_REG *)  0x00000010) // (TWI_CWGR) Clock Waveform Generator Register
+#define TWI_SR          (AT91_CAST(AT91_REG *)  0x00000020) // (TWI_SR) Status Register
+#define TWI_IER         (AT91_CAST(AT91_REG *)  0x00000024) // (TWI_IER) Interrupt Enable Register
+#define TWI_IDR         (AT91_CAST(AT91_REG *)  0x00000028) // (TWI_IDR) Interrupt Disable Register
+#define TWI_IMR         (AT91_CAST(AT91_REG *)  0x0000002C) // (TWI_IMR) Interrupt Mask Register
+#define TWI_RHR         (AT91_CAST(AT91_REG *)  0x00000030) // (TWI_RHR) Receive Holding Register
+#define TWI_THR         (AT91_CAST(AT91_REG *)  0x00000034) // (TWI_THR) Transmit Holding Register
 
 #endif
 // -------- TWI_CR : (TWI Offset: 0x0) TWI Control Register --------
@@ -1351,10 +1351,10 @@ typedef struct _AT91S_TWI {
 #define AT91C_TWI_SWRST       (0x1 <<  7) // (TWI) Software Reset
 // -------- TWI_MMR : (TWI Offset: 0x4) TWI Master Mode Register --------
 #define AT91C_TWI_IADRSZ      (0x3 <<  8) // (TWI) Internal Device Address Size
-#define 	AT91C_TWI_IADRSZ_NO                   (0x0 <<  8) // (TWI) No internal device address
-#define 	AT91C_TWI_IADRSZ_1_BYTE               (0x1 <<  8) // (TWI) One-byte internal device address
-#define 	AT91C_TWI_IADRSZ_2_BYTE               (0x2 <<  8) // (TWI) Two-byte internal device address
-#define 	AT91C_TWI_IADRSZ_3_BYTE               (0x3 <<  8) // (TWI) Three-byte internal device address
+#define     AT91C_TWI_IADRSZ_NO                   (0x0 <<  8) // (TWI) No internal device address
+#define     AT91C_TWI_IADRSZ_1_BYTE               (0x1 <<  8) // (TWI) One-byte internal device address
+#define     AT91C_TWI_IADRSZ_2_BYTE               (0x2 <<  8) // (TWI) Two-byte internal device address
+#define     AT91C_TWI_IADRSZ_3_BYTE               (0x3 <<  8) // (TWI) Three-byte internal device address
 #define AT91C_TWI_MREAD       (0x1 << 12) // (TWI) Master Read Direction
 #define AT91C_TWI_DADR        (0x7F << 16) // (TWI) Device Address
 // -------- TWI_CWGR : (TWI Offset: 0x10) TWI Clock Waveform Generator Register --------
@@ -1381,29 +1381,29 @@ typedef struct _AT91S_TWI {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_TC {
-    AT91_REG	 TC_CCR; 	// Channel Control Register
-    AT91_REG	 TC_CMR; 	// Channel Mode Register (Capture Mode / Waveform Mode)
-    AT91_REG	 Reserved0[2]; 	//
-    AT91_REG	 TC_CV; 	// Counter Value
-    AT91_REG	 TC_RA; 	// Register A
-    AT91_REG	 TC_RB; 	// Register B
-    AT91_REG	 TC_RC; 	// Register C
-    AT91_REG	 TC_SR; 	// Status Register
-    AT91_REG	 TC_IER; 	// Interrupt Enable Register
-    AT91_REG	 TC_IDR; 	// Interrupt Disable Register
-    AT91_REG	 TC_IMR; 	// Interrupt Mask Register
+    AT91_REG     TC_CCR;    // Channel Control Register
+    AT91_REG     TC_CMR;    // Channel Mode Register (Capture Mode / Waveform Mode)
+    AT91_REG     Reserved0[2];  //
+    AT91_REG     TC_CV;     // Counter Value
+    AT91_REG     TC_RA;     // Register A
+    AT91_REG     TC_RB;     // Register B
+    AT91_REG     TC_RC;     // Register C
+    AT91_REG     TC_SR;     // Status Register
+    AT91_REG     TC_IER;    // Interrupt Enable Register
+    AT91_REG     TC_IDR;    // Interrupt Disable Register
+    AT91_REG     TC_IMR;    // Interrupt Mask Register
 } AT91S_TC, *AT91PS_TC;
 #else
-#define TC_CCR          (AT91_CAST(AT91_REG *) 	0x00000000) // (TC_CCR) Channel Control Register
-#define TC_CMR          (AT91_CAST(AT91_REG *) 	0x00000004) // (TC_CMR) Channel Mode Register (Capture Mode / Waveform Mode)
-#define TC_CV           (AT91_CAST(AT91_REG *) 	0x00000010) // (TC_CV) Counter Value
-#define TC_RA           (AT91_CAST(AT91_REG *) 	0x00000014) // (TC_RA) Register A
-#define TC_RB           (AT91_CAST(AT91_REG *) 	0x00000018) // (TC_RB) Register B
-#define TC_RC           (AT91_CAST(AT91_REG *) 	0x0000001C) // (TC_RC) Register C
-#define TC_SR           (AT91_CAST(AT91_REG *) 	0x00000020) // (TC_SR) Status Register
-#define TC_IER          (AT91_CAST(AT91_REG *) 	0x00000024) // (TC_IER) Interrupt Enable Register
-#define TC_IDR          (AT91_CAST(AT91_REG *) 	0x00000028) // (TC_IDR) Interrupt Disable Register
-#define TC_IMR          (AT91_CAST(AT91_REG *) 	0x0000002C) // (TC_IMR) Interrupt Mask Register
+#define TC_CCR          (AT91_CAST(AT91_REG *)  0x00000000) // (TC_CCR) Channel Control Register
+#define TC_CMR          (AT91_CAST(AT91_REG *)  0x00000004) // (TC_CMR) Channel Mode Register (Capture Mode / Waveform Mode)
+#define TC_CV           (AT91_CAST(AT91_REG *)  0x00000010) // (TC_CV) Counter Value
+#define TC_RA           (AT91_CAST(AT91_REG *)  0x00000014) // (TC_RA) Register A
+#define TC_RB           (AT91_CAST(AT91_REG *)  0x00000018) // (TC_RB) Register B
+#define TC_RC           (AT91_CAST(AT91_REG *)  0x0000001C) // (TC_RC) Register C
+#define TC_SR           (AT91_CAST(AT91_REG *)  0x00000020) // (TC_SR) Status Register
+#define TC_IER          (AT91_CAST(AT91_REG *)  0x00000024) // (TC_IER) Interrupt Enable Register
+#define TC_IDR          (AT91_CAST(AT91_REG *)  0x00000028) // (TC_IDR) Interrupt Disable Register
+#define TC_IMR          (AT91_CAST(AT91_REG *)  0x0000002C) // (TC_IMR) Interrupt Mask Register
 
 #endif
 // -------- TC_CCR : (TC Offset: 0x0) TC Channel Control Register --------
@@ -1525,65 +1525,65 @@ typedef struct _AT91S_TC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_TCB {
-    AT91S_TC	 TCB_TC0; 	// TC Channel 0
-    AT91_REG	 Reserved0[4]; 	//
-    AT91S_TC	 TCB_TC1; 	// TC Channel 1
-    AT91_REG	 Reserved1[4]; 	//
-    AT91S_TC	 TCB_TC2; 	// TC Channel 2
-    AT91_REG	 Reserved2[4]; 	//
-    AT91_REG	 TCB_BCR; 	// TC Block Control Register
-    AT91_REG	 TCB_BMR; 	// TC Block Mode Register
+    AT91S_TC     TCB_TC0;   // TC Channel 0
+    AT91_REG     Reserved0[4];  //
+    AT91S_TC     TCB_TC1;   // TC Channel 1
+    AT91_REG     Reserved1[4];  //
+    AT91S_TC     TCB_TC2;   // TC Channel 2
+    AT91_REG     Reserved2[4];  //
+    AT91_REG     TCB_BCR;   // TC Block Control Register
+    AT91_REG     TCB_BMR;   // TC Block Mode Register
 } AT91S_TCB, *AT91PS_TCB;
 #else
-#define TCB_BCR         (AT91_CAST(AT91_REG *) 	0x000000C0) // (TCB_BCR) TC Block Control Register
-#define TCB_BMR         (AT91_CAST(AT91_REG *) 	0x000000C4) // (TCB_BMR) TC Block Mode Register
+#define TCB_BCR         (AT91_CAST(AT91_REG *)  0x000000C0) // (TCB_BCR) TC Block Control Register
+#define TCB_BMR         (AT91_CAST(AT91_REG *)  0x000000C4) // (TCB_BMR) TC Block Mode Register
 
 #endif
 // -------- TCB_BCR : (TCB Offset: 0xc0) TC Block Control Register --------
 #define AT91C_TCB_SYNC        (0x1 <<  0) // (TCB) Synchro Command
 // -------- TCB_BMR : (TCB Offset: 0xc4) TC Block Mode Register --------
 #define AT91C_TCB_TC0XC0S     (0x3 <<  0) // (TCB) External Clock Signal 0 Selection
-#define 	AT91C_TCB_TC0XC0S_TCLK0                (0x0) // (TCB) TCLK0 connected to XC0
-#define 	AT91C_TCB_TC0XC0S_NONE                 (0x1) // (TCB) None signal connected to XC0
-#define 	AT91C_TCB_TC0XC0S_TIOA1                (0x2) // (TCB) TIOA1 connected to XC0
-#define 	AT91C_TCB_TC0XC0S_TIOA2                (0x3) // (TCB) TIOA2 connected to XC0
+#define     AT91C_TCB_TC0XC0S_TCLK0                (0x0) // (TCB) TCLK0 connected to XC0
+#define     AT91C_TCB_TC0XC0S_NONE                 (0x1) // (TCB) None signal connected to XC0
+#define     AT91C_TCB_TC0XC0S_TIOA1                (0x2) // (TCB) TIOA1 connected to XC0
+#define     AT91C_TCB_TC0XC0S_TIOA2                (0x3) // (TCB) TIOA2 connected to XC0
 #define AT91C_TCB_TC1XC1S     (0x3 <<  2) // (TCB) External Clock Signal 1 Selection
-#define 	AT91C_TCB_TC1XC1S_TCLK1                (0x0 <<  2) // (TCB) TCLK1 connected to XC1
-#define 	AT91C_TCB_TC1XC1S_NONE                 (0x1 <<  2) // (TCB) None signal connected to XC1
-#define 	AT91C_TCB_TC1XC1S_TIOA0                (0x2 <<  2) // (TCB) TIOA0 connected to XC1
-#define 	AT91C_TCB_TC1XC1S_TIOA2                (0x3 <<  2) // (TCB) TIOA2 connected to XC1
+#define     AT91C_TCB_TC1XC1S_TCLK1                (0x0 <<  2) // (TCB) TCLK1 connected to XC1
+#define     AT91C_TCB_TC1XC1S_NONE                 (0x1 <<  2) // (TCB) None signal connected to XC1
+#define     AT91C_TCB_TC1XC1S_TIOA0                (0x2 <<  2) // (TCB) TIOA0 connected to XC1
+#define     AT91C_TCB_TC1XC1S_TIOA2                (0x3 <<  2) // (TCB) TIOA2 connected to XC1
 #define AT91C_TCB_TC2XC2S     (0x3 <<  4) // (TCB) External Clock Signal 2 Selection
-#define 	AT91C_TCB_TC2XC2S_TCLK2                (0x0 <<  4) // (TCB) TCLK2 connected to XC2
-#define 	AT91C_TCB_TC2XC2S_NONE                 (0x1 <<  4) // (TCB) None signal connected to XC2
-#define 	AT91C_TCB_TC2XC2S_TIOA0                (0x2 <<  4) // (TCB) TIOA0 connected to XC2
-#define 	AT91C_TCB_TC2XC2S_TIOA1                (0x3 <<  4) // (TCB) TIOA2 connected to XC2
+#define     AT91C_TCB_TC2XC2S_TCLK2                (0x0 <<  4) // (TCB) TCLK2 connected to XC2
+#define     AT91C_TCB_TC2XC2S_NONE                 (0x1 <<  4) // (TCB) None signal connected to XC2
+#define     AT91C_TCB_TC2XC2S_TIOA0                (0x2 <<  4) // (TCB) TIOA0 connected to XC2
+#define     AT91C_TCB_TC2XC2S_TIOA1                (0x3 <<  4) // (TCB) TIOA2 connected to XC2
 
 // *****************************************************************************
 //              SOFTWARE API DEFINITION  FOR PWMC Channel Interface
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_PWMC_CH {
-    AT91_REG	 PWMC_CMR; 		// Channel Mode Register
-    AT91_REG	 PWMC_CDTYR; 	// Channel Duty Cycle Register
-    AT91_REG	 PWMC_CPRDR; 	// Channel Period Register
-    AT91_REG	 PWMC_CCNTR; 	// Channel Counter Register
-    AT91_REG	 PWMC_CUPDR; 	// Channel Update Register
-    AT91_REG	 PWMC_Reserved[3]; 	// Reserved
+    AT91_REG     PWMC_CMR;      // Channel Mode Register
+    AT91_REG     PWMC_CDTYR;    // Channel Duty Cycle Register
+    AT91_REG     PWMC_CPRDR;    // Channel Period Register
+    AT91_REG     PWMC_CCNTR;    // Channel Counter Register
+    AT91_REG     PWMC_CUPDR;    // Channel Update Register
+    AT91_REG     PWMC_Reserved[3];  // Reserved
 } AT91S_PWMC_CH, *AT91PS_PWMC_CH;
 #else
-#define PWMC_CMR        (AT91_CAST(AT91_REG *) 	0x00000000) // (PWMC_CMR) Channel Mode Register
-#define PWMC_CDTYR      (AT91_CAST(AT91_REG *) 	0x00000004) // (PWMC_CDTYR) Channel Duty Cycle Register
-#define PWMC_CPRDR      (AT91_CAST(AT91_REG *) 	0x00000008) // (PWMC_CPRDR) Channel Period Register
-#define PWMC_CCNTR      (AT91_CAST(AT91_REG *) 	0x0000000C) // (PWMC_CCNTR) Channel Counter Register
-#define PWMC_CUPDR      (AT91_CAST(AT91_REG *) 	0x00000010) // (PWMC_CUPDR) Channel Update Register
-#define Reserved        (AT91_CAST(AT91_REG *) 	0x00000014) // (Reserved) Reserved
+#define PWMC_CMR        (AT91_CAST(AT91_REG *)  0x00000000) // (PWMC_CMR) Channel Mode Register
+#define PWMC_CDTYR      (AT91_CAST(AT91_REG *)  0x00000004) // (PWMC_CDTYR) Channel Duty Cycle Register
+#define PWMC_CPRDR      (AT91_CAST(AT91_REG *)  0x00000008) // (PWMC_CPRDR) Channel Period Register
+#define PWMC_CCNTR      (AT91_CAST(AT91_REG *)  0x0000000C) // (PWMC_CCNTR) Channel Counter Register
+#define PWMC_CUPDR      (AT91_CAST(AT91_REG *)  0x00000010) // (PWMC_CUPDR) Channel Update Register
+#define Reserved        (AT91_CAST(AT91_REG *)  0x00000014) // (Reserved) Reserved
 
 #endif
 // -------- PWMC_CMR : (PWMC_CH Offset: 0x0) PWMC Channel Mode Register --------
 #define AT91C_PWMC_CPRE       (0xF <<  0) // (PWMC_CH) Channel Pre-scaler : PWMC_CLKx
-#define 	AT91C_PWMC_CPRE_MCK                  (0x0) // (PWMC_CH)
-#define 	AT91C_PWMC_CPRE_MCKA                 (0xB) // (PWMC_CH)
-#define 	AT91C_PWMC_CPRE_MCKB                 (0xC) // (PWMC_CH)
+#define     AT91C_PWMC_CPRE_MCK                  (0x0) // (PWMC_CH)
+#define     AT91C_PWMC_CPRE_MCKA                 (0xB) // (PWMC_CH)
+#define     AT91C_PWMC_CPRE_MCKB                 (0xC) // (PWMC_CH)
 #define AT91C_PWMC_CALG       (0x1 <<  8) // (PWMC_CH) Channel Alignment
 #define AT91C_PWMC_CPOL       (0x1 <<  9) // (PWMC_CH) Channel Polarity
 #define AT91C_PWMC_CPD        (0x1 << 10) // (PWMC_CH) Channel Update Period
@@ -1601,38 +1601,38 @@ typedef struct _AT91S_PWMC_CH {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_PWMC {
-    AT91_REG	 PWMC_MR; 	// PWMC Mode Register
-    AT91_REG	 PWMC_ENA; 	// PWMC Enable Register
-    AT91_REG	 PWMC_DIS; 	// PWMC Disable Register
-    AT91_REG	 PWMC_SR; 	// PWMC Status Register
-    AT91_REG	 PWMC_IER; 	// PWMC Interrupt Enable Register
-    AT91_REG	 PWMC_IDR; 	// PWMC Interrupt Disable Register
-    AT91_REG	 PWMC_IMR; 	// PWMC Interrupt Mask Register
-    AT91_REG	 PWMC_ISR; 	// PWMC Interrupt Status Register
-    AT91_REG	 Reserved0[55]; 	//
-    AT91_REG	 PWMC_VR; 	// PWMC Version Register
-    AT91_REG	 Reserved1[64]; 	//
-    AT91S_PWMC_CH	 PWMC_CH[4]; 	// PWMC Channel
+    AT91_REG     PWMC_MR;   // PWMC Mode Register
+    AT91_REG     PWMC_ENA;  // PWMC Enable Register
+    AT91_REG     PWMC_DIS;  // PWMC Disable Register
+    AT91_REG     PWMC_SR;   // PWMC Status Register
+    AT91_REG     PWMC_IER;  // PWMC Interrupt Enable Register
+    AT91_REG     PWMC_IDR;  // PWMC Interrupt Disable Register
+    AT91_REG     PWMC_IMR;  // PWMC Interrupt Mask Register
+    AT91_REG     PWMC_ISR;  // PWMC Interrupt Status Register
+    AT91_REG     Reserved0[55];     //
+    AT91_REG     PWMC_VR;   // PWMC Version Register
+    AT91_REG     Reserved1[64];     //
+    AT91S_PWMC_CH    PWMC_CH[4];    // PWMC Channel
 } AT91S_PWMC, *AT91PS_PWMC;
 #else
-#define PWMC_MR         (AT91_CAST(AT91_REG *) 	0x00000000) // (PWMC_MR) PWMC Mode Register
-#define PWMC_ENA        (AT91_CAST(AT91_REG *) 	0x00000004) // (PWMC_ENA) PWMC Enable Register
-#define PWMC_DIS        (AT91_CAST(AT91_REG *) 	0x00000008) // (PWMC_DIS) PWMC Disable Register
-#define PWMC_SR         (AT91_CAST(AT91_REG *) 	0x0000000C) // (PWMC_SR) PWMC Status Register
-#define PWMC_IER        (AT91_CAST(AT91_REG *) 	0x00000010) // (PWMC_IER) PWMC Interrupt Enable Register
-#define PWMC_IDR        (AT91_CAST(AT91_REG *) 	0x00000014) // (PWMC_IDR) PWMC Interrupt Disable Register
-#define PWMC_IMR        (AT91_CAST(AT91_REG *) 	0x00000018) // (PWMC_IMR) PWMC Interrupt Mask Register
-#define PWMC_ISR        (AT91_CAST(AT91_REG *) 	0x0000001C) // (PWMC_ISR) PWMC Interrupt Status Register
-#define PWMC_VR         (AT91_CAST(AT91_REG *) 	0x000000FC) // (PWMC_VR) PWMC Version Register
+#define PWMC_MR         (AT91_CAST(AT91_REG *)  0x00000000) // (PWMC_MR) PWMC Mode Register
+#define PWMC_ENA        (AT91_CAST(AT91_REG *)  0x00000004) // (PWMC_ENA) PWMC Enable Register
+#define PWMC_DIS        (AT91_CAST(AT91_REG *)  0x00000008) // (PWMC_DIS) PWMC Disable Register
+#define PWMC_SR         (AT91_CAST(AT91_REG *)  0x0000000C) // (PWMC_SR) PWMC Status Register
+#define PWMC_IER        (AT91_CAST(AT91_REG *)  0x00000010) // (PWMC_IER) PWMC Interrupt Enable Register
+#define PWMC_IDR        (AT91_CAST(AT91_REG *)  0x00000014) // (PWMC_IDR) PWMC Interrupt Disable Register
+#define PWMC_IMR        (AT91_CAST(AT91_REG *)  0x00000018) // (PWMC_IMR) PWMC Interrupt Mask Register
+#define PWMC_ISR        (AT91_CAST(AT91_REG *)  0x0000001C) // (PWMC_ISR) PWMC Interrupt Status Register
+#define PWMC_VR         (AT91_CAST(AT91_REG *)  0x000000FC) // (PWMC_VR) PWMC Version Register
 
 #endif
 // -------- PWMC_MR : (PWMC Offset: 0x0) PWMC Mode Register --------
 #define AT91C_PWMC_DIVA       (0xFF <<  0) // (PWMC) CLKA divide factor.
 #define AT91C_PWMC_PREA       (0xF <<  8) // (PWMC) Divider Input Clock Prescaler A
-#define 	AT91C_PWMC_PREA_MCK                  (0x0 <<  8) // (PWMC)
+#define     AT91C_PWMC_PREA_MCK                  (0x0 <<  8) // (PWMC)
 #define AT91C_PWMC_DIVB       (0xFF << 16) // (PWMC) CLKB divide factor.
 #define AT91C_PWMC_PREB       (0xF << 24) // (PWMC) Divider Input Clock Prescaler B
-#define 	AT91C_PWMC_PREB_MCK                  (0x0 << 24) // (PWMC)
+#define     AT91C_PWMC_PREB_MCK                  (0x0 << 24) // (PWMC)
 // -------- PWMC_ENA : (PWMC Offset: 0x4) PWMC Enable Register --------
 #define AT91C_PWMC_CHID0      (0x1 <<  0) // (PWMC) Channel ID 0
 #define AT91C_PWMC_CHID1      (0x1 <<  1) // (PWMC) Channel ID 1
@@ -1650,37 +1650,37 @@ typedef struct _AT91S_PWMC {
 // *****************************************************************************
 #ifndef __ASSEMBLY__
 typedef struct _AT91S_UDP {
-    AT91_REG	 UDP_NUM; 	// Frame Number Register
-    AT91_REG	 UDP_GLBSTATE; 	// Global State Register
-    AT91_REG	 UDP_FADDR; 	// Function Address Register
-    AT91_REG	 Reserved0[1]; 	//
-    AT91_REG	 UDP_IER; 	// Interrupt Enable Register
-    AT91_REG	 UDP_IDR; 	// Interrupt Disable Register
-    AT91_REG	 UDP_IMR; 	// Interrupt Mask Register
-    AT91_REG	 UDP_ISR; 	// Interrupt Status Register
-    AT91_REG	 UDP_ICR; 	// Interrupt Clear Register
-    AT91_REG	 Reserved1[1]; 	//
-    AT91_REG	 UDP_RSTEP; 	// Reset Endpoint Register
-    AT91_REG	 Reserved2[1]; 	//
-    AT91_REG	 UDP_CSR[4]; 	// Endpoint Control and Status Register
-    AT91_REG	 Reserved3[4]; 	//
-    AT91_REG	 UDP_FDR[4]; 	// Endpoint FIFO Data Register
-    AT91_REG	 Reserved4[5]; 	//
-    AT91_REG	 UDP_TXVC; 	// Transceiver Control Register
+    AT91_REG     UDP_NUM;   // Frame Number Register
+    AT91_REG     UDP_GLBSTATE;  // Global State Register
+    AT91_REG     UDP_FADDR;     // Function Address Register
+    AT91_REG     Reserved0[1];  //
+    AT91_REG     UDP_IER;   // Interrupt Enable Register
+    AT91_REG     UDP_IDR;   // Interrupt Disable Register
+    AT91_REG     UDP_IMR;   // Interrupt Mask Register
+    AT91_REG     UDP_ISR;   // Interrupt Status Register
+    AT91_REG     UDP_ICR;   // Interrupt Clear Register
+    AT91_REG     Reserved1[1];  //
+    AT91_REG     UDP_RSTEP;     // Reset Endpoint Register
+    AT91_REG     Reserved2[1];  //
+    AT91_REG     UDP_CSR[4];    // Endpoint Control and Status Register
+    AT91_REG     Reserved3[4];  //
+    AT91_REG     UDP_FDR[4];    // Endpoint FIFO Data Register
+    AT91_REG     Reserved4[5];  //
+    AT91_REG     UDP_TXVC;  // Transceiver Control Register
 } AT91S_UDP, *AT91PS_UDP;
 #else
-#define UDP_FRM_NUM     (AT91_CAST(AT91_REG *) 	0x00000000) // (UDP_FRM_NUM) Frame Number Register
-#define UDP_GLBSTATE    (AT91_CAST(AT91_REG *) 	0x00000004) // (UDP_GLBSTATE) Global State Register
-#define UDP_FADDR       (AT91_CAST(AT91_REG *) 	0x00000008) // (UDP_FADDR) Function Address Register
-#define UDP_IER         (AT91_CAST(AT91_REG *) 	0x00000010) // (UDP_IER) Interrupt Enable Register
-#define UDP_IDR         (AT91_CAST(AT91_REG *) 	0x00000014) // (UDP_IDR) Interrupt Disable Register
-#define UDP_IMR         (AT91_CAST(AT91_REG *) 	0x00000018) // (UDP_IMR) Interrupt Mask Register
-#define UDP_ISR         (AT91_CAST(AT91_REG *) 	0x0000001C) // (UDP_ISR) Interrupt Status Register
-#define UDP_ICR         (AT91_CAST(AT91_REG *) 	0x00000020) // (UDP_ICR) Interrupt Clear Register
-#define UDP_RSTEP       (AT91_CAST(AT91_REG *) 	0x00000028) // (UDP_RSTEP) Reset Endpoint Register
-#define UDP_CSR         (AT91_CAST(AT91_REG *) 	0x00000030) // (UDP_CSR) Endpoint Control and Status Register
-#define UDP_FDR         (AT91_CAST(AT91_REG *) 	0x00000050) // (UDP_FDR) Endpoint FIFO Data Register
-#define UDP_TXVC        (AT91_CAST(AT91_REG *) 	0x00000074) // (UDP_TXVC) Transceiver Control Register
+#define UDP_FRM_NUM     (AT91_CAST(AT91_REG *)  0x00000000) // (UDP_FRM_NUM) Frame Number Register
+#define UDP_GLBSTATE    (AT91_CAST(AT91_REG *)  0x00000004) // (UDP_GLBSTATE) Global State Register
+#define UDP_FADDR       (AT91_CAST(AT91_REG *)  0x00000008) // (UDP_FADDR) Function Address Register
+#define UDP_IER         (AT91_CAST(AT91_REG *)  0x00000010) // (UDP_IER) Interrupt Enable Register
+#define UDP_IDR         (AT91_CAST(AT91_REG *)  0x00000014) // (UDP_IDR) Interrupt Disable Register
+#define UDP_IMR         (AT91_CAST(AT91_REG *)  0x00000018) // (UDP_IMR) Interrupt Mask Register
+#define UDP_ISR         (AT91_CAST(AT91_REG *)  0x0000001C) // (UDP_ISR) Interrupt Status Register
+#define UDP_ICR         (AT91_CAST(AT91_REG *)  0x00000020) // (UDP_ICR) Interrupt Clear Register
+#define UDP_RSTEP       (AT91_CAST(AT91_REG *)  0x00000028) // (UDP_RSTEP) Reset Endpoint Register
+#define UDP_CSR         (AT91_CAST(AT91_REG *)  0x00000030) // (UDP_CSR) Endpoint Control and Status Register
+#define UDP_FDR         (AT91_CAST(AT91_REG *)  0x00000050) // (UDP_FDR) Endpoint FIFO Data Register
+#define UDP_TXVC        (AT91_CAST(AT91_REG *)  0x00000074) // (UDP_TXVC) Transceiver Control Register
 
 #endif
 // -------- UDP_FRM_NUM : (UDP Offset: 0x0) USB Frame Number Register --------
@@ -1745,362 +1745,362 @@ typedef struct _AT91S_UDP {
 // *****************************************************************************
 // ========== Register definition for SYS peripheral ==========
 // ========== Register definition for AIC peripheral ==========
-#define AT91C_AIC_IVR   (AT91_CAST(AT91_REG *) 	0xFFFFF100) // (AIC) IRQ Vector Register
-#define AT91C_AIC_SMR   (AT91_CAST(AT91_REG *) 	0xFFFFF000) // (AIC) Source Mode Register
-#define AT91C_AIC_FVR   (AT91_CAST(AT91_REG *) 	0xFFFFF104) // (AIC) FIQ Vector Register
-#define AT91C_AIC_DCR   (AT91_CAST(AT91_REG *) 	0xFFFFF138) // (AIC) Debug Control Register (Protect)
-#define AT91C_AIC_EOICR (AT91_CAST(AT91_REG *) 	0xFFFFF130) // (AIC) End of Interrupt Command Register
-#define AT91C_AIC_SVR   (AT91_CAST(AT91_REG *) 	0xFFFFF080) // (AIC) Source Vector Register
-#define AT91C_AIC_FFSR  (AT91_CAST(AT91_REG *) 	0xFFFFF148) // (AIC) Fast Forcing Status Register
-#define AT91C_AIC_ICCR  (AT91_CAST(AT91_REG *) 	0xFFFFF128) // (AIC) Interrupt Clear Command Register
-#define AT91C_AIC_ISR   (AT91_CAST(AT91_REG *) 	0xFFFFF108) // (AIC) Interrupt Status Register
-#define AT91C_AIC_IMR   (AT91_CAST(AT91_REG *) 	0xFFFFF110) // (AIC) Interrupt Mask Register
-#define AT91C_AIC_IPR   (AT91_CAST(AT91_REG *) 	0xFFFFF10C) // (AIC) Interrupt Pending Register
-#define AT91C_AIC_FFER  (AT91_CAST(AT91_REG *) 	0xFFFFF140) // (AIC) Fast Forcing Enable Register
-#define AT91C_AIC_IECR  (AT91_CAST(AT91_REG *) 	0xFFFFF120) // (AIC) Interrupt Enable Command Register
-#define AT91C_AIC_ISCR  (AT91_CAST(AT91_REG *) 	0xFFFFF12C) // (AIC) Interrupt Set Command Register
-#define AT91C_AIC_FFDR  (AT91_CAST(AT91_REG *) 	0xFFFFF144) // (AIC) Fast Forcing Disable Register
-#define AT91C_AIC_CISR  (AT91_CAST(AT91_REG *) 	0xFFFFF114) // (AIC) Core Interrupt Status Register
-#define AT91C_AIC_IDCR  (AT91_CAST(AT91_REG *) 	0xFFFFF124) // (AIC) Interrupt Disable Command Register
-#define AT91C_AIC_SPU   (AT91_CAST(AT91_REG *) 	0xFFFFF134) // (AIC) Spurious Vector Register
+#define AT91C_AIC_IVR   (AT91_CAST(AT91_REG *)  0xFFFFF100) // (AIC) IRQ Vector Register
+#define AT91C_AIC_SMR   (AT91_CAST(AT91_REG *)  0xFFFFF000) // (AIC) Source Mode Register
+#define AT91C_AIC_FVR   (AT91_CAST(AT91_REG *)  0xFFFFF104) // (AIC) FIQ Vector Register
+#define AT91C_AIC_DCR   (AT91_CAST(AT91_REG *)  0xFFFFF138) // (AIC) Debug Control Register (Protect)
+#define AT91C_AIC_EOICR (AT91_CAST(AT91_REG *)  0xFFFFF130) // (AIC) End of Interrupt Command Register
+#define AT91C_AIC_SVR   (AT91_CAST(AT91_REG *)  0xFFFFF080) // (AIC) Source Vector Register
+#define AT91C_AIC_FFSR  (AT91_CAST(AT91_REG *)  0xFFFFF148) // (AIC) Fast Forcing Status Register
+#define AT91C_AIC_ICCR  (AT91_CAST(AT91_REG *)  0xFFFFF128) // (AIC) Interrupt Clear Command Register
+#define AT91C_AIC_ISR   (AT91_CAST(AT91_REG *)  0xFFFFF108) // (AIC) Interrupt Status Register
+#define AT91C_AIC_IMR   (AT91_CAST(AT91_REG *)  0xFFFFF110) // (AIC) Interrupt Mask Register
+#define AT91C_AIC_IPR   (AT91_CAST(AT91_REG *)  0xFFFFF10C) // (AIC) Interrupt Pending Register
+#define AT91C_AIC_FFER  (AT91_CAST(AT91_REG *)  0xFFFFF140) // (AIC) Fast Forcing Enable Register
+#define AT91C_AIC_IECR  (AT91_CAST(AT91_REG *)  0xFFFFF120) // (AIC) Interrupt Enable Command Register
+#define AT91C_AIC_ISCR  (AT91_CAST(AT91_REG *)  0xFFFFF12C) // (AIC) Interrupt Set Command Register
+#define AT91C_AIC_FFDR  (AT91_CAST(AT91_REG *)  0xFFFFF144) // (AIC) Fast Forcing Disable Register
+#define AT91C_AIC_CISR  (AT91_CAST(AT91_REG *)  0xFFFFF114) // (AIC) Core Interrupt Status Register
+#define AT91C_AIC_IDCR  (AT91_CAST(AT91_REG *)  0xFFFFF124) // (AIC) Interrupt Disable Command Register
+#define AT91C_AIC_SPU   (AT91_CAST(AT91_REG *)  0xFFFFF134) // (AIC) Spurious Vector Register
 // ========== Register definition for PDC_DBGU peripheral ==========
-#define AT91C_DBGU_TCR  (AT91_CAST(AT91_REG *) 	0xFFFFF30C) // (PDC_DBGU) Transmit Counter Register
-#define AT91C_DBGU_RNPR (AT91_CAST(AT91_REG *) 	0xFFFFF310) // (PDC_DBGU) Receive Next Pointer Register
-#define AT91C_DBGU_TNPR (AT91_CAST(AT91_REG *) 	0xFFFFF318) // (PDC_DBGU) Transmit Next Pointer Register
-#define AT91C_DBGU_TPR  (AT91_CAST(AT91_REG *) 	0xFFFFF308) // (PDC_DBGU) Transmit Pointer Register
-#define AT91C_DBGU_RPR  (AT91_CAST(AT91_REG *) 	0xFFFFF300) // (PDC_DBGU) Receive Pointer Register
-#define AT91C_DBGU_RCR  (AT91_CAST(AT91_REG *) 	0xFFFFF304) // (PDC_DBGU) Receive Counter Register
-#define AT91C_DBGU_RNCR (AT91_CAST(AT91_REG *) 	0xFFFFF314) // (PDC_DBGU) Receive Next Counter Register
-#define AT91C_DBGU_PTCR (AT91_CAST(AT91_REG *) 	0xFFFFF320) // (PDC_DBGU) PDC Transfer Control Register
-#define AT91C_DBGU_PTSR (AT91_CAST(AT91_REG *) 	0xFFFFF324) // (PDC_DBGU) PDC Transfer Status Register
-#define AT91C_DBGU_TNCR (AT91_CAST(AT91_REG *) 	0xFFFFF31C) // (PDC_DBGU) Transmit Next Counter Register
+#define AT91C_DBGU_TCR  (AT91_CAST(AT91_REG *)  0xFFFFF30C) // (PDC_DBGU) Transmit Counter Register
+#define AT91C_DBGU_RNPR (AT91_CAST(AT91_REG *)  0xFFFFF310) // (PDC_DBGU) Receive Next Pointer Register
+#define AT91C_DBGU_TNPR (AT91_CAST(AT91_REG *)  0xFFFFF318) // (PDC_DBGU) Transmit Next Pointer Register
+#define AT91C_DBGU_TPR  (AT91_CAST(AT91_REG *)  0xFFFFF308) // (PDC_DBGU) Transmit Pointer Register
+#define AT91C_DBGU_RPR  (AT91_CAST(AT91_REG *)  0xFFFFF300) // (PDC_DBGU) Receive Pointer Register
+#define AT91C_DBGU_RCR  (AT91_CAST(AT91_REG *)  0xFFFFF304) // (PDC_DBGU) Receive Counter Register
+#define AT91C_DBGU_RNCR (AT91_CAST(AT91_REG *)  0xFFFFF314) // (PDC_DBGU) Receive Next Counter Register
+#define AT91C_DBGU_PTCR (AT91_CAST(AT91_REG *)  0xFFFFF320) // (PDC_DBGU) PDC Transfer Control Register
+#define AT91C_DBGU_PTSR (AT91_CAST(AT91_REG *)  0xFFFFF324) // (PDC_DBGU) PDC Transfer Status Register
+#define AT91C_DBGU_TNCR (AT91_CAST(AT91_REG *)  0xFFFFF31C) // (PDC_DBGU) Transmit Next Counter Register
 // ========== Register definition for DBGU peripheral ==========
-#define AT91C_DBGU_EXID (AT91_CAST(AT91_REG *) 	0xFFFFF244) // (DBGU) Chip ID Extension Register
-#define AT91C_DBGU_BRGR (AT91_CAST(AT91_REG *) 	0xFFFFF220) // (DBGU) Baud Rate Generator Register
-#define AT91C_DBGU_IDR  (AT91_CAST(AT91_REG *) 	0xFFFFF20C) // (DBGU) Interrupt Disable Register
-#define AT91C_DBGU_CSR  (AT91_CAST(AT91_REG *) 	0xFFFFF214) // (DBGU) Channel Status Register
-#define AT91C_DBGU_CIDR (AT91_CAST(AT91_REG *) 	0xFFFFF240) // (DBGU) Chip ID Register
-#define AT91C_DBGU_MR   (AT91_CAST(AT91_REG *) 	0xFFFFF204) // (DBGU) Mode Register
-#define AT91C_DBGU_IMR  (AT91_CAST(AT91_REG *) 	0xFFFFF210) // (DBGU) Interrupt Mask Register
-#define AT91C_DBGU_CR   (AT91_CAST(AT91_REG *) 	0xFFFFF200) // (DBGU) Control Register
-#define AT91C_DBGU_FNTR (AT91_CAST(AT91_REG *) 	0xFFFFF248) // (DBGU) Force NTRST Register
-#define AT91C_DBGU_THR  (AT91_CAST(AT91_REG *) 	0xFFFFF21C) // (DBGU) Transmitter Holding Register
-#define AT91C_DBGU_RHR  (AT91_CAST(AT91_REG *) 	0xFFFFF218) // (DBGU) Receiver Holding Register
-#define AT91C_DBGU_IER  (AT91_CAST(AT91_REG *) 	0xFFFFF208) // (DBGU) Interrupt Enable Register
+#define AT91C_DBGU_EXID (AT91_CAST(AT91_REG *)  0xFFFFF244) // (DBGU) Chip ID Extension Register
+#define AT91C_DBGU_BRGR (AT91_CAST(AT91_REG *)  0xFFFFF220) // (DBGU) Baud Rate Generator Register
+#define AT91C_DBGU_IDR  (AT91_CAST(AT91_REG *)  0xFFFFF20C) // (DBGU) Interrupt Disable Register
+#define AT91C_DBGU_CSR  (AT91_CAST(AT91_REG *)  0xFFFFF214) // (DBGU) Channel Status Register
+#define AT91C_DBGU_CIDR (AT91_CAST(AT91_REG *)  0xFFFFF240) // (DBGU) Chip ID Register
+#define AT91C_DBGU_MR   (AT91_CAST(AT91_REG *)  0xFFFFF204) // (DBGU) Mode Register
+#define AT91C_DBGU_IMR  (AT91_CAST(AT91_REG *)  0xFFFFF210) // (DBGU) Interrupt Mask Register
+#define AT91C_DBGU_CR   (AT91_CAST(AT91_REG *)  0xFFFFF200) // (DBGU) Control Register
+#define AT91C_DBGU_FNTR (AT91_CAST(AT91_REG *)  0xFFFFF248) // (DBGU) Force NTRST Register
+#define AT91C_DBGU_THR  (AT91_CAST(AT91_REG *)  0xFFFFF21C) // (DBGU) Transmitter Holding Register
+#define AT91C_DBGU_RHR  (AT91_CAST(AT91_REG *)  0xFFFFF218) // (DBGU) Receiver Holding Register
+#define AT91C_DBGU_IER  (AT91_CAST(AT91_REG *)  0xFFFFF208) // (DBGU) Interrupt Enable Register
 // ========== Register definition for PIOA peripheral ==========
-#define AT91C_PIOA_ODR  (AT91_CAST(AT91_REG *) 	0xFFFFF414) // (PIOA) Output Disable Registerr
-#define AT91C_PIOA_SODR (AT91_CAST(AT91_REG *) 	0xFFFFF430) // (PIOA) Set Output Data Register
-#define AT91C_PIOA_ISR  (AT91_CAST(AT91_REG *) 	0xFFFFF44C) // (PIOA) Interrupt Status Register
-#define AT91C_PIOA_ABSR (AT91_CAST(AT91_REG *) 	0xFFFFF478) // (PIOA) AB Select Status Register
-#define AT91C_PIOA_IER  (AT91_CAST(AT91_REG *) 	0xFFFFF440) // (PIOA) Interrupt Enable Register
+#define AT91C_PIOA_ODR  (AT91_CAST(AT91_REG *)  0xFFFFF414) // (PIOA) Output Disable Registerr
+#define AT91C_PIOA_SODR (AT91_CAST(AT91_REG *)  0xFFFFF430) // (PIOA) Set Output Data Register
+#define AT91C_PIOA_ISR  (AT91_CAST(AT91_REG *)  0xFFFFF44C) // (PIOA) Interrupt Status Register
+#define AT91C_PIOA_ABSR (AT91_CAST(AT91_REG *)  0xFFFFF478) // (PIOA) AB Select Status Register
+#define AT91C_PIOA_IER  (AT91_CAST(AT91_REG *)  0xFFFFF440) // (PIOA) Interrupt Enable Register
 #define AT91C_PIOA_PPUDR (AT91_CAST(AT91_REG *) 0xFFFFF460) // (PIOA) Pull-up Disable Register
-#define AT91C_PIOA_IMR  (AT91_CAST(AT91_REG *) 	0xFFFFF448) // (PIOA) Interrupt Mask Register
-#define AT91C_PIOA_PER  (AT91_CAST(AT91_REG *) 	0xFFFFF400) // (PIOA) PIO Enable Register
-#define AT91C_PIOA_IFDR (AT91_CAST(AT91_REG *) 	0xFFFFF424) // (PIOA) Input Filter Disable Register
-#define AT91C_PIOA_OWDR (AT91_CAST(AT91_REG *) 	0xFFFFF4A4) // (PIOA) Output Write Disable Register
-#define AT91C_PIOA_MDSR (AT91_CAST(AT91_REG *) 	0xFFFFF458) // (PIOA) Multi-driver Status Register
-#define AT91C_PIOA_IDR  (AT91_CAST(AT91_REG *) 	0xFFFFF444) // (PIOA) Interrupt Disable Register
-#define AT91C_PIOA_ODSR (AT91_CAST(AT91_REG *) 	0xFFFFF438) // (PIOA) Output Data Status Register
-#define AT91C_PIOA_PPUSR (AT91_CAST(AT91_REG *)	0xFFFFF468) // (PIOA) Pull-up Status Register
-#define AT91C_PIOA_OWSR (AT91_CAST(AT91_REG *) 	0xFFFFF4A8) // (PIOA) Output Write Status Register
-#define AT91C_PIOA_BSR  (AT91_CAST(AT91_REG *) 	0xFFFFF474) // (PIOA) Select B Register
-#define AT91C_PIOA_OWER (AT91_CAST(AT91_REG *) 	0xFFFFF4A0) // (PIOA) Output Write Enable Register
-#define AT91C_PIOA_IFER (AT91_CAST(AT91_REG *) 	0xFFFFF420) // (PIOA) Input Filter Enable Register
-#define AT91C_PIOA_PDSR (AT91_CAST(AT91_REG *) 	0xFFFFF43C) // (PIOA) Pin Data Status Register
-#define AT91C_PIOA_PPUER (AT91_CAST(AT91_REG *)	0xFFFFF464) // (PIOA) Pull-up Enable Register
-#define AT91C_PIOA_OSR  (AT91_CAST(AT91_REG *) 	0xFFFFF418) // (PIOA) Output Status Register
-#define AT91C_PIOA_ASR  (AT91_CAST(AT91_REG *) 	0xFFFFF470) // (PIOA) Select A Register
-#define AT91C_PIOA_MDDR (AT91_CAST(AT91_REG *) 	0xFFFFF454) // (PIOA) Multi-driver Disable Register
-#define AT91C_PIOA_CODR (AT91_CAST(AT91_REG *) 	0xFFFFF434) // (PIOA) Clear Output Data Register
-#define AT91C_PIOA_MDER (AT91_CAST(AT91_REG *) 	0xFFFFF450) // (PIOA) Multi-driver Enable Register
-#define AT91C_PIOA_PDR  (AT91_CAST(AT91_REG *) 	0xFFFFF404) // (PIOA) PIO Disable Register
-#define AT91C_PIOA_IFSR (AT91_CAST(AT91_REG *) 	0xFFFFF428) // (PIOA) Input Filter Status Register
-#define AT91C_PIOA_OER  (AT91_CAST(AT91_REG *) 	0xFFFFF410) // (PIOA) Output Enable Register
-#define AT91C_PIOA_PSR  (AT91_CAST(AT91_REG *) 	0xFFFFF408) // (PIOA) PIO Status Register
+#define AT91C_PIOA_IMR  (AT91_CAST(AT91_REG *)  0xFFFFF448) // (PIOA) Interrupt Mask Register
+#define AT91C_PIOA_PER  (AT91_CAST(AT91_REG *)  0xFFFFF400) // (PIOA) PIO Enable Register
+#define AT91C_PIOA_IFDR (AT91_CAST(AT91_REG *)  0xFFFFF424) // (PIOA) Input Filter Disable Register
+#define AT91C_PIOA_OWDR (AT91_CAST(AT91_REG *)  0xFFFFF4A4) // (PIOA) Output Write Disable Register
+#define AT91C_PIOA_MDSR (AT91_CAST(AT91_REG *)  0xFFFFF458) // (PIOA) Multi-driver Status Register
+#define AT91C_PIOA_IDR  (AT91_CAST(AT91_REG *)  0xFFFFF444) // (PIOA) Interrupt Disable Register
+#define AT91C_PIOA_ODSR (AT91_CAST(AT91_REG *)  0xFFFFF438) // (PIOA) Output Data Status Register
+#define AT91C_PIOA_PPUSR (AT91_CAST(AT91_REG *) 0xFFFFF468) // (PIOA) Pull-up Status Register
+#define AT91C_PIOA_OWSR (AT91_CAST(AT91_REG *)  0xFFFFF4A8) // (PIOA) Output Write Status Register
+#define AT91C_PIOA_BSR  (AT91_CAST(AT91_REG *)  0xFFFFF474) // (PIOA) Select B Register
+#define AT91C_PIOA_OWER (AT91_CAST(AT91_REG *)  0xFFFFF4A0) // (PIOA) Output Write Enable Register
+#define AT91C_PIOA_IFER (AT91_CAST(AT91_REG *)  0xFFFFF420) // (PIOA) Input Filter Enable Register
+#define AT91C_PIOA_PDSR (AT91_CAST(AT91_REG *)  0xFFFFF43C) // (PIOA) Pin Data Status Register
+#define AT91C_PIOA_PPUER (AT91_CAST(AT91_REG *) 0xFFFFF464) // (PIOA) Pull-up Enable Register
+#define AT91C_PIOA_OSR  (AT91_CAST(AT91_REG *)  0xFFFFF418) // (PIOA) Output Status Register
+#define AT91C_PIOA_ASR  (AT91_CAST(AT91_REG *)  0xFFFFF470) // (PIOA) Select A Register
+#define AT91C_PIOA_MDDR (AT91_CAST(AT91_REG *)  0xFFFFF454) // (PIOA) Multi-driver Disable Register
+#define AT91C_PIOA_CODR (AT91_CAST(AT91_REG *)  0xFFFFF434) // (PIOA) Clear Output Data Register
+#define AT91C_PIOA_MDER (AT91_CAST(AT91_REG *)  0xFFFFF450) // (PIOA) Multi-driver Enable Register
+#define AT91C_PIOA_PDR  (AT91_CAST(AT91_REG *)  0xFFFFF404) // (PIOA) PIO Disable Register
+#define AT91C_PIOA_IFSR (AT91_CAST(AT91_REG *)  0xFFFFF428) // (PIOA) Input Filter Status Register
+#define AT91C_PIOA_OER  (AT91_CAST(AT91_REG *)  0xFFFFF410) // (PIOA) Output Enable Register
+#define AT91C_PIOA_PSR  (AT91_CAST(AT91_REG *)  0xFFFFF408) // (PIOA) PIO Status Register
 // ========== Register definition for CKGR peripheral ==========
-#define AT91C_CKGR_MOR  (AT91_CAST(AT91_REG *) 	0xFFFFFC20) // (CKGR) Main Oscillator Register
-#define AT91C_CKGR_PLLR (AT91_CAST(AT91_REG *) 	0xFFFFFC2C) // (CKGR) PLL Register
-#define AT91C_CKGR_MCFR (AT91_CAST(AT91_REG *) 	0xFFFFFC24) // (CKGR) Main Clock  Frequency Register
+#define AT91C_CKGR_MOR  (AT91_CAST(AT91_REG *)  0xFFFFFC20) // (CKGR) Main Oscillator Register
+#define AT91C_CKGR_PLLR (AT91_CAST(AT91_REG *)  0xFFFFFC2C) // (CKGR) PLL Register
+#define AT91C_CKGR_MCFR (AT91_CAST(AT91_REG *)  0xFFFFFC24) // (CKGR) Main Clock  Frequency Register
 // ========== Register definition for PMC peripheral ==========
-#define AT91C_PMC_IDR   (AT91_CAST(AT91_REG *) 	0xFFFFFC64) // (PMC) Interrupt Disable Register
-#define AT91C_PMC_MOR   (AT91_CAST(AT91_REG *) 	0xFFFFFC20) // (PMC) Main Oscillator Register
-#define AT91C_PMC_PLLR  (AT91_CAST(AT91_REG *) 	0xFFFFFC2C) // (PMC) PLL Register
-#define AT91C_PMC_PCER  (AT91_CAST(AT91_REG *) 	0xFFFFFC10) // (PMC) Peripheral Clock Enable Register
-#define AT91C_PMC_PCKR  (AT91_CAST(AT91_REG *) 	0xFFFFFC40) // (PMC) Programmable Clock Register
-#define AT91C_PMC_MCKR  (AT91_CAST(AT91_REG *) 	0xFFFFFC30) // (PMC) Master Clock Register
-#define AT91C_PMC_SCDR  (AT91_CAST(AT91_REG *) 	0xFFFFFC04) // (PMC) System Clock Disable Register
-#define AT91C_PMC_PCDR  (AT91_CAST(AT91_REG *) 	0xFFFFFC14) // (PMC) Peripheral Clock Disable Register
-#define AT91C_PMC_SCSR  (AT91_CAST(AT91_REG *) 	0xFFFFFC08) // (PMC) System Clock Status Register
-#define AT91C_PMC_PCSR  (AT91_CAST(AT91_REG *) 	0xFFFFFC18) // (PMC) Peripheral Clock Status Register
-#define AT91C_PMC_MCFR  (AT91_CAST(AT91_REG *) 	0xFFFFFC24) // (PMC) Main Clock  Frequency Register
-#define AT91C_PMC_SCER  (AT91_CAST(AT91_REG *) 	0xFFFFFC00) // (PMC) System Clock Enable Register
-#define AT91C_PMC_IMR   (AT91_CAST(AT91_REG *) 	0xFFFFFC6C) // (PMC) Interrupt Mask Register
-#define AT91C_PMC_IER   (AT91_CAST(AT91_REG *) 	0xFFFFFC60) // (PMC) Interrupt Enable Register
-#define AT91C_PMC_SR    (AT91_CAST(AT91_REG *) 	0xFFFFFC68) // (PMC) Status Register
+#define AT91C_PMC_IDR   (AT91_CAST(AT91_REG *)  0xFFFFFC64) // (PMC) Interrupt Disable Register
+#define AT91C_PMC_MOR   (AT91_CAST(AT91_REG *)  0xFFFFFC20) // (PMC) Main Oscillator Register
+#define AT91C_PMC_PLLR  (AT91_CAST(AT91_REG *)  0xFFFFFC2C) // (PMC) PLL Register
+#define AT91C_PMC_PCER  (AT91_CAST(AT91_REG *)  0xFFFFFC10) // (PMC) Peripheral Clock Enable Register
+#define AT91C_PMC_PCKR  (AT91_CAST(AT91_REG *)  0xFFFFFC40) // (PMC) Programmable Clock Register
+#define AT91C_PMC_MCKR  (AT91_CAST(AT91_REG *)  0xFFFFFC30) // (PMC) Master Clock Register
+#define AT91C_PMC_SCDR  (AT91_CAST(AT91_REG *)  0xFFFFFC04) // (PMC) System Clock Disable Register
+#define AT91C_PMC_PCDR  (AT91_CAST(AT91_REG *)  0xFFFFFC14) // (PMC) Peripheral Clock Disable Register
+#define AT91C_PMC_SCSR  (AT91_CAST(AT91_REG *)  0xFFFFFC08) // (PMC) System Clock Status Register
+#define AT91C_PMC_PCSR  (AT91_CAST(AT91_REG *)  0xFFFFFC18) // (PMC) Peripheral Clock Status Register
+#define AT91C_PMC_MCFR  (AT91_CAST(AT91_REG *)  0xFFFFFC24) // (PMC) Main Clock  Frequency Register
+#define AT91C_PMC_SCER  (AT91_CAST(AT91_REG *)  0xFFFFFC00) // (PMC) System Clock Enable Register
+#define AT91C_PMC_IMR   (AT91_CAST(AT91_REG *)  0xFFFFFC6C) // (PMC) Interrupt Mask Register
+#define AT91C_PMC_IER   (AT91_CAST(AT91_REG *)  0xFFFFFC60) // (PMC) Interrupt Enable Register
+#define AT91C_PMC_SR    (AT91_CAST(AT91_REG *)  0xFFFFFC68) // (PMC) Status Register
 // ========== Register definition for RSTC peripheral ==========
-#define AT91C_RSTC_RCR  (AT91_CAST(AT91_REG *) 	0xFFFFFD00) // (RSTC) Reset Control Register
-#define AT91C_RSTC_RMR  (AT91_CAST(AT91_REG *) 	0xFFFFFD08) // (RSTC) Reset Mode Register
-#define AT91C_RSTC_RSR  (AT91_CAST(AT91_REG *) 	0xFFFFFD04) // (RSTC) Reset Status Register
+#define AT91C_RSTC_RCR  (AT91_CAST(AT91_REG *)  0xFFFFFD00) // (RSTC) Reset Control Register
+#define AT91C_RSTC_RMR  (AT91_CAST(AT91_REG *)  0xFFFFFD08) // (RSTC) Reset Mode Register
+#define AT91C_RSTC_RSR  (AT91_CAST(AT91_REG *)  0xFFFFFD04) // (RSTC) Reset Status Register
 // ========== Register definition for RTTC peripheral ==========
-#define AT91C_RTTC_RTSR (AT91_CAST(AT91_REG *) 	0xFFFFFD2C) // (RTTC) Real-time Status Register
-#define AT91C_RTTC_RTMR (AT91_CAST(AT91_REG *) 	0xFFFFFD20) // (RTTC) Real-time Mode Register
-#define AT91C_RTTC_RTVR (AT91_CAST(AT91_REG *) 	0xFFFFFD28) // (RTTC) Real-time Value Register
-#define AT91C_RTTC_RTAR (AT91_CAST(AT91_REG *) 	0xFFFFFD24) // (RTTC) Real-time Alarm Register
+#define AT91C_RTTC_RTSR (AT91_CAST(AT91_REG *)  0xFFFFFD2C) // (RTTC) Real-time Status Register
+#define AT91C_RTTC_RTMR (AT91_CAST(AT91_REG *)  0xFFFFFD20) // (RTTC) Real-time Mode Register
+#define AT91C_RTTC_RTVR (AT91_CAST(AT91_REG *)  0xFFFFFD28) // (RTTC) Real-time Value Register
+#define AT91C_RTTC_RTAR (AT91_CAST(AT91_REG *)  0xFFFFFD24) // (RTTC) Real-time Alarm Register
 // ========== Register definition for PITC peripheral ==========
-#define AT91C_PITC_PIVR (AT91_CAST(AT91_REG *) 	0xFFFFFD38) // (PITC) Period Interval Value Register
-#define AT91C_PITC_PISR (AT91_CAST(AT91_REG *) 	0xFFFFFD34) // (PITC) Period Interval Status Register
-#define AT91C_PITC_PIIR (AT91_CAST(AT91_REG *) 	0xFFFFFD3C) // (PITC) Period Interval Image Register
-#define AT91C_PITC_PIMR (AT91_CAST(AT91_REG *) 	0xFFFFFD30) // (PITC) Period Interval Mode Register
+#define AT91C_PITC_PIVR (AT91_CAST(AT91_REG *)  0xFFFFFD38) // (PITC) Period Interval Value Register
+#define AT91C_PITC_PISR (AT91_CAST(AT91_REG *)  0xFFFFFD34) // (PITC) Period Interval Status Register
+#define AT91C_PITC_PIIR (AT91_CAST(AT91_REG *)  0xFFFFFD3C) // (PITC) Period Interval Image Register
+#define AT91C_PITC_PIMR (AT91_CAST(AT91_REG *)  0xFFFFFD30) // (PITC) Period Interval Mode Register
 // ========== Register definition for WDTC peripheral ==========
-#define AT91C_WDTC_WDCR (AT91_CAST(AT91_REG *) 	0xFFFFFD40) // (WDTC) Watchdog Control Register
-#define AT91C_WDTC_WDSR (AT91_CAST(AT91_REG *) 	0xFFFFFD48) // (WDTC) Watchdog Status Register
-#define AT91C_WDTC_WDMR (AT91_CAST(AT91_REG *) 	0xFFFFFD44) // (WDTC) Watchdog Mode Register
+#define AT91C_WDTC_WDCR (AT91_CAST(AT91_REG *)  0xFFFFFD40) // (WDTC) Watchdog Control Register
+#define AT91C_WDTC_WDSR (AT91_CAST(AT91_REG *)  0xFFFFFD48) // (WDTC) Watchdog Status Register
+#define AT91C_WDTC_WDMR (AT91_CAST(AT91_REG *)  0xFFFFFD44) // (WDTC) Watchdog Mode Register
 // ========== Register definition for VREG peripheral ==========
-#define AT91C_VREG_MR   (AT91_CAST(AT91_REG *) 	0xFFFFFD60) // (VREG) Voltage Regulator Mode Register
+#define AT91C_VREG_MR   (AT91_CAST(AT91_REG *)  0xFFFFFD60) // (VREG) Voltage Regulator Mode Register
 // ========== Register definition for EFC0 peripheral ==========
-#define AT91C_EFC0_FCR  (AT91_CAST(AT91_REG *) 	0xFFFFFF64) // (EFC0) MC Flash Command Register
-#define AT91C_EFC0_FSR  (AT91_CAST(AT91_REG *) 	0xFFFFFF68) // (EFC0) MC Flash Status Register
-#define AT91C_EFC0_VR   (AT91_CAST(AT91_REG *) 	0xFFFFFF6C) // (EFC0) MC Flash Version Register
-#define AT91C_EFC0_FMR  (AT91_CAST(AT91_REG *) 	0xFFFFFF60) // (EFC0) MC Flash Mode Register
+#define AT91C_EFC0_FCR  (AT91_CAST(AT91_REG *)  0xFFFFFF64) // (EFC0) MC Flash Command Register
+#define AT91C_EFC0_FSR  (AT91_CAST(AT91_REG *)  0xFFFFFF68) // (EFC0) MC Flash Status Register
+#define AT91C_EFC0_VR   (AT91_CAST(AT91_REG *)  0xFFFFFF6C) // (EFC0) MC Flash Version Register
+#define AT91C_EFC0_FMR  (AT91_CAST(AT91_REG *)  0xFFFFFF60) // (EFC0) MC Flash Mode Register
 // ========== Register definition for EFC1 peripheral ==========
-#define AT91C_EFC1_VR   (AT91_CAST(AT91_REG *) 	0xFFFFFF7C) // (EFC1) MC Flash Version Register
-#define AT91C_EFC1_FCR  (AT91_CAST(AT91_REG *) 	0xFFFFFF74) // (EFC1) MC Flash Command Register
-#define AT91C_EFC1_FSR  (AT91_CAST(AT91_REG *) 	0xFFFFFF78) // (EFC1) MC Flash Status Register
-#define AT91C_EFC1_FMR  (AT91_CAST(AT91_REG *) 	0xFFFFFF70) // (EFC1) MC Flash Mode Register
+#define AT91C_EFC1_VR   (AT91_CAST(AT91_REG *)  0xFFFFFF7C) // (EFC1) MC Flash Version Register
+#define AT91C_EFC1_FCR  (AT91_CAST(AT91_REG *)  0xFFFFFF74) // (EFC1) MC Flash Command Register
+#define AT91C_EFC1_FSR  (AT91_CAST(AT91_REG *)  0xFFFFFF78) // (EFC1) MC Flash Status Register
+#define AT91C_EFC1_FMR  (AT91_CAST(AT91_REG *)  0xFFFFFF70) // (EFC1) MC Flash Mode Register
 // ========== Register definition for MC peripheral ==========
-#define AT91C_MC_ASR    (AT91_CAST(AT91_REG *) 	0xFFFFFF04) // (MC) MC Abort Status Register
-#define AT91C_MC_RCR    (AT91_CAST(AT91_REG *) 	0xFFFFFF00) // (MC) MC Remap Control Register
-#define AT91C_MC_PUP    (AT91_CAST(AT91_REG *) 	0xFFFFFF50) // (MC) MC Protection Unit Peripherals
-#define AT91C_MC_PUIA   (AT91_CAST(AT91_REG *) 	0xFFFFFF10) // (MC) MC Protection Unit Area
-#define AT91C_MC_AASR   (AT91_CAST(AT91_REG *) 	0xFFFFFF08) // (MC) MC Abort Address Status Register
-#define AT91C_MC_PUER   (AT91_CAST(AT91_REG *) 	0xFFFFFF54) // (MC) MC Protection Unit Enable Register
+#define AT91C_MC_ASR    (AT91_CAST(AT91_REG *)  0xFFFFFF04) // (MC) MC Abort Status Register
+#define AT91C_MC_RCR    (AT91_CAST(AT91_REG *)  0xFFFFFF00) // (MC) MC Remap Control Register
+#define AT91C_MC_PUP    (AT91_CAST(AT91_REG *)  0xFFFFFF50) // (MC) MC Protection Unit Peripherals
+#define AT91C_MC_PUIA   (AT91_CAST(AT91_REG *)  0xFFFFFF10) // (MC) MC Protection Unit Area
+#define AT91C_MC_AASR   (AT91_CAST(AT91_REG *)  0xFFFFFF08) // (MC) MC Abort Address Status Register
+#define AT91C_MC_PUER   (AT91_CAST(AT91_REG *)  0xFFFFFF54) // (MC) MC Protection Unit Enable Register
 // ========== Register definition for PDC_SPI peripheral ==========
-#define AT91C_SPI_PTCR  (AT91_CAST(AT91_REG *) 	0xFFFE0120) // (PDC_SPI) PDC Transfer Control Register
-#define AT91C_SPI_TPR   (AT91_CAST(AT91_REG *) 	0xFFFE0108) // (PDC_SPI) Transmit Pointer Register
-#define AT91C_SPI_TCR   (AT91_CAST(AT91_REG *) 	0xFFFE010C) // (PDC_SPI) Transmit Counter Register
-#define AT91C_SPI_RCR   (AT91_CAST(AT91_REG *) 	0xFFFE0104) // (PDC_SPI) Receive Counter Register
-#define AT91C_SPI_PTSR  (AT91_CAST(AT91_REG *) 	0xFFFE0124) // (PDC_SPI) PDC Transfer Status Register
-#define AT91C_SPI_RNPR  (AT91_CAST(AT91_REG *) 	0xFFFE0110) // (PDC_SPI) Receive Next Pointer Register
-#define AT91C_SPI_RPR   (AT91_CAST(AT91_REG *) 	0xFFFE0100) // (PDC_SPI) Receive Pointer Register
-#define AT91C_SPI_TNCR  (AT91_CAST(AT91_REG *) 	0xFFFE011C) // (PDC_SPI) Transmit Next Counter Register
-#define AT91C_SPI_RNCR  (AT91_CAST(AT91_REG *) 	0xFFFE0114) // (PDC_SPI) Receive Next Counter Register
-#define AT91C_SPI_TNPR  (AT91_CAST(AT91_REG *) 	0xFFFE0118) // (PDC_SPI) Transmit Next Pointer Register
+#define AT91C_SPI_PTCR  (AT91_CAST(AT91_REG *)  0xFFFE0120) // (PDC_SPI) PDC Transfer Control Register
+#define AT91C_SPI_TPR   (AT91_CAST(AT91_REG *)  0xFFFE0108) // (PDC_SPI) Transmit Pointer Register
+#define AT91C_SPI_TCR   (AT91_CAST(AT91_REG *)  0xFFFE010C) // (PDC_SPI) Transmit Counter Register
+#define AT91C_SPI_RCR   (AT91_CAST(AT91_REG *)  0xFFFE0104) // (PDC_SPI) Receive Counter Register
+#define AT91C_SPI_PTSR  (AT91_CAST(AT91_REG *)  0xFFFE0124) // (PDC_SPI) PDC Transfer Status Register
+#define AT91C_SPI_RNPR  (AT91_CAST(AT91_REG *)  0xFFFE0110) // (PDC_SPI) Receive Next Pointer Register
+#define AT91C_SPI_RPR   (AT91_CAST(AT91_REG *)  0xFFFE0100) // (PDC_SPI) Receive Pointer Register
+#define AT91C_SPI_TNCR  (AT91_CAST(AT91_REG *)  0xFFFE011C) // (PDC_SPI) Transmit Next Counter Register
+#define AT91C_SPI_RNCR  (AT91_CAST(AT91_REG *)  0xFFFE0114) // (PDC_SPI) Receive Next Counter Register
+#define AT91C_SPI_TNPR  (AT91_CAST(AT91_REG *)  0xFFFE0118) // (PDC_SPI) Transmit Next Pointer Register
 // ========== Register definition for SPI peripheral ==========
-#define AT91C_SPI_IER   (AT91_CAST(AT91_REG *) 	0xFFFE0014) // (SPI) Interrupt Enable Register
-#define AT91C_SPI_SR    (AT91_CAST(AT91_REG *) 	0xFFFE0010) // (SPI) Status Register
-#define AT91C_SPI_IDR   (AT91_CAST(AT91_REG *) 	0xFFFE0018) // (SPI) Interrupt Disable Register
-#define AT91C_SPI_CR    (AT91_CAST(AT91_REG *) 	0xFFFE0000) // (SPI) Control Register
-#define AT91C_SPI_MR    (AT91_CAST(AT91_REG *) 	0xFFFE0004) // (SPI) Mode Register
-#define AT91C_SPI_IMR   (AT91_CAST(AT91_REG *) 	0xFFFE001C) // (SPI) Interrupt Mask Register
-#define AT91C_SPI_TDR   (AT91_CAST(AT91_REG *) 	0xFFFE000C) // (SPI) Transmit Data Register
-#define AT91C_SPI_RDR   (AT91_CAST(AT91_REG *) 	0xFFFE0008) // (SPI) Receive Data Register
-#define AT91C_SPI_CSR   (AT91_CAST(AT91_REG *) 	0xFFFE0030) // (SPI) Chip Select Register
+#define AT91C_SPI_IER   (AT91_CAST(AT91_REG *)  0xFFFE0014) // (SPI) Interrupt Enable Register
+#define AT91C_SPI_SR    (AT91_CAST(AT91_REG *)  0xFFFE0010) // (SPI) Status Register
+#define AT91C_SPI_IDR   (AT91_CAST(AT91_REG *)  0xFFFE0018) // (SPI) Interrupt Disable Register
+#define AT91C_SPI_CR    (AT91_CAST(AT91_REG *)  0xFFFE0000) // (SPI) Control Register
+#define AT91C_SPI_MR    (AT91_CAST(AT91_REG *)  0xFFFE0004) // (SPI) Mode Register
+#define AT91C_SPI_IMR   (AT91_CAST(AT91_REG *)  0xFFFE001C) // (SPI) Interrupt Mask Register
+#define AT91C_SPI_TDR   (AT91_CAST(AT91_REG *)  0xFFFE000C) // (SPI) Transmit Data Register
+#define AT91C_SPI_RDR   (AT91_CAST(AT91_REG *)  0xFFFE0008) // (SPI) Receive Data Register
+#define AT91C_SPI_CSR   (AT91_CAST(AT91_REG *)  0xFFFE0030) // (SPI) Chip Select Register
 // ========== Register definition for PDC_ADC peripheral ==========
-#define AT91C_ADC_PTSR  (AT91_CAST(AT91_REG *) 	0xFFFD8124) // (PDC_ADC) PDC Transfer Status Register
-#define AT91C_ADC_PTCR  (AT91_CAST(AT91_REG *) 	0xFFFD8120) // (PDC_ADC) PDC Transfer Control Register
-#define AT91C_ADC_TNPR  (AT91_CAST(AT91_REG *) 	0xFFFD8118) // (PDC_ADC) Transmit Next Pointer Register
-#define AT91C_ADC_TNCR  (AT91_CAST(AT91_REG *) 	0xFFFD811C) // (PDC_ADC) Transmit Next Counter Register
-#define AT91C_ADC_RNPR  (AT91_CAST(AT91_REG *) 	0xFFFD8110) // (PDC_ADC) Receive Next Pointer Register
-#define AT91C_ADC_RNCR  (AT91_CAST(AT91_REG *) 	0xFFFD8114) // (PDC_ADC) Receive Next Counter Register
-#define AT91C_ADC_RPR   (AT91_CAST(AT91_REG *) 	0xFFFD8100) // (PDC_ADC) Receive Pointer Register
-#define AT91C_ADC_TCR   (AT91_CAST(AT91_REG *) 	0xFFFD810C) // (PDC_ADC) Transmit Counter Register
-#define AT91C_ADC_TPR   (AT91_CAST(AT91_REG *) 	0xFFFD8108) // (PDC_ADC) Transmit Pointer Register
-#define AT91C_ADC_RCR   (AT91_CAST(AT91_REG *) 	0xFFFD8104) // (PDC_ADC) Receive Counter Register
+#define AT91C_ADC_PTSR  (AT91_CAST(AT91_REG *)  0xFFFD8124) // (PDC_ADC) PDC Transfer Status Register
+#define AT91C_ADC_PTCR  (AT91_CAST(AT91_REG *)  0xFFFD8120) // (PDC_ADC) PDC Transfer Control Register
+#define AT91C_ADC_TNPR  (AT91_CAST(AT91_REG *)  0xFFFD8118) // (PDC_ADC) Transmit Next Pointer Register
+#define AT91C_ADC_TNCR  (AT91_CAST(AT91_REG *)  0xFFFD811C) // (PDC_ADC) Transmit Next Counter Register
+#define AT91C_ADC_RNPR  (AT91_CAST(AT91_REG *)  0xFFFD8110) // (PDC_ADC) Receive Next Pointer Register
+#define AT91C_ADC_RNCR  (AT91_CAST(AT91_REG *)  0xFFFD8114) // (PDC_ADC) Receive Next Counter Register
+#define AT91C_ADC_RPR   (AT91_CAST(AT91_REG *)  0xFFFD8100) // (PDC_ADC) Receive Pointer Register
+#define AT91C_ADC_TCR   (AT91_CAST(AT91_REG *)  0xFFFD810C) // (PDC_ADC) Transmit Counter Register
+#define AT91C_ADC_TPR   (AT91_CAST(AT91_REG *)  0xFFFD8108) // (PDC_ADC) Transmit Pointer Register
+#define AT91C_ADC_RCR   (AT91_CAST(AT91_REG *)  0xFFFD8104) // (PDC_ADC) Receive Counter Register
 // ========== Register definition for ADC peripheral ==========
-#define AT91C_ADC_CDR2  (AT91_CAST(AT91_REG *) 	0xFFFD8038) // (ADC) ADC Channel Data Register 2
-#define AT91C_ADC_CDR3  (AT91_CAST(AT91_REG *) 	0xFFFD803C) // (ADC) ADC Channel Data Register 3
-#define AT91C_ADC_CDR0  (AT91_CAST(AT91_REG *) 	0xFFFD8030) // (ADC) ADC Channel Data Register 0
-#define AT91C_ADC_CDR5  (AT91_CAST(AT91_REG *) 	0xFFFD8044) // (ADC) ADC Channel Data Register 5
-#define AT91C_ADC_CHDR  (AT91_CAST(AT91_REG *) 	0xFFFD8014) // (ADC) ADC Channel Disable Register
-#define AT91C_ADC_SR    (AT91_CAST(AT91_REG *) 	0xFFFD801C) // (ADC) ADC Status Register
-#define AT91C_ADC_CDR4  (AT91_CAST(AT91_REG *) 	0xFFFD8040) // (ADC) ADC Channel Data Register 4
-#define AT91C_ADC_CDR1  (AT91_CAST(AT91_REG *) 	0xFFFD8034) // (ADC) ADC Channel Data Register 1
-#define AT91C_ADC_LCDR  (AT91_CAST(AT91_REG *) 	0xFFFD8020) // (ADC) ADC Last Converted Data Register
-#define AT91C_ADC_IDR   (AT91_CAST(AT91_REG *) 	0xFFFD8028) // (ADC) ADC Interrupt Disable Register
-#define AT91C_ADC_CR    (AT91_CAST(AT91_REG *) 	0xFFFD8000) // (ADC) ADC Control Register
-#define AT91C_ADC_CDR7  (AT91_CAST(AT91_REG *) 	0xFFFD804C) // (ADC) ADC Channel Data Register 7
-#define AT91C_ADC_CDR6  (AT91_CAST(AT91_REG *) 	0xFFFD8048) // (ADC) ADC Channel Data Register 6
-#define AT91C_ADC_IER   (AT91_CAST(AT91_REG *) 	0xFFFD8024) // (ADC) ADC Interrupt Enable Register
-#define AT91C_ADC_CHER  (AT91_CAST(AT91_REG *) 	0xFFFD8010) // (ADC) ADC Channel Enable Register
-#define AT91C_ADC_CHSR  (AT91_CAST(AT91_REG *) 	0xFFFD8018) // (ADC) ADC Channel Status Register
-#define AT91C_ADC_MR    (AT91_CAST(AT91_REG *) 	0xFFFD8004) // (ADC) ADC Mode Register
-#define AT91C_ADC_IMR   (AT91_CAST(AT91_REG *) 	0xFFFD802C) // (ADC) ADC Interrupt Mask Register
+#define AT91C_ADC_CDR2  (AT91_CAST(AT91_REG *)  0xFFFD8038) // (ADC) ADC Channel Data Register 2
+#define AT91C_ADC_CDR3  (AT91_CAST(AT91_REG *)  0xFFFD803C) // (ADC) ADC Channel Data Register 3
+#define AT91C_ADC_CDR0  (AT91_CAST(AT91_REG *)  0xFFFD8030) // (ADC) ADC Channel Data Register 0
+#define AT91C_ADC_CDR5  (AT91_CAST(AT91_REG *)  0xFFFD8044) // (ADC) ADC Channel Data Register 5
+#define AT91C_ADC_CHDR  (AT91_CAST(AT91_REG *)  0xFFFD8014) // (ADC) ADC Channel Disable Register
+#define AT91C_ADC_SR    (AT91_CAST(AT91_REG *)  0xFFFD801C) // (ADC) ADC Status Register
+#define AT91C_ADC_CDR4  (AT91_CAST(AT91_REG *)  0xFFFD8040) // (ADC) ADC Channel Data Register 4
+#define AT91C_ADC_CDR1  (AT91_CAST(AT91_REG *)  0xFFFD8034) // (ADC) ADC Channel Data Register 1
+#define AT91C_ADC_LCDR  (AT91_CAST(AT91_REG *)  0xFFFD8020) // (ADC) ADC Last Converted Data Register
+#define AT91C_ADC_IDR   (AT91_CAST(AT91_REG *)  0xFFFD8028) // (ADC) ADC Interrupt Disable Register
+#define AT91C_ADC_CR    (AT91_CAST(AT91_REG *)  0xFFFD8000) // (ADC) ADC Control Register
+#define AT91C_ADC_CDR7  (AT91_CAST(AT91_REG *)  0xFFFD804C) // (ADC) ADC Channel Data Register 7
+#define AT91C_ADC_CDR6  (AT91_CAST(AT91_REG *)  0xFFFD8048) // (ADC) ADC Channel Data Register 6
+#define AT91C_ADC_IER   (AT91_CAST(AT91_REG *)  0xFFFD8024) // (ADC) ADC Interrupt Enable Register
+#define AT91C_ADC_CHER  (AT91_CAST(AT91_REG *)  0xFFFD8010) // (ADC) ADC Channel Enable Register
+#define AT91C_ADC_CHSR  (AT91_CAST(AT91_REG *)  0xFFFD8018) // (ADC) ADC Channel Status Register
+#define AT91C_ADC_MR    (AT91_CAST(AT91_REG *)  0xFFFD8004) // (ADC) ADC Mode Register
+#define AT91C_ADC_IMR   (AT91_CAST(AT91_REG *)  0xFFFD802C) // (ADC) ADC Interrupt Mask Register
 // ========== Register definition for PDC_SSC peripheral ==========
-#define AT91C_SSC_TNCR  (AT91_CAST(AT91_REG *) 	0xFFFD411C) // (PDC_SSC) Transmit Next Counter Register
-#define AT91C_SSC_RPR   (AT91_CAST(AT91_REG *) 	0xFFFD4100) // (PDC_SSC) Receive Pointer Register
-#define AT91C_SSC_RNCR  (AT91_CAST(AT91_REG *) 	0xFFFD4114) // (PDC_SSC) Receive Next Counter Register
-#define AT91C_SSC_TPR   (AT91_CAST(AT91_REG *) 	0xFFFD4108) // (PDC_SSC) Transmit Pointer Register
-#define AT91C_SSC_PTCR  (AT91_CAST(AT91_REG *) 	0xFFFD4120) // (PDC_SSC) PDC Transfer Control Register
-#define AT91C_SSC_TCR   (AT91_CAST(AT91_REG *) 	0xFFFD410C) // (PDC_SSC) Transmit Counter Register
-#define AT91C_SSC_RCR   (AT91_CAST(AT91_REG *) 	0xFFFD4104) // (PDC_SSC) Receive Counter Register
-#define AT91C_SSC_RNPR  (AT91_CAST(AT91_REG *) 	0xFFFD4110) // (PDC_SSC) Receive Next Pointer Register
-#define AT91C_SSC_TNPR  (AT91_CAST(AT91_REG *) 	0xFFFD4118) // (PDC_SSC) Transmit Next Pointer Register
-#define AT91C_SSC_PTSR  (AT91_CAST(AT91_REG *) 	0xFFFD4124) // (PDC_SSC) PDC Transfer Status Register
+#define AT91C_SSC_TNCR  (AT91_CAST(AT91_REG *)  0xFFFD411C) // (PDC_SSC) Transmit Next Counter Register
+#define AT91C_SSC_RPR   (AT91_CAST(AT91_REG *)  0xFFFD4100) // (PDC_SSC) Receive Pointer Register
+#define AT91C_SSC_RNCR  (AT91_CAST(AT91_REG *)  0xFFFD4114) // (PDC_SSC) Receive Next Counter Register
+#define AT91C_SSC_TPR   (AT91_CAST(AT91_REG *)  0xFFFD4108) // (PDC_SSC) Transmit Pointer Register
+#define AT91C_SSC_PTCR  (AT91_CAST(AT91_REG *)  0xFFFD4120) // (PDC_SSC) PDC Transfer Control Register
+#define AT91C_SSC_TCR   (AT91_CAST(AT91_REG *)  0xFFFD410C) // (PDC_SSC) Transmit Counter Register
+#define AT91C_SSC_RCR   (AT91_CAST(AT91_REG *)  0xFFFD4104) // (PDC_SSC) Receive Counter Register
+#define AT91C_SSC_RNPR  (AT91_CAST(AT91_REG *)  0xFFFD4110) // (PDC_SSC) Receive Next Pointer Register
+#define AT91C_SSC_TNPR  (AT91_CAST(AT91_REG *)  0xFFFD4118) // (PDC_SSC) Transmit Next Pointer Register
+#define AT91C_SSC_PTSR  (AT91_CAST(AT91_REG *)  0xFFFD4124) // (PDC_SSC) PDC Transfer Status Register
 // ========== Register definition for SSC peripheral ==========
-#define AT91C_SSC_RHR   (AT91_CAST(AT91_REG *) 	0xFFFD4020) // (SSC) Receive Holding Register
-#define AT91C_SSC_RSHR  (AT91_CAST(AT91_REG *) 	0xFFFD4030) // (SSC) Receive Sync Holding Register
-#define AT91C_SSC_TFMR  (AT91_CAST(AT91_REG *) 	0xFFFD401C) // (SSC) Transmit Frame Mode Register
-#define AT91C_SSC_IDR   (AT91_CAST(AT91_REG *) 	0xFFFD4048) // (SSC) Interrupt Disable Register
-#define AT91C_SSC_THR   (AT91_CAST(AT91_REG *) 	0xFFFD4024) // (SSC) Transmit Holding Register
-#define AT91C_SSC_RCMR  (AT91_CAST(AT91_REG *) 	0xFFFD4010) // (SSC) Receive Clock ModeRegister
-#define AT91C_SSC_IER   (AT91_CAST(AT91_REG *) 	0xFFFD4044) // (SSC) Interrupt Enable Register
-#define AT91C_SSC_TSHR  (AT91_CAST(AT91_REG *) 	0xFFFD4034) // (SSC) Transmit Sync Holding Register
-#define AT91C_SSC_SR    (AT91_CAST(AT91_REG *) 	0xFFFD4040) // (SSC) Status Register
-#define AT91C_SSC_CMR   (AT91_CAST(AT91_REG *) 	0xFFFD4004) // (SSC) Clock Mode Register
-#define AT91C_SSC_TCMR  (AT91_CAST(AT91_REG *) 	0xFFFD4018) // (SSC) Transmit Clock Mode Register
-#define AT91C_SSC_CR    (AT91_CAST(AT91_REG *) 	0xFFFD4000) // (SSC) Control Register
-#define AT91C_SSC_IMR   (AT91_CAST(AT91_REG *) 	0xFFFD404C) // (SSC) Interrupt Mask Register
-#define AT91C_SSC_RFMR  (AT91_CAST(AT91_REG *) 	0xFFFD4014) // (SSC) Receive Frame Mode Register
+#define AT91C_SSC_RHR   (AT91_CAST(AT91_REG *)  0xFFFD4020) // (SSC) Receive Holding Register
+#define AT91C_SSC_RSHR  (AT91_CAST(AT91_REG *)  0xFFFD4030) // (SSC) Receive Sync Holding Register
+#define AT91C_SSC_TFMR  (AT91_CAST(AT91_REG *)  0xFFFD401C) // (SSC) Transmit Frame Mode Register
+#define AT91C_SSC_IDR   (AT91_CAST(AT91_REG *)  0xFFFD4048) // (SSC) Interrupt Disable Register
+#define AT91C_SSC_THR   (AT91_CAST(AT91_REG *)  0xFFFD4024) // (SSC) Transmit Holding Register
+#define AT91C_SSC_RCMR  (AT91_CAST(AT91_REG *)  0xFFFD4010) // (SSC) Receive Clock ModeRegister
+#define AT91C_SSC_IER   (AT91_CAST(AT91_REG *)  0xFFFD4044) // (SSC) Interrupt Enable Register
+#define AT91C_SSC_TSHR  (AT91_CAST(AT91_REG *)  0xFFFD4034) // (SSC) Transmit Sync Holding Register
+#define AT91C_SSC_SR    (AT91_CAST(AT91_REG *)  0xFFFD4040) // (SSC) Status Register
+#define AT91C_SSC_CMR   (AT91_CAST(AT91_REG *)  0xFFFD4004) // (SSC) Clock Mode Register
+#define AT91C_SSC_TCMR  (AT91_CAST(AT91_REG *)  0xFFFD4018) // (SSC) Transmit Clock Mode Register
+#define AT91C_SSC_CR    (AT91_CAST(AT91_REG *)  0xFFFD4000) // (SSC) Control Register
+#define AT91C_SSC_IMR   (AT91_CAST(AT91_REG *)  0xFFFD404C) // (SSC) Interrupt Mask Register
+#define AT91C_SSC_RFMR  (AT91_CAST(AT91_REG *)  0xFFFD4014) // (SSC) Receive Frame Mode Register
 // ========== Register definition for PDC_US1 peripheral ==========
-#define AT91C_US1_RNCR  (AT91_CAST(AT91_REG *) 	0xFFFC4114) // (PDC_US1) Receive Next Counter Register
-#define AT91C_US1_PTCR  (AT91_CAST(AT91_REG *) 	0xFFFC4120) // (PDC_US1) PDC Transfer Control Register
-#define AT91C_US1_TCR   (AT91_CAST(AT91_REG *) 	0xFFFC410C) // (PDC_US1) Transmit Counter Register
-#define AT91C_US1_PTSR  (AT91_CAST(AT91_REG *) 	0xFFFC4124) // (PDC_US1) PDC Transfer Status Register
-#define AT91C_US1_TNPR  (AT91_CAST(AT91_REG *) 	0xFFFC4118) // (PDC_US1) Transmit Next Pointer Register
-#define AT91C_US1_RCR   (AT91_CAST(AT91_REG *) 	0xFFFC4104) // (PDC_US1) Receive Counter Register
-#define AT91C_US1_RNPR  (AT91_CAST(AT91_REG *) 	0xFFFC4110) // (PDC_US1) Receive Next Pointer Register
-#define AT91C_US1_RPR   (AT91_CAST(AT91_REG *) 	0xFFFC4100) // (PDC_US1) Receive Pointer Register
-#define AT91C_US1_TNCR  (AT91_CAST(AT91_REG *) 	0xFFFC411C) // (PDC_US1) Transmit Next Counter Register
-#define AT91C_US1_TPR   (AT91_CAST(AT91_REG *) 	0xFFFC4108) // (PDC_US1) Transmit Pointer Register
+#define AT91C_US1_RNCR  (AT91_CAST(AT91_REG *)  0xFFFC4114) // (PDC_US1) Receive Next Counter Register
+#define AT91C_US1_PTCR  (AT91_CAST(AT91_REG *)  0xFFFC4120) // (PDC_US1) PDC Transfer Control Register
+#define AT91C_US1_TCR   (AT91_CAST(AT91_REG *)  0xFFFC410C) // (PDC_US1) Transmit Counter Register
+#define AT91C_US1_PTSR  (AT91_CAST(AT91_REG *)  0xFFFC4124) // (PDC_US1) PDC Transfer Status Register
+#define AT91C_US1_TNPR  (AT91_CAST(AT91_REG *)  0xFFFC4118) // (PDC_US1) Transmit Next Pointer Register
+#define AT91C_US1_RCR   (AT91_CAST(AT91_REG *)  0xFFFC4104) // (PDC_US1) Receive Counter Register
+#define AT91C_US1_RNPR  (AT91_CAST(AT91_REG *)  0xFFFC4110) // (PDC_US1) Receive Next Pointer Register
+#define AT91C_US1_RPR   (AT91_CAST(AT91_REG *)  0xFFFC4100) // (PDC_US1) Receive Pointer Register
+#define AT91C_US1_TNCR  (AT91_CAST(AT91_REG *)  0xFFFC411C) // (PDC_US1) Transmit Next Counter Register
+#define AT91C_US1_TPR   (AT91_CAST(AT91_REG *)  0xFFFC4108) // (PDC_US1) Transmit Pointer Register
 // ========== Register definition for US1 peripheral ==========
-#define AT91C_US1_IF    (AT91_CAST(AT91_REG *) 	0xFFFC404C) // (US1) IRDA_FILTER Register
-#define AT91C_US1_NER   (AT91_CAST(AT91_REG *) 	0xFFFC4044) // (US1) Nb Errors Register
-#define AT91C_US1_RTOR  (AT91_CAST(AT91_REG *) 	0xFFFC4024) // (US1) Receiver Time-out Register
-#define AT91C_US1_CSR   (AT91_CAST(AT91_REG *) 	0xFFFC4014) // (US1) Channel Status Register
-#define AT91C_US1_IDR   (AT91_CAST(AT91_REG *) 	0xFFFC400C) // (US1) Interrupt Disable Register
-#define AT91C_US1_IER   (AT91_CAST(AT91_REG *) 	0xFFFC4008) // (US1) Interrupt Enable Register
-#define AT91C_US1_THR   (AT91_CAST(AT91_REG *) 	0xFFFC401C) // (US1) Transmitter Holding Register
-#define AT91C_US1_TTGR  (AT91_CAST(AT91_REG *) 	0xFFFC4028) // (US1) Transmitter Time-guard Register
-#define AT91C_US1_RHR   (AT91_CAST(AT91_REG *) 	0xFFFC4018) // (US1) Receiver Holding Register
-#define AT91C_US1_BRGR  (AT91_CAST(AT91_REG *) 	0xFFFC4020) // (US1) Baud Rate Generator Register
-#define AT91C_US1_IMR   (AT91_CAST(AT91_REG *) 	0xFFFC4010) // (US1) Interrupt Mask Register
-#define AT91C_US1_FIDI  (AT91_CAST(AT91_REG *) 	0xFFFC4040) // (US1) FI_DI_Ratio Register
-#define AT91C_US1_CR    (AT91_CAST(AT91_REG *) 	0xFFFC4000) // (US1) Control Register
-#define AT91C_US1_MR    (AT91_CAST(AT91_REG *) 	0xFFFC4004) // (US1) Mode Register
+#define AT91C_US1_IF    (AT91_CAST(AT91_REG *)  0xFFFC404C) // (US1) IRDA_FILTER Register
+#define AT91C_US1_NER   (AT91_CAST(AT91_REG *)  0xFFFC4044) // (US1) Nb Errors Register
+#define AT91C_US1_RTOR  (AT91_CAST(AT91_REG *)  0xFFFC4024) // (US1) Receiver Time-out Register
+#define AT91C_US1_CSR   (AT91_CAST(AT91_REG *)  0xFFFC4014) // (US1) Channel Status Register
+#define AT91C_US1_IDR   (AT91_CAST(AT91_REG *)  0xFFFC400C) // (US1) Interrupt Disable Register
+#define AT91C_US1_IER   (AT91_CAST(AT91_REG *)  0xFFFC4008) // (US1) Interrupt Enable Register
+#define AT91C_US1_THR   (AT91_CAST(AT91_REG *)  0xFFFC401C) // (US1) Transmitter Holding Register
+#define AT91C_US1_TTGR  (AT91_CAST(AT91_REG *)  0xFFFC4028) // (US1) Transmitter Time-guard Register
+#define AT91C_US1_RHR   (AT91_CAST(AT91_REG *)  0xFFFC4018) // (US1) Receiver Holding Register
+#define AT91C_US1_BRGR  (AT91_CAST(AT91_REG *)  0xFFFC4020) // (US1) Baud Rate Generator Register
+#define AT91C_US1_IMR   (AT91_CAST(AT91_REG *)  0xFFFC4010) // (US1) Interrupt Mask Register
+#define AT91C_US1_FIDI  (AT91_CAST(AT91_REG *)  0xFFFC4040) // (US1) FI_DI_Ratio Register
+#define AT91C_US1_CR    (AT91_CAST(AT91_REG *)  0xFFFC4000) // (US1) Control Register
+#define AT91C_US1_MR    (AT91_CAST(AT91_REG *)  0xFFFC4004) // (US1) Mode Register
 // ========== Register definition for PDC_US0 peripheral ==========
-#define AT91C_US0_TNPR  (AT91_CAST(AT91_REG *) 	0xFFFC0118) // (PDC_US0) Transmit Next Pointer Register
-#define AT91C_US0_RNPR  (AT91_CAST(AT91_REG *) 	0xFFFC0110) // (PDC_US0) Receive Next Pointer Register
-#define AT91C_US0_TCR   (AT91_CAST(AT91_REG *) 	0xFFFC010C) // (PDC_US0) Transmit Counter Register
-#define AT91C_US0_PTCR  (AT91_CAST(AT91_REG *) 	0xFFFC0120) // (PDC_US0) PDC Transfer Control Register
-#define AT91C_US0_PTSR  (AT91_CAST(AT91_REG *) 	0xFFFC0124) // (PDC_US0) PDC Transfer Status Register
-#define AT91C_US0_TNCR  (AT91_CAST(AT91_REG *) 	0xFFFC011C) // (PDC_US0) Transmit Next Counter Register
-#define AT91C_US0_TPR   (AT91_CAST(AT91_REG *) 	0xFFFC0108) // (PDC_US0) Transmit Pointer Register
-#define AT91C_US0_RCR   (AT91_CAST(AT91_REG *) 	0xFFFC0104) // (PDC_US0) Receive Counter Register
-#define AT91C_US0_RPR   (AT91_CAST(AT91_REG *) 	0xFFFC0100) // (PDC_US0) Receive Pointer Register
-#define AT91C_US0_RNCR  (AT91_CAST(AT91_REG *) 	0xFFFC0114) // (PDC_US0) Receive Next Counter Register
+#define AT91C_US0_TNPR  (AT91_CAST(AT91_REG *)  0xFFFC0118) // (PDC_US0) Transmit Next Pointer Register
+#define AT91C_US0_RNPR  (AT91_CAST(AT91_REG *)  0xFFFC0110) // (PDC_US0) Receive Next Pointer Register
+#define AT91C_US0_TCR   (AT91_CAST(AT91_REG *)  0xFFFC010C) // (PDC_US0) Transmit Counter Register
+#define AT91C_US0_PTCR  (AT91_CAST(AT91_REG *)  0xFFFC0120) // (PDC_US0) PDC Transfer Control Register
+#define AT91C_US0_PTSR  (AT91_CAST(AT91_REG *)  0xFFFC0124) // (PDC_US0) PDC Transfer Status Register
+#define AT91C_US0_TNCR  (AT91_CAST(AT91_REG *)  0xFFFC011C) // (PDC_US0) Transmit Next Counter Register
+#define AT91C_US0_TPR   (AT91_CAST(AT91_REG *)  0xFFFC0108) // (PDC_US0) Transmit Pointer Register
+#define AT91C_US0_RCR   (AT91_CAST(AT91_REG *)  0xFFFC0104) // (PDC_US0) Receive Counter Register
+#define AT91C_US0_RPR   (AT91_CAST(AT91_REG *)  0xFFFC0100) // (PDC_US0) Receive Pointer Register
+#define AT91C_US0_RNCR  (AT91_CAST(AT91_REG *)  0xFFFC0114) // (PDC_US0) Receive Next Counter Register
 // ========== Register definition for US0 peripheral ==========
-#define AT91C_US0_BRGR  (AT91_CAST(AT91_REG *) 	0xFFFC0020) // (US0) Baud Rate Generator Register
-#define AT91C_US0_NER   (AT91_CAST(AT91_REG *) 	0xFFFC0044) // (US0) Nb Errors Register
-#define AT91C_US0_CR    (AT91_CAST(AT91_REG *) 	0xFFFC0000) // (US0) Control Register
-#define AT91C_US0_IMR   (AT91_CAST(AT91_REG *) 	0xFFFC0010) // (US0) Interrupt Mask Register
-#define AT91C_US0_FIDI  (AT91_CAST(AT91_REG *) 	0xFFFC0040) // (US0) FI_DI_Ratio Register
-#define AT91C_US0_TTGR  (AT91_CAST(AT91_REG *) 	0xFFFC0028) // (US0) Transmitter Time-guard Register
-#define AT91C_US0_MR    (AT91_CAST(AT91_REG *) 	0xFFFC0004) // (US0) Mode Register
-#define AT91C_US0_RTOR  (AT91_CAST(AT91_REG *) 	0xFFFC0024) // (US0) Receiver Time-out Register
-#define AT91C_US0_CSR   (AT91_CAST(AT91_REG *) 	0xFFFC0014) // (US0) Channel Status Register
-#define AT91C_US0_RHR   (AT91_CAST(AT91_REG *) 	0xFFFC0018) // (US0) Receiver Holding Register
-#define AT91C_US0_IDR   (AT91_CAST(AT91_REG *) 	0xFFFC000C) // (US0) Interrupt Disable Register
-#define AT91C_US0_THR   (AT91_CAST(AT91_REG *) 	0xFFFC001C) // (US0) Transmitter Holding Register
-#define AT91C_US0_IF    (AT91_CAST(AT91_REG *) 	0xFFFC004C) // (US0) IRDA_FILTER Register
-#define AT91C_US0_IER   (AT91_CAST(AT91_REG *) 	0xFFFC0008) // (US0) Interrupt Enable Register
+#define AT91C_US0_BRGR  (AT91_CAST(AT91_REG *)  0xFFFC0020) // (US0) Baud Rate Generator Register
+#define AT91C_US0_NER   (AT91_CAST(AT91_REG *)  0xFFFC0044) // (US0) Nb Errors Register
+#define AT91C_US0_CR    (AT91_CAST(AT91_REG *)  0xFFFC0000) // (US0) Control Register
+#define AT91C_US0_IMR   (AT91_CAST(AT91_REG *)  0xFFFC0010) // (US0) Interrupt Mask Register
+#define AT91C_US0_FIDI  (AT91_CAST(AT91_REG *)  0xFFFC0040) // (US0) FI_DI_Ratio Register
+#define AT91C_US0_TTGR  (AT91_CAST(AT91_REG *)  0xFFFC0028) // (US0) Transmitter Time-guard Register
+#define AT91C_US0_MR    (AT91_CAST(AT91_REG *)  0xFFFC0004) // (US0) Mode Register
+#define AT91C_US0_RTOR  (AT91_CAST(AT91_REG *)  0xFFFC0024) // (US0) Receiver Time-out Register
+#define AT91C_US0_CSR   (AT91_CAST(AT91_REG *)  0xFFFC0014) // (US0) Channel Status Register
+#define AT91C_US0_RHR   (AT91_CAST(AT91_REG *)  0xFFFC0018) // (US0) Receiver Holding Register
+#define AT91C_US0_IDR   (AT91_CAST(AT91_REG *)  0xFFFC000C) // (US0) Interrupt Disable Register
+#define AT91C_US0_THR   (AT91_CAST(AT91_REG *)  0xFFFC001C) // (US0) Transmitter Holding Register
+#define AT91C_US0_IF    (AT91_CAST(AT91_REG *)  0xFFFC004C) // (US0) IRDA_FILTER Register
+#define AT91C_US0_IER   (AT91_CAST(AT91_REG *)  0xFFFC0008) // (US0) Interrupt Enable Register
 // ========== Register definition for TWI peripheral ==========
-#define AT91C_TWI_IER   (AT91_CAST(AT91_REG *) 	0xFFFB8024) // (TWI) Interrupt Enable Register
-#define AT91C_TWI_CR    (AT91_CAST(AT91_REG *) 	0xFFFB8000) // (TWI) Control Register
-#define AT91C_TWI_SR    (AT91_CAST(AT91_REG *) 	0xFFFB8020) // (TWI) Status Register
-#define AT91C_TWI_IMR   (AT91_CAST(AT91_REG *) 	0xFFFB802C) // (TWI) Interrupt Mask Register
-#define AT91C_TWI_THR   (AT91_CAST(AT91_REG *) 	0xFFFB8034) // (TWI) Transmit Holding Register
-#define AT91C_TWI_IDR   (AT91_CAST(AT91_REG *) 	0xFFFB8028) // (TWI) Interrupt Disable Register
-#define AT91C_TWI_IADR  (AT91_CAST(AT91_REG *) 	0xFFFB800C) // (TWI) Internal Address Register
-#define AT91C_TWI_MMR   (AT91_CAST(AT91_REG *) 	0xFFFB8004) // (TWI) Master Mode Register
-#define AT91C_TWI_CWGR  (AT91_CAST(AT91_REG *) 	0xFFFB8010) // (TWI) Clock Waveform Generator Register
-#define AT91C_TWI_RHR   (AT91_CAST(AT91_REG *) 	0xFFFB8030) // (TWI) Receive Holding Register
+#define AT91C_TWI_IER   (AT91_CAST(AT91_REG *)  0xFFFB8024) // (TWI) Interrupt Enable Register
+#define AT91C_TWI_CR    (AT91_CAST(AT91_REG *)  0xFFFB8000) // (TWI) Control Register
+#define AT91C_TWI_SR    (AT91_CAST(AT91_REG *)  0xFFFB8020) // (TWI) Status Register
+#define AT91C_TWI_IMR   (AT91_CAST(AT91_REG *)  0xFFFB802C) // (TWI) Interrupt Mask Register
+#define AT91C_TWI_THR   (AT91_CAST(AT91_REG *)  0xFFFB8034) // (TWI) Transmit Holding Register
+#define AT91C_TWI_IDR   (AT91_CAST(AT91_REG *)  0xFFFB8028) // (TWI) Interrupt Disable Register
+#define AT91C_TWI_IADR  (AT91_CAST(AT91_REG *)  0xFFFB800C) // (TWI) Internal Address Register
+#define AT91C_TWI_MMR   (AT91_CAST(AT91_REG *)  0xFFFB8004) // (TWI) Master Mode Register
+#define AT91C_TWI_CWGR  (AT91_CAST(AT91_REG *)  0xFFFB8010) // (TWI) Clock Waveform Generator Register
+#define AT91C_TWI_RHR   (AT91_CAST(AT91_REG *)  0xFFFB8030) // (TWI) Receive Holding Register
 // ========== Register definition for TC0 peripheral ==========
-#define AT91C_TC0_SR    (AT91_CAST(AT91_REG *) 	0xFFFA0020) // (TC0) Status Register
-#define AT91C_TC0_RC    (AT91_CAST(AT91_REG *) 	0xFFFA001C) // (TC0) Register C
-#define AT91C_TC0_RB    (AT91_CAST(AT91_REG *) 	0xFFFA0018) // (TC0) Register B
-#define AT91C_TC0_CCR   (AT91_CAST(AT91_REG *) 	0xFFFA0000) // (TC0) Channel Control Register
-#define AT91C_TC0_CMR   (AT91_CAST(AT91_REG *) 	0xFFFA0004) // (TC0) Channel Mode Register (Capture Mode / Waveform Mode)
-#define AT91C_TC0_IER   (AT91_CAST(AT91_REG *) 	0xFFFA0024) // (TC0) Interrupt Enable Register
-#define AT91C_TC0_RA    (AT91_CAST(AT91_REG *) 	0xFFFA0014) // (TC0) Register A
-#define AT91C_TC0_IDR   (AT91_CAST(AT91_REG *) 	0xFFFA0028) // (TC0) Interrupt Disable Register
-#define AT91C_TC0_CV    (AT91_CAST(AT91_REG *) 	0xFFFA0010) // (TC0) Counter Value
-#define AT91C_TC0_IMR   (AT91_CAST(AT91_REG *) 	0xFFFA002C) // (TC0) Interrupt Mask Register
+#define AT91C_TC0_SR    (AT91_CAST(AT91_REG *)  0xFFFA0020) // (TC0) Status Register
+#define AT91C_TC0_RC    (AT91_CAST(AT91_REG *)  0xFFFA001C) // (TC0) Register C
+#define AT91C_TC0_RB    (AT91_CAST(AT91_REG *)  0xFFFA0018) // (TC0) Register B
+#define AT91C_TC0_CCR   (AT91_CAST(AT91_REG *)  0xFFFA0000) // (TC0) Channel Control Register
+#define AT91C_TC0_CMR   (AT91_CAST(AT91_REG *)  0xFFFA0004) // (TC0) Channel Mode Register (Capture Mode / Waveform Mode)
+#define AT91C_TC0_IER   (AT91_CAST(AT91_REG *)  0xFFFA0024) // (TC0) Interrupt Enable Register
+#define AT91C_TC0_RA    (AT91_CAST(AT91_REG *)  0xFFFA0014) // (TC0) Register A
+#define AT91C_TC0_IDR   (AT91_CAST(AT91_REG *)  0xFFFA0028) // (TC0) Interrupt Disable Register
+#define AT91C_TC0_CV    (AT91_CAST(AT91_REG *)  0xFFFA0010) // (TC0) Counter Value
+#define AT91C_TC0_IMR   (AT91_CAST(AT91_REG *)  0xFFFA002C) // (TC0) Interrupt Mask Register
 // ========== Register definition for TC1 peripheral ==========
-#define AT91C_TC1_RB    (AT91_CAST(AT91_REG *) 	0xFFFA0058) // (TC1) Register B
-#define AT91C_TC1_CCR   (AT91_CAST(AT91_REG *) 	0xFFFA0040) // (TC1) Channel Control Register
-#define AT91C_TC1_IER   (AT91_CAST(AT91_REG *) 	0xFFFA0064) // (TC1) Interrupt Enable Register
-#define AT91C_TC1_IDR   (AT91_CAST(AT91_REG *) 	0xFFFA0068) // (TC1) Interrupt Disable Register
-#define AT91C_TC1_SR    (AT91_CAST(AT91_REG *) 	0xFFFA0060) // (TC1) Status Register
-#define AT91C_TC1_CMR   (AT91_CAST(AT91_REG *) 	0xFFFA0044) // (TC1) Channel Mode Register (Capture Mode / Waveform Mode)
-#define AT91C_TC1_RA    (AT91_CAST(AT91_REG *) 	0xFFFA0054) // (TC1) Register A
-#define AT91C_TC1_RC    (AT91_CAST(AT91_REG *) 	0xFFFA005C) // (TC1) Register C
-#define AT91C_TC1_IMR   (AT91_CAST(AT91_REG *) 	0xFFFA006C) // (TC1) Interrupt Mask Register
-#define AT91C_TC1_CV    (AT91_CAST(AT91_REG *) 	0xFFFA0050) // (TC1) Counter Value
+#define AT91C_TC1_RB    (AT91_CAST(AT91_REG *)  0xFFFA0058) // (TC1) Register B
+#define AT91C_TC1_CCR   (AT91_CAST(AT91_REG *)  0xFFFA0040) // (TC1) Channel Control Register
+#define AT91C_TC1_IER   (AT91_CAST(AT91_REG *)  0xFFFA0064) // (TC1) Interrupt Enable Register
+#define AT91C_TC1_IDR   (AT91_CAST(AT91_REG *)  0xFFFA0068) // (TC1) Interrupt Disable Register
+#define AT91C_TC1_SR    (AT91_CAST(AT91_REG *)  0xFFFA0060) // (TC1) Status Register
+#define AT91C_TC1_CMR   (AT91_CAST(AT91_REG *)  0xFFFA0044) // (TC1) Channel Mode Register (Capture Mode / Waveform Mode)
+#define AT91C_TC1_RA    (AT91_CAST(AT91_REG *)  0xFFFA0054) // (TC1) Register A
+#define AT91C_TC1_RC    (AT91_CAST(AT91_REG *)  0xFFFA005C) // (TC1) Register C
+#define AT91C_TC1_IMR   (AT91_CAST(AT91_REG *)  0xFFFA006C) // (TC1) Interrupt Mask Register
+#define AT91C_TC1_CV    (AT91_CAST(AT91_REG *)  0xFFFA0050) // (TC1) Counter Value
 // ========== Register definition for TC2 peripheral ==========
-#define AT91C_TC2_CMR   (AT91_CAST(AT91_REG *) 	0xFFFA0084) // (TC2) Channel Mode Register (Capture Mode / Waveform Mode)
-#define AT91C_TC2_CCR   (AT91_CAST(AT91_REG *) 	0xFFFA0080) // (TC2) Channel Control Register
-#define AT91C_TC2_CV    (AT91_CAST(AT91_REG *) 	0xFFFA0090) // (TC2) Counter Value
-#define AT91C_TC2_RA    (AT91_CAST(AT91_REG *) 	0xFFFA0094) // (TC2) Register A
-#define AT91C_TC2_RB    (AT91_CAST(AT91_REG *) 	0xFFFA0098) // (TC2) Register B
-#define AT91C_TC2_IDR   (AT91_CAST(AT91_REG *) 	0xFFFA00A8) // (TC2) Interrupt Disable Register
-#define AT91C_TC2_IMR   (AT91_CAST(AT91_REG *) 	0xFFFA00AC) // (TC2) Interrupt Mask Register
-#define AT91C_TC2_RC    (AT91_CAST(AT91_REG *) 	0xFFFA009C) // (TC2) Register C
-#define AT91C_TC2_IER   (AT91_CAST(AT91_REG *) 	0xFFFA00A4) // (TC2) Interrupt Enable Register
-#define AT91C_TC2_SR    (AT91_CAST(AT91_REG *) 	0xFFFA00A0) // (TC2) Status Register
+#define AT91C_TC2_CMR   (AT91_CAST(AT91_REG *)  0xFFFA0084) // (TC2) Channel Mode Register (Capture Mode / Waveform Mode)
+#define AT91C_TC2_CCR   (AT91_CAST(AT91_REG *)  0xFFFA0080) // (TC2) Channel Control Register
+#define AT91C_TC2_CV    (AT91_CAST(AT91_REG *)  0xFFFA0090) // (TC2) Counter Value
+#define AT91C_TC2_RA    (AT91_CAST(AT91_REG *)  0xFFFA0094) // (TC2) Register A
+#define AT91C_TC2_RB    (AT91_CAST(AT91_REG *)  0xFFFA0098) // (TC2) Register B
+#define AT91C_TC2_IDR   (AT91_CAST(AT91_REG *)  0xFFFA00A8) // (TC2) Interrupt Disable Register
+#define AT91C_TC2_IMR   (AT91_CAST(AT91_REG *)  0xFFFA00AC) // (TC2) Interrupt Mask Register
+#define AT91C_TC2_RC    (AT91_CAST(AT91_REG *)  0xFFFA009C) // (TC2) Register C
+#define AT91C_TC2_IER   (AT91_CAST(AT91_REG *)  0xFFFA00A4) // (TC2) Interrupt Enable Register
+#define AT91C_TC2_SR    (AT91_CAST(AT91_REG *)  0xFFFA00A0) // (TC2) Status Register
 // ========== Register definition for TCB peripheral ==========
-#define AT91C_TCB_BMR   (AT91_CAST(AT91_REG *) 	0xFFFA00C4) // (TCB) TC Block Mode Register
-#define AT91C_TCB_BCR   (AT91_CAST(AT91_REG *) 	0xFFFA00C0) // (TCB) TC Block Control Register
+#define AT91C_TCB_BMR   (AT91_CAST(AT91_REG *)  0xFFFA00C4) // (TCB) TC Block Mode Register
+#define AT91C_TCB_BCR   (AT91_CAST(AT91_REG *)  0xFFFA00C0) // (TCB) TC Block Control Register
 // ========== Register definition for PWMC_CH3 peripheral ==========
-#define AT91C_PWMC_CH3_CUPDR (AT91_CAST(AT91_REG *) 	0xFFFCC270) // (PWMC_CH3) Channel Update Register
-#define AT91C_PWMC_CH3_Reserved (AT91_CAST(AT91_REG *) 	0xFFFCC274) // (PWMC_CH3) Reserved
-#define AT91C_PWMC_CH3_CPRDR (AT91_CAST(AT91_REG *) 	0xFFFCC268) // (PWMC_CH3) Channel Period Register
-#define AT91C_PWMC_CH3_CDTYR (AT91_CAST(AT91_REG *) 	0xFFFCC264) // (PWMC_CH3) Channel Duty Cycle Register
-#define AT91C_PWMC_CH3_CCNTR (AT91_CAST(AT91_REG *) 	0xFFFCC26C) // (PWMC_CH3) Channel Counter Register
-#define AT91C_PWMC_CH3_CMR (AT91_CAST(AT91_REG *) 	0xFFFCC260) // (PWMC_CH3) Channel Mode Register
+#define AT91C_PWMC_CH3_CUPDR (AT91_CAST(AT91_REG *)     0xFFFCC270) // (PWMC_CH3) Channel Update Register
+#define AT91C_PWMC_CH3_Reserved (AT91_CAST(AT91_REG *)  0xFFFCC274) // (PWMC_CH3) Reserved
+#define AT91C_PWMC_CH3_CPRDR (AT91_CAST(AT91_REG *)     0xFFFCC268) // (PWMC_CH3) Channel Period Register
+#define AT91C_PWMC_CH3_CDTYR (AT91_CAST(AT91_REG *)     0xFFFCC264) // (PWMC_CH3) Channel Duty Cycle Register
+#define AT91C_PWMC_CH3_CCNTR (AT91_CAST(AT91_REG *)     0xFFFCC26C) // (PWMC_CH3) Channel Counter Register
+#define AT91C_PWMC_CH3_CMR (AT91_CAST(AT91_REG *)   0xFFFCC260) // (PWMC_CH3) Channel Mode Register
 // ========== Register definition for PWMC_CH2 peripheral ==========
-#define AT91C_PWMC_CH2_Reserved (AT91_CAST(AT91_REG *) 	0xFFFCC254) // (PWMC_CH2) Reserved
-#define AT91C_PWMC_CH2_CMR (AT91_CAST(AT91_REG *) 	0xFFFCC240) // (PWMC_CH2) Channel Mode Register
-#define AT91C_PWMC_CH2_CCNTR (AT91_CAST(AT91_REG *) 	0xFFFCC24C) // (PWMC_CH2) Channel Counter Register
-#define AT91C_PWMC_CH2_CPRDR (AT91_CAST(AT91_REG *) 	0xFFFCC248) // (PWMC_CH2) Channel Period Register
-#define AT91C_PWMC_CH2_CUPDR (AT91_CAST(AT91_REG *) 	0xFFFCC250) // (PWMC_CH2) Channel Update Register
-#define AT91C_PWMC_CH2_CDTYR (AT91_CAST(AT91_REG *) 	0xFFFCC244) // (PWMC_CH2) Channel Duty Cycle Register
+#define AT91C_PWMC_CH2_Reserved (AT91_CAST(AT91_REG *)  0xFFFCC254) // (PWMC_CH2) Reserved
+#define AT91C_PWMC_CH2_CMR (AT91_CAST(AT91_REG *)   0xFFFCC240) // (PWMC_CH2) Channel Mode Register
+#define AT91C_PWMC_CH2_CCNTR (AT91_CAST(AT91_REG *)     0xFFFCC24C) // (PWMC_CH2) Channel Counter Register
+#define AT91C_PWMC_CH2_CPRDR (AT91_CAST(AT91_REG *)     0xFFFCC248) // (PWMC_CH2) Channel Period Register
+#define AT91C_PWMC_CH2_CUPDR (AT91_CAST(AT91_REG *)     0xFFFCC250) // (PWMC_CH2) Channel Update Register
+#define AT91C_PWMC_CH2_CDTYR (AT91_CAST(AT91_REG *)     0xFFFCC244) // (PWMC_CH2) Channel Duty Cycle Register
 // ========== Register definition for PWMC_CH1 peripheral ==========
-#define AT91C_PWMC_CH1_Reserved (AT91_CAST(AT91_REG *) 	0xFFFCC234) // (PWMC_CH1) Reserved
-#define AT91C_PWMC_CH1_CUPDR (AT91_CAST(AT91_REG *) 	0xFFFCC230) // (PWMC_CH1) Channel Update Register
-#define AT91C_PWMC_CH1_CPRDR (AT91_CAST(AT91_REG *) 	0xFFFCC228) // (PWMC_CH1) Channel Period Register
-#define AT91C_PWMC_CH1_CCNTR (AT91_CAST(AT91_REG *) 	0xFFFCC22C) // (PWMC_CH1) Channel Counter Register
-#define AT91C_PWMC_CH1_CDTYR (AT91_CAST(AT91_REG *) 	0xFFFCC224) // (PWMC_CH1) Channel Duty Cycle Register
-#define AT91C_PWMC_CH1_CMR (AT91_CAST(AT91_REG *) 	0xFFFCC220) // (PWMC_CH1) Channel Mode Register
+#define AT91C_PWMC_CH1_Reserved (AT91_CAST(AT91_REG *)  0xFFFCC234) // (PWMC_CH1) Reserved
+#define AT91C_PWMC_CH1_CUPDR (AT91_CAST(AT91_REG *)     0xFFFCC230) // (PWMC_CH1) Channel Update Register
+#define AT91C_PWMC_CH1_CPRDR (AT91_CAST(AT91_REG *)     0xFFFCC228) // (PWMC_CH1) Channel Period Register
+#define AT91C_PWMC_CH1_CCNTR (AT91_CAST(AT91_REG *)     0xFFFCC22C) // (PWMC_CH1) Channel Counter Register
+#define AT91C_PWMC_CH1_CDTYR (AT91_CAST(AT91_REG *)     0xFFFCC224) // (PWMC_CH1) Channel Duty Cycle Register
+#define AT91C_PWMC_CH1_CMR (AT91_CAST(AT91_REG *)   0xFFFCC220) // (PWMC_CH1) Channel Mode Register
 // ========== Register definition for PWMC_CH0 peripheral ==========
-#define AT91C_PWMC_CH0_Reserved (AT91_CAST(AT91_REG *) 	0xFFFCC214) // (PWMC_CH0) Reserved
-#define AT91C_PWMC_CH0_CPRDR (AT91_CAST(AT91_REG *) 	0xFFFCC208) // (PWMC_CH0) Channel Period Register
-#define AT91C_PWMC_CH0_CDTYR (AT91_CAST(AT91_REG *) 	0xFFFCC204) // (PWMC_CH0) Channel Duty Cycle Register
-#define AT91C_PWMC_CH0_CMR (AT91_CAST(AT91_REG *) 		0xFFFCC200) // (PWMC_CH0) Channel Mode Register
-#define AT91C_PWMC_CH0_CUPDR (AT91_CAST(AT91_REG *) 	0xFFFCC210) // (PWMC_CH0) Channel Update Register
-#define AT91C_PWMC_CH0_CCNTR (AT91_CAST(AT91_REG *) 	0xFFFCC20C) // (PWMC_CH0) Channel Counter Register
+#define AT91C_PWMC_CH0_Reserved (AT91_CAST(AT91_REG *)  0xFFFCC214) // (PWMC_CH0) Reserved
+#define AT91C_PWMC_CH0_CPRDR (AT91_CAST(AT91_REG *)     0xFFFCC208) // (PWMC_CH0) Channel Period Register
+#define AT91C_PWMC_CH0_CDTYR (AT91_CAST(AT91_REG *)     0xFFFCC204) // (PWMC_CH0) Channel Duty Cycle Register
+#define AT91C_PWMC_CH0_CMR (AT91_CAST(AT91_REG *)       0xFFFCC200) // (PWMC_CH0) Channel Mode Register
+#define AT91C_PWMC_CH0_CUPDR (AT91_CAST(AT91_REG *)     0xFFFCC210) // (PWMC_CH0) Channel Update Register
+#define AT91C_PWMC_CH0_CCNTR (AT91_CAST(AT91_REG *)     0xFFFCC20C) // (PWMC_CH0) Channel Counter Register
 // ========== Register definition for PWMC peripheral ==========
-#define AT91C_PWMC_IDR  (AT91_CAST(AT91_REG *) 	0xFFFCC014) // (PWMC) PWMC Interrupt Disable Register
-#define AT91C_PWMC_DIS  (AT91_CAST(AT91_REG *) 	0xFFFCC008) // (PWMC) PWMC Disable Register
-#define AT91C_PWMC_IER  (AT91_CAST(AT91_REG *) 	0xFFFCC010) // (PWMC) PWMC Interrupt Enable Register
-#define AT91C_PWMC_VR   (AT91_CAST(AT91_REG *) 	0xFFFCC0FC) // (PWMC) PWMC Version Register
-#define AT91C_PWMC_ISR  (AT91_CAST(AT91_REG *) 	0xFFFCC01C) // (PWMC) PWMC Interrupt Status Register
-#define AT91C_PWMC_SR   (AT91_CAST(AT91_REG *) 	0xFFFCC00C) // (PWMC) PWMC Status Register
-#define AT91C_PWMC_IMR  (AT91_CAST(AT91_REG *) 	0xFFFCC018) // (PWMC) PWMC Interrupt Mask Register
-#define AT91C_PWMC_MR   (AT91_CAST(AT91_REG *) 	0xFFFCC000) // (PWMC) PWMC Mode Register
-#define AT91C_PWMC_ENA  (AT91_CAST(AT91_REG *) 	0xFFFCC004) // (PWMC) PWMC Enable Register
+#define AT91C_PWMC_IDR  (AT91_CAST(AT91_REG *)  0xFFFCC014) // (PWMC) PWMC Interrupt Disable Register
+#define AT91C_PWMC_DIS  (AT91_CAST(AT91_REG *)  0xFFFCC008) // (PWMC) PWMC Disable Register
+#define AT91C_PWMC_IER  (AT91_CAST(AT91_REG *)  0xFFFCC010) // (PWMC) PWMC Interrupt Enable Register
+#define AT91C_PWMC_VR   (AT91_CAST(AT91_REG *)  0xFFFCC0FC) // (PWMC) PWMC Version Register
+#define AT91C_PWMC_ISR  (AT91_CAST(AT91_REG *)  0xFFFCC01C) // (PWMC) PWMC Interrupt Status Register
+#define AT91C_PWMC_SR   (AT91_CAST(AT91_REG *)  0xFFFCC00C) // (PWMC) PWMC Status Register
+#define AT91C_PWMC_IMR  (AT91_CAST(AT91_REG *)  0xFFFCC018) // (PWMC) PWMC Interrupt Mask Register
+#define AT91C_PWMC_MR   (AT91_CAST(AT91_REG *)  0xFFFCC000) // (PWMC) PWMC Mode Register
+#define AT91C_PWMC_ENA  (AT91_CAST(AT91_REG *)  0xFFFCC004) // (PWMC) PWMC Enable Register
 // ========== Register definition for UDP peripheral ==========
-#define AT91C_UDP_IMR   (AT91_CAST(AT91_REG *) 	0xFFFB0018) // (UDP) Interrupt Mask Register
-#define AT91C_UDP_FADDR (AT91_CAST(AT91_REG *) 	0xFFFB0008) // (UDP) Function Address Register
-#define AT91C_UDP_NUM   (AT91_CAST(AT91_REG *) 	0xFFFB0000) // (UDP) Frame Number Register
-#define AT91C_UDP_FDR   (AT91_CAST(AT91_REG *) 	0xFFFB0050) // (UDP) Endpoint FIFO Data Register
-#define AT91C_UDP_ISR   (AT91_CAST(AT91_REG *) 	0xFFFB001C) // (UDP) Interrupt Status Register
-#define AT91C_UDP_CSR   (AT91_CAST(AT91_REG *) 	0xFFFB0030) // (UDP) Endpoint Control and Status Register
-#define AT91C_UDP_IDR   (AT91_CAST(AT91_REG *) 	0xFFFB0014) // (UDP) Interrupt Disable Register
-#define AT91C_UDP_ICR   (AT91_CAST(AT91_REG *) 	0xFFFB0020) // (UDP) Interrupt Clear Register
-#define AT91C_UDP_RSTEP (AT91_CAST(AT91_REG *) 	0xFFFB0028) // (UDP) Reset Endpoint Register
-#define AT91C_UDP_TXVC  (AT91_CAST(AT91_REG *) 	0xFFFB0074) // (UDP) Transceiver Control Register
-#define AT91C_UDP_GLBSTATE (AT91_CAST(AT91_REG *) 	0xFFFB0004) // (UDP) Global State Register
-#define AT91C_UDP_IER   (AT91_CAST(AT91_REG *) 	0xFFFB0010) // (UDP) Interrupt Enable Register
+#define AT91C_UDP_IMR   (AT91_CAST(AT91_REG *)  0xFFFB0018) // (UDP) Interrupt Mask Register
+#define AT91C_UDP_FADDR (AT91_CAST(AT91_REG *)  0xFFFB0008) // (UDP) Function Address Register
+#define AT91C_UDP_NUM   (AT91_CAST(AT91_REG *)  0xFFFB0000) // (UDP) Frame Number Register
+#define AT91C_UDP_FDR   (AT91_CAST(AT91_REG *)  0xFFFB0050) // (UDP) Endpoint FIFO Data Register
+#define AT91C_UDP_ISR   (AT91_CAST(AT91_REG *)  0xFFFB001C) // (UDP) Interrupt Status Register
+#define AT91C_UDP_CSR   (AT91_CAST(AT91_REG *)  0xFFFB0030) // (UDP) Endpoint Control and Status Register
+#define AT91C_UDP_IDR   (AT91_CAST(AT91_REG *)  0xFFFB0014) // (UDP) Interrupt Disable Register
+#define AT91C_UDP_ICR   (AT91_CAST(AT91_REG *)  0xFFFB0020) // (UDP) Interrupt Clear Register
+#define AT91C_UDP_RSTEP (AT91_CAST(AT91_REG *)  0xFFFB0028) // (UDP) Reset Endpoint Register
+#define AT91C_UDP_TXVC  (AT91_CAST(AT91_REG *)  0xFFFB0074) // (UDP) Transceiver Control Register
+#define AT91C_UDP_GLBSTATE (AT91_CAST(AT91_REG *)   0xFFFB0004) // (UDP) Global State Register
+#define AT91C_UDP_IER   (AT91_CAST(AT91_REG *)  0xFFFB0010) // (UDP) Interrupt Enable Register
 
 // *****************************************************************************
 //               PIO DEFINITIONS FOR AT91SAM7S512
@@ -2242,55 +2242,55 @@ typedef struct _AT91S_UDP {
 // *****************************************************************************
 //               BASE ADDRESS DEFINITIONS FOR AT91SAM7S512
 // *****************************************************************************
-#define AT91C_BASE_SYS       (AT91_CAST(AT91PS_SYS) 	0xFFFFF000) // (SYS) Base Address
-#define AT91C_BASE_AIC       (AT91_CAST(AT91PS_AIC) 	0xFFFFF000) // (AIC) Base Address
-#define AT91C_BASE_PDC_DBGU  (AT91_CAST(AT91PS_PDC) 	0xFFFFF300) // (PDC_DBGU) Base Address
-#define AT91C_BASE_DBGU      (AT91_CAST(AT91PS_DBGU) 	0xFFFFF200) // (DBGU) Base Address
-#define AT91C_BASE_PIOA      (AT91_CAST(AT91PS_PIO) 	0xFFFFF400) // (PIOA) Base Address
-#define AT91C_BASE_CKGR      (AT91_CAST(AT91PS_CKGR) 	0xFFFFFC20) // (CKGR) Base Address
-#define AT91C_BASE_PMC       (AT91_CAST(AT91PS_PMC) 	0xFFFFFC00) // (PMC) Base Address
-#define AT91C_BASE_RSTC      (AT91_CAST(AT91PS_RSTC) 	0xFFFFFD00) // (RSTC) Base Address
-#define AT91C_BASE_RTTC      (AT91_CAST(AT91PS_RTTC) 	0xFFFFFD20) // (RTTC) Base Address
-#define AT91C_BASE_PITC      (AT91_CAST(AT91PS_PITC) 	0xFFFFFD30) // (PITC) Base Address
-#define AT91C_BASE_WDTC      (AT91_CAST(AT91PS_WDTC) 	0xFFFFFD40) // (WDTC) Base Address
-#define AT91C_BASE_VREG      (AT91_CAST(AT91PS_VREG) 	0xFFFFFD60) // (VREG) Base Address
-#define AT91C_BASE_EFC0      (AT91_CAST(AT91PS_EFC) 	0xFFFFFF60) // (EFC0) Base Address
-#define AT91C_BASE_EFC1      (AT91_CAST(AT91PS_EFC) 	0xFFFFFF70) // (EFC1) Base Address
-#define AT91C_BASE_MC        (AT91_CAST(AT91PS_MC) 		0xFFFFFF00) // (MC) Base Address
-#define AT91C_BASE_PDC_SPI   (AT91_CAST(AT91PS_PDC) 	0xFFFE0100) // (PDC_SPI) Base Address
-#define AT91C_BASE_SPI       (AT91_CAST(AT91PS_SPI) 	0xFFFE0000) // (SPI) Base Address
-#define AT91C_BASE_PDC_ADC   (AT91_CAST(AT91PS_PDC) 	0xFFFD8100) // (PDC_ADC) Base Address
-#define AT91C_BASE_ADC       (AT91_CAST(AT91PS_ADC) 	0xFFFD8000) // (ADC) Base Address
-#define AT91C_BASE_PDC_SSC   (AT91_CAST(AT91PS_PDC) 	0xFFFD4100) // (PDC_SSC) Base Address
-#define AT91C_BASE_SSC       (AT91_CAST(AT91PS_SSC) 	0xFFFD4000) // (SSC) Base Address
-#define AT91C_BASE_PDC_US1   (AT91_CAST(AT91PS_PDC) 	0xFFFC4100) // (PDC_US1) Base Address
-#define AT91C_BASE_US1       (AT91_CAST(AT91PS_USART) 	0xFFFC4000) // (US1) Base Address
-#define AT91C_BASE_PDC_US0   (AT91_CAST(AT91PS_PDC) 	0xFFFC0100) // (PDC_US0) Base Address
-#define AT91C_BASE_US0       (AT91_CAST(AT91PS_USART) 	0xFFFC0000) // (US0) Base Address
-#define AT91C_BASE_TWI       (AT91_CAST(AT91PS_TWI) 	0xFFFB8000) // (TWI) Base Address
-#define AT91C_BASE_TC0       (AT91_CAST(AT91PS_TC) 		0xFFFA0000) // (TC0) Base Address
-#define AT91C_BASE_TC1       (AT91_CAST(AT91PS_TC) 		0xFFFA0040) // (TC1) Base Address
-#define AT91C_BASE_TC2       (AT91_CAST(AT91PS_TC) 		0xFFFA0080) // (TC2) Base Address
-#define AT91C_BASE_TCB       (AT91_CAST(AT91PS_TCB) 	0xFFFA0000) // (TCB) Base Address
+#define AT91C_BASE_SYS       (AT91_CAST(AT91PS_SYS)     0xFFFFF000) // (SYS) Base Address
+#define AT91C_BASE_AIC       (AT91_CAST(AT91PS_AIC)     0xFFFFF000) // (AIC) Base Address
+#define AT91C_BASE_PDC_DBGU  (AT91_CAST(AT91PS_PDC)     0xFFFFF300) // (PDC_DBGU) Base Address
+#define AT91C_BASE_DBGU      (AT91_CAST(AT91PS_DBGU)    0xFFFFF200) // (DBGU) Base Address
+#define AT91C_BASE_PIOA      (AT91_CAST(AT91PS_PIO)     0xFFFFF400) // (PIOA) Base Address
+#define AT91C_BASE_CKGR      (AT91_CAST(AT91PS_CKGR)    0xFFFFFC20) // (CKGR) Base Address
+#define AT91C_BASE_PMC       (AT91_CAST(AT91PS_PMC)     0xFFFFFC00) // (PMC) Base Address
+#define AT91C_BASE_RSTC      (AT91_CAST(AT91PS_RSTC)    0xFFFFFD00) // (RSTC) Base Address
+#define AT91C_BASE_RTTC      (AT91_CAST(AT91PS_RTTC)    0xFFFFFD20) // (RTTC) Base Address
+#define AT91C_BASE_PITC      (AT91_CAST(AT91PS_PITC)    0xFFFFFD30) // (PITC) Base Address
+#define AT91C_BASE_WDTC      (AT91_CAST(AT91PS_WDTC)    0xFFFFFD40) // (WDTC) Base Address
+#define AT91C_BASE_VREG      (AT91_CAST(AT91PS_VREG)    0xFFFFFD60) // (VREG) Base Address
+#define AT91C_BASE_EFC0      (AT91_CAST(AT91PS_EFC)     0xFFFFFF60) // (EFC0) Base Address
+#define AT91C_BASE_EFC1      (AT91_CAST(AT91PS_EFC)     0xFFFFFF70) // (EFC1) Base Address
+#define AT91C_BASE_MC        (AT91_CAST(AT91PS_MC)      0xFFFFFF00) // (MC) Base Address
+#define AT91C_BASE_PDC_SPI   (AT91_CAST(AT91PS_PDC)     0xFFFE0100) // (PDC_SPI) Base Address
+#define AT91C_BASE_SPI       (AT91_CAST(AT91PS_SPI)     0xFFFE0000) // (SPI) Base Address
+#define AT91C_BASE_PDC_ADC   (AT91_CAST(AT91PS_PDC)     0xFFFD8100) // (PDC_ADC) Base Address
+#define AT91C_BASE_ADC       (AT91_CAST(AT91PS_ADC)     0xFFFD8000) // (ADC) Base Address
+#define AT91C_BASE_PDC_SSC   (AT91_CAST(AT91PS_PDC)     0xFFFD4100) // (PDC_SSC) Base Address
+#define AT91C_BASE_SSC       (AT91_CAST(AT91PS_SSC)     0xFFFD4000) // (SSC) Base Address
+#define AT91C_BASE_PDC_US1   (AT91_CAST(AT91PS_PDC)     0xFFFC4100) // (PDC_US1) Base Address
+#define AT91C_BASE_US1       (AT91_CAST(AT91PS_USART)   0xFFFC4000) // (US1) Base Address
+#define AT91C_BASE_PDC_US0   (AT91_CAST(AT91PS_PDC)     0xFFFC0100) // (PDC_US0) Base Address
+#define AT91C_BASE_US0       (AT91_CAST(AT91PS_USART)   0xFFFC0000) // (US0) Base Address
+#define AT91C_BASE_TWI       (AT91_CAST(AT91PS_TWI)     0xFFFB8000) // (TWI) Base Address
+#define AT91C_BASE_TC0       (AT91_CAST(AT91PS_TC)      0xFFFA0000) // (TC0) Base Address
+#define AT91C_BASE_TC1       (AT91_CAST(AT91PS_TC)      0xFFFA0040) // (TC1) Base Address
+#define AT91C_BASE_TC2       (AT91_CAST(AT91PS_TC)      0xFFFA0080) // (TC2) Base Address
+#define AT91C_BASE_TCB       (AT91_CAST(AT91PS_TCB)     0xFFFA0000) // (TCB) Base Address
 #define AT91C_BASE_PWMC_CH3  (AT91_CAST(AT91PS_PWMC_CH) 0xFFFCC260) // (PWMC_CH3) Base Address
 #define AT91C_BASE_PWMC_CH2  (AT91_CAST(AT91PS_PWMC_CH) 0xFFFCC240) // (PWMC_CH2) Base Address
 #define AT91C_BASE_PWMC_CH1  (AT91_CAST(AT91PS_PWMC_CH) 0xFFFCC220) // (PWMC_CH1) Base Address
 #define AT91C_BASE_PWMC_CH0  (AT91_CAST(AT91PS_PWMC_CH) 0xFFFCC200) // (PWMC_CH0) Base Address
-#define AT91C_BASE_PWMC      (AT91_CAST(AT91PS_PWMC) 	0xFFFCC000) // (PWMC) Base Address
-#define AT91C_BASE_UDP       (AT91_CAST(AT91PS_UDP) 	0xFFFB0000) // (UDP) Base Address
+#define AT91C_BASE_PWMC      (AT91_CAST(AT91PS_PWMC)    0xFFFCC000) // (PWMC) Base Address
+#define AT91C_BASE_UDP       (AT91_CAST(AT91PS_UDP)     0xFFFB0000) // (UDP) Base Address
 
 // *****************************************************************************
 //               MEMORY MAPPING DEFINITIONS FOR AT91SAM7S512
 // *****************************************************************************
 // ISRAM
-#define AT91C_ISRAM	 (0x00200000) // Internal SRAM base address
-#define AT91C_ISRAM_SIZE	 (0x00010000) // Internal SRAM size in byte (64 Kbytes)
+#define AT91C_ISRAM  (0x00200000) // Internal SRAM base address
+#define AT91C_ISRAM_SIZE     (0x00010000) // Internal SRAM size in byte (64 Kbytes)
 // IFLASH
-#define AT91C_IFLASH	 (0x00100000) // Internal FLASH base address
-#define AT91C_IFLASH_SIZE	 (0x00080000) // Internal FLASH size in byte (512 Kbytes)
-#define AT91C_IFLASH_PAGE_SIZE	 (256) // Internal FLASH Page Size: 256 bytes
-#define AT91C_IFLASH_LOCK_REGION_SIZE	 (16384) // Internal FLASH Lock Region Size: 16 Kbytes
-#define AT91C_IFLASH_NB_OF_PAGES	 (2048) // Internal FLASH Number of Pages: 2048 bytes
-#define AT91C_IFLASH_NB_OF_LOCK_BITS	 (32) // Internal FLASH Number of Lock Bits: 32 bytes
+#define AT91C_IFLASH     (0x00100000) // Internal FLASH base address
+#define AT91C_IFLASH_SIZE    (0x00080000) // Internal FLASH size in byte (512 Kbytes)
+#define AT91C_IFLASH_PAGE_SIZE   (256) // Internal FLASH Page Size: 256 bytes
+#define AT91C_IFLASH_LOCK_REGION_SIZE    (16384) // Internal FLASH Lock Region Size: 16 Kbytes
+#define AT91C_IFLASH_NB_OF_PAGES     (2048) // Internal FLASH Number of Pages: 2048 bytes
+#define AT91C_IFLASH_NB_OF_LOCK_BITS     (32) // Internal FLASH Number of Lock Bits: 32 bytes
 
 #endif
diff --git a/include/common.h b/include/common.h
index 3dbf77622..dec2261ec 100644
--- a/include/common.h
+++ b/include/common.h
@@ -19,9 +19,20 @@ extern "C" {
 #include <stddef.h>
 #include <stdint.h>
 #include <stdbool.h>
-#include <at91sam7s512.h>
+
 typedef unsigned char byte_t;
 
+#ifdef _MSC_VER
+typedef DWORD uint32_t;
+typedef BYTE uint8_t;
+#define PACKED
+// stuff
+#else
+#include <stdint.h>
+#include <stdbool.h>
+#define PACKED __attribute__((packed))
+#endif
+
 // debug
 #define DBG_NONE          0 // no messages
 #define DBG_ERROR         1 // errors only
@@ -30,9 +41,6 @@ typedef unsigned char byte_t;
 #define DBG_EXTENDED      4 // errors + info + debug + breaking debug messages
 extern int DBGLEVEL;
 
-// Flashmem spi baudrate
-extern uint32_t FLASHMEM_SPIBAUDRATE;
-
 // reader voltage field detector
 #define MF_MINFIELDV      4000
 
@@ -49,74 +57,87 @@ extern uint32_t FLASHMEM_SPIBAUDRATE;
 #endif
 #define RAMFUNC __attribute((long_call, section(".ramfunc")))
 
-// RDV40 Section
-// 256kb divided into 4k sectors.
-//
-// 0x3F000 - 1 4kb sector = signature
-// 0x3E000 - 1 4kb sector = settings
-// 0x3D000 - 1 4kb sector = default T55XX keys dictionary
-// 0x3B000 - 1 4kb sector = default ICLASS keys dictionary
-// 0x39000 - 2 4kb sectors = default MFC keys dictionary
-//
-#ifndef FLASH_MEM_BLOCK_SIZE
-# define FLASH_MEM_BLOCK_SIZE   256
+#ifndef ROTR
+# define ROTR(x,n) (((uintmax_t)(x) >> (n)) | ((uintmax_t)(x) << ((sizeof(x) * 8) - (n))))
 #endif
 
-#ifndef FLASH_MEM_MAX_SIZE
-# define FLASH_MEM_MAX_SIZE     0x40000  // (262144)
+#ifndef ROTL
+# define ROTL(x,n) (((uintmax_t)(x) << (n)) | ((uintmax_t)(x) >> ((sizeof(x) * 8) - (n))))
 #endif
 
-#ifndef FLASH_MEM_MAX_4K_SECTOR
-# define FLASH_MEM_MAX_4K_SECTOR   0x3F000
+// endian change for 64bit
+#ifdef __GNUC__
+#ifndef BSWAP_64
+#define BSWAP_64(x) __builtin_bswap64(x)
+#endif
+#else
+#ifdef _MSC_VER
+#ifndef BSWAP_64
+#define BSWAP_64(x) _byteswap_uint64(x)
+#endif
+#else
+#ifndef BSWAP_64
+#define BSWAP_64(x) \
+    (((uint64_t)(x) << 56) | \
+     (((uint64_t)(x) << 40) & 0xff000000000000ULL) | \
+     (((uint64_t)(x) << 24) & 0xff0000000000ULL) | \
+     (((uint64_t)(x) << 8)  & 0xff00000000ULL) | \
+     (((uint64_t)(x) >> 8)  & 0xff000000ULL) | \
+     (((uint64_t)(x) >> 24) & 0xff0000ULL) | \
+     (((uint64_t)(x) >> 40) & 0xff00ULL) | \
+     ((uint64_t)(x)  >> 56))
+#endif
+#endif
 #endif
 
-
-#ifndef FLASH_MEM_ID_LEN
-# define FLASH_MEM_ID_LEN 8
+// endian change for 32bit
+#ifdef __GNUC__
+#ifndef BSWAP_32
+#define BSWAP_32(x) __builtin_bswap32(x)
+#endif
+#else
+#ifdef _MSC_VER
+#ifndef BSWAP_32
+#define BSWAP_32(x) _byteswap_ulong(x)
+#endif
+#else
+#ifndef BSWAP_32
+# define BSWAP_32(x) \
+    ((((x) & 0xff000000) >> 24) | (((x) & 0x00ff0000) >>  8) | \
+     (((x) & 0x0000ff00) <<  8) | (((x) & 0x000000ff) << 24))
+#endif
+#endif
 #endif
 
-#ifndef FLASH_MEM_SIGNATURE_LEN
-# define FLASH_MEM_SIGNATURE_LEN 128
+#define EVEN                        0
+#define ODD                         1
+
+// Nibble logic
+#ifndef NIBBLE_HIGH
+# define NIBBLE_HIGH(b) ( (b & 0xF0) >> 4 )
 #endif
 
-#ifndef FLASH_MEM_SIGNATURE_OFFSET
-// -1 for historical compatibility with already released Proxmark3 RDV4.0 devices
-# define FLASH_MEM_SIGNATURE_OFFSET (FLASH_MEM_MAX_SIZE - FLASH_MEM_SIGNATURE_LEN - 1)
+#ifndef NIBBLE_LOW
+# define NIBBLE_LOW(b)  ( b & 0x0F )
 #endif
 
-#ifndef T55XX_CONFIG_LEN
-# define T55XX_CONFIG_LEN sizeof( t55xx_config )
+#ifndef CRUMB
+# define CRUMB(b,p)    (((b & (0x3 << p) ) >> p ) & 0xF)
 #endif
 
-#ifndef T55XX_CONFIG_OFFSET
-# define T55XX_CONFIG_OFFSET (FLASH_MEM_MAX_4K_SECTOR - 0x2000)
+#ifndef SWAP_NIBBLE
+# define SWAP_NIBBLE(b)  ( (NIBBLE_LOW(b)<< 4) | NIBBLE_HIGH(b))
 #endif
 
-// Reserved space for T55XX PWD = 4 kb
-#ifndef DEFAULT_T55XX_KEYS_OFFSET
-# define DEFAULT_T55XX_KEYS_OFFSET (FLASH_MEM_MAX_4K_SECTOR - 0x3000)
+// Binary Encoded Digit
+#ifndef BCD2DEC
+# define BCD2DEC(bcd) HornerScheme(bcd, 0x10, 10)
 #endif
 
-// Reserved space for iClass keys = 4 kb
-#ifndef DEFAULT_ICLASS_KEYS_OFFSET
-# define DEFAULT_ICLASS_KEYS_OFFSET (FLASH_MEM_MAX_4K_SECTOR - 0x4000)
+#ifndef DEC2BCD
+# define DEC2BCD(dec) HornerScheme(dec, 10, 0x10)
 #endif
 
-// Reserved space for MIFARE Keys = 8 kb
-#ifndef DEFAULT_MF_KEYS_OFFSET
-# define DEFAULT_MF_KEYS_OFFSET (FLASH_MEM_MAX_4K_SECTOR - 0x6000)
-#endif
-
-
-
-// RDV40,  validation structure to help identifying that client/firmware is talking with RDV40
-typedef struct {
-    uint8_t magic[4];
-    uint8_t flashid[FLASH_MEM_ID_LEN];
-    uint8_t signature[FLASH_MEM_SIGNATURE_LEN];
-} __attribute__((__packed__)) rdv40_validation_t;
-
-
 #ifdef __cplusplus
 }
 #endif
diff --git a/include/hitag.h b/include/hitag.h
index f57f1180a..9df93fcf7 100644
--- a/include/hitag.h
+++ b/include/hitag.h
@@ -14,11 +14,7 @@
 #ifndef HITAG_H__
 #define HITAG_H__
 
-#ifdef _MSC_VER
-#define PACKED
-#else
-#define PACKED __attribute__((packed))
-#endif
+#include "common.h"
 
 typedef enum {
     RHTSF_CHALLENGE           = 01,
diff --git a/include/mifare.h b/include/mifare.h
index 87ba3a74e..9d444e833 100644
--- a/include/mifare.h
+++ b/include/mifare.h
@@ -46,7 +46,7 @@ typedef struct {
     uint8_t sak;
     uint8_t ats_len;
     uint8_t ats[256];
-} __attribute__((__packed__)) iso14a_card_select_t;
+} PACKED iso14a_card_select_t;
 
 typedef enum ISO14A_COMMAND {
     ISO14A_CONNECT = (1 << 0),
@@ -129,7 +129,7 @@ typedef struct {
 typedef struct {
     uint8_t atr_len;
     uint8_t atr[30];
-} __attribute__((__packed__)) smart_card_atr_t;
+} PACKED smart_card_atr_t;
 
 typedef enum SMARTCARD_COMMAND {
     SC_CONNECT = (1 << 0),
@@ -167,7 +167,7 @@ typedef struct {
     uint8_t iccode[2];
     uint8_t mrt[6];
     uint8_t servicecode[2];
-} __attribute__((__packed__)) felica_card_select_t;
+} PACKED felica_card_select_t;
 
 typedef enum FELICA_COMMAND {
     FELICA_CONNECT = (1 << 0),
diff --git a/include/pm3_cmd.h b/include/pm3_cmd.h
index 8eef7e75d..be5d75b21 100644
--- a/include/pm3_cmd.h
+++ b/include/pm3_cmd.h
@@ -13,20 +13,11 @@
 #ifndef __PM3_CMD_H
 #define __PM3_CMD_H
 
+#include "common.h"
+
 // Use it e.g. when using slow links such as BT
 #define USART_SLOW_LINK
 
-#ifdef _MSC_VER
-typedef DWORD uint32_t;
-typedef BYTE uint8_t;
-#define PACKED
-// stuff
-#else
-#include <stdint.h>
-#include <stdbool.h>
-#define PACKED __attribute__((packed))
-#endif
-
 #define PM3_CMD_DATA_SIZE 512
 #define PM3_CMD_DATA_SIZE_MIX ( PM3_CMD_DATA_SIZE - 3 * sizeof(uint64_t) )
 
@@ -128,7 +119,7 @@ typedef struct {
     int divisor;
     int trigger_threshold;
 } sample_config;
-
+/*
 typedef struct {
     uint16_t start_gap;
     uint16_t write_gap;
@@ -136,7 +127,35 @@ typedef struct {
     uint16_t write_1;
     uint16_t read_gap;
 } t55xx_config;
+*/
 
+// Extended to support 1 of 4 timing
+typedef struct  {
+    uint16_t start_gap ;
+    uint16_t write_gap ;
+    uint16_t write_0   ;
+    uint16_t write_1   ;
+    uint16_t read_gap  ;
+    uint16_t write_2   ;
+    uint16_t write_3   ;
+
+} t55xx_config_t;
+// This setup will allow for the 4 downlink modes "m" as well as other items if needed.
+// Given the one struct we can then read/write to flash/client in one go.
+typedef struct {
+    t55xx_config_t m[4]; // mode
+} t55xx_config;
+
+/*typedef struct  {
+    uint16_t start_gap [4];
+    uint16_t write_gap [4];
+    uint16_t write_0   [4];
+    uint16_t write_1   [4];
+    uint16_t write_2   [4];
+    uint16_t write_3   [4];
+    uint16_t read_gap  [4];
+} t55xx_config;
+*/
 typedef struct {
     uint8_t version;
     uint32_t baudrate;
@@ -217,6 +236,7 @@ typedef struct {
 #define CMD_HARDWARE_RESET                                                0x0004
 #define CMD_START_FLASH                                                   0x0005
 #define CMD_CHIP_INFO                                                     0x0006
+#define CMD_BL_VERSION                                                    0x0007
 #define CMD_NACK                                                          0x00fe
 #define CMD_ACK                                                           0x00ff
 
@@ -236,9 +256,9 @@ typedef struct {
 #define CMD_CAPABILITIES                                                  0x0112
 #define CMD_QUIT_SESSION                                                  0x0113
 #define CMD_SET_DBGMODE                                                   0x0114
+#define CMD_STANDALONE                                                    0x0115
 
 // RDV40, Flash memory operations
-#define CMD_FLASHMEM_READ                                                 0x0120
 #define CMD_FLASHMEM_WRITE                                                0x0121
 #define CMD_FLASHMEM_WIPE                                                 0x0122
 #define CMD_FLASHMEM_DOWNLOAD                                             0x0123
@@ -246,6 +266,40 @@ typedef struct {
 #define CMD_FLASHMEM_INFO                                                 0x0125
 #define CMD_FLASHMEM_SET_SPIBAUDRATE                                      0x0126
 
+// RDV40, High level flashmem SPIFFS Manipulation
+// ALL function will have a lazy or Safe version
+// that will be handled as argument of safety level [0..2] respectiveley normal / lazy / safe
+// However as how design is, MOUNT and UNMOUNT only need/have lazy as safest level so a safe level will still execute a lazy version
+// see spiffs.c for more about the normal/lazy/safety information)
+#define CMD_SPIFFS_MOUNT                                                  0x0130
+#define CMD_SPIFFS_UNMOUNT                                                0x0131
+#define CMD_SPIFFS_WRITE                                                  0x0132
+// We take +0x1000 when having a variant of similar function (todo : make it an argument!)
+#define CMD_SPIFFS_APPEND                                                 0x1132
+
+#define CMD_SPIFFS_READ                                                   0x0133
+//We use no open/close instruvtion, as they are handled internally.
+#define CMD_SPIFFS_REMOVE                                                 0x0134
+#define CMD_SPIFFS_RM                                                     CMD_SPIFFS_REMOVE
+#define CMD_SPIFFS_RENAME                                                 0x0135
+#define CMD_SPIFFS_MV                                                     CMD_SPIFFS_RENAME
+#define CMD_SPIFFS_COPY                                                   0x0136
+#define CMD_SPIFFS_CP                                                     CMD_SPIFFS_COPY
+#define CMD_SPIFFS_STAT                                                   0x0137
+#define CMD_SPIFFS_FSTAT                                                  0x0138
+#define CMD_SPIFFS_INFO                                                   0x0139
+#define CMD_SPIFFS_FORMAT                                                 CMD_FLASHMEM_WIPE
+// This take a +0x2000 as they are high level helper and special functions
+// As the others, they may have safety level argument if it makkes sense
+#define CMD_SPIFFS_PRINT_TREE                                             0x2130
+#define CMD_SPIFFS_GET_TREE                                               0x2131
+#define CMD_SPIFFS_TEST                                                   0x2132
+#define CMD_SPIFFS_PRINT_FSINFO                                           0x2133
+#define CMD_SPIFFS_DOWNLOAD                                               0x2134
+#define CMD_SPIFFS_DOWNLOADED                                             0x2135
+// more ?
+
+
 // RDV40,  Smart card operations
 #define CMD_SMART_RAW                                                     0x0140
 #define CMD_SMART_UPGRADE                                                 0x0141
@@ -545,6 +599,21 @@ typedef struct {
 /* Set if this device understands the chip info command */
 #define DEVICE_INFO_FLAG_UNDERSTANDS_CHIP_INFO       (1<<5)
 
+/* Set if this device understands the version command */
+#define DEVICE_INFO_FLAG_UNDERSTANDS_VERSION         (1<<6)
+
+#define BL_VERSION_MAJOR(version) ((uint32_t)(version) >> 22)
+#define BL_VERSION_MINOR(version) (((uint32_t)(version) >> 12) & 0x3ff)
+#define BL_VERSION_PATCH(version) ((uint32_t)(version) & 0xfff)
+#define BL_MAKE_VERSION(major, minor, patch) (((major) << 22) | ((minor) << 12) | (patch))
+// Some boundaries to distinguish valid versions from corrupted info
+#define BL_VERSION_FIRST_MAJOR    1
+#define BL_VERSION_LAST_MAJOR     99
+#define BL_VERSION_INVALID  0
+// Different versions here. Each version should increase the numbers
+#define BL_VERSION_1_0_0    BL_MAKE_VERSION(1, 0, 0)
+
+
 /* CMD_START_FLASH may have three arguments: start of area to flash,
    end of area to flash, optional magic.
    The bootrom will not allow to overwrite itself unless this magic
diff --git a/include/pmflash.h b/include/pmflash.h
new file mode 100644
index 000000000..b7bf39ddc
--- /dev/null
+++ b/include/pmflash.h
@@ -0,0 +1,97 @@
+//-----------------------------------------------------------------------------
+// (c) RFID Research Group - 2019
+//
+// This code is licensed to you under the terms of the GNU GPL, version 2 or,
+// at your option, any later version. See the LICENSE.txt file for the text of
+// the license.
+//
+//-----------------------------------------------------------------------------
+// RDV4 flash constants
+//-----------------------------------------------------------------------------
+
+#ifndef __PMFLASH_H
+#define __PMFLASH_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#include <stddef.h>
+#include <stdint.h>
+#include <stdbool.h>
+
+#include "common.h"
+
+// Flashmem spi baudrate
+extern uint32_t FLASHMEM_SPIBAUDRATE;
+
+// RDV40 Section
+// 256kb divided into 4k sectors.
+//
+// 0x3F000 - 1 4kb sector = signature
+// 0x3E000 - 1 4kb sector = settings
+// 0x3D000 - 1 4kb sector = default T55XX keys dictionary
+// 0x3B000 - 1 4kb sector = default ICLASS keys dictionary
+// 0x39000 - 2 4kb sectors = default MFC keys dictionary
+//
+#ifndef FLASH_MEM_BLOCK_SIZE
+# define FLASH_MEM_BLOCK_SIZE   256
+#endif
+
+#ifndef FLASH_MEM_MAX_SIZE
+# define FLASH_MEM_MAX_SIZE     0x40000  // (262144)
+#endif
+
+#ifndef FLASH_MEM_MAX_4K_SECTOR
+# define FLASH_MEM_MAX_4K_SECTOR   0x3F000
+#endif
+
+
+#ifndef FLASH_MEM_ID_LEN
+# define FLASH_MEM_ID_LEN 8
+#endif
+
+#ifndef FLASH_MEM_SIGNATURE_LEN
+# define FLASH_MEM_SIGNATURE_LEN 128
+#endif
+
+#ifndef FLASH_MEM_SIGNATURE_OFFSET
+// -1 for historical compatibility with already released Proxmark3 RDV4.0 devices
+# define FLASH_MEM_SIGNATURE_OFFSET (FLASH_MEM_MAX_SIZE - FLASH_MEM_SIGNATURE_LEN - 1)
+#endif
+
+#ifndef T55XX_CONFIG_LEN
+# define T55XX_CONFIG_LEN sizeof( t55xx_config )
+#endif
+
+#ifndef T55XX_CONFIG_OFFSET
+# define T55XX_CONFIG_OFFSET (FLASH_MEM_MAX_4K_SECTOR - 0x2000)
+#endif
+
+// Reserved space for T55XX PWD = 4 kb
+#ifndef DEFAULT_T55XX_KEYS_OFFSET
+# define DEFAULT_T55XX_KEYS_OFFSET (FLASH_MEM_MAX_4K_SECTOR - 0x3000)
+#endif
+
+// Reserved space for iClass keys = 4 kb
+#ifndef DEFAULT_ICLASS_KEYS_OFFSET
+# define DEFAULT_ICLASS_KEYS_OFFSET (FLASH_MEM_MAX_4K_SECTOR - 0x4000)
+#endif
+
+// Reserved space for MIFARE Keys = 8 kb
+#ifndef DEFAULT_MF_KEYS_OFFSET
+# define DEFAULT_MF_KEYS_OFFSET (FLASH_MEM_MAX_4K_SECTOR - 0x6000)
+#endif
+
+// RDV40,  validation structure to help identifying that client/firmware is talking with RDV40
+typedef struct {
+    uint8_t magic[4];
+    uint8_t flashid[FLASH_MEM_ID_LEN];
+    uint8_t signature[FLASH_MEM_SIGNATURE_LEN];
+} PACKED rdv40_validation_t;
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif // __PMFLASH_H
diff --git a/include/proxmark3.h b/include/proxmark3.h
index 8748e82d8..469713b5b 100644
--- a/include/proxmark3.h
+++ b/include/proxmark3.h
@@ -15,6 +15,7 @@
 #include "at91sam7s512.h"
 #include "config_gpio.h"
 #include "pm3_cmd.h"
+#include "common.h"
 
 #define WDT_HIT()                               AT91C_BASE_WDTC->WDTC_WDCR = 0xa5000001
 
@@ -71,8 +72,6 @@
 #define COTAG_BITS 264
 #endif
 
-//#define PACKED __attribute__((__packed__))
-
 #define LED_A_ON()        HIGH(GPIO_LED_A)
 #define LED_A_OFF()       LOW(GPIO_LED_A)
 #define LED_A_INV()       INVBIT(GPIO_LED_A)
@@ -116,7 +115,7 @@ struct version_information {
     char clean; /* 1: Tree was clean, no local changes. 0: Tree was unclean. 2: Couldn't be determined */
     char gitversion[50]; /* String with the git revision */
     char buildtime[30]; /* string with the build time */
-} __attribute__((packed));
+} PACKED;
 
 #define COMMON_AREA_MAGIC 0x43334d50 // "PM3C"
 #define COMMON_AREA_COMMAND_NONE 0
@@ -129,8 +128,8 @@ struct common_area {
         unsigned int bootrom_present: 1; /* Set when a bootrom that is capable of parsing the common area is present */
         unsigned int osimage_present: 1; /* Set when a osimage that is capable of parsing the common area is present */
         unsigned int button_pressed: 1;
-    } __attribute__((packed)) flags;
+    } PACKED flags;
     int arg1, arg2;
-} __attribute__((packed));
+} PACKED;
 
 #endif
diff --git a/proxmark3.sh b/proxmark3.sh
index b4aa51ffb..4cd6d76bc 100755
--- a/proxmark3.sh
+++ b/proxmark3.sh
@@ -3,29 +3,31 @@
 FULLIMAGE="armsrc/obj/fullimage.elf"
 BOOTIMAGE="bootrom/obj/bootrom.elf"
 
-cd $(dirname "$0")
+PM3PATH=$(dirname "$0")
+cd "$PM3PATH" || exit 1
 
 function wait4proxmark_Linux {
     echo >&2 "Waiting for Proxmark to appear..."
-    while [ ! -c /dev/ttyACM? -a ! -c /dev/pm3-? ]; do
+    while true; do
+        PM3=$(find /dev/pm3-* /dev/ttyACM* 2>/dev/null | head -1)
+        if [[ $PM3 != "" ]]; then
+            break
+        fi
         sleep .1
     done
-    local PM3=`ls -1 /dev/pm3-? /dev/ttyACM? 2>/dev/null | head -1`
-    echo >&2 -e "Found proxmark on ${PM3}\n"
-    echo $PM3
+    echo "$PM3"
 }
 
 function wait4proxmark_macOS {
     echo >&2 "Waiting for Proxmark to appear..."
     while true; do
-        PM3=$(ls /dev/pm3-* /dev/cu.usbmodem* 2>/dev/null | head -1)
+        PM3=$(find /dev/pm3-* /dev/cu.usbmodem* 2>/dev/null | head -1)
         if [[ $PM3 != "" ]]; then
-            #echo >&2 -e "Found proxmark on $(ls /dev/pm3-* /dev/cu.usbmodem* 2>/dev/null | head -1)\n"
             break
         fi
         sleep .1
     done
-    echo $PM3
+    echo "$PM3"
 }
 
 function wait4proxmark_Windows {
@@ -38,30 +40,48 @@ function wait4proxmark_Windows {
         fi
         sleep .1
     done
-    echo $PM3
+    echo "$PM3"
+}
+
+function wait4proxmark_WSL {
+    echo >&2 "Waiting for Proxmark to appear..."
+    while true; do
+        device=$(wmic.exe path Win32_SerialPort where "PNPDeviceID like '%VID_9AC4&PID_4B8F%'" get DeviceID,PNPDeviceID 2>/dev/null | awk 'NR==2')
+        if [[ $device != "" ]]; then
+            PM3=${device/ */}
+            PM3="/dev/ttyS${PM3#COM}"
+            break
+        fi
+        sleep .1
+    done
+    if [ -e "$PM3" ] && [ ! -w "$PM3" ]; then
+        echo "We need to give current user read/write access to $PM3"
+        sudo chmod 666 "$PM3"
+    fi
+    echo "$PM3"
 }
 
 SCRIPT=$(basename -- "$0")
 
 if [ "$SCRIPT" = "proxmark3.sh" ]; then
-  CMD=client/proxmark3
+  CMD() { client/proxmark3 "$@"; }
 elif [ "$SCRIPT" = "flash-all.sh" ]; then
-  CMD=client/flasher
-  ARG1="-b $BOOTIMAGE"
-  ARG2="$FULLIMAGE"
+  CMD() { client/flasher "$1" -b "$BOOTIMAGE" "$FULLIMAGE"; }
 elif [ "$SCRIPT" = "flash-fullimage.sh" ]; then
-  CMD=client/flasher
-  ARG2="$FULLIMAGE"
+  CMD() { client/flasher "$1" "$FULLIMAGE"; }
 elif [ "$SCRIPT" = "flash-bootrom.sh" ]; then
-  CMD=client/flasher
-  ARG1="-b $BOOTIMAGE"
+  CMD() { client/flasher "$1" -b "$BOOTIMAGE"; }
 else
   echo "Script ran under unknown name, abort: $SCRIPT"
   exit 1
 fi
 HOSTOS=$(uname | awk '{print toupper($0)}')
 if [ "$HOSTOS" = "LINUX" ]; then
-    PORT=$(wait4proxmark_Linux)
+    if uname -a|grep -q Microsoft; then
+        PORT=$(wait4proxmark_WSL)
+    else
+        PORT=$(wait4proxmark_Linux)
+    fi
 elif [ "$HOSTOS" = "DARWIN" ]; then
     PORT=$(wait4proxmark_macOS)
 elif [[ "$HOSTOS" =~ MINGW(32|64)_NT* ]]; then
@@ -75,6 +95,5 @@ if [ "$PORT" = "" ]; then
     exit 1
 fi
 
-#echo Running "$CMD" "$PORT" $ARG1 $ARG2 "$@"
-"$CMD" "$PORT" $ARG1 $ARG2 "$@"
+CMD "$PORT" "$@"
 exit $?
diff --git a/tools/analyzesize.py b/tools/analyzesize.py
new file mode 100755
index 000000000..4a5211c67
--- /dev/null
+++ b/tools/analyzesize.py
@@ -0,0 +1,33 @@
+#! /usr/bin/python3
+
+import json
+import subprocess
+import sys
+
+def print_increase(x, y, name):
+    if x > y:
+        print("{} increase by: {} (0x{:08X}) bytes ({}%)".format(name, x-y, x-y, (x-y)*100/y))
+    else:
+        print("{} decrease by: {} (0x{:08X}) bytes ({}%)".format(name, y-x, y-x, (y-x)*100/x))
+dbname = "tools/data.json"
+try:
+    db = json.load(open(dbname,"r"))
+except FileNotFoundError:
+    db = dict()
+
+if len(sys.argv) < 3:
+    print("Usage: analazysize.py <info|add|diff> <datasetname>")
+    exit(-1)
+action, name = sys.argv[1:3]
+currentdata = subprocess.run(["arm-none-eabi-size","armsrc/obj/fullimage.stage1.elf"], stdout=subprocess.PIPE).stdout
+currentdata = currentdata.split(b"\n")[1].strip()
+text,data,bss = [int(x) for x in currentdata.split(b"\t")[:3]]
+if action.lower() == "add":
+    db[name] = [text, data, bss]
+    json.dump(db, open(dbname, "w"))
+elif action.lower() == "diff":
+    text_ref, data_ref, bss_ref = db[name]
+    flash_ref = text_ref+data_ref
+    flash = text+data
+    print_increase(flash, flash_ref, "Flash")
+    print_increase(bss, bss_ref, "RAM")
diff --git a/tools/jtag_openocd/chip-at91sam7s.cfg b/tools/jtag_openocd/chip-at91sam7s.cfg
index a0787770a..7c546e45a 100644
--- a/tools/jtag_openocd/chip-at91sam7s.cfg
+++ b/tools/jtag_openocd/chip-at91sam7s.cfg
@@ -7,18 +7,18 @@ jtag newtap sam7x cpu -irlen 4 -ircapture 0x1 -irmask 0xf -expected-id 0x3f0f0f0
 target create sam7x.cpu arm7tdmi -endian little -chain-position sam7x.cpu
 
 sam7x.cpu configure -event reset-init {
-	soft_reset_halt
-	mww 0xfffffd00 0xa5000004	# RSTC_CR: Reset peripherals
-	mww 0xfffffd44 0x00008000	# WDT_MR: disable watchdog
-	mww 0xfffffd08 0xa5000001	# RSTC_MR enable user reset
-	mww 0xfffffc20 0x00005001	# CKGR_MOR : enable the main oscillator
-	sleep 10
-	mww 0xfffffc2c 0x000b1c02	# CKGR_PLLR: 16MHz * 12/2 = 96MHz
-	sleep 10
-	mww 0xfffffc30 0x00000007	# PMC_MCKR : MCK = PLL / 2 = 48 MHz
-	sleep 10
-	mww 0xffffff60 0x00480100	# MC_FMR: flash mode (FWS=1,FMCN=72)
-	sleep 100
+    soft_reset_halt
+    mww 0xfffffd00 0xa5000004   # RSTC_CR: Reset peripherals
+    mww 0xfffffd44 0x00008000   # WDT_MR: disable watchdog
+    mww 0xfffffd08 0xa5000001   # RSTC_MR enable user reset
+    mww 0xfffffc20 0x00005001   # CKGR_MOR : enable the main oscillator
+    sleep 10
+    mww 0xfffffc2c 0x000b1c02   # CKGR_PLLR: 16MHz * 12/2 = 96MHz
+    sleep 10
+    mww 0xfffffc30 0x00000007   # PMC_MCKR : MCK = PLL / 2 = 48 MHz
+    sleep 10
+    mww 0xffffff60 0x00480100   # MC_FMR: flash mode (FWS=1,FMCN=72)
+    sleep 100
 }
 
 gdb_memory_map enable
diff --git a/tools/mfkey/example_trace.txt b/tools/mfkey/example_trace.txt
index 694a738f6..a236706b9 100644
--- a/tools/mfkey/example_trace.txt
+++ b/tools/mfkey/example_trace.txt
@@ -35,25 +35,25 @@
 New functionality from @zhovner,  
 -----------------------------------------------------------------------------------------------------
 ### Communication decryption
-RDR	26
-TAG	04 00
-RDR	93 20
-TAG	14 57 9f 69 b5
-RDR	93 70 14 57 9f 69 b5 2e 51
-TAG	08 b6 dd
-RDR	60 14 50 2d
-TAG	ce 84 42 61
-RDR	f8 04 9c cb 05 25 c8 4f
-TAG	94 31 cc 40
-RDR	70 93 df 99
-TAG	99 72 42 8c e2 e8 52 3f 45 6b 99 c8 31 e7 69 dc ed 09
-RDR	8c a6 82 7b
-TAG	ab 79 7f d3 69 e8 b9 3a 86 77 6b 40 da e3 ef 68 6e fd
-RDR	c3 c3 81 ba
-TAG	49 e2 c9 de f4 86 8d 17 77 67 0e 58 4c 27 23 02 86 f4
-RDR	fb dc d7 c1
-TAG	4a bd 96 4b 07 d3 56 3a a0 66 ed 0a 2e ac 7f 63 12 bf
-RDR	9f 91 49 ea
+RDR 26
+TAG 04 00
+RDR 93 20
+TAG 14 57 9f 69 b5
+RDR 93 70 14 57 9f 69 b5 2e 51
+TAG 08 b6 dd
+RDR 60 14 50 2d
+TAG ce 84 42 61
+RDR f8 04 9c cb 05 25 c8 4f
+TAG 94 31 cc 40
+RDR 70 93 df 99
+TAG 99 72 42 8c e2 e8 52 3f 45 6b 99 c8 31 e7 69 dc ed 09
+RDR 8c a6 82 7b
+TAG ab 79 7f d3 69 e8 b9 3a 86 77 6b 40 da e3 ef 68 6e fd
+RDR c3 c3 81 ba
+TAG 49 e2 c9 de f4 86 8d 17 77 67 0e 58 4c 27 23 02 86 f4
+RDR fb dc d7 c1
+TAG 4a bd 96 4b 07 d3 56 3a a0 66 ed 0a 2e ac 7f 63 12 bf
+RDR 9f 91 49 ea
 
 
 ./mfkey64 14579f69 ce844261 f8049ccb 0525c84f 9431cc40 7093df99 9972428ce2e8523f456b99c831e769dced09 8ca6827b ab797fd369e8b93a86776b40dae3ef686efd c3c381ba 49e2c9def4868d1777670e584c27230286f4 fbdcd7c1 4abd964b07d3563aa066ed0a2eac7f6312bf 9f9149ea
@@ -93,4 +93,4 @@ Decrypted communication:
 {dec7}: 0000000000007e178869000000000000c4f2
 {dec8}: 61148834
 
-Found Key: [091e639cb715]
\ No newline at end of file
+Found Key: [091e639cb715]
diff --git a/uart/uart_posix.c b/uart/uart_posix.c
index 93b7229ce..ea535fd73 100644
--- a/uart/uart_posix.c
+++ b/uart/uart_posix.c
@@ -238,7 +238,8 @@ void uart_close(const serial_port sp) {
     // Does the system allows us to place a lock on this file descriptor
     int err = fcntl(spu->fd, F_SETLK, &fl);
     if (err == -1) {
-        //perror("fcntl");
+        //silent error message as it can be called from uart_open failing modes, e.g. when waiting for port to appear
+        //printf("[!] UART error while closing port\n");
     }
     close(spu->fd);
     free(sp);