#include <gtest/gtest.h>
#include <hardware/hardware.h>
#include <chrono>
#include "worker.h"
using android::Worker;
struct TestWorker : public Worker {
TestWorker()
: Worker("test-worker", HAL_PRIORITY_URGENT_DISPLAY),
value(0),
enabled_(false) {
}
int Init() {
return InitWorker();
}
void Routine() {
Lock();
if (!enabled_) {
int ret = WaitForSignalOrExitLocked();
if (ret == -EINTR) {
Unlock();
return;
}
// should only reached here if it was enabled
if (!enabled_)
printf("Shouldn't reach here while disabled %d %d\n", value, ret);
}
value++;
Unlock();
}
void Control(bool enable) {
bool changed = false;
Lock();
if (enabled_ != enable) {
enabled_ = enable;
changed = true;
}
Unlock();
if (enable && changed)
Signal();
}
int value;
private:
bool enabled_;
};
struct WorkerTest : public testing::Test {
TestWorker worker;
virtual void SetUp() {
worker.Init();
}
void small_delay() {
std::this_thread::sleep_for(std::chrono::milliseconds(20));
}
};
TEST_F(WorkerTest, test_worker) {
// already isInitialized so should succeed
ASSERT_TRUE(worker.initialized());
int val = worker.value;
small_delay();
// value shouldn't change when isInitialized
ASSERT_EQ(val, worker.value);
worker.Control(true);
small_delay();
// while locked, value shouldn't be changing
worker.Lock();
val = worker.value;
small_delay();
ASSERT_EQ(val, worker.value);
worker.Unlock();
small_delay();
// value should be different now
ASSERT_NE(val, worker.value);
worker.Control(false);
worker.Lock();
val = worker.value;
worker.Unlock();
small_delay();
// value should be same
ASSERT_EQ(val, worker.value);
worker.Exit();
ASSERT_FALSE(worker.initialized());
}
TEST_F(WorkerTest, exit_while_running) {
worker.Control(true);
std::this_thread::sleep_for(std::chrono::milliseconds(50));
worker.Exit();
}