/* * Copyright (C) 2011 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <stdio.h> #include <errno.h> #include <stdarg.h> #include <stdlib.h> #include <string.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <unistd.h> #include <cryptfs.h> #include "edify/expr.h" #include "bootloader.h" Value* WriteBootloaderFn(const char* name, State* state, int argc, Expr* argv[]) { int result = -1; Value* img; Value* xloader_loc; Value* sbl_loc; if (argc != 3) { return ErrorAbort(state, "%s() expects 3 args, got %d", name, argc); } if (ReadValueArgs(state, argv, 3, &img, &xloader_loc, &sbl_loc) < 0) { return NULL; } if(img->type != VAL_BLOB || xloader_loc->type != VAL_STRING || sbl_loc->type != VAL_STRING) { FreeValue(img); FreeValue(xloader_loc); FreeValue(sbl_loc); return ErrorAbort(state, "%s(): argument types are incorrect", name); } result = update_bootloader(img->data, img->size, xloader_loc->data, sbl_loc->data); FreeValue(img); FreeValue(xloader_loc); FreeValue(sbl_loc); return StringValue(strdup(result == 0 ? "t" : "")); } /* * The size of the userdata partition for HSPA Galaxy Nexus devices is incorrect * in the partition as it comes from the factory. Updating the bootloader fixes * the partition table, and makes the size of the userdata partition 1 sector * smaller. However, if the user had encrypted their device with the original * incorrect size of the partition table, the crypto footer has saved that * size, and tries to map that much data when decrypting. However, with the * new partition table, that size is too big to be mapped, and the kernel * throws an error, and the user can't decrypt and boot the device after the * OTA is installed. Oops! * * The fix here is to recognize a crypto footer that has the wrong size, and * update it to the new correct size. This program should be run as part of * the recovery script for HSPA Galaxy Nexus devices. */ #define BAD_SIZE 0x01b14fdfULL #define GOOD_SIZE 0x01b14fdeULL #define HSPA_PRIME_KEY_PARTITION "/dev/block/platform/omap/omap_hsmmc.0/by-name/metadata" Value* FsSizeFixFn(const char* name, State* state, int argc, Expr* argv[]) { struct crypt_mnt_ftr ftr; int fd; if (argc != 0) { return ErrorAbort(state, "%s() expects 0 args, got %d", name, argc); } if ((fd = open(HSPA_PRIME_KEY_PARTITION, O_RDWR)) == -1) { return ErrorAbort(state, "%s() Cannot open %s\n", name, HSPA_PRIME_KEY_PARTITION); } if (read(fd, &ftr, sizeof(ftr)) != sizeof(ftr)) { close(fd); return ErrorAbort(state, "%s() Cannot read crypto footer %s\n", name, HSPA_PRIME_KEY_PARTITION); } if ((ftr.magic == CRYPT_MNT_MAGIC) && (ftr.fs_size == BAD_SIZE)) { ftr.fs_size = GOOD_SIZE; if (lseek(fd, 0, SEEK_SET) == 0) { if (write(fd, &ftr, sizeof(ftr)) == sizeof(ftr)) { fsync(fd); /* Make sure it gets to the disk */ fprintf(stderr, "Footer updated\n"); close(fd); return StringValue(strdup("t")); } } close(fd); return ErrorAbort(state, "%s() Cannot seek or write crypto footer %s\n", name, HSPA_PRIME_KEY_PARTITION); } /* Nothing to do */ fprintf(stderr, "Footer doesn't need updating\n"); close(fd); return StringValue(strdup("t")); } void Register_librecovery_updater_tuna() { fprintf(stderr, "installing samsung updater extensions\n"); RegisterFunction("samsung.write_bootloader", WriteBootloaderFn); RegisterFunction("samsung.fs_size_fix", FsSizeFixFn); }