* fbt: reworking targets & assets handling WIP * fbt: dist fixes * fbt: moved SD card resources to owning apps * unit_tests: moved resources to app folder * github: updated unit_tests paths * github: packaging fixes * unit_tests: fixes * fbt: assets: internal cleanup * fbt: reworked assets handling * github: unit_tests: reintroducing fixes * minor cleanup * fbt: naming changes to reflect private nature of scons tools * fbt: resources: fixed dist archive paths * docs: updated paths * docs: updated more paths * docs: included "resources" parameter in app manifest docs; updated assets readme * updated gitignore for assets * github: updated action versions * unit_tests: restored timeout; scripts: assets: logging changes * gh: don't upload desktop animations for unit test run Co-authored-by: あく <alleteam@gmail.com>
		
			
				
	
	
		
			230 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			230 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#include "furi_hal_nfc_i.h"
 | 
						|
#include "furi_hal_nfc_tech_i.h"
 | 
						|
 | 
						|
#include <stm32wbxx_ll_tim.h>
 | 
						|
 | 
						|
#include <furi_hal_interrupt.h>
 | 
						|
#include <furi_hal_resources.h>
 | 
						|
#include <furi_hal_bus.h>
 | 
						|
 | 
						|
#define TAG "FuriHalNfcTimer"
 | 
						|
 | 
						|
#define FURI_HAL_NFC_TIMER_US_IN_S (1000000UL)
 | 
						|
 | 
						|
/**
 | 
						|
 * To enable timer debug output on GPIO, define the FURI_HAL_NFC_TIMER_DEBUG macro
 | 
						|
 * Example: ./fbt --extra-define=FURI_HAL_NFC_TIMER_DEBUG
 | 
						|
 */
 | 
						|
 | 
						|
typedef enum {
 | 
						|
    FuriHalNfcTimerFwt,
 | 
						|
    FuriHalNfcTimerBlockTx,
 | 
						|
    FuriHalNfcTimerCount,
 | 
						|
} FuriHalNfcTimer;
 | 
						|
 | 
						|
typedef struct {
 | 
						|
    TIM_TypeDef* timer;
 | 
						|
    FuriHalBus bus;
 | 
						|
    uint32_t prescaler;
 | 
						|
    uint32_t freq_khz;
 | 
						|
    FuriHalNfcEventInternalType event;
 | 
						|
    FuriHalInterruptId irq_id;
 | 
						|
    IRQn_Type irq_type;
 | 
						|
#ifdef FURI_HAL_NFC_TIMER_DEBUG
 | 
						|
    const GpioPin* pin;
 | 
						|
#endif
 | 
						|
} FuriHalNfcTimerConfig;
 | 
						|
 | 
						|
static const FuriHalNfcTimerConfig furi_hal_nfc_timers[FuriHalNfcTimerCount] = {
 | 
						|
    [FuriHalNfcTimerFwt] =
 | 
						|
        {
 | 
						|
            .timer = TIM1,
 | 
						|
            .bus = FuriHalBusTIM1,
 | 
						|
            .event = FuriHalNfcEventInternalTypeTimerFwtExpired,
 | 
						|
            .irq_id = FuriHalInterruptIdTim1UpTim16,
 | 
						|
            .irq_type = TIM1_UP_TIM16_IRQn,
 | 
						|
#ifdef FURI_HAL_NFC_TIMER_DEBUG
 | 
						|
            .pin = &gpio_ext_pa7,
 | 
						|
#endif
 | 
						|
        },
 | 
						|
    [FuriHalNfcTimerBlockTx] =
 | 
						|
        {
 | 
						|
            .timer = TIM17,
 | 
						|
            .bus = FuriHalBusTIM17,
 | 
						|
            .event = FuriHalNfcEventInternalTypeTimerBlockTxExpired,
 | 
						|
            .irq_id = FuriHalInterruptIdTim1TrgComTim17,
 | 
						|
            .irq_type = TIM1_TRG_COM_TIM17_IRQn,
 | 
						|
#ifdef FURI_HAL_NFC_TIMER_DEBUG
 | 
						|
            .pin = &gpio_ext_pa6,
 | 
						|
#endif
 | 
						|
        },
 | 
						|
};
 | 
						|
 | 
						|
static void furi_hal_nfc_timer_irq_callback(void* context) {
 | 
						|
    // Returning removed const-ness
 | 
						|
    const FuriHalNfcTimerConfig* config = context;
 | 
						|
    if(LL_TIM_IsActiveFlag_UPDATE(config->timer)) {
 | 
						|
        LL_TIM_ClearFlag_UPDATE(config->timer);
 | 
						|
        furi_hal_nfc_event_set(config->event);
 | 
						|
#ifdef FURI_HAL_NFC_TIMER_DEBUG
 | 
						|
        furi_hal_gpio_write(timer_config->pin, false);
 | 
						|
#endif
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void furi_hal_nfc_timer_init(FuriHalNfcTimer timer) {
 | 
						|
    const FuriHalNfcTimerConfig* config = &furi_hal_nfc_timers[timer];
 | 
						|
 | 
						|
    furi_hal_bus_enable(config->bus);
 | 
						|
 | 
						|
    LL_TIM_SetOnePulseMode(config->timer, LL_TIM_ONEPULSEMODE_SINGLE);
 | 
						|
    LL_TIM_EnableUpdateEvent(config->timer);
 | 
						|
    LL_TIM_SetCounterMode(config->timer, LL_TIM_COUNTERMODE_UP);
 | 
						|
    LL_TIM_SetClockSource(config->timer, LL_TIM_CLOCKSOURCE_INTERNAL);
 | 
						|
 | 
						|
    furi_hal_interrupt_set_isr(
 | 
						|
        config->irq_id,
 | 
						|
        furi_hal_nfc_timer_irq_callback,
 | 
						|
        // Warning: casting const-ness away
 | 
						|
        (FuriHalNfcTimerConfig*)config);
 | 
						|
    NVIC_SetPriority(config->irq_type, NVIC_EncodePriority(NVIC_GetPriorityGrouping(), 5, 0));
 | 
						|
    NVIC_EnableIRQ(config->irq_type);
 | 
						|
#ifdef FURI_HAL_NFC_TIMER_DEBUG
 | 
						|
    furi_hal_gpio_init(config->pin, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
 | 
						|
    furi_hal_gpio_write(config->pin, false);
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static void furi_hal_nfc_timer_deinit(FuriHalNfcTimer timer) {
 | 
						|
    const FuriHalNfcTimerConfig* config = &furi_hal_nfc_timers[timer];
 | 
						|
 | 
						|
    LL_TIM_ClearFlag_UPDATE(config->timer);
 | 
						|
    furi_hal_interrupt_set_isr(config->irq_id, NULL, NULL);
 | 
						|
    NVIC_DisableIRQ(config->irq_type);
 | 
						|
 | 
						|
    if(furi_hal_bus_is_enabled(config->bus)) {
 | 
						|
        furi_hal_bus_disable(config->bus);
 | 
						|
    }
 | 
						|
#ifdef FURI_HAL_NFC_TIMER_DEBUG
 | 
						|
    furi_hal_gpio_init_simple(config->pin, GpioModeAnalog);
 | 
						|
    furi_hal_gpio_write(config->pin, false);
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static int32_t furi_hal_nfc_timer_get_compensation(FuriHalNfcTimer timer) {
 | 
						|
    const FuriHalNfcTechBase* current_tech = furi_hal_nfc_tech[furi_hal_nfc.tech];
 | 
						|
 | 
						|
    if(furi_hal_nfc.mode == FuriHalNfcModePoller) {
 | 
						|
        const FuriHalNfcPollerCompensation* comp = ¤t_tech->poller.compensation;
 | 
						|
        if(timer == FuriHalNfcTimerFwt)
 | 
						|
            return comp->fwt;
 | 
						|
        else if(timer == FuriHalNfcTimerBlockTx)
 | 
						|
            return comp->fdt;
 | 
						|
 | 
						|
    } else if(furi_hal_nfc.mode == FuriHalNfcModeListener) {
 | 
						|
        const FuriHalNfcListenerCompensation* comp = ¤t_tech->listener.compensation;
 | 
						|
        if(timer == FuriHalNfcTimerBlockTx) return comp->fdt;
 | 
						|
    }
 | 
						|
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool furi_hal_nfc_timer_is_running(FuriHalNfcTimer timer) {
 | 
						|
    return LL_TIM_IsEnabledCounter(furi_hal_nfc_timers[timer].timer) != 0;
 | 
						|
}
 | 
						|
 | 
						|
static void furi_hal_nfc_timer_start_core_ticks(FuriHalNfcTimer timer, uint64_t core_ticks) {
 | 
						|
    furi_check(!furi_hal_nfc_timer_is_running(timer));
 | 
						|
 | 
						|
    const FuriHalNfcTimerConfig* config = &furi_hal_nfc_timers[timer];
 | 
						|
    furi_check(furi_hal_bus_is_enabled(config->bus));
 | 
						|
 | 
						|
    const uint32_t prescaler = (core_ticks - 1) / UINT16_MAX;
 | 
						|
    furi_check(prescaler <= UINT16_MAX);
 | 
						|
 | 
						|
    const uint32_t arr_reg = core_ticks / (prescaler + 1);
 | 
						|
    furi_check(arr_reg <= UINT16_MAX);
 | 
						|
 | 
						|
    LL_TIM_DisableIT_UPDATE(config->timer);
 | 
						|
 | 
						|
    LL_TIM_SetPrescaler(config->timer, prescaler);
 | 
						|
    LL_TIM_SetAutoReload(config->timer, arr_reg);
 | 
						|
 | 
						|
    LL_TIM_GenerateEvent_UPDATE(config->timer);
 | 
						|
    while(!LL_TIM_IsActiveFlag_UPDATE(config->timer))
 | 
						|
        ;
 | 
						|
    LL_TIM_ClearFlag_UPDATE(config->timer);
 | 
						|
 | 
						|
    LL_TIM_EnableIT_UPDATE(config->timer);
 | 
						|
    LL_TIM_EnableCounter(config->timer);
 | 
						|
#ifdef FURI_HAL_NFC_TIMER_DEBUG
 | 
						|
    furi_hal_gpio_write(config->pin, true);
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static void furi_hal_nfc_timer_start_us(FuriHalNfcTimer timer, uint32_t time_us) {
 | 
						|
    furi_hal_nfc_timer_start_core_ticks(
 | 
						|
        timer, SystemCoreClock / FURI_HAL_NFC_TIMER_US_IN_S * time_us);
 | 
						|
}
 | 
						|
 | 
						|
static void furi_hal_nfc_timer_start_fc(FuriHalNfcTimer timer, uint32_t time_fc) {
 | 
						|
    const int32_t comp_fc = furi_hal_nfc_timer_get_compensation(timer);
 | 
						|
    // Not starting the timer if the compensation value is greater than the requested delay
 | 
						|
    if(comp_fc >= (int32_t)time_fc) return;
 | 
						|
 | 
						|
    furi_hal_nfc_timer_start_core_ticks(
 | 
						|
        timer, ((uint64_t)SystemCoreClock * (time_fc - comp_fc)) / FURI_HAL_NFC_CARRIER_HZ);
 | 
						|
}
 | 
						|
 | 
						|
static void furi_hal_nfc_timer_stop(FuriHalNfcTimer timer) {
 | 
						|
    const FuriHalNfcTimerConfig* config = &furi_hal_nfc_timers[timer];
 | 
						|
 | 
						|
    LL_TIM_DisableIT_UPDATE(config->timer);
 | 
						|
    LL_TIM_DisableCounter(config->timer);
 | 
						|
    LL_TIM_SetCounter(config->timer, 0);
 | 
						|
    LL_TIM_SetAutoReload(config->timer, 0);
 | 
						|
 | 
						|
    if(LL_TIM_IsActiveFlag_UPDATE(config->timer)) {
 | 
						|
        LL_TIM_ClearFlag_UPDATE(config->timer);
 | 
						|
    }
 | 
						|
#ifdef FURI_HAL_NFC_TIMER_DEBUG
 | 
						|
    furi_hal_gpio_write(config->pin, false);
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
void furi_hal_nfc_timers_init() {
 | 
						|
    for(size_t i = 0; i < FuriHalNfcTimerCount; i++) {
 | 
						|
        furi_hal_nfc_timer_init(i);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void furi_hal_nfc_timers_deinit() {
 | 
						|
    for(size_t i = 0; i < FuriHalNfcTimerCount; i++) {
 | 
						|
        furi_hal_nfc_timer_deinit(i);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void furi_hal_nfc_timer_fwt_start(uint32_t time_fc) {
 | 
						|
    furi_hal_nfc_timer_start_fc(FuriHalNfcTimerFwt, time_fc);
 | 
						|
}
 | 
						|
 | 
						|
void furi_hal_nfc_timer_fwt_stop() {
 | 
						|
    furi_hal_nfc_timer_stop(FuriHalNfcTimerFwt);
 | 
						|
}
 | 
						|
 | 
						|
void furi_hal_nfc_timer_block_tx_start(uint32_t time_fc) {
 | 
						|
    furi_hal_nfc_timer_start_fc(FuriHalNfcTimerBlockTx, time_fc);
 | 
						|
}
 | 
						|
 | 
						|
void furi_hal_nfc_timer_block_tx_start_us(uint32_t time_us) {
 | 
						|
    furi_hal_nfc_timer_start_us(FuriHalNfcTimerBlockTx, time_us);
 | 
						|
}
 | 
						|
 | 
						|
void furi_hal_nfc_timer_block_tx_stop() {
 | 
						|
    furi_hal_nfc_timer_stop(FuriHalNfcTimerBlockTx);
 | 
						|
}
 | 
						|
 | 
						|
bool furi_hal_nfc_timer_block_tx_is_running() {
 | 
						|
    return furi_hal_nfc_timer_is_running(FuriHalNfcTimerBlockTx);
 | 
						|
}
 |