我正在编写一个 C 程序来更改屏幕亮度,因为 xbacklight 在我的情况下不起作用。解决方案应该是本机C(没有system((函数(,因为该程序应该可以通过setuid作为普通用户执行。调用外部 shell 命令或脚本会导致内核忽略此位。
读取控制亮度的 proc 文件工作正常,但使用 C 写入该文件不会产生任何结果,即使我以 root 身份运行程序也是如此。fprintf 调用返回 -130,表示错误。作为健全性检查,我包含一个使用 system(( 作为注释的工作解决方案。
[...]
const char* brightness = "/sys/class/backlight/intel_backlight/brightness";
f = fopen(brightness, (!strncmp(argv[1], "get", 3)) ? "r" : "rw");
[...]
int get_brightness() {
int buff;
fscanf(f, "%d", &buff);
return buff;
}
int set(int i) {
i = MAX(0, MIN(255, i));
fprintf(f, "%d", i);
printf("%d", i);
//char *cmd = (char*) malloc(59 *sizeof(char));
//snprintf(cmd, 59, "echo %d > %s", i, brightness);
//system(cmd);
//free(cmd);
}
f = fopen(brightness, (!strncmp(argv[1], "get", 3)) ? "r" : "rw");
"rw"
不是 fopen(3)
的mode
参数的有效参数。要使用 fopen(3)
以读/写模式打开文件,您应该使用 "r+"
。
使用 "rw"
是未定义的行为 - 在 Linux/glibc 中,它将被视为单个"r"
,文件将以只读模式打开,并且 printf
-> write
将失败。
通常,对如此小的文件使用缓冲 I/O 并不是一个非常明智的主意。如果需要将格式化数据写入文件描述符,则应查看dprintf(3)
。
此外,我只会在 setuid 程序中使用固定值列表,而不必验证参数,并注意验证代码本身不会成为负担等。