Files
tinyUSB/hw/bsp/pca10056/board_pca10056.c

259 lines
7.3 KiB
C
Raw Normal View History

/**************************************************************************/
/*!
@file board_pca10056.c
@author hathach
@section LICENSE
Software License Agreement (BSD License)
Copyright (c) 2018, hathach (tinyusb.org)
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. Neither the name of the copyright holders nor the
names of its contributors may be used to endorse or promote products
derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
2018-07-27 16:59:57 +07:00
This file is part of the tinyusb stack.
*/
/**************************************************************************/
2018-03-29 18:10:01 +07:00
#include "bsp/board.h"
#include "nrf_gpio.h"
#include "nrfx_power.h"
2018-09-08 00:15:34 +07:00
#include "nrfx_qspi.h"
2018-09-25 12:57:46 +07:00
#include "tusb.h"
/*------------------------------------------------------------------*/
/* MACRO TYPEDEF CONSTANT ENUM
*------------------------------------------------------------------*/
#define LED_1 13
#define LED_STATE_ON 0
2018-03-13 16:30:53 +07:00
#define LED_STATE_OFF (1-LED_STATE_ON)
uint8_t _button_pins[] = { 11, 12, 24, 25 };
2018-09-08 00:15:34 +07:00
#define BOARD_BUTTON_COUNT sizeof(_button_pins)
/*------------------------------------------------------------------*/
2018-03-29 18:10:01 +07:00
/* TUSB HAL MILLISECOND
*------------------------------------------------------------------*/
2018-08-23 14:11:21 +07:00
#if CFG_TUSB_OS == OPT_OS_NONE
2018-03-29 18:10:01 +07:00
volatile uint32_t system_ticks = 0;
void SysTick_Handler (void)
{
system_ticks++;
}
uint32_t tusb_hal_millis(void)
{
2018-03-30 15:33:36 +07:00
return board_tick2ms(system_ticks);
2018-03-29 18:10:01 +07:00
}
2018-08-23 14:11:21 +07:00
#endif
/*------------------------------------------------------------------*/
2018-03-29 18:10:01 +07:00
/* BOARD API
*------------------------------------------------------------------*/
2018-09-25 12:57:46 +07:00
enum {
QSPI_CMD_RSTEN = 0x66,
QSPI_CMD_RST = 0x99,
QSPI_CMD_WRSR = 0x01,
QSPI_CMD_READID = 0x90
};
2018-09-08 16:39:42 +07:00
extern void qspi_flash_complete (void);
void qflash_hdl (nrfx_qspi_evt_t event, void * p_context)
{
(void) p_context;
(void) event;
qspi_flash_complete();
}
/* tinyusb function that handles power event (detected, ready, removed)
* We must call it within SD's SOC event handler, or set it as power event handler if SD is not enabled.
*/
extern void tusb_hal_nrf_power_event(uint32_t event);
void board_init(void)
{
// Config clock source: XTAL or RC in sdk_config.h
NRF_CLOCK->LFCLKSRC = (uint32_t)((CLOCK_LFCLKSRC_SRC_Xtal << CLOCK_LFCLKSRC_SRC_Pos) & CLOCK_LFCLKSRC_SRC_Msk);
NRF_CLOCK->TASKS_LFCLKSTART = 1UL;
// LED
nrf_gpio_cfg_output(LED_1);
// Button
for(uint8_t i=0; i<BOARD_BUTTON_COUNT; i++) nrf_gpio_cfg_input(_button_pins[i], NRF_GPIO_PIN_PULLUP);
2018-08-23 14:11:21 +07:00
#if CFG_TUSB_OS == OPT_OS_NONE
// Tick init
SysTick_Config(SystemCoreClock/1000);
2018-08-23 14:11:21 +07:00
#endif
2018-09-08 00:15:34 +07:00
// 64 Mbit qspi flash
#ifdef BOARD_MSC_FLASH_QSPI
2018-09-08 16:39:42 +07:00
nrfx_qspi_config_t qspi_cfg = {
.xip_offset = 0,
2018-09-08 00:15:34 +07:00
.pins = {
2018-09-08 16:39:42 +07:00
.sck_pin = 19,
.csn_pin = 17,
.io0_pin = 20,
.io1_pin = 21,
.io2_pin = 22,
.io3_pin = 23,
2018-09-08 00:15:34 +07:00
},
.prot_if = {
2018-09-08 16:39:42 +07:00
.readoc = NRF_QSPI_READOC_READ4IO,
2018-09-11 22:37:44 +07:00
.writeoc = NRF_QSPI_WRITEOC_PP4IO,
2018-09-08 16:39:42 +07:00
.addrmode = NRF_QSPI_ADDRMODE_24BIT,
.dpmconfig = false, // deep power down
2018-09-08 00:15:34 +07:00
},
.phy_if = {
2018-09-11 22:37:44 +07:00
.sck_freq = NRF_QSPI_FREQ_32MDIV1,
2018-09-08 16:39:42 +07:00
.sck_delay = 1,
.spi_mode = NRF_QSPI_MODE_0,
.dpmen = false
2018-09-08 00:15:34 +07:00
},
.irq_priority = 7,
};
2018-09-11 22:37:44 +07:00
nrfx_qspi_init(&qspi_cfg, qflash_hdl, NULL);
2018-09-08 00:15:34 +07:00
nrf_qspi_cinstr_conf_t cinstr_cfg = {
2018-09-08 16:39:42 +07:00
.opcode = 0,
.length = 0,
.io2_level = true,
.io3_level = true,
2018-09-25 12:57:46 +07:00
.wipwait = false,
.wren = false
2018-09-08 00:15:34 +07:00
};
// Send reset enable
2018-09-25 12:57:46 +07:00
cinstr_cfg.opcode = QSPI_CMD_RSTEN;
2018-09-08 00:15:34 +07:00
cinstr_cfg.length = NRF_QSPI_CINSTR_LEN_1B;
nrfx_qspi_cinstr_xfer(&cinstr_cfg, NULL, NULL);
// Send reset command
2018-09-25 12:57:46 +07:00
cinstr_cfg.opcode = QSPI_CMD_RST;
2018-09-08 00:15:34 +07:00
cinstr_cfg.length = NRF_QSPI_CINSTR_LEN_1B;
nrfx_qspi_cinstr_xfer(&cinstr_cfg, NULL, NULL);
2018-09-25 12:57:46 +07:00
NRFX_DELAY_US(100); // wait for flash reset
// Send (Read ID + 3 dummy bytes) + Receive 2 bytes of Manufacture + Device ID
uint8_t dummy[6] = { 0 };
uint8_t id_resp[6] = { 0 };
cinstr_cfg.opcode = QSPI_CMD_READID;
cinstr_cfg.length = 6;
// Bug with -nrf_qspi_cinstrdata_get() didn't combine data.
// https://devzone.nordicsemi.com/f/nordic-q-a/38540/bug-nrf_qspi_cinstrdata_get-didn-t-collect-data-from-both-cinstrdat1-and-cinstrdat0
nrfx_qspi_cinstr_xfer(&cinstr_cfg, dummy, id_resp);
// Due to the bug, we collect data manually
uint8_t dev_id = (uint8_t) NRF_QSPI->CINSTRDAT1;
uint8_t mfgr_id = (uint8_t) ( NRF_QSPI->CINSTRDAT0 >> 24 );
2018-09-08 00:15:34 +07:00
// Switch to qspi mode
uint8_t sr_quad_en = 0x40;
2018-09-25 12:57:46 +07:00
cinstr_cfg.opcode = QSPI_CMD_WRSR;
2018-09-08 00:15:34 +07:00
cinstr_cfg.length = NRF_QSPI_CINSTR_LEN_2B;
2018-09-25 12:57:46 +07:00
cinstr_cfg.wipwait = cinstr_cfg.wren = true;
2018-09-08 00:15:34 +07:00
nrfx_qspi_cinstr_xfer(&cinstr_cfg, &sr_quad_en, NULL);
#endif
// USB power may already be ready at this time -> no event generated
// We need to invoke the handler based on the status initially
uint32_t usb_reg;
#ifdef SOFTDEVICE_PRESENT
uint8_t sd_en = false;
(void) sd_softdevice_is_enabled(&sd_en);
if ( sd_en ) {
sd_power_usbdetected_enable(true);
sd_power_usbpwrrdy_enable(true);
sd_power_usbremoved_enable(true);
sd_power_usbregstatus_get(&usb_reg);
}else
#else
{
// Power module init
const nrfx_power_config_t pwr_cfg = { 0 };
nrfx_power_init(&pwr_cfg);
// Register tusb function as USB power handler
const nrfx_power_usbevt_config_t config = { .handler = (nrfx_power_usb_event_handler_t) tusb_hal_nrf_power_event };
nrfx_power_usbevt_init(&config);
nrfx_power_usbevt_enable();
usb_reg = NRF_POWER->USBREGSTATUS;
}
#endif
if ( usb_reg & POWER_USBREGSTATUS_VBUSDETECT_Msk ) {
tusb_hal_nrf_power_event(NRFX_POWER_USB_EVT_DETECTED);
}
if ( usb_reg & POWER_USBREGSTATUS_OUTPUTRDY_Msk ) {
tusb_hal_nrf_power_event(NRFX_POWER_USB_EVT_READY);
}
}
2018-03-13 16:30:53 +07:00
void board_led_control(uint32_t led_id, bool state)
{
2018-03-13 16:30:53 +07:00
(void) led_id;
nrf_gpio_pin_write(LED_1, state ? LED_STATE_ON : LED_STATE_OFF);
}
uint32_t board_buttons(void)
{
uint32_t ret = 0;
for(uint8_t i=0; i<BOARD_BUTTON_COUNT; i++)
{
// button is active LOW
2018-09-08 00:15:34 +07:00
ret |= ( nrf_gpio_pin_read(_button_pins[i]) ? 0 : (1 << i));
}
return ret;
}
uint8_t board_uart_getchar(void)
{
return 0;
}
void board_uart_putchar(uint8_t c)
{
}