tinyusb/hw/bsp/pca10056/board_pca10056.c

309 lines
8.1 KiB
C
Raw Normal View History

/*
* The MIT License (MIT)
*
* Copyright (c) 2018, hathach (tinyusb.org)
*
* 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.
*
* This file is part of the TinyUSB stack.
*/
#ifdef BOARD_PCA10056
2018-03-29 18:10:01 +07:00
#include "bsp/board.h"
2018-11-13 12:32:15 +07:00
#include "nrfx/hal/nrf_gpio.h"
#include "nrfx/drivers/include/nrfx_power.h"
#include "nrfx/drivers/include/nrfx_qspi.h"
2019-01-29 14:15:13 +07:00
#ifdef SOFTDEVICE_PRESENT
#include "nrf_sdm.h"
#include "nrf_soc.h"
#endif
2018-09-25 12:57:46 +07:00
#include "tusb.h"
/*------------------------------------------------------------------*/
/* MACRO TYPEDEF CONSTANT ENUM
*------------------------------------------------------------------*/
2018-12-03 23:39:34 +07:00
#define LED_PIN 13
#define LED_STATE_ON 0
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 {
2018-10-23 19:54:06 +07:00
QSPI_CMD_RSTEN = 0x66,
QSPI_CMD_RST = 0x99,
QSPI_CMD_WRSR = 0x01,
QSPI_CMD_READID = 0x90
2018-09-25 12:57:46 +07:00
};
/* 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;
2018-11-13 12:32:15 +07:00
// LEDs
2018-12-03 23:39:34 +07:00
nrf_gpio_cfg_output(LED_PIN);
board_led_control(false);
// 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
#if 0 // def 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-10-23 19:54:06 +07:00
// NULL callback for blocking API
nrfx_qspi_init(&qspi_cfg, NULL, 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-10-23 19:54:06 +07:00
// Switch to quad mode
uint16_t sr_quad_en = 0x40;
2018-09-25 12:57:46 +07:00
cinstr_cfg.opcode = QSPI_CMD_WRSR;
2018-10-23 19:54:06 +07:00
cinstr_cfg.length = 3;
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
NVIC_SetPriority(USBD_IRQn, 2);
// 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
2019-01-29 18:02:25 +07:00
// Enable to test enable SD before USB scenario
#if 1
extern void nrf_error_cb(uint32_t id, uint32_t pc, uint32_t info);
nrf_clock_lf_cfg_t clock_cfg =
{
// LFXO
.source = NRF_CLOCK_LF_SRC_XTAL,
.rc_ctiv = 0,
.rc_temp_ctiv = 0,
.accuracy = NRF_CLOCK_LF_ACCURACY_20_PPM
};
sd_softdevice_enable(&clock_cfg, nrf_error_cb);
NVIC_EnableIRQ(SD_EVT_IRQn);
#endif
uint8_t sd_en = false;
2019-01-29 18:02:25 +07:00
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
2019-01-29 15:39:47 +07:00
#endif
{
// 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;
}
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-12-03 23:39:34 +07:00
void board_led_control(bool state)
{
2018-12-03 23:39:34 +07:00
nrf_gpio_pin_write(LED_PIN, state ? LED_STATE_ON : (1-LED_STATE_ON));
}
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;
}
2019-03-20 20:58:27 +07:00
int board_uart_read(uint8_t* buf, int len)
{
2019-03-20 20:58:27 +07:00
(void) buf;
(void) len;
return 0;
}
2019-03-20 20:58:27 +07:00
int board_uart_write(void const * buf, int len)
{
2019-03-20 20:58:27 +07:00
(void) buf;
(void) len;
return 0;
}
2019-01-29 15:39:47 +07:00
#ifdef SOFTDEVICE_PRESENT
2019-01-29 18:02:25 +07:00
// process SOC event from SD
uint32_t proc_soc(void)
{
uint32_t soc_evt;
uint32_t err = sd_evt_get(&soc_evt);
if (NRF_SUCCESS == err)
{
/*------------- usb power event handler -------------*/
int32_t usbevt = (soc_evt == NRF_EVT_POWER_USB_DETECTED ) ? NRFX_POWER_USB_EVT_DETECTED:
(soc_evt == NRF_EVT_POWER_USB_POWER_READY) ? NRFX_POWER_USB_EVT_READY :
(soc_evt == NRF_EVT_POWER_USB_REMOVED ) ? NRFX_POWER_USB_EVT_REMOVED : -1;
if ( usbevt >= 0) tusb_hal_nrf_power_event(usbevt);
}
return err;
}
uint32_t proc_ble(void)
{
// do nothing with ble
return NRF_ERROR_NOT_FOUND;
}
2019-01-29 15:39:47 +07:00
void SD_EVT_IRQHandler(void)
{
2019-01-29 18:02:25 +07:00
// process BLE and SOC until there is no more events
while( (NRF_ERROR_NOT_FOUND != proc_ble()) || (NRF_ERROR_NOT_FOUND != proc_soc()) )
{
}
2019-01-29 15:39:47 +07:00
}
2019-01-29 18:02:25 +07:00
void nrf_error_cb(uint32_t id, uint32_t pc, uint32_t info)
2019-01-29 15:39:47 +07:00
{
2019-01-29 18:02:25 +07:00
(void) id;
(void) pc;
(void) info;
2019-01-29 15:39:47 +07:00
}
#endif
#endif