blob: 9a349965435b64bd9251b329308512aeed449faa [file] [log] [blame] [edit]
// SPDX-License-Identifier: GPL-2.0-only
/*
* rt5575-spi.c -- ALC5575 SPI driver
*
* Copyright(c) 2025 Realtek Semiconductor Corp.
*
*/
#include <linux/firmware.h>
#include <linux/of.h>
#include <linux/spi/spi.h>
#include "rt5575-spi.h"
#define RT5575_SPI_CMD_BURST_WRITE 5
#define RT5575_SPI_BUF_LEN 240
struct rt5575_spi_burst_write {
u8 cmd;
u32 addr;
u8 data[RT5575_SPI_BUF_LEN];
u8 dummy;
} __packed;
struct spi_device *rt5575_spi_get_device(struct device *dev)
{
struct spi_device *spi;
struct spi_controller *ctlr;
struct device_node *spi_np;
u32 cs;
spi_np = of_parse_phandle(dev->of_node, "spi-parent", 0);
if (!spi_np) {
dev_err(dev, "Failed to get spi-parent phandle\n");
return NULL;
}
if (of_property_read_u32_index(dev->of_node, "spi-parent", 1, &cs))
cs = 0;
ctlr = of_find_spi_controller_by_node(spi_np);
of_node_put(spi_np);
if (!ctlr) {
dev_err(dev, "Failed to get spi_controller\n");
return NULL;
}
if (cs >= ctlr->num_chipselect) {
dev_err(dev, "Chip select has wrong number %d\n", cs);
spi_controller_put(ctlr);
return NULL;
}
spi = spi_new_device(ctlr, &(struct spi_board_info){
.modalias = "rt5575",
.chip_select = cs,
.max_speed_hz = 10000000,
});
spi_controller_put(ctlr);
return spi;
}
/**
* rt5575_spi_burst_write - Write data to SPI by rt5575 address.
* @spi: SPI device.
* @addr: Start address.
* @txbuf: Data buffer for writing.
* @len: Data length.
*
*/
static void rt5575_spi_burst_write(struct spi_device *spi, u32 addr, const u8 *txbuf, size_t len)
{
struct rt5575_spi_burst_write buf = {
.cmd = RT5575_SPI_CMD_BURST_WRITE,
};
unsigned int end, offset = 0;
while (offset < len) {
if (offset + RT5575_SPI_BUF_LEN <= len)
end = RT5575_SPI_BUF_LEN;
else
end = len % RT5575_SPI_BUF_LEN;
buf.addr = cpu_to_le32(addr + offset);
memcpy(&buf.data, &txbuf[offset], end);
spi_write(spi, &buf, sizeof(buf));
offset += RT5575_SPI_BUF_LEN;
}
}
int rt5575_spi_fw_load(struct spi_device *spi)
{
struct device *dev = &spi->dev;
const struct firmware *firmware;
int i, ret;
static const char * const fw_path[] = {
"realtek/rt5575/rt5575_fw1.bin",
"realtek/rt5575/rt5575_fw2.bin",
"realtek/rt5575/rt5575_fw3.bin",
"realtek/rt5575/rt5575_fw4.bin",
};
static const u32 fw_addr[] = { 0x5f400000, 0x5f600000, 0x5f7fe000, 0x5f7ff000 };
for (i = 0; i < ARRAY_SIZE(fw_addr); i++) {
ret = request_firmware(&firmware, fw_path[i], dev);
if (ret) {
dev_err(dev, "Request firmware failure: %d\n", ret);
return ret;
}
rt5575_spi_burst_write(spi, fw_addr[i], firmware->data, firmware->size);
release_firmware(firmware);
}
return 0;
}