STM32 休眠模式下如何喂狗?

发布者:TranquilSmile最新更新时间:2018-05-14 来源: eefocus关键字:STM32  休眠模式  喂狗 手机看文章 扫描二维码
随时随地手机看文章

在STM32开发中经常会用到独立看门狗(IWDG)和低功耗模式,看门狗是为了检测和解决由软件错误引起的故障,低功耗模式是为了在CPU不需要继续运行时进入到休眠模式用以节省电能。其中独立看门狗的时钟由独立的RC振荡器(STM32F10x一般为40kHz)提供,即使在主时钟出现故障时,也仍然有效,因此可以在停止和待机模式下工作。而且独立看门狗一旦启动,除了系统复位,它不能再被停止。但这样引发的一个问题是当MCU进入到低功耗模式后由于CPU停止运行无法喂狗,会导致系统频繁复位。那如何解决这个问题呢,难道独立看门狗和低功耗模式没法同时使用?

一个很好的方式是在休眠模式下通过RTC定时唤醒来喂狗,喂完够在进入继续进入到休眠模式。比如看门狗复位的时间间隔为10s。那么在进入休眠模式前设置RTC闹钟中断时间为5s。这样每隔5s唤醒一次喂一次狗。便可以很好的解决这个问题。

while(1)  

  {  

    // 执行任务  

    Task1();  

    Task2();  

    // ..  

  

    // 喂狗  

    dev_iwdg_feed();  

  

    // 进入待机模式开关  

    if(m_bEnterStandByMode)  

    {     

        // 使能外部中断,GPIOB3,用以MCU从待机模式唤醒  

        dev_exti_enable(TRUE);  

ENTERSTOPMODE:    

        // 设置RTC闹钟,5秒钟产生一次RTC闹钟中断*/  

        dev_rtc_setAlarm(5);  

      

        // 进入停止模式(低功耗),直至外部中断触发时被唤醒  

        PWR_EnterSTOPMode(PWR_Regulator_LowPower, PWR_STOPEntry_WFI);  

      

        // 是否是RTC闹钟中断唤醒  

        if(dev_rtc_isAlarm())  

        {  

            // 喂狗  

            dev_iwdg_feed();  

            // 喂完狗继续进入停止模式  

            goto ENTERSTOPMODE;   

        }  

        // 禁止外部中断   

        dev_exti_enable(FALSE);  

        // 从停止模式唤醒后恢复系统时钟  

        dev_clk_restore();  

    }               

  }  


以下是完整的参考代码:

 

//**********************************************************************************************       

//  STM32F10x StopMode RTC Feed Dog   

//  compiler: Keil UV3       

//  2013-01-04 , By friehood       

//**********************************************************************************************    

#include "stm32f10x_lib.h"  

#include "platform_config.h"  

static Boolean g_bRTCAlarm = FALSE;  

  

/*******************************************************************************  

* Function Name  : RCC_Configuration  

* Description    : Configures the different system clocks.  

* Input          : None  

* Output         : None  

* Return         : None  

*******************************************************************************/  

void RCC_Configuration(void)  

{  

    /* RCC system reset(for debug purpose) */  

    RCC_DeInit();  

  

    /* Enable HSE */  

    RCC_HSEConfig(RCC_HSE_ON);  

  

    /* Wait till HSE is ready */  

    if(RCC_WaitForHSEStartUp() == SUCCESS)  

    {  

        /* Enable Prefetch Buffer */  

        FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);  

  

        //FLASH时序控制   

        //推荐值:SYSCLK = 0~24MHz   Latency=0   

        //        SYSCLK = 24~48MHz  Latency=1   

        //        SYSCLK = 48~72MHz  Latency=2  

        //FLASH_SetLatency(FLASH_Latency_1);        //警告:修改为1会对DMA值有影响(如ADC采集值会错位)  

        FLASH_SetLatency(FLASH_Latency_2);  

  

        /* HCLK = SYSCLK */  

        RCC_HCLKConfig(RCC_SYSCLK_Div1);   

  

        /* PCLK2 = HCLK */  

        RCC_PCLK2Config(RCC_HCLK_Div1);   

  

        /* PCLK1 = HCLK/2 */  

        RCC_PCLK1Config(RCC_HCLK_Div2);  

  

        /* PLLCLK = 12MHz * 3 = 36 MHz */  

        RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_3);  

  

        /* Enable PLL */   

        RCC_PLLCmd(ENABLE);  

  

        /* Wait till PLL is ready */  

        while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)  

        {  

        }  

  

        /* Select PLL as system clock source */  

        RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);  

  

        /* Wait till PLL is used as system clock source */  

        while(RCC_GetSYSCLKSource() != 0x08)  

        {  

        }  

    }  

    /* Enable PWR and BKP clock */  

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE);  

  

    /* Enable AFIO clock */  

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);  

}  

  

/******************************************************************************* 

* Function Name  : NVIC_Configuration 

* Description    : Configures the nested vectored interrupt controller. 

* Input          : None 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void NVIC_Configuration(void)  

{  

  NVIC_InitTypeDef NVIC_InitStructure;  

  

#ifdef  VECT_TAB_RAM  

  /* Set the Vector Table base location at 0x20000000 */  

  NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0);  

#else  /* VECT_TAB_FLASH  */  

  /* Set the Vector Table base location at 0x08000000 */  

  NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);  

#endif  

  

  /* Configure one bit for preemption priority */  

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);  

}  

  

/******************************************************************************* 

* Function Name  : SysTick_Configuration 

* Description    : Configures the SysTick to generate an interrupt each 1 millisecond. 

* Input          : None 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void SysTick_Configuration(void)  

{  

  /* Select AHB clock(HCLK) as SysTick clock source */  

  SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK);  

  

  /* Set SysTick Priority to 3 */  

  NVIC_SystemHandlerPriorityConfig(SystemHandler_SysTick, 3, 0);  

     

  /* SysTick interrupt each 1ms with HCLK equal to 72MHz */  

  SysTick_SetReload(72000);  

  

  /* Enable the SysTick Interrupt */  

  SysTick_ITConfig(ENABLE);  

}  

  

/******************************************************************************* 

* Function Name  : Delay 

* Description    : Inserts a delay time. 

* Input          : nTime: specifies the delay time length, in milliseconds. 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void Delay(u32 nTime)  

{  

  /* Enable the SysTick Counter */  

  SysTick_CounterCmd(SysTick_Counter_Enable);  

    

  TimingDelay = nTime;  

  

  while(TimingDelay != 0);  

  

  /* Disable the SysTick Counter */  

  SysTick_CounterCmd(SysTick_Counter_Disable);  

  /* Clear the SysTick Counter */  

  SysTick_CounterCmd(SysTick_Counter_Clear);  

}  

  

  

/******************************************************************************* 

* Function Name  : RTC_Configuration 

* Description    : Configures RTC clock source and prescaler. 

* Input          : None 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void RTC_Configuration(void)  

{  

    EXTI_InitTypeDef EXTI_InitStructure;  

    NVIC_InitTypeDef NVIC_InitStructure;  

      

    /* RTC clock source configuration ------------------------------------------*/  

    /* Allow access to BKP Domain */  

    PWR_BackupAccessCmd(ENABLE);  

  

    /* Reset Backup Domain */  

    BKP_DeInit();  

  

    /* Enable the LSI OSC */  

    RCC_LSICmd(ENABLE);  

  

    /* Wait till LSI is ready */  

    while (RCC_GetFlagStatus(RCC_FLAG_LSIRDY) == RESET){}  

      

    /* Select the RTC Clock Source */  

    RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);  

  

    /* Enable the RTC Clock */  

    RCC_RTCCLKCmd(ENABLE);  

  

    /* RTC configuration -------------------------------------------------------*/  

    /* Wait for RTC APB registers synchronisation */  

    RTC_WaitForSynchro();  

  

    /* Set RTC prescaler: set RTC period to 1sec */  

    RTC_SetPrescaler(40000);  

      

    /* Wait until last write operation on RTC registers has finished */  

    RTC_WaitForLastTask();  

  

    /* Enable the RTC Alarm interrupt */  

    RTC_ITConfig(RTC_IT_ALR, ENABLE);  

      

    /* Wait until last write operation on RTC registers has finished */  

    RTC_WaitForLastTask();  

  

    /* Configure EXTI Line17(RTC Alarm) to generate an interrupt on rising edge */  

    EXTI_ClearITPendingBit(EXTI_Line17);  

    EXTI_InitStructure.EXTI_Line = EXTI_Line17;  

    EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;  

    EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;  

    EXTI_InitStructure.EXTI_LineCmd = ENABLE;  

    EXTI_Init(&EXTI_InitStructure);  

  

    /* Enable the RTC Interrupt */  

    NVIC_InitStructure.NVIC_IRQChannel = RTCAlarm_IRQChannel;  

    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;  

    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;  

    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  

    NVIC_Init(&NVIC_InitStructure);  

}  

  

/******************************************************************************* 

* Function Name  : RTCAlarm_IRQHandler 

* Description    : This function handles RTC Alarm interrupt request. 

* Input          : None 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void RTCAlarm_IRQHandler(void)  

{  

    if(RTC_GetITStatus(RTC_IT_ALR) != RESET)  

    {  

        /* Set the RTC alarm flag */  

        g_bRTCAlarm = TRUE;  

  

        /* Clear EXTI line17 pending bit */  

        EXTI_ClearITPendingBit(EXTI_Line17);  

  

        /* Check if the Wake-Up flag is set */  

        if(PWR_GetFlagStatus(PWR_FLAG_WU) != RESET)  

        {  

            /* Clear Wake Up flag */  

            PWR_ClearFlag(PWR_FLAG_WU);  

        }                                                                         

                                                           

        /* Wait until last write operation on RTC registers has finished */  

        RTC_WaitForLastTask();     

        /* Clear RTC Alarm interrupt pending bit */  

        RTC_ClearITPendingBit(RTC_IT_ALR);  

        /* Wait until last write operation on RTC registers has finished */  

        RTC_WaitForLastTask();  

    }  

}  

  

/******************************************************************************* 

* Function Name  : dev_rtc_setAlarm 

* Description    : 设置RTC闹钟. 

* Input          : 闹钟时间 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void dev_rtc_setAlarm(u32 AlarmValue)  

{  

    /* Clear the RTC SEC flag */  

    RTC_ClearFlag(RTC_FLAG_SEC);  

    /* Wait clear RTC flag sccess */  

    while(RTC_GetFlagStatus(RTC_FLAG_SEC) == RESET);  

    /* Wait until last write operation on RTC registers has finished */  

    RTC_WaitForLastTask();   

  

    /* Sets the RTC alarm value */  

    RTC_SetAlarm(RTC_GetCounter() + AlarmValue);  

    /* Wait until last write operation on RTC registers has finished */  

    RTC_WaitForLastTask();   

}  

  

/******************************************************************************* 

* Function Name  : dev_rtc_isAlarm 

* Description    : RTC闹钟是否触发 

* Input          : None 

* Output         : None 

* Return         : TRUE:已触发,FALSE,未触发 

*******************************************************************************/  

Boolean dev_rtc_isAlarm(void)  

{  

    if(g_bRTCAlarm)  

    {  

        /* Clear the RTC alarm flag */  

        g_bRTCAlarm = FALSE;  

        return TRUE;  

    }  

    return FALSE;             

}  

  

void dev_iwdg_init(void)  

{  

    /* Enable write access to IWDG_PR and IWDG_RLR registers */  

    IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);  

    /* IWDG counter clock: 40KHz(LSI) / 256 = 0.15625 KHz */  

    IWDG_SetPrescaler(IWDG_Prescaler_256);  

    /* Set counter reload value to 1562 */  

    IWDG_SetReload(1562);   // 10s  

    /* Reload IWDG counter */  

    IWDG_ReloadCounter();  

    /* Enable IWDG (the LSI oscillator will be enabled by hardware) */  

    IWDG_Enable();  

}  

  

void dev_iwdg_feed(void)  

{  

    IWDG_ReloadCounter();  

}  

  

/******************************************************************************* 

* Function Name  : dev_clk_restore 

* Description    : Restore system clock after wake-up from STOP: enable HSE, PLL 

*                  and select PLL as system clock source. 

* Input          : None 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void dev_clk_restore(void)  

{  

  /* Enable HSE */  

  RCC_HSEConfig(RCC_HSE_ON);  

  

  /* Wait till HSE is ready */  

  HSEStartUpStatus = RCC_WaitForHSEStartUp();  

  

  if(HSEStartUpStatus == SUCCESS)  

  {  

    /* Enable PLL */   

    RCC_PLLCmd(ENABLE);  

  

    /* Wait till PLL is ready */  

    while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)  

    {  

    }  

  

    /* Select PLL as system clock source */  

    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);  

  

    /* Wait till PLL is used as system clock source */  

    while(RCC_GetSYSCLKSource() != 0x08)  

    {  

    }  

  }  

}  

  

/******************************************************************************* 

* Function Name  : EXTI_Configuration 

* Description    : Configures EXTI Line3. 

* Input          : None 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void EXIT_Configuration(void)  

{  

    EXTI_InitTypeDef EXTI_InitStructure;  

  

    GPIO_EXTILineConfig(GPIO_PortSourceGPIOB,GPIO_PinSource3);  

    EXTI_ClearITPendingBit(EXTI_Line3);  

    EXTI_InitStructure.EXTI_Line = EXTI_Line3;  

    EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;                                         

    EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;   

    EXTI_InitStructure.EXTI_LineCmd = ENABLE;  

    EXTI_Init(&EXTI_InitStructure);  

}  

  

void dev_exti_enable(Boolean bEnable)  

{  

    NVIC_InitTypeDef NVIC_InitStructure;  

  

    /* Clear the Key Button EXTI line pending bit */  

    EXTI_ClearITPendingBit(EXTI_Line3);  

  

    NVIC_ClearIRQChannelPendingBit(EXTI3_IRQChannel);  

    NVIC_InitStructure.NVIC_IRQChannel = EXTI3_IRQChannel;  

    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;  

    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;  

    NVIC_InitStructure.NVIC_IRQChannelCmd = bEnable ? ENABLE : DISABLE;  

    NVIC_Init(&NVIC_InitStructure);   

}  

  

/******************************************************************************* 

* Function Name  : main 

* Description    : Main program. 

* Input          : None 

* Output         : None 

* Return         : None 

*******************************************************************************/  

int main(void)  

{  

  /* System Clocks Configuration */  

  RCC_Configuration();  

  

  /* NVIC configuration */  

  NVIC_Configuration();  

  

  /* Configure RTC clock source and prescaler */  

  RTC_Configuration();  

  

  /* Configure the SysTick to generate an interrupt each 1 millisecond */  

  SysTick_Configuration();  

  

  /* Configures EXTI Line3 */  

  EXIT_Configuration();  

  

  /* IWDG initialize*/  

  dev_iwdg_init();  

  

  while(1)  

  {  

    // 执行任务  

    Task1();  

    Task2();  

    // ..  

  

    // 喂狗  

    dev_iwdg_feed();  

  

    // 进入待机模式开关  

    if(m_bEnterStandByMode)  

    {     

        // 使能外部中断,GPIOB3,用以MCU从待机模式唤醒  

        dev_exti_enable(TRUE);  

ENTERSTOPMODE:    

        // 设置RTC闹钟,5秒钟产生一次RTC闹钟中断*/  

        dev_rtc_setAlarm(5);  

      

        // 进入停止模式(低功耗),直至外部中断触发时被唤醒  

        PWR_EnterSTOPMode(PWR_Regulator_LowPower, PWR_STOPEntry_WFI);  

      

        // 是否是RTC闹钟中断唤醒  

        if(dev_rtc_isAlarm())  

        {  

            // 喂狗  

            dev_iwdg_feed();  

            // 喂完狗继续进入停止模式  

            goto ENTERSTOPMODE;   

        }  

        // 禁止外部中断   

        dev_exti_enable(FALSE);  

        // 从停止模式唤醒后恢复系统时钟  

        dev_clk_restore();  

    }               

  }  

}  


关键字:STM32  休眠模式  喂狗 引用地址:STM32 休眠模式下如何喂狗?

上一篇:STM32F4 ADC1 模拟看门狗【库函数操作】
下一篇:STM32f030c8t6单片机ADC模拟看门狗使用

推荐阅读最新更新时间:2024-03-16 16:02

STM32之GPIO及第一个STM32程序(跑马灯)
今天来说一说,GPIO,对于我这个新手来说,GPIO就好比我在学习开车之前得学会如何开门一样,由此可以看出这对于我学习STM32 的重要性,好废话不多说,先总结一下STM32F103ZE的开发板里总共有7组IO口,每组IO口有16个IO,即这块板子总共有112个IO口分别是GPIOA~GPIOG。 GPIO的工作模式主要有八种:4种输入方式,4种输出方式,分别为输入浮空,输入上拉,输入下拉,模拟输入;输出方式为开漏输出,开漏复用输出,推挽输出,推挽复用输出。对应的为: (1)GPIO_Mode_AIN 模拟输入 (2)GPIO_Mode_IN_FLOATING 浮空输入 (3)GPIO_Mode_IPD 下拉输入 (4)GPIO_
[单片机]
STM32开发 -- UART应用层通信协议分析
拿到一份UART的通信协议,上手来操作之前先做一下分析。 一、帧格式说明 先看一下它的帧格式说明: 1、 帧头标志Head 不论是命令帧还是响应帧,帧头标志都是0x92。 2、 协议版本 协议版本号(4bit),目前值为1 加密方式(4bit),0表示采取“数据不加密+校验和”方式。 所以,当前此字段完整值为0x10 3、 控制字段中的C/R比特 用于指示该帧是命令帧还是应答帧,1表示命令帧,0表示应答帧。 4、 控制字段中的T/F比特 用于指示传输数据类型,1表示透明的非结构化数据,0表示正常的数据帧。 5、 虚拟通道 虚拟地址(4bit):代表数据帧的源地址标识。目前为保留位,取值为0。 通道序号(4bit):0表示虚通道0
[单片机]
STM32开发 -- Git的详细使用
在GPS部分有用到DMA,接下来看一下它的使用。 一、DMA简介 直接存储器存取(DMA) 用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输。无须CPU干预,数据可以通过DMA快速地移动,这就节省了CPU的资源来做其他操作。 两个DMA控制器有12个通道(DMA1有7个通道, DMA2有5个通道),每个通道专门用来管理来自于一个或多个外设对存储器访问的请求。还有一个仲裁器来协调各个DMA请求的优先权。 二、DMA主要特性 ● 12个独立的可配置的通道(请求): DMA1有7个通道, DMA2有5个通道 ● 每个通道都直接连接专用的硬件DMA请求,每个通道都同样支持软件触发。这些功能通过软件来配置。 ● 在同一
[单片机]
<font color='red'>STM32</font>开发 -- Git的详细使用
使用STM32的USB模块中后对USB缓冲区的认识
最近在使用STM32的USB模块开发个项目,还以为挺简单,结果搞了快两天才把USB的包缓冲区的访问搞定,在此做个小总结吧。 STM32的USB模块包缓冲区有512B,但是在STM32的参考手册中的存储器映像中却表明0x40006000-0x400063ff,整整多了512B,怎么会这样呢,同时在尝试着编程时也遇到了一个问题: 在usb_core.c文件的Setup0_Process(void)这个函数中,有这么一段: uint16_t offset = 1; if (pInformation- ControlState != PAUSE) { pInformation- USBbmRequestType = *pBuf.b
[单片机]
STM32 定时器详解
前言 STM32 单片机的定时器分为高级定时器、 通用定时器 、基本定时器三种。这三个定时器成上下级的关系,即基本定时器有的功能通用定时器都有,而且还增加了向下、向上/向下计数器、PWM生成、输出比较、输入捕获等功能; 而高级定时器又包含了通用定时器的所有功能,另外还增加了死区互补输出、刹车信号。(此处我们暂时忽略,暂时我们暂时还不需要接触这些功能) 一、STM32 通用定时器简介 通用定时器是一个通过可编程预分频器驱动的16位自动装载计数器构成。它适用于多种场合,包括测量输入信号的脉冲长度(输入捕获)或者产生输出波形(输出比较和PWM)。使用定时器预分频器和RCC时钟控制器预分频器,脉冲长度和波形周期可以在几个微秒到几个
[单片机]
STM32入门开发--按键模块实现按键点灯
1、实现效果 通过I/O口,操作按键,使得按键与LED一一对应,实现按键按下灯亮,再按下灯灭。支持连续按键。 2、实现思路 a. 要实现灯亮,首先应该初始化LED模块的时钟,使得该端口的时钟使能。 b. 初始化按键,按键对应的端口PORTA,PORTE使能,其次设置为输入模式, KEY0,KEY1为低电平有效,KEY_UP高电平有效 硬件图可知,在设计中,应该在内部设置为上拉下拉输入,即KEY0,KEY1为上拉(输入高电平,本身为低电平有效),KEY_UP为下拉(输入低电平,本身为高电平有效)。 根据模式配置图,即要设置GOIPx—ODR寄存器的3/4位为1,才能保证为上拉输入。 c. 实现相应的初始化
[单片机]
<font color='red'>STM32</font>入门开发--按键模块实现按键点灯
STM32 keil5编译优化
本来是写个IAP程序,Flash中只规划了4K字节,但是代码编写完后有6k,所以寻找优化方案,优化完在1.7k左右。 STM32 keil5编译优化,可以勾选一下几项: 1、Options for Taeget-- Target 勾选使用微库 2、Options for Taeget-- Target ①选择优化级别Level 3(-O3) ②勾选One ELF Section per Function 3、关于One ELF Section per Function的作用特别说明下: ①编译器在处理一个 c 文件的时候呢,如果这个选项不选,那么这个C文件中的所有函数在编译后只会产生一个叫 .text 的输出节:
[单片机]
基于STM32对DS1302的驱动
// 程序名: STM32驱动DS1302 //头文件 #include “stm32f10x.h” #include “usart.h” #define uchar unsigned char #define uint unsigned int ////DS1302引脚定义,可根据实际情况自行修改端口定义 #define RST PAout(5) #define IO PAout(6) #define SCK PAout(7) //DS1302地址定义 #define ds1302_sec_add 0x80 //秒数据地址 #define ds1302_min_add 0x82 //分数据地址 #define ds1302_
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved