blob: fadc7cad675e1e5ded08dddcc335f2e9c6f17d7b [file] [log] [blame]
Ken Sumrallc1bf8962012-01-06 19:09:42 -08001/*
2 * Copyright (C) 2012 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
Ken Sumrallc1bf8962012-01-06 19:09:42 -080017#include <stdio.h>
18#include <stdlib.h>
19#include <string.h>
20#include <unistd.h>
21#include <fcntl.h>
22#include <ctype.h>
23#include <sys/mount.h>
24#include <sys/stat.h>
25#include <errno.h>
26#include <sys/types.h>
27#include <sys/wait.h>
28#include <libgen.h>
29#include <time.h>
Ken Sumrall5bc31a22013-07-08 19:11:55 -070030#include <sys/swap.h>
Christoffer Dall82982342014-12-17 21:26:54 +010031#include <dirent.h>
32#include <ext4.h>
33#include <ext4_sb.h>
Paul Lawrence806d10b2015-04-28 22:07:10 +000034#include <ext4_crypt_init_extensions.h>
Ken Sumrallc1bf8962012-01-06 19:09:42 -080035
Geremy Condra3ad3d1c2013-02-22 18:11:41 -080036#include <linux/loop.h>
Ken Sumrallc1bf8962012-01-06 19:09:42 -080037#include <private/android_filesystem_config.h>
JP Abgrall4bb7bba2014-06-19 22:12:20 -070038#include <cutils/android_reboot.h>
Ken Sumrallc1bf8962012-01-06 19:09:42 -080039#include <cutils/partition_utils.h>
40#include <cutils/properties.h>
Ken Sumrallbf021b42013-03-19 19:38:44 -070041#include <logwrap/logwrap.h>
Ken Sumrallc1bf8962012-01-06 19:09:42 -080042
Geremy Condra3ad3d1c2013-02-22 18:11:41 -080043#include "mincrypt/rsa.h"
44#include "mincrypt/sha.h"
45#include "mincrypt/sha256.h"
46
Chris Fries79f33842013-09-05 13:19:21 -050047#include "ext4_utils.h"
48#include "wipe.h"
49
Ken Sumrallc1bf8962012-01-06 19:09:42 -080050#include "fs_mgr_priv.h"
Geremy Condra3ad3d1c2013-02-22 18:11:41 -080051#include "fs_mgr_priv_verity.h"
Ken Sumrallc1bf8962012-01-06 19:09:42 -080052
53#define KEY_LOC_PROP "ro.crypto.keyfile.userdata"
54#define KEY_IN_FOOTER "footer"
55
56#define E2FSCK_BIN "/system/bin/e2fsck"
JP Abgrall12351582014-06-17 17:01:14 -070057#define F2FS_FSCK_BIN "/system/bin/fsck.f2fs"
Ken Sumrall5bc31a22013-07-08 19:11:55 -070058#define MKSWAP_BIN "/system/bin/mkswap"
59
Ken Sumrall4eaf9052013-09-18 17:49:21 -070060#define FSCK_LOG_FILE "/dev/fscklogs/log"
61
Ken Sumrall5bc31a22013-07-08 19:11:55 -070062#define ZRAM_CONF_DEV "/sys/block/zram0/disksize"
Ken Sumrallc1bf8962012-01-06 19:09:42 -080063
Ken Sumrallbf021b42013-03-19 19:38:44 -070064#define ARRAY_SIZE(a) (sizeof(a) / sizeof(*(a)))
65
Ken Sumrallc1bf8962012-01-06 19:09:42 -080066/*
67 * gettime() - returns the time in seconds of the system's monotonic clock or
68 * zero on error.
69 */
70static time_t gettime(void)
71{
72 struct timespec ts;
73 int ret;
74
75 ret = clock_gettime(CLOCK_MONOTONIC, &ts);
76 if (ret < 0) {
77 ERROR("clock_gettime(CLOCK_MONOTONIC) failed: %s\n", strerror(errno));
78 return 0;
79 }
80
81 return ts.tv_sec;
82}
83
84static int wait_for_file(const char *filename, int timeout)
85{
86 struct stat info;
87 time_t timeout_time = gettime() + timeout;
88 int ret = -1;
89
90 while (gettime() < timeout_time && ((ret = stat(filename, &info)) < 0))
91 usleep(10000);
92
93 return ret;
94}
95
Ken Sumrallab6b8522013-02-13 12:58:40 -080096static void check_fs(char *blk_device, char *fs_type, char *target)
Ken Sumrallc1bf8962012-01-06 19:09:42 -080097{
Ken Sumrallc1bf8962012-01-06 19:09:42 -080098 int status;
Ken Sumrall5dc5bfe2012-07-23 19:34:00 -070099 int ret;
100 long tmpmnt_flags = MS_NOATIME | MS_NOEXEC | MS_NOSUID;
Oreste Salerno6ed84c92015-05-20 17:01:39 +0000101 char tmpmnt_opts[64] = "errors=remount-ro";
Ken Sumrallbf021b42013-03-19 19:38:44 -0700102 char *e2fsck_argv[] = {
103 E2FSCK_BIN,
104 "-y",
105 blk_device
106 };
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800107
108 /* Check for the types of filesystems we know how to check */
Ken Sumrallab6b8522013-02-13 12:58:40 -0800109 if (!strcmp(fs_type, "ext2") || !strcmp(fs_type, "ext3") || !strcmp(fs_type, "ext4")) {
Ken Sumrall5dc5bfe2012-07-23 19:34:00 -0700110 /*
111 * First try to mount and unmount the filesystem. We do this because
112 * the kernel is more efficient than e2fsck in running the journal and
113 * processing orphaned inodes, and on at least one device with a
114 * performance issue in the emmc firmware, it can take e2fsck 2.5 minutes
115 * to do what the kernel does in about a second.
116 *
117 * After mounting and unmounting the filesystem, run e2fsck, and if an
118 * error is recorded in the filesystem superblock, e2fsck will do a full
119 * check. Otherwise, it does nothing. If the kernel cannot mount the
120 * filesytsem due to an error, e2fsck is still run to do a full check
121 * fix the filesystem.
122 */
Elliott Hughes5e7dd442015-04-24 11:05:48 -0700123 errno = 0;
Oreste Salerno6ed84c92015-05-20 17:01:39 +0000124 if (!strcmp(fs_type, "ext4")) {
125 // This option is only valid with ext4
126 strlcat(tmpmnt_opts, ",nomblk_io_submit", sizeof(tmpmnt_opts));
127 }
Ken Sumrallab6b8522013-02-13 12:58:40 -0800128 ret = mount(blk_device, target, fs_type, tmpmnt_flags, tmpmnt_opts);
Elliott Hughes5e7dd442015-04-24 11:05:48 -0700129 INFO("%s(): mount(%s,%s,%s)=%d: %s\n",
130 __func__, blk_device, target, fs_type, ret, strerror(errno));
Ken Sumrallab6b8522013-02-13 12:58:40 -0800131 if (!ret) {
Nick Kralevich7294eb62015-02-05 16:02:42 -0800132 int i;
133 for (i = 0; i < 5; i++) {
134 // Try to umount 5 times before continuing on.
135 // Should we try rebooting if all attempts fail?
136 int result = umount(target);
137 if (result == 0) {
Elliott Hughes5e7dd442015-04-24 11:05:48 -0700138 INFO("%s(): unmount(%s) succeeded\n", __func__, target);
Nick Kralevich7294eb62015-02-05 16:02:42 -0800139 break;
140 }
141 ERROR("%s(): umount(%s)=%d: %s\n", __func__, target, result, strerror(errno));
142 sleep(1);
143 }
Ken Sumrall5dc5bfe2012-07-23 19:34:00 -0700144 }
145
David 'Digit' Turner28483d72014-02-17 11:14:44 +0100146 /*
147 * Some system images do not have e2fsck for licensing reasons
148 * (e.g. recent SDK system images). Detect these and skip the check.
149 */
150 if (access(E2FSCK_BIN, X_OK)) {
151 INFO("Not running %s on %s (executable not in system image)\n",
152 E2FSCK_BIN, blk_device);
153 } else {
154 INFO("Running %s on %s\n", E2FSCK_BIN, blk_device);
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800155
David 'Digit' Turner28483d72014-02-17 11:14:44 +0100156 ret = android_fork_execvp_ext(ARRAY_SIZE(e2fsck_argv), e2fsck_argv,
Yusuke Satod81c3c62015-08-14 01:22:53 -0700157 &status, true, LOG_KLOG | LOG_FILE,
158 true, FSCK_LOG_FILE, NULL, 0);
Ken Sumrallbf021b42013-03-19 19:38:44 -0700159
David 'Digit' Turner28483d72014-02-17 11:14:44 +0100160 if (ret < 0) {
161 /* No need to check for error in fork, we can't really handle it now */
162 ERROR("Failed trying to run %s\n", E2FSCK_BIN);
163 }
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800164 }
JP Abgrall12351582014-06-17 17:01:14 -0700165 } else if (!strcmp(fs_type, "f2fs")) {
166 char *f2fs_fsck_argv[] = {
167 F2FS_FSCK_BIN,
Yusuke Sato0df08272015-07-08 14:57:07 -0700168 "-a",
JP Abgrall12351582014-06-17 17:01:14 -0700169 blk_device
170 };
Yusuke Sato0df08272015-07-08 14:57:07 -0700171 INFO("Running %s -a %s\n", F2FS_FSCK_BIN, blk_device);
JP Abgrall12351582014-06-17 17:01:14 -0700172
173 ret = android_fork_execvp_ext(ARRAY_SIZE(f2fs_fsck_argv), f2fs_fsck_argv,
174 &status, true, LOG_KLOG | LOG_FILE,
Yusuke Satod81c3c62015-08-14 01:22:53 -0700175 true, FSCK_LOG_FILE, NULL, 0);
JP Abgrall12351582014-06-17 17:01:14 -0700176 if (ret < 0) {
177 /* No need to check for error in fork, we can't really handle it now */
178 ERROR("Failed trying to run %s\n", F2FS_FSCK_BIN);
179 }
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800180 }
181
182 return;
183}
184
185static void remove_trailing_slashes(char *n)
186{
187 int len;
188
189 len = strlen(n) - 1;
190 while ((*(n + len) == '/') && len) {
191 *(n + len) = '\0';
192 len--;
193 }
194}
195
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700196/*
197 * Mark the given block device as read-only, using the BLKROSET ioctl.
198 * Return 0 on success, and -1 on error.
199 */
Sami Tolvanen214f33b2014-12-18 16:15:30 +0000200int fs_mgr_set_blk_ro(const char *blockdev)
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700201{
202 int fd;
Sami Tolvanen214f33b2014-12-18 16:15:30 +0000203 int rc = -1;
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700204 int ON = 1;
205
Sami Tolvanen214f33b2014-12-18 16:15:30 +0000206 fd = TEMP_FAILURE_RETRY(open(blockdev, O_RDONLY | O_CLOEXEC));
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700207 if (fd < 0) {
208 // should never happen
Sami Tolvanen214f33b2014-12-18 16:15:30 +0000209 return rc;
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700210 }
211
Sami Tolvanen214f33b2014-12-18 16:15:30 +0000212 rc = ioctl(fd, BLKROSET, &ON);
Elliott Hughes9fc83432015-05-15 19:16:40 -0700213 close(fd);
Sami Tolvanen214f33b2014-12-18 16:15:30 +0000214
215 return rc;
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700216}
217
218/*
219 * __mount(): wrapper around the mount() system call which also
220 * sets the underlying block device to read-only if the mount is read-only.
221 * See "man 2 mount" for return values.
222 */
JP Abgrall5c01dac2014-06-18 14:54:37 -0700223static int __mount(const char *source, const char *target, const struct fstab_rec *rec)
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700224{
JP Abgrall5c01dac2014-06-18 14:54:37 -0700225 unsigned long mountflags = rec->flags;
226 int ret;
227 int save_errno;
Daniel Rosenbergf67d6bd2014-06-26 14:55:04 -0700228
229 /* We need this because sometimes we have legacy symlinks
230 * that are lingering around and need cleaning up.
231 */
232 struct stat info;
233 if (!lstat(target, &info))
234 if ((info.st_mode & S_IFMT) == S_IFLNK)
235 unlink(target);
Daniel Rosenbergf530c932014-05-28 14:10:01 -0700236 mkdir(target, 0755);
JP Abgrall5c01dac2014-06-18 14:54:37 -0700237 ret = mount(source, target, rec->fs_type, mountflags, rec->fs_options);
238 save_errno = errno;
239 INFO("%s(source=%s,target=%s,type=%s)=%d\n", __func__, source, target, rec->fs_type, ret);
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700240 if ((ret == 0) && (mountflags & MS_RDONLY) != 0) {
Sami Tolvanen214f33b2014-12-18 16:15:30 +0000241 fs_mgr_set_blk_ro(source);
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700242 }
JP Abgrall5c01dac2014-06-18 14:54:37 -0700243 errno = save_errno;
Nick Kraleviche18c0d52013-04-16 16:41:32 -0700244 return ret;
245}
246
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800247static int fs_match(char *in1, char *in2)
248{
249 char *n1;
250 char *n2;
251 int ret;
252
253 n1 = strdup(in1);
254 n2 = strdup(in2);
255
256 remove_trailing_slashes(n1);
257 remove_trailing_slashes(n2);
258
259 ret = !strcmp(n1, n2);
260
261 free(n1);
262 free(n2);
263
264 return ret;
265}
266
Geremy Condracd642fc2014-04-02 13:42:06 -0700267static int device_is_debuggable() {
268 int ret = -1;
269 char value[PROP_VALUE_MAX];
270 ret = __system_property_get("ro.debuggable", value);
271 if (ret < 0)
272 return ret;
273 return strcmp(value, "1") ? 0 : 1;
274}
275
Paul Lawrencebbb36312014-10-09 14:22:49 +0000276static int device_is_secure() {
277 int ret = -1;
278 char value[PROP_VALUE_MAX];
279 ret = __system_property_get("ro.secure", value);
280 /* If error, we want to fail secure */
281 if (ret < 0)
282 return 1;
283 return strcmp(value, "0") ? 1 : 0;
284}
285
Paul Lawrence703b87d2015-01-07 11:44:51 -0800286static int device_is_force_encrypted() {
287 int ret = -1;
288 char value[PROP_VALUE_MAX];
289 ret = __system_property_get("ro.vold.forceencryption", value);
290 if (ret < 0)
291 return 0;
292 return strcmp(value, "1") ? 0 : 1;
293}
294
JP Abgrallf22b7452014-07-02 13:16:04 -0700295/*
296 * Tries to mount any of the consecutive fstab entries that match
297 * the mountpoint of the one given by fstab->recs[start_idx].
298 *
299 * end_idx: On return, will be the last rec that was looked at.
300 * attempted_idx: On return, will indicate which fstab rec
301 * succeeded. In case of failure, it will be the start_idx.
302 * Returns
303 * -1 on failure with errno set to match the 1st mount failure.
304 * 0 on success.
305 */
306static int mount_with_alternatives(struct fstab *fstab, int start_idx, int *end_idx, int *attempted_idx)
JP Abgrall4bb7bba2014-06-19 22:12:20 -0700307{
JP Abgrallf22b7452014-07-02 13:16:04 -0700308 int i;
309 int mount_errno = 0;
310 int mounted = 0;
311
312 if (!end_idx || !attempted_idx || start_idx >= fstab->num_entries) {
313 errno = EINVAL;
314 if (end_idx) *end_idx = start_idx;
315 if (attempted_idx) *end_idx = start_idx;
316 return -1;
JP Abgrall4bb7bba2014-06-19 22:12:20 -0700317 }
JP Abgrallf22b7452014-07-02 13:16:04 -0700318
319 /* Hunt down an fstab entry for the same mount point that might succeed */
320 for (i = start_idx;
321 /* We required that fstab entries for the same mountpoint be consecutive */
322 i < fstab->num_entries && !strcmp(fstab->recs[start_idx].mount_point, fstab->recs[i].mount_point);
323 i++) {
324 /*
325 * Don't try to mount/encrypt the same mount point again.
326 * Deal with alternate entries for the same point which are required to be all following
327 * each other.
328 */
329 if (mounted) {
330 ERROR("%s(): skipping fstab dup mountpoint=%s rec[%d].fs_type=%s already mounted as %s.\n", __func__,
331 fstab->recs[i].mount_point, i, fstab->recs[i].fs_type, fstab->recs[*attempted_idx].fs_type);
332 continue;
333 }
334
335 if (fstab->recs[i].fs_mgr_flags & MF_CHECK) {
336 check_fs(fstab->recs[i].blk_device, fstab->recs[i].fs_type,
337 fstab->recs[i].mount_point);
338 }
339 if (!__mount(fstab->recs[i].blk_device, fstab->recs[i].mount_point, &fstab->recs[i])) {
340 *attempted_idx = i;
341 mounted = 1;
342 if (i != start_idx) {
343 ERROR("%s(): Mounted %s on %s with fs_type=%s instead of %s\n", __func__,
344 fstab->recs[i].blk_device, fstab->recs[i].mount_point, fstab->recs[i].fs_type,
345 fstab->recs[start_idx].fs_type);
346 }
347 } else {
348 /* back up errno for crypto decisions */
349 mount_errno = errno;
350 }
351 }
352
353 /* Adjust i for the case where it was still withing the recs[] */
354 if (i < fstab->num_entries) --i;
355
356 *end_idx = i;
357 if (!mounted) {
358 *attempted_idx = start_idx;
359 errno = mount_errno;
360 return -1;
361 }
362 return 0;
JP Abgrall4bb7bba2014-06-19 22:12:20 -0700363}
364
Christoffer Dall82982342014-12-17 21:26:54 +0100365static int translate_ext_labels(struct fstab_rec *rec)
366{
367 DIR *blockdir = NULL;
368 struct dirent *ent;
369 char *label;
370 size_t label_len;
371 int ret = -1;
372
373 if (strncmp(rec->blk_device, "LABEL=", 6))
374 return 0;
375
376 label = rec->blk_device + 6;
377 label_len = strlen(label);
378
379 if (label_len > 16) {
380 ERROR("FS label is longer than allowed by filesystem\n");
381 goto out;
382 }
383
384
385 blockdir = opendir("/dev/block");
386 if (!blockdir) {
387 ERROR("couldn't open /dev/block\n");
388 goto out;
389 }
390
391 while ((ent = readdir(blockdir))) {
392 int fd;
393 char super_buf[1024];
394 struct ext4_super_block *sb;
395
396 if (ent->d_type != DT_BLK)
397 continue;
398
399 fd = openat(dirfd(blockdir), ent->d_name, O_RDONLY);
400 if (fd < 0) {
401 ERROR("Cannot open block device /dev/block/%s\n", ent->d_name);
402 goto out;
403 }
404
405 if (TEMP_FAILURE_RETRY(lseek(fd, 1024, SEEK_SET)) < 0 ||
406 TEMP_FAILURE_RETRY(read(fd, super_buf, 1024)) != 1024) {
407 /* Probably a loopback device or something else without a readable
408 * superblock.
409 */
410 close(fd);
411 continue;
412 }
413
414 sb = (struct ext4_super_block *)super_buf;
415 if (sb->s_magic != EXT4_SUPER_MAGIC) {
416 INFO("/dev/block/%s not ext{234}\n", ent->d_name);
417 continue;
418 }
419
420 if (!strncmp(label, sb->s_volume_name, label_len)) {
421 char *new_blk_device;
422
423 if (asprintf(&new_blk_device, "/dev/block/%s", ent->d_name) < 0) {
424 ERROR("Could not allocate block device string\n");
425 goto out;
426 }
427
428 INFO("resolved label %s to %s\n", rec->blk_device, new_blk_device);
429
430 free(rec->blk_device);
431 rec->blk_device = new_blk_device;
432 ret = 0;
433 break;
434 }
435 }
436
437out:
438 closedir(blockdir);
439 return ret;
440}
441
Paul Lawrenceb8c9d272015-03-26 15:49:42 +0000442// Check to see if a mountable volume has encryption requirements
443static int handle_encryptable(struct fstab *fstab, const struct fstab_rec* rec)
444{
Paul Lawrenceaecb1e22015-11-06 11:18:48 -0800445 /* Check for existence of convert_fbe breadcrumb file */
446 char convert_fbe_name[PATH_MAX];
447 snprintf(convert_fbe_name, sizeof(convert_fbe_name),
448 "%s/convert_fbe", rec->mount_point);
449 bool convert_fbe = (access(convert_fbe_name, F_OK) == 0);
450
Paul Lawrence8d633832015-11-20 07:07:12 -0800451 /* Check for existence of convert_fbe breadcrumb file */
452 char convert_fde_name[PATH_MAX];
453 snprintf(convert_fde_name, sizeof(convert_fbe_name),
454 "%s/misc/vold/convert_fde", rec->mount_point);
455 bool convert_fde = (access(convert_fde_name, F_OK) == 0);
456
Paul Lawrenceb8c9d272015-03-26 15:49:42 +0000457 /* If this is block encryptable, need to trigger encryption */
458 if ( (rec->fs_mgr_flags & MF_FORCECRYPT)
Paul Lawrence8d633832015-11-20 07:07:12 -0800459 || ((rec->fs_mgr_flags & MF_CRYPT) && convert_fde)
Paul Lawrenceaecb1e22015-11-06 11:18:48 -0800460 || ((rec->fs_mgr_flags & MF_FORCEFDEORFBE) && !convert_fbe)
Paul Lawrenceb8c9d272015-03-26 15:49:42 +0000461 || (device_is_force_encrypted() && fs_mgr_is_encryptable(rec))) {
462 if (umount(rec->mount_point) == 0) {
463 return FS_MGR_MNTALL_DEV_NEEDS_ENCRYPTION;
464 } else {
465 WARNING("Could not umount %s (%s) - allow continue unencrypted\n",
466 rec->mount_point, strerror(errno));
467 return FS_MGR_MNTALL_DEV_NOT_ENCRYPTED;
468 }
469 }
470
471 // Deal with file level encryption
Paul Lawrenceaecb1e22015-11-06 11:18:48 -0800472 if ( (rec->fs_mgr_flags & MF_FILEENCRYPTION)
473 || ((rec->fs_mgr_flags & MF_FORCEFDEORFBE) && convert_fbe)) {
Paul Lawrenceb8c9d272015-03-26 15:49:42 +0000474 // Default or not yet initialized encryption requires no more work here
475 if (!e4crypt_non_default_key(rec->mount_point)) {
476 INFO("%s is default file encrypted\n", rec->mount_point);
477 return FS_MGR_MNTALL_DEV_DEFAULT_FILE_ENCRYPTED;
478 }
479
480 INFO("%s is non-default file encrypted\n", rec->mount_point);
481
482 // Uses non-default key, so must unmount and set up temp file system
483 if (umount(rec->mount_point)) {
484 ERROR("Failed to umount %s - rebooting\n", rec->mount_point);
485 return FS_MGR_MNTALL_FAIL;
486 }
487
488 if (fs_mgr_do_tmpfs_mount(rec->mount_point) != 0) {
489 ERROR("Failed to mount a tmpfs at %s\n", rec->mount_point);
490 return FS_MGR_MNTALL_FAIL;
491 }
492
493 // Mount data temporarily so we can access unencrypted dir
494 char tmp_mnt[PATH_MAX];
495 strlcpy(tmp_mnt, rec->mount_point, sizeof(tmp_mnt));
496 strlcat(tmp_mnt, "/tmp_mnt", sizeof(tmp_mnt));
497 if (mkdir(tmp_mnt, 0700)) {
498 ERROR("Failed to create temp mount point\n");
499 return FS_MGR_MNTALL_FAIL;
500 }
501
502 if (fs_mgr_do_mount(fstab, rec->mount_point,
503 rec->blk_device, tmp_mnt)) {
504 ERROR("Error temp mounting encrypted file system\n");
505 return FS_MGR_MNTALL_FAIL;
506 }
507
Paul Lawrenceb8c9d272015-03-26 15:49:42 +0000508 return FS_MGR_MNTALL_DEV_NON_DEFAULT_FILE_ENCRYPTED;
509 }
510
511 return FS_MGR_MNTALL_DEV_NOT_ENCRYPTED;
512}
513
JP Abgrall5c01dac2014-06-18 14:54:37 -0700514/* When multiple fstab records share the same mount_point, it will
515 * try to mount each one in turn, and ignore any duplicates after a
516 * first successful mount.
JP Abgrallf22b7452014-07-02 13:16:04 -0700517 * Returns -1 on error, and FS_MGR_MNTALL_* otherwise.
JP Abgrall5c01dac2014-06-18 14:54:37 -0700518 */
Ken Sumrallab6b8522013-02-13 12:58:40 -0800519int fs_mgr_mount_all(struct fstab *fstab)
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800520{
JP Abgrallf22b7452014-07-02 13:16:04 -0700521 int i = 0;
522 int encryptable = FS_MGR_MNTALL_DEV_NOT_ENCRYPTED;
Mohamad Ayyash38afe5f2014-03-10 15:40:29 -0700523 int error_count = 0;
JP Abgrall5c01dac2014-06-18 14:54:37 -0700524 int mret = -1;
525 int mount_errno = 0;
JP Abgrallf22b7452014-07-02 13:16:04 -0700526 int attempted_idx = -1;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800527
Ken Sumrallab6b8522013-02-13 12:58:40 -0800528 if (!fstab) {
Mohamad Ayyash38afe5f2014-03-10 15:40:29 -0700529 return -1;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800530 }
531
Ken Sumrallab6b8522013-02-13 12:58:40 -0800532 for (i = 0; i < fstab->num_entries; i++) {
533 /* Don't mount entries that are managed by vold */
Ken Sumrall6c2c1212013-02-22 17:36:21 -0800534 if (fstab->recs[i].fs_mgr_flags & (MF_VOLDMANAGED | MF_RECOVERYONLY)) {
Ken Sumrallab6b8522013-02-13 12:58:40 -0800535 continue;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800536 }
537
Ken Sumrall5bc31a22013-07-08 19:11:55 -0700538 /* Skip swap and raw partition entries such as boot, recovery, etc */
539 if (!strcmp(fstab->recs[i].fs_type, "swap") ||
540 !strcmp(fstab->recs[i].fs_type, "emmc") ||
Ken Sumrallab6b8522013-02-13 12:58:40 -0800541 !strcmp(fstab->recs[i].fs_type, "mtd")) {
542 continue;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800543 }
544
Daniel Rosenberg31a4faf2015-06-29 17:33:05 -0700545 /* Skip mounting the root partition, as it will already have been mounted */
546 if (!strcmp(fstab->recs[i].mount_point, "/")) {
547 if ((fstab->recs[i].fs_mgr_flags & MS_RDONLY) != 0) {
548 fs_mgr_set_blk_ro(fstab->recs[i].blk_device);
549 }
550 continue;
551 }
552
Christoffer Dall82982342014-12-17 21:26:54 +0100553 /* Translate LABEL= file system labels into block devices */
554 if (!strcmp(fstab->recs[i].fs_type, "ext2") ||
555 !strcmp(fstab->recs[i].fs_type, "ext3") ||
556 !strcmp(fstab->recs[i].fs_type, "ext4")) {
557 int tret = translate_ext_labels(&fstab->recs[i]);
558 if (tret < 0) {
559 ERROR("Could not translate label to block device\n");
560 continue;
561 }
562 }
563
Ken Sumrallab6b8522013-02-13 12:58:40 -0800564 if (fstab->recs[i].fs_mgr_flags & MF_WAIT) {
565 wait_for_file(fstab->recs[i].blk_device, WAIT_TIMEOUT);
566 }
567
Paul Lawrencebbb36312014-10-09 14:22:49 +0000568 if ((fstab->recs[i].fs_mgr_flags & MF_VERIFY) && device_is_secure()) {
569 int rc = fs_mgr_setup_verity(&fstab->recs[i]);
570 if (device_is_debuggable() && rc == FS_MGR_SETUP_VERITY_DISABLED) {
571 INFO("Verity disabled");
572 } else if (rc != FS_MGR_SETUP_VERITY_SUCCESS) {
JP Abgrall4bb7bba2014-06-19 22:12:20 -0700573 ERROR("Could not set up verified partition, skipping!\n");
Geremy Condra3ad3d1c2013-02-22 18:11:41 -0800574 continue;
575 }
576 }
JP Abgrallf22b7452014-07-02 13:16:04 -0700577 int last_idx_inspected;
Chris Fries79f33842013-09-05 13:19:21 -0500578 int top_idx = i;
579
JP Abgrallf22b7452014-07-02 13:16:04 -0700580 mret = mount_with_alternatives(fstab, i, &last_idx_inspected, &attempted_idx);
581 i = last_idx_inspected;
582 mount_errno = errno;
JP Abgrallf786fe52014-06-18 07:28:14 +0000583
JP Abgrall5c01dac2014-06-18 14:54:37 -0700584 /* Deal with encryptability. */
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800585 if (!mret) {
Paul Lawrenceb8c9d272015-03-26 15:49:42 +0000586 int status = handle_encryptable(fstab, &fstab->recs[attempted_idx]);
587
588 if (status == FS_MGR_MNTALL_FAIL) {
589 /* Fatal error - no point continuing */
590 return status;
Paul Lawrence166fa3d2014-02-03 13:27:49 -0800591 }
Paul Lawrenceb8c9d272015-03-26 15:49:42 +0000592
593 if (status != FS_MGR_MNTALL_DEV_NOT_ENCRYPTED) {
594 if (encryptable != FS_MGR_MNTALL_DEV_NOT_ENCRYPTED) {
595 // Log and continue
596 ERROR("Only one encryptable/encrypted partition supported\n");
597 }
598 encryptable = status;
599 }
600
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800601 /* Success! Go get the next one */
602 continue;
603 }
604
Chris Fries79f33842013-09-05 13:19:21 -0500605 /* mount(2) returned an error, handle the encryptable/formattable case */
606 bool wiped = partition_wiped(fstab->recs[top_idx].blk_device);
607 if (mret && mount_errno != EBUSY && mount_errno != EACCES &&
608 fs_mgr_is_formattable(&fstab->recs[top_idx]) && wiped) {
609 /* top_idx and attempted_idx point at the same partition, but sometimes
610 * at two different lines in the fstab. Use the top one for formatting
611 * as that is the preferred one.
612 */
613 ERROR("%s(): %s is wiped and %s %s is formattable. Format it.\n", __func__,
614 fstab->recs[top_idx].blk_device, fstab->recs[top_idx].mount_point,
615 fstab->recs[top_idx].fs_type);
616 if (fs_mgr_is_encryptable(&fstab->recs[top_idx]) &&
617 strcmp(fstab->recs[top_idx].key_loc, KEY_IN_FOOTER)) {
618 int fd = open(fstab->recs[top_idx].key_loc, O_WRONLY, 0644);
619 if (fd >= 0) {
620 INFO("%s(): also wipe %s\n", __func__, fstab->recs[top_idx].key_loc);
621 wipe_block_device(fd, get_file_size(fd));
622 close(fd);
623 } else {
624 ERROR("%s(): %s wouldn't open (%s)\n", __func__,
625 fstab->recs[top_idx].key_loc, strerror(errno));
626 }
627 }
628 if (fs_mgr_do_format(&fstab->recs[top_idx]) == 0) {
629 /* Let's replay the mount actions. */
630 i = top_idx - 1;
631 continue;
632 }
633 }
JP Abgrall5c01dac2014-06-18 14:54:37 -0700634 if (mret && mount_errno != EBUSY && mount_errno != EACCES &&
JP Abgrallcee20682014-07-02 14:26:54 -0700635 fs_mgr_is_encryptable(&fstab->recs[attempted_idx])) {
Chris Fries79f33842013-09-05 13:19:21 -0500636 if (wiped) {
JP Abgrallcee20682014-07-02 14:26:54 -0700637 ERROR("%s(): %s is wiped and %s %s is encryptable. Suggest recovery...\n", __func__,
638 fstab->recs[attempted_idx].blk_device, fstab->recs[attempted_idx].mount_point,
639 fstab->recs[attempted_idx].fs_type);
640 encryptable = FS_MGR_MNTALL_DEV_NEEDS_RECOVERY;
Mohamad Ayyash38afe5f2014-03-10 15:40:29 -0700641 continue;
JP Abgrall4bb7bba2014-06-19 22:12:20 -0700642 } else {
643 /* Need to mount a tmpfs at this mountpoint for now, and set
644 * properties that vold will query later for decrypting
645 */
JP Abgrallf22b7452014-07-02 13:16:04 -0700646 ERROR("%s(): possibly an encryptable blkdev %s for mount %s type %s )\n", __func__,
647 fstab->recs[attempted_idx].blk_device, fstab->recs[attempted_idx].mount_point,
648 fstab->recs[attempted_idx].fs_type);
649 if (fs_mgr_do_tmpfs_mount(fstab->recs[attempted_idx].mount_point) < 0) {
JP Abgrall4bb7bba2014-06-19 22:12:20 -0700650 ++error_count;
651 continue;
652 }
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800653 }
JP Abgrallf22b7452014-07-02 13:16:04 -0700654 encryptable = FS_MGR_MNTALL_DEV_MIGHT_BE_ENCRYPTED;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800655 } else {
William Roberts071f28a2014-01-14 14:33:44 -0500656 ERROR("Failed to mount an un-encryptable or wiped partition on"
Mohamad Ayyash38afe5f2014-03-10 15:40:29 -0700657 "%s at %s options: %s error: %s\n",
JP Abgrallf22b7452014-07-02 13:16:04 -0700658 fstab->recs[attempted_idx].blk_device, fstab->recs[attempted_idx].mount_point,
659 fstab->recs[attempted_idx].fs_options, strerror(mount_errno));
Mohamad Ayyash38afe5f2014-03-10 15:40:29 -0700660 ++error_count;
661 continue;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800662 }
663 }
664
Mohamad Ayyash38afe5f2014-03-10 15:40:29 -0700665 if (error_count) {
666 return -1;
Mohamad Ayyash38afe5f2014-03-10 15:40:29 -0700667 } else {
Paul Lawrence166fa3d2014-02-03 13:27:49 -0800668 return encryptable;
Mohamad Ayyash38afe5f2014-03-10 15:40:29 -0700669 }
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800670}
671
Ken Sumrallab6b8522013-02-13 12:58:40 -0800672/* If tmp_mount_point is non-null, mount the filesystem there. This is for the
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800673 * tmp mount we do to check the user password
JP Abgrall5c01dac2014-06-18 14:54:37 -0700674 * If multiple fstab entries are to be mounted on "n_name", it will try to mount each one
675 * in turn, and stop on 1st success, or no more match.
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800676 */
Ken Sumrallab6b8522013-02-13 12:58:40 -0800677int fs_mgr_do_mount(struct fstab *fstab, char *n_name, char *n_blk_device,
678 char *tmp_mount_point)
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800679{
680 int i = 0;
Paul Lawrencecf234dc2014-09-09 10:44:51 -0700681 int ret = FS_MGR_DOMNT_FAILED;
JP Abgrall5c01dac2014-06-18 14:54:37 -0700682 int mount_errors = 0;
683 int first_mount_errno = 0;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800684 char *m;
685
Ken Sumrallab6b8522013-02-13 12:58:40 -0800686 if (!fstab) {
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800687 return ret;
688 }
689
Ken Sumrallab6b8522013-02-13 12:58:40 -0800690 for (i = 0; i < fstab->num_entries; i++) {
691 if (!fs_match(fstab->recs[i].mount_point, n_name)) {
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800692 continue;
693 }
694
695 /* We found our match */
Ken Sumrall5bc31a22013-07-08 19:11:55 -0700696 /* If this swap or a raw partition, report an error */
697 if (!strcmp(fstab->recs[i].fs_type, "swap") ||
698 !strcmp(fstab->recs[i].fs_type, "emmc") ||
Ken Sumrallab6b8522013-02-13 12:58:40 -0800699 !strcmp(fstab->recs[i].fs_type, "mtd")) {
700 ERROR("Cannot mount filesystem of type %s on %s\n",
701 fstab->recs[i].fs_type, n_blk_device);
702 goto out;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800703 }
704
Ken Sumrallab6b8522013-02-13 12:58:40 -0800705 /* First check the filesystem if requested */
706 if (fstab->recs[i].fs_mgr_flags & MF_WAIT) {
707 wait_for_file(n_blk_device, WAIT_TIMEOUT);
708 }
709
710 if (fstab->recs[i].fs_mgr_flags & MF_CHECK) {
711 check_fs(n_blk_device, fstab->recs[i].fs_type,
712 fstab->recs[i].mount_point);
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800713 }
714
Paul Lawrencebbb36312014-10-09 14:22:49 +0000715 if ((fstab->recs[i].fs_mgr_flags & MF_VERIFY) && device_is_secure()) {
716 int rc = fs_mgr_setup_verity(&fstab->recs[i]);
717 if (device_is_debuggable() && rc == FS_MGR_SETUP_VERITY_DISABLED) {
718 INFO("Verity disabled");
719 } else if (rc != FS_MGR_SETUP_VERITY_SUCCESS) {
JP Abgrall4bb7bba2014-06-19 22:12:20 -0700720 ERROR("Could not set up verified partition, skipping!\n");
Geremy Condra3ad3d1c2013-02-22 18:11:41 -0800721 continue;
722 }
723 }
724
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800725 /* Now mount it where requested */
Ken Sumrallab6b8522013-02-13 12:58:40 -0800726 if (tmp_mount_point) {
727 m = tmp_mount_point;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800728 } else {
Ken Sumrallab6b8522013-02-13 12:58:40 -0800729 m = fstab->recs[i].mount_point;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800730 }
JP Abgrall5c01dac2014-06-18 14:54:37 -0700731 if (__mount(n_blk_device, m, &fstab->recs[i])) {
732 if (!first_mount_errno) first_mount_errno = errno;
733 mount_errors++;
734 continue;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800735 } else {
736 ret = 0;
737 goto out;
738 }
739 }
JP Abgrall5c01dac2014-06-18 14:54:37 -0700740 if (mount_errors) {
741 ERROR("Cannot mount filesystem on %s at %s. error: %s\n",
742 n_blk_device, m, strerror(first_mount_errno));
Paul Lawrencecf234dc2014-09-09 10:44:51 -0700743 if (first_mount_errno == EBUSY) {
744 ret = FS_MGR_DOMNT_BUSY;
745 } else {
746 ret = FS_MGR_DOMNT_FAILED;
747 }
JP Abgrall5c01dac2014-06-18 14:54:37 -0700748 } else {
749 /* We didn't find a match, say so and return an error */
750 ERROR("Cannot find mount point %s in fstab\n", fstab->recs[i].mount_point);
751 }
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800752
753out:
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800754 return ret;
755}
756
757/*
758 * mount a tmpfs filesystem at the given point.
759 * return 0 on success, non-zero on failure.
760 */
761int fs_mgr_do_tmpfs_mount(char *n_name)
762{
763 int ret;
764
765 ret = mount("tmpfs", n_name, "tmpfs",
766 MS_NOATIME | MS_NOSUID | MS_NODEV, CRYPTO_TMPFS_OPTIONS);
767 if (ret < 0) {
768 ERROR("Cannot mount tmpfs filesystem at %s\n", n_name);
769 return -1;
770 }
771
772 /* Success */
773 return 0;
774}
775
Ken Sumrallab6b8522013-02-13 12:58:40 -0800776int fs_mgr_unmount_all(struct fstab *fstab)
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800777{
778 int i = 0;
779 int ret = 0;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800780
Ken Sumrallab6b8522013-02-13 12:58:40 -0800781 if (!fstab) {
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800782 return -1;
783 }
784
Ken Sumrallab6b8522013-02-13 12:58:40 -0800785 while (fstab->recs[i].blk_device) {
786 if (umount(fstab->recs[i].mount_point)) {
787 ERROR("Cannot unmount filesystem at %s\n", fstab->recs[i].mount_point);
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800788 ret = -1;
789 }
790 i++;
791 }
792
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800793 return ret;
794}
Ken Sumrall5bc31a22013-07-08 19:11:55 -0700795
796/* This must be called after mount_all, because the mkswap command needs to be
797 * available.
798 */
799int fs_mgr_swapon_all(struct fstab *fstab)
800{
801 int i = 0;
802 int flags = 0;
803 int err = 0;
804 int ret = 0;
805 int status;
806 char *mkswap_argv[2] = {
807 MKSWAP_BIN,
808 NULL
809 };
810
811 if (!fstab) {
812 return -1;
813 }
814
815 for (i = 0; i < fstab->num_entries; i++) {
816 /* Skip non-swap entries */
817 if (strcmp(fstab->recs[i].fs_type, "swap")) {
818 continue;
819 }
820
821 if (fstab->recs[i].zram_size > 0) {
822 /* A zram_size was specified, so we need to configure the
823 * device. There is no point in having multiple zram devices
824 * on a system (all the memory comes from the same pool) so
825 * we can assume the device number is 0.
826 */
827 FILE *zram_fp;
828
829 zram_fp = fopen(ZRAM_CONF_DEV, "r+");
830 if (zram_fp == NULL) {
JP Abgrall4bb7bba2014-06-19 22:12:20 -0700831 ERROR("Unable to open zram conf device %s\n", ZRAM_CONF_DEV);
Ken Sumrall5bc31a22013-07-08 19:11:55 -0700832 ret = -1;
833 continue;
834 }
835 fprintf(zram_fp, "%d\n", fstab->recs[i].zram_size);
836 fclose(zram_fp);
837 }
838
839 if (fstab->recs[i].fs_mgr_flags & MF_WAIT) {
840 wait_for_file(fstab->recs[i].blk_device, WAIT_TIMEOUT);
841 }
842
843 /* Initialize the swap area */
844 mkswap_argv[1] = fstab->recs[i].blk_device;
845 err = android_fork_execvp_ext(ARRAY_SIZE(mkswap_argv), mkswap_argv,
Yusuke Satod81c3c62015-08-14 01:22:53 -0700846 &status, true, LOG_KLOG, false, NULL,
847 NULL, 0);
Ken Sumrall5bc31a22013-07-08 19:11:55 -0700848 if (err) {
849 ERROR("mkswap failed for %s\n", fstab->recs[i].blk_device);
850 ret = -1;
851 continue;
852 }
853
854 /* If -1, then no priority was specified in fstab, so don't set
855 * SWAP_FLAG_PREFER or encode the priority */
856 if (fstab->recs[i].swap_prio >= 0) {
857 flags = (fstab->recs[i].swap_prio << SWAP_FLAG_PRIO_SHIFT) &
858 SWAP_FLAG_PRIO_MASK;
859 flags |= SWAP_FLAG_PREFER;
860 } else {
861 flags = 0;
862 }
863 err = swapon(fstab->recs[i].blk_device, flags);
864 if (err) {
865 ERROR("swapon failed for %s\n", fstab->recs[i].blk_device);
866 ret = -1;
867 }
868 }
869
870 return ret;
871}
872
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800873/*
874 * key_loc must be at least PROPERTY_VALUE_MAX bytes long
875 *
Ken Sumrallab6b8522013-02-13 12:58:40 -0800876 * real_blk_device must be at least PROPERTY_VALUE_MAX bytes long
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800877 */
Ken Sumrallab6b8522013-02-13 12:58:40 -0800878int fs_mgr_get_crypt_info(struct fstab *fstab, char *key_loc, char *real_blk_device, int size)
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800879{
880 int i = 0;
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800881
Ken Sumrallab6b8522013-02-13 12:58:40 -0800882 if (!fstab) {
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800883 return -1;
884 }
885 /* Initialize return values to null strings */
886 if (key_loc) {
887 *key_loc = '\0';
888 }
Ken Sumrallab6b8522013-02-13 12:58:40 -0800889 if (real_blk_device) {
890 *real_blk_device = '\0';
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800891 }
892
893 /* Look for the encryptable partition to find the data */
Ken Sumrallab6b8522013-02-13 12:58:40 -0800894 for (i = 0; i < fstab->num_entries; i++) {
895 /* Don't deal with vold managed enryptable partitions here */
896 if (fstab->recs[i].fs_mgr_flags & MF_VOLDMANAGED) {
897 continue;
898 }
Paul Lawrenceb262d682015-10-29 10:31:02 -0700899 if (!(fstab->recs[i].fs_mgr_flags
900 & (MF_CRYPT | MF_FORCECRYPT | MF_FORCEFDEORFBE))) {
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800901 continue;
902 }
903
904 /* We found a match */
905 if (key_loc) {
Ken Sumrallab6b8522013-02-13 12:58:40 -0800906 strlcpy(key_loc, fstab->recs[i].key_loc, size);
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800907 }
Ken Sumrallab6b8522013-02-13 12:58:40 -0800908 if (real_blk_device) {
909 strlcpy(real_blk_device, fstab->recs[i].blk_device, size);
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800910 }
911 break;
912 }
913
Ken Sumrallc1bf8962012-01-06 19:09:42 -0800914 return 0;
915}