/* * This file is part of the flashrom project. * * Copyright 2021 Google LLC * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; version 2 of the License. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. */ #include #include #include "io_mock.h" #include "programmer.h" static void run_lifecycle(void **state, const struct programmer_entry *prog, const char *param) { (void) state; /* unused */ char *param_dup = strdup(param); printf("Testing programmer_init for programmer=%s ...\n", prog->name); assert_int_equal(0, programmer_init(prog, param_dup)); printf("... programmer_init for programmer=%s successful\n", prog->name); printf("Testing programmer_shutdown for programmer=%s ...\n", prog->name); assert_int_equal(0, programmer_shutdown()); printf("... programmer_shutdown for programmer=%s successful\n", prog->name); free(param_dup); } void dummy_init_and_shutdown_test_success(void **state) { #if CONFIG_DUMMY == 1 run_lifecycle(state, &programmer_dummy, "bus=parallel+lpc+fwh+spi"); #else skip(); #endif } void nicrealtek_init_and_shutdown_test_success(void **state) { #if CONFIG_NICREALTEK == 1 run_lifecycle(state, &programmer_nicrealtek, ""); #else skip(); #endif } int dediprog_libusb_init(void *state, libusb_context **ctx) { *ctx = not_null(); return 0; } int dediprog_libusb_control_transfer(void *state, libusb_device_handle *devh, uint8_t bmRequestType, uint8_t bRequest, uint16_t wValue, uint16_t wIndex, unsigned char *data, uint16_t wLength, unsigned int timeout) { if (bRequest == 0x08 /* dediprog_cmds CMD_READ_PROG_INFO */) { /* Provide dediprog Device String into data buffer */ memcpy(data, "SF600 V:7.2.2 ", wLength); } return wLength; } void dediprog_init_and_shutdown_test_success(void **state) { #if CONFIG_DEDIPROG == 1 const struct io_mock dediprog_io = { .libusb_init = dediprog_libusb_init, .libusb_control_transfer = dediprog_libusb_control_transfer, }; io_mock_register(&dediprog_io); run_lifecycle(state, &programmer_dediprog, "voltage=3.5V"); io_mock_register(NULL); #else skip(); #endif } struct linux_mtd_io_state { char *fopen_path; }; FILE *linux_mtd_fopen(void *state, const char *pathname, const char *mode) { struct linux_mtd_io_state *io_state = state; io_state->fopen_path = strdup(pathname); return not_null(); } size_t linux_mtd_fread(void *state, void *buf, size_t size, size_t len, FILE *fp) { struct linux_mtd_fread_mock_entry { const char *path; const char *data; }; const struct linux_mtd_fread_mock_entry fread_mock_map[] = { { "/sys/class/mtd/mtd0//type", "nor" }, { "/sys/class/mtd/mtd0//name", "Device" }, { "/sys/class/mtd/mtd0//flags", "" }, { "/sys/class/mtd/mtd0//size", "1024" }, { "/sys/class/mtd/mtd0//erasesize", "512" }, { "/sys/class/mtd/mtd0//numeraseregions", "0" }, }; struct linux_mtd_io_state *io_state = state; unsigned int i; if (!io_state->fopen_path) return 0; for (i = 0; i < ARRAY_SIZE(fread_mock_map); i++) { const struct linux_mtd_fread_mock_entry *entry = &fread_mock_map[i]; if (!strcmp(io_state->fopen_path, entry->path)) { size_t data_len = min(size * len, strlen(entry->data)); memcpy(buf, entry->data, data_len); return data_len; } } return 0; } int linux_mtd_fclose(void *state, FILE *fp) { struct linux_mtd_io_state *io_state = state; free(io_state->fopen_path); return 0; } void linux_mtd_init_and_shutdown_test_success(void **state) { #if CONFIG_LINUX_MTD == 1 struct linux_mtd_io_state linux_mtd_io_state = { NULL }; const struct io_mock linux_mtd_io = { .state = &linux_mtd_io_state, .fopen = linux_mtd_fopen, .fread = linux_mtd_fread, .fclose = linux_mtd_fclose, }; io_mock_register(&linux_mtd_io); run_lifecycle(state, &programmer_linux_mtd, ""); io_mock_register(NULL); #else skip(); #endif } char *linux_spi_fgets(void *state, char *buf, int len, FILE *fp) { /* Emulate reading max buffer size from sysfs. */ const char *max_buf_size = "1048576"; return memcpy(buf, max_buf_size, min(len, strlen(max_buf_size) + 1)); } void linux_spi_init_and_shutdown_test_success(void **state) { /* * Current implementation tests a particular path of the init procedure. * Specifically, it is reading the buffer size from sysfs. */ #if CONFIG_LINUX_SPI == 1 const struct io_mock linux_spi_io = { .fgets = linux_spi_fgets, }; io_mock_register(&linux_spi_io); run_lifecycle(state, &programmer_linux_spi, "dev=/dev/null"); io_mock_register(NULL); #else skip(); #endif } #define REALTEK_MST_MOCK_FD 0x10ec static int realtek_mst_open(void *state, const char *pathname, int flags) { assert_string_equal(pathname, "/dev/i2c-254"); assert_int_equal(flags & O_RDWR, O_RDWR); return REALTEK_MST_MOCK_FD; } static int realtek_mst_ioctl(void *state, int fd, unsigned long request, va_list args) { assert_int_equal(fd, REALTEK_MST_MOCK_FD); assert_int_equal(request, I2C_SLAVE); /* Only access to I2C address 0x4a is expected */ unsigned long addr = va_arg(args, unsigned long); assert_int_equal(addr, 0x4a); return 0; } static int realtek_mst_read(void *state, int fd, void *buf, size_t sz) { assert_int_equal(fd, REALTEK_MST_MOCK_FD); assert_int_equal(sz, 1); return sz; } static int realtek_mst_write(void *state, int fd, const void *buf, size_t sz) { assert_int_equal(fd, REALTEK_MST_MOCK_FD); const LargestIntegralType accepted_sizes[] = {1, 2}; assert_in_set(sz, accepted_sizes, ARRAY_SIZE(accepted_sizes)); return sz; } void realtek_mst_init_and_shutdown_test_success(void **state) { #if CONFIG_REALTEK_MST_I2C_SPI == 1 const struct io_mock realtek_mst_io = { .open = realtek_mst_open, .ioctl = realtek_mst_ioctl, .read = realtek_mst_read, .write = realtek_mst_write, }; io_mock_register(&realtek_mst_io); run_lifecycle(state, &programmer_realtek_mst_i2c_spi, "bus=254,enter-isp=0"); io_mock_register(NULL); #else skip(); #endif /* CONFIG_REALTEK_I2C_SPI */ }