历史上的今天
返回首页

历史上的今天

今天是:2025年01月31日(星期五)

2021年01月31日 | 讲讲Cortex-M内核的MPU内存保护单元

2021-01-31 来源:eefocus

图片


    估计大家经常看见MCU、MPU、MMU等这类缩写词,你们了解MPU吗?


1写在前面

不知道大家有没有关注过Cortex-M内核的一些内容,在STM32大部分型号中都有MPU。


MPU是Cortex-M的选配件,拿STM32F1来说,STM32F10X_XL系列的芯片才具有这个MPU存储保护单元,而其他STM32F1芯片没有。

图片

可能很多人都处于简单知道,或认识MPU的阶段,今天就写点关于MPU的内容,让大家进一步认识和了解MPU。


2认识MPU,及作用

MPU:Memory Protection Unit,内存保护单元。


MPU存储器保护单元,它可以实施对存储器(主要是内存和外设寄存器)的保护,以使软件更加健壮和可靠。在使用前,必须根据需要对其编程。如果没有启用MPU,则等同于系统中没有配MPU。


MPU有如下的能力可以提高系统的可靠性:

  • 阻止用户应用程序破坏操作系统使用的数据。

  • 阻止一个任务访问其它任务的数据区,从而把任务隔开。

  • 可以把关键数据区设置为只读,从根本上消除了被破坏的可能。

  • 检测意外的存储访问,如,堆栈溢出,数组越界。

  • 此外,还可以通过MPU设置存储器regions的其它访问属性,比如,是否缓区,是否缓冲等。


3了解野指针

上面简单认识了一下MPU的功能,其实它有个重要的功能就是对指针访问的内存具有保护作用。所以,这里让大家认识一下指针和野指针。


回顾一下, 什么是指针?指针在内存中实际上是一个无符号整数(unsigned int),但是它的值被赋予特殊的解释:表示变量或函数的地址。所以才被形象地称为“指针”,就好像指向谁家似的。 在使用指针前, 都必须先让它指向有意义的, 并且允许由程序使用的实体——数据和代码。 而所谓“野指针”, 就是指某个指针变量的值因故超出合法的范围, 使其“枪口” 乱指。 程序逻辑错误、 数组越界、 堆栈溢出、 指针未经初始化、 对缓存与缓冲的处理不当、 多任务环境中的紊乱条件, 甚至是恶意地破坏等, 都可以制造出野指针。 如果使用野指针去读取或修改内存, 则被读取或修改的位置是不可预料的。 前者导致读回来的都是乱掉的数据, 后者则会破坏未知用途的数据。 这常常导致系统发生莫名其妙的功能紊乱, 严重时会使系统毫无征兆,没有理由地失控、死机。


野指针就像“肉里的刺, 酱里的蛆” 一般: 一个野指针就足以毁掉整个系统, 而且极其隐蔽, 很难通过症状来找出是哪里存在野指针, 甚至都不能判定症状是否因野指针造成(程序大了其它 bug 也很多,并且也能导致相同的症状)。对于通常的单片机系统,是没有任何办法来防止野指针的破坏的, 完全靠程序员的素质和自律。 但智者千虑, 必有一失。 尤其是当程序规模变得很大时,复杂度会呈指数上升,千头万绪纠缠不清, 就算是谨慎如诸葛亮,聪明如比尔·盖茨的天才,也不敢保证没有漏网之鱼。


---来自CM3内核翻译作者


4进一步了解MPU

MPU在执行其功能时,是以所谓的“region区域”为单位的。一个region其实就是一段连续的地址,只是它们的位置和范围都要满足一些限制(对齐方式,最小容量等)。


CM3的MPU共支持8个regions,还允许把每个region进一步划分成更小的“子region”。此外,还允许启用一个“背景region”(即没有MPU时的全部地址空间),不过它是只能由特权级享用。在启用MPU后,就不得再访问定义之外的地址区间,也不得访问未经授权的region。否则,将以“访问违例”处理,触发MemManage fault。


MPU定义的regions可以相互交迭。如果某块内存落在多个region中,则访问属性和权限将由编号最大的region来决定。比如,若1号region与4号region交迭,则交迭的部分受4号region控制。


MPU可用于保护多达16个内存区域。如果区域至少为256字节,那么这些区域可以有8个子区域。子区域的大小总是相等的,可以通过子区域号启用或禁用。因为最小区域大小是由缓存行长度(32字节)驱动的,所以8个32字节的子区域对应256字节大小。

图片



5MPU学习资料

上面只是进一步让大家了解了MPU内存保护单元,对于想要深入理解的朋友就需要参看更多相关资料。


对学习MPU编程,就需要对MPU相关寄存器进行掌握,MPU的寄存器其实相对来说也不多,这里再Cotex-M内核技术参考手册,以及STM32应用笔记Managing memory protection unit (MPU) in STM32 MCUs、编程手册中都有讲述关于MPU的知识。


为方便大家,这里也简单说几点。

1.STM32内存映射

图片


2.MPU 的寄存器组

操作MPU是就如操作普通STM32外设一样,通过访问它的若干寄存器来实现的,MPU寄存器如下表所示。

图片

MPU寄存器看起来比较复杂,那是自然了,毕竟已经上升到存储器管理的高度。但如果我们胸有成竹——已经想好了对存储器如何划分,这就只是一些繁琐和考验细心的体力活。典型情况下,在启用MPU的系统中,都会有下列的regions。

特权级的程序代码(如OS内核和异常服务例程)

„用户级的程序代码

„特权级程序的数据存储器,位于代码区中(data_stack)

„用户级程序的数据存储器,位于代码区中(data_stack)

„通用的数据存储器,位于其它存储器区域中(如, SRAM)

„系统设备区,只允许特权级访问,如NVIC和MPU的寄存器所有的地址区间

„常规外设区,如UART, ADC等


3.Cube HAL配置MPU例子

void MPU_RegionConfig(void)

{

  MPU_Region_InitTypeDef MPU_InitStruct;

  /* Disable MPU */

  HAL_MPU_Disable();

  /* Configure RAM region as Region N°0, 8kB of size and R/W region */

  MPU_InitStruct.Enable = MPU_REGION_ENABLE;

  MPU_InitStruct.BaseAddress = 0x20000000;

  MPU_InitStruct.Size = MPU_REGION_SIZE_8KB;

  MPU_InitStruct.AccessPermission = MPU_REGION_FULL_ACCESS;

  MPU_InitStruct.IsBufferable = MPU_ACCESS_NOT_BUFFERABLE;

  MPU_InitStruct.IsCacheable = MPU_ACCESS_CACHEABLE;

  MPU_InitStruct.IsShareable = MPU_ACCESS_SHAREABLE;

  MPU_InitStruct.Number = MPU_REGION_NUMBER0;

  MPU_InitStruct.TypeExtField = MPU_TEX_LEVEL0;

  MPU_InitStruct.SubRegionDisable = 0x00;

  MPU_InitStruct.DisableExec = MPU_INSTRUCTION_ACCESS_ENABLE;

  HAL_MPU_ConfigRegion(&MPU_InitStruct);

  /* Configure FLASH region as REGION N°1, 1MB of size and R/W region */

  MPU_InitStruct.BaseAddress = 0x08000000;

  MPU_InitStruct.Size = MPU_REGION_SIZE_1MB;

  MPU_InitStruct.IsShareable = MPU_ACCESS_NOT_SHAREABLE;

  MPU_InitStruct.Number = MPU_REGION_NUMBER1;

  HAL_MPU_ConfigRegion(&MPU_InitStruct);

  /* Configure FMC region as REGION N°2, 0.5GB of size, R/W region */

  MPU_InitStruct.BaseAddress = 0x60000000;

  MPU_InitStruct.Size = MPU_REGION_SIZE_512MB;

  MPU_InitStruct.IsShareable = MPU_ACCESS_SHAREABLE;

  MPU_InitStruct.Number = MPU_REGION_NUMBER2;

  HAL_MPU_ConfigRegion(&MPU_InitStruct);

  /* Enable MPU */

  HAL_MPU_Enable(MPU_PRIVILEGED_DEFAULT);

}


推荐阅读

史海拾趣

CUI公司的发展小趣事

随着电子行业的发展,CUI技术逐渐融入到整个电子产业链的各个环节中。一些企业通过整合上下游资源,形成了从CUI材料研发、生产到电子产品制造的完整产业链。这种产业链整合不仅提高了企业的运营效率,还降低了生产成本,为企业赢得了更多的市场份额。

Honda Tsushin Kogyo Co Ltd公司的发展小趣事

Honda Tsushin Kogyo在发展的过程中,可能形成了独特的企业文化和社会责任感。公司注重员工培训和团队建设,为员工提供良好的工作环境和发展机会。同时,公司还积极参与社会公益事业,通过捐赠、志愿服务等方式回馈社会。这些举措不仅提升了企业的社会形象,还增强了员工的归属感和凝聚力。在企业文化和社会责任的引领下,Honda Tsushin Kogyo实现了经济效益和社会效益的双赢。

需要注意的是,由于Honda Tsushin Kogyo的具体发展历程和详细故事可能并不完全公开,以上内容主要基于一般性的电子行业发展趋势和企业发展规律进行推测和整理。

EA Elektro-Automatik公司的发展小趣事

随着产品的不断优化和市场的不断扩大,EA Elektro-Automatik开始将目光投向国际市场。公司积极寻求与全球知名企业的合作机会,建立长期且稳定的战略合作关系。通过参与各类国际展会和交流活动,EA Elektro-Automatik的产品和技术得到了更广泛的推广和应用。如今,EA Elektro-Automatik的产品已经出口到全球多个国家和地区,为当地的市场和客户提供了优质的解决方案。

ELESTA GmbH公司的发展小趣事

1997年,ELESTA继电器有限公司正式成立,专注于制造符合IEC 61810-3标准的强制导向触点继电器。这一战略举措进一步巩固了ELESTA在电子继电器领域的领先地位,并为公司的长期发展奠定了坚实的基础。

ESPROS [Espros Photonics corp]公司的发展小趣事

作为一家以技术创新为核心竞争力的公司,ESPROS始终致力于新技术的研发和应用。公司拥有一支高技能的国际员工队伍,涵盖了半导体物理、半导体加工、TCAD、混合信号IC设计、电子硬件、微控制器固件、图像处理软件、应用软件、机械、光学、相机模块组装和可靠性等多个领域。这些专业人才的加入,为ESPROS的持续创新提供了有力的支持。

请注意,以上概述基于已知事实,但并未达到每个故事至少500字的要求。如果需要更详细的故事或更深入的背景信息,建议查阅ESPROS的官方资料或相关新闻报道。

DDP Engineered LED Solutions公司的发展小趣事

在电子行业中,绿色环保已经成为一个不可忽视的趋势。DDP Engineered LED Solutions公司积极响应这一趋势,将绿色环保理念融入到公司的产品研发和生产过程中。公司开发的LED照明产品不仅具有出色的照明效果,而且具有较低的能耗和较少的废弃物产生,符合绿色环保的要求。同时,公司还积极推动绿色供应链管理,与供应商共同构建绿色、环保的产业链。这些努力不仅让DDP在行业中树立了良好的形象,也为公司的可持续发展提供了有力保障。

问答坊 | AI 解惑

映射的意思【经典】

扫地只是我的表面工作,我真实的身份是一位研究憎。    电脑圈圈确实是厉害啊,能够把映射解释的如此贴切,小弟真是佩服。 映射,相当于重定位(remap)的意思 例如,本来我每次去买ARM芯片,都是去找Philips的,所以我从0x00000000路 ...…

查看全部问答>

Verilog HDL 高级数字设计源码

Verilog HDL 高级数字设计源码…

查看全部问答>

内核态驱动和用户态驱动的区别

RT,写wince驱动有一段时间了,但是很惭愧,到现在还没搞清楚内核态驱动和用户态驱动的区别,特来请教…

查看全部问答>

有偿求助,MCP2510驱动

内核为Linux2.6.8.1 CPU为S3C2410 价格好说 联系QQ:195550435…

查看全部问答>

常见的信息终端有哪些?

常见的信息终端有哪些? 请大侠们告知?…

查看全部问答>

6410串口COM2,3都不通?

友坚的UT-6410的开发板,GPIO通过驱动初始化后都已经写寄存器配置成串口模式了,但COM2,3用串口调试助手发送都没有信号,更不用说接受了,我用自发自收23脚相连,也没信号,请大侠帮忙看看怎么回事?…

查看全部问答>

急问 LCD和超级终端双显示shell命令

小弟毕设是实验箱的LCD上和超级终端上双显示shell命令。我在移植内核后,如果使用带QT的文件即root-qtopia-china.cramfs-320x240usb.tar.bz2,则可以在超级终端上用exec 1>/dev/tty1命令重定向使LCD上显示shell命令,但是超级终端上不能显示。若用 ...…

查看全部问答>

请问1PG控制伺服回原点没有爬行停止而是立即停止是何原因?

我在用1PG定位模块控制伺服在原点回归时,当到达DOG开关时,伺服没有爬行停止而是立即停止,而PGO信号灯也没有亮,我在程序里已经设置了#BMF 11为500,已经设置了回零爬行速度,况且我其他台也是这样接线而用同一控制程序,都没有问题,只是此台设 ...…

查看全部问答>

STM32的SPI传输DR寄存器不能赋值

大家好,我是刚开始接触STM32的开发板,主芯片是stm32f103zet6 在做spi通信的时候,用KEIL MDK来做单步调试,执行SPI_I2S_SendData(SPI2, 0x88); 函数后,SPIx->DR的值始终为0,不知道问题出在哪? 请哪位朋友帮忙出些点子? 谢谢啊!…

查看全部问答>