mirror of
https://github.com/torvalds/linux.git
synced 2024-12-11 13:41:55 +00:00
5e6f78cb5f
Use the function dev_err_probe can simplify code, but
the error return should not be deleted, that is
unreasonable, thus fix it.
Fixes: 3d2528d6c0
("ASoC: loongson: Simplify with dev_err_probe()")
Signed-off-by: Tang Bin <tangbin@cmss.chinamobile.com>
Link: https://patch.msgid.link/20240910021104.3400-1-tangbin@cmss.chinamobile.com
Signed-off-by: Mark Brown <broonie@kernel.org>
162 lines
3.9 KiB
C
162 lines
3.9 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
//
|
|
// loongson_i2s_pci.c -- Loongson I2S controller driver
|
|
//
|
|
// Copyright (C) 2023 Loongson Technology Corporation Limited
|
|
// Author: Yingkun Meng <mengyingkun@loongson.cn>
|
|
//
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/pm_runtime.h>
|
|
#include <linux/dma-mapping.h>
|
|
#include <linux/acpi.h>
|
|
#include <linux/pci.h>
|
|
#include <sound/soc.h>
|
|
#include "loongson_i2s.h"
|
|
#include "loongson_dma.h"
|
|
|
|
static bool loongson_i2s_wr_reg(struct device *dev, unsigned int reg)
|
|
{
|
|
switch (reg) {
|
|
case LS_I2S_CFG:
|
|
case LS_I2S_CTRL:
|
|
case LS_I2S_RX_DATA:
|
|
case LS_I2S_TX_DATA:
|
|
case LS_I2S_CFG1:
|
|
return true;
|
|
default:
|
|
return false;
|
|
};
|
|
}
|
|
|
|
static bool loongson_i2s_rd_reg(struct device *dev, unsigned int reg)
|
|
{
|
|
switch (reg) {
|
|
case LS_I2S_VER:
|
|
case LS_I2S_CFG:
|
|
case LS_I2S_CTRL:
|
|
case LS_I2S_RX_DATA:
|
|
case LS_I2S_TX_DATA:
|
|
case LS_I2S_CFG1:
|
|
return true;
|
|
default:
|
|
return false;
|
|
};
|
|
}
|
|
|
|
static bool loongson_i2s_volatile_reg(struct device *dev, unsigned int reg)
|
|
{
|
|
switch (reg) {
|
|
case LS_I2S_CFG:
|
|
case LS_I2S_CTRL:
|
|
case LS_I2S_RX_DATA:
|
|
case LS_I2S_TX_DATA:
|
|
case LS_I2S_CFG1:
|
|
return true;
|
|
default:
|
|
return false;
|
|
};
|
|
}
|
|
|
|
static const struct regmap_config loongson_i2s_regmap_config = {
|
|
.reg_bits = 32,
|
|
.reg_stride = 4,
|
|
.val_bits = 32,
|
|
.max_register = LS_I2S_CFG1,
|
|
.writeable_reg = loongson_i2s_wr_reg,
|
|
.readable_reg = loongson_i2s_rd_reg,
|
|
.volatile_reg = loongson_i2s_volatile_reg,
|
|
.cache_type = REGCACHE_FLAT,
|
|
};
|
|
|
|
static int loongson_i2s_pci_probe(struct pci_dev *pdev,
|
|
const struct pci_device_id *pid)
|
|
{
|
|
const struct fwnode_handle *fwnode = pdev->dev.fwnode;
|
|
struct loongson_dma_data *tx_data, *rx_data;
|
|
struct device *dev = &pdev->dev;
|
|
struct loongson_i2s *i2s;
|
|
int ret;
|
|
|
|
if (pcim_enable_device(pdev)) {
|
|
dev_err(dev, "pci_enable_device failed\n");
|
|
return -ENODEV;
|
|
}
|
|
|
|
i2s = devm_kzalloc(dev, sizeof(*i2s), GFP_KERNEL);
|
|
if (!i2s)
|
|
return -ENOMEM;
|
|
|
|
i2s->rev_id = pdev->revision;
|
|
i2s->dev = dev;
|
|
pci_set_drvdata(pdev, i2s);
|
|
|
|
ret = pcim_iomap_regions(pdev, 1 << 0, dev_name(dev));
|
|
if (ret < 0) {
|
|
dev_err(dev, "iomap_regions failed\n");
|
|
return ret;
|
|
}
|
|
|
|
i2s->reg_base = pcim_iomap_table(pdev)[0];
|
|
i2s->regmap = devm_regmap_init_mmio(dev, i2s->reg_base,
|
|
&loongson_i2s_regmap_config);
|
|
if (IS_ERR(i2s->regmap))
|
|
return dev_err_probe(dev, PTR_ERR(i2s->regmap), "regmap_init_mmio failed\n");
|
|
|
|
tx_data = &i2s->tx_dma_data;
|
|
rx_data = &i2s->rx_dma_data;
|
|
|
|
tx_data->dev_addr = pci_resource_start(pdev, 0) + LS_I2S_TX_DATA;
|
|
tx_data->order_addr = i2s->reg_base + LS_I2S_TX_ORDER;
|
|
|
|
rx_data->dev_addr = pci_resource_start(pdev, 0) + LS_I2S_RX_DATA;
|
|
rx_data->order_addr = i2s->reg_base + LS_I2S_RX_ORDER;
|
|
|
|
tx_data->irq = fwnode_irq_get_byname(fwnode, "tx");
|
|
if (tx_data->irq < 0)
|
|
return dev_err_probe(dev, tx_data->irq, "dma tx irq invalid\n");
|
|
|
|
rx_data->irq = fwnode_irq_get_byname(fwnode, "rx");
|
|
if (rx_data->irq < 0)
|
|
return dev_err_probe(dev, rx_data->irq, "dma rx irq invalid\n");
|
|
|
|
ret = device_property_read_u32(dev, "clock-frequency", &i2s->clk_rate);
|
|
if (ret)
|
|
return dev_err_probe(dev, ret, "clock-frequency property invalid\n");
|
|
|
|
dma_set_mask_and_coherent(dev, DMA_BIT_MASK(64));
|
|
|
|
if (i2s->rev_id == 1) {
|
|
regmap_write(i2s->regmap, LS_I2S_CTRL, I2S_CTRL_RESET);
|
|
udelay(200);
|
|
}
|
|
|
|
ret = devm_snd_soc_register_component(dev, &loongson_i2s_component,
|
|
&loongson_i2s_dai, 1);
|
|
if (ret)
|
|
return dev_err_probe(dev, ret, "register DAI failed\n");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct pci_device_id loongson_i2s_ids[] = {
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_LOONGSON, 0x7a27) },
|
|
{ },
|
|
};
|
|
MODULE_DEVICE_TABLE(pci, loongson_i2s_ids);
|
|
|
|
static struct pci_driver loongson_i2s_driver = {
|
|
.name = "loongson-i2s-pci",
|
|
.id_table = loongson_i2s_ids,
|
|
.probe = loongson_i2s_pci_probe,
|
|
.driver = {
|
|
.pm = pm_sleep_ptr(&loongson_i2s_pm),
|
|
},
|
|
};
|
|
module_pci_driver(loongson_i2s_driver);
|
|
|
|
MODULE_DESCRIPTION("Loongson I2S Master Mode ASoC Driver");
|
|
MODULE_AUTHOR("Loongson Technology Corporation Limited");
|
|
MODULE_LICENSE("GPL");
|