blob: acc0f5bb06a662b0b5d6ff797da793676137c083 [file] [log] [blame]
Yifan Hongb7cd45f2017-11-13 18:47:03 -08001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "charger_test"
18#include <android/log.h>
19
20#include <chrono>
21#include <condition_variable>
22#include <fstream>
23#include <iostream>
24#include <mutex>
25#include <streambuf>
26#include <string>
27#include <thread>
28#include <vector>
29
30#include <healthd/healthd.h>
31
32#define LOG_THIS(fmt, ...) \
33 ALOGE(fmt, ##__VA_ARGS__); \
34 printf(fmt "\n", ##__VA_ARGS__);
35
36template <typename T>
37class Atomic {
38 public:
39 Atomic(T&& init) : mValue(std::move(init)) {}
40 void set(T&& newVal) {
41 {
42 std::lock_guard<std::mutex> lock(mMutex);
43 mValue = std::move(newVal);
44 }
45 mChanged.notify_all();
46 }
47 bool waitFor(long ms, const T& expectVal) {
48 std::unique_lock<std::mutex> lock(mMutex);
49 return mChanged.wait_for(lock, std::chrono::milliseconds(ms),
50 [this, &expectVal] { return mValue == expectVal; });
51 }
52 private:
53 std::mutex mMutex;
54 std::condition_variable mChanged;
55 T mValue;
56};
57
58Atomic<bool>& getUpdateNotifier() {
59 static Atomic<bool> val(false);
60 return val;
61}
62
63int energyCounter(int64_t* counter) {
64 *counter = 0xEC12345;
65 return 0;
66}
67
68const char* createFile(const char* path, const char* content) {
69 std::ofstream stream(path);
70 if (!stream.is_open()) {
71 LOG_THIS("Cannot create file %s", path);
72 return NULL;
73 }
74 stream << content << std::endl;
75 stream.close();
76 return path;
77}
78
79std::string openToString(const char* path) {
80 std::ifstream stream(path);
81 if (!stream.is_open()) {
82 LOG_THIS("Cannot open file %s", path);
83 return "";
84 }
85 return std::string(std::istreambuf_iterator<char>(stream), std::istreambuf_iterator<char>());
86}
87
88int expectContains(const std::string& content, const std::vector<std::string>& fields) {
89 int status = 0;
90 for (const auto& field : fields) {
91 auto pos = content.find(field);
92 if (pos == std::string::npos) {
93 LOG_THIS("Cannot find substr '%s'", field.c_str());
94 status = 1;
95 }
96 }
97 return status;
98}
99
100void healthd_board_init(struct healthd_config* config) {
101 config->periodic_chores_interval_fast = 60;
102 config->periodic_chores_interval_slow = 600;
103
104 config->batteryStatusPath = createFile("/data/local/tmp/batteryStatus", "Not charging");
105 config->batteryHealthPath = createFile("/data/local/tmp/batteryHealth", "Unspecified failure");
106 config->batteryPresentPath = createFile("/data/local/tmp/batteryPresent", "1");
107 config->batteryCapacityPath = createFile("/data/local/tmp/batteryCapacity", "47");
108 config->batteryVoltagePath = createFile("/data/local/tmp/batteryVoltage", "45000");
109 config->batteryTemperaturePath = createFile("/data/local/tmp/batteryTemperature", "987");
110 config->batteryTechnologyPath = createFile("/data/local/tmp/batteryTechnology", "NiCd");
111 config->batteryCurrentNowPath = createFile("/data/local/tmp/batteryCurrentNow", "99000");
112 config->batteryCurrentAvgPath = createFile("/data/local/tmp/batteryCurrentAvg", "98000");
113 config->batteryChargeCounterPath = createFile("/data/local/tmp/batteryChargeCounter", "600");
114 config->batteryFullChargePath = createFile("/data/local/tmp/batteryFullCharge", "3515547");
115 config->batteryCycleCountPath = createFile("/data/local/tmp/batteryCycleCount", "77");
116
117 config->energyCounter = energyCounter;
118 config->boot_min_cap = 50;
119 config->screen_on = NULL;
120}
121
122int healthd_board_battery_update(struct android::BatteryProperties*) {
123 getUpdateNotifier().set(true /* updated */);
124
125 // return 0 to log periodic polled battery status to kernel log
126 return 0;
127}
128
129extern int healthd_charger_main(int argc, char** argv);
130
131int main(int argc, char** argv) {
132 const char* dumpFile = "/data/local/tmp/dump.txt";
133
134 std::thread bgThread([=] {
135 healthd_charger_main(argc, argv);
136 });
137
138 // wait for healthd_init to finish
139 if (!getUpdateNotifier().waitFor(1000 /* wait ms */, true /* updated */)) {
140 LOG_THIS("Time out.");
141 exit(1);
142 }
143
144 int fd = creat(dumpFile, S_IRUSR | S_IWUSR);
145 if (fd < 0) {
146 exit(errno);
147 }
148 healthd_dump_battery_state(fd);
149 close(fd);
150
151 std::string content = openToString(dumpFile);
152 int status = expectContains(content, {
153 "status: 4",
154 "health: 6",
155 "present: 1",
156 "level: 47",
157 "voltage: 45",
158 "temp: 987",
159 "current now: 99000",
160 "current avg: 98000",
161 "charge counter: 600",
162 "current now: 99",
163 "cycle count: 77",
164 "Full charge: 3515547"
165 });
166
167 if (status == 0) {
168 LOG_THIS("Test success.");
169 } else {
170 LOG_THIS("Actual dump:\n%s", content.c_str());
171 }
172
173 exit(status); // force bgThread to exit
174}