MCU外设初始化:为什么参数配置必须优先于使能

在微控制器领域,初始化参数配置阶段至关重要。此时,虽无电源驱动,但微控制器在使能信号到来前,借初始化参数配置这一精细步骤,开启关键准备进程。初始化参数配置如同物理坐标锚定、逻辑指令部署、内在秩序预设,各参数像深埋沃土的种子,坐标、朝向、深度经精密计算,为未来指令运行奠定基础。

下面以国科安芯的MCU芯片AS32A601为例,详细展示下MCU这一严格的设计特性:

‌1. ‌外设检测阶段‌:MCU会尝试检测外设可用性,然后才开始执行用户代码。

2. 时钟树配置‌:系统时钟(CK_SYS)、AHB、APB等总线时钟必须在其他外设初始化前完成配置。

为什么参数要在使能前配置?

避免电平跳变‌:

  • GPIO复用模式下,若先使能外设再配置复用选择器,会导致短暂电平变化。
  • 普通输出IO默认输出低电平,若先使能再设置高电平,会出现短暂低脉冲。

‌ 防止硬件冲突‌:

  • 时钟使能必须在外设初始化之前,否则会导致外设无法正常工作。
  • 寄存器默认值可能不符合应用需求,直接使能可能导致意外行为。

确保稳定状态‌:

  • 外设使能前需要建立正确的时钟源、中断优先级等基础环境。
  • 参数配置需要一定时间生效,立即使能可能导致功能异常。

时钟配置

  1. 通过阅读芯片手册,确认好项目所需外设所在时钟
  2. 确保时钟最先配置,再去配置外设

01

02

03

04

05

06

07

08

09

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

void Systemclock_Init()

{

    //注意:此处需要开启系统总线级的时钟配置,具体外设时钟配置可在各模块初始化函数中具体开启,具体请参考时钟树或者下图注释

    //      1. 使用串口时,由于串口挂在APB0总线下,需要在此处开启AXIBUS3时钟、AXI4TOAPB0时钟以及APBBUS0时钟。

    //      2. 使用延时函数时,需要开启CLINT时钟

    //      3. 使用eflash、qspi时,需要开启AXIBUS3时钟、AXILITEBUS2时钟

     

    /* AXIBus3 clock operation Guide*/

    AXIBUS3_CLK_ENABLE();

    AXI4TOAPB0_CLK_ENABLE();

    APBBUS0_CLK_ENABLE();

    AXI4TOAPB1_CLK_ENABLE();

    APBBUS1_CLK_ENABLE();

    AXILITEBUS1_CLK_ENABLE();

    AXILITEBUS2_CLK_ENABLE();

    EFLASH_CLK_ENABLE();  

    PLIC_CLK_ENABLE();

    CLINT_CLK_ENABLE();

     

    SMU_PLLInitTypeDef SMU_PLLInitStruct;

    SMU_ClockInitTypeDef SMU_ClockInitStruct;

   

    /* Set PLL parameters values */

    SMU_PLLInitStruct.OscillatorType = SMU_OSCILLATORTYPE_OSC;

    SMU_PLLInitStruct.FIRCOscState = DISABLE;

    SMU_PLLInitStruct.FIRCCalibrationValue = 0x00;

    SMU_PLLInitStruct.PLLConfig.PLLState = ENABLE;

    SMU_PLLInitStruct.PLLConfig.PLLSource = SMU_PLLCLK_OSC;

    SMU_PLLInitStruct.PLLConfig.PLLDivR = 0x01;

    SMU_PLLInitStruct.PLLConfig.PLLDivQ = 0x01;

    SMU_PLLInitStruct.PLLConfig.PLLDivN = 0x14;

    SMU_PLLInitStruct.PLLConfig.PLLDivF = 0xA0;

    SMU_PLLInit(&SMU_PLLInitStruct);

     

    /* Ensure that the EFLASH is consistent with the system clock */

    FLASH_UnlockCtrl();

    FLASH_SetCLKFreq(0xA0);

     

    /* Set System Clock parameters values */

    SMU_ClockInitStruct.SYSCLKSelect = SMU_SYSCLK_PLL;

    SMU_ClockInitStruct.AXI4Bus3CLKDiv = AXI4Bus3CLKDiv1;

    SMU_ClockInitStruct.APBBus0CLKDiv = APBBus0CLKDiv1;

    SMU_ClockInitStruct.APBBus1CLKDiv = APBBus1CLKDiv8;

    SMU_ClockInitStruct.CANX2CLKDiv = CANX2CLKDiv1;

   

    SMU_ClockInit(&SMU_ClockInitStruct);

     

    EFLASH_CLK_UPDATE_ENABLE();

    EFLASH_CLK_UPDATE_DISABLE();

 

    FLASH_LockCtrl();

     

    /* Get System Clock values */

    SMU_GetClocksFreq(&SMU_ClocksStruct);

}

GPIO初始化

  1. 开始GPIO对应时钟

  2. 如果是复用IO,首先要配置复用

01

02

03

04

05

06

07

08

09

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

void GPIO_Init(void)

{

   GPIO_InitTypeDef  GPIO_InitStructure;

   /*开启GPIO所在时钟*/

   GPIOD_CLK_ENABLE();

   GPIOG_CLK_ENABLE();

   GPIOF_CLK_ENABLE();

   /* Set GPIO multiplex mapping */

   GPIO_PinAFConfig(GPIOD, GPIO_PinSource13, GPIO_AF_CAN1);//先开启复用模式

   GPIO_PinAFConfig(GPIOD, GPIO_PinSource14, GPIO_AF_CAN1);

  

   /* GPIO Configure */

   GPIO_StructInit(&GPIO_InitStructure);

   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;

   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;

   GPIO_InitStructure.GPIO_OType = GPIO_Out_PP;

   GPIO_InitStructure.GPIO_OStrength = GPIO_OStrength_9mA;

   

   GPIO_Init(GPIOD, &GPIO_InitStructure);

   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14;

   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;

   GPIO_InitStructure.GPIO_IType = GPIO_IPU;

   GPIO_InitStructure.GPIO_OStrength = GPIO_OStrength_9mA;

   GPIO_Init(GPIOD, &GPIO_InitStructure);

    

    /* GPIOB Configure */

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10 | GPIO_Pin_14 | GPIO_Pin_15;

    GPIO_InitStructure.GPIO_Mode      = GPIO_Mode_OUT;

    GPIO_InitStructure.GPIO_OType     = GPIO_Out_PP;

    GPIO_InitStructure.GPIO_OStrength = GPIO_OStrength_9mA;

     

    GPIO_Init(GPIOG, &GPIO_InitStructure);

   

    /* GPIOB Configure */

    GPIO_InitStructure.GPIO_Pin       = GPIO_Pin_1;

    GPIO_InitStructure.GPIO_Mode      = GPIO_Mode_IN;

    GPIO_InitStructure.GPIO_IType     = GPIO_IPU;

    GPIO_InitStructure.GPIO_OType     = GPIO_Out_PP;

    GPIO_InitStructure.GPIO_OStrength = GPIO_OStrength_9mA;

     

    GPIO_Init(GPIOF, &GPIO_InitStructure); 

}

部分外设参数配置

Usart
  1. 最后使能外设
  2. 配置外设参数
  3. 配置GPIO先配置复用
  4. 开启GPIO和外设时钟

01

02

03

04

05

06

07

08

09

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

void User_Print_Init(uint32_t BaudRate)

{

    USART_InitTypeDef USART_InitStructure;

    GPIO_InitTypeDef  GPIO_InitStructure;

    PLIC_InitTypeDef PLIC_InitStructure;

    /*GOPI/外设时钟使能*/

    GPIOD_CLK_ENABLE();

    USART0_CLK_ENABLE();

    /* Set GPIO multiplex mapping */

    GPIO_PinAFConfig(GPIOD, GPIO_PinSource8, GPIO_AF_USART0);       /* USART0_TX */ 开启复用模式

    GPIO_PinAFConfig(GPIOD, GPIO_PinSource9, GPIO_AF_USART0);       /* USART0_RX */

    /* GPIO Configure */

    GPIO_InitStructure.GPIO_Pin       = GPIO_Pin_8;            

    GPIO_InitStructure.GPIO_Mode      = GPIO_Mode_OUT;

    GPIO_InitStructure.GPIO_OType     = GPIO_Out_PP;

    GPIO_InitStructure.GPIO_OStrength = GPIO_OStrength_4_5mA;

    GPIO_Init(GPIOD, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin       = GPIO_Pin_9;            

    GPIO_InitStructure.GPIO_Mode      = GPIO_Mode_IN;

    GPIO_InitStructure.GPIO_IType     = GPIO_IN_FLOATING;

    GPIO_InitStructure.GPIO_OStrength = GPIO_OStrength_4_5mA;

    GPIO_Init(GPIOD, &GPIO_InitStructure);

    /*防止配置冲突*/

    USART_DeInit(USART0);

    USART_StructInit(&USART_InitStructure);

 

    /* Initializes the USART0 */

    USART_InitStructure.USART_BaudRate     = BaudRate;

    USART_InitStructure.USART_WordLength   = USART_WordLength_8b;

    USART_InitStructure.USART_StopBits     = USART_StopBits_1;

    USART_InitStructure.USART_Parity       = USART_Parity_No;

    USART_InitStructure.USART_Mode         = USART_Mode_Rx | USART_Mode_Tx;

    USART_InitStructure.USART_OverSampling = USART_OverSampling_16;

USART_Init(USART0, &USART_InitStructure);

/*配置好相关参数后,使能USART*/

    USART_Cmd(USART0, ENABLE);

    USART_ITConfig(USART0, USART_IT_RXNE, ENABLE);

     

     /* Configer the USART0 interrupt */

    PLIC_InitStructure.PLIC_IRQChannel = USART0_IRQn;

    PLIC_InitStructure.PLIC_IRQPriority = 1;

    PLIC_InitStructure.PLIC_IRQChannelCmd = ENABLE;

    PLIC_Init(&PLIC_InitStructure);

}

CAN

01

02

03

04

05

06

07

08

09

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

void User_CANFD3_Init()

{

    CANFD3_CLK_ENABLE();

    GPIOC_CLK_ENABLE();

    

    GPIO_InitTypeDef  GPIO_InitStructure;

    CANFD_InitTypeDef CANFD_InitStructure;

    PLIC_InitTypeDef  PLIC_InitStructure;

 

    /* Set GPIO multiplex mapping */

    GPIO_PinAFConfig(GPIOC, GPIO_PinSource7, GPIO_AF_CAN3);

    GPIO_PinAFConfig(GPIOC, GPIO_PinSource8, GPIO_AF_CAN3);

 

    /* GPIO Configure */

    GPIO_StructInit(&GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;

    GPIO_InitStructure.GPIO_OType = GPIO_Out_PP;

    GPIO_InitStructure.GPIO_OStrength = GPIO_OStrength_18mA;

     

    GPIO_Init(GPIOC, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;

    GPIO_InitStructure.GPIO_IType = GPIO_IPU;

    GPIO_InitStructure.GPIO_OStrength = GPIO_OStrength_18mA;

    GPIO_Init(GPIOC, &GPIO_InitStructure);

     

    /* Initializes the CANFD1 */

    /* Arbitration Phase (Nominal) Baud Rate 500KHz */

    /* Data Phase Baud Rate 2MHz */

    CANFD_StructInit(&CANFD_InitStructure);

    CANFD_InitStructure.CANFD_SRR = CANFD_SRR_RESET;

    CANFD_InitStructure.CANFD_APBRPR = CANFD_APBRPR_10tp;

    CANFD_InitStructure.CANFD_APBTR_APTS1 = CANFD_APBTR_TS1_11tp;

    CANFD_InitStructure.CANFD_APBTR_APTS2 = CANFD_APBTR_TS2_4tp;

    CANFD_InitStructure.CANFD_APBTR_APSJW = CANFD_APBTR_SJW_2tp;

     

    CANFD_InitStructure.CANFD_DPBRPR = CANFD_DPBRPR_2tp;

    CANFD_InitStructure.CANFD_DPBTR_DPTS1 = CANFD_DPBTR_TS1_7tp;

    CANFD_InitStructure.CANFD_DPBTR_DPTS2 = CANFD_DPBTR_TS2_2tp;

    CANFD_InitStructure.CANFD_DPBTR_DPSJW = CANFD_DPBTR_SJW_2tp;

    CANFD_Init(CANFD3, &CANFD_InitStructure);

         

    /* CANFD receive filter configure */

    CANFD_FilterInit(CANFD3, TB0, 0xFFE00000, 0X62E00000);       

 

    CANFD_AutoRetransConfig(CANFD3,ENABLE);

    /* Enable new message received interrupt */

    CANFD_ITConfig(CANFD3, CANFD_IT_ERXOK, ENABLE);

    /* CANFD Enable */

    CANFD_Enable(CANFD3);

     

    PLIC_StructInit(&PLIC_InitStructure);

 

    /* Configer the CANFD1 interrupt */

    PLIC_InitStructure.PLIC_IRQChannel = CANFD3_IRQn;

    PLIC_InitStructure.PLIC_IRQPriority = 2;

    PLIC_InitStructure.PLIC_IRQChannelCmd = ENABLE;

    PLIC_Init(&PLIC_InitStructure);

    CANFD_ClearITPendingBit(CANFD3, CANFD_CLEAR_ALL);

}

通过遵循"参数配置在先,外设使能在后"的原则,并采用结构化初始化流程,可以显著提高MCU系统的稳定性和可靠性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/918609.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/918609.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AI一周事件(2025年8月6日-8月12日)

(以下借助 DeepSeek-R1 & ChatGPT-5 辅助整理) 一、AI 模型与算法进展 1. OpenAI 正式发布 GPT-5(8月7日) 事件:OpenAI 于 2025 年 8 月 7 日推出 GPT-5——其自称拥有“PhD 级别”的智能,通过内置…

快速了解自然语言处理

在这个智能时代,我们每天都在和机器 “对话”—— 用语音助手查询天气、让翻译软件跨越语言障碍、靠智能客服解决问题…… 这些便捷体验的背后,都离不开自然语言处理(Natural Language Processing,NLP) 技术。作为人工…

洛谷 P2607 [ZJOI2008] 骑士-提高+/省选-

题目描述 Z 国的骑士团是一个很有势力的组织,帮会中汇聚了来自各地的精英。他们劫富济贫,惩恶扬善,受到社会各界的赞扬。 最近发生了一件可怕的事情,邪恶的 Y 国发动了一场针对 Z 国的侵略战争。战火绵延五百里,在和平…

不止于GET:掌握POST报错注入的精髓

文章目录引言POST请求简述报错注入核心思想关键前提实战演练POST报错注入与GET报错注入的区别防御之道:如何避免POST报错注入?引言 SQL注入是Web安全领域危害性最大、最常见、最持久的高危漏洞之一。它直接威胁到应用程序核心数据库的安全,可…

01数据结构-Prim算法

01数据结构-Prim算法1.普利姆(Prim)算法1.1Prim算法定义1.2Prim算法逻辑1.3Prim代码分析2.Prim算法代码实现1.普利姆(Prim)算法 1.1Prim算法定义 Prim算法在找最小生成树的时候,将顶点分为两类,一类是在查找的过程中已经包含在生成树中的顶点(假设为A类…

CacheBlend:结合缓存知识融合的快速RAG大语言模型推理服务

温馨提示: 本篇文章已同步至"AI专题精讲" CacheBlend:结合缓存知识融合的快速RAG大语言模型推理服务 摘要 大语言模型(LLMs)通常在输入中包含多个文本片段,以提供必要的上下文。为了加速对较长LLM输入的预…

Docker 在 Linux 中的额外资源占用分析

Docker 本身作为一个运行时环境,除了容器应用本身消耗的资源外,还会引入一些额外的开销。主要体现在以下几个方面: 1. 存储空间占用 (Disk Space) 这是最显著的额外开销,主要来源于 Docker 的存储驱动(如 overlay2&…

[激光原理与应用-264]:理论 - 几何光学 - 什么是焦距,长焦与短焦的比较

长焦与短焦透镜是光学系统中两类核心组件&#xff0c;其成像特性在焦距、视角、景深、像场特性及典型应用中存在显著差异。以下从多个维度进行详细对比&#xff1a;一、核心参数对比参数长焦透镜短焦透镜焦距范围通常 >50mm&#xff08;全画幅相机标准&#xff09;通常 <…

el-input 复制大量数据导致页面卡顿问题解决

问题根源 复制粘贴操作会瞬间触发大量 input 事件&#xff0c;导致 Vue 频繁更新响应式数据&#xff0c;引发性能瓶颈。 解决方案&#xff1a;使用 .lazy 修饰符 <el-input v-model.lazy"inputValue" />

PCIe Electrical Idle Sequences ( EIOS and EIEOS )

前言 PCI Express (PCIe)协议中&#xff0c;EIOS (Electrical Idle Ordered Set) 和 EIEOS (Electrical Idle Exit Ordered Set) 是在高速链路管理和状态切换过程中极为重要的特殊序列。下面做详细解释&#xff1a; 一、EIOS&#xff08;Electrical Idle Ordered Set&#xff0…

【GPT入门】第45课 无梯子,linux/win下载huggingface模型方法

【GPT入门】第45课 无梯子&#xff0c;下载huggingface模型方法1.下载模型代码2. linux 设置镜像与加速3.windows1.下载模型代码 from transformers import AutoModelForCausalLM, BertTokenizer, BertForSequenceClassificationmodel_dir /root/autodl-tmp/model_hf# 加载模…

计算机网络摘星题库800题笔记 第5章 传输层

第5章 传输层5.1 传输层概述题组闯关1.Internet 传输层滑动窗口协议规定 ( )。 A. 网络接收分组的最低效率&#xff0c;只需要重传未被确认的分组 B. 固定的窗口大小&#xff0c;只需要重传未被确认的分组 C. 网络接收分组的最低效率&#xff0c;固定的窗口大小 D. 未被确认的分…

Apache虚拟主机三种配置实战

一、虚拟主机概述 目的&#xff1a;实现单台服务器部署多个独立站点 三种部署方式&#xff1a; 相同IP 不同端口不同IP 相同端口相同IP和端口 不同域名&#xff08;FQDN&#xff09; 示例目标&#xff1a;在服务器上部署 baidu 和 taobao 两个站点方式1&#xff1a;相同IP …

【SpringBoot】04 基础入门 - 自动配置原理入门:依赖管理 + 自动配置

文章目录前言一、Spring Boot Maven项目POM文件解析1. 基础项目信息2. 父项目继承3. 依赖管理4. 构建配置5. 属性配置Spring Boot特性体现典型Spring Boot项目特点二、依赖管理1、父项目做依赖管理无需关注版本号&#xff0c;自动版本仲裁修改自动仲裁的版本官网文档2、依赖项引…

机器学习—— TF-IDF文本特征提取评估权重 + Jieba 库进行分词(以《红楼梦》为例)

使用 Jieba 库进行 TF-IDF 关键词提取&#xff08;以《红楼梦》为例&#xff09;在中文文本分析中&#xff0c;TF-IDF&#xff08;Term Frequency - Inverse Document Frequency&#xff09; 是最常用的关键词提取方法之一。它通过评估词在单个文档中的出现频率和在所有文档中的…

Kotlin语法整理

Kotlin语法整理 Kotlin语法整理 一、基本数据类型 共8种 二、变量的声明三、条件 1. if…else if…else语句2. when 语句 四、循环 1. while 语句2. do…while 语句3. for 语句4. repeat 语句5. break 语句6. continue 语句 五、数组 1. 创建元素未初始化的数组2. 创建元素初始…

跨平台低延迟的RTMP推流播放在无纸化会议与智慧教室的技术设计和架构实践

✳️ 引言&#xff1a;让每一块屏幕“同频”的核心技术 无纸化会议与智慧教室&#xff0c;正在从“辅助工具”走向“核心基础设施”&#xff0c;成为政企数字化与教育信息化建设的标配。它们的核心诉求并不只是替代纸质文档或黑板&#xff0c;而是要在多终端、多地点、多网络环…

最优扩展大型语言模型测试时计算量可能比扩展模型参数更有效

摘要 通过增加测试时计算量使大型语言模型&#xff08;LLMs&#xff09;提升输出效果&#xff0c;是构建能基于开放自然语言自主改进的通用智能体的重要步骤。本文研究LLMs推理阶段计算量的扩展规律&#xff0c;重点回答以下问题&#xff1a;若允许LLM使用固定但可观的推理阶段…

GPT5评测对比与使用

经过长达一年的技术迭代&#xff0c;OpenAI正式推出GPT-5系列模型&#xff0c;包含GPT-5&#xff08;标准版&#xff09;、GPT-5-mini&#xff08;轻量版&#xff09;和GPT-5-nano&#xff08;极简版&#xff09;三个版本&#xff0c;定价策略保持统一。本次升级在性能、效率与…

Git与CI/CD相关知识点总结

Git与CI/CD相关知识点总结 1. Git对象模型与存储机制 1.1 Git对象类型 Commit对象&#xff1a;包含提交信息、作者、时间、父commit引用、树对象引用Tree对象&#xff1a;描述目录结构和文件引用Blob对象&#xff1a;实际的文件内容 1.2 存储机制特点 增量存储&#xff1a;每次…