Skip to content
Snippets Groups Projects
arm_kinetis_debug.cpp 5.98 KiB
Newer Older
  • Learn to ignore specific revisions
  • /*
     * Simple ARM debug interface for Arduino, using the SWD (Serial Wire Debug) port.
     * Extensions for Freescale Kinetis chips.
     * 
     * 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 <Arduino.h>
    #include "arm_kinetis_debug.h"
    #include "arm_kinetis_reg.h"
    
    
    bool ARMKinetisDebug::startup()
    {
        uint32_t idr;
        uint32_t status;
    
        // Make sure we're on a compatible chip. The MDM-AP peripheral is Freescale-specific.
    
        if (!apRead(REG_MDM_IDR, idr))
    
            return false;
        if (idr != 0x001C0000) {
            log(LOG_ERROR, "ARMKinetisDebug: Didn't find a supported MDM-AP peripheral");
            return false;
        }
    
        // Reset the system, and hold the core in reset when it comes back
    
        if (!apWrite(REG_MDM_CONTROL, REG_MDM_CONTROL_SYS_RESET_REQ | REG_MDM_CONTROL_CORE_HOLD_RESET))
    
        if (!apReadPoll(REG_MDM_STATUS, status, REG_MDM_STATUS_SYS_NRESET, 0))
    
        if (!apWrite(REG_MDM_CONTROL, REG_MDM_CONTROL_CORE_HOLD_RESET))
    
            return false;
    
        // Wait until the flash controller is ready & system is out of reset
    
        if (!apReadPoll(REG_MDM_STATUS, status, REG_MDM_STATUS_SYS_NRESET | REG_MDM_STATUS_FLASH_READY, -1))
            return false;
    
        // Re-initialize the AHB-AP after reset
        if (!initMemPort())
            return false;
    
        // Test AHB-AP: Can we successfully write to RAM?
        if (!memStoreAndVerify(0x20000000, 0x31415927))
            return false;
        if (!memStoreAndVerify(0x20000000, 0x76543210))
    
            return false;
    
        /*
         * The rest of system startup looks a lot like what the bootloader does. This is based
         * on the startup sequence used by Teensyduino.
         */
    
        // Enable peripheral clocks
        if (!memStore(REG_SIM_SCGC5, 0x00043F82))
            return false;
        if (!memStore(REG_SIM_SCGC6, REG_SIM_SCGC6_FTM0 | REG_SIM_SCGC6_FTM1 | REG_SIM_SCGC6_FTFL))
            return false;
    
        // Oscillator starts up in FEI mode.
        // Turn on crystal capacitors
        if (!memStore(REG_OSC0_CR, REG_OSC_SC8P | REG_OSC_SC2P))
            return false;
        // Enable osc, 8-32 MHz range, low power
        if (!memStore(REG_MCG_C2, REG_MCG_C2_RANGE0(2) | REG_MCG_C2_EREFS))
            return false;
        // Switch to crystal as clock source, FLL input = 16 MHz / 512
        if (!memStore(REG_MCG_C1, REG_MCG_C1_CLKS(2) | REG_MCG_C1_FRDIV(4)))
            return false;
    
        // Wait for crystal oscillator to begin
        uint32_t mcg;
        if (!memPoll(REG_MCG_S, mcg, REG_MCG_S_OSCINIT0, -1))
            return false;
    
        // wait for FLL to use oscillator
        if (!memPoll(REG_MCG_S, mcg, REG_MCG_S_IREFST, 0))
            return false;
    
        // wait for MCGOUT to use oscillator
        if (!memPoll(REG_MCG_S, mcg, REG_MCG_S_CLKST_MASK, REG_MCG_S_CLKST(2)))
            return false;
    
        // Now we're in FBE mode
        // config PLL input for 16 MHz Crystal / 4 = 4 MHz
        if (!memStore(REG_MCG_C5, REG_MCG_C5_PRDIV0(3)))
            return false;
    
        // config PLL for 96 MHz output
        if (!memStore(REG_MCG_C6, REG_MCG_C6_PLLS | REG_MCG_C6_VDIV0(0)))
            return false;
    
        // wait for PLL to start using xtal as its input
        if (!memPoll(REG_MCG_S, mcg, REG_MCG_S_PLLST, -1))
            return false;
        if (!memPoll(REG_MCG_S, mcg, REG_MCG_S_LOCK0, -1))
            return false;
    
        // Now we're in PBE mode
        // config divisors: 48 MHz core, 48 MHz bus, 24 MHz flash
        if (!memStore(REG_SIM_CLKDIV1, REG_SIM_CLKDIV1_OUTDIV1(1) | REG_SIM_CLKDIV1_OUTDIV2(1) | REG_SIM_CLKDIV1_OUTDIV4(3)))
            return false;
        // switch to PLL as clock source, FLL input = 16 MHz / 512
        if (!memStore(REG_MCG_C1, REG_MCG_C1_CLKS(0) | REG_MCG_C1_FRDIV(4)))
            return false;
    
        // wait for PLL clock to be used
        if (!memPoll(REG_MCG_S, mcg, MCG_S_CLKST_MASK, MCG_S_CLKST(3)))
            return false;
    
        // Now we're in PEE mode. Ready to go!
        return true;
    }
    
    bool ARMKinetisDebug::flashMassErase()
    {
        // Erase all flash, even if some of it is protected.
    
        uint32_t status;
    
        if (!apRead(REG_MDM_STATUS, status))
    
        if (!(status & REG_MDM_STATUS_FLASH_READY)) {
    
            log(LOG_ERROR, "FLASH: Flash controller not ready before mass erase");
            return false;
        }
    
        if ((status & REG_MDM_STATUS_FLASH_ERASE_ACK)) {
    
            log(LOG_ERROR, "FLASH: Mass erase already in progress");
            return false;
        }
    
        if (!(status & REG_MDM_STATUS_MASS_ERASE_ENABLE)) {
            log(LOG_ERROR, "FLASH: Mass erase is disabled!");
            return false;
        }
    
    
        log(LOG_NORMAL, "FLASH: Beginning mass erase operation");
    
        if (!apWrite(REG_MDM_CONTROL, REG_MDM_CONTROL_CORE_HOLD_RESET | REG_MDM_CONTROL_MASS_ERASE))
    
            return false;
    
        // Wait for the mass erase to complete
    
        if (!apReadPoll(REG_MDM_STATUS, status, REG_MDM_STATUS_FLASH_ERASE_ACK, 0, 10000)) {
    
            log(LOG_ERROR, "FLASH: Timed out waiting for mass erase to complete");
            return false;
        }
    
    
        if (!(status & REG_MDM_STATUS_FLASH_READY)) {
    
            log(LOG_ERROR, "FLASH: Flash controller not ready after mass erase");
            return false;
        }
    
        return true;
    }