264 lines
6.3 KiB
C
264 lines
6.3 KiB
C
|
/* Copyright 2015 The Chromium OS Authors. All rights reserved.
|
||
|
* Use of this source code is governed by a BSD-style license that can be
|
||
|
* found in the LICENSE file.
|
||
|
*/
|
||
|
|
||
|
#include <err.h>
|
||
|
#include <errno.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <ftw.h>
|
||
|
#include <inttypes.h>
|
||
|
#include <linux/major.h>
|
||
|
#include <stdbool.h>
|
||
|
#include <stdarg.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <stdint.h>
|
||
|
#include <stdio.h>
|
||
|
#include <string.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/wait.h>
|
||
|
#include <unistd.h>
|
||
|
|
||
|
#include "cgpt.h"
|
||
|
#include "cgpt_nor.h"
|
||
|
|
||
|
static const char FLASHROM_PATH[] = "/usr/sbin/flashrom";
|
||
|
|
||
|
// Obtain the MTD size from its sysfs node.
|
||
|
int GetMtdSize(const char *mtd_device, uint64_t *size) {
|
||
|
mtd_device = strrchr(mtd_device, '/');
|
||
|
if (mtd_device == NULL) {
|
||
|
errno = EINVAL;
|
||
|
return 1;
|
||
|
}
|
||
|
char *sysfs_name;
|
||
|
if (asprintf(&sysfs_name, "/sys/class/mtd%s/size", mtd_device) == -1) {
|
||
|
return 1;
|
||
|
}
|
||
|
FILE *fp = fopen(sysfs_name, "r");
|
||
|
free(sysfs_name);
|
||
|
if (fp == NULL) {
|
||
|
return 1;
|
||
|
}
|
||
|
int ret = (fscanf(fp, "%" PRIu64 "\n", size) != 1);
|
||
|
fclose(fp);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int ForkExecV(const char *cwd, const char *const argv[]) {
|
||
|
pid_t pid = fork();
|
||
|
if (pid == -1) {
|
||
|
return -1;
|
||
|
}
|
||
|
int status = -1;
|
||
|
if (pid == 0) {
|
||
|
if (cwd && chdir(cwd) != 0) {
|
||
|
return -1;
|
||
|
}
|
||
|
execv(argv[0], (char *const *)argv);
|
||
|
// If this is reached, execv fails.
|
||
|
err(-1, "Cannot exec %s in %s.", argv[0], cwd);
|
||
|
} else {
|
||
|
if (waitpid(pid, &status, 0) != -1 && WIFEXITED(status))
|
||
|
return WEXITSTATUS(status);
|
||
|
}
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
int ForkExecL(const char *cwd, const char *cmd, ...) {
|
||
|
int argc;
|
||
|
va_list ap;
|
||
|
va_start(ap, cmd);
|
||
|
for (argc = 1; va_arg(ap, char *) != NULL; ++argc);
|
||
|
va_end(ap);
|
||
|
|
||
|
va_start(ap, cmd);
|
||
|
const char **argv = calloc(argc + 1, sizeof(char *));
|
||
|
if (argv == NULL) {
|
||
|
errno = ENOMEM;
|
||
|
va_end(ap);
|
||
|
return -1;
|
||
|
}
|
||
|
argv[0] = cmd;
|
||
|
int i;
|
||
|
for (i = 1; i < argc; ++i) {
|
||
|
argv[i] = va_arg(ap, char *);
|
||
|
}
|
||
|
va_end(ap);
|
||
|
|
||
|
int ret = ForkExecV(cwd, argv);
|
||
|
free(argv);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static int read_write(int source_fd,
|
||
|
uint64_t size,
|
||
|
const char *src_name,
|
||
|
int idx) {
|
||
|
int ret = 1;
|
||
|
const int bufsize = 4096;
|
||
|
char *buf = malloc(bufsize);
|
||
|
if (buf == NULL) {
|
||
|
goto clean_exit;
|
||
|
}
|
||
|
|
||
|
ret++;
|
||
|
char *dest;
|
||
|
if (asprintf(&dest, "%s_%d", src_name, idx) == -1) {
|
||
|
goto free_buf;
|
||
|
}
|
||
|
|
||
|
ret++;
|
||
|
int dest_fd = open(dest, O_WRONLY | O_CLOEXEC | O_CREAT, 0600);
|
||
|
if (dest_fd < 0) {
|
||
|
goto free_dest;
|
||
|
}
|
||
|
|
||
|
ret++;
|
||
|
uint64_t copied = 0;
|
||
|
ssize_t nr_read;
|
||
|
ssize_t nr_write;
|
||
|
while (copied < size) {
|
||
|
size_t to_read = size - copied;
|
||
|
if (to_read > bufsize) {
|
||
|
to_read = bufsize;
|
||
|
}
|
||
|
nr_read = read(source_fd, buf, to_read);
|
||
|
if (nr_read < 0) {
|
||
|
goto close_dest_fd;
|
||
|
}
|
||
|
nr_write = 0;
|
||
|
while (nr_write < nr_read) {
|
||
|
ssize_t s = write(dest_fd, buf + nr_write, nr_read - nr_write);
|
||
|
if (s < 0) {
|
||
|
goto close_dest_fd;
|
||
|
}
|
||
|
nr_write += s;
|
||
|
}
|
||
|
copied += nr_read;
|
||
|
}
|
||
|
|
||
|
ret = 0;
|
||
|
|
||
|
close_dest_fd:
|
||
|
close(dest_fd);
|
||
|
free_dest:
|
||
|
free(dest);
|
||
|
free_buf:
|
||
|
free(buf);
|
||
|
clean_exit:
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static int split_gpt(const char *dir_name, const char *file_name) {
|
||
|
int ret = 1;
|
||
|
char *source;
|
||
|
if (asprintf(&source, "%s/%s", dir_name, file_name) == -1) {
|
||
|
goto clean_exit;
|
||
|
}
|
||
|
|
||
|
ret++;
|
||
|
int fd = open(source, O_RDONLY | O_CLOEXEC);
|
||
|
if (fd < 0) {
|
||
|
goto free_source;
|
||
|
}
|
||
|
|
||
|
ret++;
|
||
|
struct stat stat;
|
||
|
if (fstat(fd, &stat) != 0 || (stat.st_size & 1) != 0) {
|
||
|
goto close_fd;
|
||
|
}
|
||
|
uint64_t half_size = stat.st_size / 2;
|
||
|
|
||
|
ret++;
|
||
|
if (read_write(fd, half_size, source, 1) != 0 ||
|
||
|
read_write(fd, half_size, source, 2) != 0) {
|
||
|
goto close_fd;
|
||
|
}
|
||
|
|
||
|
ret = 0;
|
||
|
close_fd:
|
||
|
close(fd);
|
||
|
free_source:
|
||
|
free(source);
|
||
|
clean_exit:
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static int remove_file_or_dir(const char *fpath, const struct stat *sb,
|
||
|
int typeflag, struct FTW *ftwbuf) {
|
||
|
return remove(fpath);
|
||
|
}
|
||
|
|
||
|
int RemoveDir(const char *dir) {
|
||
|
return nftw(dir, remove_file_or_dir, 20, FTW_DEPTH | FTW_PHYS);
|
||
|
}
|
||
|
|
||
|
// Read RW_GPT from NOR flash to "rw_gpt" in a temp dir |temp_dir_template|.
|
||
|
// |temp_dir_template| is passed to mkdtemp() so it must satisfy all
|
||
|
// requirements by mkdtemp.
|
||
|
int ReadNorFlash(char *temp_dir_template) {
|
||
|
int ret = 0;
|
||
|
|
||
|
// Create a temp dir to work in.
|
||
|
ret++;
|
||
|
if (mkdtemp(temp_dir_template) == NULL) {
|
||
|
Error("Cannot create a temporary directory.\n");
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
// Read RW_GPT section from NOR flash to "rw_gpt".
|
||
|
ret++;
|
||
|
int fd_flags = fcntl(1, F_GETFD);
|
||
|
// Close stdout on exec so that flashrom does not muck up cgpt's output.
|
||
|
if (0 != fcntl(1, F_SETFD, FD_CLOEXEC))
|
||
|
Warning("Can't stop flashrom from mucking up our output\n");
|
||
|
if (ForkExecL(temp_dir_template, FLASHROM_PATH, "-i", "RW_GPT:rw_gpt", "-r",
|
||
|
NULL) != 0) {
|
||
|
Error("Cannot exec flashrom to read from RW_GPT section.\n");
|
||
|
RemoveDir(temp_dir_template);
|
||
|
} else {
|
||
|
ret = 0;
|
||
|
}
|
||
|
|
||
|
// Restore stdout flags
|
||
|
if (0 != fcntl(1, F_SETFD, fd_flags))
|
||
|
Warning("Can't restore stdout flags\n");
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
// Write "rw_gpt" back to NOR flash. We write the file in two parts for safety.
|
||
|
int WriteNorFlash(const char *dir) {
|
||
|
int ret = 0;
|
||
|
ret++;
|
||
|
if (split_gpt(dir, "rw_gpt") != 0) {
|
||
|
Error("Cannot split rw_gpt in two.\n");
|
||
|
return ret;
|
||
|
}
|
||
|
ret++;
|
||
|
int nr_fails = 0;
|
||
|
int fd_flags = fcntl(1, F_GETFD);
|
||
|
// Close stdout on exec so that flashrom does not muck up cgpt's output.
|
||
|
if (0 != fcntl(1, F_SETFD, FD_CLOEXEC))
|
||
|
Warning("Can't stop flashrom from mucking up our output\n");
|
||
|
if (ForkExecL(dir, FLASHROM_PATH, "-i", "RW_GPT_PRIMARY:rw_gpt_1",
|
||
|
"-w", "--fast-verify", NULL) != 0) {
|
||
|
Warning("Cannot write the 1st half of rw_gpt back with flashrom.\n");
|
||
|
nr_fails++;
|
||
|
}
|
||
|
if (ForkExecL(dir, FLASHROM_PATH, "-i", "RW_GPT_SECONDARY:rw_gpt_2",
|
||
|
"-w", "--fast-verify", NULL) != 0) {
|
||
|
Warning("Cannot write the 2nd half of rw_gpt back with flashrom.\n");
|
||
|
nr_fails++;
|
||
|
}
|
||
|
if (0 != fcntl(1, F_SETFD, fd_flags))
|
||
|
Warning("Can't restore stdout flags\n");
|
||
|
switch (nr_fails) {
|
||
|
case 0: ret = 0; break;
|
||
|
case 1: Warning("It might still be okay.\n"); break;
|
||
|
case 2: Error("Cannot write both parts back with flashrom.\n"); break;
|
||
|
}
|
||
|
return ret;
|
||
|
}
|