/* * Copyright 2017 The UBports project * * This program is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License version 3, as published * by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranties of * MERCHANTABILITY, SATISFACTORY QUALITY, 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 . * * Authors: * Marius Gripsgard */ #include "flashlight.h" #include #include #include #include #define QCOM_ENABLE "255" #define QCOM_DISABLE "0" #define SIMPLE_ENABLE "1" #define SIMPLE_DISABLE "0" const size_t qcom_sysfs_size = 7; const char* const qcom_sysfs[] = {"/sys/class/leds/torch-light/brightness", "/sys/class/leds/led:flash_torch/brightness", "/sys/class/leds/flashlight/brightness", "/sys/class/leds/torch-light0/brightness", "/sys/class/leds/torch-light1/brightness", "/sys/class/leds/led:torch_0/brightness", "/sys/class/leds/led:torch_1/brightness"}; const size_t qcom_switch_size = 2; const char* const qcom_switch[] = {"/sys/class/leds/led:switch/brightness", "/sys/class/leds/led:switch_0/brightness"}; const size_t simple_sysfs_size = 2; const char* const simple_sysfs[] = {"/sys/class/flashlight_core/flashlight/flashlight_torch", "/sys/class/leds/white:flash/brightness"}; char* flash_sysfs_path = NULL; char* qcom_switch_path = NULL; enum TorchType torch_type = SIMPLE; gboolean activated = 0; int set_sysfs_path() { for (size_t i = 0; i < qcom_sysfs_size; i++) { if (access(qcom_sysfs[i], F_OK ) != -1){ flash_sysfs_path = (char*)qcom_sysfs[i]; torch_type = QCOM; /* Qualcomm torch; determine switch file (if one is needed) */ for (size_t i = 0; i < qcom_switch_size; i++) { if (access(qcom_switch[i], F_OK ) != -1) qcom_switch_path = (char*)qcom_switch[i]; } return 1; } } for (size_t i = 0; i < simple_sysfs_size; i++) { if (access(simple_sysfs[i], F_OK ) != -1){ flash_sysfs_path = (char*)simple_sysfs[i]; return 1; } } return 0; } gboolean flashlight_activated() { return activated; } int toggle_flashlight_action_qcom() { FILE *fd1 = NULL, *fd2 = NULL; int needs_enable; fd1 = fopen(flash_sysfs_path, "w"); if (fd1 != NULL) { needs_enable = access(qcom_switch_path, F_OK ) != -1; if (needs_enable) fd2 = fopen(qcom_switch_path, "w"); if (activated) if (needs_enable && fd2 != NULL) fprintf(fd2, "0"); else fprintf(fd1, QCOM_DISABLE); else { fprintf(fd1, QCOM_ENABLE); if (needs_enable && fd2 != NULL) fprintf(fd2, "1"); } fclose(fd1); if (fd2 !=NULL) fclose(fd2); return 1; } return 0; } int toggle_flashlight_action_simple() { FILE *fd = NULL; fd = fopen(flash_sysfs_path, "w"); if (fd != NULL) { fprintf(fd, activated ? SIMPLE_DISABLE : SIMPLE_ENABLE); fclose(fd); return 1; } return 0; } void toggle_flashlight_action(GAction *action, GVariant *parameter G_GNUC_UNUSED, gpointer data G_GNUC_UNUSED) { GVariant *state; int toggled; if (!set_sysfs_path()) return; state = g_action_get_state(action); activated = g_variant_get_boolean(state); g_variant_unref(state); if (torch_type == QCOM) toggled = toggle_flashlight_action_qcom(); else toggled = toggle_flashlight_action_simple(); if (toggled) g_action_change_state(action, g_variant_new_boolean(!activated)); } int flashlight_supported() { return set_sysfs_path(); }