blob: 815657d453b8f7eff31a603eac3a603aeb8c3d8b [file] [log] [blame]
/* Copyright 2020 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.
*
* Tests for host flashrom utilities.
*/
/* For strdup */
#define _POSIX_C_SOURCE 200809L
#include <fcntl.h>
#include <getopt.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "2common.h"
#include "2return_codes.h"
#include "host_misc.h"
#include "flashrom.h"
#include "subprocess.h"
#include "test_common.h"
#define MOCK_TMPFILE_NAME "/tmp/vb2_unittest"
#define MOCK_ROM_CONTENTS "bloop123"
static bool flashrom_mock_success = true;
static enum { FLASHROM_NONE, FLASHROM_READ, FLASHROM_WRITE } captured_operation;
static enum {
FLASHROM_VERIFY_UNSPECIFIED,
FLASHROM_VERIFY_FAST,
} captured_verify;
static const char *captured_op_filename;
static const char *captured_region_param;
static const char *captured_programmer;
static uint8_t *captured_rom_contents;
static uint32_t captured_rom_size;
/* Mocked mkstemp for tests. */
int mkstemp(char *template_name)
{
strncpy(template_name, MOCK_TMPFILE_NAME, strlen(template_name));
return open(template_name, O_RDWR | O_CREAT | O_TRUNC, 0666);
}
/* Mocked subprocess_run for tests. */
int subprocess_run(const char *const argv[],
struct subprocess_target *input,
struct subprocess_target *output,
struct subprocess_target *error)
{
int argc;
int opt;
int rv;
/* getopt_long wants an int instead of an enum, bummer... */
int captured_verify_int = FLASHROM_VERIFY_UNSPECIFIED;
struct option long_opts[] = {
{
.name = "noverify-all",
.has_arg = no_argument,
.flag = &captured_verify_int,
.val = FLASHROM_VERIFY_FAST,
},
};
/* Reset static variables to their defaults. */
captured_operation = FLASHROM_NONE;
captured_operation = FLASHROM_VERIFY_UNSPECIFIED;
captured_op_filename = NULL;
captured_region_param = NULL;
captured_programmer = NULL;
captured_rom_contents = NULL;
captured_rom_size = 0;
optind = 0;
/* Count the number of arguments, a required formalism for
getopt. */
for (argc = 0; argv[argc]; argc++)
continue;
/* We only understand the subset of arguments used by the
wrapper library. If it's updated to support more modes of
operation, this unit test code should be updated too. */
while ((opt = getopt_long(argc, (char *const *)argv,
":p:r:w:i:", long_opts, NULL)) != -1) {
/* Always consume the next argument if it does not
start with a dash. We have to muck with getopt's
global variables to make this happen. */
if (opt == ':' && argv[optind] && argv[optind][0] != '-') {
optarg = strdup(argv[optind]);
optind++;
opt = optopt;
} else if (optarg && optarg[0] == '-') {
optarg = NULL;
optind--;
} else if (optarg) {
optarg = strdup(optarg);
}
switch (opt) {
case 'p':
captured_programmer = optarg;
break;
case 'r':
captured_operation = FLASHROM_READ;
captured_op_filename = optarg;
break;
case 'w':
captured_operation = FLASHROM_WRITE;
captured_op_filename = optarg;
break;
case 'i':
captured_region_param = optarg;
break;
case 0:
/* long option */
break;
default:
return 1;
}
}
if (optind != argc) {
/* Extra arguments we don't understand. */
return 1;
}
rv = !flashrom_mock_success;
captured_verify = captured_verify_int;
if (captured_operation == FLASHROM_READ) {
/* Write the mocked string we read from the ROM. */
rv |= vb2_write_file(MOCK_TMPFILE_NAME, MOCK_ROM_CONTENTS,
strlen(MOCK_ROM_CONTENTS));
} else if (captured_operation == FLASHROM_WRITE) {
/* Capture the buffer contents we wrote to the ROM. */
rv |= vb2_read_file(MOCK_TMPFILE_NAME, &captured_rom_contents,
&captured_rom_size);
}
return rv;
}
static void test_read_whole_chip(void)
{
struct firmware_image image = {
.programmer = "someprog",
};
TEST_SUCC(flashrom_read(&image, NULL),
"Flashrom read succeeds");
TEST_STR_EQ(captured_programmer, "someprog",
"Using specified programmer");
TEST_EQ(captured_operation, FLASHROM_READ, "Doing a read operation");
TEST_EQ(captured_verify, FLASHROM_VERIFY_UNSPECIFIED,
"Verification not enabled");
TEST_STR_EQ(captured_op_filename, MOCK_TMPFILE_NAME,
"Reading to correct file");
TEST_PTR_EQ(captured_region_param, NULL, "Not operating on a region");
TEST_EQ(image.size, strlen(MOCK_ROM_CONTENTS), "Contents correct size");
TEST_SUCC(memcmp(image.data, MOCK_ROM_CONTENTS, image.size),
"Buffer has correct contents");
free(image.data);
}
static void test_read_region(void)
{
struct firmware_image image = {
.programmer = "someprog",
};
TEST_SUCC(flashrom_read(&image, "SOME_REGION"),
"Flashrom read succeeds");
TEST_STR_EQ(captured_programmer, "someprog",
"Using specified programmer");
TEST_EQ(captured_operation, FLASHROM_READ, "Doing a read operation");
TEST_EQ(captured_verify, FLASHROM_VERIFY_UNSPECIFIED,
"Verification not enabled");
TEST_PTR_EQ(captured_op_filename, NULL,
"Not doing a read of the whole ROM");
TEST_STR_EQ(captured_region_param, "SOME_REGION:" MOCK_TMPFILE_NAME,
"Reading to correct file and from correct region");
TEST_EQ(image.size, strlen(MOCK_ROM_CONTENTS), "Contents correct size");
TEST_SUCC(memcmp(image.data, MOCK_ROM_CONTENTS, image.size),
"Buffer has correct contents");
free(image.data);
}
static void test_read_failure(void)
{
struct firmware_image image = {
.programmer = "someprog",
};
flashrom_mock_success = false;
TEST_NEQ(flashrom_read(&image, "SOME_REGION"),
VB2_SUCCESS, "Flashrom read fails");
flashrom_mock_success = true;
}
static void test_write_whole_chip(void)
{
uint8_t buf[sizeof(MOCK_ROM_CONTENTS) - 1];
struct firmware_image image = {
.programmer = "someprog",
.data = buf,
.size = sizeof(buf),
};
memcpy(buf, MOCK_ROM_CONTENTS, sizeof(buf));
TEST_SUCC(flashrom_write(&image, NULL),
"Flashrom write succeeds");
TEST_STR_EQ(captured_programmer, "someprog",
"Using specified programmer");
TEST_EQ(captured_operation, FLASHROM_WRITE, "Doing a write operation");
TEST_EQ(captured_verify, FLASHROM_VERIFY_FAST,
"Fast verification enabled");
TEST_STR_EQ(captured_op_filename, MOCK_TMPFILE_NAME,
"Writing to correct file");
TEST_PTR_EQ(captured_region_param, NULL, "Not operating on a region");
TEST_EQ(captured_rom_size, strlen(MOCK_ROM_CONTENTS),
"Contents correct size");
TEST_SUCC(memcmp(captured_rom_contents, MOCK_ROM_CONTENTS,
captured_rom_size), "Buffer has correct contents");
}
static void test_write_region(void)
{
uint8_t buf[sizeof(MOCK_ROM_CONTENTS) - 1];
struct firmware_image image = {
.programmer = "someprog",
.data = buf,
.size = sizeof(buf),
};
memcpy(buf, MOCK_ROM_CONTENTS, sizeof(buf));
TEST_SUCC(flashrom_write(&image, "SOME_REGION"),
"Flashrom write succeeds");
TEST_STR_EQ(captured_programmer, "someprog",
"Using specified programmer");
TEST_EQ(captured_operation, FLASHROM_WRITE, "Doing a write operation");
TEST_EQ(captured_verify, FLASHROM_VERIFY_FAST,
"Fast verification enabled");
TEST_PTR_EQ(captured_op_filename, NULL,
"Not doing a write of the whole ROM");
TEST_STR_EQ(captured_region_param, "SOME_REGION:" MOCK_TMPFILE_NAME,
"Writing to correct file and from correct region");
TEST_EQ(captured_rom_size, strlen(MOCK_ROM_CONTENTS),
"Contents correct size");
TEST_SUCC(memcmp(captured_rom_contents, MOCK_ROM_CONTENTS,
captured_rom_size), "Buffer has correct contents");
}
static void test_write_failure(void)
{
uint8_t buf[20] = { 0 };
struct firmware_image image = {
.programmer = "someprog",
.data = buf,
.size = sizeof(buf),
};
flashrom_mock_success = false;
TEST_NEQ(flashrom_write(&image, "SOME_REGION"),
VB2_SUCCESS, "Flashrom write fails");
flashrom_mock_success = true;
}
int main(int argc, char *argv[])
{
test_read_whole_chip();
test_read_region();
test_read_failure();
test_write_whole_chip();
test_write_region();
test_write_failure();
return gTestSuccess ? 0 : 255;
}