Repostiory containing DAPLink source code with Reset Pin workaround for HANI_IOT board.
Upstream: https://github.com/ARMmbed/DAPLink
Diff: source/hic_hal/freescale/k26f/hic_init.c
- Revision:
- 0:01f31e923fe2
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/source/hic_hal/freescale/k26f/hic_init.c Tue Apr 07 12:55:42 2020 +0200 @@ -0,0 +1,156 @@ +/** + * @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); + } +}