`
helloyesyes
  • 浏览: 1327231 次
  • 性别: Icon_minigender_2
  • 来自: 武汉
文章分类
社区版块
存档分类
最新评论

RDTSC指令实现纳秒级计时器

阅读更多

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 <zhangjinw@gmail.com> * 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 中就可找到。
分享到:
评论

相关推荐

    RDTSC指令的应用之检测CPU主频

    RDTSC(Read Time-Stamp Counter)指令是Intel x86架构中用于获取处理器时钟周期的一个重要工具,它在检测CPU主频的过程中扮演着关键角色。本文将深入探讨RDTSC指令的应用以及如何利用它来检测CPU的主频。 RDTSC...

    rdtsc_intel_instruction

    rdtsc指令是x86架构中的一条指令,全称是read time-stamp counter,中文翻译为读取时间戳计数器。该指令用于读取处理器的时钟计数器,这个计数器是一个64位的寄存器,其值随着处理器的每个时钟周期而增加。这个指令...

    使用CPU时间戳进行高精度计时

    - **实现细节**:RDTSC指令可以直接获取到当前CPU的时钟周期数,从而实现精确的计时。这对于那些需要高频率、高精度时间测量的应用来说是非常有价值的。 - **注意事项**:虽然RDTSC指令能够提供极高的精度,但它的...

    CPU 的精确计时器

    - `cputicker.cpp` 和 `cputicker.h`:可能包含实现CPU计时器功能的类或函数,包括RDTSC或其他计时技术的封装。 - `Ticktest.cpp`:可能是用于测试计时器精度或性能的程序。 - `StdAfx.cpp` 和 `StdAfx.h`:通常在...

    Windows + Intel CPU的高精度计时器

    获得CPU时钟晶振的计时器。该软件包用VC6.0开发,以cdecl的方式提供C编译器生成的dll。源码分为4部分:WinTimerDll,封装了RDTSC汇编实现过程的DLL源码,提供了3个API函数;WinTimerTest,加载WinTimerDll生成的DLL...

    采用汇编和C两种语言,分别求取CPU当前的速度(RDTSC)

    1. 开始一个新的汇编程序,设置好指令集(例如,使用NASM或GAS汇编器)。 2. 使用RDTSC指令读取TSC值,将结果分别存储在EAX和EDX/RAX和RDX中。 3. 可以选择将这两个寄存器的值保存到内存中或进一步处理,如计算两个...

    模拟非法指令的性能分析

    #### 分析方法:利用RDTSC指令进行精确计时 为了准确衡量模拟非法指令的性能损失,我们采用了一种精密的时间测量技术,即利用Intel处理器的RDTSC指令。RDTSC(读取时间戳寄存器)提供了一个自处理器上电以来经历的...

    zperf:纳秒级精度(0.3ns)和纳秒级统计分析(10ns一下)的工具可常态化嵌入常见C ++项目进行函数调用分析内存增长分析等

    zperf记录特色皮秒级计时精度,跨Windows / mac / linux:10纳秒以下统计分析的总性能损耗: rdtsc intel平台下计数有效精度小于1纳秒计数消耗偏差7 10纳秒统计消耗总偏差10 13ns以下(默认使用,可根据实际环境选择...

    kvm-rdtsc-hack:内核模块可通过RDTSC计时器规避KVM的检测

    KVM RDTSC计时器稳定器该项目旨在稳定并最小化在KVM虚拟机中运行的程序中2个RDTSC调用和vmexit(特别是cpuid)的感知时间差。 您可能需要配置constant_tsc_offset值,默认情况下为1000。 在AMD Ryzen平台上,〜1600...

    TrapRDTSC:用于 Mac OS X 的 rdtsc 的内核级仿真

    TrapRDTSC 在 Mac OS X/x86-64 上捕获并模拟rdtsc (读取时间戳计数器)和rdtscp指令。 这在使用 rdtsc 检测外部仪器的逆向工程软件时主要有用。 除了在使用, rdtsc Mac OS X 的 iCloud/Spotlight/iMessage/apsd...

    超高精度计时器-易语言

    通常,计时器的精度受到硬件时钟频率的限制,而现代计算机系统通常使用硬件定时器,如Intel的RDTSC(循环计数器)指令,来获取微秒甚至纳秒级的时间戳。在易语言中,通过调用底层API或者封装相应的系统调用来实现对...

    用VC 编程实现对CPU主频的检测.pdf

    RDTSC(Read Time Stamp Counter)指令是Intel 80386微处理器提供的一种功能,用于读取CPU的时间戳计数器(Timestamp Counter)。这个计数器在每个时钟周期内都会递增,并在处理器复位时清零。RDTSC指令返回的64位...

    C++中精确计时的方法

    接下来,我们来看几个具体的例子,演示如何在实际应用中使用RDTSC指令来实现精确计时。 ##### 使用RDTSC的示例 ```cpp #include class KTimer { public: void Start() { startTime = GetCycleCount(); } ...

    Linux下代码运行时间的高精度测量

    ### Linux下代码运行时间的高精度测量 ...纳秒级测量方法利用`RDTSC`指令,适用于需要极高精度测量的应用场景,如实时系统中的性能优化。这两种方法各有优缺点,根据具体的应用需求选择合适的方法至关重要。

    miaobiao.rar_ miaobiao_miaobiao_汇编语言秒表_电子秒表_秒表

    在汇编语言中,我们首先要设置PIT或者使用RDTSC指令初始化计时器。然后,通过捕获定时器的中断,更新秒表的显示值。这个过程可能涉及中断向量表、中断服务程序以及计时器的配置寄存器操作。 电子秒表的暂停与继续...

    IA 32/64 Benchmark Code Execution Paper

    本文档是一篇关于如何在Linux环境下使用RDTSC指令精确测量执行特定C代码时所花费的时钟周期数的白皮书。文档由Gabriele Paoloni撰写,他是Linux内核/嵌入式软件工程师,在Intel公司任职。文档的发布日期是2010年9月...

    编程检测CPU主频.pdf

    RDTSC指令是Intel 80386微处理器提供的一种特殊指令,它用于读取CPU的时间戳计数器,该计数器在每个时钟周期都会递增。RDTSC指令返回的结果分为两个部分,高32位存储在EDX寄存器,低32位存储在EAX寄存器。然而,为了...

    Windows下 VC 高精度计时和高频事件的产生

    - **RDTSC指令**:在x86/x64架构上,可以通过汇编语言直接调用处理器的Time Stamp Counter(TSC),获取CPU自启动以来的周期数。不过,由于不同CPU速度可能不同,跨CPU的比较需要谨慎处理。 2. **高频事件的产生**...

    rdtsc:我的C ++包装器,辅助函数,摘要和实验

    `rdtsc`(Read Time-Stamp Counter)指令是Intel x86架构处理器提供的一种硬件计时机制,它允许程序员访问处理器内部的时钟周期计数器,从而获取非常精确的执行时间。这个`rdtsc`包装器项目是针对C++编程语言设计的...

    最最精确计时

    在操作系统层面,常见的计时器有系统级定时器和用户级定时器。系统级定时器通常由硬件中断驱动,而用户级定时器则更多依赖于操作系统提供的API。 CPU单位时间频率,也就是CPU的时钟周期,是指CPU执行一个基本操作所...

Global site tag (gtag.js) - Google Analytics