当前位置: 首页 > news >正文

61、【OS】【Nuttx】【构建】向量表

【声明】本博客所有内容均为个人业余时间创作,所述技术案例均来自公开开源项目(如Github,Apache基金会),不涉及任何企业机密或未公开技术,如有侵权请联系删除

背景

接之前 blog
【OS】【Nuttx】【构建】启动文件
简单分析了下启动文件(具体说应该是描述向量表的文件),下面继续看这个向量表

向量表

向量表是一块特殊的内存区域,用于存储指向中断服务程序 ISR 的地址(interrupt service routine)

  • 当中断发生时,处理器暂停当前的任务,查找向量表并跳转到对应的 ISR 执行
  • 处理完中断后,处理器返回到之前被中断的地方继续执行
  • 向量表位于内存的固定位置,该位置取决于具体的芯片架构,构建时可通过链接脚本将向量表映射到这块区域
  • 向量表不仅包含 ISR 地址,还可以包含堆栈指针,复位处理函数地址(程序复位后第一个执行的函数),不可屏蔽中断(NMI)等特殊入口点地址

下面来看下这个核心的向量表,首先是注释
在这里插入图片描述
有几个关键点:

  • 向量表是一个函数指针数组
  • 第一个元素不是中断处理函数,而是初始栈指针
  • 所有的异常和中断都通过通用处理函数 exception_common 函数进行处理(这里有点问题,应该是代码更新了,注释没更新,应该是 PENSV 之前的系统服务通过 exception_common,其他外设中断处理函数通过 exception_direct)
  • 使用了 gcc 的扩展语法 [2 … NVIC_IRQ_PENDSV] = &exception_common 来初始化连续的多个数组元素

下面来看这个函数指针数组
在这里插入图片描述
这里面有比较多的关键点,首先来看第一个,修饰前缀 const void* const,这涉及到一个非常经典的 C 指针与常量结合的问题

  • const void* 和 void* const 有什么区别?

const void*:指向常量内容的指针

这个修饰前缀有两个含义:

  • 这个指针可以指向不同的地址(指针本身可变)
  • 这个指针所指向的内容不能变,不能通过这个指针去修改内容

举个例子

const void* ptr = &x;
ptr = &y;         // 正确,可以改变指针指向的地址
*(int*)ptr = 10;  // 错误,不能通过 ptr 修改内容

void* const:常量指针

同样有两个含义:

  • 这个指针本身是常量,不能改变指向的地址
  • 这个指针指向的内容可以修改

举个例子

int x = 0;
void* const ptr = &x;ptr = &y;         // 错误,不能改变 ptr 的值,指针本身是 const
*(int*)ptr = 10;  // 正确,可以修改 ptr 所指向的内容

最后总结,const void* const 是两者的结合,修饰的 _vectors[] 数组,数组中的每个元素都是一个 指向常量数据的常量指针,即:

  • 数组成员本身(函数指针)不能被修改
  • 函数指针指向的内容也不允许通过这些指针修改

下面来看下一个点:void* vectos[]

void* vectors[]

C 语言支持基于初始化列表自动推断数组大小的功能,这种不带大小的数组声明方式,通常见于如下两种场景:

  • 作为函数参数传递数组,比如下面这个例子
void func(void* vectors[]) {// 函数体,这里可以通过 sizeof(vectors) 求得数组大小
}
  • 作为全局或静态数组定义的一部分,并结合初始化器使用,比如在现在这个例子中,向量表通过初始化列表来确定其大小
    在这里插入图片描述
    在这里,数组的实际大小由初始化列表中的元素数量决定。编译器会根据初始化列表自动推断数组的大小

除了上述两种场景,如果不在声明的同时进行初始化,只是简单地声明一个不带大小的数组,比如

void* vectors[];

那就需要在其他地方明确指定其大小或通过链接脚本等手段为其分配空间,不然会被优化掉
在这里插入图片描述
在这里插入图片描述
先分析到这,下篇继续

http://www.lryc.cn/news/578543.html

相关文章:

  • 每日一练:找到初始输入字符串 I
  • 新版本 Spring Data Jpa + QueryDSL 使用教程
  • Zephyr RTOS 信号量 (Semaphore)
  • GitHub已破4.5w star,从“零样本”到“少样本”TTS,5秒克隆声音,冲击传统录音棚!
  • MySQL 8.4 备份与恢复完全指南
  • JVM调优实战 Day 14 :大数据处理中的JVM调优
  • 文心一言开源版测评:能力、易用性与价值的全面解析
  • 磁盘的访问算法有哪些?
  • HTTPS安全传输时采用的顶级阳谋
  • [密码学实战]国密TLCP协议报文解析代码实现(三十)
  • [C#] WPF - 自定义样式(Slider篇)
  • 腾讯 iOA 零信任产品:安全远程访问的革新者
  • 数据结构day4——栈
  • 回转体水下航行器简单运动控制的奥秘:PID 控制和水动力方程的运用
  • 信息安全相关算法
  • 蓝牙音频传输协议深度解析:A2DP、HFP、AVRCP 对比与面试核心考点
  • 【机器学习2】正则化regularizaiton(降低模型过拟合)
  • 【cv视觉】标注工具的使用和数据集的创建
  • 2.SQL语句执行慢,如何分析
  • 07CSRF 漏洞保护
  • 事件监听器 + 回调处理器的事件循环系统
  • OpenCV CUDA模块设备层-----二值化阈值操作函数thresh_binary_func()
  • 设计模式精讲 Day 21:策略模式(Strategy Pattern)
  • 【STM32】 STM32低功耗模式详解:睡眠模式与唤醒机制【待测试】
  • 单元测试详解
  • 记录一个 Linux中脚本无法执行的问题
  • 构建淘宝评论监控系统:API 接口开发与实时数据采集教程
  • Camera相机人脸识别系列专题分析之十五:人脸特征检测FFD算法之libcvface_api.so算法API详细注释解析
  • Docker制作python环境
  • C++ 11 中 condition_variable 的探索与实践