blob: a3ae543fd60a5b91b5b34486f19d10fa77ed91ce [file] [log] [blame] [edit]
// Copyright 2021 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 <memory>
#include <string>
#include <utility>
#include <vector>
#include <base/memory/scoped_refptr.h>
#include <base/test/task_environment.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include "rmad/state_handler/state_handler_test_common.h"
#include "rmad/state_handler/write_protect_disable_physical_state_handler.h"
#include "rmad/system/mock_cryptohome_client.h"
#include "rmad/utils/mock_cr50_utils.h"
#include "rmad/utils/mock_crossystem_utils.h"
using testing::_;
using testing::Assign;
using testing::DoAll;
using testing::InSequence;
using testing::IsFalse;
using testing::NiceMock;
using testing::Return;
using testing::SetArgPointee;
using testing::StrictMock;
namespace rmad {
class WriteProtectDisablePhysicalStateHandlerTest : public StateHandlerTest {
public:
// Helper class to mock the callback function to send signal.
class SignalSender {
public:
MOCK_METHOD(bool, SendHardwareWriteProtectSignal, (bool), (const));
};
scoped_refptr<WriteProtectDisablePhysicalStateHandler> CreateStateHandler(
const std::vector<int> wp_status_list,
bool factory_mode_enabled,
bool is_enrolled,
bool* factory_mode_toggled = nullptr) {
// Mock |Cr50Utils|, |CrosSystemUtils| and |CryptohomeClient|.
auto mock_crossystem_utils =
std::make_unique<StrictMock<MockCrosSystemUtils>>();
{
InSequence seq;
for (int i = 0; i < wp_status_list.size(); ++i) {
EXPECT_CALL(*mock_crossystem_utils, GetInt(_, _))
.WillOnce(DoAll(SetArgPointee<1>(wp_status_list[i]), Return(true)));
}
}
auto mock_cr50_utils = std::make_unique<NiceMock<MockCr50Utils>>();
ON_CALL(*mock_cr50_utils, IsFactoryModeEnabled())
.WillByDefault(Return(factory_mode_enabled));
if (factory_mode_toggled) {
ON_CALL(*mock_cr50_utils, EnableFactoryMode())
.WillByDefault(
DoAll(Assign(factory_mode_toggled, true), Return(true)));
}
auto mock_cryptohome_client =
std::make_unique<NiceMock<MockCryptohomeClient>>();
ON_CALL(*mock_cryptohome_client, IsEnrolled())
.WillByDefault(Return(is_enrolled));
auto handler =
base::MakeRefCounted<WriteProtectDisablePhysicalStateHandler>(
json_store_, std::move(mock_cr50_utils),
std::move(mock_crossystem_utils),
std::move(mock_cryptohome_client));
auto callback = std::make_unique<base::RepeatingCallback<bool(bool)>>(
base::BindRepeating(&SignalSender::SendHardwareWriteProtectSignal,
base::Unretained(&signal_sender_)));
handler->RegisterSignalSender(std::move(callback));
return handler;
}
protected:
StrictMock<SignalSender> signal_sender_;
bool factory_mode_enabled_;
// Variables for TaskRunner.
base::test::SingleThreadTaskEnvironment task_environment_{
base::test::TaskEnvironment::TimeSource::MOCK_TIME};
};
TEST_F(WriteProtectDisablePhysicalStateHandlerTest, InitializeState_Success) {
auto handler = CreateStateHandler({}, false, false);
EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK);
}
TEST_F(WriteProtectDisablePhysicalStateHandlerTest,
GetNextStateCase_Success_FactoryModeEnabled) {
bool factory_mode_toggled = false;
auto handler = CreateStateHandler({0}, true, false, &factory_mode_toggled);
EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK);
RmadState state;
state.set_allocated_wp_disable_physical(new WriteProtectDisablePhysicalState);
auto [error, state_case] = handler->GetNextStateCase(state);
EXPECT_EQ(error, RMAD_ERROR_OK);
EXPECT_EQ(state_case, RmadState::StateCase::kWpDisableComplete);
EXPECT_FALSE(factory_mode_toggled);
}
TEST_F(WriteProtectDisablePhysicalStateHandlerTest,
GetNextStateCase_Success_FactoryModeDisabled_NoFwmp) {
bool factory_mode_toggled = false;
auto handler = CreateStateHandler({0}, false, false, &factory_mode_toggled);
EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK);
RmadState state;
state.set_allocated_wp_disable_physical(new WriteProtectDisablePhysicalState);
auto [error, state_case] = handler->GetNextStateCase(state);
EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT);
EXPECT_EQ(state_case, RmadState::StateCase::kWpDisablePhysical);
EXPECT_TRUE(factory_mode_toggled);
}
TEST_F(WriteProtectDisablePhysicalStateHandlerTest,
GetNextStateCase_Success_FactoryModeDisabled_HasFwmp) {
bool factory_mode_toggled = false;
auto handler = CreateStateHandler({0}, false, true, &factory_mode_toggled);
EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK);
RmadState state;
state.set_allocated_wp_disable_physical(new WriteProtectDisablePhysicalState);
auto [error, state_case] = handler->GetNextStateCase(state);
EXPECT_EQ(error, RMAD_ERROR_OK);
EXPECT_EQ(state_case, RmadState::StateCase::kWpDisableComplete);
EXPECT_FALSE(factory_mode_toggled);
}
TEST_F(WriteProtectDisablePhysicalStateHandlerTest,
GetNextStateCase_MissingState) {
auto handler = CreateStateHandler({}, false, false);
EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK);
// No WriteProtectDisablePhysicalState.
RmadState state;
auto [error, state_case] = handler->GetNextStateCase(state);
EXPECT_EQ(error, RMAD_ERROR_REQUEST_INVALID);
EXPECT_EQ(state_case, RmadState::StateCase::kWpDisablePhysical);
}
TEST_F(WriteProtectDisablePhysicalStateHandlerTest, GetNextStateCase_Wait) {
auto handler = CreateStateHandler({1, 1, 1, 0}, false, false);
EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK);
RmadState state;
state.set_allocated_wp_disable_physical(new WriteProtectDisablePhysicalState);
// First call to |mock_crossystem_utils_|, get 1.
auto [error, state_case] = handler->GetNextStateCase(state);
EXPECT_EQ(error, RMAD_ERROR_WAIT);
EXPECT_EQ(state_case, RmadState::StateCase::kWpDisablePhysical);
bool signal_sent = false;
EXPECT_CALL(signal_sender_, SendHardwareWriteProtectSignal(IsFalse()))
.WillOnce(DoAll(Assign(&signal_sent, true), Return(true)));
// Second call to |mock_crossystem_utils_| during polling, get 1.
task_environment_.FastForwardBy(
WriteProtectDisablePhysicalStateHandler::kPollInterval);
EXPECT_FALSE(signal_sent);
// Third call to |mock_crossystem_utils_| during polling, get 1.
task_environment_.FastForwardBy(
WriteProtectDisablePhysicalStateHandler::kPollInterval);
EXPECT_FALSE(signal_sent);
// Forth call to |mock_crossystem_utils_| during polling, get 0.
task_environment_.FastForwardBy(
WriteProtectDisablePhysicalStateHandler::kPollInterval);
EXPECT_TRUE(signal_sent);
}
} // namespace rmad