Lines Matching +full:idma +full:- +full:addr
1 // SPDX-License-Identifier: GPL-2.0
3 // ALSA SoC Audio Layer - Samsung I2S Controller driver
8 #include <dt-bindings/sound/samsung-i2s.h>
12 #include <linux/clk-provider.h>
23 #include <linux/platform_data/asoc-s3c.h>
26 #include "idma.h"
28 #include "i2s-regs.h"
124 void __iomem *addr; member
133 return i2s->drv->id == SAMSUNG_I2S_ID_SECONDARY; in is_secondary()
144 active = readl(i2s->priv->addr + I2SCON); in tx_active()
157 return i2s->pri_dai ? : i2s->sec_dai; in get_other_dai()
182 active = readl(i2s->priv->addr + I2SCON) & CON_RXDMA_ACTIVE; in rx_active()
223 return &priv->dai[dai->id - 1]; in to_info()
228 if (i2s && (i2s->mode & DAI_OPENED)) in is_opened()
236 if (is_opened(i2s) && (i2s->mode & DAI_MANAGER)) in is_manager()
245 struct samsung_i2s_priv *priv = i2s->priv; in get_rfs()
248 rfs = readl(priv->addr + I2SMOD) >> priv->variant_regs->rfs_off; in get_rfs()
249 rfs &= priv->variant_regs->rfs_mask; in get_rfs()
266 struct samsung_i2s_priv *priv = i2s->priv; in set_rfs()
267 u32 mod = readl(priv->addr + I2SMOD); in set_rfs()
268 int rfs_shift = priv->variant_regs->rfs_off; in set_rfs()
270 mod &= ~(priv->variant_regs->rfs_mask << rfs_shift); in set_rfs()
299 writel(mod, priv->addr + I2SMOD); in set_rfs()
302 /* Read bit-clock of I2S (in multiples of LRCLK) */
305 struct samsung_i2s_priv *priv = i2s->priv; in get_bfs()
308 bfs = readl(priv->addr + I2SMOD) >> priv->variant_regs->bfs_off; in get_bfs()
309 bfs &= priv->variant_regs->bfs_mask; in get_bfs()
324 /* Write bit-clock of I2S (in multiples of LRCLK) */
327 struct samsung_i2s_priv *priv = i2s->priv; in set_bfs()
328 u32 mod = readl(priv->addr + I2SMOD); in set_bfs()
329 int tdm = priv->quirks & QUIRK_SUPPORTS_TDM; in set_bfs()
330 int bfs_shift = priv->variant_regs->bfs_off; in set_bfs()
332 /* Non-TDM I2S controllers do not support BCLK > 48 * FS */ in set_bfs()
334 dev_err(&i2s->pdev->dev, "Unsupported BCLK divider\n"); in set_bfs()
338 mod &= ~(priv->variant_regs->bfs_mask << bfs_shift); in set_bfs()
369 dev_err(&i2s->pdev->dev, "Wrong BCLK Divider!\n"); in set_bfs()
373 writel(mod, priv->addr + I2SMOD); in set_bfs()
379 int blc = readl(i2s->priv->addr + I2SMOD); in get_blc()
393 struct samsung_i2s_priv *priv = i2s->priv; in i2s_txctrl()
394 void __iomem *addr = priv->addr; in i2s_txctrl() local
395 int txr_off = priv->variant_regs->txr_off; in i2s_txctrl()
396 u32 con = readl(addr + I2SCON); in i2s_txctrl()
397 u32 mod = readl(addr + I2SMOD) & ~(3 << txr_off); in i2s_txctrl()
425 writel(con, addr + I2SCON); in i2s_txctrl()
437 writel(mod, addr + I2SMOD); in i2s_txctrl()
438 writel(con, addr + I2SCON); in i2s_txctrl()
444 struct samsung_i2s_priv *priv = i2s->priv; in i2s_rxctrl()
445 void __iomem *addr = priv->addr; in i2s_rxctrl() local
446 int txr_off = priv->variant_regs->txr_off; in i2s_rxctrl()
447 u32 con = readl(addr + I2SCON); in i2s_rxctrl()
448 u32 mod = readl(addr + I2SMOD) & ~(3 << txr_off); in i2s_rxctrl()
468 writel(mod, addr + I2SMOD); in i2s_rxctrl()
469 writel(con, addr + I2SCON); in i2s_rxctrl()
482 fic = i2s->priv->addr + I2SFICS; in i2s_fifo()
484 fic = i2s->priv->addr + I2SFIC; in i2s_fifo()
491 while (--val) in i2s_fifo()
503 const struct samsung_i2s_variant_regs *i2s_regs = priv->variant_regs; in i2s_set_sysclk()
504 unsigned int cdcon_mask = 1 << i2s_regs->cdclkcon_off; in i2s_set_sysclk()
505 unsigned int rsrc_mask = 1 << i2s_regs->rclksrc_off; in i2s_set_sysclk()
510 pm_runtime_get_sync(dai->dev); in i2s_set_sysclk()
512 spin_lock_irqsave(&priv->lock, flags); in i2s_set_sysclk()
513 mod = readl(priv->addr + I2SMOD); in i2s_set_sysclk()
514 spin_unlock_irqrestore(&priv->lock, flags); in i2s_set_sysclk()
522 mask = 1 << i2s_regs->cdclkcon_off; in i2s_set_sysclk()
527 if ((rfs && other && other->rfs && (other->rfs != rfs)) || in i2s_set_sysclk()
533 dev_err(&i2s->pdev->dev, in i2s_set_sysclk()
535 ret = -EAGAIN; in i2s_set_sysclk()
540 val = 1 << i2s_regs->cdclkcon_off; in i2s_set_sysclk()
542 i2s->rfs = rfs; in i2s_set_sysclk()
547 mask = 1 << i2s_regs->rclksrc_off; in i2s_set_sysclk()
549 if ((priv->quirks & QUIRK_NO_MUXPSR) in i2s_set_sysclk()
556 if (priv->op_clk && !IS_ERR(priv->op_clk)) { in i2s_set_sysclk()
559 clk_disable_unprepare(priv->op_clk); in i2s_set_sysclk()
560 clk_put(priv->op_clk); in i2s_set_sysclk()
562 priv->rclk_srcrate = in i2s_set_sysclk()
563 clk_get_rate(priv->op_clk); in i2s_set_sysclk()
569 priv->op_clk = clk_get(&i2s->pdev->dev, in i2s_set_sysclk()
572 priv->op_clk = clk_get(&i2s->pdev->dev, in i2s_set_sysclk()
575 if (WARN_ON(IS_ERR(priv->op_clk))) { in i2s_set_sysclk()
576 ret = PTR_ERR(priv->op_clk); in i2s_set_sysclk()
577 priv->op_clk = NULL; in i2s_set_sysclk()
581 ret = clk_prepare_enable(priv->op_clk); in i2s_set_sysclk()
583 clk_put(priv->op_clk); in i2s_set_sysclk()
584 priv->op_clk = NULL; in i2s_set_sysclk()
587 priv->rclk_srcrate = clk_get_rate(priv->op_clk); in i2s_set_sysclk()
591 dev_err(&i2s->pdev->dev, in i2s_set_sysclk()
593 ret = -EAGAIN; in i2s_set_sysclk()
601 val = 1 << i2s_regs->rclksrc_off; in i2s_set_sysclk()
604 dev_err(&i2s->pdev->dev, "We don't serve that!\n"); in i2s_set_sysclk()
605 ret = -EINVAL; in i2s_set_sysclk()
609 spin_lock_irqsave(&priv->lock, flags); in i2s_set_sysclk()
610 mod = readl(priv->addr + I2SMOD); in i2s_set_sysclk()
612 writel(mod, priv->addr + I2SMOD); in i2s_set_sysclk()
613 spin_unlock_irqrestore(&priv->lock, flags); in i2s_set_sysclk()
615 pm_runtime_put(dai->dev); in i2s_set_sysclk()
619 pm_runtime_put(dai->dev); in i2s_set_sysclk()
631 lrp_shift = priv->variant_regs->lrp_off; in i2s_set_fmt()
632 sdf_shift = priv->variant_regs->sdf_off; in i2s_set_fmt()
633 mod_slave = 1 << priv->variant_regs->mss_off; in i2s_set_fmt()
652 dev_err(&i2s->pdev->dev, "Format not supported\n"); in i2s_set_fmt()
653 return -EINVAL; in i2s_set_fmt()
657 * INV flag is relative to the FORMAT flag - if set it simply in i2s_set_fmt()
670 dev_err(&i2s->pdev->dev, "Polarity not supported\n"); in i2s_set_fmt()
671 return -EINVAL; in i2s_set_fmt()
684 if (priv->rclk_srcrate == 0 && priv->clk_data.clks == NULL) in i2s_set_fmt()
689 dev_err(&i2s->pdev->dev, "master/slave format not supported\n"); in i2s_set_fmt()
690 return -EINVAL; in i2s_set_fmt()
693 pm_runtime_get_sync(dai->dev); in i2s_set_fmt()
694 spin_lock_irqsave(&priv->lock, flags); in i2s_set_fmt()
695 mod = readl(priv->addr + I2SMOD); in i2s_set_fmt()
702 spin_unlock_irqrestore(&priv->lock, flags); in i2s_set_fmt()
703 pm_runtime_put(dai->dev); in i2s_set_fmt()
704 dev_err(&i2s->pdev->dev, in i2s_set_fmt()
706 return -EAGAIN; in i2s_set_fmt()
711 writel(mod, priv->addr + I2SMOD); in i2s_set_fmt()
712 priv->slave_mode = (mod & mod_slave); in i2s_set_fmt()
713 spin_unlock_irqrestore(&priv->lock, flags); in i2s_set_fmt()
714 pm_runtime_put(dai->dev); in i2s_set_fmt()
728 WARN_ON(!pm_runtime_active(dai->dev)); in i2s_hw_params()
741 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) in i2s_hw_params()
742 i2s->dma_playback.addr_width = 4; in i2s_hw_params()
744 i2s->dma_capture.addr_width = 4; in i2s_hw_params()
747 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) in i2s_hw_params()
748 i2s->dma_playback.addr_width = 2; in i2s_hw_params()
750 i2s->dma_capture.addr_width = 2; in i2s_hw_params()
754 dev_err(&i2s->pdev->dev, "%d channels not supported\n", in i2s_hw_params()
756 return -EINVAL; in i2s_hw_params()
793 dev_err(&i2s->pdev->dev, "Format(%d) not supported\n", in i2s_hw_params()
795 return -EINVAL; in i2s_hw_params()
798 spin_lock_irqsave(&priv->lock, flags); in i2s_hw_params()
799 mod = readl(priv->addr + I2SMOD); in i2s_hw_params()
801 writel(mod, priv->addr + I2SMOD); in i2s_hw_params()
802 spin_unlock_irqrestore(&priv->lock, flags); in i2s_hw_params()
804 snd_soc_dai_init_dma_data(dai, &i2s->dma_playback, &i2s->dma_capture); in i2s_hw_params()
806 i2s->frmclk = params_rate(params); in i2s_hw_params()
808 rclksrc = priv->clk_table[CLK_I2S_RCLK_SRC]; in i2s_hw_params()
810 priv->rclk_srcrate = clk_get_rate(rclksrc); in i2s_hw_params()
824 pm_runtime_get_sync(dai->dev); in i2s_startup()
826 spin_lock_irqsave(&priv->pcm_lock, flags); in i2s_startup()
828 i2s->mode |= DAI_OPENED; in i2s_startup()
831 i2s->mode &= ~DAI_MANAGER; in i2s_startup()
833 i2s->mode |= DAI_MANAGER; in i2s_startup()
835 if (!any_active(i2s) && (priv->quirks & QUIRK_NEED_RSTCLR)) in i2s_startup()
836 writel(CON_RSTCLR, i2s->priv->addr + I2SCON); in i2s_startup()
838 spin_unlock_irqrestore(&priv->pcm_lock, flags); in i2s_startup()
851 spin_lock_irqsave(&priv->pcm_lock, flags); in i2s_shutdown()
853 i2s->mode &= ~DAI_OPENED; in i2s_shutdown()
854 i2s->mode &= ~DAI_MANAGER; in i2s_shutdown()
857 other->mode |= DAI_MANAGER; in i2s_shutdown()
860 i2s->rfs = 0; in i2s_shutdown()
861 i2s->bfs = 0; in i2s_shutdown()
863 spin_unlock_irqrestore(&priv->pcm_lock, flags); in i2s_shutdown()
865 pm_runtime_put(dai->dev); in i2s_shutdown()
870 struct samsung_i2s_priv *priv = i2s->priv; in config_setup()
877 bfs = i2s->bfs; in config_setup()
880 bfs = other->bfs; in config_setup()
886 rfs = i2s->rfs; in config_setup()
889 rfs = other->rfs; in config_setup()
892 dev_err(&i2s->pdev->dev, in config_setup()
893 "%d-RFS not supported for 24-blc\n", rfs); in config_setup()
894 return -EINVAL; in config_setup()
906 dev_err(&i2s->pdev->dev, in config_setup()
908 return -EAGAIN; in config_setup()
915 if (priv->slave_mode) in config_setup()
918 if (!(priv->quirks & QUIRK_NO_MUXPSR)) { in config_setup()
919 psr = priv->rclk_srcrate / i2s->frmclk / rfs; in config_setup()
920 writel(((psr - 1) << 8) | PSR_PSREN, priv->addr + I2SPSR); in config_setup()
921 dev_dbg(&i2s->pdev->dev, in config_setup()
923 priv->rclk_srcrate, psr, rfs, bfs); in config_setup()
933 int capture = (substream->stream == SNDRV_PCM_STREAM_CAPTURE); in i2s_trigger()
942 pm_runtime_get_sync(dai->dev); in i2s_trigger()
943 spin_lock_irqsave(&priv->lock, flags); in i2s_trigger()
946 spin_unlock_irqrestore(&priv->lock, flags); in i2s_trigger()
947 return -EINVAL; in i2s_trigger()
955 spin_unlock_irqrestore(&priv->lock, flags); in i2s_trigger()
960 spin_lock_irqsave(&priv->lock, flags); in i2s_trigger()
970 spin_unlock_irqrestore(&priv->lock, flags); in i2s_trigger()
971 pm_runtime_put(dai->dev); in i2s_trigger()
986 pm_runtime_get_sync(dai->dev); in i2s_set_clkdiv()
988 || (other && other->bfs && (other->bfs != div))) { in i2s_set_clkdiv()
989 pm_runtime_put(dai->dev); in i2s_set_clkdiv()
990 dev_err(&i2s->pdev->dev, in i2s_set_clkdiv()
992 return -EAGAIN; in i2s_set_clkdiv()
994 i2s->bfs = div; in i2s_set_clkdiv()
995 pm_runtime_put(dai->dev); in i2s_set_clkdiv()
998 dev_err(&i2s->pdev->dev, in i2s_set_clkdiv()
1000 return -EINVAL; in i2s_set_clkdiv()
1011 u32 reg = readl(priv->addr + I2SFIC); in i2s_delay()
1014 WARN_ON(!pm_runtime_active(dai->dev)); in i2s_delay()
1016 if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) in i2s_delay()
1019 delay = FICS_TXCOUNT(readl(priv->addr + I2SFICS)); in i2s_delay()
1021 delay = (reg >> priv->variant_regs->ftx0cnt_off) & 0x7f; in i2s_delay()
1029 return pm_runtime_force_suspend(component->dev); in i2s_suspend()
1034 return pm_runtime_force_resume(component->dev); in i2s_resume()
1048 pm_runtime_get_sync(dai->dev); in samsung_i2s_dai_probe()
1052 snd_soc_dai_init_dma_data(dai, &i2s->dma_playback, NULL); in samsung_i2s_dai_probe()
1054 snd_soc_dai_init_dma_data(dai, &i2s->dma_playback, in samsung_i2s_dai_probe()
1055 &i2s->dma_capture); in samsung_i2s_dai_probe()
1057 if (priv->quirks & QUIRK_NEED_RSTCLR) in samsung_i2s_dai_probe()
1058 writel(CON_RSTCLR, priv->addr + I2SCON); in samsung_i2s_dai_probe()
1060 if (priv->quirks & QUIRK_SUPPORTS_IDMA) in samsung_i2s_dai_probe()
1061 idma_reg_addr_init(priv->addr, in samsung_i2s_dai_probe()
1062 other->idma_playback.addr); in samsung_i2s_dai_probe()
1066 i2s->rfs = 0; in samsung_i2s_dai_probe()
1067 i2s->bfs = 0; in samsung_i2s_dai_probe()
1069 spin_lock_irqsave(&priv->lock, flags); in samsung_i2s_dai_probe()
1075 spin_unlock_irqrestore(&priv->lock, flags); in samsung_i2s_dai_probe()
1081 pm_runtime_put(dai->dev); in samsung_i2s_dai_probe()
1092 pm_runtime_get_sync(dai->dev); in samsung_i2s_dai_remove()
1095 if (priv->quirks & QUIRK_NEED_RSTCLR) { in samsung_i2s_dai_remove()
1096 spin_lock_irqsave(&priv->lock, flags); in samsung_i2s_dai_remove()
1097 writel(0, priv->addr + I2SCON); in samsung_i2s_dai_remove()
1098 spin_unlock_irqrestore(&priv->lock, flags); in samsung_i2s_dai_remove()
1102 pm_runtime_put(dai->dev); in samsung_i2s_dai_remove()
1136 .name = "samsung-i2s",
1155 static const char *dai_names[] = { "samsung-i2s", "samsung-i2s-sec" }; in i2s_alloc_dais()
1162 priv->dai = devm_kcalloc(&priv->pdev->dev, num_dais, in i2s_alloc_dais()
1164 if (!priv->dai) in i2s_alloc_dais()
1165 return -ENOMEM; in i2s_alloc_dais()
1167 priv->dai_drv = devm_kcalloc(&priv->pdev->dev, num_dais, in i2s_alloc_dais()
1169 if (!priv->dai_drv) in i2s_alloc_dais()
1170 return -ENOMEM; in i2s_alloc_dais()
1173 dai_drv = &priv->dai_drv[i]; in i2s_alloc_dais()
1175 dai_drv->probe = samsung_i2s_dai_probe; in i2s_alloc_dais()
1176 dai_drv->remove = samsung_i2s_dai_remove; in i2s_alloc_dais()
1178 dai_drv->symmetric_rates = 1; in i2s_alloc_dais()
1179 dai_drv->ops = &samsung_i2s_dai_ops; in i2s_alloc_dais()
1181 dai_drv->playback.channels_min = 1; in i2s_alloc_dais()
1182 dai_drv->playback.channels_max = 2; in i2s_alloc_dais()
1183 dai_drv->playback.rates = i2s_dai_data->pcm_rates; in i2s_alloc_dais()
1184 dai_drv->playback.formats = SAMSUNG_I2S_FMTS; in i2s_alloc_dais()
1185 dai_drv->playback.stream_name = stream_names[i]; in i2s_alloc_dais()
1187 dai_drv->id = i + 1; in i2s_alloc_dais()
1188 dai_drv->name = dai_names[i]; in i2s_alloc_dais()
1190 priv->dai[i].drv = &priv->dai_drv[i]; in i2s_alloc_dais()
1191 priv->dai[i].pdev = priv->pdev; in i2s_alloc_dais()
1195 dai_drv = &priv->dai_drv[SAMSUNG_I2S_ID_PRIMARY - 1]; in i2s_alloc_dais()
1197 dai_drv->capture.channels_min = 1; in i2s_alloc_dais()
1198 dai_drv->capture.channels_max = 2; in i2s_alloc_dais()
1199 dai_drv->capture.rates = i2s_dai_data->pcm_rates; in i2s_alloc_dais()
1200 dai_drv->capture.formats = SAMSUNG_I2S_FMTS; in i2s_alloc_dais()
1201 dai_drv->capture.stream_name = "Primary Capture"; in i2s_alloc_dais()
1211 priv->suspend_i2smod = readl(priv->addr + I2SMOD); in i2s_runtime_suspend()
1212 priv->suspend_i2scon = readl(priv->addr + I2SCON); in i2s_runtime_suspend()
1213 priv->suspend_i2spsr = readl(priv->addr + I2SPSR); in i2s_runtime_suspend()
1215 if (priv->op_clk) in i2s_runtime_suspend()
1216 clk_disable_unprepare(priv->op_clk); in i2s_runtime_suspend()
1217 clk_disable_unprepare(priv->clk); in i2s_runtime_suspend()
1227 ret = clk_prepare_enable(priv->clk); in i2s_runtime_resume()
1231 if (priv->op_clk) { in i2s_runtime_resume()
1232 ret = clk_prepare_enable(priv->op_clk); in i2s_runtime_resume()
1234 clk_disable_unprepare(priv->clk); in i2s_runtime_resume()
1239 writel(priv->suspend_i2scon, priv->addr + I2SCON); in i2s_runtime_resume()
1240 writel(priv->suspend_i2smod, priv->addr + I2SMOD); in i2s_runtime_resume()
1241 writel(priv->suspend_i2spsr, priv->addr + I2SPSR); in i2s_runtime_resume()
1251 for (i = 0; i < priv->clk_data.clk_num; i++) { in i2s_unregister_clocks()
1252 if (!IS_ERR(priv->clk_table[i])) in i2s_unregister_clocks()
1253 clk_unregister(priv->clk_table[i]); in i2s_unregister_clocks()
1259 of_clk_del_provider(priv->pdev->dev.of_node); in i2s_unregister_clock_provider()
1270 struct device *dev = &priv->pdev->dev; in i2s_register_clock_provider()
1271 const struct samsung_i2s_variant_regs *reg_info = priv->variant_regs; in i2s_register_clock_provider()
1277 if (!of_find_property(dev->of_node, "#clock-cells", NULL)) in i2s_register_clock_provider()
1293 return -ENOMEM; in i2s_register_clock_provider()
1296 if (!(priv->quirks & QUIRK_NO_MUXPSR)) { in i2s_register_clock_provider()
1298 u32 val = readl(priv->addr + I2SPSR); in i2s_register_clock_provider()
1299 writel(val | PSR_PSREN, priv->addr + I2SPSR); in i2s_register_clock_provider()
1301 priv->clk_table[CLK_I2S_RCLK_SRC] = clk_register_mux(dev, in i2s_register_clock_provider()
1305 priv->addr + I2SMOD, reg_info->rclksrc_off, in i2s_register_clock_provider()
1306 1, 0, &priv->lock); in i2s_register_clock_provider()
1308 priv->clk_table[CLK_I2S_RCLK_PSR] = clk_register_divider(dev, in i2s_register_clock_provider()
1312 priv->addr + I2SPSR, 8, 6, 0, &priv->lock); in i2s_register_clock_provider()
1315 priv->clk_data.clk_num = 2; in i2s_register_clock_provider()
1318 priv->clk_table[CLK_I2S_CDCLK] = clk_register_gate(dev, in i2s_register_clock_provider()
1321 priv->addr + I2SMOD, reg_info->cdclkcon_off, in i2s_register_clock_provider()
1322 CLK_GATE_SET_TO_DISABLE, &priv->lock); in i2s_register_clock_provider()
1324 priv->clk_data.clk_num += 1; in i2s_register_clock_provider()
1325 priv->clk_data.clks = priv->clk_table; in i2s_register_clock_provider()
1327 ret = of_clk_add_provider(dev->of_node, of_clk_src_onecell_get, in i2s_register_clock_provider()
1328 &priv->clk_data); in i2s_register_clock_provider()
1344 devname = devm_kasprintf(&priv->pdev->dev, GFP_KERNEL, "%s-sec", in i2s_create_secondary_device()
1345 dev_name(&priv->pdev->dev)); in i2s_create_secondary_device()
1347 return -ENOMEM; in i2s_create_secondary_device()
1349 pdev_sec = platform_device_alloc(devname, -1); in i2s_create_secondary_device()
1351 return -ENOMEM; in i2s_create_secondary_device()
1353 pdev_sec->driver_override = kstrdup("samsung-i2s", GFP_KERNEL); in i2s_create_secondary_device()
1361 ret = device_attach(&pdev_sec->dev); in i2s_create_secondary_device()
1363 platform_device_unregister(priv->pdev_sec); in i2s_create_secondary_device()
1364 dev_info(&pdev_sec->dev, "device_attach() failed\n"); in i2s_create_secondary_device()
1368 priv->pdev_sec = pdev_sec; in i2s_create_secondary_device()
1375 platform_device_unregister(priv->pdev_sec); in i2s_delete_secondary_device()
1376 priv->pdev_sec = NULL; in i2s_delete_secondary_device()
1382 struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data; in samsung_i2s_probe()
1384 struct device_node *np = pdev->dev.of_node; in samsung_i2s_probe()
1391 if (IS_ENABLED(CONFIG_OF) && pdev->dev.of_node) { in samsung_i2s_probe()
1392 i2s_dai_data = of_device_get_match_data(&pdev->dev); in samsung_i2s_probe()
1400 i2s_dai_data = (struct samsung_i2s_dai_data *)id->driver_data; in samsung_i2s_probe()
1403 priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL); in samsung_i2s_probe()
1405 return -ENOMEM; in samsung_i2s_probe()
1408 priv->quirks = i2s_dai_data->quirks; in samsung_i2s_probe()
1411 dev_err(&pdev->dev, "Missing platform data\n"); in samsung_i2s_probe()
1412 return -EINVAL; in samsung_i2s_probe()
1414 priv->quirks = i2s_pdata->type.quirks; in samsung_i2s_probe()
1417 num_dais = (priv->quirks & QUIRK_SEC_DAI) ? 2 : 1; in samsung_i2s_probe()
1418 priv->pdev = pdev; in samsung_i2s_probe()
1419 priv->variant_regs = i2s_dai_data->i2s_variant_regs; in samsung_i2s_probe()
1425 pri_dai = &priv->dai[SAMSUNG_I2S_ID_PRIMARY - 1]; in samsung_i2s_probe()
1427 spin_lock_init(&priv->lock); in samsung_i2s_probe()
1428 spin_lock_init(&priv->pcm_lock); in samsung_i2s_probe()
1431 pri_dai->dma_playback.filter_data = i2s_pdata->dma_playback; in samsung_i2s_probe()
1432 pri_dai->dma_capture.filter_data = i2s_pdata->dma_capture; in samsung_i2s_probe()
1433 pri_dai->filter = i2s_pdata->dma_filter; in samsung_i2s_probe()
1435 idma_addr = i2s_pdata->type.idma_addr; in samsung_i2s_probe()
1437 if (of_property_read_u32(np, "samsung,idma-addr", in samsung_i2s_probe()
1439 if (priv->quirks & QUIRK_SUPPORTS_IDMA) { in samsung_i2s_probe()
1440 dev_info(&pdev->dev, "idma address is not"\ in samsung_i2s_probe()
1447 priv->addr = devm_ioremap_resource(&pdev->dev, res); in samsung_i2s_probe()
1448 if (IS_ERR(priv->addr)) in samsung_i2s_probe()
1449 return PTR_ERR(priv->addr); in samsung_i2s_probe()
1451 regs_base = res->start; in samsung_i2s_probe()
1453 priv->clk = devm_clk_get(&pdev->dev, "iis"); in samsung_i2s_probe()
1454 if (IS_ERR(priv->clk)) { in samsung_i2s_probe()
1455 dev_err(&pdev->dev, "Failed to get iis clock\n"); in samsung_i2s_probe()
1456 return PTR_ERR(priv->clk); in samsung_i2s_probe()
1459 ret = clk_prepare_enable(priv->clk); in samsung_i2s_probe()
1461 dev_err(&pdev->dev, "failed to enable clock: %d\n", ret); in samsung_i2s_probe()
1464 pri_dai->dma_playback.addr = regs_base + I2STXD; in samsung_i2s_probe()
1465 pri_dai->dma_capture.addr = regs_base + I2SRXD; in samsung_i2s_probe()
1466 pri_dai->dma_playback.chan_name = "tx"; in samsung_i2s_probe()
1467 pri_dai->dma_capture.chan_name = "rx"; in samsung_i2s_probe()
1468 pri_dai->dma_playback.addr_width = 4; in samsung_i2s_probe()
1469 pri_dai->dma_capture.addr_width = 4; in samsung_i2s_probe()
1470 pri_dai->priv = priv; in samsung_i2s_probe()
1472 if (priv->quirks & QUIRK_PRI_6CHAN) in samsung_i2s_probe()
1473 pri_dai->drv->playback.channels_max = 6; in samsung_i2s_probe()
1475 ret = samsung_asoc_dma_platform_register(&pdev->dev, pri_dai->filter, in samsung_i2s_probe()
1480 if (priv->quirks & QUIRK_SEC_DAI) { in samsung_i2s_probe()
1481 sec_dai = &priv->dai[SAMSUNG_I2S_ID_SECONDARY - 1]; in samsung_i2s_probe()
1483 sec_dai->dma_playback.addr = regs_base + I2STXDS; in samsung_i2s_probe()
1484 sec_dai->dma_playback.chan_name = "tx-sec"; in samsung_i2s_probe()
1487 sec_dai->dma_playback.filter_data = i2s_pdata->dma_play_sec; in samsung_i2s_probe()
1488 sec_dai->filter = i2s_pdata->dma_filter; in samsung_i2s_probe()
1491 sec_dai->dma_playback.addr_width = 4; in samsung_i2s_probe()
1492 sec_dai->idma_playback.addr = idma_addr; in samsung_i2s_probe()
1493 sec_dai->pri_dai = pri_dai; in samsung_i2s_probe()
1494 sec_dai->priv = priv; in samsung_i2s_probe()
1495 pri_dai->sec_dai = sec_dai; in samsung_i2s_probe()
1501 ret = samsung_asoc_dma_platform_register(&priv->pdev_sec->dev, in samsung_i2s_probe()
1502 sec_dai->filter, "tx-sec", NULL, in samsung_i2s_probe()
1503 &pdev->dev); in samsung_i2s_probe()
1509 if (i2s_pdata && i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) { in samsung_i2s_probe()
1510 dev_err(&pdev->dev, "Unable to configure gpio\n"); in samsung_i2s_probe()
1511 ret = -EINVAL; in samsung_i2s_probe()
1515 dev_set_drvdata(&pdev->dev, priv); in samsung_i2s_probe()
1517 ret = devm_snd_soc_register_component(&pdev->dev, in samsung_i2s_probe()
1519 priv->dai_drv, num_dais); in samsung_i2s_probe()
1523 pm_runtime_set_active(&pdev->dev); in samsung_i2s_probe()
1524 pm_runtime_enable(&pdev->dev); in samsung_i2s_probe()
1530 priv->op_clk = clk_get_parent(priv->clk_table[CLK_I2S_RCLK_SRC]); in samsung_i2s_probe()
1535 pm_runtime_disable(&pdev->dev); in samsung_i2s_probe()
1539 clk_disable_unprepare(priv->clk); in samsung_i2s_probe()
1545 struct samsung_i2s_priv *priv = dev_get_drvdata(&pdev->dev); in samsung_i2s_remove()
1551 pm_runtime_get_sync(&pdev->dev); in samsung_i2s_remove()
1552 pm_runtime_disable(&pdev->dev); in samsung_i2s_remove()
1556 clk_disable_unprepare(priv->clk); in samsung_i2s_remove()
1558 pm_runtime_put_noidle(&pdev->dev); in samsung_i2s_remove()
1654 .name = "samsung-i2s",
1664 .compatible = "samsung,s3c6410-i2s",
1667 .compatible = "samsung,s5pv210-i2s",
1670 .compatible = "samsung,exynos5420-i2s",
1673 .compatible = "samsung,exynos7-i2s",
1676 .compatible = "samsung,exynos7-i2s1",
1696 .name = "samsung-i2s",
1707 MODULE_ALIAS("platform:samsung-i2s");