Skip to content
Snippets Groups Projects
dfu.c 3.63 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 "usb_dev.h"
    #include "serial.h"
    
    
    static dfu_state_t dfu_state = dfuIDLE;
    static dfu_status_t dfu_status = OK;
    static unsigned dfu_poll_timeout = 1;
    
    
    static uint32_t debug_a, debug_b, debug_c;
    
    // 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;
    }
    
    static bool ftfl_busy()
    {
    	// Is the flash memory controller busy?
    	return 0 == (FTFL_FSTAT_CCIF & FTFL_FSTAT);
    }
    
    static void ftfl_busy_wait()
    {
    	// 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
    	
    	ftfl_busy_wait();
    	FTFL_FCCOB0 = 0x81;
    	FTFL_FCCOB1 = control_code;
    	ftfl_launch_command();
    	ftfl_busy_wait();
    }
    
    	// Use FlexRAM (dfu_buffer) as normal RAM.
    	ftfl_set_flexram_function(0xFF);
    
    
        serial_begin(BAUD2DIV(115200));
        serial_print("Hello from DFU!\n");
    }
    
    
    void dfu_debug()
    {
    	serial_phex32(debug_a);
    	serial_putchar(' ');
    	serial_phex32(debug_b);
    	serial_putchar(' ');
    	serial_phex32(debug_c);
    	serial_putchar('\n');
    }
    
    void 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;
    	}
    
    	debug_a = blockNum;
    	debug_b = packetOffset;
    	debug_c = blockLength;
    
    	memcpy(dfu_buffer + packetOffset, data, packetLength);
    
    	if (packetOffset + packetLength == blockLength) {
    		// Full block received
    		dfu_state = dfuDNLOAD_SYNC;
    		dfu_status = OK;
    	}
    
    }
    
    void dfu_getstatus(uint8_t *status)
    {
    
    	status[0] = dfu_status;
    	status[1] = dfu_poll_timeout >> 16;
    	status[2] = dfu_poll_timeout >> 8;
    	status[3] = dfu_poll_timeout;
    	status[4] = dfu_state;
    	status[5] = 0;  // iString
    
    }
    
    void dfu_clrstatus()
    {
    
    	if (dfu_state == dfuERROR) {
    		dfu_state = dfuIDLE;
    	}
    
    }
    
    uint8_t dfu_getstate()
    {