Path: blob/master/sound/soc/loongson/loongson_i2s_pci.c
26437 views
// SPDX-License-Identifier: GPL-2.01//2// loongson_i2s_pci.c -- Loongson I2S controller driver3//4// Copyright (C) 2023 Loongson Technology Corporation Limited5// Author: Yingkun Meng <[email protected]>6//78#include <linux/module.h>9#include <linux/delay.h>10#include <linux/pm_runtime.h>11#include <linux/dma-mapping.h>12#include <linux/acpi.h>13#include <linux/pci.h>14#include <sound/soc.h>15#include "loongson_i2s.h"16#include "loongson_dma.h"1718#define DRIVER_NAME "loongson-i2s-pci"1920static bool loongson_i2s_wr_reg(struct device *dev, unsigned int reg)21{22switch (reg) {23case LS_I2S_CFG:24case LS_I2S_CTRL:25case LS_I2S_RX_DATA:26case LS_I2S_TX_DATA:27case LS_I2S_CFG1:28return true;29default:30return false;31};32}3334static bool loongson_i2s_rd_reg(struct device *dev, unsigned int reg)35{36switch (reg) {37case LS_I2S_VER:38case LS_I2S_CFG:39case LS_I2S_CTRL:40case LS_I2S_RX_DATA:41case LS_I2S_TX_DATA:42case LS_I2S_CFG1:43return true;44default:45return false;46};47}4849static bool loongson_i2s_volatile_reg(struct device *dev, unsigned int reg)50{51switch (reg) {52case LS_I2S_CFG:53case LS_I2S_CTRL:54case LS_I2S_RX_DATA:55case LS_I2S_TX_DATA:56case LS_I2S_CFG1:57return true;58default:59return false;60};61}6263static const struct regmap_config loongson_i2s_regmap_config = {64.reg_bits = 32,65.reg_stride = 4,66.val_bits = 32,67.max_register = LS_I2S_CFG1,68.writeable_reg = loongson_i2s_wr_reg,69.readable_reg = loongson_i2s_rd_reg,70.volatile_reg = loongson_i2s_volatile_reg,71.cache_type = REGCACHE_FLAT,72};7374static int loongson_i2s_pci_probe(struct pci_dev *pdev,75const struct pci_device_id *pid)76{77const struct fwnode_handle *fwnode = pdev->dev.fwnode;78struct loongson_dma_data *tx_data, *rx_data;79struct device *dev = &pdev->dev;80struct loongson_i2s *i2s;81int ret;8283if (pcim_enable_device(pdev)) {84dev_err(dev, "pci_enable_device failed\n");85return -ENODEV;86}8788i2s = devm_kzalloc(dev, sizeof(*i2s), GFP_KERNEL);89if (!i2s)90return -ENOMEM;9192i2s->rev_id = pdev->revision;93i2s->dev = dev;94pci_set_drvdata(pdev, i2s);9596i2s->reg_base = pcim_iomap_region(pdev, 0, DRIVER_NAME);97if (IS_ERR(i2s->reg_base)) {98dev_err(dev, "iomap_region failed\n");99return PTR_ERR(i2s->reg_base);100}101102i2s->regmap = devm_regmap_init_mmio(dev, i2s->reg_base,103&loongson_i2s_regmap_config);104if (IS_ERR(i2s->regmap))105return dev_err_probe(dev, PTR_ERR(i2s->regmap), "regmap_init_mmio failed\n");106107tx_data = &i2s->tx_dma_data;108rx_data = &i2s->rx_dma_data;109110tx_data->dev_addr = pci_resource_start(pdev, 0) + LS_I2S_TX_DATA;111tx_data->order_addr = i2s->reg_base + LS_I2S_TX_ORDER;112113rx_data->dev_addr = pci_resource_start(pdev, 0) + LS_I2S_RX_DATA;114rx_data->order_addr = i2s->reg_base + LS_I2S_RX_ORDER;115116tx_data->irq = fwnode_irq_get_byname(fwnode, "tx");117if (tx_data->irq < 0)118return dev_err_probe(dev, tx_data->irq, "dma tx irq invalid\n");119120rx_data->irq = fwnode_irq_get_byname(fwnode, "rx");121if (rx_data->irq < 0)122return dev_err_probe(dev, rx_data->irq, "dma rx irq invalid\n");123124ret = device_property_read_u32(dev, "clock-frequency", &i2s->clk_rate);125if (ret)126return dev_err_probe(dev, ret, "clock-frequency property invalid\n");127128dma_set_mask_and_coherent(dev, DMA_BIT_MASK(64));129130if (i2s->rev_id == 1) {131regmap_write(i2s->regmap, LS_I2S_CTRL, I2S_CTRL_RESET);132udelay(200);133}134135ret = devm_snd_soc_register_component(dev, &loongson_i2s_component,136&loongson_i2s_dai, 1);137if (ret)138return dev_err_probe(dev, ret, "register DAI failed\n");139140return 0;141}142143static const struct pci_device_id loongson_i2s_ids[] = {144{ PCI_DEVICE(PCI_VENDOR_ID_LOONGSON, 0x7a27) },145{ },146};147MODULE_DEVICE_TABLE(pci, loongson_i2s_ids);148149static struct pci_driver loongson_i2s_driver = {150.name = DRIVER_NAME,151.id_table = loongson_i2s_ids,152.probe = loongson_i2s_pci_probe,153.driver = {154.pm = pm_sleep_ptr(&loongson_i2s_pm),155},156};157module_pci_driver(loongson_i2s_driver);158159MODULE_DESCRIPTION("Loongson I2S Master Mode ASoC Driver");160MODULE_AUTHOR("Loongson Technology Corporation Limited");161MODULE_LICENSE("GPL");162163164