可以读取但不能写入 C 语言中的 proc 文件



我正在编写一个 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 程序中使用固定值列表,而不必验证参数,并注意验证代码本身不会成为负担等。

最新更新