xref: /aosp_15_r20/external/flashrom/tests/linux_spi.c (revision 0d6140be3aa665ecc836e8907834fcd3e3b018fc)
1*0d6140beSAndroid Build Coastguard Worker /*
2*0d6140beSAndroid Build Coastguard Worker  * This file is part of the flashrom project.
3*0d6140beSAndroid Build Coastguard Worker  *
4*0d6140beSAndroid Build Coastguard Worker  * Copyright 2021 Google LLC
5*0d6140beSAndroid Build Coastguard Worker  *
6*0d6140beSAndroid Build Coastguard Worker  * This program is free software; you can redistribute it and/or modify
7*0d6140beSAndroid Build Coastguard Worker  * it under the terms of the GNU General Public License as published by
8*0d6140beSAndroid Build Coastguard Worker  * the Free Software Foundation; version 2 of the License.
9*0d6140beSAndroid Build Coastguard Worker  *
10*0d6140beSAndroid Build Coastguard Worker  * This program is distributed in the hope that it will be useful,
11*0d6140beSAndroid Build Coastguard Worker  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12*0d6140beSAndroid Build Coastguard Worker  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13*0d6140beSAndroid Build Coastguard Worker  * GNU General Public License for more details.
14*0d6140beSAndroid Build Coastguard Worker  */
15*0d6140beSAndroid Build Coastguard Worker 
16*0d6140beSAndroid Build Coastguard Worker #include "lifecycle.h"
17*0d6140beSAndroid Build Coastguard Worker 
18*0d6140beSAndroid Build Coastguard Worker #if CONFIG_LINUX_SPI == 1
linux_spi_ioctl(void * state,int fd,unsigned long request,va_list args)19*0d6140beSAndroid Build Coastguard Worker static int linux_spi_ioctl(void *state, int fd, unsigned long request, va_list args) {
20*0d6140beSAndroid Build Coastguard Worker 
21*0d6140beSAndroid Build Coastguard Worker 	if (request == SPI_IOC_MESSAGE(2)) { /* ioctl code for read request */
22*0d6140beSAndroid Build Coastguard Worker 		struct spi_ioc_transfer* msg = va_arg(args, struct spi_ioc_transfer*);
23*0d6140beSAndroid Build Coastguard Worker 
24*0d6140beSAndroid Build Coastguard Worker 		/* First message has write array and write count */
25*0d6140beSAndroid Build Coastguard Worker 		unsigned int writecnt = msg[0].len;
26*0d6140beSAndroid Build Coastguard Worker 		unsigned char *writearr = (unsigned char *)(uintptr_t)msg[0].tx_buf;
27*0d6140beSAndroid Build Coastguard Worker 		/* Second message has read array and read count */
28*0d6140beSAndroid Build Coastguard Worker 		unsigned int readcnt = msg[1].len;
29*0d6140beSAndroid Build Coastguard Worker 
30*0d6140beSAndroid Build Coastguard Worker 		/* Detect probing */
31*0d6140beSAndroid Build Coastguard Worker 		if (writecnt == 1 && writearr[0] == JEDEC_RDID && readcnt == 3) {
32*0d6140beSAndroid Build Coastguard Worker 			/* We need to populate read array. */
33*0d6140beSAndroid Build Coastguard Worker 			unsigned char *readarr = (unsigned char *)(uintptr_t)msg[1].rx_buf;
34*0d6140beSAndroid Build Coastguard Worker 			readarr[0] = 0xEF; /* WINBOND_NEX_ID */
35*0d6140beSAndroid Build Coastguard Worker 			readarr[1] = 0x40; /* WINBOND_NEX_W25Q128_V left byte */
36*0d6140beSAndroid Build Coastguard Worker 			readarr[2] = 0x18; /* WINBOND_NEX_W25Q128_V right byte */
37*0d6140beSAndroid Build Coastguard Worker 		}
38*0d6140beSAndroid Build Coastguard Worker 	}
39*0d6140beSAndroid Build Coastguard Worker 
40*0d6140beSAndroid Build Coastguard Worker 	return 0;
41*0d6140beSAndroid Build Coastguard Worker }
42*0d6140beSAndroid Build Coastguard Worker 
linux_spi_fgets(void * state,char * buf,int len,FILE * fp)43*0d6140beSAndroid Build Coastguard Worker static char *linux_spi_fgets(void *state, char *buf, int len, FILE *fp)
44*0d6140beSAndroid Build Coastguard Worker {
45*0d6140beSAndroid Build Coastguard Worker 	/* Emulate reading max buffer size from sysfs. */
46*0d6140beSAndroid Build Coastguard Worker 	const char *max_buf_size = "1048576";
47*0d6140beSAndroid Build Coastguard Worker 
48*0d6140beSAndroid Build Coastguard Worker 	return memcpy(buf, max_buf_size, min(len, strlen(max_buf_size) + 1));
49*0d6140beSAndroid Build Coastguard Worker }
50*0d6140beSAndroid Build Coastguard Worker 
linux_spi_probe_lifecycle_test_success(void ** state)51*0d6140beSAndroid Build Coastguard Worker void linux_spi_probe_lifecycle_test_success(void **state)
52*0d6140beSAndroid Build Coastguard Worker {
53*0d6140beSAndroid Build Coastguard Worker 	/*
54*0d6140beSAndroid Build Coastguard Worker 	 * Current implementation tests a particular path of the init procedure.
55*0d6140beSAndroid Build Coastguard Worker 	 * Specifically, it is reading the buffer size from sysfs.
56*0d6140beSAndroid Build Coastguard Worker 	 */
57*0d6140beSAndroid Build Coastguard Worker 	struct io_mock_fallback_open_state linux_spi_fallback_open_state = {
58*0d6140beSAndroid Build Coastguard Worker 		.noc = 0,
59*0d6140beSAndroid Build Coastguard Worker 		.paths = { LOCK_FILE, "/dev/null", NULL },
60*0d6140beSAndroid Build Coastguard Worker 		.flags = { O_RDWR, O_RDWR },
61*0d6140beSAndroid Build Coastguard Worker 	};
62*0d6140beSAndroid Build Coastguard Worker 	const struct io_mock linux_spi_io = {
63*0d6140beSAndroid Build Coastguard Worker 		.iom_fgets	= linux_spi_fgets,
64*0d6140beSAndroid Build Coastguard Worker 		.iom_ioctl	= linux_spi_ioctl,
65*0d6140beSAndroid Build Coastguard Worker 		.fallback_open_state = &linux_spi_fallback_open_state,
66*0d6140beSAndroid Build Coastguard Worker 	};
67*0d6140beSAndroid Build Coastguard Worker 
68*0d6140beSAndroid Build Coastguard Worker 	run_probe_lifecycle(state, &linux_spi_io, &programmer_linux_spi, "dev=/dev/null", "W25Q128.V");
69*0d6140beSAndroid Build Coastguard Worker }
70*0d6140beSAndroid Build Coastguard Worker #else
71*0d6140beSAndroid Build Coastguard Worker 	SKIP_TEST(linux_spi_probe_lifecycle_test_success)
72*0d6140beSAndroid Build Coastguard Worker #endif /* CONFIG_LINUX_SPI */
73