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