Repostiory containing DAPLink source code with Reset Pin workaround for HANI_IOT board.
Upstream: https://github.com/ARMmbed/DAPLink
source/family/arm/musca_b/power_ctrl.c
- Committer:
- Pawel Zarembski
- Date:
- 2020-04-07
- Revision:
- 0:01f31e923fe2
File content as of revision 0:01f31e923fe2:
/** * @file power_ctrl.c * @brief power control sequence logic for Musca B * * DAPLink Interface Firmware * Copyright (c) 2009-2019, ARM Limited, All Rights Reserved * 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 "string.h" #include "stdio.h" #include "stdint.h" #include "syscon.h" #include "gpio.h" #include "utils.h" #include "uart.h" #include "i2c_gpio.h" #include "daplink_debug.h" #include "settings.h" #include "region_defs.h" // I2CIO expander init (PCA9537) void i2cio_init(void) { unsigned int data; // Default I2C state LPC_GPIO->DIR[PIN_I2C_SCL_PORT] &= ~PIN_I2C_SCL; // Wait 1ms delay(1); LPC_GPIO->DIR[PIN_I2C_SDA_PORT] &= ~PIN_I2C_SDA; // Wait 1ms delay(1); // Dummy read i2c_gpio_rbyte(0, &data); // Wait 1ms delay(1); // Set default values // Output register (all PSUs OFF) i2c_gpio_wbyte(1, 0x00); // Wait 1ms delay(1); // Inversion register i2c_gpio_wbyte(2, 0x00); // Wait 1ms delay(1); // Configuration register i2c_gpio_wbyte(3, 0xF0); // Wait 100ms delay(100); } // I2CIO expander power OFF // Sequence: ON > FLASH_PDM + /PWR_OK > /VDD_IO > /VDD_CORE > + /VDD_FLASH + /FLASH_PDM > OFF void i2cio_power_off(void) { i2c_gpio_wbyte(1, FL_ON | FLASH_PDM | CR_ON | IO_ON); LPC_GPIO->CLR[PIN_PWROK_PORT] = PIN_PWROK; i2c_gpio_wbyte(1, FL_ON | FLASH_PDM | CR_ON); i2c_gpio_wbyte(1, FL_ON | FLASH_PDM); i2c_gpio_wbyte(1, 0); } // I2CIO expander power ON // Sequence: OFF > VDD_FLASH + FLASH_PDM > VDD_CORE > /FLASH_PDM + PWR_OK > VDD_IO > ON void i2cio_power_on(void) { i2c_gpio_wbyte(1, FL_ON | FLASH_PDM); // Wait 100ms delay(100); i2c_gpio_wbyte(1, FL_ON | FLASH_PDM | CR_ON); // Wait 100ms delay(100); i2c_gpio_wbyte(1, FL_ON | CR_ON); LPC_GPIO->SET[PIN_PWROK_PORT] = PIN_PWROK; // Wait 100ms delay(100); i2c_gpio_wbyte(1, FL_ON | CR_ON | IO_ON); // Wait 100ms delay(100); } // Configure SCC void configure_syscon(unsigned int pc) { unsigned int din, dout; // Dummy read syscon_readreg(0x004, &din); // CPU0 boot from QSPI/eFlash0/eFlash1/SRAM // pc = 0x10000000/0x1A000000/0x1A200000/0x1A400000; syscon_writereg(0x058, pc); // CPU1 boot from QSPI/eFlash0/eFlash1/SRAM // pc = 0x10000000/0x1A000000/0x1A200000/0x1A400000; syscon_writereg(0x060, pc); // QSPI divider for 20MHz dout = 0x00000001; syscon_writereg(0x010, dout); // Set PA[37:0] IO drive strength (4/8/4mA) and slew rate (Slow) syscon_writereg(0x0E8, 0xFFF00000); syscon_writereg(0x0EC, 0xFFFFFFC0); syscon_writereg(0x0F0, 0x000FFFFF); syscon_writereg(0x0F4, 0xFFFFFFFF); syscon_writereg(0x108, 0xFFFFFFFF); syscon_writereg(0x10C, 0x0000003F); } // Brown Out Detect void enable_BOD(int enable) { if (enable) { // Set BOD interrupt to 2.80-2.90V LPC_SYSCON->BODCTRL |= 0x1C; NVIC_EnableIRQ(BOD_IRQn); /* Enable BOD interrupt */ } else { LPC_SYSCON->BODCTRL = 0x00; NVIC_DisableIRQ(BOD_IRQn); NVIC_ClearPendingIRQ(BOD_IRQn); } } //Power off / shutdown sequence void power_off_sequence() { // Disable Brown Out Detection enable_BOD(0); // Apply CS_nSRST LPC_GPIO->CLR[PIN_nRESET_PORT] = PIN_nRESET; // Wait 10ms delay(10); // Apply CB_nRST LPC_GPIO->CLR[PIN_CB_nRST_PORT] = PIN_CB_nRST; // Wait 10ms delay(10); // Apply CFG_nRST LPC_GPIO->CLR[PIN_CFG_nRST_PORT] = PIN_CFG_nRST; // Wait 10ms delay(10); } //Power on sequence void power_on_sequence() { // Drive SCC signals LPC_GPIO->DIR[PIN_SCC_CLK_PORT] |= PIN_SCC_CLK; LPC_GPIO->DIR[PIN_SCC_DATAIN_PORT] |= PIN_SCC_DATAIN; LPC_GPIO->DIR[PIN_SCC_DATAOUT_PORT] &= ~PIN_SCC_DATAOUT; LPC_GPIO->DIR[PIN_SCC_WNR_PORT] |= PIN_SCC_WNR; LPC_GPIO->DIR[PIN_SCC_LOAD_PORT] |= PIN_SCC_LOAD; // Wait 10ms delay(10); // Release CFG_nRST to allow SCC config LPC_GPIO->SET[PIN_CFG_nRST_PORT] = PIN_CFG_nRST; // Wait 10ms delay(10); // Configure SCC #if (defined MUSCA_B_BOOT_CODE_SRAM) configure_syscon(MUSCA_B_S_CODE_SRAM_START); #elif (defined MUSCA_B_BOOT_EFLASH1) configure_syscon(MUSCA_B_S_EFLASH1_START); #elif (defined MUSCA_B_BOOT_EFLASH0) configure_syscon(MUSCA_B_S_EFLASH0_START); #elif (defined MUSCA_B_BOOT_QSPI_FLASH) configure_syscon(MUSCA_B_S_FLASH_START); #endif // Wait 10ms delay(10); // Release SCC signals LPC_GPIO->DIR[PIN_SCC_CLK_PORT] &= ~PIN_SCC_CLK; LPC_GPIO->DIR[PIN_SCC_DATAIN_PORT] &= ~PIN_SCC_DATAIN; LPC_GPIO->DIR[PIN_SCC_DATAOUT_PORT] &= ~PIN_SCC_DATAOUT; LPC_GPIO->DIR[PIN_SCC_WNR_PORT] &= ~PIN_SCC_WNR; LPC_GPIO->DIR[PIN_SCC_LOAD_PORT] &= ~PIN_SCC_LOAD; // Wait 10ms delay(10); // Release CB_nRST LPC_GPIO->SET[PIN_CB_nRST_PORT] = PIN_CB_nRST; // Wait 10ms delay(10); // Release CS_nSRST LPC_GPIO->SET[PIN_nRESET_PORT] = PIN_nRESET; // Wait 10ms delay(10); // Enable Brown Out Detection enable_BOD(1); } // BOD Interrupt Service Routine void BOD_IRQHandler(void) { NVIC_DisableIRQ(BOD_IRQn); gpio_set_cdc_led(GPIO_LED_OFF); // ON GREEN // go into controlled shutdown power_off_sequence(); // Turn OFF power i2cio_power_off(); while(1) { delay(100); gpio_set_hid_led(GPIO_LED_ON); delay(100); gpio_set_hid_led(GPIO_LED_OFF); } } // Function to wait till PBON button is pressed and released void wait_for_pbon(void) { // Standby - wait for PBON while (!gpio_get_pbon_btn()) { // Do something with leds? gpio_set_cdc_led(GPIO_LED_ON); delay(100); gpio_set_cdc_led(GPIO_LED_OFF); delay(100); } // Wait for PBON to go low then high while (gpio_get_pbon_btn()) { // Do something with leds? gpio_set_hid_led(GPIO_LED_ON); delay(100); gpio_set_hid_led(GPIO_LED_OFF); delay(100); } // Wait 10ms delay(10); }