c - time() 和 gettimeofday() 之间的区别以及为什么会导致段错误

标签 c linux time segmentation-fault

我正在尝试测量系统调用的时间量,我尝试在该程序中使用 time(0)gettimeofday(),但是每当我使用 gettimeofday() 来分割错误。我想我可以只使用 time(0) 但我想知道为什么会这样。我知道你们可以看看它,看看问题所在。请不要对我大喊大叫!

我想得到时间但不想把它保存在任何地方。

我已经尝试了所有我能想到的代码组合,但我在这里粘贴了最简单的版本。我是 C 和 Linux 的新手。我查看了 .stackdump 文件,但它对我来说毫无意义。

GetRDTSC 在 util.h 中,它执行 rdtsc(),正如人们所期望的那样。现在它设置为 10 次迭代,但稍后循环将运行 1000 次,没有 printf

#include <stdio.h>
#include <time.h>
#include "util.h"

int main() {

    int i;
    uint64_t cycles[10];

    for (i = 0; i < 10; ++i) {

         // get initial cycles
         uint64_t init = GetRDTSC();

         gettimeofday(); // <== time(0) will work here without a seg fault.

         // get cycles after
         uint64_t after = GetRDTSC();   

         // save cycles for each operation in an array
         cycles[i] = after - init;

         printf("%i\n", (int)(cycles[i]));
    }  
}

最佳答案

简短版

gettimeofday() 需要一个指向 struct timeval 的指针来填充时间数据。

所以,例如,你会做这样的事情:

#include <sys/time.h>
#include <stdio.h>
int main() {  
    struct timeval tv;
    gettimeofday(&tv, NULL); // timezone should be NULL
    printf("%d seconds\n", tv.tv_secs);
    return 0;
}

长版

真正的 问题是 gcc 会自动在您的系统上包含 vdso,它包含系统调用 gettimeofday 的符号。考虑这个程序(整个文件):

int main() {
  gettimeofday();
  return 0;
}

默认情况下,gcc 会在没有警告的情况下编译这个。如果您检查它所链接的符号,您会看到:

ternus@event-horizon ~> gcc -o foo foo.c
ternus@event-horizon ~> ldd foo
        linux-vdso.so.1 =>  (0x00007ffff33fe000)
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f56a5255000)
        /lib64/ld-linux-x86-64.so.2 (0x00007f56a562b000)

您只是碰巧使用了一个具有已定义符号的函数,但没有原型(prototype),就无法判断它应该有多少个命令行参数。

如果你用-Wall编译它,你会看到:

ternus@event-horizon ~> gcc -Wall -o foo foo.c
foo.c: In function ‘main’:
foo.c:2:3: warning: implicit declaration of function ‘gettimeofday’ [-Wimplicit-function-declaration]

当然,当您尝试运行它时,它会出现段错误。有趣的是,它会在内核空间出现段错误(这是在 MacOS 上):

cternus@astarael ~/foo> gcc -o foo -g foo.c
cternus@astarael ~/foo> gdb foo
GNU gdb 6.3.50-20050815 (Apple version gdb-1822) (Sun Aug  5 03:00:42 UTC 2012)
[etc]

(gdb) run
Starting program: /Users/cternus/foo/foo
Reading symbols for shared libraries +.............................. done

Program received signal EXC_BAD_ACCESS, Could not access memory.
Reason: KERN_INVALID_ADDRESS at address: 0x0000000000000001
0x00007fff87eeab73 in __commpage_gettimeofday ()

现在考虑这个程序(同样,没有头文件):

typedef struct {
  long tv_sec;
  long tv_usec;
} timeval;

int main() {
  timeval tv;
  gettimeofday(&tv, 0);
  return 0;
}

这将编译和运行得很好——没有段错误。您已经为它提供了它期望的内存位置,即使仍然没有提供 gettimeofday 原型(prototype)。

更多信息:

Can anyone understand how gettimeofday works?

Is there a faster equivalent of gettimeofday?

The POSIX gettimeofday specification

关于c - time() 和 gettimeofday() 之间的区别以及为什么会导致段错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19528665/

相关文章:

c - 键盘 Hook : change the key code

c - 使用 malloc 创建多维数组时的奇怪行为

Eclipse 中的 C 代码返回对符号 cvSaveImage 的 undefined reference

linux - 需要命令在 perl expect 中并行运行

linux - 在 Linux 上运行 Java FX 2.0 应用程序时出现黑屏

linux - Linux(Ubuntu 服务器)上带有 Apache 的 Lua 错误?

java.util.Date 在 JDK 5 和 JDK 6 中返回不同的日期

c - 即使使用 `noexecstack`,堆栈也是可执行的

time - 为什么 time.Parse 会错误地解析时间?

c++ - 时区之间的时间转换