Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Yingkun Meng | 674 | 93.09% | 1 | 20.00% |
Binbin Zhou | 28 | 3.87% | 2 | 40.00% |
Arnd Bergmann | 12 | 1.66% | 1 | 20.00% |
tangbin | 10 | 1.38% | 1 | 20.00% |
Total | 724 | 5 |
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
// 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");
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1