xref: /linux/sound/soc/loongson/loongson_i2s_pci.c (revision c335412c)
1d84881e0SYingkun Meng // SPDX-License-Identifier: GPL-2.0
2d84881e0SYingkun Meng //
3d84881e0SYingkun Meng // loongson_i2s_pci.c -- Loongson I2S controller driver
4d84881e0SYingkun Meng //
5d84881e0SYingkun Meng // Copyright (C) 2023 Loongson Technology Corporation Limited
6d84881e0SYingkun Meng // Author: Yingkun Meng <mengyingkun@loongson.cn>
7d84881e0SYingkun Meng //
8d84881e0SYingkun Meng 
9d84881e0SYingkun Meng #include <linux/module.h>
10d84881e0SYingkun Meng #include <linux/delay.h>
11d84881e0SYingkun Meng #include <linux/pm_runtime.h>
12d84881e0SYingkun Meng #include <linux/dma-mapping.h>
13d84881e0SYingkun Meng #include <linux/acpi.h>
14d84881e0SYingkun Meng #include <linux/pci.h>
15d84881e0SYingkun Meng #include <sound/soc.h>
16d84881e0SYingkun Meng #include "loongson_i2s.h"
17d84881e0SYingkun Meng #include "loongson_dma.h"
18d84881e0SYingkun Meng 
loongson_i2s_wr_reg(struct device * dev,unsigned int reg)19d84881e0SYingkun Meng static bool loongson_i2s_wr_reg(struct device *dev, unsigned int reg)
20d84881e0SYingkun Meng {
21d84881e0SYingkun Meng 	switch (reg) {
22d84881e0SYingkun Meng 	case LS_I2S_CFG:
23d84881e0SYingkun Meng 	case LS_I2S_CTRL:
24d84881e0SYingkun Meng 	case LS_I2S_RX_DATA:
25d84881e0SYingkun Meng 	case LS_I2S_TX_DATA:
26d84881e0SYingkun Meng 	case LS_I2S_CFG1:
27d84881e0SYingkun Meng 		return true;
28d84881e0SYingkun Meng 	default:
29d84881e0SYingkun Meng 		return false;
30d84881e0SYingkun Meng 	};
31d84881e0SYingkun Meng }
32d84881e0SYingkun Meng 
loongson_i2s_rd_reg(struct device * dev,unsigned int reg)33d84881e0SYingkun Meng static bool loongson_i2s_rd_reg(struct device *dev, unsigned int reg)
34d84881e0SYingkun Meng {
35d84881e0SYingkun Meng 	switch (reg) {
36d84881e0SYingkun Meng 	case LS_I2S_VER:
37d84881e0SYingkun Meng 	case LS_I2S_CFG:
38d84881e0SYingkun Meng 	case LS_I2S_CTRL:
39d84881e0SYingkun Meng 	case LS_I2S_RX_DATA:
40d84881e0SYingkun Meng 	case LS_I2S_TX_DATA:
41d84881e0SYingkun Meng 	case LS_I2S_CFG1:
42d84881e0SYingkun Meng 		return true;
43d84881e0SYingkun Meng 	default:
44d84881e0SYingkun Meng 		return false;
45d84881e0SYingkun Meng 	};
46d84881e0SYingkun Meng }
47d84881e0SYingkun Meng 
loongson_i2s_volatile_reg(struct device * dev,unsigned int reg)48d84881e0SYingkun Meng static bool loongson_i2s_volatile_reg(struct device *dev, unsigned int reg)
49d84881e0SYingkun Meng {
50d84881e0SYingkun Meng 	switch (reg) {
51d84881e0SYingkun Meng 	case LS_I2S_CFG:
52d84881e0SYingkun Meng 	case LS_I2S_CTRL:
53d84881e0SYingkun Meng 	case LS_I2S_RX_DATA:
54d84881e0SYingkun Meng 	case LS_I2S_TX_DATA:
55d84881e0SYingkun Meng 	case LS_I2S_CFG1:
56d84881e0SYingkun Meng 		return true;
57d84881e0SYingkun Meng 	default:
58d84881e0SYingkun Meng 		return false;
59d84881e0SYingkun Meng 	};
60d84881e0SYingkun Meng }
61d84881e0SYingkun Meng 
62d84881e0SYingkun Meng static const struct regmap_config loongson_i2s_regmap_config = {
63d84881e0SYingkun Meng 	.reg_bits = 32,
64d84881e0SYingkun Meng 	.reg_stride = 4,
65d84881e0SYingkun Meng 	.val_bits = 32,
66d84881e0SYingkun Meng 	.max_register = LS_I2S_CFG1,
67d84881e0SYingkun Meng 	.writeable_reg = loongson_i2s_wr_reg,
68d84881e0SYingkun Meng 	.readable_reg = loongson_i2s_rd_reg,
69d84881e0SYingkun Meng 	.volatile_reg = loongson_i2s_volatile_reg,
70d84881e0SYingkun Meng 	.cache_type = REGCACHE_FLAT,
71d84881e0SYingkun Meng };
72d84881e0SYingkun Meng 
loongson_i2s_pci_probe(struct pci_dev * pdev,const struct pci_device_id * pid)73d84881e0SYingkun Meng static int loongson_i2s_pci_probe(struct pci_dev *pdev,
74d84881e0SYingkun Meng 				  const struct pci_device_id *pid)
75d84881e0SYingkun Meng {
76d84881e0SYingkun Meng 	const struct fwnode_handle *fwnode = pdev->dev.fwnode;
77d84881e0SYingkun Meng 	struct loongson_dma_data *tx_data, *rx_data;
78d84881e0SYingkun Meng 	struct loongson_i2s *i2s;
79d84881e0SYingkun Meng 	int ret;
80d84881e0SYingkun Meng 
81d84881e0SYingkun Meng 	if (pcim_enable_device(pdev)) {
82d84881e0SYingkun Meng 		dev_err(&pdev->dev, "pci_enable_device failed\n");
83d84881e0SYingkun Meng 		return -ENODEV;
84d84881e0SYingkun Meng 	}
85d84881e0SYingkun Meng 
86d84881e0SYingkun Meng 	i2s = devm_kzalloc(&pdev->dev, sizeof(*i2s), GFP_KERNEL);
87d84881e0SYingkun Meng 	if (!i2s)
88d84881e0SYingkun Meng 		return -ENOMEM;
89d84881e0SYingkun Meng 
90d84881e0SYingkun Meng 	i2s->rev_id = pdev->revision;
91d84881e0SYingkun Meng 	i2s->dev = &pdev->dev;
92d84881e0SYingkun Meng 	pci_set_drvdata(pdev, i2s);
93d84881e0SYingkun Meng 
94d84881e0SYingkun Meng 	ret = pcim_iomap_regions(pdev, 1 << 0, dev_name(&pdev->dev));
95d84881e0SYingkun Meng 	if (ret < 0) {
96d84881e0SYingkun Meng 		dev_err(&pdev->dev, "iomap_regions failed\n");
97d84881e0SYingkun Meng 		return ret;
98d84881e0SYingkun Meng 	}
99d84881e0SYingkun Meng 	i2s->reg_base = pcim_iomap_table(pdev)[0];
100d84881e0SYingkun Meng 	i2s->regmap = devm_regmap_init_mmio(&pdev->dev, i2s->reg_base,
101d84881e0SYingkun Meng 					    &loongson_i2s_regmap_config);
102d84881e0SYingkun Meng 	if (IS_ERR(i2s->regmap)) {
103d84881e0SYingkun Meng 		dev_err(&pdev->dev, "regmap_init_mmio failed\n");
104d84881e0SYingkun Meng 		return PTR_ERR(i2s->regmap);
105d84881e0SYingkun Meng 	}
106d84881e0SYingkun Meng 
107d84881e0SYingkun Meng 	tx_data = &i2s->tx_dma_data;
108d84881e0SYingkun Meng 	rx_data = &i2s->rx_dma_data;
109d84881e0SYingkun Meng 
110*012fa262SArnd Bergmann 	tx_data->dev_addr = pci_resource_start(pdev, 0) + LS_I2S_TX_DATA;
111d84881e0SYingkun Meng 	tx_data->order_addr = i2s->reg_base + LS_I2S_TX_ORDER;
112d84881e0SYingkun Meng 
113*012fa262SArnd Bergmann 	rx_data->dev_addr = pci_resource_start(pdev, 0) + LS_I2S_RX_DATA;
114d84881e0SYingkun Meng 	rx_data->order_addr = i2s->reg_base + LS_I2S_RX_ORDER;
115d84881e0SYingkun Meng 
116d84881e0SYingkun Meng 	tx_data->irq = fwnode_irq_get_byname(fwnode, "tx");
117d84881e0SYingkun Meng 	if (tx_data->irq < 0) {
118d84881e0SYingkun Meng 		dev_err(&pdev->dev, "dma tx irq invalid\n");
119d84881e0SYingkun Meng 		return tx_data->irq;
120d84881e0SYingkun Meng 	}
121d84881e0SYingkun Meng 
122d84881e0SYingkun Meng 	rx_data->irq = fwnode_irq_get_byname(fwnode, "rx");
123d84881e0SYingkun Meng 	if (rx_data->irq < 0) {
124d84881e0SYingkun Meng 		dev_err(&pdev->dev, "dma rx irq invalid\n");
125d84881e0SYingkun Meng 		return rx_data->irq;
126d84881e0SYingkun Meng 	}
127d84881e0SYingkun Meng 
128d84881e0SYingkun Meng 	device_property_read_u32(&pdev->dev, "clock-frequency", &i2s->clk_rate);
129d84881e0SYingkun Meng 	if (!i2s->clk_rate) {
130d84881e0SYingkun Meng 		dev_err(&pdev->dev, "clock-frequency property invalid\n");
131d84881e0SYingkun Meng 		return -EINVAL;
132d84881e0SYingkun Meng 	}
133d84881e0SYingkun Meng 
134d84881e0SYingkun Meng 	dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(64));
135d84881e0SYingkun Meng 
136d84881e0SYingkun Meng 	if (i2s->rev_id == 1) {
137d84881e0SYingkun Meng 		regmap_write(i2s->regmap, LS_I2S_CTRL, I2S_CTRL_RESET);
138d84881e0SYingkun Meng 		udelay(200);
139d84881e0SYingkun Meng 	}
140d84881e0SYingkun Meng 
141d84881e0SYingkun Meng 	ret = devm_snd_soc_register_component(&pdev->dev,
142d84881e0SYingkun Meng 					      &loongson_i2s_component,
143d84881e0SYingkun Meng 					      &loongson_i2s_dai, 1);
144d84881e0SYingkun Meng 	if (ret) {
145d84881e0SYingkun Meng 		dev_err(&pdev->dev, "register DAI failed %d\n", ret);
146d84881e0SYingkun Meng 		return ret;
147d84881e0SYingkun Meng 	}
148d84881e0SYingkun Meng 
149d84881e0SYingkun Meng 	return 0;
150d84881e0SYingkun Meng }
151d84881e0SYingkun Meng 
152d84881e0SYingkun Meng static const struct pci_device_id loongson_i2s_ids[] = {
153d84881e0SYingkun Meng 	{ PCI_DEVICE(PCI_VENDOR_ID_LOONGSON, 0x7a27) },
154d84881e0SYingkun Meng 	{ },
155d84881e0SYingkun Meng };
156d84881e0SYingkun Meng MODULE_DEVICE_TABLE(pci, loongson_i2s_ids);
157d84881e0SYingkun Meng 
158d84881e0SYingkun Meng static struct pci_driver loongson_i2s_driver = {
159d84881e0SYingkun Meng 	.name = "loongson-i2s-pci",
160d84881e0SYingkun Meng 	.id_table = loongson_i2s_ids,
161d84881e0SYingkun Meng 	.probe = loongson_i2s_pci_probe,
162d84881e0SYingkun Meng 	.driver = {
163d84881e0SYingkun Meng 		.pm = pm_sleep_ptr(&loongson_i2s_pm),
164d84881e0SYingkun Meng 	},
165d84881e0SYingkun Meng };
166d84881e0SYingkun Meng module_pci_driver(loongson_i2s_driver);
167d84881e0SYingkun Meng 
168d84881e0SYingkun Meng MODULE_DESCRIPTION("Loongson I2S Master Mode ASoC Driver");
169d84881e0SYingkun Meng MODULE_AUTHOR("Loongson Technology Corporation Limited");
170d84881e0SYingkun Meng MODULE_LICENSE("GPL");
171