/* Copyright (C) 2023 Aiden Gall
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see . */
#include
#include
#include
#include
#include
#include
#include
#include
static const char VIDEO_GROUP[] = "video";
static const char BRIGHTNESS[] =
"/sys/class/backlight/intel_backlight/brightness";
static const char MAX_BRIGHTNESS[] =
"/sys/class/backlight/intel_backlight/max_brightness";
static void die(const char *fmt, ...);
static long strtonum(const char *arg, long minval, long maxval);
static int check_group(void);
static long read_sys_file_num(const char *path);
static void write_sys_file_num(const char *path, long val);
static void
die(const char *const fmt, ...)
{
va_list ap;
va_start(ap, fmt);
vfprintf(stderr, fmt, ap);
va_end(ap);
exit(EXIT_FAILURE);
}
static long
strtonum(const char *const arg, const long minval, const long maxval)
{
long ret;
char *endptr;
if (minval > maxval)
die("strtonum: Invalid range\n");
if (!arg)
die("strtonum: Missing argument\n");
errno = 0;
ret = strtol(arg, &endptr, 10);
if (errno)
die("strtol: %s\n", strerror(errno));
if (arg == endptr || *endptr)
die("strtol: Invalid integer\n");
if (ret < minval || ret > maxval)
die("strtonum: Argument out of range\n");
return ret;
}
static int
check_group(void)
{
uid_t uid, euid;
gid_t video_gid;
gid_t *glist;
int glist_size;
struct group *grp;
int i;
uid = getuid();
euid = geteuid();
if (uid == 0)
return 1;
if (euid != 0)
die("check_group: setuid is not set\n");
errno = 0;
grp = getgrnam(VIDEO_GROUP);
if (errno)
die("getgrnam: %s\n", strerror(errno));
if (!grp)
die("getgrnam: Group '%s' not found\n", VIDEO_GROUP);
video_gid = grp->gr_gid;
errno = 0;
glist_size = getgroups(0, NULL);
if (errno)
die("getgroups: %s\n", strerror(errno));
glist = calloc(glist_size, sizeof(*glist));
if (!glist)
die("check_group: Out of memory\n");
errno = 0;
if (getgroups(glist_size, glist) != glist_size)
die("getgroups: Inconsistent number of groups\n");
if (errno)
die("getgroups: %s\n", strerror(errno));
for (i = 0; i < glist_size; i++) {
if (glist[i] == video_gid) {
free(glist);
return 1;
}
}
free(glist);
return 0;
}
static long
read_sys_file_num(const char *const path)
{
FILE *fd;
char buf[64];
errno = 0;
fd = fopen(path, "r");
if (errno)
die("fopen: %s\n", strerror(errno));
if (!fd)
die("fopen: Returned NULL\n");
if (!fgets(buf, sizeof(buf), fd))
die("fgets: Returned NULL\n");
fclose(fd);
buf[strcspn(buf, "\n")] = '\0';
return strtonum(buf, 0, LONG_MAX/2);
}
static void
write_sys_file_num(const char *const path, const long val)
{
FILE *fd;
errno = 0;
fd = fopen(path, "w");
if (errno)
die("fopen: %s\n", strerror(errno));
if (!fd)
die("fopen: Returned NULL\n");
if (!fprintf(fd, "%ld\n", val))
die("fprintf: Failed to write\n");
fclose(fd);
}
int
main(int argc, char *argv[])
{
long max_brightness;
long val;
if (!check_group())
die("%s: User not in '%s' group\n", argv[0], VIDEO_GROUP);
if (argc != 3)
die("%s: Invalid number of arguments\n", argv[0]);
val = strtonum(argv[2], 0, LONG_MAX/2);
if (!argv[1][0] || argv[1][1])
goto invalid_mode;
switch (argv[1][0]) {
case '-':
val = -val;
/* FALLTHROUGH */
case '+':
val += read_sys_file_num(BRIGHTNESS);
/* FALLTHROUGH */
case '=':
max_brightness = read_sys_file_num(MAX_BRIGHTNESS);
if (val < 0)
val = 0;
if (val > max_brightness)
val = max_brightness;
write_sys_file_num(BRIGHTNESS, val);
break;
default:
invalid_mode:
die("%s: Invalid mode\n", argv[0]);
}
return 0;
}