espressif_tinyusb/src/portable/nordic/nrf5x/hal_nrf5x.c

256 lines
7.0 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.
*/
2018-04-12 08:14:59 +02:00
#include "tusb_option.h"
2018-07-23 10:25:45 +02:00
#if TUSB_OPT_DEVICE_ENABLED && CFG_TUSB_MCU == OPT_MCU_NRF5X
2018-03-26 17:54:34 +02:00
#include "nrf.h"
#include "nrf_gpio.h"
2018-03-20 14:57:49 +01:00
#include "nrf_clock.h"
2018-04-04 11:48:52 +02:00
#include "nrf_usbd.h"
#include "nrf_drv_usbd_errata.h"
2018-04-04 15:02:54 +02:00
#ifdef SOFTDEVICE_PRESENT
#include "nrf_sdm.h"
#include "nrf_soc.h"
2019-01-29 12:47:29 +01:00
#endif
2018-07-02 06:09:22 +02:00
#include "nrfx_power.h"
2018-11-23 09:05:40 +01:00
#include "device/dcd.h"
/*------------------------------------------------------------------*/
/* MACRO TYPEDEF CONSTANT ENUM
*------------------------------------------------------------------*/
2018-05-14 08:40:41 +02:00
void tusb_hal_nrf_power_event(uint32_t event);
/*------------------------------------------------------------------*/
/* HFCLK helper
*------------------------------------------------------------------*/
2018-04-04 15:02:54 +02:00
2018-07-02 12:32:09 +02:00
#ifdef SOFTDEVICE_PRESENT
2018-04-04 15:02:54 +02:00
// check if SD is present and enabled
static bool is_sd_enabled(void)
{
uint8_t sd_en = false;
(void) sd_softdevice_is_enabled(&sd_en);
return sd_en;
}
2018-07-02 12:32:09 +02:00
#endif
2018-04-04 15:02:54 +02:00
2018-04-04 11:48:52 +02:00
static bool hfclk_running(void)
{
#ifdef SOFTDEVICE_PRESENT
if ( is_sd_enabled() )
{
uint32_t is_running;
(void) sd_clock_hfclk_is_running(&is_running);
return (is_running ? true : false);
}
#endif
return nrf_clock_hf_is_running(NRF_CLOCK_HFCLK_HIGH_ACCURACY);
}
static void hfclk_enable(void)
{
// already running, nothing to do
if ( hfclk_running() ) return;
#ifdef SOFTDEVICE_PRESENT
if ( is_sd_enabled() )
{
(void)sd_clock_hfclk_request();
return;
}
#endif
nrf_clock_event_clear(NRF_CLOCK_EVENT_HFCLKSTARTED);
nrf_clock_task_trigger(NRF_CLOCK_TASK_HFCLKSTART);
}
static void hfclk_disable(void)
{
#ifdef SOFTDEVICE_PRESENT
if ( is_sd_enabled() )
{
(void)sd_clock_hfclk_release();
return;
}
#endif
nrf_clock_task_trigger(NRF_CLOCK_TASK_HFCLKSTOP);
}
2018-05-14 08:40:41 +02:00
/*------------------------------------------------------------------*/
/* Controller Start up Sequence (USBD 51.4 specs)
*------------------------------------------------------------------*/
// tusb_hal_nrf_power_event must be called by SOC event handler
2018-06-06 08:40:02 +02:00
void tusb_hal_nrf_power_event (uint32_t event)
2018-05-14 08:40:41 +02:00
{
switch ( event )
{
case NRFX_POWER_USB_EVT_DETECTED:
if ( !NRF_USBD->ENABLE )
2018-04-18 13:05:08 +02:00
{
2018-05-14 08:40:41 +02:00
/* Prepare for READY event receiving */
nrf_usbd_eventcause_clear(NRF_USBD_EVENTCAUSE_READY_MASK);
/* Enable the peripheral */
2018-06-06 08:40:02 +02:00
// ERRATA 171, 187, 166
if ( nrf_drv_usbd_errata_187() )
2018-04-18 13:05:08 +02:00
{
// CRITICAL_REGION_ENTER();
2018-06-06 08:40:02 +02:00
if ( *((volatile uint32_t *) (0x4006EC00)) == 0x00000000 )
2018-05-14 08:40:41 +02:00
{
2018-06-06 08:40:02 +02:00
*((volatile uint32_t *) (0x4006EC00)) = 0x00009375;
*((volatile uint32_t *) (0x4006ED14)) = 0x00000003;
*((volatile uint32_t *) (0x4006EC00)) = 0x00009375;
2018-05-14 08:40:41 +02:00
}
else
{
2018-06-06 08:40:02 +02:00
*((volatile uint32_t *) (0x4006ED14)) = 0x00000003;
2018-05-14 08:40:41 +02:00
}
// CRITICAL_REGION_EXIT();
}
2018-05-14 08:40:41 +02:00
2018-06-06 08:40:02 +02:00
if ( nrf_drv_usbd_errata_171() )
{
// CRITICAL_REGION_ENTER();
2018-06-06 08:40:02 +02:00
if ( *((volatile uint32_t *) (0x4006EC00)) == 0x00000000 )
2018-05-14 08:40:41 +02:00
{
2018-06-06 08:40:02 +02:00
*((volatile uint32_t *) (0x4006EC00)) = 0x00009375;
*((volatile uint32_t *) (0x4006EC14)) = 0x000000C0;
*((volatile uint32_t *) (0x4006EC00)) = 0x00009375;
2018-05-14 08:40:41 +02:00
}
else
{
2018-06-06 08:40:02 +02:00
*((volatile uint32_t *) (0x4006EC14)) = 0x000000C0;
2018-05-14 08:40:41 +02:00
}
// CRITICAL_REGION_EXIT();
}
2018-05-14 08:40:41 +02:00
nrf_usbd_enable();
// Enable HFCLK
hfclk_enable();
2018-04-04 11:48:52 +02:00
}
2018-05-14 08:40:41 +02:00
break;
case NRFX_POWER_USB_EVT_READY:
/* Waiting for USBD peripheral enabled */
while ( !(USBD_EVENTCAUSE_READY_Msk & NRF_USBD->EVENTCAUSE) ) { }
nrf_usbd_eventcause_clear(USBD_EVENTCAUSE_READY_Msk);
nrf_usbd_event_clear(USBD_EVENTCAUSE_READY_Msk);
2018-04-04 11:48:52 +02:00
2018-06-06 08:40:02 +02:00
if ( nrf_drv_usbd_errata_171() )
{
// CRITICAL_REGION_ENTER();
2018-06-06 08:40:02 +02:00
if ( *((volatile uint32_t *) (0x4006EC00)) == 0x00000000 )
{
*((volatile uint32_t *) (0x4006EC00)) = 0x00009375;
*((volatile uint32_t *) (0x4006EC14)) = 0x00000000;
*((volatile uint32_t *) (0x4006EC00)) = 0x00009375;
}
else
{
*((volatile uint32_t *) (0x4006EC14)) = 0x00000000;
}
2018-05-14 08:40:41 +02:00
// CRITICAL_REGION_EXIT();
}
if ( nrf_drv_usbd_errata_187() )
2018-05-14 08:40:41 +02:00
{
// CRITICAL_REGION_ENTER();
2018-06-06 08:40:02 +02:00
if ( *((volatile uint32_t *) (0x4006EC00)) == 0x00000000 )
{
*((volatile uint32_t *) (0x4006EC00)) = 0x00009375;
*((volatile uint32_t *) (0x4006ED14)) = 0x00000000;
*((volatile uint32_t *) (0x4006EC00)) = 0x00009375;
}
else
{
*((volatile uint32_t *) (0x4006ED14)) = 0x00000000;
}
// CRITICAL_REGION_EXIT();
2018-05-14 08:40:41 +02:00
}
2018-05-14 08:40:41 +02:00
if ( nrf_drv_usbd_errata_166() )
{
*((volatile uint32_t *) (NRF_USBD_BASE + 0x800)) = 0x7E3;
*((volatile uint32_t *) (NRF_USBD_BASE + 0x804)) = 0x40;
2018-04-04 11:48:52 +02:00
2018-06-06 08:40:02 +02:00
__ISB();
__DSB();
2018-05-14 08:40:41 +02:00
}
2018-04-04 11:48:52 +02:00
2018-09-25 07:57:46 +02:00
nrf_usbd_isosplit_set(USBD_ISOSPLIT_SPLIT_HalfIN);
2018-04-04 11:48:52 +02:00
// Enable interrupt
NRF_USBD->INTENSET = USBD_INTEN_USBRESET_Msk | USBD_INTEN_EPDATA_Msk |
2018-07-02 12:32:09 +02:00
USBD_INTEN_EP0SETUP_Msk | USBD_INTEN_EP0DATADONE_Msk | USBD_INTEN_ENDEPIN0_Msk | USBD_INTEN_ENDEPOUT0_Msk;
// Enable interrupt, priorities should be set by application
2018-05-14 08:40:41 +02:00
NVIC_ClearPendingIRQ(USBD_IRQn);
NVIC_EnableIRQ(USBD_IRQn);
2018-04-04 11:48:52 +02:00
2018-05-14 08:40:41 +02:00
// Wait for HFCLK
2018-12-05 08:30:47 +01:00
while ( !hfclk_running() ) { }
2018-04-05 13:36:59 +02:00
2018-05-14 08:40:41 +02:00
// Enable pull up
nrf_usbd_pullup_enable();
break;
2018-04-04 11:48:52 +02:00
2018-05-14 08:40:41 +02:00
case NRFX_POWER_USB_EVT_REMOVED:
if ( NRF_USBD->ENABLE )
{
// Abort all transfers
2018-04-04 11:48:52 +02:00
2018-05-14 08:40:41 +02:00
// Disable pull up
nrf_usbd_pullup_disable();
2018-04-04 11:48:52 +02:00
2018-05-14 08:40:41 +02:00
// Disable Interrupt
NVIC_DisableIRQ(USBD_IRQn);
2018-04-04 11:48:52 +02:00
2018-05-14 08:40:41 +02:00
// disable all interrupt
NRF_USBD->INTENCLR = NRF_USBD->INTEN;
2018-04-04 11:48:52 +02:00
2018-05-14 08:40:41 +02:00
nrf_usbd_disable();
hfclk_disable();
2018-11-23 09:05:40 +01:00
dcd_event_bus_signal(0, DCD_EVENT_UNPLUGGED, true);
2018-05-14 08:40:41 +02:00
}
break;
2018-04-04 11:48:52 +02:00
2018-05-14 08:40:41 +02:00
default: break;
2018-04-04 11:48:52 +02:00
}
}
2018-03-26 17:54:34 +02:00
#endif