182 lines
5.7 KiB
C++
182 lines
5.7 KiB
C++
/*
|
|
* 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 "charger_test"
|
|
#include <android/log.h>
|
|
|
|
#include <chrono>
|
|
#include <condition_variable>
|
|
#include <fstream>
|
|
#include <iostream>
|
|
#include <mutex>
|
|
#include <streambuf>
|
|
#include <string>
|
|
#include <thread>
|
|
#include <vector>
|
|
|
|
#include <health2/Health.h>
|
|
|
|
#define LOG_THIS(fmt, ...) \
|
|
ALOGE(fmt, ##__VA_ARGS__); \
|
|
printf(fmt "\n", ##__VA_ARGS__);
|
|
|
|
template <typename T>
|
|
class Atomic {
|
|
public:
|
|
Atomic(T&& init) : mValue(std::move(init)) {}
|
|
void set(T&& newVal) {
|
|
{
|
|
std::lock_guard<std::mutex> lock(mMutex);
|
|
mValue = std::move(newVal);
|
|
}
|
|
mChanged.notify_all();
|
|
}
|
|
bool waitFor(long ms, const T& expectVal) {
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
|
return mChanged.wait_for(lock, std::chrono::milliseconds(ms),
|
|
[this, &expectVal] { return mValue == expectVal; });
|
|
}
|
|
private:
|
|
std::mutex mMutex;
|
|
std::condition_variable mChanged;
|
|
T mValue;
|
|
};
|
|
|
|
Atomic<bool>& getUpdateNotifier() {
|
|
static Atomic<bool> val(false);
|
|
return val;
|
|
}
|
|
|
|
int energyCounter(int64_t* counter) {
|
|
*counter = 0xEC12345;
|
|
return 0;
|
|
}
|
|
|
|
const char* createFile(const char* path, const char* content) {
|
|
std::ofstream stream(path);
|
|
if (!stream.is_open()) {
|
|
LOG_THIS("Cannot create file %s", path);
|
|
return NULL;
|
|
}
|
|
stream << content << std::endl;
|
|
stream.close();
|
|
return path;
|
|
}
|
|
|
|
std::string openToString(const char* path) {
|
|
std::ifstream stream(path);
|
|
if (!stream.is_open()) {
|
|
LOG_THIS("Cannot open file %s", path);
|
|
return "";
|
|
}
|
|
return std::string(std::istreambuf_iterator<char>(stream), std::istreambuf_iterator<char>());
|
|
}
|
|
|
|
int expectContains(const std::string& content, const std::vector<std::string>& fields) {
|
|
int status = 0;
|
|
for (const auto& field : fields) {
|
|
auto pos = content.find(field);
|
|
if (pos == std::string::npos) {
|
|
LOG_THIS("Cannot find substr '%s'", field.c_str());
|
|
status = 1;
|
|
}
|
|
}
|
|
return status;
|
|
}
|
|
|
|
::android::hardware::hidl_handle createHidlHandle(const char* filepath) {
|
|
int fd = creat(filepath, S_IRUSR | S_IWUSR);
|
|
if (fd < 0) return {};
|
|
native_handle_t* nativeHandle = native_handle_create(1, 0);
|
|
nativeHandle->data[0] = fd;
|
|
::android::hardware::hidl_handle handle;
|
|
handle.setTo(nativeHandle, true /* shouldOwn */);
|
|
return handle;
|
|
}
|
|
|
|
void healthd_board_init(struct healthd_config* config) {
|
|
config->periodic_chores_interval_fast = 60;
|
|
config->periodic_chores_interval_slow = 600;
|
|
|
|
config->batteryStatusPath = createFile("/data/local/tmp/batteryStatus", "Not charging");
|
|
config->batteryHealthPath = createFile("/data/local/tmp/batteryHealth", "Unspecified failure");
|
|
config->batteryPresentPath = createFile("/data/local/tmp/batteryPresent", "1");
|
|
config->batteryCapacityPath = createFile("/data/local/tmp/batteryCapacity", "47");
|
|
config->batteryVoltagePath = createFile("/data/local/tmp/batteryVoltage", "45000");
|
|
config->batteryTemperaturePath = createFile("/data/local/tmp/batteryTemperature", "987");
|
|
config->batteryTechnologyPath = createFile("/data/local/tmp/batteryTechnology", "NiCd");
|
|
config->batteryCurrentNowPath = createFile("/data/local/tmp/batteryCurrentNow", "99000");
|
|
config->batteryCurrentAvgPath = createFile("/data/local/tmp/batteryCurrentAvg", "98000");
|
|
config->batteryChargeCounterPath = createFile("/data/local/tmp/batteryChargeCounter", "600");
|
|
config->batteryFullChargePath = createFile("/data/local/tmp/batteryFullCharge", "3515547");
|
|
config->batteryCycleCountPath = createFile("/data/local/tmp/batteryCycleCount", "77");
|
|
|
|
config->energyCounter = energyCounter;
|
|
config->boot_min_cap = 50;
|
|
config->screen_on = NULL;
|
|
}
|
|
|
|
int healthd_board_battery_update(struct android::BatteryProperties*) {
|
|
getUpdateNotifier().set(true /* updated */);
|
|
|
|
// return 0 to log periodic polled battery status to kernel log
|
|
return 0;
|
|
}
|
|
|
|
extern int healthd_charger_main(int argc, char** argv);
|
|
|
|
int main(int argc, char** argv) {
|
|
using android::hardware::health::V2_0::implementation::Health;
|
|
|
|
const char* dumpFile = "/data/local/tmp/dump.txt";
|
|
|
|
std::thread bgThread([=] {
|
|
healthd_charger_main(argc, argv);
|
|
});
|
|
|
|
// wait for healthd_init to finish
|
|
if (!getUpdateNotifier().waitFor(1000 /* wait ms */, true /* updated */)) {
|
|
LOG_THIS("Time out.");
|
|
exit(1);
|
|
}
|
|
|
|
Health::getImplementation()->debug(createHidlHandle(dumpFile), {} /* options */);
|
|
|
|
std::string content = openToString(dumpFile);
|
|
int status = expectContains(content, {
|
|
"status: 4",
|
|
"health: 6",
|
|
"present: 1",
|
|
"level: 47",
|
|
"voltage: 45",
|
|
"temp: 987",
|
|
"current now: 99000",
|
|
"current avg: 98000",
|
|
"charge counter: 600",
|
|
"current now: 99",
|
|
"cycle count: 77",
|
|
"Full charge: 3515547"
|
|
});
|
|
|
|
if (status == 0) {
|
|
LOG_THIS("Test success.");
|
|
} else {
|
|
LOG_THIS("Actual dump:\n%s", content.c_str());
|
|
}
|
|
|
|
exit(status); // force bgThread to exit
|
|
}
|