/*
* Copyright (C) 2016 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.
*/
#include <general_test/timer_set_test.h>
#include <cinttypes>
#include <cstddef>
#include <new>
#include <shared/send_message.h>
#include <shared/time_util.h>
#include <chre.h>
using nanoapp_testing::kOneMillisecondInNanoseconds;
using nanoapp_testing::kOneSecondInNanoseconds;
using nanoapp_testing::sendFatalFailureToHost;
using nanoapp_testing::sendInternalFailureToHost;
using nanoapp_testing::sendSuccessToHost;
/*
* We have various "stages" for different timer setups we want to test.
* To speed up the test, we run all our stages simultaneously. That
* requires having 6 timers available, but with a 32 timer minimum
* and a presumption that tests aren't going to be run alongside a lot
* of other nanoapps, this should be fine.
*
* See initStages() for the description of each stage. Since these all
* happen in parallel, we leave it to each stage to mark itself has having
* succeeded, and have markSuccess() tell the Host when all stages have
* reported in.
*
* Note that we intentionally place the one-shot timers first, to give
* us more time to notice them (incorrectly) firing multiple times.
*/
static uint64_t kShortDuration = 10 * kOneMillisecondInNanoseconds;
static uint64_t kLongDuration = kOneSecondInNanoseconds;
namespace general_test {
TimerSetTest::Stage::Stage(uint32_t stage, uint64_t duration,
const void *cookie, bool oneShot)
: mSetTime(0), mDuration(duration), mStage(stage), mEventCount(0),
mCookie(cookie), mOneShot(oneShot) {}
void TimerSetTest::Stage::start() {
mSetTime = chreGetTime();
mTimerHandle = chreTimerSet(mDuration, mCookie, mOneShot);
if (mTimerHandle == CHRE_TIMER_INVALID) {
sendFatalFailureToHost("Unable to set timer ", &mStage);
}
if (mSetTime == 0) {
sendFatalFailureToHost("chreGetTime() gave 0");
}
}
void TimerSetTest::Stage::processEvent(uint64_t timestamp, TimerSetTest *test) {
if (mSetTime == 0) {
sendInternalFailureToHost("Didn't initialize mSetTime");
}
mEventCount++;
uint64_t expectedTime = mSetTime + (mEventCount * mDuration);
if (timestamp < expectedTime) {
sendFatalFailureToHost("Timer triggered too soon ", &mStage);
}
// TODO(b/32179037): Make this check much stricter.
if (timestamp > (expectedTime + kOneSecondInNanoseconds)) {
sendFatalFailureToHost("Timer triggered over a second late ", &mStage);
}
if (mOneShot) {
if (mEventCount > 1) {
sendFatalFailureToHost("One shot timer called multiple times ",
&mStage);
} else {
test->markSuccess(mStage);
}
} else if (mEventCount == 3) {
// We mark recurring timers as successful on their third firing, if we
// can cancel it.
if (chreTimerCancel(mTimerHandle)) {
test->markSuccess(mStage);
} else {
sendFatalFailureToHost("Could not cancel recurring timer", &mStage);
}
}
}
void TimerSetTest::initStages() {
// To avoid fragmentation, we do one large allocation, and use
// placement new to initialize it.
mStages = static_cast<Stage*>(chreHeapAlloc(sizeof(*mStages) *
kStageCount));
if (mStages == nullptr) {
sendFatalFailureToHost("Insufficient heap");
}
#define COOKIE(num) reinterpret_cast<const void*>(num)
// Stage 0: Test NULL cookie
new(&mStages[0]) Stage(0, kShortDuration, nullptr, true);
// Stage 1: Test (void*)-1 cookie
new(&mStages[1]) Stage(1, kShortDuration, COOKIE(-1), true);
// Stage 2: Test one shot with short duration
new(&mStages[2]) Stage(2, kShortDuration, COOKIE(2), true);
// Stage 3: Test one shot with long duration
new(&mStages[3]) Stage(3, kLongDuration, COOKIE(3), true);
// Stage 4: Test recurring with long duration
new(&mStages[4]) Stage(4, kLongDuration, COOKIE(4), false);
// Stage 5: Test recurring with short duration
new(&mStages[5]) Stage(5, kShortDuration, COOKIE(5), false);
static_assert((5 + 1) == kStageCount, "Missized array");
#undef COOKIE
}
TimerSetTest::TimerSetTest()
: Test(CHRE_API_VERSION_1_0), mInMethod(false), mFinishedBitmask(0) {
}
void TimerSetTest::setUp(uint32_t messageSize, const void * /* message */) {
mInMethod = true;
if (messageSize != 0) {
sendFatalFailureToHost(
"TimerSet message expects 0 additional bytes, got ",
&messageSize);
}
initStages();
for (size_t i = 0; i < kStageCount; i++) {
mStages[i].start();
}
mInMethod = false;
}
TimerSetTest::~TimerSetTest() {
chreHeapFree(mStages);
}
void TimerSetTest::handleEvent(uint32_t senderInstanceId,
uint16_t eventType, const void* eventData) {
uint64_t timestamp = chreGetTime();
if (mInMethod) {
sendFatalFailureToHost("handleEvent invoked while another nanoapp "
"method is running");
}
mInMethod = true;
if (senderInstanceId != CHRE_INSTANCE_ID) {
sendFatalFailureToHost("handleEvent got event from unexpected sender:",
&senderInstanceId);
}
if (eventType != CHRE_EVENT_TIMER) {
unexpectedEvent(eventType);
}
Stage *stage = getStageFromCookie(eventData);
if (stage == nullptr) {
sendFatalFailureToHost("handleEvent got invalid eventData");
}
stage->processEvent(timestamp, this);
mInMethod = false;
}
void TimerSetTest::markSuccess(uint32_t stage) {
chreLog(CHRE_LOG_DEBUG, "Stage %" PRIu32 " succeeded", stage);
uint32_t finishedBit = (1 << stage);
if ((kAllFinished & finishedBit) == 0) {
sendFatalFailureToHost("markSuccess bad stage", &stage);
}
mFinishedBitmask |= finishedBit;
if (mFinishedBitmask == kAllFinished) {
sendSuccessToHost();
}
}
TimerSetTest::Stage *TimerSetTest::getStageFromCookie(const void *cookie) {
Stage *ret = nullptr;
for (size_t i = 0; i < kStageCount; i++) {
if (mStages[i].getCookie() == cookie) {
if (ret != nullptr) {
sendInternalFailureToHost("Multiple stages with the same "
"cookie");
}
ret = &mStages[i];
// It's cheap enough to go through the whole array, and will
// catch if we screw up this test setup by duplicating a cookie.
}
}
return ret;
}
} // namespace general_test