Skip to content
Snippets Groups Projects
dfu.c 7.8 KiB
Newer Older
  • Learn to ignore specific revisions
  • /*
     * Fadecandy DFU Bootloader
     * 
     * Copyright (c) 2013 Micah Elizabeth Scott
     * 
     * Permission is hereby granted, free of charge, to any person obtaining a copy of
     * this software and associated documentation files (the "Software"), to deal in
     * the Software without restriction, including without limitation the rights to
     * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
     * the Software, and to permit persons to whom the Software is furnished to do so,
     * subject to the following conditions:
     * 
     * The above copyright notice and this permission notice shall be included in all
     * copies or substantial portions of the Software.
     * 
     * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
     * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
     * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
     * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
     * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
     * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
     */
    
    
    #include "mk20dx128.h"
    
    #include "usb_dev.h"
    
    // Internal flash-programming state machine
    static unsigned fl_current_addr = 0;
    static enum {
    
        flsIDLE = 0,
        flsERASING,
        flsPROGRAMMING
    
    static dfu_state_t dfu_state = dfuIDLE;
    static dfu_status_t dfu_status = OK;
    static unsigned dfu_poll_timeout = 1;
    
    
    // Programming buffer in MK20DX128 FlexRAM, where the flash controller can quickly access it.
    static __attribute__ ((section(".flexram"))) uint8_t dfu_buffer[DFU_TRANSFER_SIZE];
    
    static void *memcpy(void *dst, const void *src, size_t cnt) {
    
        uint8_t *dst8 = dst;
        const uint8_t *src8 = src;
        while (cnt > 0) {
            cnt--;
            *(dst8++) = *(src8++);
        }
        return dst;
    
        // Is the flash memory controller busy?
        return 0 == (FTFL_FSTAT_CCIF & FTFL_FSTAT);
    
        // Wait for the flash memory controller to finish any pending operation.
        while (ftfl_busy());
    
    }
    
    static void ftfl_launch_command()
    {
    
        // Begin a flash memory controller command
        FTFL_FSTAT = FTFL_FSTAT_ACCERR | FTFL_FSTAT_FPVIOL | FTFL_FSTAT_RDCOLERR;
        FTFL_FSTAT = FTFL_FSTAT_CCIF;
    
    }
    
    static void ftfl_set_flexram_function(uint8_t control_code)
    {
    
        // Issue a Set FlexRAM Function command. Busy-waits until the command is done.
        
        ftfl_busy_wait();
        FTFL_FCCOB0 = 0x81;
        FTFL_FCCOB1 = control_code;
        ftfl_launch_command();
        ftfl_busy_wait();
    
    static void ftfl_begin_erase_sector(uint32_t address)
    {
    
        FTFL_FCCOB0 = 0x09;
        FTFL_FCCOB1 = address >> 16;
        FTFL_FCCOB2 = address >> 8;
        FTFL_FCCOB3 = address;
        ftfl_launch_command();
    
    static void ftfl_begin_program_section(uint32_t address, uint32_t numLWords)
    {
    
        FTFL_FCCOB0 = 0x0B;
        FTFL_FCCOB1 = address >> 16;
        FTFL_FCCOB2 = address >> 8;
        FTFL_FCCOB3 = address;
        FTFL_FCCOB4 = numLWords >> 8;
        FTFL_FCCOB5 = numLWords;
        ftfl_launch_command();
    
    static uint32_t address_for_block(unsigned blockNum)
    {
    
        return 0x1000 + (blockNum << 10);
    
        // Use FlexRAM (dfu_buffer) as normal RAM.
        ftfl_set_flexram_function(0xFF);
    
    uint8_t dfu_getstate()
    
        return dfu_state;
    
    bool dfu_download(unsigned blockNum, unsigned blockLength,
    
        unsigned packetOffset, unsigned packetLength, const uint8_t *data)
    
        if (packetOffset + packetLength > DFU_TRANSFER_SIZE ||
            packetOffset + packetLength > blockLength) {
    
            // Overflow!
            dfu_state = dfuERROR;
            dfu_status = errADDRESS;
            return false;
        }
    
        // Store more data...
        memcpy(dfu_buffer + packetOffset, data, packetLength);
    
        if (packetOffset + packetLength != blockLength) {
            // Still waiting for more data.
            return true;
        }
    
        if (dfu_state != dfuIDLE && dfu_state != dfuDNLOAD_IDLE) {
            // Wrong state! Oops.
            dfu_state = dfuERROR;
            dfu_status = errSTALLEDPKT;
            return false;
        }
    
        if (ftfl_busy() || fl_state != flsIDLE) {
            // Flash controller shouldn't be busy now!
            dfu_state = dfuERROR;
            dfu_status = errUNKNOWN;
            return false;       
        }
    
        if (!blockLength) {
            // End of download
            dfu_state = dfuMANIFEST_SYNC;
            dfu_status = OK;
            return true;
        }
    
        // Start programming a block by erasing the corresponding flash sector
        fl_state = flsERASING;
        fl_current_addr = address_for_block(blockNum);
        ftfl_begin_erase_sector(fl_current_addr);
    
        dfu_state = dfuDNLOAD_SYNC;
        dfu_status = OK;
        return true;
    
    static bool fl_handle_status(uint8_t fstat, unsigned specificError)
    {
    
        /*
         * Handle common errors from an FSTAT register value.
         * The indicated "specificError" is used for reporting a command-specific
         * error from MGSTAT0.
         *
         * Returns true if handled, false if not.
         */
    
    
        if (0 == (fstat & FTFL_FSTAT_CCIF)) {
            // Still working...
            return true;
        }
    
        if (fstat & FTFL_FSTAT_RDCOLERR) {
            // Bus collision. We did something wrong internally.
            dfu_state = dfuERROR;
            dfu_status = errUNKNOWN;
            fl_state = flsIDLE;
            return true;
        }
    
        if (fstat & (FTFL_FSTAT_FPVIOL | FTFL_FSTAT_ACCERR)) {
            // Address or protection error
            dfu_state = dfuERROR;
            dfu_status = errADDRESS;
            fl_state = flsIDLE;
            return true;
        }
    
        if (fstat & FTFL_FSTAT_MGSTAT0) {
            // Command-specifid error
            dfu_state = dfuERROR;
            dfu_status = specificError;
            fl_state = flsIDLE;
            return true;
        }
    
        return false;
    
    }
    
    static void fl_state_poll()
    {
    
        // Try to advance the state of our own flash programming state machine.
    
        uint8_t fstat = FTFL_FSTAT;
        switch (fl_state) {
    
            case flsIDLE:
                break;
    
            case flsERASING:
                if (!fl_handle_status(fstat, errERASE)) {
                    // Done! Move on to programming the sector.
                    fl_state = flsPROGRAMMING;
                    ftfl_begin_program_section(fl_current_addr, DFU_TRANSFER_SIZE/4);
                }
                break;
    
            case flsPROGRAMMING:
                if (!fl_handle_status(fstat, errVERIFY)) {
                    // Done!
                    fl_state = flsIDLE;
                }
                break;
        }
    
    bool dfu_getstatus(uint8_t *status)
    
        switch (dfu_state) {
    
            case dfuDNLOAD_SYNC:
            case dfuDNBUSY:
                // Programming operation in progress. Advance our private flash state machine.
                fl_state_poll();
    
                if (dfu_state == dfuERROR) {
                    // An error occurred inside fl_state_poll();
                } else if (fl_state == flsIDLE) {
                    dfu_state = dfuDNLOAD_IDLE;
                } else {
                    dfu_state = dfuDNBUSY;
                }
                break;
    
            case dfuMANIFEST_SYNC:
                // Ready to reboot. The main thread will take care of this. Also let the DFU tool
                // know to leave us alone until this happens.
                dfu_state = dfuMANIFEST;
                dfu_poll_timeout = 1000;
                break;
    
            default:
                break;
        }
    
        status[0] = dfu_status;
        status[1] = dfu_poll_timeout;
        status[2] = dfu_poll_timeout >> 8;
        status[3] = dfu_poll_timeout >> 16;
        status[4] = dfu_state;
        status[5] = 0;  // iString
    
        return true;
    
    bool dfu_clrstatus()
    
        switch (dfu_state) {
    
            case dfuERROR:
                // Clear an error
                dfu_state = dfuIDLE;
                dfu_status = OK;
                return true;
    
            default:
                // Unexpected request
                dfu_state = dfuERROR;
                dfu_status = errSTALLEDPKT;
                return false;
        }
    
    bool dfu_abort()
    
        dfu_state = dfuIDLE;
        dfu_status = OK;
        return true;