363 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			363 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
/****************************************************************************
 | 
						|
 | 
						|
Copyright(c) 2019 by Aerospace C.Power (Chongqing) Microelectronics. ALL RIGHTS RESERVED.
 | 
						|
 | 
						|
This Information is proprietary to Aerospace C.Power (Chongqing) Microelectronics and MAY NOT
 | 
						|
be copied by any method or incorporated into another program without
 | 
						|
the express written consent of Aerospace C.Power. This Information or any portion
 | 
						|
thereof remains the property of Aerospace C.Power. The Information contained herein
 | 
						|
is believed to be accurate and Aerospace C.Power assumes no responsibility or
 | 
						|
liability for its use in any way and conveys no license or title under
 | 
						|
any patent or copyright and makes no representation or warranty that this
 | 
						|
Information is free from patent or copyright infringement.
 | 
						|
 | 
						|
****************************************************************************/
 | 
						|
 | 
						|
/* os shim includes */
 | 
						|
#include "iot_config.h"
 | 
						|
#include "os_types.h"
 | 
						|
#include "hw_reg_api.h"
 | 
						|
#include "iot_mem.h"
 | 
						|
 | 
						|
/* plc public api includes */
 | 
						|
#include "plc_fr.h"
 | 
						|
#include "plc_const.h"
 | 
						|
#include "iot_bitops.h"
 | 
						|
 | 
						|
/* driver includes */
 | 
						|
#include "iot_system.h"
 | 
						|
#include "iot_irq.h"
 | 
						|
#include "iot_gpio_api.h"
 | 
						|
#include "iot_board_api.h"
 | 
						|
#include "iot_errno_api.h"
 | 
						|
#include "pin_rf.h"
 | 
						|
 | 
						|
#include "mpdu_header.h"
 | 
						|
#include "mac_sys_reg.h"
 | 
						|
#include "plc_protocol.h"
 | 
						|
#include "mac_raw_reg.h"
 | 
						|
#include "mac_rawdata_hw.h"
 | 
						|
#include "phy_reg.h"
 | 
						|
#include "phy_bb.h"
 | 
						|
#include "phy_phase.h"
 | 
						|
#include "phy_isr.h"
 | 
						|
#include "ahb.h"
 | 
						|
#include "os_timer_api.h"
 | 
						|
#include "phy_overstress.h"
 | 
						|
#include "hw_phy_api.h"
 | 
						|
 | 
						|
iot_plc_over_stress_cb_t mac_ovr_stress_isr_cb = NULL;
 | 
						|
 | 
						|
/* MAC callback func register */
 | 
						|
void register_over_stress_cb(iot_plc_over_stress_cb_t cb)
 | 
						|
{
 | 
						|
    mac_ovr_stress_isr_cb = cb;
 | 
						|
}
 | 
						|
 | 
						|
bool_t phy_overstress_is_on(void)
 | 
						|
{
 | 
						|
    bool_t overstress_is_on = false;
 | 
						|
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint8_t over_stress_gpio = 0;
 | 
						|
    uint32_t irq_type_reg  = 0;
 | 
						|
    uint32_t reg_int = 0;
 | 
						|
 | 
						|
    /* lic with geode in and other geode out */
 | 
						|
    if(iot_chip_check_lic()) {/* internal PA */
 | 
						|
        /* clr intr*/
 | 
						|
        reg_int = PHY_READ_REG(CFG_BB_INT_CLR_1_ADDR);
 | 
						|
        reg_int |=PHY_LIC_OVR_STRESS;
 | 
						|
        PHY_WRITE_REG(CFG_BB_INT_CLR_1_ADDR, reg_int);
 | 
						|
 | 
						|
        /* check intr src */
 | 
						|
        irq_type_reg = PHY_READ_REG(CFG_BB_INT_RAW_1_ADDR);
 | 
						|
        if(irq_type_reg & PHY_LIC_OVR_STRESS){
 | 
						|
            return true;
 | 
						|
        } else {
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
    } else {/* external PA */
 | 
						|
        over_stress_gpio = iot_board_get_gpio(GPIO_GEODE_OVT);
 | 
						|
        if(iot_gpio_value_get(over_stress_gpio)) {
 | 
						|
            return false;
 | 
						|
        } else {
 | 
						|
            return true;
 | 
						|
        }
 | 
						|
    }
 | 
						|
#endif
 | 
						|
 | 
						|
    return overstress_is_on;
 | 
						|
}
 | 
						|
 | 
						|
/* TODO: use macro distinguish WARID, as WAR_BUGID_244 in iot_config.h */
 | 
						|
uint32_t IRAM_ATTR phy_get_hw_interrupt_type(void)
 | 
						|
{
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint32_t irq_type_reg  = 0;
 | 
						|
    uint32_t irq_type = 0;
 | 
						|
 | 
						|
    /* get int 1 mask */
 | 
						|
    irq_type_reg = PHY_READ_REG(CFG_BB_INT_MASK_1_ADDR);
 | 
						|
    if(irq_type_reg & PHY_LIC_OVR_STRESS){
 | 
						|
        irq_type |= INTR_TYPE_PHY_LIC_OVR_STRESS;
 | 
						|
    }
 | 
						|
    return irq_type;
 | 
						|
#else
 | 
						|
    return 0;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
void IRAM_ATTR phy_disable_hw_interrupt(uint32_t irq_type)
 | 
						|
{
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint32_t reg_int = 0;
 | 
						|
 | 
						|
    /* disable int 1 */
 | 
						|
    if (irq_type & INTR_TYPE_PHY_LIC_OVR_STRESS) {
 | 
						|
        reg_int = PHY_READ_REG(CFG_BB_INT_EN_1_ADDR);
 | 
						|
        reg_int &= (~PHY_LIC_OVR_STRESS);
 | 
						|
        PHY_WRITE_REG(CFG_BB_INT_EN_1_ADDR, reg_int);
 | 
						|
    }
 | 
						|
#else
 | 
						|
    (void)irq_type;
 | 
						|
#endif
 | 
						|
    return;
 | 
						|
}
 | 
						|
 | 
						|
void IRAM_ATTR phy_clear_hw_interrupt(uint32_t irq_type)
 | 
						|
{
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint32_t reg_int = 0;
 | 
						|
 | 
						|
    /* clear int 1 */
 | 
						|
    if (irq_type & INTR_TYPE_PHY_LIC_OVR_STRESS) {
 | 
						|
        reg_int = PHY_READ_REG(CFG_BB_INT_CLR_1_ADDR);
 | 
						|
        reg_int |=PHY_LIC_OVR_STRESS;
 | 
						|
        PHY_WRITE_REG(CFG_BB_INT_CLR_1_ADDR, reg_int);
 | 
						|
    }
 | 
						|
#else
 | 
						|
    (void)irq_type;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
void IRAM_ATTR phy_enable_hw_interrupt(uint32_t irq_type)
 | 
						|
{
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint32_t reg_int = 0;
 | 
						|
 | 
						|
    /* enable int 1 */
 | 
						|
    if (irq_type & INTR_TYPE_PHY_LIC_OVR_STRESS) {
 | 
						|
        reg_int = PHY_READ_REG(CFG_BB_INT_EN_1_ADDR);
 | 
						|
        reg_int |=PHY_LIC_OVR_STRESS;
 | 
						|
        PHY_WRITE_REG(CFG_BB_INT_EN_1_ADDR, reg_int);
 | 
						|
    }
 | 
						|
#else
 | 
						|
    (void)irq_type;
 | 
						|
#endif
 | 
						|
    return;
 | 
						|
}
 | 
						|
 | 
						|
uint32_t IRAM_ATTR phy_interrupt_handler(uint32_t vector, iot_addrword_t data)
 | 
						|
{
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint32_t isr_flag = 0;
 | 
						|
 | 
						|
    isr_flag = phy_get_hw_interrupt_type();
 | 
						|
    /* clear cpu0 int */
 | 
						|
    phy_clear_hw_interrupt(isr_flag);
 | 
						|
 | 
						|
    /* disable interrupt mask */
 | 
						|
    phy_disable_hw_interrupt(isr_flag);
 | 
						|
 | 
						|
    /* turn down gain */
 | 
						|
    if (isr_flag & INTR_TYPE_PHY_LIC_OVR_STRESS) {
 | 
						|
        /* update cnt */
 | 
						|
        phy_add_overstress_cnt();
 | 
						|
        /* turn down power */
 | 
						|
        if (!g_phy_ctxt.indep.ovs_ctxt.ovr_stress) {
 | 
						|
            /* update global flag */
 | 
						|
            g_phy_ctxt.indep.ovs_ctxt.ovr_stress = 1;
 | 
						|
        }
 | 
						|
        /* mac callback */
 | 
						|
        if (mac_ovr_stress_isr_cb != NULL) {
 | 
						|
            mac_ovr_stress_isr_cb(NULL, 1);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (isr_flag & INTR_TYPE_PHY_LIC_OVR_STRESS){
 | 
						|
        /* disable over stress interrupt */
 | 
						|
        isr_flag &= ~INTR_TYPE_PHY_LIC_OVR_STRESS;
 | 
						|
        phy_enable_hw_interrupt(isr_flag);
 | 
						|
    }
 | 
						|
    else{
 | 
						|
        phy_enable_hw_interrupt(isr_flag);
 | 
						|
    }
 | 
						|
#else
 | 
						|
    (void)vector;
 | 
						|
    (void)data;
 | 
						|
#endif
 | 
						|
    return ERR_OK;
 | 
						|
}
 | 
						|
 | 
						|
void phy_gpio_interrupt_handler(int arg)
 | 
						|
{
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint8_t over_stress_gpio = iot_board_get_gpio(GPIO_GEODE_OVT);
 | 
						|
 | 
						|
    /* disable overstress gpio interrupt */
 | 
						|
    iot_gpio_interrupt_enable(over_stress_gpio, 0);
 | 
						|
 | 
						|
    /* add event cnt */
 | 
						|
    phy_add_overstress_cnt();
 | 
						|
 | 
						|
    /* turn down tx power with rms */
 | 
						|
    if(!g_phy_ctxt.indep.ovs_ctxt.ovr_stress) {
 | 
						|
        /* update global flag */
 | 
						|
        g_phy_ctxt.indep.ovs_ctxt.ovr_stress = 1;
 | 
						|
    }
 | 
						|
 | 
						|
    /* mac callback */
 | 
						|
    if(mac_ovr_stress_isr_cb != NULL){
 | 
						|
        mac_ovr_stress_isr_cb(NULL, 0);
 | 
						|
    }
 | 
						|
#endif
 | 
						|
 | 
						|
    (void)arg;
 | 
						|
    return;
 | 
						|
}
 | 
						|
 | 
						|
uint32_t phy_gpio_interrupt_init()
 | 
						|
{
 | 
						|
    uint32_t ret = ERR_FAIL;
 | 
						|
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint8_t over_stress_gpio = iot_board_get_gpio(GPIO_GEODE_OVT);
 | 
						|
 | 
						|
    ret = iot_gpio_open_as_interrupt(over_stress_gpio);
 | 
						|
    ret |= iot_gpio_interrupt_config(over_stress_gpio,
 | 
						|
        GPIO_INT_LEVEL_LOW, phy_gpio_interrupt_handler, 0,
 | 
						|
        GPIO_INT_FUNC_ENABLE_AUTOSTOP);
 | 
						|
    ret |= iot_gpio_interrupt_enable(over_stress_gpio, 1);
 | 
						|
 | 
						|
    if(ret){
 | 
						|
        iot_gpio_close(over_stress_gpio);
 | 
						|
    }
 | 
						|
#endif
 | 
						|
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 | 
						|
static void phy_isr_clean(void)
 | 
						|
{
 | 
						|
    /* disable all interrupt and clean up any pending interrupt */
 | 
						|
    PHY_WRITE_REG(CFG_BB_INT_EN_0_ADDR, 0);
 | 
						|
    PHY_WRITE_REG(CFG_BB_INT_EN_1_ADDR, 0);
 | 
						|
    PHY_WRITE_REG(CFG_BB_INT_EN_2_ADDR, 0);
 | 
						|
    //INT_EN_3 map to cpu1
 | 
						|
 | 
						|
    /* clean up all pending interrupt */
 | 
						|
    PHY_WRITE_REG(CFG_BB_INT_CLR_0_ADDR, BB_INT_CLR_0_MASK);
 | 
						|
    PHY_WRITE_REG(CFG_BB_INT_CLR_1_ADDR, BB_INT_CLR_1_MASK);
 | 
						|
    PHY_WRITE_REG(CFG_BB_INT_CLR_2_ADDR, BB_INT_CLR_2_MASK);
 | 
						|
}
 | 
						|
 | 
						|
void phy_isr_clean_cpu1(void)
 | 
						|
{
 | 
						|
    /* disable all interrupt and clean up any pending interrupt */
 | 
						|
    PHY_WRITE_REG(CFG_BB_INT_EN_3_ADDR, 0);
 | 
						|
 | 
						|
    /* clean up all pending interrupt */
 | 
						|
    PHY_WRITE_REG(CFG_BB_INT_CLR_3_ADDR, BB_INT_CLR_3_MASK);
 | 
						|
}
 | 
						|
 | 
						|
void phy_isr_init()
 | 
						|
{
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    iot_irq_t phy_fd_fc_err_irqhdl;
 | 
						|
    iot_irq_t phy_lic_ovr_stress_irqhdl;
 | 
						|
 | 
						|
    /* reset phy interrupt reg */
 | 
						|
    phy_isr_clean();
 | 
						|
 | 
						|
    /* k48 or k68 */
 | 
						|
    if(iot_chip_check_lic()) {/* internal PA */
 | 
						|
        phy_lic_ovr_stress_irqhdl = \
 | 
						|
            iot_interrupt_create(IRQ_NUM_PHY_LIC_OVR_STRESS, HAL_INTR_PRI_0,
 | 
						|
                                0, phy_interrupt_handler);
 | 
						|
        iot_interrupt_attach(phy_lic_ovr_stress_irqhdl);
 | 
						|
        iot_interrupt_unmask(phy_lic_ovr_stress_irqhdl);
 | 
						|
    } else {/* external PA */
 | 
						|
        phy_gpio_interrupt_init();
 | 
						|
    }
 | 
						|
 | 
						|
    phy_fd_fc_err_irqhdl = \
 | 
						|
        iot_interrupt_create(HAL_VECTOR_PHY_2, HAL_INTR_PRI_6,
 | 
						|
                            0, phy_interrupt_handler);
 | 
						|
    iot_interrupt_attach(phy_fd_fc_err_irqhdl);
 | 
						|
    iot_interrupt_unmask(phy_fd_fc_err_irqhdl);
 | 
						|
 | 
						|
#endif
 | 
						|
 | 
						|
    return;
 | 
						|
}
 | 
						|
 | 
						|
void phy_isr_start()
 | 
						|
{
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint32_t reg_int;
 | 
						|
 | 
						|
    /* bind phy int 1 */
 | 
						|
    if(iot_chip_check_lic()){/* internal PA */
 | 
						|
        reg_int = PHY_READ_REG(CFG_BB_INT_EN_1_ADDR);
 | 
						|
        reg_int |= PHY_LIC_OVR_STRESS;
 | 
						|
        PHY_WRITE_REG(CFG_BB_INT_EN_1_ADDR, reg_int);
 | 
						|
    }
 | 
						|
#endif
 | 
						|
    return;
 | 
						|
}
 | 
						|
 | 
						|
void phy_isr_start_cpu1(void)
 | 
						|
{
 | 
						|
#if HW_PLATFORM >= HW_PLATFORM_FPGA
 | 
						|
    uint32_t reg_int = 0;
 | 
						|
 | 
						|
    phy_isr_clean_cpu1();
 | 
						|
    /* TODO: enable isr need use api phy_enable_hw_interrupt_cpu1() */
 | 
						|
    /* bind phy int 3 */
 | 
						|
    reg_int = PHY_RECV_FD_FC_OK;
 | 
						|
    reg_int |= PHY_RECV_FD_FC_FAIL;
 | 
						|
    reg_int |= PHY_TX_FD_TX_DONE;
 | 
						|
 | 
						|
#if ENA_WAR_NSG_EXTMI || MAC_SYM_NUM_FIX
 | 
						|
        reg_int |= PHY_RECV_FD_CH_EST_DONE;
 | 
						|
#endif/* ENA_WAR_NSG_EXTMI || MAC_SYM_NUM_FIX */
 | 
						|
 | 
						|
#if ENA_WAR_NSG_EXTMI
 | 
						|
        reg_int |= PHY_RECV_FD_PB_OK;
 | 
						|
        reg_int |= PHY_RECV_FD_PB_FAIL;
 | 
						|
        reg_int |= PHY_RECV_FC_RAW_RECEIVE;
 | 
						|
#endif/* ENA_WAR_NSG_EXTMI */
 | 
						|
 | 
						|
#if (ENA_WAR_NSG_EXTMI || ENA_SW_SYNC_PPM_WAR)
 | 
						|
    reg_int |= PHY_TX_FD_INSERT_PREAM_DONE;
 | 
						|
#endif
 | 
						|
 | 
						|
#if (ENA_WAR_SPG_TX_OK || DEBUG_NID_ERR || ENA_CCO_RX_THR_PHASE_CHANGE)
 | 
						|
    reg_int |= PHY_TX_TD_FC_DONE;
 | 
						|
#endif
 | 
						|
 | 
						|
#if ENA_WAR_244
 | 
						|
    reg_int |= PHY_FD_TX_ABORT;
 | 
						|
    reg_int |= PHY_FD_TX_STUCK;
 | 
						|
    reg_int |= PHY_TX_TD_START;
 | 
						|
#endif
 | 
						|
#if MAC_SYM_NUM_FIX || MAC_TIMESTAMPING || ENA_CCO_RX_THR_PHASE_CHANGE
 | 
						|
    reg_int |= PHY_RECV_FD_PLD_OK;
 | 
						|
    reg_int |= PHY_RECV_FD_PLD_FAIL;
 | 
						|
#endif
 | 
						|
    PHY_WRITE_REG(CFG_BB_INT_EN_3_ADDR, reg_int);
 | 
						|
#endif
 | 
						|
    return;
 | 
						|
}
 |