blob: 65ce3979f95d0d092a57ec6fad78eebdd96d5536 [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 "rmad/state_handler/repair_complete_state_handler.h"
#include <memory>
#include <utility>
#include <base/files/file_path.h>
#include <base/logging.h>
#include <base/notreached.h>
#include <brillo/file_utils.h>
#include "rmad/constants.h"
#include "rmad/system/power_manager_client_impl.h"
#include "rmad/utils/dbus_utils.h"
namespace rmad {
RepairCompleteStateHandler::RepairCompleteStateHandler(
scoped_refptr<JsonStore> json_store)
: BaseStateHandler(json_store), working_dir_path_(kDefaultWorkingDirPath) {
power_manager_client_ =
std::make_unique<PowerManagerClientImpl>(GetSystemBus());
}
RepairCompleteStateHandler::RepairCompleteStateHandler(
scoped_refptr<JsonStore> json_store,
const base::FilePath& working_dir_path,
std::unique_ptr<PowerManagerClient> power_manager_client)
: BaseStateHandler(json_store),
working_dir_path_(working_dir_path),
power_manager_client_(std::move(power_manager_client)) {}
RmadErrorCode RepairCompleteStateHandler::InitializeState() {
if (!state_.has_repair_complete()) {
state_.set_allocated_repair_complete(new RepairCompleteState);
}
return RMAD_ERROR_OK;
}
BaseStateHandler::GetNextStateCaseReply
RepairCompleteStateHandler::GetNextStateCase(const RmadState& state) {
if (!state.has_repair_complete()) {
LOG(ERROR) << "RmadState missing |repair_complete| state.";
return {.error = RMAD_ERROR_REQUEST_INVALID, .state_case = GetStateCase()};
}
if (state.repair_complete().shutdown() ==
RepairCompleteState::RMAD_REPAIR_COMPLETE_UNKNOWN) {
return {.error = RMAD_ERROR_REQUEST_ARGS_MISSING,
.state_case = GetStateCase()};
}
// Clear the state file.
if (!json_store_->ClearAndDeleteFile()) {
LOG(ERROR) << "RepairCompleteState: Failed to clear RMA state file";
return {.error = RMAD_ERROR_TRANSITION_FAILED,
.state_case = GetStateCase()};
}
switch (state.repair_complete().shutdown()) {
case RepairCompleteState::RMAD_REPAIR_COMPLETE_REBOOT:
// Wait for a while before reboot.
timer_.Start(FROM_HERE, kShutdownDelay, this,
&RepairCompleteStateHandler::Reboot);
return {.error = RMAD_ERROR_EXPECT_REBOOT, .state_case = GetStateCase()};
case RepairCompleteState::RMAD_REPAIR_COMPLETE_SHUTDOWN:
// Wait for a while before shutdown.
timer_.Start(FROM_HERE, kShutdownDelay, this,
&RepairCompleteStateHandler::Shutdown);
return {.error = RMAD_ERROR_EXPECT_SHUTDOWN,
.state_case = GetStateCase()};
case RepairCompleteState::RMAD_REPAIR_COMPLETE_BATTERY_CUTOFF:
// Wait for a while before cutoff.
timer_.Start(FROM_HERE, kShutdownDelay, this,
&RepairCompleteStateHandler::Cutoff);
return {.error = RMAD_ERROR_EXPECT_SHUTDOWN,
.state_case = GetStateCase()};
default:
break;
}
NOTREACHED();
return {.error = RMAD_ERROR_NOT_SET,
.state_case = RmadState::StateCase::STATE_NOT_SET};
}
void RepairCompleteStateHandler::Reboot() {
LOG(INFO) << "RMA flow complete. Rebooting.";
if (!power_manager_client_->Restart()) {
LOG(ERROR) << "Failed to reboot";
}
}
void RepairCompleteStateHandler::Shutdown() {
LOG(INFO) << "RMA flow complete. Shutting down.";
if (!power_manager_client_->Shutdown()) {
LOG(ERROR) << "Failed to shut down";
}
}
void RepairCompleteStateHandler::Cutoff() {
LOG(INFO) << "RMA flow complete. Doing battery cutoff.";
// The pre-stop script picks up the file before shutdown/reboot, and requests
// a battery cutoff by crossystem.
if (!brillo::TouchFile(
working_dir_path_.AppendASCII(kCutoffRequestFilePath))) {
LOG(ERROR) << "Failed to request battery cutoff";
return;
}
// Battery cutoff requires a reboot (not shutdown) after the request.
if (!power_manager_client_->Restart()) {
LOG(ERROR) << "Failed to reboot";
}
}
} // namespace rmad