main.c 9.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386
  1. /*
  2. * Licensed under the GNU Public License version 3.
  3. * Read attached LICENSE
  4. * Copyright (C) 2012 - Fredrik Haikarainen
  5. * Fredrik.haikarainen@gmail.com
  6. */
  7. #define _GNU_SOURCE
  8. #include <stdio.h>
  9. #include <stdlib.h>
  10. #include <dirent.h>
  11. #include <string.h>
  12. #include <unistd.h>
  13. #include <limits.h>
  14. typedef enum LBOOL{
  15. TRUE = 0,
  16. FALSE = 1
  17. } LBOOL;
  18. typedef enum LOPTYPE{
  19. SET = 0,
  20. ADD = 1,
  21. SUB = 2
  22. } LOPTYPE;
  23. /* Flags */
  24. LBOOL q; /* Quiet, supresses output */
  25. LBOOL c; /* print unprecise current value in percentage */
  26. LBOOL m; /* read max value*/
  27. LBOOL p; /* precision read current value */
  28. int wbright;
  29. LOPTYPE ot;
  30. LBOOL readint(const char* f, unsigned int *i){
  31. FILE* ff = fopen(f, "r");
  32. if(ff){
  33. fscanf(ff, "%u", i);
  34. fclose(ff);
  35. return TRUE;
  36. }else{
  37. return FALSE;
  38. }
  39. }
  40. LBOOL writeint(const char* f, unsigned int i){
  41. FILE* ff = fopen(f, "w");
  42. if(ff){
  43. fprintf(ff, "%u", i);
  44. fclose(ff);
  45. return TRUE;
  46. }else{
  47. return FALSE;
  48. }
  49. }
  50. LBOOL is_dir(const char* d){
  51. DIR* dd = opendir(d);
  52. if(dd){
  53. closedir(dd);
  54. return TRUE;
  55. }else{
  56. return FALSE;
  57. }
  58. }
  59. LBOOL is_writable(const char* f){
  60. FILE* dd = fopen(f, "w");
  61. if(dd){
  62. fclose(dd);
  63. return TRUE;
  64. }else{
  65. return FALSE;
  66. }
  67. }
  68. typedef struct {
  69. char* name;
  70. unsigned int current_brightness;
  71. unsigned int max_brightness;
  72. char* c_path; /* Controller-path */
  73. char* cb_path; /* Current brightness-path */
  74. char* mb_path; /* Max brightness-path */
  75. char* b_path; /* Brightness-path */
  76. enum LBOOL is_ok;
  77. } controller;
  78. typedef struct {
  79. controller controllers[256];
  80. unsigned int num_controllers;
  81. } fetch_result;
  82. fetch_result fetch_controllers(const char* ctrldir){
  83. fetch_result returner;
  84. struct dirent* ep;
  85. DIR* lcdir = opendir(ctrldir);
  86. returner.num_controllers = 0;
  87. if(!lcdir){
  88. if(q == FALSE)
  89. printf("Error: Could not open directory '%s'!\n", ctrldir);
  90. }else{
  91. while( ( ep = readdir(lcdir) ) ){
  92. char* currctrldir = "";
  93. char* currctrldir_curr = "";
  94. char* currctrldir_max = "";
  95. char* currctrldir_f = "";
  96. if( ep->d_name[0] != '.'){
  97. returner.controllers[returner.num_controllers].name = ep->d_name;
  98. /* Set some default values, in case something fails we dont just get null */
  99. returner.controllers[returner.num_controllers].current_brightness = 0;
  100. returner.controllers[returner.num_controllers].max_brightness = 0;
  101. returner.controllers[returner.num_controllers].is_ok = TRUE;
  102. returner.controllers[returner.num_controllers].c_path = NULL;
  103. returner.controllers[returner.num_controllers].cb_path = NULL;
  104. returner.controllers[returner.num_controllers].b_path = NULL;
  105. returner.controllers[returner.num_controllers].mb_path = NULL;
  106. /* Get path to the current controller dir */
  107. asprintf(&currctrldir, "%s/%s", ctrldir, ep->d_name);
  108. returner.controllers[returner.num_controllers].c_path = currctrldir;
  109. if(is_dir(currctrldir) == FALSE){
  110. if(q == FALSE)
  111. printf("Warning: '%s' is not a directory, check your system.\n", currctrldir);
  112. returner.controllers[returner.num_controllers].is_ok = FALSE;
  113. }
  114. /* Get path to current actual_brightness-file */
  115. asprintf(&currctrldir_curr, "%s/%s", currctrldir, "actual_brightness");
  116. returner.controllers[returner.num_controllers].cb_path = currctrldir_curr;
  117. if( readint(currctrldir_curr, &returner.controllers[returner.num_controllers].current_brightness) == FALSE ){
  118. if(q == FALSE)
  119. printf("Warning: Can't read actual_brightness-file of '%s'. Will ignore this controller.\n", ep->d_name);
  120. returner.controllers[returner.num_controllers].is_ok = FALSE;
  121. }
  122. /* Get path to current max_brightness-file*/
  123. asprintf(&currctrldir_max, "%s/%s", currctrldir, "max_brightness");
  124. returner.controllers[returner.num_controllers].mb_path = currctrldir_max;
  125. if( readint(currctrldir_max, &returner.controllers[returner.num_controllers].max_brightness) == FALSE ){
  126. if(q == FALSE)
  127. printf("Warning: Can't read max_brightness-file of '%s'. Will ignore this controller.\n", ep->d_name);
  128. returner.controllers[returner.num_controllers].is_ok = FALSE;
  129. }
  130. /* Get path to current brightness-file */
  131. asprintf(&currctrldir_f, "%s/%s", currctrldir, "brightness");
  132. returner.controllers[returner.num_controllers].b_path = currctrldir_f;
  133. if( is_writable(currctrldir_f) == FALSE){
  134. if(q == FALSE)
  135. printf("Warning: Controllerfile of '%s' is not writable. Will ignore this controller.\n", ep->d_name);
  136. returner.controllers[returner.num_controllers].is_ok = FALSE;
  137. }
  138. returner.num_controllers++;
  139. }
  140. }
  141. closedir(lcdir);
  142. }
  143. return returner;
  144. }
  145. controller* get_best_controller(fetch_result* res){
  146. unsigned int it;
  147. unsigned int cmax;
  148. LBOOL foundokctrl;
  149. controller* returner;
  150. it = 0;
  151. cmax = 0;
  152. foundokctrl = FALSE;
  153. returner = NULL;
  154. while(it < res->num_controllers){
  155. if(res->controllers[it].is_ok == TRUE){
  156. if(foundokctrl != TRUE){
  157. foundokctrl = TRUE;
  158. returner = &res->controllers[it];
  159. }
  160. if(res->controllers[it].max_brightness > cmax){
  161. cmax = res->controllers[it].max_brightness;
  162. returner = &res->controllers[it];
  163. }
  164. }
  165. it++;
  166. }
  167. return returner;
  168. }
  169. void usage(){
  170. printf("Usage: light [-qcaspm] <value>\n\n\t-q:\t Run quiet, supresses output.\n\t-c:\t Prints the current brightness in percent and exits.(Not precise)\n\t-p:\t Prints the current brightness directly from controller and exits. (Precise)\n\t-m:\t Prints the max brightness directly from controller and exits. \n\t-a:\t Add the value instead of setting it.\n\t-s:\t Subtract the value instead of setting it.\n\n\t<value>\t Brightness wanted in percent.\n\n");
  171. }
  172. int main(int argc, char **argv) {
  173. unsigned int argsit;
  174. fetch_result res;
  175. controller* best_ctrl;
  176. unsigned int citr;
  177. uid_t uid;
  178. unsigned int minlight;
  179. LBOOL given;
  180. unsigned int real_wbright;
  181. unsigned int minlight_nonp;
  182. unsigned int curr_bright;
  183. unsigned int curr_brightp;
  184. /* Get UID */
  185. uid = getuid();
  186. /* Parse arguments */
  187. q=FALSE;
  188. c=FALSE;
  189. m=FALSE;
  190. p=FALSE;
  191. ot=SET;
  192. wbright=0;
  193. argsit = 1;
  194. given=FALSE;
  195. while(argsit < argc){
  196. char* carg = argv[argsit];
  197. if(carg[0] == '-'){
  198. unsigned int cargit = 1;
  199. while(cargit < strlen(carg)){
  200. switch(carg[cargit]){
  201. case 'q':
  202. q = TRUE;
  203. break;
  204. case 'a':
  205. ot = ADD;
  206. break;
  207. case 's':
  208. ot = SUB;
  209. break;
  210. case 'c':
  211. c = TRUE;
  212. break;
  213. case 'p':
  214. p = TRUE;
  215. break;
  216. case 'm':
  217. m = TRUE;
  218. break;
  219. default:
  220. break;
  221. }
  222. cargit++;
  223. }
  224. }else{
  225. wbright = atoi(carg);
  226. given=TRUE;
  227. }
  228. argsit++;
  229. }
  230. if(c == TRUE || m == TRUE || p == TRUE){
  231. q = TRUE;
  232. }
  233. if(given == FALSE && c == FALSE && m == FALSE && p == FALSE){
  234. usage();
  235. return 0;
  236. }
  237. if(q == FALSE)
  238. printf("Light 0.4 - Fredrik Haikarainen\n");
  239. /* Get and check minlight */
  240. if(readint("/etc/light/minlight", &minlight) == FALSE){
  241. minlight = 5;
  242. if(q == FALSE)
  243. printf("Warning: Couldn't read /etc/light/minlight, using 5 as default.\n");
  244. }
  245. /* Fetch controllers */
  246. if(q == FALSE)
  247. printf("Fetching controllers..\n");
  248. res = fetch_controllers("/sys/class/backlight");
  249. citr = 0;
  250. while(citr < res.num_controllers){
  251. controller* currc = &res.controllers[citr];
  252. if(currc->is_ok == TRUE){
  253. if(q == FALSE)
  254. printf("\tFound '%s' (%u/%u)\n", currc->name, currc->current_brightness, currc->max_brightness);
  255. }else{
  256. if(q == FALSE)
  257. printf("\tFound '%s', but ignoring\n", currc->name);
  258. }
  259. citr++;
  260. }
  261. if(q == FALSE)
  262. printf("\n");
  263. /* Get the best controller */
  264. best_ctrl = get_best_controller(&res);
  265. if(best_ctrl == NULL){
  266. if(uid == 0){
  267. if(q == FALSE)
  268. printf("No okay controller found, even though you are root! Check your system.\n");
  269. }else{
  270. if(q == FALSE)
  271. printf("No okay controller found, check your permissions or try to run as root.\n");
  272. }
  273. return 1;
  274. }
  275. if(p == TRUE){
  276. printf("%u\n", best_ctrl->current_brightness);
  277. return 0;
  278. }
  279. if(m == TRUE){
  280. printf("%u\n", best_ctrl->max_brightness);
  281. return 0;
  282. }
  283. if(q == FALSE)
  284. printf("Using controller '%s' ..\n", best_ctrl->name);
  285. if(wbright < 0){ wbright = 0;}
  286. if(wbright > 100){wbright=100;}
  287. curr_bright = best_ctrl->current_brightness;
  288. curr_brightp = (float)((float)curr_bright / (float)best_ctrl->max_brightness) * 100;
  289. if(c == TRUE){
  290. printf("%u\n", curr_brightp);
  291. return 0;
  292. }
  293. switch(ot){
  294. case SET:
  295. real_wbright = best_ctrl->max_brightness * ( (float)wbright / 100 );
  296. break;
  297. case ADD:
  298. real_wbright = ( best_ctrl->max_brightness * ( (float)( curr_brightp + wbright +1) / 100 ));
  299. break;
  300. case SUB:
  301. real_wbright = ( best_ctrl->max_brightness * ( (float)( curr_brightp - wbright + 1) / 100 ));
  302. break;
  303. default:
  304. break;
  305. }
  306. minlight_nonp = best_ctrl->max_brightness * ( (float)minlight / 100);
  307. /* FIXME
  308. Line below makes sure the value never wraps around and gets higher. Puts a (high) limit on max brightness.
  309. Not sure if safe for portabilities sake.
  310. */
  311. if(real_wbright > ((UINT_MAX/2) - best_ctrl->max_brightness)){ real_wbright = minlight_nonp; }
  312. if(real_wbright > best_ctrl->max_brightness){real_wbright = best_ctrl->max_brightness;}
  313. if(real_wbright < minlight_nonp){real_wbright = minlight_nonp;}
  314. if(q == FALSE)
  315. printf("Writing %u to file '%s'..\n", real_wbright, best_ctrl->b_path);
  316. if(writeint(best_ctrl->b_path, real_wbright) == FALSE){
  317. if(q == FALSE){
  318. printf("Error: Could not write to file %s, check your permissions!\n", best_ctrl->b_path);
  319. }
  320. return 1;
  321. }
  322. return 0;
  323. }