1 //
2 // Copyright (C) 2010 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 //
16 
17 #ifndef UPDATE_ENGINE_PAYLOAD_CONSUMER_POSTINSTALL_RUNNER_ACTION_H_
18 #define UPDATE_ENGINE_PAYLOAD_CONSUMER_POSTINSTALL_RUNNER_ACTION_H_
19 
20 #include <memory>
21 #include <string>
22 #include <utility>
23 #include <vector>
24 
25 #include <base/files/file_descriptor_watcher_posix.h>
26 #include <brillo/message_loops/message_loop.h>
27 #include <gtest/gtest_prod.h>
28 
29 #include "update_engine/common/action.h"
30 #include "update_engine/common/boot_control_interface.h"
31 #include "update_engine/common/hardware_interface.h"
32 #include "update_engine/payload_consumer/install_plan.h"
33 
34 // The Postinstall Runner Action is responsible for running the postinstall
35 // script of a successfully downloaded update.
36 
37 namespace chromeos_update_engine {
38 
39 class BootControlInterface;
40 
41 class PostinstallRunnerAction : public InstallPlanAction {
42  public:
43   PostinstallRunnerAction(BootControlInterface* boot_control,
44                           HardwareInterface* hardware);
45   ~PostinstallRunnerAction();
46 
47   // InstallPlanAction overrides.
48   void PerformAction() override;
49   void SuspendAction() override;
50   void ResumeAction() override;
51   void TerminateProcessing() override;
52 
53   class DelegateInterface {
54    public:
55     virtual ~DelegateInterface() = default;
56 
57     // Called whenever there is an overall progress update from the postinstall
58     // programs.
59     virtual void ProgressUpdate(double progress) = 0;
60   };
61 
set_delegate(DelegateInterface * delegate)62   void set_delegate(DelegateInterface* delegate) { delegate_ = delegate; }
63 
64   // Debugging/logging
StaticType()65   static std::string StaticType() { return "PostinstallRunnerAction"; }
Type()66   std::string Type() const override { return StaticType(); }
67 
68  private:
69   friend class PostinstallRunnerActionTest;
70   FRIEND_TEST(PostinstallRunnerActionTest, ProcessProgressLineTest);
71 
72   // exposed for testing purposes only
SetMountDir(std::string dir)73   void SetMountDir(std::string dir) { fs_mount_dir_ = std::move(dir); }
74   void EnsureUnmounted();
75 
76   void PerformPartitionPostinstall();
77   [[nodiscard]] bool MountPartition(
78       const InstallPlan::Partition& partition) noexcept;
79 
80   // Called whenever the |progress_fd_| has data available to read.
81   void OnProgressFdReady();
82 
83   // Updates the action progress according to the |line| passed from the
84   // postinstall program. Valid lines are:
85   //     global_progress <frac>
86   //         <frac> should be between 0.0 and 1.0; sets the progress to the
87   //         <frac> value.
88   bool ProcessProgressLine(const std::string& line);
89 
90   // Report the progress to the delegate given that the postinstall operation
91   // for |current_partition_| has a current progress of |frac|, a value between
92   // 0 and 1 for that step.
93   void ReportProgress(double frac);
94 
95   // Cleanup the setup made when running postinstall for a given partition.
96   // Unmount and remove the mountpoint directory if needed and cleanup the
97   // status file descriptor and message loop task watching for it.
98   void Cleanup();
99 
100   // Subprocess::Exec callback.
101   void CompletePartitionPostinstall(int return_code, const std::string& output);
102 
103   // Complete the Action with the passed |error_code| and mark the new slot as
104   // ready. Called when the post-install script was run for all the partitions.
105   void CompletePostinstall(ErrorCode error_code);
106 
107   // The path where the filesystem will be mounted during post-install.
108   std::string fs_mount_dir_;
109 
110   // The partition being processed on the list of partitions specified in the
111   // InstallPlan.
112   size_t current_partition_{0};
113 
114   // A non-negative value representing the estimated weight of each partition
115   // passed in the install plan. The weight is used to predict the overall
116   // progress from the individual progress of each partition and should
117   // correspond to the time it takes to run it.
118   std::vector<double> partition_weight_;
119 
120   // The sum of all the weights in |partition_weight_|.
121   double total_weight_{0};
122 
123   // The sum of all the weights in |partition_weight_| up to but not including
124   // the |current_partition_|.
125   double accumulated_weight_{0};
126 
127   // The delegate used to notify of progress updates, if any.
128   DelegateInterface* delegate_{nullptr};
129 
130   // The BootControlInerface used to mark the new slot as ready.
131   BootControlInterface* boot_control_;
132 
133   // HardwareInterface used to signal powerwash.
134   HardwareInterface* hardware_;
135 
136   // Whether the Powerwash was scheduled before invoking post-install script.
137   // Used for cleaning up if post-install fails.
138   bool powerwash_scheduled_{false};
139 
140   // Postinstall command currently running, or 0 if no program running.
141   pid_t current_command_{0};
142 
143   // True if |current_command_| has been suspended by SuspendAction().
144   bool is_current_command_suspended_{false};
145 
146   // The parent progress file descriptor used to watch for progress reports from
147   // the postinstall program and the task watching for them.
148   int progress_fd_{-1};
149 
150   std::unique_ptr<base::FileDescriptorWatcher::Controller> progress_controller_;
151 
152   // A buffer of a partial read line from the progress file descriptor.
153   std::string progress_buffer_;
154 
155   DISALLOW_COPY_AND_ASSIGN(PostinstallRunnerAction);
156 };
157 
158 }  // namespace chromeos_update_engine
159 
160 #endif  // UPDATE_ENGINE_PAYLOAD_CONSUMER_POSTINSTALL_RUNNER_ACTION_H_
161