xref: /aosp_15_r20/system/core/init/init.cpp (revision 00c7fec1bb09f3284aad6a6f96d2f63dfc3650ad)
1*00c7fec1SAndroid Build Coastguard Worker /*
2*00c7fec1SAndroid Build Coastguard Worker  * Copyright (C) 2008 The Android Open Source Project
3*00c7fec1SAndroid Build Coastguard Worker  *
4*00c7fec1SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*00c7fec1SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*00c7fec1SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*00c7fec1SAndroid Build Coastguard Worker  *
8*00c7fec1SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*00c7fec1SAndroid Build Coastguard Worker  *
10*00c7fec1SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*00c7fec1SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*00c7fec1SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*00c7fec1SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*00c7fec1SAndroid Build Coastguard Worker  * limitations under the License.
15*00c7fec1SAndroid Build Coastguard Worker  */
16*00c7fec1SAndroid Build Coastguard Worker 
17*00c7fec1SAndroid Build Coastguard Worker #include "init.h"
18*00c7fec1SAndroid Build Coastguard Worker 
19*00c7fec1SAndroid Build Coastguard Worker #include <dirent.h>
20*00c7fec1SAndroid Build Coastguard Worker #include <fcntl.h>
21*00c7fec1SAndroid Build Coastguard Worker #include <paths.h>
22*00c7fec1SAndroid Build Coastguard Worker #include <pthread.h>
23*00c7fec1SAndroid Build Coastguard Worker #include <signal.h>
24*00c7fec1SAndroid Build Coastguard Worker #include <stdlib.h>
25*00c7fec1SAndroid Build Coastguard Worker #include <string.h>
26*00c7fec1SAndroid Build Coastguard Worker #include <sys/eventfd.h>
27*00c7fec1SAndroid Build Coastguard Worker #include <sys/mount.h>
28*00c7fec1SAndroid Build Coastguard Worker #include <sys/signalfd.h>
29*00c7fec1SAndroid Build Coastguard Worker #include <sys/system_properties.h>
30*00c7fec1SAndroid Build Coastguard Worker #include <sys/types.h>
31*00c7fec1SAndroid Build Coastguard Worker #include <sys/utsname.h>
32*00c7fec1SAndroid Build Coastguard Worker #include <unistd.h>
33*00c7fec1SAndroid Build Coastguard Worker 
34*00c7fec1SAndroid Build Coastguard Worker #include <filesystem>
35*00c7fec1SAndroid Build Coastguard Worker #include <fstream>
36*00c7fec1SAndroid Build Coastguard Worker #include <functional>
37*00c7fec1SAndroid Build Coastguard Worker #include <iostream>
38*00c7fec1SAndroid Build Coastguard Worker #include <map>
39*00c7fec1SAndroid Build Coastguard Worker #include <memory>
40*00c7fec1SAndroid Build Coastguard Worker #include <mutex>
41*00c7fec1SAndroid Build Coastguard Worker #include <optional>
42*00c7fec1SAndroid Build Coastguard Worker #include <thread>
43*00c7fec1SAndroid Build Coastguard Worker #include <vector>
44*00c7fec1SAndroid Build Coastguard Worker 
45*00c7fec1SAndroid Build Coastguard Worker #include <android-base/chrono_utils.h>
46*00c7fec1SAndroid Build Coastguard Worker #include <android-base/file.h>
47*00c7fec1SAndroid Build Coastguard Worker #include <android-base/logging.h>
48*00c7fec1SAndroid Build Coastguard Worker #include <android-base/parseint.h>
49*00c7fec1SAndroid Build Coastguard Worker #include <android-base/properties.h>
50*00c7fec1SAndroid Build Coastguard Worker #include <android-base/stringprintf.h>
51*00c7fec1SAndroid Build Coastguard Worker #include <android-base/strings.h>
52*00c7fec1SAndroid Build Coastguard Worker #include <android-base/thread_annotations.h>
53*00c7fec1SAndroid Build Coastguard Worker #include <fs_avb/fs_avb.h>
54*00c7fec1SAndroid Build Coastguard Worker #include <fs_mgr_vendor_overlay.h>
55*00c7fec1SAndroid Build Coastguard Worker #include <libavb/libavb.h>
56*00c7fec1SAndroid Build Coastguard Worker #include <libgsi/libgsi.h>
57*00c7fec1SAndroid Build Coastguard Worker #include <libsnapshot/snapshot.h>
58*00c7fec1SAndroid Build Coastguard Worker #include <logwrap/logwrap.h>
59*00c7fec1SAndroid Build Coastguard Worker #include <processgroup/processgroup.h>
60*00c7fec1SAndroid Build Coastguard Worker #include <processgroup/setup.h>
61*00c7fec1SAndroid Build Coastguard Worker #include <selinux/android.h>
62*00c7fec1SAndroid Build Coastguard Worker #include <unwindstack/AndroidUnwinder.h>
63*00c7fec1SAndroid Build Coastguard Worker 
64*00c7fec1SAndroid Build Coastguard Worker #include "action.h"
65*00c7fec1SAndroid Build Coastguard Worker #include "action_manager.h"
66*00c7fec1SAndroid Build Coastguard Worker #include "action_parser.h"
67*00c7fec1SAndroid Build Coastguard Worker #include "apex_init_util.h"
68*00c7fec1SAndroid Build Coastguard Worker #include "epoll.h"
69*00c7fec1SAndroid Build Coastguard Worker #include "first_stage_init.h"
70*00c7fec1SAndroid Build Coastguard Worker #include "first_stage_mount.h"
71*00c7fec1SAndroid Build Coastguard Worker #include "import_parser.h"
72*00c7fec1SAndroid Build Coastguard Worker #include "keychords.h"
73*00c7fec1SAndroid Build Coastguard Worker #include "lmkd_service.h"
74*00c7fec1SAndroid Build Coastguard Worker #include "mount_handler.h"
75*00c7fec1SAndroid Build Coastguard Worker #include "mount_namespace.h"
76*00c7fec1SAndroid Build Coastguard Worker #include "property_service.h"
77*00c7fec1SAndroid Build Coastguard Worker #include "proto_utils.h"
78*00c7fec1SAndroid Build Coastguard Worker #include "reboot.h"
79*00c7fec1SAndroid Build Coastguard Worker #include "reboot_utils.h"
80*00c7fec1SAndroid Build Coastguard Worker #include "second_stage_resources.h"
81*00c7fec1SAndroid Build Coastguard Worker #include "security.h"
82*00c7fec1SAndroid Build Coastguard Worker #include "selabel.h"
83*00c7fec1SAndroid Build Coastguard Worker #include "selinux.h"
84*00c7fec1SAndroid Build Coastguard Worker #include "service.h"
85*00c7fec1SAndroid Build Coastguard Worker #include "service_list.h"
86*00c7fec1SAndroid Build Coastguard Worker #include "service_parser.h"
87*00c7fec1SAndroid Build Coastguard Worker #include "sigchld_handler.h"
88*00c7fec1SAndroid Build Coastguard Worker #include "snapuserd_transition.h"
89*00c7fec1SAndroid Build Coastguard Worker #include "subcontext.h"
90*00c7fec1SAndroid Build Coastguard Worker #include "system/core/init/property_service.pb.h"
91*00c7fec1SAndroid Build Coastguard Worker #include "util.h"
92*00c7fec1SAndroid Build Coastguard Worker 
93*00c7fec1SAndroid Build Coastguard Worker #ifndef RECOVERY
94*00c7fec1SAndroid Build Coastguard Worker #include "com_android_apex.h"
95*00c7fec1SAndroid Build Coastguard Worker #endif  // RECOVERY
96*00c7fec1SAndroid Build Coastguard Worker 
97*00c7fec1SAndroid Build Coastguard Worker using namespace std::chrono_literals;
98*00c7fec1SAndroid Build Coastguard Worker using namespace std::string_literals;
99*00c7fec1SAndroid Build Coastguard Worker 
100*00c7fec1SAndroid Build Coastguard Worker using android::base::boot_clock;
101*00c7fec1SAndroid Build Coastguard Worker using android::base::ConsumePrefix;
102*00c7fec1SAndroid Build Coastguard Worker using android::base::GetProperty;
103*00c7fec1SAndroid Build Coastguard Worker using android::base::ReadFileToString;
104*00c7fec1SAndroid Build Coastguard Worker using android::base::SetProperty;
105*00c7fec1SAndroid Build Coastguard Worker using android::base::StringPrintf;
106*00c7fec1SAndroid Build Coastguard Worker using android::base::Timer;
107*00c7fec1SAndroid Build Coastguard Worker using android::base::Trim;
108*00c7fec1SAndroid Build Coastguard Worker using android::base::unique_fd;
109*00c7fec1SAndroid Build Coastguard Worker using android::fs_mgr::AvbHandle;
110*00c7fec1SAndroid Build Coastguard Worker using android::snapshot::SnapshotManager;
111*00c7fec1SAndroid Build Coastguard Worker 
112*00c7fec1SAndroid Build Coastguard Worker namespace android {
113*00c7fec1SAndroid Build Coastguard Worker namespace init {
114*00c7fec1SAndroid Build Coastguard Worker 
115*00c7fec1SAndroid Build Coastguard Worker static int property_triggers_enabled = 0;
116*00c7fec1SAndroid Build Coastguard Worker 
117*00c7fec1SAndroid Build Coastguard Worker static int sigterm_fd = -1;
118*00c7fec1SAndroid Build Coastguard Worker static int property_fd = -1;
119*00c7fec1SAndroid Build Coastguard Worker 
120*00c7fec1SAndroid Build Coastguard Worker struct PendingControlMessage {
121*00c7fec1SAndroid Build Coastguard Worker     std::string message;
122*00c7fec1SAndroid Build Coastguard Worker     std::string name;
123*00c7fec1SAndroid Build Coastguard Worker     pid_t pid;
124*00c7fec1SAndroid Build Coastguard Worker     int fd;
125*00c7fec1SAndroid Build Coastguard Worker };
126*00c7fec1SAndroid Build Coastguard Worker static std::mutex pending_control_messages_lock;
127*00c7fec1SAndroid Build Coastguard Worker static std::queue<PendingControlMessage> pending_control_messages;
128*00c7fec1SAndroid Build Coastguard Worker 
129*00c7fec1SAndroid Build Coastguard Worker // Init epolls various FDs to wait for various inputs.  It previously waited on property changes
130*00c7fec1SAndroid Build Coastguard Worker // with a blocking socket that contained the information related to the change, however, it was easy
131*00c7fec1SAndroid Build Coastguard Worker // to fill that socket and deadlock the system.  Now we use locks to handle the property changes
132*00c7fec1SAndroid Build Coastguard Worker // directly in the property thread, however we still must wake the epoll to inform init that there
133*00c7fec1SAndroid Build Coastguard Worker // is a change to process, so we use this FD.  It is non-blocking, since we do not care how many
134*00c7fec1SAndroid Build Coastguard Worker // times WakeMainInitThread() is called, only that the epoll will wake.
135*00c7fec1SAndroid Build Coastguard Worker static int wake_main_thread_fd = -1;
InstallInitNotifier(Epoll * epoll)136*00c7fec1SAndroid Build Coastguard Worker static void InstallInitNotifier(Epoll* epoll) {
137*00c7fec1SAndroid Build Coastguard Worker     wake_main_thread_fd = eventfd(0, EFD_CLOEXEC);
138*00c7fec1SAndroid Build Coastguard Worker     if (wake_main_thread_fd == -1) {
139*00c7fec1SAndroid Build Coastguard Worker         PLOG(FATAL) << "Failed to create eventfd for waking init";
140*00c7fec1SAndroid Build Coastguard Worker     }
141*00c7fec1SAndroid Build Coastguard Worker     auto clear_eventfd = [] {
142*00c7fec1SAndroid Build Coastguard Worker         uint64_t counter;
143*00c7fec1SAndroid Build Coastguard Worker         TEMP_FAILURE_RETRY(read(wake_main_thread_fd, &counter, sizeof(counter)));
144*00c7fec1SAndroid Build Coastguard Worker     };
145*00c7fec1SAndroid Build Coastguard Worker 
146*00c7fec1SAndroid Build Coastguard Worker     if (auto result = epoll->RegisterHandler(wake_main_thread_fd, clear_eventfd); !result.ok()) {
147*00c7fec1SAndroid Build Coastguard Worker         LOG(FATAL) << result.error();
148*00c7fec1SAndroid Build Coastguard Worker     }
149*00c7fec1SAndroid Build Coastguard Worker }
150*00c7fec1SAndroid Build Coastguard Worker 
WakeMainInitThread()151*00c7fec1SAndroid Build Coastguard Worker static void WakeMainInitThread() {
152*00c7fec1SAndroid Build Coastguard Worker     uint64_t counter = 1;
153*00c7fec1SAndroid Build Coastguard Worker     TEMP_FAILURE_RETRY(write(wake_main_thread_fd, &counter, sizeof(counter)));
154*00c7fec1SAndroid Build Coastguard Worker }
155*00c7fec1SAndroid Build Coastguard Worker 
156*00c7fec1SAndroid Build Coastguard Worker static class PropWaiterState {
157*00c7fec1SAndroid Build Coastguard Worker   public:
StartWaiting(const char * name,const char * value)158*00c7fec1SAndroid Build Coastguard Worker     bool StartWaiting(const char* name, const char* value) {
159*00c7fec1SAndroid Build Coastguard Worker         auto lock = std::lock_guard{lock_};
160*00c7fec1SAndroid Build Coastguard Worker         if (waiting_for_prop_) {
161*00c7fec1SAndroid Build Coastguard Worker             return false;
162*00c7fec1SAndroid Build Coastguard Worker         }
163*00c7fec1SAndroid Build Coastguard Worker         if (GetProperty(name, "") != value) {
164*00c7fec1SAndroid Build Coastguard Worker             // Current property value is not equal to expected value
165*00c7fec1SAndroid Build Coastguard Worker             wait_prop_name_ = name;
166*00c7fec1SAndroid Build Coastguard Worker             wait_prop_value_ = value;
167*00c7fec1SAndroid Build Coastguard Worker             waiting_for_prop_.reset(new Timer());
168*00c7fec1SAndroid Build Coastguard Worker         } else {
169*00c7fec1SAndroid Build Coastguard Worker             LOG(INFO) << "start_waiting_for_property(\"" << name << "\", \"" << value
170*00c7fec1SAndroid Build Coastguard Worker                       << "\"): already set";
171*00c7fec1SAndroid Build Coastguard Worker         }
172*00c7fec1SAndroid Build Coastguard Worker         return true;
173*00c7fec1SAndroid Build Coastguard Worker     }
174*00c7fec1SAndroid Build Coastguard Worker 
ResetWaitForProp()175*00c7fec1SAndroid Build Coastguard Worker     void ResetWaitForProp() {
176*00c7fec1SAndroid Build Coastguard Worker         auto lock = std::lock_guard{lock_};
177*00c7fec1SAndroid Build Coastguard Worker         ResetWaitForPropLocked();
178*00c7fec1SAndroid Build Coastguard Worker     }
179*00c7fec1SAndroid Build Coastguard Worker 
CheckAndResetWait(const std::string & name,const std::string & value)180*00c7fec1SAndroid Build Coastguard Worker     void CheckAndResetWait(const std::string& name, const std::string& value) {
181*00c7fec1SAndroid Build Coastguard Worker         auto lock = std::lock_guard{lock_};
182*00c7fec1SAndroid Build Coastguard Worker         // We always record how long init waited for ueventd to tell us cold boot finished.
183*00c7fec1SAndroid Build Coastguard Worker         // If we aren't waiting on this property, it means that ueventd finished before we even
184*00c7fec1SAndroid Build Coastguard Worker         // started to wait.
185*00c7fec1SAndroid Build Coastguard Worker         if (name == kColdBootDoneProp) {
186*00c7fec1SAndroid Build Coastguard Worker             auto time_waited = waiting_for_prop_ ? waiting_for_prop_->duration().count() : 0;
187*00c7fec1SAndroid Build Coastguard Worker             std::thread([time_waited] {
188*00c7fec1SAndroid Build Coastguard Worker                 SetProperty("ro.boottime.init.cold_boot_wait", std::to_string(time_waited));
189*00c7fec1SAndroid Build Coastguard Worker             }).detach();
190*00c7fec1SAndroid Build Coastguard Worker         }
191*00c7fec1SAndroid Build Coastguard Worker 
192*00c7fec1SAndroid Build Coastguard Worker         if (waiting_for_prop_) {
193*00c7fec1SAndroid Build Coastguard Worker             if (wait_prop_name_ == name && wait_prop_value_ == value) {
194*00c7fec1SAndroid Build Coastguard Worker                 LOG(INFO) << "Wait for property '" << wait_prop_name_ << "=" << wait_prop_value_
195*00c7fec1SAndroid Build Coastguard Worker                           << "' took " << *waiting_for_prop_;
196*00c7fec1SAndroid Build Coastguard Worker                 ResetWaitForPropLocked();
197*00c7fec1SAndroid Build Coastguard Worker                 WakeMainInitThread();
198*00c7fec1SAndroid Build Coastguard Worker             }
199*00c7fec1SAndroid Build Coastguard Worker         }
200*00c7fec1SAndroid Build Coastguard Worker     }
201*00c7fec1SAndroid Build Coastguard Worker 
202*00c7fec1SAndroid Build Coastguard Worker     // This is not thread safe because it releases the lock when it returns, so the waiting state
203*00c7fec1SAndroid Build Coastguard Worker     // may change.  However, we only use this function to prevent running commands in the main
204*00c7fec1SAndroid Build Coastguard Worker     // thread loop when we are waiting, so we do not care about false positives; only false
205*00c7fec1SAndroid Build Coastguard Worker     // negatives.  StartWaiting() and this function are always called from the same thread, so false
206*00c7fec1SAndroid Build Coastguard Worker     // negatives are not possible and therefore we're okay.
MightBeWaiting()207*00c7fec1SAndroid Build Coastguard Worker     bool MightBeWaiting() {
208*00c7fec1SAndroid Build Coastguard Worker         auto lock = std::lock_guard{lock_};
209*00c7fec1SAndroid Build Coastguard Worker         return static_cast<bool>(waiting_for_prop_);
210*00c7fec1SAndroid Build Coastguard Worker     }
211*00c7fec1SAndroid Build Coastguard Worker 
212*00c7fec1SAndroid Build Coastguard Worker   private:
ResetWaitForPropLocked()213*00c7fec1SAndroid Build Coastguard Worker     void ResetWaitForPropLocked() EXCLUSIVE_LOCKS_REQUIRED(lock_) {
214*00c7fec1SAndroid Build Coastguard Worker         wait_prop_name_.clear();
215*00c7fec1SAndroid Build Coastguard Worker         wait_prop_value_.clear();
216*00c7fec1SAndroid Build Coastguard Worker         waiting_for_prop_.reset();
217*00c7fec1SAndroid Build Coastguard Worker     }
218*00c7fec1SAndroid Build Coastguard Worker 
219*00c7fec1SAndroid Build Coastguard Worker     std::mutex lock_;
GUARDED_BY(lock_)220*00c7fec1SAndroid Build Coastguard Worker     GUARDED_BY(lock_) std::unique_ptr<Timer> waiting_for_prop_{nullptr};
221*00c7fec1SAndroid Build Coastguard Worker     GUARDED_BY(lock_) std::string wait_prop_name_;
222*00c7fec1SAndroid Build Coastguard Worker     GUARDED_BY(lock_) std::string wait_prop_value_;
223*00c7fec1SAndroid Build Coastguard Worker 
224*00c7fec1SAndroid Build Coastguard Worker } prop_waiter_state;
225*00c7fec1SAndroid Build Coastguard Worker 
start_waiting_for_property(const char * name,const char * value)226*00c7fec1SAndroid Build Coastguard Worker bool start_waiting_for_property(const char* name, const char* value) {
227*00c7fec1SAndroid Build Coastguard Worker     return prop_waiter_state.StartWaiting(name, value);
228*00c7fec1SAndroid Build Coastguard Worker }
229*00c7fec1SAndroid Build Coastguard Worker 
ResetWaitForProp()230*00c7fec1SAndroid Build Coastguard Worker void ResetWaitForProp() {
231*00c7fec1SAndroid Build Coastguard Worker     prop_waiter_state.ResetWaitForProp();
232*00c7fec1SAndroid Build Coastguard Worker }
233*00c7fec1SAndroid Build Coastguard Worker 
234*00c7fec1SAndroid Build Coastguard Worker static class ShutdownState {
235*00c7fec1SAndroid Build Coastguard Worker   public:
TriggerShutdown(const std::string & command)236*00c7fec1SAndroid Build Coastguard Worker     void TriggerShutdown(const std::string& command) {
237*00c7fec1SAndroid Build Coastguard Worker         // We can't call HandlePowerctlMessage() directly in this function,
238*00c7fec1SAndroid Build Coastguard Worker         // because it modifies the contents of the action queue, which can cause the action queue
239*00c7fec1SAndroid Build Coastguard Worker         // to get into a bad state if this function is called from a command being executed by the
240*00c7fec1SAndroid Build Coastguard Worker         // action queue.  Instead we set this flag and ensure that shutdown happens before the next
241*00c7fec1SAndroid Build Coastguard Worker         // command is run in the main init loop.
242*00c7fec1SAndroid Build Coastguard Worker         auto lock = std::lock_guard{shutdown_command_lock_};
243*00c7fec1SAndroid Build Coastguard Worker         shutdown_command_ = command;
244*00c7fec1SAndroid Build Coastguard Worker         do_shutdown_ = true;
245*00c7fec1SAndroid Build Coastguard Worker         WakeMainInitThread();
246*00c7fec1SAndroid Build Coastguard Worker     }
247*00c7fec1SAndroid Build Coastguard Worker 
CheckShutdown()248*00c7fec1SAndroid Build Coastguard Worker     std::optional<std::string> CheckShutdown() __attribute__((warn_unused_result)) {
249*00c7fec1SAndroid Build Coastguard Worker         auto lock = std::lock_guard{shutdown_command_lock_};
250*00c7fec1SAndroid Build Coastguard Worker         if (do_shutdown_ && !IsShuttingDown()) {
251*00c7fec1SAndroid Build Coastguard Worker             do_shutdown_ = false;
252*00c7fec1SAndroid Build Coastguard Worker             return shutdown_command_;
253*00c7fec1SAndroid Build Coastguard Worker         }
254*00c7fec1SAndroid Build Coastguard Worker         return {};
255*00c7fec1SAndroid Build Coastguard Worker     }
256*00c7fec1SAndroid Build Coastguard Worker 
257*00c7fec1SAndroid Build Coastguard Worker   private:
258*00c7fec1SAndroid Build Coastguard Worker     std::mutex shutdown_command_lock_;
259*00c7fec1SAndroid Build Coastguard Worker     std::string shutdown_command_ GUARDED_BY(shutdown_command_lock_);
260*00c7fec1SAndroid Build Coastguard Worker     bool do_shutdown_ = false;
261*00c7fec1SAndroid Build Coastguard Worker } shutdown_state;
262*00c7fec1SAndroid Build Coastguard Worker 
DumpState()263*00c7fec1SAndroid Build Coastguard Worker void DumpState() {
264*00c7fec1SAndroid Build Coastguard Worker     ServiceList::GetInstance().DumpState();
265*00c7fec1SAndroid Build Coastguard Worker     ActionManager::GetInstance().DumpState();
266*00c7fec1SAndroid Build Coastguard Worker }
267*00c7fec1SAndroid Build Coastguard Worker 
CreateParser(ActionManager & action_manager,ServiceList & service_list)268*00c7fec1SAndroid Build Coastguard Worker Parser CreateParser(ActionManager& action_manager, ServiceList& service_list) {
269*00c7fec1SAndroid Build Coastguard Worker     Parser parser;
270*00c7fec1SAndroid Build Coastguard Worker 
271*00c7fec1SAndroid Build Coastguard Worker     parser.AddSectionParser("service",
272*00c7fec1SAndroid Build Coastguard Worker                             std::make_unique<ServiceParser>(&service_list, GetSubcontext()));
273*00c7fec1SAndroid Build Coastguard Worker     parser.AddSectionParser("on", std::make_unique<ActionParser>(&action_manager, GetSubcontext()));
274*00c7fec1SAndroid Build Coastguard Worker     parser.AddSectionParser("import", std::make_unique<ImportParser>(&parser));
275*00c7fec1SAndroid Build Coastguard Worker 
276*00c7fec1SAndroid Build Coastguard Worker     return parser;
277*00c7fec1SAndroid Build Coastguard Worker }
278*00c7fec1SAndroid Build Coastguard Worker 
279*00c7fec1SAndroid Build Coastguard Worker #ifndef RECOVERY
280*00c7fec1SAndroid Build Coastguard Worker template <typename T>
281*00c7fec1SAndroid Build Coastguard Worker struct LibXmlErrorHandler {
282*00c7fec1SAndroid Build Coastguard Worker     T handler_;
283*00c7fec1SAndroid Build Coastguard Worker     template <typename Handler>
LibXmlErrorHandlerandroid::init::LibXmlErrorHandler284*00c7fec1SAndroid Build Coastguard Worker     LibXmlErrorHandler(Handler&& handler) : handler_(std::move(handler)) {
285*00c7fec1SAndroid Build Coastguard Worker         xmlSetGenericErrorFunc(nullptr, &ErrorHandler);
286*00c7fec1SAndroid Build Coastguard Worker     }
~LibXmlErrorHandlerandroid::init::LibXmlErrorHandler287*00c7fec1SAndroid Build Coastguard Worker     ~LibXmlErrorHandler() { xmlSetGenericErrorFunc(nullptr, nullptr); }
ErrorHandlerandroid::init::LibXmlErrorHandler288*00c7fec1SAndroid Build Coastguard Worker     static void ErrorHandler(void*, const char* msg, ...) {
289*00c7fec1SAndroid Build Coastguard Worker         va_list args;
290*00c7fec1SAndroid Build Coastguard Worker         va_start(args, msg);
291*00c7fec1SAndroid Build Coastguard Worker         char* formatted;
292*00c7fec1SAndroid Build Coastguard Worker         if (vasprintf(&formatted, msg, args) >= 0) {
293*00c7fec1SAndroid Build Coastguard Worker             LOG(ERROR) << formatted;
294*00c7fec1SAndroid Build Coastguard Worker         }
295*00c7fec1SAndroid Build Coastguard Worker         free(formatted);
296*00c7fec1SAndroid Build Coastguard Worker         va_end(args);
297*00c7fec1SAndroid Build Coastguard Worker     }
298*00c7fec1SAndroid Build Coastguard Worker };
299*00c7fec1SAndroid Build Coastguard Worker 
300*00c7fec1SAndroid Build Coastguard Worker template <typename Handler>
301*00c7fec1SAndroid Build Coastguard Worker LibXmlErrorHandler(Handler&&) -> LibXmlErrorHandler<Handler>;
302*00c7fec1SAndroid Build Coastguard Worker #endif  // RECOVERY
303*00c7fec1SAndroid Build Coastguard Worker 
304*00c7fec1SAndroid Build Coastguard Worker // Returns a Parser that accepts scripts from APEX modules. It supports `service` and `on`.
CreateApexConfigParser(ActionManager & action_manager,ServiceList & service_list)305*00c7fec1SAndroid Build Coastguard Worker Parser CreateApexConfigParser(ActionManager& action_manager, ServiceList& service_list) {
306*00c7fec1SAndroid Build Coastguard Worker     Parser parser;
307*00c7fec1SAndroid Build Coastguard Worker     auto subcontext = GetSubcontext();
308*00c7fec1SAndroid Build Coastguard Worker #ifndef RECOVERY
309*00c7fec1SAndroid Build Coastguard Worker     if (subcontext) {
310*00c7fec1SAndroid Build Coastguard Worker         const auto apex_info_list_file = "/apex/apex-info-list.xml";
311*00c7fec1SAndroid Build Coastguard Worker         auto error_handler = LibXmlErrorHandler([&](const auto& error_message) {
312*00c7fec1SAndroid Build Coastguard Worker             LOG(ERROR) << "Failed to read " << apex_info_list_file << ":" << error_message;
313*00c7fec1SAndroid Build Coastguard Worker         });
314*00c7fec1SAndroid Build Coastguard Worker         const auto apex_info_list = com::android::apex::readApexInfoList(apex_info_list_file);
315*00c7fec1SAndroid Build Coastguard Worker         if (apex_info_list.has_value()) {
316*00c7fec1SAndroid Build Coastguard Worker             std::vector<std::string> subcontext_apexes;
317*00c7fec1SAndroid Build Coastguard Worker             for (const auto& info : apex_info_list->getApexInfo()) {
318*00c7fec1SAndroid Build Coastguard Worker                 if (subcontext->PartitionMatchesSubcontext(info.getPartition())) {
319*00c7fec1SAndroid Build Coastguard Worker                     subcontext_apexes.push_back(info.getModuleName());
320*00c7fec1SAndroid Build Coastguard Worker                 }
321*00c7fec1SAndroid Build Coastguard Worker             }
322*00c7fec1SAndroid Build Coastguard Worker             subcontext->SetApexList(std::move(subcontext_apexes));
323*00c7fec1SAndroid Build Coastguard Worker         }
324*00c7fec1SAndroid Build Coastguard Worker     }
325*00c7fec1SAndroid Build Coastguard Worker #endif  // RECOVERY
326*00c7fec1SAndroid Build Coastguard Worker     parser.AddSectionParser("service", std::make_unique<ServiceParser>(&service_list, subcontext));
327*00c7fec1SAndroid Build Coastguard Worker     parser.AddSectionParser("on", std::make_unique<ActionParser>(&action_manager, subcontext));
328*00c7fec1SAndroid Build Coastguard Worker 
329*00c7fec1SAndroid Build Coastguard Worker     return parser;
330*00c7fec1SAndroid Build Coastguard Worker }
331*00c7fec1SAndroid Build Coastguard Worker 
LoadBootScripts(ActionManager & action_manager,ServiceList & service_list)332*00c7fec1SAndroid Build Coastguard Worker static void LoadBootScripts(ActionManager& action_manager, ServiceList& service_list) {
333*00c7fec1SAndroid Build Coastguard Worker     Parser parser = CreateParser(action_manager, service_list);
334*00c7fec1SAndroid Build Coastguard Worker 
335*00c7fec1SAndroid Build Coastguard Worker     std::string bootscript = GetProperty("ro.boot.init_rc", "");
336*00c7fec1SAndroid Build Coastguard Worker     if (bootscript.empty()) {
337*00c7fec1SAndroid Build Coastguard Worker         parser.ParseConfig("/system/etc/init/hw/init.rc");
338*00c7fec1SAndroid Build Coastguard Worker         if (!parser.ParseConfig("/system/etc/init")) {
339*00c7fec1SAndroid Build Coastguard Worker             late_import_paths.emplace_back("/system/etc/init");
340*00c7fec1SAndroid Build Coastguard Worker         }
341*00c7fec1SAndroid Build Coastguard Worker         // late_import is available only in Q and earlier release. As we don't
342*00c7fec1SAndroid Build Coastguard Worker         // have system_ext in those versions, skip late_import for system_ext.
343*00c7fec1SAndroid Build Coastguard Worker         parser.ParseConfig("/system_ext/etc/init");
344*00c7fec1SAndroid Build Coastguard Worker         if (!parser.ParseConfig("/vendor/etc/init")) {
345*00c7fec1SAndroid Build Coastguard Worker             late_import_paths.emplace_back("/vendor/etc/init");
346*00c7fec1SAndroid Build Coastguard Worker         }
347*00c7fec1SAndroid Build Coastguard Worker         if (!parser.ParseConfig("/odm/etc/init")) {
348*00c7fec1SAndroid Build Coastguard Worker             late_import_paths.emplace_back("/odm/etc/init");
349*00c7fec1SAndroid Build Coastguard Worker         }
350*00c7fec1SAndroid Build Coastguard Worker         if (!parser.ParseConfig("/product/etc/init")) {
351*00c7fec1SAndroid Build Coastguard Worker             late_import_paths.emplace_back("/product/etc/init");
352*00c7fec1SAndroid Build Coastguard Worker         }
353*00c7fec1SAndroid Build Coastguard Worker     } else {
354*00c7fec1SAndroid Build Coastguard Worker         parser.ParseConfig(bootscript);
355*00c7fec1SAndroid Build Coastguard Worker     }
356*00c7fec1SAndroid Build Coastguard Worker }
357*00c7fec1SAndroid Build Coastguard Worker 
PropertyChanged(const std::string & name,const std::string & value)358*00c7fec1SAndroid Build Coastguard Worker void PropertyChanged(const std::string& name, const std::string& value) {
359*00c7fec1SAndroid Build Coastguard Worker     // If the property is sys.powerctl, we bypass the event queue and immediately handle it.
360*00c7fec1SAndroid Build Coastguard Worker     // This is to ensure that init will always and immediately shutdown/reboot, regardless of
361*00c7fec1SAndroid Build Coastguard Worker     // if there are other pending events to process or if init is waiting on an exec service or
362*00c7fec1SAndroid Build Coastguard Worker     // waiting on a property.
363*00c7fec1SAndroid Build Coastguard Worker     // In non-thermal-shutdown case, 'shutdown' trigger will be fired to let device specific
364*00c7fec1SAndroid Build Coastguard Worker     // commands to be executed.
365*00c7fec1SAndroid Build Coastguard Worker     if (name == "sys.powerctl") {
366*00c7fec1SAndroid Build Coastguard Worker         trigger_shutdown(value);
367*00c7fec1SAndroid Build Coastguard Worker     }
368*00c7fec1SAndroid Build Coastguard Worker 
369*00c7fec1SAndroid Build Coastguard Worker     if (property_triggers_enabled) {
370*00c7fec1SAndroid Build Coastguard Worker         ActionManager::GetInstance().QueuePropertyChange(name, value);
371*00c7fec1SAndroid Build Coastguard Worker         WakeMainInitThread();
372*00c7fec1SAndroid Build Coastguard Worker     }
373*00c7fec1SAndroid Build Coastguard Worker 
374*00c7fec1SAndroid Build Coastguard Worker     prop_waiter_state.CheckAndResetWait(name, value);
375*00c7fec1SAndroid Build Coastguard Worker }
376*00c7fec1SAndroid Build Coastguard Worker 
HandleProcessActions()377*00c7fec1SAndroid Build Coastguard Worker static std::optional<boot_clock::time_point> HandleProcessActions() {
378*00c7fec1SAndroid Build Coastguard Worker     std::optional<boot_clock::time_point> next_process_action_time;
379*00c7fec1SAndroid Build Coastguard Worker     for (const auto& s : ServiceList::GetInstance()) {
380*00c7fec1SAndroid Build Coastguard Worker         if ((s->flags() & SVC_RUNNING) && s->timeout_period()) {
381*00c7fec1SAndroid Build Coastguard Worker             auto timeout_time = s->time_started() + *s->timeout_period();
382*00c7fec1SAndroid Build Coastguard Worker             if (boot_clock::now() > timeout_time) {
383*00c7fec1SAndroid Build Coastguard Worker                 s->Timeout();
384*00c7fec1SAndroid Build Coastguard Worker             } else {
385*00c7fec1SAndroid Build Coastguard Worker                 if (!next_process_action_time || timeout_time < *next_process_action_time) {
386*00c7fec1SAndroid Build Coastguard Worker                     next_process_action_time = timeout_time;
387*00c7fec1SAndroid Build Coastguard Worker                 }
388*00c7fec1SAndroid Build Coastguard Worker             }
389*00c7fec1SAndroid Build Coastguard Worker         }
390*00c7fec1SAndroid Build Coastguard Worker 
391*00c7fec1SAndroid Build Coastguard Worker         if (!(s->flags() & SVC_RESTARTING)) continue;
392*00c7fec1SAndroid Build Coastguard Worker 
393*00c7fec1SAndroid Build Coastguard Worker         auto restart_time = s->time_started() + s->restart_period();
394*00c7fec1SAndroid Build Coastguard Worker         if (boot_clock::now() > restart_time) {
395*00c7fec1SAndroid Build Coastguard Worker             if (auto result = s->Start(); !result.ok()) {
396*00c7fec1SAndroid Build Coastguard Worker                 LOG(ERROR) << "Could not restart process '" << s->name() << "': " << result.error();
397*00c7fec1SAndroid Build Coastguard Worker             }
398*00c7fec1SAndroid Build Coastguard Worker         } else {
399*00c7fec1SAndroid Build Coastguard Worker             if (!next_process_action_time || restart_time < *next_process_action_time) {
400*00c7fec1SAndroid Build Coastguard Worker                 next_process_action_time = restart_time;
401*00c7fec1SAndroid Build Coastguard Worker             }
402*00c7fec1SAndroid Build Coastguard Worker         }
403*00c7fec1SAndroid Build Coastguard Worker     }
404*00c7fec1SAndroid Build Coastguard Worker     return next_process_action_time;
405*00c7fec1SAndroid Build Coastguard Worker }
406*00c7fec1SAndroid Build Coastguard Worker 
DoControlStart(Service * service)407*00c7fec1SAndroid Build Coastguard Worker static Result<void> DoControlStart(Service* service) {
408*00c7fec1SAndroid Build Coastguard Worker     return service->Start();
409*00c7fec1SAndroid Build Coastguard Worker }
410*00c7fec1SAndroid Build Coastguard Worker 
DoControlStop(Service * service)411*00c7fec1SAndroid Build Coastguard Worker static Result<void> DoControlStop(Service* service) {
412*00c7fec1SAndroid Build Coastguard Worker     service->Stop();
413*00c7fec1SAndroid Build Coastguard Worker     return {};
414*00c7fec1SAndroid Build Coastguard Worker }
415*00c7fec1SAndroid Build Coastguard Worker 
DoControlRestart(Service * service)416*00c7fec1SAndroid Build Coastguard Worker static Result<void> DoControlRestart(Service* service) {
417*00c7fec1SAndroid Build Coastguard Worker     service->Restart();
418*00c7fec1SAndroid Build Coastguard Worker     return {};
419*00c7fec1SAndroid Build Coastguard Worker }
420*00c7fec1SAndroid Build Coastguard Worker 
StopServicesFromApex(const std::string & apex_name)421*00c7fec1SAndroid Build Coastguard Worker int StopServicesFromApex(const std::string& apex_name) {
422*00c7fec1SAndroid Build Coastguard Worker     auto services = ServiceList::GetInstance().FindServicesByApexName(apex_name);
423*00c7fec1SAndroid Build Coastguard Worker     if (services.empty()) {
424*00c7fec1SAndroid Build Coastguard Worker         LOG(INFO) << "No service found for APEX: " << apex_name;
425*00c7fec1SAndroid Build Coastguard Worker         return 0;
426*00c7fec1SAndroid Build Coastguard Worker     }
427*00c7fec1SAndroid Build Coastguard Worker     std::set<std::string> service_names;
428*00c7fec1SAndroid Build Coastguard Worker     for (const auto& service : services) {
429*00c7fec1SAndroid Build Coastguard Worker         service_names.emplace(service->name());
430*00c7fec1SAndroid Build Coastguard Worker     }
431*00c7fec1SAndroid Build Coastguard Worker     constexpr std::chrono::milliseconds kServiceStopTimeout = 10s;
432*00c7fec1SAndroid Build Coastguard Worker     int still_running = StopServicesAndLogViolations(service_names, kServiceStopTimeout,
433*00c7fec1SAndroid Build Coastguard Worker                         true /*SIGTERM*/);
434*00c7fec1SAndroid Build Coastguard Worker     // Send SIGKILL to ones that didn't terminate cleanly.
435*00c7fec1SAndroid Build Coastguard Worker     if (still_running > 0) {
436*00c7fec1SAndroid Build Coastguard Worker         still_running = StopServicesAndLogViolations(service_names, 0ms, false /*SIGKILL*/);
437*00c7fec1SAndroid Build Coastguard Worker     }
438*00c7fec1SAndroid Build Coastguard Worker     return still_running;
439*00c7fec1SAndroid Build Coastguard Worker }
440*00c7fec1SAndroid Build Coastguard Worker 
RemoveServiceAndActionFromApex(const std::string & apex_name)441*00c7fec1SAndroid Build Coastguard Worker void RemoveServiceAndActionFromApex(const std::string& apex_name) {
442*00c7fec1SAndroid Build Coastguard Worker     // Remove services and actions that match apex name
443*00c7fec1SAndroid Build Coastguard Worker     ActionManager::GetInstance().RemoveActionIf([&](const std::unique_ptr<Action>& action) -> bool {
444*00c7fec1SAndroid Build Coastguard Worker         if (GetApexNameFromFileName(action->filename()) == apex_name) {
445*00c7fec1SAndroid Build Coastguard Worker             return true;
446*00c7fec1SAndroid Build Coastguard Worker         }
447*00c7fec1SAndroid Build Coastguard Worker         return false;
448*00c7fec1SAndroid Build Coastguard Worker     });
449*00c7fec1SAndroid Build Coastguard Worker     ServiceList::GetInstance().RemoveServiceIf([&](const std::unique_ptr<Service>& s) -> bool {
450*00c7fec1SAndroid Build Coastguard Worker         if (GetApexNameFromFileName(s->filename()) == apex_name) {
451*00c7fec1SAndroid Build Coastguard Worker             return true;
452*00c7fec1SAndroid Build Coastguard Worker         }
453*00c7fec1SAndroid Build Coastguard Worker         return false;
454*00c7fec1SAndroid Build Coastguard Worker     });
455*00c7fec1SAndroid Build Coastguard Worker }
456*00c7fec1SAndroid Build Coastguard Worker 
DoUnloadApex(const std::string & apex_name)457*00c7fec1SAndroid Build Coastguard Worker static Result<void> DoUnloadApex(const std::string& apex_name) {
458*00c7fec1SAndroid Build Coastguard Worker     if (StopServicesFromApex(apex_name) > 0) {
459*00c7fec1SAndroid Build Coastguard Worker         return Error() << "Unable to stop all service from " << apex_name;
460*00c7fec1SAndroid Build Coastguard Worker     }
461*00c7fec1SAndroid Build Coastguard Worker     RemoveServiceAndActionFromApex(apex_name);
462*00c7fec1SAndroid Build Coastguard Worker     return {};
463*00c7fec1SAndroid Build Coastguard Worker }
464*00c7fec1SAndroid Build Coastguard Worker 
UpdateApexLinkerConfig(const std::string & apex_name)465*00c7fec1SAndroid Build Coastguard Worker static Result<void> UpdateApexLinkerConfig(const std::string& apex_name) {
466*00c7fec1SAndroid Build Coastguard Worker     // Do not invoke linkerconfig when there's no bin/ in the apex.
467*00c7fec1SAndroid Build Coastguard Worker     const std::string bin_path = "/apex/" + apex_name + "/bin";
468*00c7fec1SAndroid Build Coastguard Worker     if (access(bin_path.c_str(), R_OK) != 0) {
469*00c7fec1SAndroid Build Coastguard Worker         return {};
470*00c7fec1SAndroid Build Coastguard Worker     }
471*00c7fec1SAndroid Build Coastguard Worker     const char* linkerconfig_binary = "/apex/com.android.runtime/bin/linkerconfig";
472*00c7fec1SAndroid Build Coastguard Worker     const char* linkerconfig_target = "/linkerconfig";
473*00c7fec1SAndroid Build Coastguard Worker     const char* arguments[] = {linkerconfig_binary, "--target", linkerconfig_target, "--apex",
474*00c7fec1SAndroid Build Coastguard Worker                                apex_name.c_str(),   "--strict"};
475*00c7fec1SAndroid Build Coastguard Worker 
476*00c7fec1SAndroid Build Coastguard Worker     if (logwrap_fork_execvp(arraysize(arguments), arguments, nullptr, false, LOG_KLOG, false,
477*00c7fec1SAndroid Build Coastguard Worker                             nullptr) != 0) {
478*00c7fec1SAndroid Build Coastguard Worker         return ErrnoError() << "failed to execute linkerconfig";
479*00c7fec1SAndroid Build Coastguard Worker     }
480*00c7fec1SAndroid Build Coastguard Worker     LOG(INFO) << "Generated linker configuration for " << apex_name;
481*00c7fec1SAndroid Build Coastguard Worker     return {};
482*00c7fec1SAndroid Build Coastguard Worker }
483*00c7fec1SAndroid Build Coastguard Worker 
DoLoadApex(const std::string & apex_name)484*00c7fec1SAndroid Build Coastguard Worker static Result<void> DoLoadApex(const std::string& apex_name) {
485*00c7fec1SAndroid Build Coastguard Worker     if (auto result = ParseRcScriptsFromApex(apex_name); !result.ok()) {
486*00c7fec1SAndroid Build Coastguard Worker         return result.error();
487*00c7fec1SAndroid Build Coastguard Worker     }
488*00c7fec1SAndroid Build Coastguard Worker 
489*00c7fec1SAndroid Build Coastguard Worker     if (auto result = UpdateApexLinkerConfig(apex_name); !result.ok()) {
490*00c7fec1SAndroid Build Coastguard Worker         return result.error();
491*00c7fec1SAndroid Build Coastguard Worker     }
492*00c7fec1SAndroid Build Coastguard Worker 
493*00c7fec1SAndroid Build Coastguard Worker     return {};
494*00c7fec1SAndroid Build Coastguard Worker }
495*00c7fec1SAndroid Build Coastguard Worker 
496*00c7fec1SAndroid Build Coastguard Worker enum class ControlTarget {
497*00c7fec1SAndroid Build Coastguard Worker     SERVICE,    // function gets called for the named service
498*00c7fec1SAndroid Build Coastguard Worker     INTERFACE,  // action gets called for every service that holds this interface
499*00c7fec1SAndroid Build Coastguard Worker };
500*00c7fec1SAndroid Build Coastguard Worker 
501*00c7fec1SAndroid Build Coastguard Worker using ControlMessageFunction = std::function<Result<void>(Service*)>;
502*00c7fec1SAndroid Build Coastguard Worker 
GetControlMessageMap()503*00c7fec1SAndroid Build Coastguard Worker static const std::map<std::string, ControlMessageFunction, std::less<>>& GetControlMessageMap() {
504*00c7fec1SAndroid Build Coastguard Worker     // clang-format off
505*00c7fec1SAndroid Build Coastguard Worker     static const std::map<std::string, ControlMessageFunction, std::less<>> control_message_functions = {
506*00c7fec1SAndroid Build Coastguard Worker         {"sigstop_on",        [](auto* service) { service->set_sigstop(true); return Result<void>{}; }},
507*00c7fec1SAndroid Build Coastguard Worker         {"sigstop_off",       [](auto* service) { service->set_sigstop(false); return Result<void>{}; }},
508*00c7fec1SAndroid Build Coastguard Worker         {"oneshot_on",        [](auto* service) { service->set_oneshot(true); return Result<void>{}; }},
509*00c7fec1SAndroid Build Coastguard Worker         {"oneshot_off",       [](auto* service) { service->set_oneshot(false); return Result<void>{}; }},
510*00c7fec1SAndroid Build Coastguard Worker         {"start",             DoControlStart},
511*00c7fec1SAndroid Build Coastguard Worker         {"stop",              DoControlStop},
512*00c7fec1SAndroid Build Coastguard Worker         {"restart",           DoControlRestart},
513*00c7fec1SAndroid Build Coastguard Worker     };
514*00c7fec1SAndroid Build Coastguard Worker     // clang-format on
515*00c7fec1SAndroid Build Coastguard Worker 
516*00c7fec1SAndroid Build Coastguard Worker     return control_message_functions;
517*00c7fec1SAndroid Build Coastguard Worker }
518*00c7fec1SAndroid Build Coastguard Worker 
HandleApexControlMessage(std::string_view action,const std::string & name,std::string_view message)519*00c7fec1SAndroid Build Coastguard Worker static Result<void> HandleApexControlMessage(std::string_view action, const std::string& name,
520*00c7fec1SAndroid Build Coastguard Worker                                              std::string_view message) {
521*00c7fec1SAndroid Build Coastguard Worker     if (action == "load") {
522*00c7fec1SAndroid Build Coastguard Worker         return DoLoadApex(name);
523*00c7fec1SAndroid Build Coastguard Worker     } else if (action == "unload") {
524*00c7fec1SAndroid Build Coastguard Worker         return DoUnloadApex(name);
525*00c7fec1SAndroid Build Coastguard Worker     } else {
526*00c7fec1SAndroid Build Coastguard Worker         return Error() << "Unknown control msg '" << message << "'";
527*00c7fec1SAndroid Build Coastguard Worker     }
528*00c7fec1SAndroid Build Coastguard Worker }
529*00c7fec1SAndroid Build Coastguard Worker 
HandleControlMessage(std::string_view message,const std::string & name,pid_t from_pid)530*00c7fec1SAndroid Build Coastguard Worker static bool HandleControlMessage(std::string_view message, const std::string& name,
531*00c7fec1SAndroid Build Coastguard Worker                                  pid_t from_pid) {
532*00c7fec1SAndroid Build Coastguard Worker     std::string cmdline_path = StringPrintf("proc/%d/cmdline", from_pid);
533*00c7fec1SAndroid Build Coastguard Worker     std::string process_cmdline;
534*00c7fec1SAndroid Build Coastguard Worker     if (ReadFileToString(cmdline_path, &process_cmdline)) {
535*00c7fec1SAndroid Build Coastguard Worker         std::replace(process_cmdline.begin(), process_cmdline.end(), '\0', ' ');
536*00c7fec1SAndroid Build Coastguard Worker         process_cmdline = Trim(process_cmdline);
537*00c7fec1SAndroid Build Coastguard Worker     } else {
538*00c7fec1SAndroid Build Coastguard Worker         process_cmdline = "unknown process";
539*00c7fec1SAndroid Build Coastguard Worker     }
540*00c7fec1SAndroid Build Coastguard Worker 
541*00c7fec1SAndroid Build Coastguard Worker     auto action = message;
542*00c7fec1SAndroid Build Coastguard Worker     if (ConsumePrefix(&action, "apex_")) {
543*00c7fec1SAndroid Build Coastguard Worker         if (auto result = HandleApexControlMessage(action, name, message); !result.ok()) {
544*00c7fec1SAndroid Build Coastguard Worker             LOG(ERROR) << "Control message: Could not ctl." << message << " for '" << name
545*00c7fec1SAndroid Build Coastguard Worker                        << "' from pid: " << from_pid << " (" << process_cmdline
546*00c7fec1SAndroid Build Coastguard Worker                        << "): " << result.error();
547*00c7fec1SAndroid Build Coastguard Worker             return false;
548*00c7fec1SAndroid Build Coastguard Worker         }
549*00c7fec1SAndroid Build Coastguard Worker         LOG(INFO) << "Control message: Processed ctl." << message << " for '" << name
550*00c7fec1SAndroid Build Coastguard Worker                   << "' from pid: " << from_pid << " (" << process_cmdline << ")";
551*00c7fec1SAndroid Build Coastguard Worker         return true;
552*00c7fec1SAndroid Build Coastguard Worker     }
553*00c7fec1SAndroid Build Coastguard Worker 
554*00c7fec1SAndroid Build Coastguard Worker     Service* service = nullptr;
555*00c7fec1SAndroid Build Coastguard Worker     if (ConsumePrefix(&action, "interface_")) {
556*00c7fec1SAndroid Build Coastguard Worker         service = ServiceList::GetInstance().FindInterface(name);
557*00c7fec1SAndroid Build Coastguard Worker     } else {
558*00c7fec1SAndroid Build Coastguard Worker         service = ServiceList::GetInstance().FindService(name);
559*00c7fec1SAndroid Build Coastguard Worker     }
560*00c7fec1SAndroid Build Coastguard Worker 
561*00c7fec1SAndroid Build Coastguard Worker     if (service == nullptr) {
562*00c7fec1SAndroid Build Coastguard Worker         LOG(ERROR) << "Control message: Could not find '" << name << "' for ctl." << message
563*00c7fec1SAndroid Build Coastguard Worker                    << " from pid: " << from_pid << " (" << process_cmdline << ")";
564*00c7fec1SAndroid Build Coastguard Worker         return false;
565*00c7fec1SAndroid Build Coastguard Worker     }
566*00c7fec1SAndroid Build Coastguard Worker 
567*00c7fec1SAndroid Build Coastguard Worker     const auto& map = GetControlMessageMap();
568*00c7fec1SAndroid Build Coastguard Worker     const auto it = map.find(action);
569*00c7fec1SAndroid Build Coastguard Worker     if (it == map.end()) {
570*00c7fec1SAndroid Build Coastguard Worker         LOG(ERROR) << "Unknown control msg '" << message << "'";
571*00c7fec1SAndroid Build Coastguard Worker         return false;
572*00c7fec1SAndroid Build Coastguard Worker     }
573*00c7fec1SAndroid Build Coastguard Worker     const auto& function = it->second;
574*00c7fec1SAndroid Build Coastguard Worker 
575*00c7fec1SAndroid Build Coastguard Worker     if (auto result = function(service); !result.ok()) {
576*00c7fec1SAndroid Build Coastguard Worker         LOG(ERROR) << "Control message: Could not ctl." << message << " for '" << name
577*00c7fec1SAndroid Build Coastguard Worker                    << "' from pid: " << from_pid << " (" << process_cmdline
578*00c7fec1SAndroid Build Coastguard Worker                    << "): " << result.error();
579*00c7fec1SAndroid Build Coastguard Worker         return false;
580*00c7fec1SAndroid Build Coastguard Worker     }
581*00c7fec1SAndroid Build Coastguard Worker 
582*00c7fec1SAndroid Build Coastguard Worker     LOG(INFO) << "Control message: Processed ctl." << message << " for '" << name
583*00c7fec1SAndroid Build Coastguard Worker               << "' from pid: " << from_pid << " (" << process_cmdline << ")";
584*00c7fec1SAndroid Build Coastguard Worker     return true;
585*00c7fec1SAndroid Build Coastguard Worker }
586*00c7fec1SAndroid Build Coastguard Worker 
QueueControlMessage(const std::string & message,const std::string & name,pid_t pid,int fd)587*00c7fec1SAndroid Build Coastguard Worker bool QueueControlMessage(const std::string& message, const std::string& name, pid_t pid, int fd) {
588*00c7fec1SAndroid Build Coastguard Worker     auto lock = std::lock_guard{pending_control_messages_lock};
589*00c7fec1SAndroid Build Coastguard Worker     if (pending_control_messages.size() > 100) {
590*00c7fec1SAndroid Build Coastguard Worker         LOG(ERROR) << "Too many pending control messages, dropped '" << message << "' for '" << name
591*00c7fec1SAndroid Build Coastguard Worker                    << "' from pid: " << pid;
592*00c7fec1SAndroid Build Coastguard Worker         return false;
593*00c7fec1SAndroid Build Coastguard Worker     }
594*00c7fec1SAndroid Build Coastguard Worker     pending_control_messages.push({message, name, pid, fd});
595*00c7fec1SAndroid Build Coastguard Worker     WakeMainInitThread();
596*00c7fec1SAndroid Build Coastguard Worker     return true;
597*00c7fec1SAndroid Build Coastguard Worker }
598*00c7fec1SAndroid Build Coastguard Worker 
HandleControlMessages()599*00c7fec1SAndroid Build Coastguard Worker static void HandleControlMessages() {
600*00c7fec1SAndroid Build Coastguard Worker     auto lock = std::unique_lock{pending_control_messages_lock};
601*00c7fec1SAndroid Build Coastguard Worker     // Init historically would only execute handle one property message, including control messages
602*00c7fec1SAndroid Build Coastguard Worker     // in each iteration of its main loop.  We retain this behavior here to prevent starvation of
603*00c7fec1SAndroid Build Coastguard Worker     // other actions in the main loop.
604*00c7fec1SAndroid Build Coastguard Worker     if (!pending_control_messages.empty()) {
605*00c7fec1SAndroid Build Coastguard Worker         auto control_message = pending_control_messages.front();
606*00c7fec1SAndroid Build Coastguard Worker         pending_control_messages.pop();
607*00c7fec1SAndroid Build Coastguard Worker         lock.unlock();
608*00c7fec1SAndroid Build Coastguard Worker 
609*00c7fec1SAndroid Build Coastguard Worker         bool success = HandleControlMessage(control_message.message, control_message.name,
610*00c7fec1SAndroid Build Coastguard Worker                                             control_message.pid);
611*00c7fec1SAndroid Build Coastguard Worker 
612*00c7fec1SAndroid Build Coastguard Worker         uint32_t response = success ? PROP_SUCCESS : PROP_ERROR_HANDLE_CONTROL_MESSAGE;
613*00c7fec1SAndroid Build Coastguard Worker         if (control_message.fd != -1) {
614*00c7fec1SAndroid Build Coastguard Worker             TEMP_FAILURE_RETRY(send(control_message.fd, &response, sizeof(response), 0));
615*00c7fec1SAndroid Build Coastguard Worker             close(control_message.fd);
616*00c7fec1SAndroid Build Coastguard Worker         }
617*00c7fec1SAndroid Build Coastguard Worker         lock.lock();
618*00c7fec1SAndroid Build Coastguard Worker     }
619*00c7fec1SAndroid Build Coastguard Worker     // If we still have items to process, make sure we wake back up to do so.
620*00c7fec1SAndroid Build Coastguard Worker     if (!pending_control_messages.empty()) {
621*00c7fec1SAndroid Build Coastguard Worker         WakeMainInitThread();
622*00c7fec1SAndroid Build Coastguard Worker     }
623*00c7fec1SAndroid Build Coastguard Worker }
624*00c7fec1SAndroid Build Coastguard Worker 
wait_for_coldboot_done_action(const BuiltinArguments & args)625*00c7fec1SAndroid Build Coastguard Worker static Result<void> wait_for_coldboot_done_action(const BuiltinArguments& args) {
626*00c7fec1SAndroid Build Coastguard Worker     if (!prop_waiter_state.StartWaiting(kColdBootDoneProp, "true")) {
627*00c7fec1SAndroid Build Coastguard Worker         LOG(FATAL) << "Could not wait for '" << kColdBootDoneProp << "'";
628*00c7fec1SAndroid Build Coastguard Worker     }
629*00c7fec1SAndroid Build Coastguard Worker 
630*00c7fec1SAndroid Build Coastguard Worker     return {};
631*00c7fec1SAndroid Build Coastguard Worker }
632*00c7fec1SAndroid Build Coastguard Worker 
SetupCgroupsAction(const BuiltinArguments &)633*00c7fec1SAndroid Build Coastguard Worker static Result<void> SetupCgroupsAction(const BuiltinArguments&) {
634*00c7fec1SAndroid Build Coastguard Worker     if (!CgroupsAvailable()) {
635*00c7fec1SAndroid Build Coastguard Worker         LOG(INFO) << "Cgroups support in kernel is not enabled";
636*00c7fec1SAndroid Build Coastguard Worker         return {};
637*00c7fec1SAndroid Build Coastguard Worker     }
638*00c7fec1SAndroid Build Coastguard Worker     if (!CgroupSetup()) {
639*00c7fec1SAndroid Build Coastguard Worker         return ErrnoError() << "Failed to setup cgroups";
640*00c7fec1SAndroid Build Coastguard Worker     }
641*00c7fec1SAndroid Build Coastguard Worker 
642*00c7fec1SAndroid Build Coastguard Worker     return {};
643*00c7fec1SAndroid Build Coastguard Worker }
644*00c7fec1SAndroid Build Coastguard Worker 
export_oem_lock_status()645*00c7fec1SAndroid Build Coastguard Worker static void export_oem_lock_status() {
646*00c7fec1SAndroid Build Coastguard Worker     if (!android::base::GetBoolProperty("ro.oem_unlock_supported", false)) {
647*00c7fec1SAndroid Build Coastguard Worker         return;
648*00c7fec1SAndroid Build Coastguard Worker     }
649*00c7fec1SAndroid Build Coastguard Worker     SetProperty(
650*00c7fec1SAndroid Build Coastguard Worker             "ro.boot.flash.locked",
651*00c7fec1SAndroid Build Coastguard Worker             android::base::GetProperty("ro.boot.verifiedbootstate", "") == "orange" ? "0" : "1");
652*00c7fec1SAndroid Build Coastguard Worker }
653*00c7fec1SAndroid Build Coastguard Worker 
property_enable_triggers_action(const BuiltinArguments & args)654*00c7fec1SAndroid Build Coastguard Worker static Result<void> property_enable_triggers_action(const BuiltinArguments& args) {
655*00c7fec1SAndroid Build Coastguard Worker     /* Enable property triggers. */
656*00c7fec1SAndroid Build Coastguard Worker     property_triggers_enabled = 1;
657*00c7fec1SAndroid Build Coastguard Worker     return {};
658*00c7fec1SAndroid Build Coastguard Worker }
659*00c7fec1SAndroid Build Coastguard Worker 
queue_property_triggers_action(const BuiltinArguments & args)660*00c7fec1SAndroid Build Coastguard Worker static Result<void> queue_property_triggers_action(const BuiltinArguments& args) {
661*00c7fec1SAndroid Build Coastguard Worker     ActionManager::GetInstance().QueueBuiltinAction(property_enable_triggers_action, "enable_property_trigger");
662*00c7fec1SAndroid Build Coastguard Worker     ActionManager::GetInstance().QueueAllPropertyActions();
663*00c7fec1SAndroid Build Coastguard Worker     return {};
664*00c7fec1SAndroid Build Coastguard Worker }
665*00c7fec1SAndroid Build Coastguard Worker 
666*00c7fec1SAndroid Build Coastguard Worker // Set the UDC controller for the ConfigFS USB Gadgets.
667*00c7fec1SAndroid Build Coastguard Worker // Read the UDC controller in use from "/sys/class/udc".
668*00c7fec1SAndroid Build Coastguard Worker // In case of multiple UDC controllers select the first one.
SetUsbController()669*00c7fec1SAndroid Build Coastguard Worker static void SetUsbController() {
670*00c7fec1SAndroid Build Coastguard Worker     static auto controller_set = false;
671*00c7fec1SAndroid Build Coastguard Worker     if (controller_set) return;
672*00c7fec1SAndroid Build Coastguard Worker     std::unique_ptr<DIR, decltype(&closedir)>dir(opendir("/sys/class/udc"), closedir);
673*00c7fec1SAndroid Build Coastguard Worker     if (!dir) return;
674*00c7fec1SAndroid Build Coastguard Worker 
675*00c7fec1SAndroid Build Coastguard Worker     dirent* dp;
676*00c7fec1SAndroid Build Coastguard Worker     while ((dp = readdir(dir.get())) != nullptr) {
677*00c7fec1SAndroid Build Coastguard Worker         if (dp->d_name[0] == '.') continue;
678*00c7fec1SAndroid Build Coastguard Worker 
679*00c7fec1SAndroid Build Coastguard Worker         SetProperty("sys.usb.controller", dp->d_name);
680*00c7fec1SAndroid Build Coastguard Worker         controller_set = true;
681*00c7fec1SAndroid Build Coastguard Worker         break;
682*00c7fec1SAndroid Build Coastguard Worker     }
683*00c7fec1SAndroid Build Coastguard Worker }
684*00c7fec1SAndroid Build Coastguard Worker 
685*00c7fec1SAndroid Build Coastguard Worker /// Set ro.kernel.version property to contain the major.minor pair as returned
686*00c7fec1SAndroid Build Coastguard Worker /// by uname(2).
SetKernelVersion()687*00c7fec1SAndroid Build Coastguard Worker static void SetKernelVersion() {
688*00c7fec1SAndroid Build Coastguard Worker     struct utsname uts;
689*00c7fec1SAndroid Build Coastguard Worker     unsigned int major, minor;
690*00c7fec1SAndroid Build Coastguard Worker 
691*00c7fec1SAndroid Build Coastguard Worker     if ((uname(&uts) != 0) || (sscanf(uts.release, "%u.%u", &major, &minor) != 2)) {
692*00c7fec1SAndroid Build Coastguard Worker         LOG(ERROR) << "Could not parse the kernel version from uname";
693*00c7fec1SAndroid Build Coastguard Worker         return;
694*00c7fec1SAndroid Build Coastguard Worker     }
695*00c7fec1SAndroid Build Coastguard Worker     SetProperty("ro.kernel.version", android::base::StringPrintf("%u.%u", major, minor));
696*00c7fec1SAndroid Build Coastguard Worker }
697*00c7fec1SAndroid Build Coastguard Worker 
HandleSigtermSignal(const signalfd_siginfo & siginfo)698*00c7fec1SAndroid Build Coastguard Worker static void HandleSigtermSignal(const signalfd_siginfo& siginfo) {
699*00c7fec1SAndroid Build Coastguard Worker     if (siginfo.ssi_pid != 0) {
700*00c7fec1SAndroid Build Coastguard Worker         // Drop any userspace SIGTERM requests.
701*00c7fec1SAndroid Build Coastguard Worker         LOG(DEBUG) << "Ignoring SIGTERM from pid " << siginfo.ssi_pid;
702*00c7fec1SAndroid Build Coastguard Worker         return;
703*00c7fec1SAndroid Build Coastguard Worker     }
704*00c7fec1SAndroid Build Coastguard Worker 
705*00c7fec1SAndroid Build Coastguard Worker     HandlePowerctlMessage("shutdown,container");
706*00c7fec1SAndroid Build Coastguard Worker }
707*00c7fec1SAndroid Build Coastguard Worker 
HandleSignalFd(int signal)708*00c7fec1SAndroid Build Coastguard Worker static void HandleSignalFd(int signal) {
709*00c7fec1SAndroid Build Coastguard Worker     signalfd_siginfo siginfo;
710*00c7fec1SAndroid Build Coastguard Worker     const int signal_fd = signal == SIGCHLD ? Service::GetSigchldFd() : sigterm_fd;
711*00c7fec1SAndroid Build Coastguard Worker     ssize_t bytes_read = TEMP_FAILURE_RETRY(read(signal_fd, &siginfo, sizeof(siginfo)));
712*00c7fec1SAndroid Build Coastguard Worker     if (bytes_read != sizeof(siginfo)) {
713*00c7fec1SAndroid Build Coastguard Worker         PLOG(ERROR) << "Failed to read siginfo from signal_fd";
714*00c7fec1SAndroid Build Coastguard Worker         return;
715*00c7fec1SAndroid Build Coastguard Worker     }
716*00c7fec1SAndroid Build Coastguard Worker 
717*00c7fec1SAndroid Build Coastguard Worker     switch (siginfo.ssi_signo) {
718*00c7fec1SAndroid Build Coastguard Worker         case SIGCHLD:
719*00c7fec1SAndroid Build Coastguard Worker             ReapAnyOutstandingChildren();
720*00c7fec1SAndroid Build Coastguard Worker             break;
721*00c7fec1SAndroid Build Coastguard Worker         case SIGTERM:
722*00c7fec1SAndroid Build Coastguard Worker             HandleSigtermSignal(siginfo);
723*00c7fec1SAndroid Build Coastguard Worker             break;
724*00c7fec1SAndroid Build Coastguard Worker         default:
725*00c7fec1SAndroid Build Coastguard Worker             LOG(ERROR) << "signal_fd: received unexpected signal " << siginfo.ssi_signo;
726*00c7fec1SAndroid Build Coastguard Worker             break;
727*00c7fec1SAndroid Build Coastguard Worker     }
728*00c7fec1SAndroid Build Coastguard Worker }
729*00c7fec1SAndroid Build Coastguard Worker 
UnblockSignals()730*00c7fec1SAndroid Build Coastguard Worker static void UnblockSignals() {
731*00c7fec1SAndroid Build Coastguard Worker     const struct sigaction act { .sa_handler = SIG_DFL };
732*00c7fec1SAndroid Build Coastguard Worker     sigaction(SIGCHLD, &act, nullptr);
733*00c7fec1SAndroid Build Coastguard Worker 
734*00c7fec1SAndroid Build Coastguard Worker     sigset_t mask;
735*00c7fec1SAndroid Build Coastguard Worker     sigemptyset(&mask);
736*00c7fec1SAndroid Build Coastguard Worker     sigaddset(&mask, SIGCHLD);
737*00c7fec1SAndroid Build Coastguard Worker     sigaddset(&mask, SIGTERM);
738*00c7fec1SAndroid Build Coastguard Worker 
739*00c7fec1SAndroid Build Coastguard Worker     if (sigprocmask(SIG_UNBLOCK, &mask, nullptr) == -1) {
740*00c7fec1SAndroid Build Coastguard Worker         PLOG(FATAL) << "failed to unblock signals for PID " << getpid();
741*00c7fec1SAndroid Build Coastguard Worker     }
742*00c7fec1SAndroid Build Coastguard Worker }
743*00c7fec1SAndroid Build Coastguard Worker 
RegisterSignalFd(Epoll * epoll,int signal,int fd)744*00c7fec1SAndroid Build Coastguard Worker static Result<void> RegisterSignalFd(Epoll* epoll, int signal, int fd) {
745*00c7fec1SAndroid Build Coastguard Worker     return epoll->RegisterHandler(
746*00c7fec1SAndroid Build Coastguard Worker             fd, [signal]() { HandleSignalFd(signal); }, EPOLLIN | EPOLLPRI);
747*00c7fec1SAndroid Build Coastguard Worker }
748*00c7fec1SAndroid Build Coastguard Worker 
CreateAndRegisterSignalFd(Epoll * epoll,int signal)749*00c7fec1SAndroid Build Coastguard Worker static Result<int> CreateAndRegisterSignalFd(Epoll* epoll, int signal) {
750*00c7fec1SAndroid Build Coastguard Worker     sigset_t mask;
751*00c7fec1SAndroid Build Coastguard Worker     sigemptyset(&mask);
752*00c7fec1SAndroid Build Coastguard Worker     sigaddset(&mask, signal);
753*00c7fec1SAndroid Build Coastguard Worker     if (sigprocmask(SIG_BLOCK, &mask, nullptr) == -1) {
754*00c7fec1SAndroid Build Coastguard Worker         return ErrnoError() << "failed to block signal " << signal;
755*00c7fec1SAndroid Build Coastguard Worker     }
756*00c7fec1SAndroid Build Coastguard Worker 
757*00c7fec1SAndroid Build Coastguard Worker     unique_fd signal_fd(signalfd(-1, &mask, SFD_CLOEXEC));
758*00c7fec1SAndroid Build Coastguard Worker     if (signal_fd.get() < 0) {
759*00c7fec1SAndroid Build Coastguard Worker         return ErrnoError() << "failed to create signalfd for signal " << signal;
760*00c7fec1SAndroid Build Coastguard Worker     }
761*00c7fec1SAndroid Build Coastguard Worker     OR_RETURN(RegisterSignalFd(epoll, signal, signal_fd.get()));
762*00c7fec1SAndroid Build Coastguard Worker 
763*00c7fec1SAndroid Build Coastguard Worker     return signal_fd.release();
764*00c7fec1SAndroid Build Coastguard Worker }
765*00c7fec1SAndroid Build Coastguard Worker 
InstallSignalFdHandler(Epoll * epoll)766*00c7fec1SAndroid Build Coastguard Worker static void InstallSignalFdHandler(Epoll* epoll) {
767*00c7fec1SAndroid Build Coastguard Worker     // Applying SA_NOCLDSTOP to a defaulted SIGCHLD handler prevents the signalfd from receiving
768*00c7fec1SAndroid Build Coastguard Worker     // SIGCHLD when a child process stops or continues (b/77867680#comment9).
769*00c7fec1SAndroid Build Coastguard Worker     const struct sigaction act { .sa_flags = SA_NOCLDSTOP, .sa_handler = SIG_DFL };
770*00c7fec1SAndroid Build Coastguard Worker     sigaction(SIGCHLD, &act, nullptr);
771*00c7fec1SAndroid Build Coastguard Worker 
772*00c7fec1SAndroid Build Coastguard Worker     // Register a handler to unblock signals in the child processes.
773*00c7fec1SAndroid Build Coastguard Worker     const int result = pthread_atfork(nullptr, nullptr, &UnblockSignals);
774*00c7fec1SAndroid Build Coastguard Worker     if (result != 0) {
775*00c7fec1SAndroid Build Coastguard Worker         LOG(FATAL) << "Failed to register a fork handler: " << strerror(result);
776*00c7fec1SAndroid Build Coastguard Worker     }
777*00c7fec1SAndroid Build Coastguard Worker 
778*00c7fec1SAndroid Build Coastguard Worker     Result<void> cs_result = RegisterSignalFd(epoll, SIGCHLD, Service::GetSigchldFd());
779*00c7fec1SAndroid Build Coastguard Worker     if (!cs_result.ok()) {
780*00c7fec1SAndroid Build Coastguard Worker         PLOG(FATAL) << cs_result.error();
781*00c7fec1SAndroid Build Coastguard Worker     }
782*00c7fec1SAndroid Build Coastguard Worker 
783*00c7fec1SAndroid Build Coastguard Worker     if (!IsRebootCapable()) {
784*00c7fec1SAndroid Build Coastguard Worker         Result<int> cs_result = CreateAndRegisterSignalFd(epoll, SIGTERM);
785*00c7fec1SAndroid Build Coastguard Worker         if (!cs_result.ok()) {
786*00c7fec1SAndroid Build Coastguard Worker             PLOG(FATAL) << cs_result.error();
787*00c7fec1SAndroid Build Coastguard Worker         }
788*00c7fec1SAndroid Build Coastguard Worker         sigterm_fd = cs_result.value();
789*00c7fec1SAndroid Build Coastguard Worker     }
790*00c7fec1SAndroid Build Coastguard Worker }
791*00c7fec1SAndroid Build Coastguard Worker 
HandleKeychord(const std::vector<int> & keycodes)792*00c7fec1SAndroid Build Coastguard Worker void HandleKeychord(const std::vector<int>& keycodes) {
793*00c7fec1SAndroid Build Coastguard Worker     // Only handle keychords if adb is enabled.
794*00c7fec1SAndroid Build Coastguard Worker     std::string adb_enabled = android::base::GetProperty("init.svc.adbd", "");
795*00c7fec1SAndroid Build Coastguard Worker     if (adb_enabled != "running") {
796*00c7fec1SAndroid Build Coastguard Worker         LOG(WARNING) << "Not starting service for keychord " << android::base::Join(keycodes, ' ')
797*00c7fec1SAndroid Build Coastguard Worker                      << " because ADB is disabled";
798*00c7fec1SAndroid Build Coastguard Worker         return;
799*00c7fec1SAndroid Build Coastguard Worker     }
800*00c7fec1SAndroid Build Coastguard Worker 
801*00c7fec1SAndroid Build Coastguard Worker     auto found = false;
802*00c7fec1SAndroid Build Coastguard Worker     for (const auto& service : ServiceList::GetInstance()) {
803*00c7fec1SAndroid Build Coastguard Worker         auto svc = service.get();
804*00c7fec1SAndroid Build Coastguard Worker         if (svc->keycodes() == keycodes) {
805*00c7fec1SAndroid Build Coastguard Worker             found = true;
806*00c7fec1SAndroid Build Coastguard Worker             LOG(INFO) << "Starting service '" << svc->name() << "' from keychord "
807*00c7fec1SAndroid Build Coastguard Worker                       << android::base::Join(keycodes, ' ');
808*00c7fec1SAndroid Build Coastguard Worker             if (auto result = svc->Start(); !result.ok()) {
809*00c7fec1SAndroid Build Coastguard Worker                 LOG(ERROR) << "Could not start service '" << svc->name() << "' from keychord "
810*00c7fec1SAndroid Build Coastguard Worker                            << android::base::Join(keycodes, ' ') << ": " << result.error();
811*00c7fec1SAndroid Build Coastguard Worker             }
812*00c7fec1SAndroid Build Coastguard Worker         }
813*00c7fec1SAndroid Build Coastguard Worker     }
814*00c7fec1SAndroid Build Coastguard Worker     if (!found) {
815*00c7fec1SAndroid Build Coastguard Worker         LOG(ERROR) << "Service for keychord " << android::base::Join(keycodes, ' ') << " not found";
816*00c7fec1SAndroid Build Coastguard Worker     }
817*00c7fec1SAndroid Build Coastguard Worker }
818*00c7fec1SAndroid Build Coastguard Worker 
UmountDebugRamdisk()819*00c7fec1SAndroid Build Coastguard Worker static void UmountDebugRamdisk() {
820*00c7fec1SAndroid Build Coastguard Worker     if (umount("/debug_ramdisk") != 0) {
821*00c7fec1SAndroid Build Coastguard Worker         PLOG(ERROR) << "Failed to umount /debug_ramdisk";
822*00c7fec1SAndroid Build Coastguard Worker     }
823*00c7fec1SAndroid Build Coastguard Worker }
824*00c7fec1SAndroid Build Coastguard Worker 
UmountSecondStageRes()825*00c7fec1SAndroid Build Coastguard Worker static void UmountSecondStageRes() {
826*00c7fec1SAndroid Build Coastguard Worker     if (umount(kSecondStageRes) != 0) {
827*00c7fec1SAndroid Build Coastguard Worker         PLOG(ERROR) << "Failed to umount " << kSecondStageRes;
828*00c7fec1SAndroid Build Coastguard Worker     }
829*00c7fec1SAndroid Build Coastguard Worker }
830*00c7fec1SAndroid Build Coastguard Worker 
MountExtraFilesystems()831*00c7fec1SAndroid Build Coastguard Worker static void MountExtraFilesystems() {
832*00c7fec1SAndroid Build Coastguard Worker #define CHECKCALL(x) \
833*00c7fec1SAndroid Build Coastguard Worker     if ((x) != 0) PLOG(FATAL) << #x " failed.";
834*00c7fec1SAndroid Build Coastguard Worker 
835*00c7fec1SAndroid Build Coastguard Worker     // /apex is used to mount APEXes
836*00c7fec1SAndroid Build Coastguard Worker     CHECKCALL(mount("tmpfs", "/apex", "tmpfs", MS_NOEXEC | MS_NOSUID | MS_NODEV,
837*00c7fec1SAndroid Build Coastguard Worker                     "mode=0755,uid=0,gid=0"));
838*00c7fec1SAndroid Build Coastguard Worker 
839*00c7fec1SAndroid Build Coastguard Worker     if (NeedsTwoMountNamespaces()) {
840*00c7fec1SAndroid Build Coastguard Worker         // /bootstrap-apex is used to mount "bootstrap" APEXes.
841*00c7fec1SAndroid Build Coastguard Worker         CHECKCALL(mount("tmpfs", "/bootstrap-apex", "tmpfs", MS_NOEXEC | MS_NOSUID | MS_NODEV,
842*00c7fec1SAndroid Build Coastguard Worker                         "mode=0755,uid=0,gid=0"));
843*00c7fec1SAndroid Build Coastguard Worker     }
844*00c7fec1SAndroid Build Coastguard Worker 
845*00c7fec1SAndroid Build Coastguard Worker     // /linkerconfig is used to keep generated linker configuration
846*00c7fec1SAndroid Build Coastguard Worker     CHECKCALL(mount("tmpfs", "/linkerconfig", "tmpfs", MS_NOEXEC | MS_NOSUID | MS_NODEV,
847*00c7fec1SAndroid Build Coastguard Worker                     "mode=0755,uid=0,gid=0"));
848*00c7fec1SAndroid Build Coastguard Worker #undef CHECKCALL
849*00c7fec1SAndroid Build Coastguard Worker }
850*00c7fec1SAndroid Build Coastguard Worker 
RecordStageBoottimes(const boot_clock::time_point & second_stage_start_time)851*00c7fec1SAndroid Build Coastguard Worker static void RecordStageBoottimes(const boot_clock::time_point& second_stage_start_time) {
852*00c7fec1SAndroid Build Coastguard Worker     int64_t first_stage_start_time_ns = -1;
853*00c7fec1SAndroid Build Coastguard Worker     if (auto first_stage_start_time_str = getenv(kEnvFirstStageStartedAt);
854*00c7fec1SAndroid Build Coastguard Worker         first_stage_start_time_str) {
855*00c7fec1SAndroid Build Coastguard Worker         SetProperty("ro.boottime.init", first_stage_start_time_str);
856*00c7fec1SAndroid Build Coastguard Worker         android::base::ParseInt(first_stage_start_time_str, &first_stage_start_time_ns);
857*00c7fec1SAndroid Build Coastguard Worker     }
858*00c7fec1SAndroid Build Coastguard Worker     unsetenv(kEnvFirstStageStartedAt);
859*00c7fec1SAndroid Build Coastguard Worker 
860*00c7fec1SAndroid Build Coastguard Worker     int64_t selinux_start_time_ns = -1;
861*00c7fec1SAndroid Build Coastguard Worker     if (auto selinux_start_time_str = getenv(kEnvSelinuxStartedAt); selinux_start_time_str) {
862*00c7fec1SAndroid Build Coastguard Worker         android::base::ParseInt(selinux_start_time_str, &selinux_start_time_ns);
863*00c7fec1SAndroid Build Coastguard Worker     }
864*00c7fec1SAndroid Build Coastguard Worker     unsetenv(kEnvSelinuxStartedAt);
865*00c7fec1SAndroid Build Coastguard Worker 
866*00c7fec1SAndroid Build Coastguard Worker     if (selinux_start_time_ns == -1) return;
867*00c7fec1SAndroid Build Coastguard Worker     if (first_stage_start_time_ns == -1) return;
868*00c7fec1SAndroid Build Coastguard Worker 
869*00c7fec1SAndroid Build Coastguard Worker     SetProperty("ro.boottime.init.first_stage",
870*00c7fec1SAndroid Build Coastguard Worker                 std::to_string(selinux_start_time_ns - first_stage_start_time_ns));
871*00c7fec1SAndroid Build Coastguard Worker     SetProperty("ro.boottime.init.selinux",
872*00c7fec1SAndroid Build Coastguard Worker                 std::to_string(second_stage_start_time.time_since_epoch().count() -
873*00c7fec1SAndroid Build Coastguard Worker                                selinux_start_time_ns));
874*00c7fec1SAndroid Build Coastguard Worker     if (auto init_module_time_str = getenv(kEnvInitModuleDurationMs); init_module_time_str) {
875*00c7fec1SAndroid Build Coastguard Worker         SetProperty("ro.boottime.init.modules", init_module_time_str);
876*00c7fec1SAndroid Build Coastguard Worker         unsetenv(kEnvInitModuleDurationMs);
877*00c7fec1SAndroid Build Coastguard Worker     }
878*00c7fec1SAndroid Build Coastguard Worker }
879*00c7fec1SAndroid Build Coastguard Worker 
SendLoadPersistentPropertiesMessage()880*00c7fec1SAndroid Build Coastguard Worker void SendLoadPersistentPropertiesMessage() {
881*00c7fec1SAndroid Build Coastguard Worker     auto init_message = InitMessage{};
882*00c7fec1SAndroid Build Coastguard Worker     init_message.set_load_persistent_properties(true);
883*00c7fec1SAndroid Build Coastguard Worker     if (auto result = SendMessage(property_fd, init_message); !result.ok()) {
884*00c7fec1SAndroid Build Coastguard Worker         LOG(ERROR) << "Failed to send load persistent properties message: " << result.error();
885*00c7fec1SAndroid Build Coastguard Worker     }
886*00c7fec1SAndroid Build Coastguard Worker }
887*00c7fec1SAndroid Build Coastguard Worker 
ConnectEarlyStageSnapuserdAction(const BuiltinArguments & args)888*00c7fec1SAndroid Build Coastguard Worker static Result<void> ConnectEarlyStageSnapuserdAction(const BuiltinArguments& args) {
889*00c7fec1SAndroid Build Coastguard Worker     auto pid = GetSnapuserdFirstStagePid();
890*00c7fec1SAndroid Build Coastguard Worker     if (!pid) {
891*00c7fec1SAndroid Build Coastguard Worker         return {};
892*00c7fec1SAndroid Build Coastguard Worker     }
893*00c7fec1SAndroid Build Coastguard Worker 
894*00c7fec1SAndroid Build Coastguard Worker     auto info = GetSnapuserdFirstStageInfo();
895*00c7fec1SAndroid Build Coastguard Worker     if (auto iter = std::find(info.begin(), info.end(), "socket"s); iter == info.end()) {
896*00c7fec1SAndroid Build Coastguard Worker         // snapuserd does not support socket handoff, so exit early.
897*00c7fec1SAndroid Build Coastguard Worker         return {};
898*00c7fec1SAndroid Build Coastguard Worker     }
899*00c7fec1SAndroid Build Coastguard Worker 
900*00c7fec1SAndroid Build Coastguard Worker     // Socket handoff is supported.
901*00c7fec1SAndroid Build Coastguard Worker     auto svc = ServiceList::GetInstance().FindService("snapuserd");
902*00c7fec1SAndroid Build Coastguard Worker     if (!svc) {
903*00c7fec1SAndroid Build Coastguard Worker         LOG(FATAL) << "Failed to find snapuserd service entry";
904*00c7fec1SAndroid Build Coastguard Worker     }
905*00c7fec1SAndroid Build Coastguard Worker 
906*00c7fec1SAndroid Build Coastguard Worker     svc->SetShutdownCritical();
907*00c7fec1SAndroid Build Coastguard Worker     svc->SetStartedInFirstStage(*pid);
908*00c7fec1SAndroid Build Coastguard Worker 
909*00c7fec1SAndroid Build Coastguard Worker     svc = ServiceList::GetInstance().FindService("snapuserd_proxy");
910*00c7fec1SAndroid Build Coastguard Worker     if (!svc) {
911*00c7fec1SAndroid Build Coastguard Worker         LOG(FATAL) << "Failed find snapuserd_proxy service entry, merge will never initiate";
912*00c7fec1SAndroid Build Coastguard Worker     }
913*00c7fec1SAndroid Build Coastguard Worker     if (!svc->MarkSocketPersistent("snapuserd")) {
914*00c7fec1SAndroid Build Coastguard Worker         LOG(FATAL) << "Could not find snapuserd socket in snapuserd_proxy service entry";
915*00c7fec1SAndroid Build Coastguard Worker     }
916*00c7fec1SAndroid Build Coastguard Worker     if (auto result = svc->Start(); !result.ok()) {
917*00c7fec1SAndroid Build Coastguard Worker         LOG(FATAL) << "Could not start snapuserd_proxy: " << result.error();
918*00c7fec1SAndroid Build Coastguard Worker     }
919*00c7fec1SAndroid Build Coastguard Worker     return {};
920*00c7fec1SAndroid Build Coastguard Worker }
921*00c7fec1SAndroid Build Coastguard Worker 
SecondStageMain(int argc,char ** argv)922*00c7fec1SAndroid Build Coastguard Worker int SecondStageMain(int argc, char** argv) {
923*00c7fec1SAndroid Build Coastguard Worker     if (REBOOT_BOOTLOADER_ON_PANIC) {
924*00c7fec1SAndroid Build Coastguard Worker         InstallRebootSignalHandlers();
925*00c7fec1SAndroid Build Coastguard Worker     }
926*00c7fec1SAndroid Build Coastguard Worker 
927*00c7fec1SAndroid Build Coastguard Worker     // No threads should be spin up until signalfd
928*00c7fec1SAndroid Build Coastguard Worker     // is registered. If the threads are indeed required,
929*00c7fec1SAndroid Build Coastguard Worker     // each of these threads _should_ make sure SIGCHLD signal
930*00c7fec1SAndroid Build Coastguard Worker     // is blocked. See b/223076262
931*00c7fec1SAndroid Build Coastguard Worker     boot_clock::time_point start_time = boot_clock::now();
932*00c7fec1SAndroid Build Coastguard Worker 
933*00c7fec1SAndroid Build Coastguard Worker     trigger_shutdown = [](const std::string& command) { shutdown_state.TriggerShutdown(command); };
934*00c7fec1SAndroid Build Coastguard Worker 
935*00c7fec1SAndroid Build Coastguard Worker     SetStdioToDevNull(argv);
936*00c7fec1SAndroid Build Coastguard Worker     InitKernelLogging(argv);
937*00c7fec1SAndroid Build Coastguard Worker     LOG(INFO) << "init second stage started!";
938*00c7fec1SAndroid Build Coastguard Worker 
939*00c7fec1SAndroid Build Coastguard Worker     SelinuxSetupKernelLogging();
940*00c7fec1SAndroid Build Coastguard Worker 
941*00c7fec1SAndroid Build Coastguard Worker     // Update $PATH in the case the second stage init is newer than first stage init, where it is
942*00c7fec1SAndroid Build Coastguard Worker     // first set.
943*00c7fec1SAndroid Build Coastguard Worker     if (setenv("PATH", _PATH_DEFPATH, 1) != 0) {
944*00c7fec1SAndroid Build Coastguard Worker         PLOG(FATAL) << "Could not set $PATH to '" << _PATH_DEFPATH << "' in second stage";
945*00c7fec1SAndroid Build Coastguard Worker     }
946*00c7fec1SAndroid Build Coastguard Worker 
947*00c7fec1SAndroid Build Coastguard Worker     // Init should not crash because of a dependence on any other process, therefore we ignore
948*00c7fec1SAndroid Build Coastguard Worker     // SIGPIPE and handle EPIPE at the call site directly.  Note that setting a signal to SIG_IGN
949*00c7fec1SAndroid Build Coastguard Worker     // is inherited across exec, but custom signal handlers are not.  Since we do not want to
950*00c7fec1SAndroid Build Coastguard Worker     // ignore SIGPIPE for child processes, we set a no-op function for the signal handler instead.
951*00c7fec1SAndroid Build Coastguard Worker     {
952*00c7fec1SAndroid Build Coastguard Worker         struct sigaction action = {.sa_flags = SA_RESTART};
953*00c7fec1SAndroid Build Coastguard Worker         action.sa_handler = [](int) {};
954*00c7fec1SAndroid Build Coastguard Worker         sigaction(SIGPIPE, &action, nullptr);
955*00c7fec1SAndroid Build Coastguard Worker     }
956*00c7fec1SAndroid Build Coastguard Worker 
957*00c7fec1SAndroid Build Coastguard Worker     // Set init and its forked children's oom_adj.
958*00c7fec1SAndroid Build Coastguard Worker     if (auto result =
959*00c7fec1SAndroid Build Coastguard Worker                 WriteFile("/proc/1/oom_score_adj", StringPrintf("%d", DEFAULT_OOM_SCORE_ADJUST));
960*00c7fec1SAndroid Build Coastguard Worker         !result.ok()) {
961*00c7fec1SAndroid Build Coastguard Worker         LOG(ERROR) << "Unable to write " << DEFAULT_OOM_SCORE_ADJUST
962*00c7fec1SAndroid Build Coastguard Worker                    << " to /proc/1/oom_score_adj: " << result.error();
963*00c7fec1SAndroid Build Coastguard Worker     }
964*00c7fec1SAndroid Build Coastguard Worker 
965*00c7fec1SAndroid Build Coastguard Worker     // Indicate that booting is in progress to background fw loaders, etc.
966*00c7fec1SAndroid Build Coastguard Worker     close(open("/dev/.booting", O_WRONLY | O_CREAT | O_CLOEXEC, 0000));
967*00c7fec1SAndroid Build Coastguard Worker 
968*00c7fec1SAndroid Build Coastguard Worker     // See if need to load debug props to allow adb root, when the device is unlocked.
969*00c7fec1SAndroid Build Coastguard Worker     const char* force_debuggable_env = getenv("INIT_FORCE_DEBUGGABLE");
970*00c7fec1SAndroid Build Coastguard Worker     bool load_debug_prop = false;
971*00c7fec1SAndroid Build Coastguard Worker     if (force_debuggable_env && AvbHandle::IsDeviceUnlocked()) {
972*00c7fec1SAndroid Build Coastguard Worker         load_debug_prop = "true"s == force_debuggable_env;
973*00c7fec1SAndroid Build Coastguard Worker     }
974*00c7fec1SAndroid Build Coastguard Worker     unsetenv("INIT_FORCE_DEBUGGABLE");
975*00c7fec1SAndroid Build Coastguard Worker 
976*00c7fec1SAndroid Build Coastguard Worker     // Umount the debug ramdisk so property service doesn't read .prop files from there, when it
977*00c7fec1SAndroid Build Coastguard Worker     // is not meant to.
978*00c7fec1SAndroid Build Coastguard Worker     if (!load_debug_prop) {
979*00c7fec1SAndroid Build Coastguard Worker         UmountDebugRamdisk();
980*00c7fec1SAndroid Build Coastguard Worker     }
981*00c7fec1SAndroid Build Coastguard Worker 
982*00c7fec1SAndroid Build Coastguard Worker     PropertyInit();
983*00c7fec1SAndroid Build Coastguard Worker 
984*00c7fec1SAndroid Build Coastguard Worker     // Umount second stage resources after property service has read the .prop files.
985*00c7fec1SAndroid Build Coastguard Worker     UmountSecondStageRes();
986*00c7fec1SAndroid Build Coastguard Worker 
987*00c7fec1SAndroid Build Coastguard Worker     // Umount the debug ramdisk after property service has read the .prop files when it means to.
988*00c7fec1SAndroid Build Coastguard Worker     if (load_debug_prop) {
989*00c7fec1SAndroid Build Coastguard Worker         UmountDebugRamdisk();
990*00c7fec1SAndroid Build Coastguard Worker     }
991*00c7fec1SAndroid Build Coastguard Worker 
992*00c7fec1SAndroid Build Coastguard Worker     // Mount extra filesystems required during second stage init
993*00c7fec1SAndroid Build Coastguard Worker     MountExtraFilesystems();
994*00c7fec1SAndroid Build Coastguard Worker 
995*00c7fec1SAndroid Build Coastguard Worker     // Now set up SELinux for second stage.
996*00c7fec1SAndroid Build Coastguard Worker     SelabelInitialize();
997*00c7fec1SAndroid Build Coastguard Worker     SelinuxRestoreContext();
998*00c7fec1SAndroid Build Coastguard Worker 
999*00c7fec1SAndroid Build Coastguard Worker     Epoll epoll;
1000*00c7fec1SAndroid Build Coastguard Worker     if (auto result = epoll.Open(); !result.ok()) {
1001*00c7fec1SAndroid Build Coastguard Worker         PLOG(FATAL) << result.error();
1002*00c7fec1SAndroid Build Coastguard Worker     }
1003*00c7fec1SAndroid Build Coastguard Worker 
1004*00c7fec1SAndroid Build Coastguard Worker     // We always reap children before responding to the other pending functions. This is to
1005*00c7fec1SAndroid Build Coastguard Worker     // prevent a race where other daemons see that a service has exited and ask init to
1006*00c7fec1SAndroid Build Coastguard Worker     // start it again via ctl.start before init has reaped it.
1007*00c7fec1SAndroid Build Coastguard Worker     epoll.SetFirstCallback(ReapAnyOutstandingChildren);
1008*00c7fec1SAndroid Build Coastguard Worker 
1009*00c7fec1SAndroid Build Coastguard Worker     InstallSignalFdHandler(&epoll);
1010*00c7fec1SAndroid Build Coastguard Worker     InstallInitNotifier(&epoll);
1011*00c7fec1SAndroid Build Coastguard Worker     StartPropertyService(&property_fd);
1012*00c7fec1SAndroid Build Coastguard Worker 
1013*00c7fec1SAndroid Build Coastguard Worker     // Make the time that init stages started available for bootstat to log.
1014*00c7fec1SAndroid Build Coastguard Worker     RecordStageBoottimes(start_time);
1015*00c7fec1SAndroid Build Coastguard Worker 
1016*00c7fec1SAndroid Build Coastguard Worker     // Set libavb version for Framework-only OTA match in Treble build.
1017*00c7fec1SAndroid Build Coastguard Worker     if (const char* avb_version = getenv("INIT_AVB_VERSION"); avb_version != nullptr) {
1018*00c7fec1SAndroid Build Coastguard Worker         SetProperty("ro.boot.avb_version", avb_version);
1019*00c7fec1SAndroid Build Coastguard Worker     }
1020*00c7fec1SAndroid Build Coastguard Worker     unsetenv("INIT_AVB_VERSION");
1021*00c7fec1SAndroid Build Coastguard Worker 
1022*00c7fec1SAndroid Build Coastguard Worker     fs_mgr_vendor_overlay_mount_all();
1023*00c7fec1SAndroid Build Coastguard Worker     export_oem_lock_status();
1024*00c7fec1SAndroid Build Coastguard Worker     MountHandler mount_handler(&epoll);
1025*00c7fec1SAndroid Build Coastguard Worker     SetUsbController();
1026*00c7fec1SAndroid Build Coastguard Worker     SetKernelVersion();
1027*00c7fec1SAndroid Build Coastguard Worker 
1028*00c7fec1SAndroid Build Coastguard Worker     const BuiltinFunctionMap& function_map = GetBuiltinFunctionMap();
1029*00c7fec1SAndroid Build Coastguard Worker     Action::set_function_map(&function_map);
1030*00c7fec1SAndroid Build Coastguard Worker 
1031*00c7fec1SAndroid Build Coastguard Worker     if (!SetupMountNamespaces()) {
1032*00c7fec1SAndroid Build Coastguard Worker         PLOG(FATAL) << "SetupMountNamespaces failed";
1033*00c7fec1SAndroid Build Coastguard Worker     }
1034*00c7fec1SAndroid Build Coastguard Worker 
1035*00c7fec1SAndroid Build Coastguard Worker     InitializeSubcontext();
1036*00c7fec1SAndroid Build Coastguard Worker 
1037*00c7fec1SAndroid Build Coastguard Worker     ActionManager& am = ActionManager::GetInstance();
1038*00c7fec1SAndroid Build Coastguard Worker     ServiceList& sm = ServiceList::GetInstance();
1039*00c7fec1SAndroid Build Coastguard Worker 
1040*00c7fec1SAndroid Build Coastguard Worker     LoadBootScripts(am, sm);
1041*00c7fec1SAndroid Build Coastguard Worker 
1042*00c7fec1SAndroid Build Coastguard Worker     // Turning this on and letting the INFO logging be discarded adds 0.2s to
1043*00c7fec1SAndroid Build Coastguard Worker     // Nexus 9 boot time, so it's disabled by default.
1044*00c7fec1SAndroid Build Coastguard Worker     if (false) DumpState();
1045*00c7fec1SAndroid Build Coastguard Worker 
1046*00c7fec1SAndroid Build Coastguard Worker     // Make the GSI status available before scripts start running.
1047*00c7fec1SAndroid Build Coastguard Worker     auto is_running = android::gsi::IsGsiRunning() ? "1" : "0";
1048*00c7fec1SAndroid Build Coastguard Worker     SetProperty(gsi::kGsiBootedProp, is_running);
1049*00c7fec1SAndroid Build Coastguard Worker     auto is_installed = android::gsi::IsGsiInstalled() ? "1" : "0";
1050*00c7fec1SAndroid Build Coastguard Worker     SetProperty(gsi::kGsiInstalledProp, is_installed);
1051*00c7fec1SAndroid Build Coastguard Worker     if (android::gsi::IsGsiRunning()) {
1052*00c7fec1SAndroid Build Coastguard Worker         std::string dsu_slot;
1053*00c7fec1SAndroid Build Coastguard Worker         if (android::gsi::GetActiveDsu(&dsu_slot)) {
1054*00c7fec1SAndroid Build Coastguard Worker             SetProperty(gsi::kDsuSlotProp, dsu_slot);
1055*00c7fec1SAndroid Build Coastguard Worker         }
1056*00c7fec1SAndroid Build Coastguard Worker     }
1057*00c7fec1SAndroid Build Coastguard Worker 
1058*00c7fec1SAndroid Build Coastguard Worker     am.QueueBuiltinAction(SetupCgroupsAction, "SetupCgroups");
1059*00c7fec1SAndroid Build Coastguard Worker     am.QueueBuiltinAction(SetKptrRestrictAction, "SetKptrRestrict");
1060*00c7fec1SAndroid Build Coastguard Worker     am.QueueBuiltinAction(TestPerfEventSelinuxAction, "TestPerfEventSelinux");
1061*00c7fec1SAndroid Build Coastguard Worker     am.QueueEventTrigger("early-init");
1062*00c7fec1SAndroid Build Coastguard Worker     am.QueueBuiltinAction(ConnectEarlyStageSnapuserdAction, "ConnectEarlyStageSnapuserd");
1063*00c7fec1SAndroid Build Coastguard Worker 
1064*00c7fec1SAndroid Build Coastguard Worker     // Queue an action that waits for coldboot done so we know ueventd has set up all of /dev...
1065*00c7fec1SAndroid Build Coastguard Worker     am.QueueBuiltinAction(wait_for_coldboot_done_action, "wait_for_coldboot_done");
1066*00c7fec1SAndroid Build Coastguard Worker     // ... so that we can start queuing up actions that require stuff from /dev.
1067*00c7fec1SAndroid Build Coastguard Worker     am.QueueBuiltinAction(SetMmapRndBitsAction, "SetMmapRndBits");
1068*00c7fec1SAndroid Build Coastguard Worker     Keychords keychords;
1069*00c7fec1SAndroid Build Coastguard Worker     am.QueueBuiltinAction(
1070*00c7fec1SAndroid Build Coastguard Worker             [&epoll, &keychords](const BuiltinArguments& args) -> Result<void> {
1071*00c7fec1SAndroid Build Coastguard Worker                 for (const auto& svc : ServiceList::GetInstance()) {
1072*00c7fec1SAndroid Build Coastguard Worker                     keychords.Register(svc->keycodes());
1073*00c7fec1SAndroid Build Coastguard Worker                 }
1074*00c7fec1SAndroid Build Coastguard Worker                 keychords.Start(&epoll, HandleKeychord);
1075*00c7fec1SAndroid Build Coastguard Worker                 return {};
1076*00c7fec1SAndroid Build Coastguard Worker             },
1077*00c7fec1SAndroid Build Coastguard Worker             "KeychordInit");
1078*00c7fec1SAndroid Build Coastguard Worker 
1079*00c7fec1SAndroid Build Coastguard Worker     // Trigger all the boot actions to get us started.
1080*00c7fec1SAndroid Build Coastguard Worker     am.QueueEventTrigger("init");
1081*00c7fec1SAndroid Build Coastguard Worker 
1082*00c7fec1SAndroid Build Coastguard Worker     // Don't mount filesystems or start core system services in charger mode.
1083*00c7fec1SAndroid Build Coastguard Worker     std::string bootmode = GetProperty("ro.bootmode", "");
1084*00c7fec1SAndroid Build Coastguard Worker     if (bootmode == "charger") {
1085*00c7fec1SAndroid Build Coastguard Worker         am.QueueEventTrigger("charger");
1086*00c7fec1SAndroid Build Coastguard Worker     } else {
1087*00c7fec1SAndroid Build Coastguard Worker         am.QueueEventTrigger("late-init");
1088*00c7fec1SAndroid Build Coastguard Worker     }
1089*00c7fec1SAndroid Build Coastguard Worker 
1090*00c7fec1SAndroid Build Coastguard Worker     // Run all property triggers based on current state of the properties.
1091*00c7fec1SAndroid Build Coastguard Worker     am.QueueBuiltinAction(queue_property_triggers_action, "queue_property_triggers");
1092*00c7fec1SAndroid Build Coastguard Worker 
1093*00c7fec1SAndroid Build Coastguard Worker     // Restore prio before main loop
1094*00c7fec1SAndroid Build Coastguard Worker     setpriority(PRIO_PROCESS, 0, 0);
1095*00c7fec1SAndroid Build Coastguard Worker     while (true) {
1096*00c7fec1SAndroid Build Coastguard Worker         // By default, sleep until something happens. Do not convert far_future into
1097*00c7fec1SAndroid Build Coastguard Worker         // std::chrono::milliseconds because that would trigger an overflow. The unit of boot_clock
1098*00c7fec1SAndroid Build Coastguard Worker         // is 1ns.
1099*00c7fec1SAndroid Build Coastguard Worker         const boot_clock::time_point far_future = boot_clock::time_point::max();
1100*00c7fec1SAndroid Build Coastguard Worker         boot_clock::time_point next_action_time = far_future;
1101*00c7fec1SAndroid Build Coastguard Worker 
1102*00c7fec1SAndroid Build Coastguard Worker         auto shutdown_command = shutdown_state.CheckShutdown();
1103*00c7fec1SAndroid Build Coastguard Worker         if (shutdown_command) {
1104*00c7fec1SAndroid Build Coastguard Worker             LOG(INFO) << "Got shutdown_command '" << *shutdown_command
1105*00c7fec1SAndroid Build Coastguard Worker                       << "' Calling HandlePowerctlMessage()";
1106*00c7fec1SAndroid Build Coastguard Worker             HandlePowerctlMessage(*shutdown_command);
1107*00c7fec1SAndroid Build Coastguard Worker         }
1108*00c7fec1SAndroid Build Coastguard Worker 
1109*00c7fec1SAndroid Build Coastguard Worker         if (!(prop_waiter_state.MightBeWaiting() || Service::is_exec_service_running())) {
1110*00c7fec1SAndroid Build Coastguard Worker             am.ExecuteOneCommand();
1111*00c7fec1SAndroid Build Coastguard Worker             // If there's more work to do, wake up again immediately.
1112*00c7fec1SAndroid Build Coastguard Worker             if (am.HasMoreCommands()) {
1113*00c7fec1SAndroid Build Coastguard Worker                 next_action_time = boot_clock::now();
1114*00c7fec1SAndroid Build Coastguard Worker             }
1115*00c7fec1SAndroid Build Coastguard Worker         }
1116*00c7fec1SAndroid Build Coastguard Worker         // Since the above code examined pending actions, no new actions must be
1117*00c7fec1SAndroid Build Coastguard Worker         // queued by the code between this line and the Epoll::Wait() call below
1118*00c7fec1SAndroid Build Coastguard Worker         // without calling WakeMainInitThread().
1119*00c7fec1SAndroid Build Coastguard Worker         if (!IsShuttingDown()) {
1120*00c7fec1SAndroid Build Coastguard Worker             auto next_process_action_time = HandleProcessActions();
1121*00c7fec1SAndroid Build Coastguard Worker 
1122*00c7fec1SAndroid Build Coastguard Worker             // If there's a process that needs restarting, wake up in time for that.
1123*00c7fec1SAndroid Build Coastguard Worker             if (next_process_action_time) {
1124*00c7fec1SAndroid Build Coastguard Worker                 next_action_time = std::min(next_action_time, *next_process_action_time);
1125*00c7fec1SAndroid Build Coastguard Worker             }
1126*00c7fec1SAndroid Build Coastguard Worker         }
1127*00c7fec1SAndroid Build Coastguard Worker 
1128*00c7fec1SAndroid Build Coastguard Worker         std::optional<std::chrono::milliseconds> epoll_timeout;
1129*00c7fec1SAndroid Build Coastguard Worker         if (next_action_time != far_future) {
1130*00c7fec1SAndroid Build Coastguard Worker             epoll_timeout = std::chrono::ceil<std::chrono::milliseconds>(
1131*00c7fec1SAndroid Build Coastguard Worker                     std::max(next_action_time - boot_clock::now(), 0ns));
1132*00c7fec1SAndroid Build Coastguard Worker         }
1133*00c7fec1SAndroid Build Coastguard Worker         auto epoll_result = epoll.Wait(epoll_timeout);
1134*00c7fec1SAndroid Build Coastguard Worker         if (!epoll_result.ok()) {
1135*00c7fec1SAndroid Build Coastguard Worker             LOG(ERROR) << epoll_result.error();
1136*00c7fec1SAndroid Build Coastguard Worker         }
1137*00c7fec1SAndroid Build Coastguard Worker         if (!IsShuttingDown()) {
1138*00c7fec1SAndroid Build Coastguard Worker             HandleControlMessages();
1139*00c7fec1SAndroid Build Coastguard Worker             SetUsbController();
1140*00c7fec1SAndroid Build Coastguard Worker         }
1141*00c7fec1SAndroid Build Coastguard Worker     }
1142*00c7fec1SAndroid Build Coastguard Worker 
1143*00c7fec1SAndroid Build Coastguard Worker     return 0;
1144*00c7fec1SAndroid Build Coastguard Worker }
1145*00c7fec1SAndroid Build Coastguard Worker 
1146*00c7fec1SAndroid Build Coastguard Worker }  // namespace init
1147*00c7fec1SAndroid Build Coastguard Worker }  // namespace android
1148