创作

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

3天内不再提示

浅谈嵌入式中的C语言编程技术分析

gh_94c30763133f 来源:STM32嵌入式开发 作者:STM32嵌入式开发 2021-08-19 14:50 次阅读

嵌入式产品的可靠性自然与硬件密不可分,但在硬件确定、并且没有第三方测试的前提下,使用防御性编程思想写出的代码,往往具有更高的稳定性。

防御性编程首先需要认清C语言的种种缺陷和陷阱,C语言对于运行时的检查十分弱小,需要程序员谨慎的考虑代码,在必要的时候增加判断;防御性编程的另一个核心思想是假设代码运行在并不可靠的硬件上,外接干扰有可能会打乱程序执行顺序、更改RAM存储数据等等。

1具有形参的函数,需判断传递来的实参是否合法

程序员可能无意识的传递了错误参数;外界的强干扰可能将传递的参数修改掉,或者使用随机参数意外的调用函数,因此在执行函数主体前,需要先确定实参是否合法。

4c99c0fa-fd3c-11eb-9bcf-12bb97331649.png

2仔细检查函数的返回值

对函数返回的错误码,要进行全面仔细处理,必要时做错误记录。

4ca27a06-fd3c-11eb-9bcf-12bb97331649.png

3防止指针越界

如果动态计算一个地址时,要保证被计算的地址是合理的并指向某个有意义的地方。特别对于指向一个结构或数组的内部的指针,当指针增加或者改变后仍然指向同一个结构或数组。

4防止数组越界

数组越界的问题前文已经讲述的很多了,由于C不会对数组进行有效的检测,因此必须在应用中显式的检测数组越界问题。下面的例子可用于中断接收通讯数据。

4cbb786c-fd3c-11eb-9bcf-12bb97331649.png

在使用一些库函数时,同样需要对边界进行检查,比如下面的memset(RecBuf,0,len)函数把RecBuf指指向的内存区的前len个字节用0填充,如果不注意len的长度,就会将数组RecBuf之外的内存区清零:

4ce8cf2e-fd3c-11eb-9bcf-12bb97331649.png

5数学数运算

5.1除法运算,只检测除数为零就可靠吗?

除法运算前,检查除数是否为零几乎已经成为共识,但是仅检查除数是否为零就够了吗?

考虑两个整数相除,对于一个signed long类型变量,它能表示的数值范围为:-2147483648 ~+2147483647,如果让-2147483648/ -1,那么结果应该是+2147483648,但是这个结果已经超出了signedlong所能表示的范围了。所以,在这种情况下,除了要检测除数是否为零外,还要检测除法是否溢出。

 #include <limits.h>  
   signed long sl1,sl2,result; /*初始化sl1和sl2*/   
  if((sl2==0)||(sl1==LONG_MIN && sl2==-1)) {  
   //处理错误     } else  
  {     result = sl1 / sl2; }

5.2检测运算溢出

整数的加减乘运算都有可能发生溢出,在讨论未定义行为时,给出过一个有符号整形加法溢出判断代码,这里再给出一个无符号整形加法溢出判断代码段:

 #include      unsigned int a,b,result; /*初始化a,b*/     if(UINT_MAX-a {     //处理溢出     } else    {     result=a+b; }

嵌入式硬件一般没有浮点处理器,浮点数运算在嵌入式也比较少见并且溢出判断严重依赖C库支持,这里不讨论。

5.3检测移

在讨论未定义行为时,提到有符号数右移、移位的数量是负值或者大于操作数的位数都是未定义行为,也提到不对有符号数进行位操作,但要检测移位的数量是否大于操作数的位数。下面给出一个无符号整数左移检测代码段:

 unsigned int ui1; unsigned int ui2; unsigned int uresult; /*初始化ui1,ui2*/   if(ui2>=sizeof(unsigned int)*CHAR_BIT) {     //处理错误   } else   {     uresult=ui1< }

6如果有硬件看门狗,则使用它

在其它一切措施都失效的情况下,看门狗可能是最后的防线。它的原理特别简单,但却能大大提高设备的可靠性。如果设备有硬件看门狗,一定要为它编写驱动程序。

  • 要尽可能早的开启看门狗

这是因为从上电复位结束到开启看门狗的这段时间内,设备有可能被干扰而跳过看门狗初始化程序,导致看门狗失效。尽可能早的开启看门狗,可以降低这种概率;

  • 不要在中断中喂狗,除非有其他联动措施

在中断程序喂狗,由于干扰的存在,程序可能一直处于中断之中,这样会导致看门狗失效。如果在主程序中设置标志位,中断程序喂狗时与这个标志位联合判断,也是允许的;

  • 喂狗间隔跟产品需求有关,并非特定的时间

产品的特性决定了喂狗间隔。对于不涉及安全性、实时性的设备,喂狗间隔比较宽松,但间隔时间不宜过长,否则被用户感知到,是影响用户体验的。对于设计安全性、有实时控制类的设备,原则是尽可能快的复位,否则会造成事故。

克莱门汀号在进行第二阶段的任务时,原本预订要从月球飞行到太空深处的Geographos小行星进行探勘,然而这艘太空探测器在飞向小行星时却由于一个软件缺陷而使其中断运作20分钟,不但未能到达小行星,也因为控制喷嘴燃烧了11分钟使电力供应降低,无法再透过远端控制探测器,最终结束这项任务,但也导致了资源与资金的浪费。

“克莱门汀太空任务失败这件事让我感到十分震惊,它其实可以透过硬件中一款简单的看门狗计时器避免掉这项意外,但由于当时的开发时间相当紧缩,程序设计人员没时间编写程序来启动它,”Ganssle说。

遗憾的是,1998年发射的近地号太空船(NEAR)也遇到了相同的问题。由于编程人员并未采纳建议,因此,当推进器减速器系统故障时,29公斤的储备燃料也随之报销──这同样是一个本来可经由看门狗定时器编程而避免的问题,同时也证明要从其他程序设计人员的错误中学习并不容易。

7 关键数据储存多个备份,取数据采用“表决法”

RAM中的数据在受到干扰情况下有可能被改变,对于系统关键数据应该进行保护。关键数据包括全局变量、静态变量以及需要保护的数据区域。备份数据与原数据不应该处于相邻位置,因此不应由编译器默认分配备份数据位置,而应该由程序员指定区域存储。

可以将RAM分为3个区域,第一个区域保存原码,第二个区域保存反码,第三个区域保存异或码,区域之间预留一定量的“空白”RAM作为隔离。可以使用编译器的“分散加载”机制将变量分别存储在这些区域。需要进行读取时,同时读出3份数据并进行表决,取至少有两个相同的那个值。

假如设备的RAM从0x1000_0000开始,我需要在RAM的0x1000_0000~0x10007FFF内存储原码,在0x1000_9000~0x10009FFF内存储反码,在0x1000_B000~0x1000BFFF内存储0xAA的异或码,编译器的分散加载可以设置为:

 LR_IROM1 0x00000000 0x00080000  {   
 ; load region size_region
ER_IROM10x000000000x00080000
{;loadaddress=execution address  
  *.o (RESET, +First)    *(InRoot$$Sections)   
 .ANY (+RO)   }   RW_IRAM1 0x10000000 0x00008000
  {  ;保存原码    .ANY (+RW +ZI )  
 }   RW_IRAM3 0x10009000 0x00001000{
    ;保存反码    .ANY (MY_BK1)  
 }   RW_IRAM2 0x1000B000 0x00001000  
{  ;保存异或码    .ANY (MY_BK2)  
 } }

如果一个关键变量需要多处备份,可以按照下面方式定义变量,将三个变量分别指定到三个不连续的RAM区中,并在定义时按照原码、反码、0xAA的异或码进行初始化。

 uint32  plc_pc=0;                                                       //原码   __attribute__((section("MY_BK1"))) uint32 plc_pc_not=~0x0;              //反码   __attribute__((section("MY_BK2"))) uint32 plc_pc_xor=0x0^0xAAAAAAAA;    //异或码

当需要写这个变量时,这三个位置都要更新;读取变量时,读取三个值做判断,取至少有两个相同的那个值。

为什么选取异或码而不是补码?这是因为MDK的整数是按照补码存储的,正数的补码与原码相同,在这种情况下,原码和补码是一致的,不但起不到冗余作用,反而对可靠性有害。比如存储的一个非零整数区因为干扰,RAM都被清零,由于原码和补码一致,按照3取2的“表决法”,会将干扰值0当做正确的数据。

8 对非易失性存储器进行备份存

非易失性存储器包括但不限于Flash、EEPROM、铁电。仅仅将写入非易失性存储器中的数据再读出校验是不够的。强干扰情况下可能导致非易失性存储器内的数据错误,在写非易失性存储器的期间系统掉电将导致数据丢失,因干扰导致程序跑飞到写非易失性存储器函数中,将导致数据存储紊乱。

一种可靠的办法是将非易失性存储器分成多个区,每个数据都将按照不同的形式写入到这些分区中,需要进行读取时,同时读出多份数据并进行表决,取相同数目较多的那个值。

9 软件

对于初始化序列或者有一定先后顺序的函数调用,为了保证调用顺序或者确保每个函数都被调用,我们可以使用环环相扣,实质上这也是一种软件锁。此外对于一些安全关键代码语句(是语句,而不是函数),可以给它们设置软件锁,只有持有特定钥匙的,才可以访问这些关键代码。也可以通俗的理解为,关键安全代码不能按照单一条件执行,要额外的多设置一个标志。

比如,向Flash写一个数据,我们会判断数据是否合法、写入的地址是否合法,计算要写入的扇区。之后调用写Flash子程序,在这个子程序中,判断扇区地址是否合法、数据长度是否合法,之后就要将数据写入Flash。

由于写Flash语句是安全关键代码,所以程序给这些语句上锁:必须具有正确的钥匙才可以写Flash。这样即使是程序跑飞到写Flash子程序,也能大大降低误写的风险。

  • * 名称:RamToFlash() * 功能:复制RAM的数据到FLASH,命令代码51。
  • * 入口参数:dst 目标地址,即FLASH起始地址。以512字节为分界
  • * src 源地址,即RAM地址。地址必须字对齐 * no 复制字节个数,为512/1024/4096/8192 * ProgStart 软件锁标志 * 出口参数:IAP返回值(paramout缓冲区) CMD_SUCCESS,SRC_ADDR_ERROR,DST_ADDR_ERROR, SRC_ADDR_NOT_MAPPED,DST_ADDR_NOT_MAPPED,COUNT_ERROR,BUSY,未选择扇区 ****************************************************************/
  • void RamToFlash(uint32 dst, uint32 src, uint32 no,uint8 ProgStart)
  • { PLC_ASSERT("Sector number",(dst>=0x00040000)&&(dst<=0x0007FFFF)); PLC_ASSERT("Copy bytes number is 512",(no==512)); PLC_ASSERT("ProgStart==0xA5",(ProgStart==0xA5)); paramin[0] = IAP_RAMTOFLASH; // 设置命令字 paramin[1] = dst; // 设置参数 paramin[2] = src; paramin[3] = no; paramin[4] = Fcclk/1000;
  • if(ProgStart==0xA5)
  • //只有软件锁标志正确时,才执行关键代码
  • { iap_entry(paramin, paramout); // 调用IAP服务程序 ProgStart=0; } else { paramout[0]=PROG_UNSTART; } }

该程序段是编程lpc1778内部Flash,其中调用IAP程序的函数iap_entry(paramin, paramout)是关键安全代码,所以在执行该代码前,先判断一个特定设置的安全锁标志ProgStart,只有这个标志符合设定值,才会执行编程Flash操作。如果因为意外程序跑飞到该函数,由于ProgStart标志不正确,是不会对Flash进行编程的。

10 通信

通讯线上的数据误码相对严重,通讯线越长,所处的环境越恶劣,误码会越严重。抛开硬件和环境的作用,我们的软件应能识别错误的通讯数据。对此有一些应用措施:

  • 制定协议时,限制每帧的字节数;

每帧字节数越多,发生误码的可能性就越大,无效的数据也会越多。对此以太网规定每帧数据不大于1500字节,高可靠性的CAN收发器规定每帧数据不得多于8字节,对于RS485,基于RS485链路应用最广泛的Modbus协议一帧数据规定不超过256字节。因此,建议制定内部通讯协议时,使用RS485时规定每帧数据不超过256字节;

  • 使用多种校验

编写程序时应使能奇偶校验,每帧超过16字节的应用,建议至少编写CRC16校验程序。

  • 增加额外判断

1)增加缓冲区溢出判断。这是因为数据接收多是在中断中完成,编译器检测不出缓冲区是否溢出,需要手动检查,在上文介绍数据溢出一节中已经详细说明。

2)增加超时判断。当一帧数据接收到一半,长时间接收不到剩余数据,则认为这帧数据无效,重新开始接收。可选,跟不同的协议有关,但缓冲区溢出判断必须实现。这是因为对于需要帧头判断的协议,上位机可能发送完帧头后突然断电,重启后上位机是从新的帧开始发送的,但是下位机已经接收到了上次未发送完的帧头,所以上位机的这次帧头会被下位机当成正常数据接收。这有可能造成数据长度字段为一个很大的值,填满该长度的缓冲区需要相当多的数据(比如一帧可能1000字节),影响响应时间;另一方面,如果程序没有缓冲区溢出判断,那么缓冲区很可能溢出,后果是灾难性的。

  • 重传机制

如果检测到通讯数据发生了错误,则要有重传机制重新发送出错的帧。

11 开关量输入的检测、确认

开关量容易受到尖脉冲干扰,如果不进行滤除,可能会造成误动作。一般情况下,需要对开关量输入信号进行多次采样,并进行逻辑判断直到确认信号无误为止。

12 开关量输出

开关信号简单的一次输出是不安全的,干扰信号可能会翻转开关量输出的状态。采取重复刷新输出可以有效防止电平的翻转。

13 初始化信息的保存和恢复

微处理器的寄存器值也可能会因外界干扰而改变,外设初始化值需要在寄存器中长期保存,最容易被破坏。由于Flash中的数据相对不易被破坏,可以将初始化信息预先写入Flash,待程序空闲时比较与初始化相关的寄存器值是否被更改,如果发现非法更改则使用Flash中的值进行恢复。

公司目前使用的4.3寸LCD显示屏抗干扰能力一般。如果显示屏与控制器之间的排线距离过长或者对使用该显示屏的设备打静电或者脉冲群,显示屏有可能会花屏或者白屏。

对此,我们可以将初始化显示屏的数据保存在Flash中,程序运行后,每隔一段时间从显示屏的寄存器读出当前值和Flash存储的值相比较,如果发现两者不同,则重新初始化显示屏。下面给出校验源码,仅供参考。

定义数据结构:

4d0349b2-fd3c-11eb-9bcf-12bb97331649.png

定义const修饰的结构体变量,存储LCD部分寄存器的初始值,这个初始值跟具体的应用初始化有关,不一定是表中的数据,通常情况下,这个结构体变量被存储到Flash中。

 /*LCD部分寄存器设置值列表*/  
 lcd_redu_list_struct const lcd_redu_list_str[]= { 

  {SSD1963_Get_Address_Mode,{0x20}                                  
 ,1}, /*1*/    {SSD1963_Get_Pll_Mn      
,{0x3b,0x02,0x04}                      
   ,3}, /*2*/   
 {SSD1963_Get_Pll_Status 
 ,{0x04}                              
     ,1}, /*3*/   
 {SSD1963_Get_Lcd_Mode  
  ,{0x24,0x20,0x01,0xdf,0x01,0x0f,0x00}    
 ,7}, /*4*/   
 {SSD1963_Get_Hori_Period ,{0x02,0x0c,0x00,0x2a,0x07,0x00,0x00,0x00},8}, /*5*/  
  {SSD1963_Get_Vert_Period ,{0x01,0x1d,0x00,0x0b,0x09,0x00,0x00}    
 ,7}, /*6*/    {SSD1963_Get_Power_Mode  ,{0x1c}                                  
 ,1}, /*7*/    {SSD1963_Get_Display_Mode,{0x03}                                 
  ,1}, /*8*/    {SSD1963_Get_Gpio_Conf   ,{0x0F,0x01}                       
       ,2}, /*9*/   
 {SSD1963_Get_Lshift_Freq ,{0x00,0xb8}                         
     ,2}, /*10*/ };

实现函数如下所示,函数会遍历结构体变量中的每一个命令,以及每一个命令下的初始值,如果有一个不正确,则跳出循环,执行重新初始化和恢复措施。这个函数中的MY_DEBUGF宏是我自己的调试函数,使用串口打印调试信息,在接下来的第五部分将详细叙述。

通过这个函数,我可以长时间监控显示屏的哪些命令、哪些位容易被干扰。程序里使用了一个被妖魔化的关键字:goto。大多数C语言书籍对goto关键字谈之色变,但你应该有自己的判断。在函数内部跳出多重循环,除了goto关键字,又有哪种方法能如此简洁高效!

 /**  * lcd 显示冗余  * 每隔一段时间调用该程序一次  */  
 void lcd_redu(void)   {  
   uint8_t  tmp[8];   
  uint32_t i,j;   
  uint32_t lcd_init_flag;   
  lcd_init_flag =0;
for(i=0;i<sizeof(lcd_redu_list_str)/sizeof(lcd_redu_list_str[0]);i+)     
{         LCD_SendCommand(lcd_redu_list_str[i].lcd_command);         
uyDelay(10);     
    for(j=0;j<lcd_redu_list_str[i].lcd_value_num;j++)     
    {             tmp[j]=LCD_ReadData();       
      if(tmp[j]!=lcd_redu_list_str[i].lcd_get_value[j])       
      {                 lcd_init_flag=0x55;                
 MY_DEBUGF(MENU_DEBUG,("读lcd寄存器值与预期不符,命令为:0x%x,第%d个参数,          
   该参数正确值为:0x%x,实际读出值为:0x%xn",lcd_redu_list_str[i].lcd_command,j+1,       
      lcd_redu_list_str[i].lcd_get_value[j],tmp[j]));                
 goto handle_lcd_init;             }         }     } 
    handle_lcd_init:  
   if(lcd_init_flag==0x55)   
  {         //重新初始化LCD        
   //一些必要的恢复措施       } }

14 陷阱


对于8051内核单片机,由于没有相应的硬件支持,可以用纯软件设置软件陷阱,用来拦截一些程序跑飞。对于ARM7或者Cortex-M系列单片机,硬件已经内建了多种异常,软件需要根据硬件异常来编写陷阱程序,用来快速定位甚至恢复错误。

15 阻塞处理

有时候程序员会使用while(!flag);语句阻塞在此等待标志flag改变,比如串口发送时用来等待一字节数据发送完成。这样的代码时存在风险的,如果因为某些原因标志位一直不改变则会造成系统死机。

一个良好冗余的程序是设置一个超时定时器,超过一定时间后,强制程序退出while循环。

2003年8月11日发生的W32.Blaster.Worm蠕虫事件导致全球经济损失高达5亿美元,这个漏洞是利用了Windows分布式组件对象模型的远程过程调用接口中的一个逻辑缺陷:在调用GetMachineName()函数时,循环只设置了一个不充分的结束条件。

原代码简化如下所示:

4d20a930-fd3c-11eb-9bcf-12bb97331649.png

软发布的安全补丁MS03-026解决了这个问题,为GetMachineName()函数设置了充分终止条件。一个解决代码简化如下所示(并非微软补丁代码):

 HRESULT GetMachineName( WCHAR *pwszPath,
   WCHARwszMachineName[MAX_COMPUTTERNAME_LENGTH_FQDN+1]) {   
     WCHAR *pwszServerName = wszMachineName;      
  WCHAR *pwszTemp = pwszPath + 2;      
  WCHAR *end_addr = pwszServerName +MAX_COMPUTTERNAME_LENGTH_FQDN;

while((*pwszTemp!=L’\’)&&(*pwszTemp!=L’’)&&(pwszServerName            
  *pwszServerName++= *pwszTemp++;       
 /*… */   }

编辑:hfy


原文标题:嵌入式开发中的防御性C语言编程

文章出处:【微信号:gh_94c30763133f,微信公众号:FPGA那点事儿】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    如何去实现一种ARM汇编语言程序设计呢

    ARM 程序设计 由于 C 语言便于理解,有大量的支持库,所以它是当前 ARM 程序设计所使用的主要编程语言 对硬件系统的初始...
    发表于 05-27 17:37 1355次 阅读

    智芯公司嵌入式操作系统枢纽4.0安全性达到国际安全性最高标准

    5月25日,智芯公司自主研发的嵌入式操作系统——枢纽4.0(以下简称枢纽4.0)获工信部赛宝实验室功....
    的头像 科技见闻网 发表于 05-27 11:11 217次 阅读
    智芯公司嵌入式操作系统枢纽4.0安全性达到国际安全性最高标准

    2022 RT-Thread全球技术大会:实时嵌入式操作系统概述

    RT-Thread全球技术大会:嵌入式实时操作系统的昨天、今天和明天 主要从四个方面进行介绍,首先对....
    的头像 牵手一起梦 发表于 05-27 10:42 121次 阅读
    2022 RT-Thread全球技术大会:实时嵌入式操作系统概述

    ATPCS基本规则

    有调用关系的所有子程序必须遵守同一种ATPCS,编译器或者汇编器在ELF格式的目标文件中设置相应的属....
    的头像 安芯教育科技 发表于 05-27 10:12 85次 阅读

    JWebFileTrans(JDownload)从网络上下载文件的程序

    ./oschina_soft/JDownload.zip
    发表于 05-27 09:10 7次 阅读
    JWebFileTrans(JDownload)从网络上下载文件的程序

    【i.MX6ULL】驱动开发10—阻塞&amp;非阻塞式按键检测

    本篇使用两种I/O模型进行按键读取:阻塞式I/O和非用阻塞式I/O,通过实际的实验,对比两者方式的实....
    的头像 码农爱学习 发表于 05-27 09:08 828次 阅读
    【i.MX6ULL】驱动开发10—阻塞&amp;非阻塞式按键检测

    自连数字一体化手术室方案提供数据高度共享的技术条件

    自连解决方案通过对手术室内各类医疗器械无线联网升级,例如呼吸机、监测仪、手术显微镜、内窥镜、麻醉站、....
    的头像 科技绿洲 发表于 05-26 16:42 229次 阅读

    详解回调函数的概念及使用步骤

    回调函数就是一个被作为参数传递的函数。在C语言中,回调函数只能使用函数指针实现,在C++、Pytho....
    的头像 嵌入式应用开发 发表于 05-26 15:20 133次 阅读

    【i.MX6ULL】驱动开发9——Linux I/O模型分析并进来类比

    本篇介绍了Linux中的五种I/O模型:阻塞式I/O模型、非阻塞式I/O模型、I/O复用模型、信号驱....
    的头像 码农爱学习 发表于 05-26 09:05 1130次 阅读
    【i.MX6ULL】驱动开发9——Linux I/O模型分析并进来类比

    全志H6开发板香橙派OrangePi 3 LTS在 Linux系统中安装Home Assistant的方法

    注意事项: 本篇介绍的是通过docker安装 Home Assistant 的方式,并且只提供在 Ubuntu 或者 Debian 系统中的安装方法, H...
    发表于 05-25 19:41 1214次 阅读
    全志H6开发板香橙派OrangePi 3 LTS在 Linux系统中安装Home Assistant的方法

    嵌入式工控主板的应用优势

    在我国加工制造领域和各种生产领域之中相关的自动化技术应用之下,要求更多专业的控制主板来提高系统运转的....
    的头像 kimkylin 发表于 05-25 15:31 34次 阅读

    嵌入式主板的常见种类及应用领域

    嵌入式主板一般理解为嵌入在设备里面做控制、数据处理使用的CPU板,也就是设备的"大脑"。嵌入式到设备....
    的头像 kimkylin 发表于 05-25 15:24 82次 阅读

    微型转换器取代电力轨道集群的方法

      市场正在提供越来越多的带有硬件加密引擎的高端通用微控制器。然而,仅仅因为微控制器具有硬件加密引擎....
    的头像 星星科技指导员 发表于 05-25 15:07 97次 阅读

    振弦采集模块VM,如何修改通讯速率?

    振弦采集模块VM系列,如何修改通讯速率?
    发表于 05-25 10:43 1491次 阅读

    Vitis HLS工具简介及设计流程

    Vitis HLS 是一种高层次综合工具,支持将 C、C++ 和 OpenCL 函数硬连线到器件逻辑....
    的头像 Xilinx赛灵思官微 发表于 05-25 09:43 239次 阅读

    【i.MX6ULL】驱动开发8—中断法检测按键 Linux中断和定时使用方法

    本篇主要介绍了Linux中断的使用方法,通过按键来进行中断实验测试,并使用Linux定时器进行按键去....
    的头像 码农爱学习 发表于 05-25 09:09 1354次 阅读
    【i.MX6ULL】驱动开发8—中断法检测按键 Linux中断和定时使用方法

    触控工业显示器内嵌安装与外嵌安装如何选

    嵌入式安装是一种常见的安装方式,它可以让两个物体贴合安装,形成一个整体,看上去简洁、美观且不占空间,....
    发表于 05-24 17:39 9次 阅读

    基于TI AM5728 + Artix-7 FPGA开发板(DSP+ARM) 5G通信测试手册

    前 言本文主要基于我司TL5728F-EVM评估板 + 移远RM500Q 5G模块演示5G网络的功能。 本指导文档适用开发环境:Window...
    发表于 05-24 17:26 1130次 阅读

    使用安全身份验证器保护您的医疗设备端点

      安全认证器通过提供基于硬件的加密、固定功能操作来应对威胁;安全存储;和攻击对策。借助这些 IC ....
    的头像 星星科技指导员 发表于 05-24 16:41 293次 阅读
    使用安全身份验证器保护您的医疗设备端点

    安全IC如何保护物联网设备中的传输数据

     为了拥有真正安全的 TLS 方案并避免我们讨论过的陷阱,需要遵循一组最低限度的规则。在使用时保护会....
    的头像 星星科技指导员 发表于 05-24 16:19 355次 阅读

    深入介绍嵌入式开发中三种不同的程序架构

    三种常用的软件架构有:顺序执行的前后台系统、时间片轮询系统和多任务操作系统:STM32单片机开发中的....
    的头像 STM32嵌入式开发 发表于 05-24 16:02 304次 阅读

    Linux驱动开发_mplayer播放器开发

    USB wifi (360) : 如何从官网下载源码进行代码移植、安装相关无线工具、让嵌入式设备支持....
    的头像 DS小龙哥-嵌入式技术 发表于 05-24 09:56 221次 阅读
    Linux驱动开发_mplayer播放器开发

    如何通过PUF技术防止侵入式攻击

      Maxim 的第一款 PUF 产品是采用 ChipDNA™ 技术的 DS28E38 DeepCo....
    的头像 星星科技指导员 发表于 05-24 09:47 145次 阅读
    如何通过PUF技术防止侵入式攻击

    VM振弦采集模块 ,使用时单次测量和连续测量的区别是什么?

    发表于 05-24 09:41 1194次 阅读

    学好嵌入式Linux,需不需要全精通?专家给的建议一语中的!

    学习之前,一定要弄清楚自己要从事的方向。对于嵌入式开发,方向就分为很多种。比较主流的方向是单片机、L....
    的头像 jf_35791075 发表于 05-24 09:27 2次 阅读
    学好嵌入式Linux,需不需要全精通?专家给的建议一语中的!

    i.MX6ULL 驱动开发7—按键输入捕获与GPIO输入配置与高低电平读取

    本篇主要介绍了i.MX6ULL的按键检测的使用,主要的知识点是设备树的修改,以及GPIO的输入配置与....
    的头像 码农爱学习 发表于 05-24 09:11 2050次 阅读
    i.MX6ULL 驱动开发7—按键输入捕获与GPIO输入配置与高低电平读取

    基于TI AM5728 Artix-7通信综合开发案例

    此案例来源于:创龙科技测试板卡为:TISitara系列AM5728+XilinxArtix-7FPG....
    发表于 05-23 18:03 8次 阅读

    创龙基于AM57X 的CPSW网口SWITCH模式使用说明

    此案例来源于:创龙科技测试板卡为:TISitara系列AM5728+XilinxArtix-7FPG....
    发表于 05-23 17:23 7次 阅读

    TI AM57X FPGA MicroBlaze裸机案例开发

    此案例来源于:创龙科技测试板卡为:TISitara系列AM5728+XilinxArtix-7FPG....
    发表于 05-23 17:23 5次 阅读

    C语言基础知识分享

    常见问题答疑 1. 学习 java为什么建议先学 语言 没学过计算机专业课程能够学懂 语言 英....
    发表于 05-23 16:07 15次 阅读

    嵌入式开发中串口通讯方案

    在嵌入式开发中,经常会用到串口通讯。面对不同应用场景,需要不同的方案。
    的头像 小麦大叔 发表于 05-23 11:48 273次 阅读

    基于non-OS和RT-Thread的按键库源码及应用

    核心的按键扫描代码仅有三行,没错,就是经典的 三行按键扫描算法。使用 C 语言标准库 API 编写,....
    的头像 硬件攻城狮 发表于 05-23 09:39 316次 阅读
    基于non-OS和RT-Thread的按键库源码及应用

    电路设计中常用的滤波方法锦集

    在整流电路输出的电压是单向脉动性电压,不能直接给电子电路使用。所以要对输出的电压进行滤波, 消除电压....
    的头像 嵌入式应用开发 发表于 05-22 16:09 453次 阅读
    电路设计中常用的滤波方法锦集

    【i.MX6ULL】驱动开发6——GPIO子系统点亮LED

    本篇介绍了使用**Pinctrl子系统与GPIO子系统**的方式来点亮LED,与之前的寄存器版点亮L....
    的头像 码农爱学习 发表于 05-21 21:50 1388次 阅读
    【i.MX6ULL】驱动开发6——GPIO子系统点亮LED

    【i.MX6ULL】驱动开发5——设备树原理与点亮LED

    本篇介绍了设备树的基本原理以及设备树的使用方法,在上一篇点亮LED的代码基础上,通过设备树的方式,实....
    的头像 码农爱学习 发表于 05-21 21:41 1345次 阅读
    【i.MX6ULL】驱动开发5——设备树原理与点亮LED

    【i.MX6ULL】驱动开发4——点亮LED(寄存器版)

    本篇主要介绍了如何通过操作寄存器来点亮i.MX6ULL开发板上的led,通过编写LED对应的驱动程序....
    的头像 码农爱学习 发表于 05-21 21:26 1272次 阅读
    【i.MX6ULL】驱动开发4——点亮LED(寄存器版)

    C语言_二维数组与一维指针

    字符串:本身属于字符数组、字符串结尾有’\0’。
    的头像 DS小龙哥-嵌入式技术 发表于 05-21 09:11 361次 阅读

    了解系统可靠性何时以及为何如此重要

      Maxim Integrated 推出了一款旨在保持系统高可靠性的新型低功耗微控制器:MAX32....
    的头像 星星科技指导员 发表于 05-21 07:19 74次 阅读

    STM32嵌入式显示器创建一个有答案的生态系统

      工程师可以通过选择是否使用触摸面板、光学或空气粘合、触摸检测的形状以及是否添加框架来定制他们的 ....
    的头像 星星科技指导员 发表于 05-21 07:02 74次 阅读

    「案例分享」基于 AM57x+ Artix-7 FPGA开发板——PRU开发手册详解

    PRU-ICSS(可编程实时单元子系统和工业通信子系统,简称PRU)由双32位RISC核(PRU可编程实时单元)、共享数据和...
    发表于 05-20 17:10 5565次 阅读

    NIST认证和测试为安全解决方案提供信心

    NIST 的成立是为了通过纠正二流的测量基础设施来提高美国企业的竞争力。由于主要优先事项是互操作性,....
    的头像 星星科技指导员 发表于 05-20 16:18 257次 阅读

    ARM怎样去设置imx6开发板的IP地址呢

    imx6 开发板网络相关配置 声明: 每位 imx6开 发板 用户的网络使用环境未必相同,本次操作基于飞凌嵌入式imx6开发板, 网络环境...
    发表于 05-20 15:10 3624次 阅读

    5g NR 入门 --带你了解新无线电通信领域的先进技术

    了解基础的5g NR (新无线电) ,包括其频带,应用,什么技术使之成为可能        5G 是无线智能手机通...
    发表于 05-20 10:33 782次 阅读
    5g NR 入门 --带你了解新无线电通信领域的先进技术

    Zbox-FS零公开、重隐私的嵌入式文件系统

    ./oschina_soft/zbox.zip
    发表于 05-20 09:33 12次 阅读
    Zbox-FS零公开、重隐私的嵌入式文件系统

    详解内存池技术的原理与实现

    最近在网上看到了几篇篇讲述内存池技术的文章,有一篇是有IBM中国研发中心的人写的,写的不错~~文章地....
    的头像 Linux内核补给站 发表于 05-20 08:58 3542次 阅读
    详解内存池技术的原理与实现

    微控制器中各式各样的编程语言

    在上一篇文章中,我们探索了微控制器中的软件与硬件,并分析了为何微控制器的软件和硬件相辅相成缺一不可。....
    的头像 东芝半导体 发表于 05-19 19:58 651次 阅读

    RT—thread线程启动详解

    就序列表实现 就序列表实际上就是一个数组,用来存放优先级不同的线程,系统默认最大的线程数量为32个,....
    的头像 嵌入式应用开发 发表于 05-19 16:17 240次 阅读
    RT—thread线程启动详解

    stm32裸机RT—thread开始创建线程详解

          在裸机系统 中,他们统统放在一个叫栈的地方,栈是单片机RAM里面一段连续的内存空间,栈的....
    的头像 嵌入式应用开发 发表于 05-19 15:02 188次 阅读
    stm32裸机RT—thread开始创建线程详解

    嵌入式ARM9 S3C2451如何配置GPIO引脚功能呢

    博主以循循渐进的方式理解 总结如下: GPxCON   配置引脚功能          &...
    发表于 05-19 14:59 3588次 阅读

    C语言—数组详解

    介绍C语言数组的知识点。
    的头像 DS小龙哥-嵌入式技术 发表于 05-19 10:15 3611次 阅读

    想请教下关于单片机io内部结构和内阻的问题

    我看有说单片机内部有很大的内阻,那么想请问下: 问题一:(1)输出时:io可否直接接地?(2)输入时:io可否直接接vcc? 现实...
    发表于 05-19 01:10 1492次 阅读

    Debian系统移植USBWIFI RTL8192EU驱动

    作者: donatello1996 来源:电子发烧友 原文标题: 【飞凌嵌入式OKMX8MP-C 开....
    的头像 柠檬籽儿 发表于 05-18 15:15 277次 阅读
    Debian系统移植USBWIFI RTL8192EU驱动

    移植RT-Thread到STM32开发板的详细步骤例程

    本章开始,先新建一个基于野火STM32全系列(包含M3/4/7)开发板的的RT-Thread的工程模....
    的头像 嵌入式应用开发 发表于 05-18 09:14 1339次 阅读
    移植RT-Thread到STM32开发板的详细步骤例程

    关于CAS等原子操作介绍 无锁队列的链表实现方法

    在开始说无锁队列之前,我们需要知道一个很重要的技术就是CAS操作——Compare & Set,或是....
    的头像 书生途 发表于 05-18 09:12 162次 阅读
    关于CAS等原子操作介绍 无锁队列的链表实现方法

    或许智能家居不再是简单的智能设备控制

    说到智能家居你会想到什么,语音声控、自动感知、人性化设置、住宅家居一体化?随着综合布线技术、网络通信....
    发表于 05-17 14:26 47次 阅读

    编程案例 使用MPLAB®X开发AVR®MCU 写C程序框架

    从新建一个空白程序开始,从头写一个程序框架,能有助于更好的理解MPLAB® X开发环境所提供的各种编....
    的头像 物联网评论 发表于 05-17 11:24 2140次 阅读
    编程案例 使用MPLAB®X开发AVR®MCU 写C程序框架

    高性能超值系列微控制器具有更多可能性

      传统上,竞争的微处理器制造商在环境温度升高时试图全速运行 CPU 时会遇到散热问题。因此,开发人....
    的头像 星星科技指导员 发表于 05-17 09:52 234次 阅读

    STM32 Step-by-Step如何加快开发速度

    STM32 Step-by-Step 的另一个基本方面是我们的新社区。多年来,许多成员都表达了学习如....
    的头像 星星科技指导员 发表于 05-16 16:23 271次 阅读

    BlueNRG-Mesh创建蓝牙网状网络解决方案

      通过 STM32Cube 的X-CUBE-BLEMESH1扩展软件包,在连接到 NUCLEO-L....
    的头像 星星科技指导员 发表于 05-16 16:02 214次 阅读

    用C语言写出简单的加密算法

    有一套四位数加密系统,输入四位数以后会自动加密。加密规则如下:每位数字都加上 5,然后用和除以 10....
    的头像 FPGA之家 发表于 05-16 10:20 166次 阅读