单片机中的_nop_()延时及其相关的基础扩展

嵌入式ARM 2023-10-30 12:01
使用 _nop_() 函数做延时遇到的一些问题 以及对此延伸出的一些需要了解的基本概念 ...... by  矜辰所致
完善文章内容结构,补充指令周期、机器周期等一些基本概念   
最近还是继续做着项目,因为在某 8051 内核芯片上使用到了 I2C 通讯,又需要 _nop_() 函数来实现 us 延时,那么正好来写一篇由_nop_() 函数引起的一系列基本概念 。
本文内容:对 nop 的认识,单片机中的 nop 函数做延时的一些注意事项,以及单片机中基本的指令周期,机器周期等一些基本概念的说明。

本文目录:

  • 一、 NOP 指令

    • 1.1 NOP 指令的作用

  • 二、单片机中的 `_nop_()` 函数

    • 2.1 C语言中的 NOP

    • 2.2 _nop_ 函数消耗的时间

  • 三、用 `_nop_()` 延时的注意事项

    • 3.1 函数调用对延时的影响

    • 3.2 调用函数中的语句对延时的影响

  • 四、指令周期、机器周期、时钟周期

  • 结语

一、 NOP 指令

_nop_() 函数产生的是 NOP 指令,先来简单介绍一下 NOP 指令,基本介绍走个流程把:

NOP 是编程语言中一个经常用到的指令,它的全称是 No Operation,即无操作指令。

NOP 是汇编语言中的一个伪指令,通过NOP一系列的编程语句,能够不改变任何程序可以访问的寄存器。

NOP 指令的作用:

  1. 我们知道,指令、数据对齐可以有效地提高程序的性能, 使用 NOP 指令,可以使得指令按字对齐,从而提高效率 。比如一条指令占用 3 个字节,再加上一个 NOP 指令,就使得指令 4 字节对齐了。

  2. 通过 NOP 指令产生一定的延迟,这与 CPU 的频率有关系,适用于一些频率低的 单片机 场合。

  3. 计算机在输入或者输出的过程中,使用 NOP 指令可可以很好的等待计算机缓冲区清空,等待总线恢复正常,其实也算是延时的一种了

二、单片机中的 _nop_() 函数

1、C语言中的 NOP

如果使用汇编语言,我们可以直接使用 NOP 指令的 ,直接写一个 nop 就可以,比如下面示例:

        .text                   ; 代码段开始
        .syntax unified         ; 

start:
        mov r0, #0x55          ; 将0x55存储在寄存器R0中
        nop                    ; 插入NOP指令
        mov r1, #0xAA          ; 将0xAA存储在寄存器R1中

        add r2, r0, r1         ; 将R0和R1相加并将结果存储在R2中

但是我们在单片机中编程,现在都是使用 C 语言,对于 C 语言本身来说,是没有空语句的。

但是我们在做51单片机的开发中,在库文件中提供了一个void _nop_(void);函数,这个函数声明一般在 intrins.h 头文件当中,我们只需要 #include 就可以使用 _nop_(); 函数了。

比如:

我们已经知道了 nop 是空语句,什么都不做,但是在这里我们还是得明确的知道 一个 _nop_()表示空循环一个机器指令的时间。

2、nop 函数消耗的时间

那么在我们的单片机中,一个 nop 的时间是多少呢?、

上面说到,一个 nop 表示一个机器周期,那么一个机器周期是多少?

机器周期当然与主频有关,在单片机中指的就是晶振的频率。

首先基本的东西还是要知道的 一个机器周期包含12个晶振周期。 所以我们可以通过下面的计算得知 nop 函数消耗的时间:

假设单片机 12M 晶振,晶振周期1/12微秒,一个机器周期包含12个晶振周期,所以12M晶振时机器周期 = 12x(1/12)us = 1us 。.

所以12M 晶振中一个 nop 表示延时1us; 

6M 晶振中延时2us,24M 晶振中延时 0.5 us

至于其他的晶振频率,我们可以按照上面的计算代入即可。

对于 _nop_() 函数 其实在我以前的文章 BH1750 传感器实战教学 —— 驱动移植篇 中有过说明:

三、用 _nop_() 延时的注意事项

至此,我们已经可以知道在我们的程序中,一个 nop 函数执行所需要时间,我们可以利用多个  _nop_() 函数来实现一些 us 级别的延时。

比如我以前一些帖子里面提到的在 51 上面的 I2C 通讯:

在上图中,就是一个简单的 I2C 其实信号的实现方式, 在上图中,有说明 多几个 nop 少几个 nop 无所谓,实际上现在看来是有问题的,这让我付出了代价,这一点我后面会在写某个传感器测试博文的时候会提到。

1、函数调用对延时的影响

那么本文这里要说明的是一些使用时候的问题,依然是我以前文中提到的,在 STM32 HAL 库中没有 us 延时,所以我一直用的是:

void delay_us(uint32_t Delay)
{
 uint32_t cnt = Delay * 8;   // 32Mhz ,其他频率其他倍数
 uint32_t i = 0;
 for(i = 0; i < cnt; i++)__NOP();
}

于是乎,对于本次使用的 16MHZ 晶振的 51 芯片,我改成了如下:

void delay_us(uint32 Delay)
{
  uint32 cnt = Delay * 4;   // 32Mhz 8 ,其他频率其他倍数     16Mhz慢一点  4
  uint32 i = 0;
  for(i = 0; i < cnt; i++)_nop_();
}

然后自然的把上面的 I2C_Start 改成如下:

void I2C_Start1(void)
{
   sda_high();
   delay_us(5);
   scl_high();
   delay_us(10);
   sda_low();
   delay_us(10);
   scl_low(); //使SCL置低,准备发送或者接受数据
   delay_us(10);
}

反正改完以后传感器通讯是不正确的,于是乎最后上了示波器,惊讶的发现,在我使用的 51 上面采用上面的方式的波形图如下(注意看波形的时间):

是不是很意外,时间周期居然可以达到 ms 级别,就是使用一个一个循环调用 nop 的函数……,我一个 I2C 传感器的初始化工作,居然持续了好几秒时间……

而在 STM32 平台下面,我观察到的波形图如下(us级别算是正常的):

虽然知道调用函数会占用时间,但是上面的情况也太离谱了点,即便我最后把循环里面的 *4 都直接删除,波形周期还是 ms 级别。

这…… 真的是有点太离谱了,一个简单的 nop 延时函数在实际上会有这么久的延时……

反正最后我还是去掉了函数,采用直接使用很多个 nop 函数直接写的方式,如下图的上面部分:

实际上,除了调用函数,在函数中的使用什么语法也决定了这个函数执行的时间长短,这个问题对于我们现在大家常用的 ARM 内核来说,可能都不太容易发现,或者影响没那么大,但是对于老一点的 51 内核,影响就大了,但大到上面这种程度,也是我没想到的。

2、调用函数中的语句对延时的影响

那说到除了调用函数,函数中的语句是如何影响时间的呢,这因为在C51编译器中,对不同的循环方法,采用不同的指令来完成的,对于不同的指令,单片机执行所需要的的时间也是不一样的。

单片机执行一条指令所需要的时间:

完成一条指令需要的时间,也就是指令周期。

指令周期就是 单片机 取出一条指令并执行这条指令所需要的时间。指令周期,是从取指令、分析指令到执行完所需的全部时间。

指令周期一般由若干个机器周期组成(我们上面讲过,一个 _nop_() 就是一个机器周期),他是以机器周期为单位的!!!

其实通过我们前文的介绍,我们已经知道如何算一个 单片机的 机器周期(一个 nop 的时间,他是由 12 个时钟周期组成的),我们只需要知道这条指令是由几个机器周期组成的就可以,这一点会在单片机的使用手册中有说明,比如下图:

上图中上面一些指令需要 12 个时钟周期,就是一个机器周期,最后一个需要 2个 机器周期。

大家都能看到其实后面有一个 6T 模式的说明,很容易理解,就是1机器周期等于 6 个时钟周期的模式,这样会使得单片机执行效率提升 2 倍 ,现代单片机有许多都有这种高效率模式。


了解完了指令周期,那我们是不是很容易的就明白了,在函数中为什么不同的语句会对延时产生不同影响了。

这里呢,我就不对不同的语句进行单独的分析了,大家有时间可以自己生成汇编文件自己研究,这里我就从网上截取了部分说明:

在选择C51中循环语句时,要注意以下几个问题 . 

第一、定义的 C51 中循环变量,尽量采用无符号字符型变量。.

第二、在 for 循环语句中,尽量采用变量减减来做循环。.

第三、在do…while,while语句中,循环体内变量也采用减减方法。

我们要知道的是,上面的做法都是为了减少额外的时间开销,使得我们想要的延时时间更加准确。

四、指令周期、机器周期、时钟周期

在文章上面部分由反复的提到过几个概念:指令周期,机器周期,时钟周期。

为了防止有的小伙伴还是迷迷糊糊的,这里来简单的总结一下(以 8051 单片机为例):

时钟周期  =   1/ 晶振频率
单片机的心跳,基本时间单位 . 

机器周期  =  时钟周期 * 12 

单片机的基本操作周期,一个机器周期,单片机完成一项基本操作,如取指令,读/写存储器 . 

指令周期 : 

CPU 执行一条指令所需要的时间, 以机器周期为单位。 

指令周期所需要的的机器周期,可以通过单片机使用手册中的指令表查询得到。

当然,其实与上面这些概念相关的还有一个状态周期,他等于 2个 时钟周期,这里也提一下。

五、结语

本文通过一个简单的 _nop_() 函数,我们探讨了在单片机中实现 us 延时的一些时间问题以及注意事项,进而引出了一些时间周期的基本概念,相信能让大家在日后使用到的时候能够更好的理解与计算自己所需要的延时时间。

END

来源:矜辰所致


版权归原作者所有,如有侵权,请联系删除。


推荐阅读

我用这个技术,干掉几千行if else!

麒麟9000s,并非来自SMIC,而是...

程序员最容易读错的单词,听到status我炸了


→点关注,不迷路←

嵌入式ARM 关注这个时代最火的嵌入式ARM,你想知道的都在这里。
评论
  • 车身域是指负责管理和控制汽车车身相关功能的一个功能域,在汽车域控系统中起着至关重要的作用。它涵盖了车门、车窗、车灯、雨刮器等各种与车身相关的功能模块。与汽车电子电气架构升级相一致,车身域发展亦可以划分为三个阶段,功能集成愈加丰富:第一阶段为分布式架构:对应BCM车身控制模块,包含灯光、雨刮、门窗等传统车身控制功能。第二阶段为域集中架构:对应BDC/CEM域控制器,在BCM基础上集成网关、PEPS等。第三阶段为SOA理念下的中央集中架构:VIU/ZCU区域控制器,在BDC/CEM基础上集成VCU、
    北汇信息 2025-01-03 16:01 173浏览
  • 在快速发展的能源领域,发电厂是发电的支柱,效率和安全性至关重要。在这种背景下,国产数字隔离器已成为现代化和优化发电厂运营的重要组成部分。本文探讨了这些设备在提高性能方面的重要性,同时展示了中国在生产可靠且具有成本效益的数字隔离器方面的进步。什么是数字隔离器?数字隔离器充当屏障,在电气上将系统的不同部分隔离开来,同时允许无缝数据传输。在发电厂中,它们保护敏感的控制电路免受高压尖峰的影响,确保准确的信号处理,并在恶劣条件下保持系统完整性。中国国产数字隔离器经历了重大创新,在许多方面达到甚至超过了全球
    克里雅半导体科技 2025-01-03 16:10 119浏览
  • 【工程师故事】+半年的经历依然忧伤,带着焦虑和绝望  对于一个企业来说,赚钱才是第一位的,对于一个人来说,赚钱也是第一位的。因为企业要活下去,因为个人也要活下去。企业打不了倒闭。个人还是要吃饭的。企业倒闭了,打不了从头再来。个人失业了,面对的不仅是房贷车贷和教育,还有找工作的焦虑。企业说,一个公司倒闭了,说明不了什么,这是正常的一个现象。个人说,一个中年男人失业了,面对的压力太大了,焦虑会摧毁你的一切。企业说,是个公司倒闭了,也不是什么大的问题,只不过是这些公司经营有问题吧。
    curton 2025-01-02 23:08 289浏览
  • 本文继续介绍Linux系统查看硬件配置及常用调试命令,方便开发者快速了解开发板硬件信息及进行相关调试。触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。查看系统版本信息查看操作系统版本信息root@ido:/# cat /etc/*releaseDISTRIB_ID=UbuntuDISTRIB_RELEASE=20.04DISTRIB_CODENAME=focalDIS
    Industio_触觉智能 2025-01-03 11:37 137浏览
  • Matter加持:新世代串流装置如何改变智能家居体验?随着现在智能家庭快速成长,串流装置(Streaming Device,以下简称Streaming Device)除了提供更卓越的影音体验,越来越多厂商开始推出支持Matter标准的串流产品,使其能作为智能家庭中枢,连结多种智能家电。消费者可以透过Matter的功能执行多样化功能,例如:开关灯、控制窗帘、对讲机开门,以及操作所有支持Matter的智能家电。此外,再搭配语音遥控器与语音助理,打造出一个更加智能、便捷的居家生活。支持Matter协议
    百佳泰测试实验室 2025-01-03 10:29 143浏览
  • 影像质量应用于多个不同领域,无论是在娱乐、医疗或工业应用中,高质量的影像都是决策的关键基础。清晰的影像不仅能提升观看体验,还能保证关键细节的准确传达,例如:在医学影像中,它对诊断结果有着直接的影响!不仅如此,影像质量还影响了:▶ 压缩技术▶ 存储需求▶ 传输效率随着技术进步,影像质量的标准不断提高,对于研究与开发领域,理解并提升影像质量已成为不可忽视的重要课题。在图像处理的过程中,硬件与软件除了各自扮演着不可或缺的基础角色,有效地协作能够确保图像处理过程既高效又具有优异的质量。软硬件各扮演了什么
    百佳泰测试实验室 2025-01-03 10:39 137浏览
  • 物联网(IoT)的快速发展彻底改变了从智能家居到工业自动化等各个行业。由于物联网系统需要高效、可靠且紧凑的组件来处理众多传感器、执行器和通信设备,国产固态继电器(SSR)已成为满足中国这些需求的关键解决方案。本文探讨了国产SSR如何满足物联网应用的需求,重点介绍了它们的优势、技术能力以及在现实场景中的应用。了解物联网中的固态继电器固态继电器是一种电子开关设备,它使用半导体而不是机械触点来控制负载。与传统的机械继电器不同,固态继电器具有以下优势:快速切换:确保精确快速的响应,这对于实时物联网系统至
    克里雅半导体科技 2025-01-03 16:11 164浏览
  • 在测试XTS时会遇到修改产品属性、SElinux权限、等一些内容,修改源码再编译很费时。今天为大家介绍一个便捷的方法,让OpenHarmony通过挂载镜像来修改镜像内容!触觉智能Purple Pi OH鸿蒙开发板演示。搭载了瑞芯微RK3566四核处理器,树莓派卡片电脑设计,支持开源鸿蒙OpenHarmony3.2-5.0系统,适合鸿蒙开发入门学习。挂载镜像首先,将要修改内容的镜像传入虚拟机当中,并创建一个要挂载镜像的文件夹,如下图:之后通过挂载命令将system.img镜像挂载到sys
    Industio_触觉智能 2025-01-03 11:39 113浏览
  • 自动化已成为现代制造业的基石,而驱动隔离器作为关键组件,在提升效率、精度和可靠性方面起到了不可或缺的作用。随着工业技术不断革新,驱动隔离器正助力自动化生产设备适应新兴趋势,并推动行业未来的发展。本文将探讨自动化的核心趋势及驱动隔离器在其中的重要角色。自动化领域的新兴趋势智能工厂的崛起智能工厂已成为自动化生产的新标杆。通过结合物联网(IoT)、人工智能(AI)和机器学习(ML),智能工厂实现了实时监控和动态决策。驱动隔离器在其中至关重要,它确保了传感器、执行器和控制单元之间的信号完整性,同时提供高
    腾恩科技-彭工 2025-01-03 16:28 159浏览
  • 光耦合器,也称为光隔离器,是一种利用光在两个隔离电路之间传输电信号的组件。在医疗领域,确保患者安全和设备可靠性至关重要。在众多有助于医疗设备安全性和效率的组件中,光耦合器起着至关重要的作用。这些紧凑型设备经常被忽视,但对于隔离高压和防止敏感医疗设备中的电气危害却是必不可少的。本文深入探讨了光耦合器的功能、其在医疗应用中的重要性以及其实际使用示例。什么是光耦合器?它通常由以下部分组成:LED(发光二极管):将电信号转换为光。光电探测器(例如光电晶体管):检测光并将其转换回电信号。这种布置确保输入和
    腾恩科技-彭工 2025-01-03 16:27 155浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦