Repostiory containing DAPLink source code with Reset Pin workaround for HANI_IOT board.

Upstream: https://github.com/ARMmbed/DAPLink

source/hic_hal/freescale/k26f/hic_init.c

Committer:
Pawel Zarembski
Date:
2020-04-07
Revision:
0:01f31e923fe2

File content as of revision 0:01f31e923fe2:

/**
 * @file    hic_init.c
 * @brief
 *
 * DAPLink Interface Firmware
 * Copyright (c) 2009-2016, ARM Limited, All Rights Reserved
 * Copyright (c) 2016-2017 NXP
 * SPDX-License-Identifier: Apache-2.0
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may
 * not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#include "hic_init.h"
#include "gpio.h"
#include "fsl_clock.h"
#include "usb_phy.h"
#include "util.h"

static void busy_wait(uint32_t cycles)
{
    volatile uint32_t i;
    i = cycles;

    while (i > 0) {
        i--;
    }
}

static void fll_delay(void)
{
    // ~2.5ms at 16MHz core clock
    busy_wait(10000);
}

// This IRQ handler will be invoked if VDD falls below the trip point.
void LVD_LVW_IRQHandler(void)
{
    if (PMC->LVDSC1 & PMC_LVDSC1_LVDF_MASK)
    {
        util_assert(false && "low voltage detect tripped");
        PMC->LVDSC1 |= PMC_LVDSC1_LVDACK_MASK;
    }
    if (PMC->LVDSC2 & PMC_LVDSC2_LVWF_MASK)
    {
        util_assert(false && "low voltage warning tripped");
        PMC->LVDSC2 |= PMC_LVDSC2_LVWACK_MASK;
    }
}

//! - MPU is disabled and gated.
//! - 8kB cache is enabled. SRAM is not cached, so no flushing is required for normal operation.
//! - Enable low voltage warning interrupt.
//! - Disable USB current limiter so the voltage doesn't drop as we enable high speed clocks.
void sdk_init(void)
{
    CLOCK_SetXtal0Freq(16000000U); // 16 MHz crystal
    CLOCK_SetXtal32Freq(0);

    // Disable the MPU if it's enabled.
    if (SIM->SCGC7 & SIM_SCGC7_MPU_MASK)
    {
        SYSMPU->CESR = 0;
        SIM->SCGC7 &= ~SIM_SCGC7_MPU_MASK;
    }

    // Invalidate and enable code cache.
    LMEM->PCCCR = LMEM_PCCCR_GO_MASK | LMEM_PCCCR_INVW1_MASK | LMEM_PCCCR_INVW0_MASK | LMEM_PCCCR_ENCACHE_MASK;

    // Enable LVD/LVW IRQ.
    PMC->LVDSC1 |= PMC_LVDSC1_LVDACK_MASK;
    PMC->LVDSC1 = PMC_LVDSC1_LVDIE_MASK | PMC_LVDSC1_LVDV(0); // low trip point
    PMC->LVDSC2 |= PMC_LVDSC2_LVWACK_MASK;
    PMC->LVDSC2 = PMC_LVDSC2_LVWIE_MASK | PMC_LVDSC2_LVWV(0); // low trip point
//     NVIC_EnableIRQ(LVD_LVW_IRQn);

    // Disable USB inrush current limiter.
    SIM->USBPHYCTL |= SIM_USBPHYCTL_USBDISILIM_MASK;
}

//! - Turn on 16MHz crystal oscillator.
//! - Turn on 32kHz IRC.
//! - Switch core clock to System PLL at 120 MHz, bus clock at 60 MHz, flash clock at 24 MHz.
//! - Enable the 480MHz USB PHY PLL.
//! - Ungate USBPHY and USBHS.
//! - Configure the USB PHY.
void hic_enable_usb_clocks(void)
{
    // Enable external oscillator and 32kHz IRC.
    MCG->C1 |= MCG_C1_IRCLKEN_MASK; // Select 32k IR.
    // Delay at least 100µs for 32kHz IRQ to stabilize.
    fll_delay();
    // Configure OSC for very high freq, low power mode.
    MCG->C2 = (MCG->C2 & ~(MCG_C2_RANGE_MASK | MCG_C2_HGO_MASK)) | MCG_C2_RANGE(2);
    OSC0->CR |= OSC_CR_ERCLKEN_MASK; // Enable OSC.
    MCG->C2 |= MCG_C2_EREFS_MASK; // Select OSC as ext ref.

    // Wait for the oscillator to stabilize.
    while (!(MCG->S & MCG_S_OSCINIT0_MASK))
    {
    }

    // Divide 16MHz xtal by 512 = 31.25kHz
    CLOCK_SetFbeMode(4, kMCG_Dmx32Default, kMCG_DrsMid, fll_delay);

    // Set dividers before switching to SYSPLL.
    SIM->CLKDIV1 = SIM_CLKDIV1_OUTDIV1(0)       // System/core  /1 = 120MHz
                    | SIM_CLKDIV1_OUTDIV2(1)    // Bus          /2 = 60Mhz
                    | SIM_CLKDIV1_OUTDIV3(4)    // FlexBus      /5 = 24Mhz
                    | SIM_CLKDIV1_OUTDIV4(4);   // Flash        /5 = 24MHz

    // 120MHz SYSPLL
    mcg_pll_config_t pllConfig;
    pllConfig.enableMode = 0;
    pllConfig.prdiv = 2 - 1;
    pllConfig.vdiv = 30 - 16;
    CLOCK_SetPbeMode(kMCG_PllClkSelPll0, &pllConfig);
    CLOCK_SetPeeMode();

    // Enable USB clock source and init phy. This turns on the 480MHz PLL.
    CLOCK_EnableUsbhs0Clock(kCLOCK_UsbSrcPll0, CLOCK_GetFreq(kCLOCK_PllFllSelClk));
    USB_EhciPhyInit(0, CPU_XTAL_CLK_HZ);

    SystemCoreClockUpdate();
}

void hic_power_target(void)
{
    // Keep powered off in bootloader mode
    // to prevent the target from effecting the state
    // of the reset line / reset button
    if (!daplink_is_bootloader()) {
        // configure pin as GPIO
        PIN_POWER_EN_PORT->PCR[PIN_POWER_EN_BIT] = PORT_PCR_MUX(1);
        // force always on logic 1
        PIN_POWER_EN_GPIO->PSOR = 1UL << PIN_POWER_EN_BIT;
        PIN_POWER_EN_GPIO->PDDR |= 1UL << PIN_POWER_EN_BIT;

        // Let the voltage rails stabilize.  This is especailly important
        // during software resets, since the target's 3.3v rail can take
        // 20-50ms to drain.  During this time the target could be driving
        // the reset pin low, causing the bootloader to think the reset
        // button is pressed.
        // Note: With optimization set to -O2 the value 5115 delays for ~1ms @ 20.9Mhz core
        busy_wait(5115 * 50);
    }
}