162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci//
362306a36Sopenharmony_ci// loongson_i2s_pci.c -- Loongson I2S controller driver
462306a36Sopenharmony_ci//
562306a36Sopenharmony_ci// Copyright (C) 2023 Loongson Technology Corporation Limited
662306a36Sopenharmony_ci// Author: Yingkun Meng <mengyingkun@loongson.cn>
762306a36Sopenharmony_ci//
862306a36Sopenharmony_ci
962306a36Sopenharmony_ci#include <linux/module.h>
1062306a36Sopenharmony_ci#include <linux/delay.h>
1162306a36Sopenharmony_ci#include <linux/pm_runtime.h>
1262306a36Sopenharmony_ci#include <linux/dma-mapping.h>
1362306a36Sopenharmony_ci#include <linux/acpi.h>
1462306a36Sopenharmony_ci#include <linux/pci.h>
1562306a36Sopenharmony_ci#include <sound/soc.h>
1662306a36Sopenharmony_ci#include "loongson_i2s.h"
1762306a36Sopenharmony_ci#include "loongson_dma.h"
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_cistatic bool loongson_i2s_wr_reg(struct device *dev, unsigned int reg)
2062306a36Sopenharmony_ci{
2162306a36Sopenharmony_ci	switch (reg) {
2262306a36Sopenharmony_ci	case LS_I2S_CFG:
2362306a36Sopenharmony_ci	case LS_I2S_CTRL:
2462306a36Sopenharmony_ci	case LS_I2S_RX_DATA:
2562306a36Sopenharmony_ci	case LS_I2S_TX_DATA:
2662306a36Sopenharmony_ci	case LS_I2S_CFG1:
2762306a36Sopenharmony_ci		return true;
2862306a36Sopenharmony_ci	default:
2962306a36Sopenharmony_ci		return false;
3062306a36Sopenharmony_ci	};
3162306a36Sopenharmony_ci}
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_cistatic bool loongson_i2s_rd_reg(struct device *dev, unsigned int reg)
3462306a36Sopenharmony_ci{
3562306a36Sopenharmony_ci	switch (reg) {
3662306a36Sopenharmony_ci	case LS_I2S_VER:
3762306a36Sopenharmony_ci	case LS_I2S_CFG:
3862306a36Sopenharmony_ci	case LS_I2S_CTRL:
3962306a36Sopenharmony_ci	case LS_I2S_RX_DATA:
4062306a36Sopenharmony_ci	case LS_I2S_TX_DATA:
4162306a36Sopenharmony_ci	case LS_I2S_CFG1:
4262306a36Sopenharmony_ci		return true;
4362306a36Sopenharmony_ci	default:
4462306a36Sopenharmony_ci		return false;
4562306a36Sopenharmony_ci	};
4662306a36Sopenharmony_ci}
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_cistatic bool loongson_i2s_volatile_reg(struct device *dev, unsigned int reg)
4962306a36Sopenharmony_ci{
5062306a36Sopenharmony_ci	switch (reg) {
5162306a36Sopenharmony_ci	case LS_I2S_CFG:
5262306a36Sopenharmony_ci	case LS_I2S_CTRL:
5362306a36Sopenharmony_ci	case LS_I2S_RX_DATA:
5462306a36Sopenharmony_ci	case LS_I2S_TX_DATA:
5562306a36Sopenharmony_ci	case LS_I2S_CFG1:
5662306a36Sopenharmony_ci		return true;
5762306a36Sopenharmony_ci	default:
5862306a36Sopenharmony_ci		return false;
5962306a36Sopenharmony_ci	};
6062306a36Sopenharmony_ci}
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_cistatic const struct regmap_config loongson_i2s_regmap_config = {
6362306a36Sopenharmony_ci	.reg_bits = 32,
6462306a36Sopenharmony_ci	.reg_stride = 4,
6562306a36Sopenharmony_ci	.val_bits = 32,
6662306a36Sopenharmony_ci	.max_register = LS_I2S_CFG1,
6762306a36Sopenharmony_ci	.writeable_reg = loongson_i2s_wr_reg,
6862306a36Sopenharmony_ci	.readable_reg = loongson_i2s_rd_reg,
6962306a36Sopenharmony_ci	.volatile_reg = loongson_i2s_volatile_reg,
7062306a36Sopenharmony_ci	.cache_type = REGCACHE_FLAT,
7162306a36Sopenharmony_ci};
7262306a36Sopenharmony_ci
7362306a36Sopenharmony_cistatic int loongson_i2s_pci_probe(struct pci_dev *pdev,
7462306a36Sopenharmony_ci				  const struct pci_device_id *pid)
7562306a36Sopenharmony_ci{
7662306a36Sopenharmony_ci	const struct fwnode_handle *fwnode = pdev->dev.fwnode;
7762306a36Sopenharmony_ci	struct loongson_dma_data *tx_data, *rx_data;
7862306a36Sopenharmony_ci	struct loongson_i2s *i2s;
7962306a36Sopenharmony_ci	int ret;
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_ci	if (pcim_enable_device(pdev)) {
8262306a36Sopenharmony_ci		dev_err(&pdev->dev, "pci_enable_device failed\n");
8362306a36Sopenharmony_ci		return -ENODEV;
8462306a36Sopenharmony_ci	}
8562306a36Sopenharmony_ci
8662306a36Sopenharmony_ci	i2s = devm_kzalloc(&pdev->dev, sizeof(*i2s), GFP_KERNEL);
8762306a36Sopenharmony_ci	if (!i2s)
8862306a36Sopenharmony_ci		return -ENOMEM;
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_ci	i2s->rev_id = pdev->revision;
9162306a36Sopenharmony_ci	i2s->dev = &pdev->dev;
9262306a36Sopenharmony_ci	pci_set_drvdata(pdev, i2s);
9362306a36Sopenharmony_ci
9462306a36Sopenharmony_ci	ret = pcim_iomap_regions(pdev, 1 << 0, dev_name(&pdev->dev));
9562306a36Sopenharmony_ci	if (ret < 0) {
9662306a36Sopenharmony_ci		dev_err(&pdev->dev, "iomap_regions failed\n");
9762306a36Sopenharmony_ci		return ret;
9862306a36Sopenharmony_ci	}
9962306a36Sopenharmony_ci	i2s->reg_base = pcim_iomap_table(pdev)[0];
10062306a36Sopenharmony_ci	i2s->regmap = devm_regmap_init_mmio(&pdev->dev, i2s->reg_base,
10162306a36Sopenharmony_ci					    &loongson_i2s_regmap_config);
10262306a36Sopenharmony_ci	if (IS_ERR(i2s->regmap)) {
10362306a36Sopenharmony_ci		dev_err(&pdev->dev, "regmap_init_mmio failed\n");
10462306a36Sopenharmony_ci		return PTR_ERR(i2s->regmap);
10562306a36Sopenharmony_ci	}
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_ci	tx_data = &i2s->tx_dma_data;
10862306a36Sopenharmony_ci	rx_data = &i2s->rx_dma_data;
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_ci	tx_data->dev_addr = pci_resource_start(pdev, 0) + LS_I2S_TX_DATA;
11162306a36Sopenharmony_ci	tx_data->order_addr = i2s->reg_base + LS_I2S_TX_ORDER;
11262306a36Sopenharmony_ci
11362306a36Sopenharmony_ci	rx_data->dev_addr = pci_resource_start(pdev, 0) + LS_I2S_RX_DATA;
11462306a36Sopenharmony_ci	rx_data->order_addr = i2s->reg_base + LS_I2S_RX_ORDER;
11562306a36Sopenharmony_ci
11662306a36Sopenharmony_ci	tx_data->irq = fwnode_irq_get_byname(fwnode, "tx");
11762306a36Sopenharmony_ci	if (tx_data->irq < 0) {
11862306a36Sopenharmony_ci		dev_err(&pdev->dev, "dma tx irq invalid\n");
11962306a36Sopenharmony_ci		return tx_data->irq;
12062306a36Sopenharmony_ci	}
12162306a36Sopenharmony_ci
12262306a36Sopenharmony_ci	rx_data->irq = fwnode_irq_get_byname(fwnode, "rx");
12362306a36Sopenharmony_ci	if (rx_data->irq < 0) {
12462306a36Sopenharmony_ci		dev_err(&pdev->dev, "dma rx irq invalid\n");
12562306a36Sopenharmony_ci		return rx_data->irq;
12662306a36Sopenharmony_ci	}
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci	device_property_read_u32(&pdev->dev, "clock-frequency", &i2s->clk_rate);
12962306a36Sopenharmony_ci	if (!i2s->clk_rate) {
13062306a36Sopenharmony_ci		dev_err(&pdev->dev, "clock-frequency property invalid\n");
13162306a36Sopenharmony_ci		return -EINVAL;
13262306a36Sopenharmony_ci	}
13362306a36Sopenharmony_ci
13462306a36Sopenharmony_ci	dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(64));
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ci	if (i2s->rev_id == 1) {
13762306a36Sopenharmony_ci		regmap_write(i2s->regmap, LS_I2S_CTRL, I2S_CTRL_RESET);
13862306a36Sopenharmony_ci		udelay(200);
13962306a36Sopenharmony_ci	}
14062306a36Sopenharmony_ci
14162306a36Sopenharmony_ci	ret = devm_snd_soc_register_component(&pdev->dev,
14262306a36Sopenharmony_ci					      &loongson_i2s_component,
14362306a36Sopenharmony_ci					      &loongson_i2s_dai, 1);
14462306a36Sopenharmony_ci	if (ret) {
14562306a36Sopenharmony_ci		dev_err(&pdev->dev, "register DAI failed %d\n", ret);
14662306a36Sopenharmony_ci		return ret;
14762306a36Sopenharmony_ci	}
14862306a36Sopenharmony_ci
14962306a36Sopenharmony_ci	return 0;
15062306a36Sopenharmony_ci}
15162306a36Sopenharmony_ci
15262306a36Sopenharmony_cistatic const struct pci_device_id loongson_i2s_ids[] = {
15362306a36Sopenharmony_ci	{ PCI_DEVICE(PCI_VENDOR_ID_LOONGSON, 0x7a27) },
15462306a36Sopenharmony_ci	{ },
15562306a36Sopenharmony_ci};
15662306a36Sopenharmony_ciMODULE_DEVICE_TABLE(pci, loongson_i2s_ids);
15762306a36Sopenharmony_ci
15862306a36Sopenharmony_cistatic struct pci_driver loongson_i2s_driver = {
15962306a36Sopenharmony_ci	.name = "loongson-i2s-pci",
16062306a36Sopenharmony_ci	.id_table = loongson_i2s_ids,
16162306a36Sopenharmony_ci	.probe = loongson_i2s_pci_probe,
16262306a36Sopenharmony_ci	.driver = {
16362306a36Sopenharmony_ci		.owner = THIS_MODULE,
16462306a36Sopenharmony_ci		.pm = pm_sleep_ptr(&loongson_i2s_pm),
16562306a36Sopenharmony_ci	},
16662306a36Sopenharmony_ci};
16762306a36Sopenharmony_cimodule_pci_driver(loongson_i2s_driver);
16862306a36Sopenharmony_ci
16962306a36Sopenharmony_ciMODULE_DESCRIPTION("Loongson I2S Master Mode ASoC Driver");
17062306a36Sopenharmony_ciMODULE_AUTHOR("Loongson Technology Corporation Limited");
17162306a36Sopenharmony_ciMODULE_LICENSE("GPL");
172