18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Copyright (C) 2018 Texas Instruments Incorporated - https://www.ti.com 48c2ecf20Sopenharmony_ci * Author: Peter Ujfalusi <peter.ujfalusi@ti.com> 58c2ecf20Sopenharmony_ci */ 68c2ecf20Sopenharmony_ci 78c2ecf20Sopenharmony_ci#include <linux/device.h> 88c2ecf20Sopenharmony_ci#include <linux/module.h> 98c2ecf20Sopenharmony_ci#include <sound/core.h> 108c2ecf20Sopenharmony_ci#include <sound/pcm.h> 118c2ecf20Sopenharmony_ci#include <sound/pcm_params.h> 128c2ecf20Sopenharmony_ci#include <sound/soc.h> 138c2ecf20Sopenharmony_ci#include <sound/dmaengine_pcm.h> 148c2ecf20Sopenharmony_ci 158c2ecf20Sopenharmony_ci#include "sdma-pcm.h" 168c2ecf20Sopenharmony_ci 178c2ecf20Sopenharmony_cistatic const struct snd_pcm_hardware sdma_pcm_hardware = { 188c2ecf20Sopenharmony_ci .info = SNDRV_PCM_INFO_MMAP | 198c2ecf20Sopenharmony_ci SNDRV_PCM_INFO_MMAP_VALID | 208c2ecf20Sopenharmony_ci SNDRV_PCM_INFO_PAUSE | SNDRV_PCM_INFO_RESUME | 218c2ecf20Sopenharmony_ci SNDRV_PCM_INFO_NO_PERIOD_WAKEUP | 228c2ecf20Sopenharmony_ci SNDRV_PCM_INFO_INTERLEAVED, 238c2ecf20Sopenharmony_ci .period_bytes_min = 32, 248c2ecf20Sopenharmony_ci .period_bytes_max = 64 * 1024, 258c2ecf20Sopenharmony_ci .buffer_bytes_max = 128 * 1024, 268c2ecf20Sopenharmony_ci .periods_min = 2, 278c2ecf20Sopenharmony_ci .periods_max = 255, 288c2ecf20Sopenharmony_ci}; 298c2ecf20Sopenharmony_ci 308c2ecf20Sopenharmony_cistatic const struct snd_dmaengine_pcm_config sdma_dmaengine_pcm_config = { 318c2ecf20Sopenharmony_ci .pcm_hardware = &sdma_pcm_hardware, 328c2ecf20Sopenharmony_ci .prepare_slave_config = snd_dmaengine_pcm_prepare_slave_config, 338c2ecf20Sopenharmony_ci .prealloc_buffer_size = 128 * 1024, 348c2ecf20Sopenharmony_ci}; 358c2ecf20Sopenharmony_ci 368c2ecf20Sopenharmony_ciint sdma_pcm_platform_register(struct device *dev, 378c2ecf20Sopenharmony_ci char *txdmachan, char *rxdmachan) 388c2ecf20Sopenharmony_ci{ 398c2ecf20Sopenharmony_ci struct snd_dmaengine_pcm_config *config; 408c2ecf20Sopenharmony_ci unsigned int flags = 0; 418c2ecf20Sopenharmony_ci 428c2ecf20Sopenharmony_ci /* Standard names for the directions: 'tx' and 'rx' */ 438c2ecf20Sopenharmony_ci if (!txdmachan && !rxdmachan) 448c2ecf20Sopenharmony_ci return devm_snd_dmaengine_pcm_register(dev, 458c2ecf20Sopenharmony_ci &sdma_dmaengine_pcm_config, 0); 468c2ecf20Sopenharmony_ci 478c2ecf20Sopenharmony_ci config = devm_kzalloc(dev, sizeof(*config), GFP_KERNEL); 488c2ecf20Sopenharmony_ci if (!config) 498c2ecf20Sopenharmony_ci return -ENOMEM; 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_ci *config = sdma_dmaengine_pcm_config; 528c2ecf20Sopenharmony_ci 538c2ecf20Sopenharmony_ci if (!txdmachan || !rxdmachan) { 548c2ecf20Sopenharmony_ci /* One direction only PCM */ 558c2ecf20Sopenharmony_ci flags |= SND_DMAENGINE_PCM_FLAG_HALF_DUPLEX; 568c2ecf20Sopenharmony_ci if (!txdmachan) { 578c2ecf20Sopenharmony_ci txdmachan = rxdmachan; 588c2ecf20Sopenharmony_ci rxdmachan = NULL; 598c2ecf20Sopenharmony_ci } 608c2ecf20Sopenharmony_ci } 618c2ecf20Sopenharmony_ci 628c2ecf20Sopenharmony_ci config->chan_names[0] = txdmachan; 638c2ecf20Sopenharmony_ci config->chan_names[1] = rxdmachan; 648c2ecf20Sopenharmony_ci 658c2ecf20Sopenharmony_ci return devm_snd_dmaengine_pcm_register(dev, config, flags); 668c2ecf20Sopenharmony_ci} 678c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(sdma_pcm_platform_register); 688c2ecf20Sopenharmony_ci 698c2ecf20Sopenharmony_ciMODULE_AUTHOR("Peter Ujfalusi <peter.ujfalusi@ti.com>"); 708c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("sDMA PCM ASoC platform driver"); 718c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2"); 72