blob: fd82c2d5c1324847ee43700801e80992cb01d03b [file] [log] [blame]
Alex Deymoaea4c1c2015-08-19 20:24:43 -07001//
2// Copyright (C) 2009 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
rspangler@google.com49fdf182009-10-10 00:57:34 +000016
Alex Deymo39910dc2015-11-09 17:04:30 -080017#ifndef UPDATE_ENGINE_COMMON_ACTION_H_
18#define UPDATE_ENGINE_COMMON_ACTION_H_
rspangler@google.com49fdf182009-10-10 00:57:34 +000019
20#include <stdio.h>
Alex Deymobc91a272014-05-20 16:45:33 -070021
Alex Deymobc91a272014-05-20 16:45:33 -070022#include <memory>
Alex Vakulenkod2779df2014-06-16 13:19:00 -070023#include <string>
Alex Deymobc91a272014-05-20 16:45:33 -070024
Alex Deymobc91a272014-05-20 16:45:33 -070025#include <base/logging.h>
Ben Chan05735a12014-09-03 07:48:22 -070026#include <base/macros.h>
Alex Deymobc91a272014-05-20 16:45:33 -070027
Alex Deymo39910dc2015-11-09 17:04:30 -080028#include "update_engine/common/action_pipe.h"
29#include "update_engine/common/action_processor.h"
rspangler@google.com49fdf182009-10-10 00:57:34 +000030
31// The structure of these classes (Action, ActionPipe, ActionProcessor, etc.)
32// is based on the KSAction* classes from the Google Update Engine code at
33// http://code.google.com/p/update-engine/ . The author of this file sends
34// a big thanks to that team for their high quality design, implementation,
35// and documentation.
36//
37// Readers may want to consult this wiki page from the Update Engine site:
38// http://code.google.com/p/update-engine/wiki/ActionProcessor
39// Although it's referring to the Objective-C KSAction* classes, much
40// applies here as well.
41//
42// How it works:
43//
44// First off, there is only one thread and all I/O should be asynchronous.
Alex Deymo0b3db6b2015-08-10 15:19:37 -070045// A message loop blocks whenever there is no work to be done. This happens
rspangler@google.com49fdf182009-10-10 00:57:34 +000046// where there is no CPU work to be done and no I/O ready to transfer in or
Alex Deymo0b3db6b2015-08-10 15:19:37 -070047// out. Two kinds of events can wake up the message loop: timer alarm or file
48// descriptors. If either of these happens, the message loop finds out the owner
49// of what fired and calls the appropriate code to handle it. As such, all the
50// code in the Action* classes and the code that is calls is non-blocking.
rspangler@google.com49fdf182009-10-10 00:57:34 +000051//
52// An ActionProcessor contains a queue of Actions to perform. When
53// ActionProcessor::StartProcessing() is called, it executes the first action.
54// Each action tells the processor when it has completed, which causes the
55// Processor to execute the next action. ActionProcessor may have a delegate
56// (an object of type ActionProcessorDelegate). If it does, the delegate
57// is called to be notified of events as they happen.
58//
59// ActionPipe classes
60//
61// See action_pipe.h
62//
63// ActionTraits
64//
65// We need to use an extra class ActionTraits. ActionTraits is a simple
66// templated class that contains only two typedefs: OutputObjectType and
67// InputObjectType. Each action class also has two typedefs of the same name
68// that are of the same type. So, to get the input/output types of, e.g., the
69// DownloadAction class, we look at the type of
70// DownloadAction::InputObjectType.
71//
72// Each concrete Action class derives from Action<T>. This means that during
Alex Vakulenkod2779df2014-06-16 13:19:00 -070073// template instantiation of Action<T>, T is declared but not defined, which
rspangler@google.com49fdf182009-10-10 00:57:34 +000074// means that T::InputObjectType (and OutputObjectType) is not defined.
75// However, the traits class is constructed in such a way that it will be
Alex Vakulenkod2779df2014-06-16 13:19:00 -070076// template instantiated first, so Action<T> *can* find the types it needs by
rspangler@google.com49fdf182009-10-10 00:57:34 +000077// consulting ActionTraits<T>::InputObjectType (and OutputObjectType).
78// This is why the ActionTraits classes are needed.
79
rspangler@google.com49fdf182009-10-10 00:57:34 +000080namespace chromeos_update_engine {
81
82// It is handy to have a non-templated base class of all Actions.
83class AbstractAction {
84 public:
Alex Vakulenko88b591f2014-08-28 16:48:57 -070085 AbstractAction() : processor_(nullptr) {}
Alex Deymoe8948702014-11-11 21:44:45 -080086 virtual ~AbstractAction() = default;
rspangler@google.com49fdf182009-10-10 00:57:34 +000087
88 // Begin performing the action. Since this code is asynchronous, when this
89 // method returns, it means only that the action has started, not necessarily
90 // completed. However, it's acceptable for this method to perform the
91 // action synchronously; Action authors should understand the implications
92 // of synchronously performing, though, because this is a single-threaded
93 // app, the entire process will be blocked while the action performs.
94 //
95 // When the action is complete, it must call
96 // ActionProcessor::ActionComplete(this); to notify the processor that it's
97 // done.
98 virtual void PerformAction() = 0;
99
David Zeuthen33bae492014-02-25 16:16:18 -0800100 // Called on ActionProcess::ActionComplete() by ActionProcessor.
101 virtual void ActionCompleted(ErrorCode code) {}
102
rspangler@google.com49fdf182009-10-10 00:57:34 +0000103 // Called by the ActionProcessor to tell this Action which processor
104 // it belongs to.
105 void SetProcessor(ActionProcessor* processor) {
106 if (processor)
107 CHECK(!processor_);
108 else
109 CHECK(processor_);
110 processor_ = processor;
111 }
112
113 // Returns true iff the action is the current action of its ActionProcessor.
114 bool IsRunning() const {
115 if (!processor_)
116 return false;
117 return processor_->current_action() == this;
118 }
119
120 // Called on asynchronous actions if canceled. Actions may implement if
121 // there's any cleanup to do. There is no need to call
122 // ActionProcessor::ActionComplete() because the processor knows this
123 // action is terminating.
124 // Only the ActionProcessor should call this.
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700125 virtual void TerminateProcessing() {}
rspangler@google.com49fdf182009-10-10 00:57:34 +0000126
Alex Deymo14fd1ec2016-02-24 22:03:57 -0800127 // Called on asynchronous actions if the processing is suspended and resumed,
128 // respectively. These methods are called by the ActionProcessor and should
129 // not be explicitly called.
130 // The action may still call ActionCompleted() once the action is completed
131 // while the processing is suspended, for example if suspend/resume is not
132 // implemented for the given action.
133 virtual void SuspendAction() {}
134 virtual void ResumeAction() {}
135
rspangler@google.com49fdf182009-10-10 00:57:34 +0000136 // These methods are useful for debugging. TODO(adlr): consider using
137 // std::type_info for this?
138 // Type() returns a string of the Action type. I.e., for DownloadAction,
139 // Type() would return "DownloadAction".
Andrew de los Reyes4fe15d02009-12-10 19:01:36 -0800140 virtual std::string Type() const = 0;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000141
142 protected:
143 // A weak pointer to the processor that owns this Action.
144 ActionProcessor* processor_;
145};
146
147// Forward declare a couple classes we use.
Amin Hassanib2689592019-01-13 17:04:28 -0800148template <typename T>
rspangler@google.com49fdf182009-10-10 00:57:34 +0000149class ActionPipe;
Amin Hassanib2689592019-01-13 17:04:28 -0800150template <typename T>
rspangler@google.com49fdf182009-10-10 00:57:34 +0000151class ActionTraits;
152
Amin Hassanib2689592019-01-13 17:04:28 -0800153template <typename SubClass>
rspangler@google.com49fdf182009-10-10 00:57:34 +0000154class Action : public AbstractAction {
155 public:
Alex Deymo610277e2014-11-11 21:18:11 -0800156 ~Action() override {}
rspangler@google.com49fdf182009-10-10 00:57:34 +0000157
158 // Attaches an input pipe to this Action. This is optional; an Action
159 // doesn't need to have an input pipe. The input pipe must be of the type
160 // of object that this class expects.
161 // This is generally called by ActionPipe::Bond()
162 void set_in_pipe(
163 // this type is a fancy way of saying: a shared_ptr to an
164 // ActionPipe<InputObjectType>.
Amin Hassanib2689592019-01-13 17:04:28 -0800165 const std::shared_ptr<
166 ActionPipe<typename ActionTraits<SubClass>::InputObjectType>>&
167 in_pipe) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000168 in_pipe_ = in_pipe;
169 }
170
171 // Attaches an output pipe to this Action. This is optional; an Action
172 // doesn't need to have an output pipe. The output pipe must be of the type
173 // of object that this class expects.
174 // This is generally called by ActionPipe::Bond()
175 void set_out_pipe(
176 // this type is a fancy way of saying: a shared_ptr to an
177 // ActionPipe<OutputObjectType>.
Amin Hassanib2689592019-01-13 17:04:28 -0800178 const std::shared_ptr<
179 ActionPipe<typename ActionTraits<SubClass>::OutputObjectType>>&
180 out_pipe) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000181 out_pipe_ = out_pipe;
182 }
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000183
rspangler@google.com49fdf182009-10-10 00:57:34 +0000184 // Returns true iff there is an associated input pipe. If there's an input
185 // pipe, there's an input object, but it may have been constructed with the
186 // default ctor if the previous action didn't call SetOutputObject().
187 bool HasInputObject() const { return in_pipe_.get(); }
188
189 // returns a const reference to the object in the input pipe.
190 const typename ActionTraits<SubClass>::InputObjectType& GetInputObject()
191 const {
192 CHECK(HasInputObject());
193 return in_pipe_->contents();
194 }
195
196 // Returns true iff there's an output pipe.
Amin Hassanib2689592019-01-13 17:04:28 -0800197 bool HasOutputPipe() const { return out_pipe_.get(); }
rspangler@google.com49fdf182009-10-10 00:57:34 +0000198
199 // Copies the object passed into the output pipe. It will be accessible to
200 // the next Action via that action's input pipe (which is the same as this
201 // Action's output pipe).
202 void SetOutputObject(
203 const typename ActionTraits<SubClass>::OutputObjectType& out_obj) {
204 CHECK(HasOutputPipe());
205 out_pipe_->set_contents(out_obj);
206 }
207
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000208 // Returns a reference to the object sitting in the output pipe.
209 const typename ActionTraits<SubClass>::OutputObjectType& GetOutputObject() {
210 CHECK(HasOutputPipe());
211 return out_pipe_->contents();
212 }
213
Andrew de los Reyesf98bff82010-05-06 13:33:25 -0700214 protected:
rspangler@google.com49fdf182009-10-10 00:57:34 +0000215 // We use a shared_ptr to the pipe. shared_ptr objects destroy what they
216 // point to when the last such shared_ptr object dies. We consider the
217 // Actions on either end of a pipe to "own" the pipe. When the last Action
218 // of the two dies, the ActionPipe will die, too.
Ben Chanf9cb98c2014-09-21 18:31:30 -0700219 std::shared_ptr<ActionPipe<typename ActionTraits<SubClass>::InputObjectType>>
rspangler@google.com49fdf182009-10-10 00:57:34 +0000220 in_pipe_;
Ben Chanf9cb98c2014-09-21 18:31:30 -0700221 std::shared_ptr<ActionPipe<typename ActionTraits<SubClass>::OutputObjectType>>
rspangler@google.com49fdf182009-10-10 00:57:34 +0000222 out_pipe_;
223};
224
Yifan Hong90965502020-02-19 15:22:47 -0800225// An action that does nothing and completes with kSuccess immediately.
226class NoOpAction : public AbstractAction {
227 public:
228 ~NoOpAction() override {}
229 void PerformAction() override {
230 processor_->ActionComplete(this, ErrorCode::kSuccess);
231 }
Yifan Hong40bb0d02020-02-24 17:33:14 -0800232 static std::string StaticType() { return "NoOpAction"; }
233 std::string Type() const override { return StaticType(); }
Yifan Hong90965502020-02-19 15:22:47 -0800234};
235
rspangler@google.com49fdf182009-10-10 00:57:34 +0000236}; // namespace chromeos_update_engine
237
Alex Deymo39910dc2015-11-09 17:04:30 -0800238#endif // UPDATE_ENGINE_COMMON_ACTION_H_