RDTSC指令实现纳秒级计时器 _rdtsc 转换 纳秒-程序员宅基地

技术标签: 汇编  编译器  performance  x86  mfc  实验室  signal  

 
 
 

X86 platform

    从pentium开始,很多80x86微处理器都引入TSC,一个用于时间戳计数器的64位的寄存器,它在每个时钟信号(CLK, CLK是微处理器中一条用于接收外部振荡器的时钟信号输入引线)到来时加一。
    通过它可以计算CPU的主频,比如:如果微处理器的主频是1MHZ的话,那么TSC就会在1秒内增加1000000。除了计算CPU的主频外,还可以通过TSC来测试微处理器其他处理单元的运算速度,资料[2]介绍了这个内容。
    那么如何获取TSC的值呢?rdtsc,一条读取TSC的指令,它把TSC的低32位存放在eax寄存器中,把TSC的高32位存放在edx中,更详细的描述见资料[1]。
    下面来看看rdtsc的具体用法,在linux源代码include/asm-i386/msr.h中,可以找到这么三个关于rdtsc的宏定义:

 

#define rdtsc(low,high) /
      __asm__ __volatile__("rdtsc" : "=a" (low), "=d" (high))
 
#define rdtscl(low) /
       __asm__ __volatile__("rdtsc" : "=a" (low) : : "edx")
 
#define rdtscll(val) /
       __asm__ __volatile__("rdtsc" : "=A" (val))

 

 


    第三个正是我们需要的,为了方便在我们自己的应用程序中使用,通过资料[3]我们把它简单的封装一下:

 

typedef unsigned long long cycles_t;
inline cycles_t currentcycles() {
    cycles_t result;
    __asm__ __volatile__ ("rdtsc" : "=A" (result));
    return result;
}

 

 



    下面来介绍一个实例,用rdtsc来计算CPU的主频。计算方法就是根据TSC的工作原理来的。我们在时间间隔1秒的前后分别记下TSC的值,然后求差并除以1000000。这样就可以计算出以MHZ为单位的主频了。大概的算法如下:

 

t1 = currentcycles();
sleep(1);
t2 = currentcycles();
printf("cpu MHz        : %lld/n", (t2-t1)/1000000);

 

 



    不过考虑到sleep是基于alarm和pause实现的,我们这里直接通过alarm来产生1秒的时间间隔了。

/**
* readTSC.c -- read the time stamp counter using the rdtsc instruction *
* falcon <[email protected]> * 2008-04-07 *
* ref: 1. [url]http://z.cs.utexas.edu/users/habals/blog/index.php/linux/70[/url]
* 2. [url]http://www.e7188.com/Article/safe/300/878/2007/2007022852353.html[/url] *
*/

#include <stdio.h> /* printf */
#include <unistd.h> /* alarm, pause */
#include <sys/types.h>
#include <signal.h> /* signal,kill */

typedef unsigned long long cycles_t;

inline cycles_t currentcycles()
{
    cycles_t result;
    __asm__ __volatile__ ("rdtsc" : "=A" (result));
        
    return result;
}

cycles_t t1, t2;

void handler(int signo)
{
    t2 = currentcycles();
    printf("cpu MHz : %lld/n", (t2-t1)/1000000);
    kill(getpid(), SIGINT);
}

int main(void)
{
    signal(SIGALRM, handler);
    t1 = currentcycles();
    alarm(1);
    while(1)
        pause();
    
    return 0;
}


这里是执行情况:

 

$ make readTSC
cc     readTSC.c   -o readTSC
$ ./readTSC
cpu MHz        : 2199

$ cat /proc/cpuinfo | grep MHz
cpu MHz        : 2200.103

 

 



    我们算出来的主频跟内核proc文件系统中记录的值差不多,不过内核里头计算的要大一些,可能内核在计算主频时用了浮点运算的缘故。实际上我们计算的值也比真实的要大,因为在t1和t2之间,除了1s外,我们调用了alarm,而且进行了除法运算,因此实际时间要大于1秒,所以实际主频就会更小一些。

参考资料:

[1] RDTSC instruction
http://www.h52617221.de/dictionary.php?id=278  
[2] Using the RDTSC Instruction for Performance Monitoring
http://cs.smu.ca/~jamuir/rdtscpm1.pdf
[3] Use RDTSC instruction to measure time on x86 architecture
http://z.cs.utexas.edu/users/habals/blog/index.php/linux/70

end of X86 platform

 

-------------------------------------------------------------------------

从别人文章中摘出来的关于MIPS记时相关的部分。

********* 描述1 ***********

在mips平台上,可以通过读取coprocessor 0中的寄存器9来获取time stamp,

#define rdtscl(dest)/
      __asm__ __volatile__("mfc0 %0, $9; nop":"=r"(dest));

 

 

********* 描述2 ***********

除了这个与体系结构无关的函数外,我们还将示例使用一段内嵌的汇编代码。为此,我们来给 MIPS 处理器实现一个 rdtscl 函数,功能就象 x86 的一样。

这个例子之所以基于 MIPS,是因为大多数 MIPS 处理器都有一个 32 位的计数器,在它们的内部“coprocessor 0”中命名为 register 9 寄存器。为了从内核空间读取该寄存器,可以定义下面的宏,它执行“从 coprocessor 0 读取”的汇编指令:*


#define rdtscl(dest) /
   _ _asm_ _ _ _volatile_ _("mfc0 %0,$9; nop" : "=r" (dest))





注:nop 指令是必需的,防止了编译器在指令mfc0之后立刻访问目标寄存器。这种互锁(interlock)在 RISC处理器中是很典型的,在延迟期间编译器仍然可以调度其它指令执行。我们在这里使用nop,是因为内嵌汇编指令对编译器来说是个黑盒,不能进行优化。



通过使用这个宏,MIPS 处理器就可以执行和前面所示用于 x86 的相同的代码了。

gcc 内嵌汇编的有趣之处在于通用寄存器的分配使用是由编译器完成的。这个宏中使用的 %0 只是“参数 0”的占位符,参数 0 由随后的“作为输出(=)使用的任意寄存器(r)”定义。该宏还说明了输出寄存器要对应于 C 表达式 dest。内嵌汇编的语法功能强大但也比较复杂,特别是在对各寄存器使用有限制的平台上更是如此,如 x86 系列。完整的语法描述在 gcc 文档中提供,一般在 info 中就可找到。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/gonxi/article/details/6104842

智能推荐

c# 调用c++ lib静态库_c#调用lib-程序员宅基地

文章浏览阅读2w次,点赞7次,收藏51次。四个步骤1.创建C++ Win32项目动态库dll 2.在Win32项目动态库中添加 外部依赖项 lib头文件和lib库3.导出C接口4.c#调用c++动态库开始你的表演...①创建一个空白的解决方案,在解决方案中添加 Visual C++ , Win32 项目空白解决方案的创建:添加Visual C++ , Win32 项目这......_c#调用lib

deepin/ubuntu安装苹方字体-程序员宅基地

文章浏览阅读4.6k次。苹方字体是苹果系统上的黑体,挺好看的。注重颜值的网站都会使用,例如知乎:font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Microsoft YaHei, Source Han Sans SC, Noto Sans CJK SC, W..._ubuntu pingfang

html表单常见操作汇总_html表单的处理程序有那些-程序员宅基地

文章浏览阅读159次。表单表单概述表单标签表单域按钮控件demo表单标签表单标签基本语法结构<form action="处理数据程序的url地址“ method=”get|post“ name="表单名称”></form><!--action,当提交表单时,向何处发送表单中的数据,地址可以是相对地址也可以是绝对地址--><!--method将表单中的数据传送给服务器处理,get方式直接显示在url地址中,数据可以被缓存,且长度有限制;而post方式数据隐藏传输,_html表单的处理程序有那些

PHP设置谷歌验证器(Google Authenticator)实现操作二步验证_php otp 验证器-程序员宅基地

文章浏览阅读1.2k次。使用说明:开启Google的登陆二步验证(即Google Authenticator服务)后用户登陆时需要输入额外由手机客户端生成的一次性密码。实现Google Authenticator功能需要服务器端和客户端的支持。服务器端负责密钥的生成、验证一次性密码是否正确。客户端记录密钥后生成一次性密码。下载谷歌验证类库文件放到项目合适位置(我这边放在项目Vender下面)https://github.com/PHPGangsta/GoogleAuthenticatorPHP代码示例://引入谷_php otp 验证器

【Python】matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距-程序员宅基地

文章浏览阅读4.3k次,点赞5次,收藏11次。matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距

docker — 容器存储_docker 保存容器-程序员宅基地

文章浏览阅读2.2k次。①Storage driver 处理各镜像层及容器层的处理细节,实现了多层数据的堆叠,为用户 提供了多层数据合并后的统一视图②所有 Storage driver 都使用可堆叠图像层和写时复制(CoW)策略③docker info 命令可查看当系统上的 storage driver主要用于测试目的,不建议用于生成环境。_docker 保存容器

随便推点

网络拓扑结构_网络拓扑csdn-程序员宅基地

文章浏览阅读834次,点赞27次,收藏13次。网络拓扑结构是指计算机网络中各组件(如计算机、服务器、打印机、路由器、交换机等设备)及其连接线路在物理布局或逻辑构型上的排列形式。这种布局不仅描述了设备间的实际物理连接方式,也决定了数据在网络中流动的路径和方式。不同的网络拓扑结构影响着网络的性能、可靠性、可扩展性及管理维护的难易程度。_网络拓扑csdn

JS重写Date函数,兼容IOS系统_date.prototype 将所有 ios-程序员宅基地

文章浏览阅读1.8k次,点赞5次,收藏8次。IOS系统Date的坑要创建一个指定时间的new Date对象时,通常的做法是:new Date("2020-09-21 11:11:00")这行代码在 PC 端和安卓端都是正常的,而在 iOS 端则会提示 Invalid Date 无效日期。在IOS年月日中间的横岗许换成斜杠,也就是new Date("2020/09/21 11:11:00")通常为了兼容IOS的这个坑,需要做一些额外的特殊处理,笔者在开发的时候经常会忘了兼容IOS系统。所以就想试着重写Date函数,一劳永逸,避免每次ne_date.prototype 将所有 ios

如何将EXCEL表导入plsql数据库中-程序员宅基地

文章浏览阅读5.3k次。方法一:用PLSQL Developer工具。 1 在PLSQL Developer的sql window里输入select * from test for update; 2 按F8执行 3 打开锁, 再按一下加号. 鼠标点到第一列的列头,使全列成选中状态,然后粘贴,最后commit提交即可。(前提..._excel导入pl/sql

Git常用命令速查手册-程序员宅基地

文章浏览阅读83次。Git常用命令速查手册1、初始化仓库git init2、将文件添加到仓库git add 文件名 # 将工作区的某个文件添加到暂存区 git add -u # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件git add -A # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件...

分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120-程序员宅基地

文章浏览阅读202次。分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120

【C++缺省函数】 空类默认产生的6个类成员函数_空类默认产生哪些类成员函数-程序员宅基地

文章浏览阅读1.8k次。版权声明:转载请注明出处 http://blog.csdn.net/irean_lau。目录(?)[+]1、缺省构造函数。2、缺省拷贝构造函数。3、 缺省析构函数。4、缺省赋值运算符。5、缺省取址运算符。6、 缺省取址运算符 const。[cpp] view plain copy_空类默认产生哪些类成员函数

推荐文章

热门文章

相关标签