// SPDX-License-Identifier: GPL-2.0+ /* * (C) Copyright 2002 * Gerald Van Baren, Custom IDEAS, vanbaren@cideas.com */ /* * SPI Read/Write Utilities */ #include #include #include #include #include #include "spi-sunxi.h" /*----------------------------------------------------------------------- * Definitions */ #ifndef MAX_SPI_BYTES # define MAX_SPI_BYTES 64 /* Maximum number of bytes we can handle */ #endif #ifndef CONFIG_DEFAULT_SPI_BUS # define CONFIG_DEFAULT_SPI_BUS 1 #endif #ifndef CONFIG_DEFAULT_SPI_MODE # define CONFIG_DEFAULT_SPI_MODE SPI_MODE_3 #endif /* * Values from last command. */ static unsigned int bus; static unsigned int cs; static unsigned int mode; static int bitlen; static uchar dout[MAX_SPI_BYTES]; static uchar din[MAX_SPI_BYTES]; static int do_sunxi_spi_xfer(int bus, int cs) { struct sunxi_spi_slave *spi_slave; int ret = 0; spi_slave = get_sspi(bus); if (spi_slave != NULL) { ret = spi_claim_bus(&spi_slave->slave); if (ret) { printf("Failed to claim SPI bus: %d\n", ret); } ret = spi_xfer(&spi_slave->slave, bitlen, dout, din, SPI_XFER_END); if (ret) { printf("Error %d during SPI transaction\n", ret); return ret; } else { int j; printf("SPI SEND: \n"); for (j = 0; j < ((bitlen + 7) / 8); j++) printf("%02X", dout[j]); printf("\n"); printf("SPI RECV: \n"); for (j = 0; j < ((bitlen + 7) / 8); j++) printf("%02X", din[j]); printf("\n"); } } return ret; } /* * SPI read/write * * Syntax: * spi {dev} {num_bits} {dout} * {dev} is the device number for controlling chip select (see TBD) * {num_bits} is the number of bits to send & receive (base 10) * {dout} is a hexadecimal string of data to send * The command prints out the hexadecimal string received via SPI. */ int do_sunxi_sspi (cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[]) { char *cp = 0; uchar tmp; int j; /* * We use the last specified parameters, unless new ones are * entered. */ if ((flag & CMD_FLAG_REPEAT) == 0) { if (argc >= 2) { mode = CONFIG_DEFAULT_SPI_MODE; bus = simple_strtoul(argv[1], &cp, 10); if (*cp == ':') { cs = simple_strtoul(cp+1, &cp, 10); } else { cs = bus; bus = CONFIG_DEFAULT_SPI_BUS; } if (*cp == '.') mode = simple_strtoul(cp+1, NULL, 10); } if (argc >= 3) bitlen = simple_strtoul(argv[2], NULL, 10); if (argc >= 4) { cp = argv[3]; for (j = 0; *cp; j++, cp++) { tmp = *cp - '0'; if (tmp > 9) tmp -= ('A' - '0') - 10; if (tmp > 15) tmp -= ('a' - 'A'); if (tmp > 15) { printf("Hex conversion error on %c\n", *cp); return 1; } if ((j % 2) == 0) dout[j / 2] = (tmp << 4); else dout[j / 2] |= tmp; } } } if ((bitlen < 0) || (bitlen > (MAX_SPI_BYTES * 8))) { printf("Invalid bitlen %d\n", bitlen); return 1; } if (do_sunxi_spi_xfer(bus, cs)) return 1; return 0; } /***************************************************/ U_BOOT_CMD( sunxi_sspi, 5, 1, do_sunxi_sspi, "SPI utility command", "[:][.] - Send and receive bits\n" " - Identifies the SPI bus\n" " - Identifies the chip select\n" " - Identifies the SPI mode to use\n" " - Number of bits to send (base 10)\n" " - Hexadecimal string that gets sent" );