blob: 4f4c2b144b6441747415ad08a117c29f154ec8ce [file] [log] [blame] [edit]
// Copyright 2020 The Chromium OS Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "iioservice/daemon/test_fakes.h"
#include <algorithm>
#include <vector>
#include <libmems/common_types.h>
#include <libmems/test_fakes.h>
#include <base/check.h>
#include <base/check_op.h>
namespace iioservice {
namespace fakes {
namespace {
int64_t CalcMovingAverage(const std::vector<int64_t>& values) {
int64_t size = values.size();
int64_t value_total = 0, sum = 0;
for (int64_t i = size - 1; i >= 0; --i) {
sum += values[i];
value_total += sum;
}
return value_total / ((size + 1) * size / 2);
}
} // namespace
// static
FakeSamplesHandler::ScopedFakeSamplesHandler FakeSamplesHandler::Create(
scoped_refptr<base::SingleThreadTaskRunner> ipc_task_runner,
scoped_refptr<base::SingleThreadTaskRunner> task_runner,
libmems::fakes::FakeIioDevice* fake_iio_device) {
ScopedFakeSamplesHandler handler(nullptr, SamplesHandlerDeleter);
double min_freq, max_freq;
if (!fake_iio_device->GetMinMaxFrequency(&min_freq, &max_freq))
return handler;
handler.reset(new FakeSamplesHandler(std::move(ipc_task_runner),
std::move(task_runner), fake_iio_device,
min_freq, max_freq));
return handler;
}
void FakeSamplesHandler::ResumeReading() {
sample_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&FakeSamplesHandler::ResumeReadingOnThread,
weak_factory_.GetWeakPtr()));
}
void FakeSamplesHandler::CheckRequestedFrequency(double max_freq) {
sample_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&FakeSamplesHandler::CheckRequestedFrequencyOnThread,
weak_factory_.GetWeakPtr(), max_freq));
}
FakeSamplesHandler::FakeSamplesHandler(
scoped_refptr<base::SingleThreadTaskRunner> ipc_task_runner,
scoped_refptr<base::SingleThreadTaskRunner> task_runner,
libmems::fakes::FakeIioDevice* fake_iio_device,
double min_freq,
double max_freq)
: SamplesHandler(std::move(ipc_task_runner),
std::move(task_runner),
fake_iio_device,
min_freq,
max_freq),
fake_iio_device_(fake_iio_device) {}
void FakeSamplesHandler::ResumeReadingOnThread() {
CHECK(sample_task_runner_->BelongsToCurrentThread());
fake_iio_device_->ResumeReadingSamples();
}
void FakeSamplesHandler::CheckRequestedFrequencyOnThread(double max_freq) {
CHECK(sample_task_runner_->BelongsToCurrentThread());
CHECK_EQ(max_freq, requested_frequency_);
}
// static
std::unique_ptr<FakeSamplesObserver> FakeSamplesObserver::Create(
libmems::IioDevice* device,
std::multiset<std::pair<int, cros::mojom::ObserverErrorType>> failures,
double frequency,
double frequency2,
double dev_frequency,
double dev_frequency2,
int pause_index) {
std::unique_ptr<FakeSamplesObserver> handler(new FakeSamplesObserver(
device, std::move(failures), frequency, frequency2, dev_frequency,
dev_frequency2, pause_index));
return handler;
}
FakeSamplesObserver::~FakeSamplesObserver() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
}
void FakeSamplesObserver::OnSampleUpdated(
const libmems::IioDevice::IioSample& sample) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
CHECK(failures_.empty() || failures_.begin()->first > sample_index_);
sample_ = sample;
int step = GetStep();
CHECK_GE(step, 1);
CHECK_GT(base::size(libmems::fakes::kFakeAccelSamples),
sample_index_ + step - 1);
if (device_->GetId() == kAccelDeviceId) {
for (int chnIndex = 0;
chnIndex < base::size(libmems::fakes::kFakeAccelChns); ++chnIndex) {
auto it = sample.find(chnIndex);
// channel: accel_y isn't enabled before |pause_index_|
if (sample_index_ + step - 1 < pause_index_ && chnIndex == 1) {
CHECK(it == sample.end());
continue;
}
CHECK(it != sample.end());
// Check timestamp channel
if (strncmp(libmems::fakes::kFakeAccelChns[chnIndex],
libmems::kTimestampAttr,
strlen(libmems::kTimestampAttr)) == 0) {
CHECK_EQ(it->second,
libmems::fakes::kFakeAccelSamples[sample_index_ + step - 1]
[chnIndex]);
continue;
}
// Check other channels
std::vector<int64_t> values;
for (int index = 0; index < step; ++index) {
if (chnIndex == 1 && sample_index_ + index < pause_index_) {
values.push_back(
libmems::fakes::kFakeAccelSamples[pause_index_][chnIndex]);
} else {
values.push_back(libmems::fakes::kFakeAccelSamples[sample_index_ +
index][chnIndex]);
}
}
CHECK_EQ(it->second, CalcMovingAverage(values));
}
} else {
auto channels = device_->GetAllChannels();
for (size_t i = 0; i < channels.size(); ++i) {
auto raw_value = channels[i]->ReadNumberAttribute(libmems::kRawAttr);
if (!raw_value.has_value())
continue;
auto it = sample.find(i);
CHECK(it != sample.end());
CHECK_EQ(raw_value.value(), it->second);
}
}
sample_index_ += step;
}
void FakeSamplesObserver::OnErrorOccurred(cros::mojom::ObserverErrorType type) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
CHECK(!failures_.empty());
CHECK_EQ(failures_.begin()->second, type);
if (type != cros::mojom::ObserverErrorType::FREQUENCY_INVALID)
CHECK_LE(failures_.begin()->first, sample_index_ + GetStep());
else
CHECK_EQ(frequency_, 0.0);
failures_.erase(failures_.begin());
}
mojo::PendingRemote<cros::mojom::SensorDeviceSamplesObserver>
FakeSamplesObserver::GetRemote() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
CHECK(!receiver_.is_bound());
auto remote = receiver_.BindNewPipeAndPassRemote();
receiver_.set_disconnect_handler(base::BindOnce(
&FakeSamplesObserver::OnObserverDisconnect, weak_factory_.GetWeakPtr()));
return remote;
}
bool FakeSamplesObserver::is_bound() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return receiver_.is_bound();
}
bool FakeSamplesObserver::FinishedObserving() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
int step = GetStep();
return (frequency2_ == 0.0 && sample_index_ + step - 1 >= pause_index_) ||
sample_index_ + step - 1 >=
base::size(libmems::fakes::kFakeAccelSamples);
}
bool FakeSamplesObserver::NoRemainingFailures() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return failures_.empty();
}
int FakeSamplesObserver::GetSampleIndex() const {
return sample_index_;
}
const libmems::IioDevice::IioSample& FakeSamplesObserver::GetLatestSample()
const {
return sample_;
}
FakeSamplesObserver::FakeSamplesObserver(
libmems::IioDevice* device,
std::multiset<std::pair<int, cros::mojom::ObserverErrorType>> failures,
double frequency,
double frequency2,
double dev_frequency,
double dev_frequency2,
int pause_index)
: device_(device),
failures_(std::move(failures)),
frequency_(frequency),
frequency2_(frequency2),
dev_frequency_(dev_frequency),
dev_frequency2_(dev_frequency2),
pause_index_(pause_index),
receiver_(this) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
CHECK_GE(frequency_, 0.0);
CHECK_GE(frequency2_, 0.0);
CHECK_GE(dev_frequency_, libmems::kFrequencyEpsilon);
CHECK_GE(dev_frequency2_, libmems::kFrequencyEpsilon);
if (frequency_ == 0.0) {
if (frequency2_ == 0.0)
sample_index_ = base::size(libmems::fakes::kFakeAccelSamples);
else
sample_index_ = pause_index_;
}
}
void FakeSamplesObserver::OnObserverDisconnect() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
receiver_.reset();
}
int FakeSamplesObserver::GetStep() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
CHECK_GE(dev_frequency_, libmems::kFrequencyEpsilon);
int step = base::size(libmems::fakes::kFakeAccelSamples);
if (frequency_ >= libmems::kFrequencyEpsilon)
step = dev_frequency_ / frequency_;
if (sample_index_ + step - 1 < pause_index_)
return step;
if (frequency2_ < libmems::kFrequencyEpsilon)
return base::size(libmems::fakes::kFakeAccelSamples);
int step2 = dev_frequency2_ / frequency2_;
return std::max(pause_index_ - sample_index_ + 1, step2);
}
} // namespace fakes
} // namespace iioservice