/* * Copyright 2018 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. */ #pragma once #include <map> #include <utility> #include <base/logging.h> namespace bluetooth { namespace common { /** * State machine used by Bluetooth native stack. */ class StateMachine { public: enum { kStateInvalid = -1 }; /** * A class to represent the state in the State Machine. */ class State { friend class StateMachine; public: /** * Constructor. * * @param sm the State Machine to use * @param state_id the unique State ID. It should be a non-negative number. */ State(StateMachine& sm, int state_id) : sm_(sm), state_id_(state_id) {} virtual ~State() = default; /** * Process an event. * TODO: The arguments are wrong - used for backward compatibility. * Will be replaced later. * * @param event the event type * @param p_data the event data * @return true if the processing was completed, otherwise false */ virtual bool ProcessEvent(uint32_t event, void* p_data) = 0; /** * Get the State ID. * * @return the State ID */ int StateId() const { return state_id_; } protected: /** * Called when a state is entered. */ virtual void OnEnter() {} /** * Called when a state is exited. */ virtual void OnExit() {} /** * Transition the State Machine to a new state. * * @param dest_state_id the state ID to transition to. It must be one * of the unique state IDs when the corresponding state was created. */ void TransitionTo(int dest_state_id) { sm_.TransitionTo(dest_state_id); } /** * Transition the State Machine to a new state. * * @param dest_state the state to transition to. It cannot be nullptr. */ void TransitionTo(StateMachine::State* dest_state) { sm_.TransitionTo(dest_state); } private: StateMachine& sm_; int state_id_; }; StateMachine() : initial_state_(nullptr), previous_state_(nullptr), current_state_(nullptr) {} ~StateMachine() { for (auto& kv : states_) delete kv.second; } /** * Start the State Machine operation. */ void Start() { TransitionTo(initial_state_); } /** * Quit the State Machine operation. */ void Quit() { previous_state_ = current_state_ = nullptr; } /** * Get the current State ID. * * @return the current State ID */ int StateId() const { if (current_state_ != nullptr) { return current_state_->StateId(); } return kStateInvalid; } /** * Get the previous current State ID. * * @return the previous State ID */ int PreviousStateId() const { if (previous_state_ != nullptr) { return previous_state_->StateId(); } return kStateInvalid; } /** * Process an event. * TODO: The arguments are wrong - used for backward compatibility. * Will be replaced later. * * @param event the event type * @param p_data the event data * @return true if the processing was completed, otherwise false */ bool ProcessEvent(uint32_t event, void* p_data) { if (current_state_ == nullptr) return false; return current_state_->ProcessEvent(event, p_data); } /** * Transition the State Machine to a new state. * * @param dest_state_id the state ID to transition to. It must be one * of the unique state IDs when the corresponding state was created. */ void TransitionTo(int dest_state_id) { auto it = states_.find(dest_state_id); CHECK(it != states_.end()) << "Unknown State ID: " << dest_state_id; State* dest_state = it->second; TransitionTo(dest_state); } /** * Transition the State Machine to a new state. * * @param dest_state the state to transition to. It cannot be nullptr. */ void TransitionTo(StateMachine::State* dest_state) { if (current_state_ != nullptr) { current_state_->OnExit(); } previous_state_ = current_state_; current_state_ = dest_state; current_state_->OnEnter(); } /** * Add a state to the State Machine. * The state machine takes ownership on the state - i.e., the state will * be deleted by the State Machine itself. * * @param state the state to add */ void AddState(State* state) { states_.insert(std::make_pair(state->StateId(), state)); } /** * Set the initial state of the State Machine. * * @param initial_state the initial state */ void SetInitialState(State* initial_state) { initial_state_ = initial_state; } private: State* initial_state_; State* previous_state_; State* current_state_; std::map<int, State*> states_; }; } // namespace common } // namespace bluetooth