/*
* 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);
}