summaryrefslogtreecommitdiff
path: root/sound/soc/codecs/rt5575-spi.c
blob: 9a349965435b64bd9251b329308512aeed449faa (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
// 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;
}