glibc源码分析之getpriority,setpriority,nice

glibc提供了getpriority,setpriority,nice函数用于获取或者设置进程的优先级。

getpriority函数定义在sysdeps/unix/sysv/linux/getpriority.c文件中。

#define PZERO 20

int
__getpriority (enum __priority_which which, id_t who)
{
  int res;

  res = INLINE_SYSCALL (getpriority, 2, (int) which, who);
  if (res >= 0)
    res = PZERO - res;
  return res;
}
libc_hidden_def (__getpriority)
weak_alias (__getpriority, getpriority)

__getpriority 函数首先调用了getpriority系统调用获取返回值。如果返回值大于0,则系统调用调用成功,将PZERO-返回值的值作为函数返回值。这样一来,__getpriority 函数无法通过返回值是否为-1来判断函数执行是否成功。因为PZERO-返回值可能为-1。

setpriority函数由脚本生成。

#define SYSCALL_NAME setpriority
#define SYSCALL_NARGS 3
#define SYSCALL_SYMBOL __setpriority
#define SYSCALL_CANCELLABLE 0
#define SYSCALL_NOERRNO 0
#define SYSCALL_ERRVAL 0
#include <syscall-template.S>
weak_alias (__setpriority, setpriority)
hidden_weak (setpriority)

nice函数定义在sysdeps/unix/sysv/linux/nice.c文件中。

#include <sysdeps/posix/nice.c>

sysdeps/posix/nice.c文件内容为

int
nice (int incr)
{
  int save;
  int prio;
  int result;

  /* -1 is a valid priority, so we use errno to check for an error.  */
  save = errno;
  __set_errno (0);
  prio = __getpriority (PRIO_PROCESS, 0);
  if (prio == -1)
    {
      if (errno != 0)
    return -1;
    }

  result = __setpriority (PRIO_PROCESS, 0, prio + incr);
  if (result == -1)
    {
      if (errno == EACCES)
    __set_errno (EPERM);
      return -1;
    }

  __set_errno (save);
  return __getpriority (PRIO_PROCESS, 0);
}

linux中存在nice系统调用,但是glibc没有直接调用,而是由其他函数实现。

发布了185 篇原创文章 · 获赞 18 · 访问量 16万+

猜你喜欢

转载自blog.csdn.net/pk_20140716/article/details/77510059