// Copyright 2011 Olivier Gillet.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with this program.  If not, see .
//
// -----------------------------------------------------------------------------
//
// Bootloader supporting MIDI SysEx update.
//
// Caveat: assumes the firmware flashing is always done from first to last
// block, in increasing order. Random access flashing is not supported!
#include "avrlibx/io/gpio.h"
#include "avrlibx/io/usart.h"
#include "avrlibx/system/init.h"
#include "avrlibx/system/time.h"
#include "avrlibx/third_party/sp_driver/sp_driver.h"
#include "edges/hardware_config.h"
using namespace avrlibx;
using namespace edges;
MidiIO midi;
Leds leds;
Switches switches;
uint16_t page = 0;
uint8_t rx_buffer[APP_SECTION_PAGE_SIZE + 1];
typedef void (*MainEntryPoint)(void) __attribute__ ((noreturn));
MainEntryPoint main_entry_point = (MainEntryPoint)0x0000;
inline void Init() {
  SysInit();
  leds.set_direction(OUTPUT);
  switches.set_direction(INPUT);
  switches.set_mode(PORT_MODE_PULL_UP);
  midi.Init();
}
void WriteBufferToFlash() {
  SP_LoadFlashPage(rx_buffer);
  SP_EraseWriteApplicationPage(page);
  SP_WaitForSPM();
  
  NVM_CMD = NVM_CMD_NO_OPERATION_gc;
}
void FlashLedsOk() {
  for (uint8_t i = 0; i < 8; ++i) {
    leds.Write(0xf);
    ConstantDelay(50);
    leds.Write(8);
    ConstantDelay(50);
  }
}
void FlashLedsError() {
  leds.Write(0);
  for (uint8_t i = 0; i < 5; ++i) {
    ConstantDelay(100);
    leds.Toggle();
  }
}
static const uint8_t sysex_header[] = {
  0xf0,  // 
  0x00, 0x21, 0x02,  // Mutable instruments manufacturer id.
  0x00, 0x0a,  // Product ID for "any product".
};
enum SysExReceptionState {
  MATCHING_HEADER = 0,
  READING_COMMAND = 1,
  READING_DATA = 2,
};
inline void MidiLoop() {
  uint8_t byte;
  uint16_t bytes_read = 0;
  uint16_t rx_buffer_index;
  uint8_t state = MATCHING_HEADER;
  uint8_t checksum;
  uint8_t sysex_commands[2];
  uint8_t page_byte = 0;
  midi.Init();
  page = 0;
  
  while (1) {
    leds.Write(8 | (4 >> ((page_byte + 3) & 0x3)));
    byte = midi.Read();
    // In case we see a realtime message in the stream, safely ignore it.
    if (byte > 0xf0 && byte != 0xf7) {
      continue;
    }
    switch (state) {
      case MATCHING_HEADER:
        if (byte == sysex_header[bytes_read]) {
          ++bytes_read;
          if (bytes_read == sizeof(sysex_header)) {
            bytes_read = 0;
            state = READING_COMMAND;
          }
        } else {
          bytes_read = 0;
        }
        break;
      case READING_COMMAND:
        if (byte < 0x80) {
          sysex_commands[bytes_read++] = byte;
          if (bytes_read == 2) {
            bytes_read = 0;
            rx_buffer_index = 0;
            checksum = 0;
            state = READING_DATA;
          }
        } else {
          state = MATCHING_HEADER;
          bytes_read = 0;
        }
        break;
      case READING_DATA:
        if (byte < 0x80) {
          if (bytes_read & 1) {
            rx_buffer[rx_buffer_index] |= byte & 0xf;
            if (rx_buffer_index < APP_SECTION_PAGE_SIZE) {
              checksum += rx_buffer[rx_buffer_index];
            }
            ++rx_buffer_index;
          } else {
            rx_buffer[rx_buffer_index] = (byte << 4);
          }
          ++bytes_read;
        } else if (byte == 0xf7) {
          if (sysex_commands[0] == 0x7f &&
              sysex_commands[1] == 0x00 &&
              bytes_read == 0) {
            // Reset.
            return;
          } else if (rx_buffer_index == APP_SECTION_PAGE_SIZE + 1 &&
                     sysex_commands[0] == 0x7e &&
                     sysex_commands[1] == 0x00 &&
                     rx_buffer[rx_buffer_index - 1] == checksum) {
            // Block write.
            WriteBufferToFlash();
            page += APP_SECTION_PAGE_SIZE;
            ++page_byte;
          } else {
            FlashLedsError();
          }
          state = MATCHING_HEADER;
          bytes_read = 0;
        }
        break;
    }
  }
}
int main(void) {
  Init();
  ConstantDelay(25);
  if (!(switches.Read() & 1)) {
    FlashLedsOk();
    MidiLoop();
    FlashLedsOk();
  }
  SP_LockSPM();
  EIND = 0x00;
  main_entry_point();
}