0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

在MCU上怎样完成动态库的制作和加载

strongerHuang 来源:工程师的废纸篓 作者:Tony Yang 2022-03-09 17:02 次阅读

作为一个嵌入式软件攻城狮,提起库首先会想到静态库和动态库。静态库一般以.a为后缀,动态库以.so为后缀(Win系统.DLL)。

poYBAGIobn6AO9DjAAA6pPX499U371.png

而作为一个单片机软件攻城狮,也会经常用到各种静态库,常见的C库有stdio,stdlib,string,time等,第三方库也有CMSIS_DSP_Library,mbedtls,60730等等。为什么要使用这些库呢?个人认为可能有以下几个原因:

系统解耦,可以减少软件模块之间的耦合,使软件结构更清晰。

加速编译,有些库源码有非常多的源文件,如果使用源码编译会大大增加编译时间。

保护知识产权,限定产品的使用范围。比如NXP提供的电机算法库rtcesl当然希望它运行在自己品牌MCU

MCU从产品实用角度上看,也是有动态库的需求,比如下面几种场景:

有些产品形态要求MCU的固件分为安全域和非安全域,安全域的逻辑一旦经过认证就无法进行升级,而非安全域的逻辑可以通过bootloader进行远程或本地的升级。比如新规电表,计量部分属于安全域不可更新,而其他固件则可以进行更新

90d538d8-9b7c-11ec-952b-dac502259ad0.jpg

有些软件/算法公司并不提供实际的硬件产品,和终端客户的合作模式如果按件收费,除非每个产品必须联网认证,就像我们所用的Windows系统激活,否则很难把控。比较好的做法是,将算法编译成二进制烧入芯片中,向终端客户提供芯片。

90ebbe50-9b7c-11ec-952b-dac502259ad0.jpg

有些产品经常需要更新固件,受限于通讯速率,Flash大小的问题,希望可更新的固件越小越好。比如内置蓝牙的MCU,其协议栈往往在100KB左右,如果通过源码或者静态库的方式编译,则升级的时候就必须将协议栈也一并升级,即使协议栈本身没有任何修改,由于升级的固件比较大,所需要的更新时间也会非常长。如果通过动态库的方式提供协议栈,就没有这个问题。同时,如果客户要求产品的固件能回退,则Flash必须能放两份固件,如果应用代码超过20K,则最终固件很容易超过128K(APP+BLE),如果使用静态库,则选型时必须使用超过256K的产品(APP+BLE)x2,而动态库则可以使用256K以内的产品(APPx2+BLE)。用过Nordic蓝牙MCU的应该都懂。

90fd66f0-9b7c-11ec-952b-dac502259ad0.jpg

今天我们就专门研究下如何在MCU上实现动态库的制作和加载。

制作动态库

首先我们先回忆下如何制作静态库,第三方的或者自己写的静态库,一般由三部分组成:

poYBAGIobnGATYg5AAAtswS9XVo508.png

在MCU中制作的动态库则需要如下几部分内容:

pYYBAGIobmmAdntvAAAoV8Oe-rw421.png

至于具体如何实现,现在以之前介绍的开源PLC为例:

规划内存,将MCU内部的Flash/RAM分出一部分留给库,具体做法需要修改链接文件中的相关地址

pYYBAGIobl-AD6b-AAA22syThwQ328.png

定义函数指针列表结构体,用于提供给用户程序调用:

poYBAGIobk2ACD9JAABMOVMhLJM686.png

初始化结构体,并实现函数原型:

pYYBAGIobkSAap75AAAy-2UA9f0984.png

poYBAGIobjiAB4rYAAAOoFGwU0M126.png

poYBAGIobiyAQNJ8AAAhcKh8-wk838.png

修改链接文件,将函数指针列表放到固定地址

poYBAGIobiGAS4_bAAAK9LZxTz8464.png

编译并下载到MCU中

加载动态库

生成动态库后,用户只需要知道函数指针列表的地址和结构体列表即可。使用过程如下:

在用户工程中定义相同的结构体,并从固定地址中加载到

poYBAGIobhiAeBJlAAARIhl1CsU766.png

通过函数指针执行相应代码即可

pYYBAGIobguAPUZ6AAAM3MhCJW4415.png

扩展知识

针对动态库的应用场景2,简单的卖芯片也只是防君子不防小人,因为终端客户可以通过JTAG/SWD把动态库读出来,有些芯片虽然支持禁用JTAG/SWD使能ISP的方式下载,但用户还是可以通过指针+地址的方式将库从芯片中读取出来,之后找芯片原厂克隆,针对这种攻击,普通的MCU往往是无力应付的,大部分的软件/算法公司是通过绑定加密芯片的方式来进行保护,但是除非加密芯片中有算法逻辑,否则这种保护也是徒劳的,因为不管是动态库,还是静态库,攻击者都可以通过反汇编看到关键的跳转点,从而bypass验证过程。

难道真的就没有更好的防守么?目前知道有两种方案:

NXP Kinetis系列产品在Flash中添加了FAC(Flash Access Control)功能,可以保证在可配置区域内只能被执行,而无法通过指针方式,DMA,JTAG/SWD,Ezport等接口获取

910c7f6e-9b7c-11ec-952b-dac502259ad0.png

ARM Cortex-M33内核以及支持了TrustZone,可以将动态库放置在Secure区域,并设置NSC访问接口,具体做法和FAC类似

9121e8cc-9b7c-11ec-952b-dac502259ad0.png


审核编辑:郭婷

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • mcu
    mcu
    +关注

    关注

    146

    文章

    17186

    浏览量

    351749
  • 嵌入式
    +关注

    关注

    5087

    文章

    19149

    浏览量

    306201
收藏 人收藏

    评论

    相关推荐

    Qt创建动态给C#调用,通过回调完成交互

    windows下做应用开发时,经常需要多种不同的语言混合编程。比如:利用Qt开发一个动态,给C#调用。 当前的需求是: 利用Qt开发一个工具,给C#调用,来
    的头像 发表于 09-09 11:37 4954次阅读
    Qt创建<b class='flag-5'>动态</b><b class='flag-5'>库</b>给C#调用,通过回调<b class='flag-5'>完成</b>交互

    单片机上实现动态加载功能

    本项目是一个单片机(如:STM32)实现动态加载功能的函数,与Windows中的dll,Linux中的so类似,可以将代码
    发表于 05-30 11:04 1890次阅读

    Linux下动态和静态制作及使用

    不必从零开始,我们要做的只是恰当的位置调用合适的库函数去实现相应的功能,充分利用前人的劳动成果,就是“站在巨人的肩膀”。本文主要简述Linux下制作以及使用方法。一、什么是
    发表于 11-18 17:05

    linux Qt 动态制作以及使用方法

    一、Qt动态制作 (1)、新建 选择目录选择需要的模块,如果需要界面则选择QtGui等。(2)、编写简单代码以及编译编写简单的打印代码,然后点击运行,之后弹出如下窗口:点击“取消”,之后可在工程
    发表于 07-05 08:01

    labview由于缺少依赖项或动态依赖,动态加载失败问题

    `设备是有光谱和电机,激光共同工作的一个程序,别人电脑可以无报错打开,自己电脑说丢失外部符号或依赖关系,或出现无效的文件格式,因此无法加载动态
    发表于 05-21 15:21

    Linux下静态动态制作与使用

    什么是静态函数动态函数库又是什么?linux静态函数怎样创建并使用的?动态函数库怎样
    发表于 04-26 06:45

    动态模块加载的调试经验分享

    cache 没能同步导致的。问题二目前动态模块加载功能还有如下待完善的地方:初始化全局指针的时候,不能将其设置为某个全局变量的地址,因为全局变量的地址是
    发表于 03-22 14:53

    有没有办法让我看到保存的项目PC和加载MCU中的项目之间的程序差异?

    有没有办法让我看到保存的项目(PC)和加载MCU 中的项目之间的程序差异?我正在使用 System Workbench IDE 对 stm32f405rgt6 进行编程。MCU集成
    发表于 12-23 07:10

    制作和使用自定义C文件

    制作和使用自定义C文件 目标 1.制作一个文件libGetMax.a ,其中包含一个外部函数GetMax 。 函数GetMax的作用是判断
    发表于 01-16 11:58 1305次阅读

    Linux下静态动态(共享)的制作与使用

    Linux下静态动态(共享)的制作与使用Linux
    发表于 07-09 14:39 1186次阅读

    Linux下的静态动态动态加载

    动态必须在编译时对编译器可见,但编译器却不将此种编译进可执行文件; b) 在运行期间,动态加载和卸载的
    发表于 04-02 14:32 827次阅读

    动态和静态制作步骤

    是一种可执行的二进制文件,是编译好的代码。使用可以提高开发效率。 Linux 下有静态动态
    的头像 发表于 07-27 11:00 826次阅读

    车规MCU的启动加载程序是什么

    加载程序负责MCU启动和加载应用程序。它通过读取存储器中的引导加载代码,并将其
    的头像 发表于 10-27 17:26 1615次阅读

    单片机上实现动态加载功能的函数介绍

    本项目是一个单片机(如:STM32)实现动态加载功能的函数,与Windows中的dll,Linux中的so类似,可以将代码
    的头像 发表于 11-09 10:55 1587次阅读

    MATLAB中如何保存和加载消息

    保存和加载消息 您可以保存消息并存储内容以供以后使用。 例如从订阅者获取一条新消息。 posedata = receive(posesub, 10 ) 然后使用MATLAB的保存函数将姿态数据保存
    的头像 发表于 11-15 15:17 418次阅读