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没有直接调用,而是由其他函数实现。