nanosleep不适用于不到一秒的值

pat*_*cek 3 c linux sleep

我有一个程序(混合C和Fortran,虽然似乎没有相关性)nanosleep.但是,如果我timespectv_sec值为0,则它​​根本不会睡眠.该tv_nsec值可以是一秒钟的微秒,但它不会睡眠.(如果tv_sec是的话1,它可以暂时休眠一下.)为什么会这样?

为了让事情更加混乱,usleep使用适当的值(即995000usec)按预期睡眠一秒钟.

我在RHEL 5.8和RHEL 6.4框中看到了这个问题.两者都在使用gcc.

这是调用nanosleep的函数:

void msleep(int *milliseconds)
{
    long usec;
    struct timespec sleep;
    usec = (*milliseconds) % 1000;
    sleep.tv_sec = (*milliseconds) / 1000;
    sleep.tv_nsec = 1000*usec;
    nanosleep(&sleep, NULL);
}
Run Code Online (Sandbox Code Playgroud)

显然,我实际上并不需要纳秒精度!

我还测试了一个版本,我检查了返回值; 它总是0(成功),因此rem输出参数(如果中断则剩余时间)从未设置.

alk*_*alk 7

你缺少一个因子1000.

试试这个:

#define _POSIX_C_SOURCE 199309L /* shall be >= 199309L */

#include <time.h>

void msleep(int *milliseconds)  
{
  int ms_remaining = (*milliseconds) % 1000;
  long usec = ms_remaining * 1000;
  struct timespec ts_sleep;

  ts_sleep.tv_sec = (*milliseconds) / 1000;
  ts_sleep.tv_nsec = 1000*usec;
  nanosleep(&ts_sleep, NULL);
}
Run Code Online (Sandbox Code Playgroud)

更紧凑:

#define _POSIX_C_SOURCE 199309L /* shall be >= 199309L */

#include <time.h>

void msleep(int * pmilliseconds)  
{
  struct timespec ts_sleep = 
  {
    *pmilliseconds / 1000,
    (*pmilliseconds % 1000) * 1000000L
  };

  nanosleep(&ts_sleep, NULL);
}
Run Code Online (Sandbox Code Playgroud)

最后是一个完整的实现,包括错误处理和nanosleep()早期中断的情况:

#define _POSIX_C_SOURCE 199309L

#include <time.h>
#include <errno.h>
#include <stdio.h>

int ms_sleep(unsigned int ms)
{
  int result = 0;

  {
    struct timespec ts_remaining =
    { 
      ms / 1000, 
      (ms % 1000) * 1000000L 
    };

    do
    {
      struct timespec ts_sleep = ts_remaining;
      result = nanosleep(&ts_sleep, &ts_remaining);
    } 
    while ((EINTR == errno) && (-1 == result));
  }

  if (-1 == result)
  {
    perror("nanosleep() failed");
  }

  return result;
}
Run Code Online (Sandbox Code Playgroud)

遵循包装器以满足OP的要求:

#include <errno.h>
#include <stdio.h>

int ms_sleep(unsigned int);

void msleep(int * pms)
{
  int result = 0;

  if ((NULL == pms) || (0 > *pms)) /* Check for valid input. */
  {
    errno = EINVAL;
    result = -1;
  }
  else 
  {
    result = ms_sleep(*pms));
  }

  if (-1 == result)
  {
    perror("ms_sleep() failed");
    /* Exit and/or log error here. */
  }
}
Run Code Online (Sandbox Code Playgroud)

更新(参考下面的chux评论):

假设至少C99,这部分上面的代码

  struct timespec ts_sleep = 
  {
    *pmilliseconds / 1000,
    (*pmilliseconds % 1000) * 1000000L
  };
Run Code Online (Sandbox Code Playgroud)

可能最好像这样写

  struct timespec ts_sleep = 
  {
    .tv_sec = *pmilliseconds / 1000,
    .tv_nsec = (*pmilliseconds % 1000) * 1000000L
  };
Run Code Online (Sandbox Code Playgroud)

不依赖于struct timespec会员的顺序.