/*
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#define LOG_TAG "android.hardware.vibrator@1.2-service.crosshatch"
#include <android/hardware/vibrator/1.2/IVibrator.h>
#include <hidl/HidlSupport.h>
#include <hidl/HidlTransportSupport.h>
#include <utils/Errors.h>
#include <utils/StrongPointer.h>
#include "Vibrator.h"
using android::hardware::configureRpcThreadpool;
using android::hardware::joinRpcThreadpool;
using android::hardware::vibrator::V1_2::IVibrator;
using android::hardware::vibrator::V1_2::implementation::Vibrator;
static constexpr char ACTIVATE_PATH[] = "/sys/class/leds/vibrator/activate";
static constexpr char DURATION_PATH[] = "/sys/class/leds/vibrator/duration";
static constexpr char STATE_PATH[] = "/sys/class/leds/vibrator/state";
static constexpr char EFFECT_INDEX_PATH[] = "/sys/class/leds/vibrator/device/cp_trigger_index";
static constexpr char EFFECT_QUEUE_PATH[] = "/sys/class/leds/vibrator/device/cp_trigger_queue";
static constexpr char DIGI_SCALE_PATH[] = "/sys/class/leds/vibrator/device/dig_scale";
// File path to the calibration file
static constexpr char CALIBRATION_FILEPATH[] = "/persist/haptics/cs40l20.cal";
// Kernel ABIs for updating the calibration data
static constexpr char F0_CONFIG[] = "f0_measured";
static constexpr char REDC_CONFIG[] = "redc_measured";
static constexpr char F0_FILEPATH[] = "/sys/class/leds/vibrator/device/f0_stored";
static constexpr char REDC_FILEPATH[] = "/sys/class/leds/vibrator/device/redc_stored";
static std::string trim(const std::string &str, const std::string &whitespace = " \t") {
const auto str_begin = str.find_first_not_of(whitespace);
if (str_begin == std::string::npos) {
return "";
}
const auto str_end = str.find_last_not_of(whitespace);
const auto str_range = str_end - str_begin + 1;
return str.substr(str_begin, str_range);
}
static bool loadCalibrationData() {
std::map<std::string, std::string> config_data;
std::ofstream f0{F0_FILEPATH};
if (!f0) {
ALOGE("Failed to open %s (%d): %s", F0_FILEPATH, errno, strerror(errno));
}
std::ofstream redc{REDC_FILEPATH};
if (!redc) {
ALOGE("Failed to open %s (%d): %s", REDC_FILEPATH, errno, strerror(errno));
}
std::ifstream cal_data{CALIBRATION_FILEPATH};
if (!cal_data) {
ALOGE("Failed to open %s (%d): %s", CALIBRATION_FILEPATH, errno, strerror(errno));
return false;
}
for (std::string line; std::getline(cal_data, line);) {
if (line.empty() || line[0] == '#') {
continue;
}
std::istringstream is_line(line);
std::string key;
if (std::getline(is_line, key, ':')) {
std::string value;
if (std::getline(is_line, value)) {
config_data[trim(key)] = trim(value);
}
}
}
if (config_data.find(F0_CONFIG) != config_data.end()) {
f0 << config_data[F0_CONFIG] << std::endl;
}
if (config_data.find(REDC_CONFIG) != config_data.end()) {
redc << config_data[REDC_CONFIG] << std::endl;
}
return true;
}
// passing out ownership, can't make sp yet
IVibrator *makeVibratorService() {
// ostreams below are required
std::ofstream activate{ACTIVATE_PATH};
if (!activate) {
ALOGE("Failed to open %s (%d): %s", ACTIVATE_PATH, errno, strerror(errno));
}
std::ofstream duration{DURATION_PATH};
if (!duration) {
ALOGE("Failed to open %s (%d): %s", DURATION_PATH, errno, strerror(errno));
}
std::ofstream state{STATE_PATH};
if (!state) {
ALOGE("Failed to open %s (%d): %s", STATE_PATH, errno, strerror(errno));
}
std::ofstream effect{EFFECT_INDEX_PATH};
if (!state) {
ALOGE("Failed to open %s (%d): %s", EFFECT_INDEX_PATH, errno, strerror(errno));
}
std::ofstream queue{EFFECT_QUEUE_PATH};
if (!state) {
ALOGE("Failed to open %s (%d): %s", EFFECT_QUEUE_PATH, errno, strerror(errno));
}
std::ofstream scale{DIGI_SCALE_PATH};
if (!scale) {
ALOGE("Failed to open %s (%d): %s", DIGI_SCALE_PATH, errno, strerror(errno));
}
state << 1 << std::endl;
if (!state) {
ALOGE("Failed to set state (%d): %s", errno, strerror(errno));
}
if (!loadCalibrationData()) {
ALOGW("Failed to load calibration data");
}
return new Vibrator(std::move(activate), std::move(duration), std::move(effect),
std::move(queue), std::move(scale));
}
extern "C" IVibrator *HIDL_FETCH_IVibrator(const char * /*instnace*/) {
return makeVibratorService();
}