Lines Matching full:spdifrx
208 * struct stm32_spdifrx_data - private data of SPDIFRX
211 * @regmap: SPDIFRX register map pointer
212 * @regmap_conf: SPDIFRX register map configuration pointer
214 * @kclk: kernel clock feeding the SPDIFRX clock generator
221 * @phys_addr: SPDIFRX registers physical base address
226 * @irq: SPDIFRX interrupt line
253 struct stm32_spdifrx_data *spdifrx = (struct stm32_spdifrx_data *)data; in stm32_spdifrx_dma_complete() local
254 struct platform_device *pdev = spdifrx->pdev; in stm32_spdifrx_dma_complete()
255 u32 *p_start = (u32 *)spdifrx->dmab->area; in stm32_spdifrx_dma_complete()
258 u16 *ub_ptr = (short *)spdifrx->ub; in stm32_spdifrx_dma_complete()
261 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_CR, in stm32_spdifrx_dma_complete()
265 if (!spdifrx->dmab->area) in stm32_spdifrx_dma_complete()
280 spdifrx->cs[i] = (unsigned char)SPDIFRX_CSR_CSGET(*ptr); in stm32_spdifrx_dma_complete()
289 complete(&spdifrx->cs_completion); in stm32_spdifrx_dma_complete()
292 static int stm32_spdifrx_dma_ctrl_start(struct stm32_spdifrx_data *spdifrx) in stm32_spdifrx_dma_ctrl_start() argument
297 spdifrx->desc = dmaengine_prep_slave_single(spdifrx->ctrl_chan, in stm32_spdifrx_dma_ctrl_start()
298 spdifrx->dmab->addr, in stm32_spdifrx_dma_ctrl_start()
302 if (!spdifrx->desc) in stm32_spdifrx_dma_ctrl_start()
305 spdifrx->desc->callback = stm32_spdifrx_dma_complete; in stm32_spdifrx_dma_ctrl_start()
306 spdifrx->desc->callback_param = spdifrx; in stm32_spdifrx_dma_ctrl_start()
307 cookie = dmaengine_submit(spdifrx->desc); in stm32_spdifrx_dma_ctrl_start()
312 dma_async_issue_pending(spdifrx->ctrl_chan); in stm32_spdifrx_dma_ctrl_start()
317 static void stm32_spdifrx_dma_ctrl_stop(struct stm32_spdifrx_data *spdifrx) in stm32_spdifrx_dma_ctrl_stop() argument
319 dmaengine_terminate_async(spdifrx->ctrl_chan); in stm32_spdifrx_dma_ctrl_stop()
322 static int stm32_spdifrx_start_sync(struct stm32_spdifrx_data *spdifrx) in stm32_spdifrx_start_sync() argument
329 ret = regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_IMR, imr, imr); in stm32_spdifrx_start_sync()
333 spin_lock_irqsave(&spdifrx->lock, flags); in stm32_spdifrx_start_sync()
335 spdifrx->refcount++; in stm32_spdifrx_start_sync()
337 regmap_read(spdifrx->regmap, STM32_SPDIFRX_CR, &cr); in stm32_spdifrx_start_sync()
341 * Start sync if SPDIFRX is still in idle state. in stm32_spdifrx_start_sync()
342 * SPDIFRX reception enabled when sync done in stm32_spdifrx_start_sync()
344 dev_dbg(&spdifrx->pdev->dev, "start synchronization\n"); in stm32_spdifrx_start_sync()
347 * SPDIFRX configuration: in stm32_spdifrx_start_sync()
361 ret = regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_CR, in stm32_spdifrx_start_sync()
364 dev_err(&spdifrx->pdev->dev, in stm32_spdifrx_start_sync()
368 spin_unlock_irqrestore(&spdifrx->lock, flags); in stm32_spdifrx_start_sync()
373 static void stm32_spdifrx_stop(struct stm32_spdifrx_data *spdifrx) in stm32_spdifrx_stop() argument
378 spin_lock_irqsave(&spdifrx->lock, flags); in stm32_spdifrx_stop()
380 if (--spdifrx->refcount) { in stm32_spdifrx_stop()
381 spin_unlock_irqrestore(&spdifrx->lock, flags); in stm32_spdifrx_stop()
388 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_CR, cr_mask, cr); in stm32_spdifrx_stop()
390 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_IMR, in stm32_spdifrx_stop()
393 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_IFCR, in stm32_spdifrx_stop()
397 regmap_read(spdifrx->regmap, STM32_SPDIFRX_DR, ®); in stm32_spdifrx_stop()
398 regmap_read(spdifrx->regmap, STM32_SPDIFRX_CSR, ®); in stm32_spdifrx_stop()
400 spin_unlock_irqrestore(&spdifrx->lock, flags); in stm32_spdifrx_stop()
404 struct stm32_spdifrx_data *spdifrx) in stm32_spdifrx_dma_ctrl_register() argument
408 spdifrx->ctrl_chan = dma_request_chan(dev, "rx-ctrl"); in stm32_spdifrx_dma_ctrl_register()
409 if (IS_ERR(spdifrx->ctrl_chan)) in stm32_spdifrx_dma_ctrl_register()
410 return dev_err_probe(dev, PTR_ERR(spdifrx->ctrl_chan), in stm32_spdifrx_dma_ctrl_register()
413 spdifrx->dmab = devm_kzalloc(dev, sizeof(struct snd_dma_buffer), in stm32_spdifrx_dma_ctrl_register()
415 if (!spdifrx->dmab) in stm32_spdifrx_dma_ctrl_register()
418 spdifrx->dmab->dev.type = SNDRV_DMA_TYPE_DEV_IRAM; in stm32_spdifrx_dma_ctrl_register()
419 spdifrx->dmab->dev.dev = dev; in stm32_spdifrx_dma_ctrl_register()
420 ret = snd_dma_alloc_pages(spdifrx->dmab->dev.type, dev, in stm32_spdifrx_dma_ctrl_register()
421 SPDIFRX_CSR_BUF_LENGTH, spdifrx->dmab); in stm32_spdifrx_dma_ctrl_register()
427 spdifrx->slave_config.direction = DMA_DEV_TO_MEM; in stm32_spdifrx_dma_ctrl_register()
428 spdifrx->slave_config.src_addr = (dma_addr_t)(spdifrx->phys_addr + in stm32_spdifrx_dma_ctrl_register()
430 spdifrx->slave_config.dst_addr = spdifrx->dmab->addr; in stm32_spdifrx_dma_ctrl_register()
431 spdifrx->slave_config.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES; in stm32_spdifrx_dma_ctrl_register()
432 spdifrx->slave_config.src_maxburst = 1; in stm32_spdifrx_dma_ctrl_register()
434 ret = dmaengine_slave_config(spdifrx->ctrl_chan, in stm32_spdifrx_dma_ctrl_register()
435 &spdifrx->slave_config); in stm32_spdifrx_dma_ctrl_register()
438 spdifrx->ctrl_chan = NULL; in stm32_spdifrx_dma_ctrl_register()
479 static int stm32_spdifrx_get_ctrl_data(struct stm32_spdifrx_data *spdifrx) in stm32_spdifrx_get_ctrl_data() argument
483 memset(spdifrx->cs, 0, SPDIFRX_CS_BYTES_NB); in stm32_spdifrx_get_ctrl_data()
484 memset(spdifrx->ub, 0, SPDIFRX_UB_BYTES_NB); in stm32_spdifrx_get_ctrl_data()
486 ret = stm32_spdifrx_dma_ctrl_start(spdifrx); in stm32_spdifrx_get_ctrl_data()
490 ret = clk_prepare_enable(spdifrx->kclk); in stm32_spdifrx_get_ctrl_data()
492 dev_err(&spdifrx->pdev->dev, "Enable kclk failed: %d\n", ret); in stm32_spdifrx_get_ctrl_data()
496 ret = regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_CR, in stm32_spdifrx_get_ctrl_data()
501 ret = stm32_spdifrx_start_sync(spdifrx); in stm32_spdifrx_get_ctrl_data()
505 if (wait_for_completion_interruptible_timeout(&spdifrx->cs_completion, in stm32_spdifrx_get_ctrl_data()
508 dev_dbg(&spdifrx->pdev->dev, "Failed to get control data\n"); in stm32_spdifrx_get_ctrl_data()
512 stm32_spdifrx_stop(spdifrx); in stm32_spdifrx_get_ctrl_data()
513 stm32_spdifrx_dma_ctrl_stop(spdifrx); in stm32_spdifrx_get_ctrl_data()
516 clk_disable_unprepare(spdifrx->kclk); in stm32_spdifrx_get_ctrl_data()
525 struct stm32_spdifrx_data *spdifrx = snd_soc_dai_get_drvdata(cpu_dai); in stm32_spdifrx_capture_get() local
527 stm32_spdifrx_get_ctrl_data(spdifrx); in stm32_spdifrx_capture_get()
529 ucontrol->value.iec958.status[0] = spdifrx->cs[0]; in stm32_spdifrx_capture_get()
530 ucontrol->value.iec958.status[1] = spdifrx->cs[1]; in stm32_spdifrx_capture_get()
531 ucontrol->value.iec958.status[2] = spdifrx->cs[2]; in stm32_spdifrx_capture_get()
532 ucontrol->value.iec958.status[3] = spdifrx->cs[3]; in stm32_spdifrx_capture_get()
533 ucontrol->value.iec958.status[4] = spdifrx->cs[4]; in stm32_spdifrx_capture_get()
542 struct stm32_spdifrx_data *spdifrx = snd_soc_dai_get_drvdata(cpu_dai); in stm32_spdif_user_bits_get() local
544 stm32_spdifrx_get_ctrl_data(spdifrx); in stm32_spdif_user_bits_get()
546 ucontrol->value.iec958.status[0] = spdifrx->ub[0]; in stm32_spdif_user_bits_get()
547 ucontrol->value.iec958.status[1] = spdifrx->ub[1]; in stm32_spdif_user_bits_get()
548 ucontrol->value.iec958.status[2] = spdifrx->ub[2]; in stm32_spdif_user_bits_get()
549 ucontrol->value.iec958.status[3] = spdifrx->ub[3]; in stm32_spdif_user_bits_get()
550 ucontrol->value.iec958.status[4] = spdifrx->ub[4]; in stm32_spdif_user_bits_get()
577 SOC_ENUM("SPDIFRX input", ctrl_enum_input),
578 SOC_ENUM("SPDIFRX CS channel", ctrl_enum_cs_channel),
597 struct stm32_spdifrx_data *spdifrx = dev_get_drvdata(cpu_dai->dev); in stm32_spdifrx_dai_probe() local
599 spdifrx->dma_params.addr = (dma_addr_t)(spdifrx->phys_addr + in stm32_spdifrx_dai_probe()
601 spdifrx->dma_params.maxburst = 1; in stm32_spdifrx_dai_probe()
603 snd_soc_dai_init_dma_data(cpu_dai, NULL, &spdifrx->dma_params); in stm32_spdifrx_dai_probe()
667 struct stm32_spdifrx_data *spdifrx = (struct stm32_spdifrx_data *)devid; in stm32_spdifrx_isr() local
668 struct platform_device *pdev = spdifrx->pdev; in stm32_spdifrx_isr()
673 regmap_read(spdifrx->regmap, STM32_SPDIFRX_SR, &sr); in stm32_spdifrx_isr()
674 regmap_read(spdifrx->regmap, STM32_SPDIFRX_IMR, &imr); in stm32_spdifrx_isr()
689 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_IFCR, in stm32_spdifrx_isr()
708 /* Enable spdifrx */ in stm32_spdifrx_isr()
710 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_CR, in stm32_spdifrx_isr()
730 regmap_read(spdifrx->regmap, STM32_SPDIFRX_CR, &cr); in stm32_spdifrx_isr()
734 /* SPDIFRX is in STATE_STOP. Disable SPDIFRX to clear errors */ in stm32_spdifrx_isr()
736 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_CR, in stm32_spdifrx_isr()
739 /* If SPDIFRX was in STATE_SYNC, retry synchro */ in stm32_spdifrx_isr()
742 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_CR, in stm32_spdifrx_isr()
747 spin_lock(&spdifrx->irq_lock); in stm32_spdifrx_isr()
748 if (spdifrx->substream) in stm32_spdifrx_isr()
749 snd_pcm_stop(spdifrx->substream, in stm32_spdifrx_isr()
751 spin_unlock(&spdifrx->irq_lock); in stm32_spdifrx_isr()
756 spin_lock(&spdifrx->irq_lock); in stm32_spdifrx_isr()
757 if (err_xrun && spdifrx->substream) in stm32_spdifrx_isr()
758 snd_pcm_stop_xrun(spdifrx->substream); in stm32_spdifrx_isr()
759 spin_unlock(&spdifrx->irq_lock); in stm32_spdifrx_isr()
767 struct stm32_spdifrx_data *spdifrx = snd_soc_dai_get_drvdata(cpu_dai); in stm32_spdifrx_startup() local
771 spin_lock_irqsave(&spdifrx->irq_lock, flags); in stm32_spdifrx_startup()
772 spdifrx->substream = substream; in stm32_spdifrx_startup()
773 spin_unlock_irqrestore(&spdifrx->irq_lock, flags); in stm32_spdifrx_startup()
775 ret = clk_prepare_enable(spdifrx->kclk); in stm32_spdifrx_startup()
777 dev_err(&spdifrx->pdev->dev, "Enable kclk failed: %d\n", ret); in stm32_spdifrx_startup()
786 struct stm32_spdifrx_data *spdifrx = snd_soc_dai_get_drvdata(cpu_dai); in stm32_spdifrx_hw_params() local
798 dev_err(&spdifrx->pdev->dev, "Unexpected data format\n"); in stm32_spdifrx_hw_params()
807 spdifrx->dma_params.addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES; in stm32_spdifrx_hw_params()
808 snd_soc_dai_init_dma_data(cpu_dai, NULL, &spdifrx->dma_params); in stm32_spdifrx_hw_params()
810 return regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_CR, in stm32_spdifrx_hw_params()
818 struct stm32_spdifrx_data *spdifrx = snd_soc_dai_get_drvdata(cpu_dai); in stm32_spdifrx_trigger() local
825 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_IMR, in stm32_spdifrx_trigger()
828 regmap_update_bits(spdifrx->regmap, STM32_SPDIFRX_CR, in stm32_spdifrx_trigger()
831 ret = stm32_spdifrx_start_sync(spdifrx); in stm32_spdifrx_trigger()
836 stm32_spdifrx_stop(spdifrx); in stm32_spdifrx_trigger()
848 struct stm32_spdifrx_data *spdifrx = snd_soc_dai_get_drvdata(cpu_dai); in stm32_spdifrx_shutdown() local
851 spin_lock_irqsave(&spdifrx->irq_lock, flags); in stm32_spdifrx_shutdown()
852 spdifrx->substream = NULL; in stm32_spdifrx_shutdown()
853 spin_unlock_irqrestore(&spdifrx->irq_lock, flags); in stm32_spdifrx_shutdown()
855 clk_disable_unprepare(spdifrx->kclk); in stm32_spdifrx_shutdown()
890 .name = "stm32-spdifrx",
901 .compatible = "st,stm32h7-spdifrx",
908 struct stm32_spdifrx_data *spdifrx) in stm32_spdifrx_parse_of() argument
916 spdifrx->regmap_conf = device_get_match_data(&pdev->dev); in stm32_spdifrx_parse_of()
917 if (!spdifrx->regmap_conf) in stm32_spdifrx_parse_of()
920 spdifrx->base = devm_platform_get_and_ioremap_resource(pdev, 0, &res); in stm32_spdifrx_parse_of()
921 if (IS_ERR(spdifrx->base)) in stm32_spdifrx_parse_of()
922 return PTR_ERR(spdifrx->base); in stm32_spdifrx_parse_of()
924 spdifrx->phys_addr = res->start; in stm32_spdifrx_parse_of()
926 spdifrx->kclk = devm_clk_get(&pdev->dev, "kclk"); in stm32_spdifrx_parse_of()
927 if (IS_ERR(spdifrx->kclk)) in stm32_spdifrx_parse_of()
928 return dev_err_probe(&pdev->dev, PTR_ERR(spdifrx->kclk), in stm32_spdifrx_parse_of()
931 spdifrx->irq = platform_get_irq(pdev, 0); in stm32_spdifrx_parse_of()
932 if (spdifrx->irq < 0) in stm32_spdifrx_parse_of()
933 return spdifrx->irq; in stm32_spdifrx_parse_of()
940 struct stm32_spdifrx_data *spdifrx = platform_get_drvdata(pdev); in stm32_spdifrx_remove() local
942 if (!IS_ERR(spdifrx->ctrl_chan)) in stm32_spdifrx_remove()
943 dma_release_channel(spdifrx->ctrl_chan); in stm32_spdifrx_remove()
945 if (spdifrx->dmab) in stm32_spdifrx_remove()
946 snd_dma_free_pages(spdifrx->dmab); in stm32_spdifrx_remove()
955 struct stm32_spdifrx_data *spdifrx; in stm32_spdifrx_probe() local
961 spdifrx = devm_kzalloc(&pdev->dev, sizeof(*spdifrx), GFP_KERNEL); in stm32_spdifrx_probe()
962 if (!spdifrx) in stm32_spdifrx_probe()
965 spdifrx->pdev = pdev; in stm32_spdifrx_probe()
966 init_completion(&spdifrx->cs_completion); in stm32_spdifrx_probe()
967 spin_lock_init(&spdifrx->lock); in stm32_spdifrx_probe()
968 spin_lock_init(&spdifrx->irq_lock); in stm32_spdifrx_probe()
970 platform_set_drvdata(pdev, spdifrx); in stm32_spdifrx_probe()
972 ret = stm32_spdifrx_parse_of(pdev, spdifrx); in stm32_spdifrx_probe()
976 spdifrx->regmap = devm_regmap_init_mmio_clk(&pdev->dev, "kclk", in stm32_spdifrx_probe()
977 spdifrx->base, in stm32_spdifrx_probe()
978 spdifrx->regmap_conf); in stm32_spdifrx_probe()
979 if (IS_ERR(spdifrx->regmap)) in stm32_spdifrx_probe()
980 return dev_err_probe(&pdev->dev, PTR_ERR(spdifrx->regmap), in stm32_spdifrx_probe()
983 ret = devm_request_irq(&pdev->dev, spdifrx->irq, stm32_spdifrx_isr, 0, in stm32_spdifrx_probe()
984 dev_name(&pdev->dev), spdifrx); in stm32_spdifrx_probe()
1013 ret = stm32_spdifrx_dma_ctrl_register(&pdev->dev, spdifrx); in stm32_spdifrx_probe()
1017 ret = regmap_read(spdifrx->regmap, STM32_SPDIFRX_IDR, &idr); in stm32_spdifrx_probe()
1022 ret = regmap_read(spdifrx->regmap, STM32_SPDIFRX_VERR, &ver); in stm32_spdifrx_probe()
1026 dev_dbg(&pdev->dev, "SPDIFRX version: %lu.%lu registered\n", in stm32_spdifrx_probe()
1046 struct stm32_spdifrx_data *spdifrx = dev_get_drvdata(dev); in stm32_spdifrx_suspend() local
1048 regcache_cache_only(spdifrx->regmap, true); in stm32_spdifrx_suspend()
1049 regcache_mark_dirty(spdifrx->regmap); in stm32_spdifrx_suspend()
1056 struct stm32_spdifrx_data *spdifrx = dev_get_drvdata(dev); in stm32_spdifrx_resume() local
1058 regcache_cache_only(spdifrx->regmap, false); in stm32_spdifrx_resume()
1060 return regcache_sync(spdifrx->regmap); in stm32_spdifrx_resume()
1070 .name = "st,stm32-spdifrx",
1080 MODULE_DESCRIPTION("STM32 Soc spdifrx Interface");
1082 MODULE_ALIAS("platform:stm32-spdifrx");