blob: ef9a4514c4241c40fb6508a9bfaa0f180ad2f7c0 [file] [log] [blame]
Tom Cherryde6bd502018-02-13 16:50:08 -08001//
2// Copyright (C) 2018 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
16
Tom Cherryb5f2ec02019-11-08 17:54:27 -080017#include "host_init_verifier.h"
18
Tom Cherry31525f52018-05-09 18:33:31 -070019#include <errno.h>
Tom Cherry3f1bce82019-06-05 09:13:11 -070020#include <getopt.h>
Tom Cherryde6bd502018-02-13 16:50:08 -080021#include <pwd.h>
Tom Cherry31525f52018-05-09 18:33:31 -070022#include <stdio.h>
Tom Cherry863d8082018-06-12 14:40:38 -070023#include <stdlib.h>
Tom Cherryde6bd502018-02-13 16:50:08 -080024
Daniel Norman3f42a762019-07-09 11:00:53 -070025#include <fstream>
Tom Cherry194b5d12018-05-09 17:38:30 -070026#include <iostream>
Tom Cherry3f1bce82019-06-05 09:13:11 -070027#include <iterator>
Tom Cherry194b5d12018-05-09 17:38:30 -070028#include <string>
Tom Cherry31525f52018-05-09 18:33:31 -070029#include <vector>
Tom Cherry194b5d12018-05-09 17:38:30 -070030
Tom Cherry31525f52018-05-09 18:33:31 -070031#include <android-base/file.h>
Tom Cherryde6bd502018-02-13 16:50:08 -080032#include <android-base/logging.h>
Tom Cherry31525f52018-05-09 18:33:31 -070033#include <android-base/parseint.h>
Tom Cherry194b5d12018-05-09 17:38:30 -070034#include <android-base/strings.h>
Tom Cherry6ad4d0a2020-03-04 13:35:28 -080035#include <generated_android_ids.h>
Steven Moreland422a7582019-10-15 14:53:19 -070036#include <hidl/metadata.h>
Tom Cherryb5f2ec02019-11-08 17:54:27 -080037#include <property_info_serializer/property_info_serializer.h>
Tom Cherryde6bd502018-02-13 16:50:08 -080038
39#include "action.h"
40#include "action_manager.h"
41#include "action_parser.h"
Tom Cherry4772f1d2019-07-30 09:34:41 -070042#include "check_builtins.h"
Tom Cherry194b5d12018-05-09 17:38:30 -070043#include "host_import_parser.h"
44#include "host_init_stubs.h"
Daniel Normand2533c32019-08-02 15:13:50 -070045#include "interface_utils.h"
Tom Cherryde6bd502018-02-13 16:50:08 -080046#include "parser.h"
47#include "result.h"
48#include "service.h"
Tom Cherry2aeb1ad2019-06-26 10:46:20 -070049#include "service_list.h"
50#include "service_parser.h"
Tom Cherryde6bd502018-02-13 16:50:08 -080051
Tom Cherry194b5d12018-05-09 17:38:30 -070052using namespace std::literals;
53
Tom Cherry31525f52018-05-09 18:33:31 -070054using android::base::ParseInt;
55using android::base::ReadFileToString;
Tom Cherry194b5d12018-05-09 17:38:30 -070056using android::base::Split;
Tom Cherryb5f2ec02019-11-08 17:54:27 -080057using android::properties::BuildTrie;
58using android::properties::ParsePropertyInfoFile;
59using android::properties::PropertyInfoArea;
60using android::properties::PropertyInfoEntry;
Tom Cherry194b5d12018-05-09 17:38:30 -070061
Tom Cherry3f1bce82019-06-05 09:13:11 -070062static std::vector<std::string> passwd_files;
Tom Cherry31525f52018-05-09 18:33:31 -070063
Tom Cherry3f1bce82019-06-05 09:13:11 -070064static std::vector<std::pair<std::string, int>> GetVendorPasswd(const std::string& passwd_file) {
Tom Cherry31525f52018-05-09 18:33:31 -070065 std::string passwd;
Tom Cherry863d8082018-06-12 14:40:38 -070066 if (!ReadFileToString(passwd_file, &passwd)) {
Tom Cherry31525f52018-05-09 18:33:31 -070067 return {};
68 }
69
70 std::vector<std::pair<std::string, int>> result;
71 auto passwd_lines = Split(passwd, "\n");
72 for (const auto& line : passwd_lines) {
73 auto split_line = Split(line, ":");
74 if (split_line.size() < 3) {
75 continue;
76 }
77 int uid = 0;
78 if (!ParseInt(split_line[2], &uid)) {
79 continue;
80 }
81 result.emplace_back(split_line[0], uid);
82 }
83 return result;
84}
85
Tom Cherry3f1bce82019-06-05 09:13:11 -070086static std::vector<std::pair<std::string, int>> GetVendorPasswd() {
87 std::vector<std::pair<std::string, int>> result;
88 for (const auto& passwd_file : passwd_files) {
89 auto individual_result = GetVendorPasswd(passwd_file);
90 std::move(individual_result.begin(), individual_result.end(),
91 std::back_insert_iterator(result));
92 }
93 return result;
94}
95
Tom Cherryde6bd502018-02-13 16:50:08 -080096passwd* getpwnam(const char* login) { // NOLINT: implementing bad function.
Tom Cherry31525f52018-05-09 18:33:31 -070097 // This isn't thread safe, but that's okay for our purposes.
98 static char static_name[32] = "";
99 static char static_dir[32] = "/";
100 static char static_shell[32] = "/system/bin/sh";
101 static passwd static_passwd = {
102 .pw_name = static_name,
103 .pw_dir = static_dir,
104 .pw_shell = static_shell,
105 .pw_uid = 0,
106 .pw_gid = 0,
Tom Cherryde6bd502018-02-13 16:50:08 -0800107 };
Tom Cherry31525f52018-05-09 18:33:31 -0700108
109 for (size_t n = 0; n < android_id_count; ++n) {
110 if (!strcmp(android_ids[n].name, login)) {
111 snprintf(static_name, sizeof(static_name), "%s", android_ids[n].name);
112 static_passwd.pw_uid = android_ids[n].aid;
113 static_passwd.pw_gid = android_ids[n].aid;
114 return &static_passwd;
115 }
116 }
117
118 static const auto vendor_passwd = GetVendorPasswd();
119
120 for (const auto& [name, uid] : vendor_passwd) {
121 if (name == login) {
122 snprintf(static_name, sizeof(static_name), "%s", name.c_str());
123 static_passwd.pw_uid = uid;
124 static_passwd.pw_gid = uid;
125 return &static_passwd;
126 }
127 }
128
Tom Cherry290427b2018-06-14 13:40:20 -0700129 unsigned int oem_uid;
130 if (sscanf(login, "oem_%u", &oem_uid) == 1) {
131 snprintf(static_name, sizeof(static_name), "%s", login);
132 static_passwd.pw_uid = oem_uid;
133 static_passwd.pw_gid = oem_uid;
134 return &static_passwd;
135 }
136
Tom Cherry31525f52018-05-09 18:33:31 -0700137 errno = ENOENT;
138 return nullptr;
Tom Cherryde6bd502018-02-13 16:50:08 -0800139}
140
141namespace android {
142namespace init {
143
Tom Cherry4772f1d2019-07-30 09:34:41 -0700144static Result<void> check_stub(const BuiltinArguments& args) {
Tom Cherrybbcbc2f2019-06-10 11:08:01 -0700145 return {};
Tom Cherryde6bd502018-02-13 16:50:08 -0800146}
147
148#include "generated_stub_builtin_function_map.h"
149
Tom Cherry3f1bce82019-06-05 09:13:11 -0700150void PrintUsage() {
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800151 std::cout << "usage: host_init_verifier [options] <init rc file>\n"
Tom Cherry3f1bce82019-06-05 09:13:11 -0700152 "\n"
153 "Tests an init script for correctness\n"
154 "\n"
155 "-p FILE\tSearch this passwd file for users and groups\n"
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800156 "--property_contexts=FILE\t Use this file for property_contexts\n"
Tom Cherry3f1bce82019-06-05 09:13:11 -0700157 << std::endl;
158}
159
Steven Moreland422a7582019-10-15 14:53:19 -0700160Result<InterfaceInheritanceHierarchyMap> ReadInterfaceInheritanceHierarchy() {
161 InterfaceInheritanceHierarchyMap result;
162 for (const HidlInterfaceMetadata& iface : HidlInterfaceMetadata::all()) {
163 std::set<FQName> inherited_interfaces;
164 for (const std::string& intf : iface.inherited) {
165 FQName fqname;
166 if (!fqname.setTo(intf)) {
167 return Error() << "Unable to parse interface '" << intf << "'";
168 }
169 inherited_interfaces.insert(fqname);
170 }
171 FQName fqname;
172 if (!fqname.setTo(iface.name)) {
173 return Error() << "Unable to parse interface '" << iface.name << "'";
174 }
175 result[fqname] = inherited_interfaces;
176 }
177
178 return result;
179}
180
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800181const PropertyInfoArea* property_info_area;
182
183void HandlePropertyContexts(const std::string& filename,
184 std::vector<PropertyInfoEntry>* property_infos) {
185 auto file_contents = std::string();
186 if (!ReadFileToString(filename, &file_contents)) {
187 PLOG(ERROR) << "Could not read properties from '" << filename << "'";
188 exit(EXIT_FAILURE);
189 }
190
191 auto errors = std::vector<std::string>{};
Tom Cherry4b077c52019-12-11 07:56:51 -0800192 ParsePropertyInfoFile(file_contents, true, property_infos, &errors);
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800193 for (const auto& error : errors) {
194 LOG(ERROR) << "Could not read line from '" << filename << "': " << error;
195 }
196 if (!errors.empty()) {
197 exit(EXIT_FAILURE);
198 }
199}
200
Tom Cherryde6bd502018-02-13 16:50:08 -0800201int main(int argc, char** argv) {
Elliott Hughes1be0d142018-05-23 09:16:46 -0700202 android::base::InitLogging(argv, &android::base::StdioLogger);
Tom Cherry194b5d12018-05-09 17:38:30 -0700203 android::base::SetMinimumLogSeverity(android::base::ERROR);
Tom Cherry863d8082018-06-12 14:40:38 -0700204
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800205 auto property_infos = std::vector<PropertyInfoEntry>();
206
Tom Cherry3f1bce82019-06-05 09:13:11 -0700207 while (true) {
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800208 static const char kPropertyContexts[] = "property-contexts=";
Tom Cherry3f1bce82019-06-05 09:13:11 -0700209 static const struct option long_options[] = {
210 {"help", no_argument, nullptr, 'h'},
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800211 {kPropertyContexts, required_argument, nullptr, 0},
Tom Cherry3f1bce82019-06-05 09:13:11 -0700212 {nullptr, 0, nullptr, 0},
213 };
214
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800215 int option_index;
216 int arg = getopt_long(argc, argv, "p:", long_options, &option_index);
Tom Cherry3f1bce82019-06-05 09:13:11 -0700217
218 if (arg == -1) {
219 break;
220 }
221
222 switch (arg) {
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800223 case 0:
224 if (long_options[option_index].name == kPropertyContexts) {
225 HandlePropertyContexts(optarg, &property_infos);
226 }
227 break;
Tom Cherry3f1bce82019-06-05 09:13:11 -0700228 case 'h':
229 PrintUsage();
230 return EXIT_FAILURE;
231 case 'p':
232 passwd_files.emplace_back(optarg);
233 break;
234 default:
235 std::cerr << "getprop: getopt returned invalid result: " << arg << std::endl;
236 return EXIT_FAILURE;
237 }
Tom Cherryde6bd502018-02-13 16:50:08 -0800238 }
Tom Cherry194b5d12018-05-09 17:38:30 -0700239
Tom Cherry3f1bce82019-06-05 09:13:11 -0700240 argc -= optind;
241 argv += optind;
242
Steven Moreland422a7582019-10-15 14:53:19 -0700243 if (argc != 1) {
Tom Cherry3f1bce82019-06-05 09:13:11 -0700244 PrintUsage();
245 return EXIT_FAILURE;
Tom Cherry194b5d12018-05-09 17:38:30 -0700246 }
247
Steven Moreland422a7582019-10-15 14:53:19 -0700248 auto interface_inheritance_hierarchy_map = ReadInterfaceInheritanceHierarchy();
Bernie Innocenticecebbb2020-02-06 03:49:33 +0900249 if (!interface_inheritance_hierarchy_map.ok()) {
Daniel Normand2533c32019-08-02 15:13:50 -0700250 LOG(ERROR) << interface_inheritance_hierarchy_map.error();
251 return EXIT_FAILURE;
252 }
253 SetKnownInterfaces(*interface_inheritance_hierarchy_map);
254
Tom Cherryb5f2ec02019-11-08 17:54:27 -0800255 std::string serialized_contexts;
256 std::string trie_error;
257 if (!BuildTrie(property_infos, "u:object_r:default_prop:s0", "string", &serialized_contexts,
258 &trie_error)) {
259 LOG(ERROR) << "Unable to serialize property contexts: " << trie_error;
260 return EXIT_FAILURE;
261 }
262
263 property_info_area = reinterpret_cast<const PropertyInfoArea*>(serialized_contexts.c_str());
264
Tom Cherryd52a5b32019-07-22 16:05:36 -0700265 const BuiltinFunctionMap& function_map = GetBuiltinFunctionMap();
Tom Cherryde6bd502018-02-13 16:50:08 -0800266 Action::set_function_map(&function_map);
267 ActionManager& am = ActionManager::GetInstance();
268 ServiceList& sl = ServiceList::GetInstance();
269 Parser parser;
Daniel Normand2533c32019-08-02 15:13:50 -0700270 parser.AddSectionParser("service", std::make_unique<ServiceParser>(
271 &sl, nullptr, *interface_inheritance_hierarchy_map));
Tom Cherryde6bd502018-02-13 16:50:08 -0800272 parser.AddSectionParser("on", std::make_unique<ActionParser>(&am, nullptr));
Tom Cherry863d8082018-06-12 14:40:38 -0700273 parser.AddSectionParser("import", std::make_unique<HostImportParser>());
Tom Cherryde6bd502018-02-13 16:50:08 -0800274
Tom Cherry3f1bce82019-06-05 09:13:11 -0700275 if (!parser.ParseConfigFileInsecure(*argv)) {
276 LOG(ERROR) << "Failed to open init rc script '" << *argv << "'";
Tom Cherry863d8082018-06-12 14:40:38 -0700277 return EXIT_FAILURE;
Tom Cherryde6bd502018-02-13 16:50:08 -0800278 }
Tom Cherry6737a6b2019-08-05 15:03:58 -0700279 size_t failures = parser.parse_error_count() + am.CheckAllCommands() + sl.CheckAllCommands();
Tom Cherry4772f1d2019-07-30 09:34:41 -0700280 if (failures > 0) {
281 LOG(ERROR) << "Failed to parse init script '" << *argv << "' with " << failures
282 << " errors";
Tom Cherry863d8082018-06-12 14:40:38 -0700283 return EXIT_FAILURE;
Tom Cherryde6bd502018-02-13 16:50:08 -0800284 }
Tom Cherry863d8082018-06-12 14:40:38 -0700285 return EXIT_SUCCESS;
Tom Cherryde6bd502018-02-13 16:50:08 -0800286}
287
288} // namespace init
289} // namespace android
290
291int main(int argc, char** argv) {
Tom Cherry863d8082018-06-12 14:40:38 -0700292 return android::init::main(argc, argv);
Tom Cherryde6bd502018-02-13 16:50:08 -0800293}