162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci// 362306a36Sopenharmony_ci// Loongson ALSA SoC Platform (DMA) driver 462306a36Sopenharmony_ci// 562306a36Sopenharmony_ci// Copyright (C) 2023 Loongson Technology Corporation Limited 662306a36Sopenharmony_ci// Author: Yingkun Meng <mengyingkun@loongson.cn> 762306a36Sopenharmony_ci// 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci#include <linux/module.h> 1062306a36Sopenharmony_ci#include <linux/io-64-nonatomic-lo-hi.h> 1162306a36Sopenharmony_ci#include <linux/delay.h> 1262306a36Sopenharmony_ci#include <linux/pm_runtime.h> 1362306a36Sopenharmony_ci#include <linux/dma-mapping.h> 1462306a36Sopenharmony_ci#include <sound/soc.h> 1562306a36Sopenharmony_ci#include <sound/pcm.h> 1662306a36Sopenharmony_ci#include <sound/pcm_params.h> 1762306a36Sopenharmony_ci#include "loongson_i2s.h" 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci/* DMA dma_order Register */ 2062306a36Sopenharmony_ci#define DMA_ORDER_STOP (1 << 4) /* DMA stop */ 2162306a36Sopenharmony_ci#define DMA_ORDER_START (1 << 3) /* DMA start */ 2262306a36Sopenharmony_ci#define DMA_ORDER_ASK_VALID (1 << 2) /* DMA ask valid flag */ 2362306a36Sopenharmony_ci#define DMA_ORDER_AXI_UNCO (1 << 1) /* Uncache access */ 2462306a36Sopenharmony_ci#define DMA_ORDER_ADDR_64 (1 << 0) /* 64bits address support */ 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_ci#define DMA_ORDER_ASK_MASK (~0x1fUL) /* Ask addr mask */ 2762306a36Sopenharmony_ci#define DMA_ORDER_CTRL_MASK (0x0fUL) /* Control mask */ 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci/* 3062306a36Sopenharmony_ci * DMA registers descriptor. 3162306a36Sopenharmony_ci */ 3262306a36Sopenharmony_cistruct loongson_dma_desc { 3362306a36Sopenharmony_ci u32 order; /* Next descriptor address register */ 3462306a36Sopenharmony_ci u32 saddr; /* Source address register */ 3562306a36Sopenharmony_ci u32 daddr; /* Device address register */ 3662306a36Sopenharmony_ci u32 length; /* Total length register */ 3762306a36Sopenharmony_ci u32 step_length; /* Memory stride register */ 3862306a36Sopenharmony_ci u32 step_times; /* Repeat time register */ 3962306a36Sopenharmony_ci u32 cmd; /* Command register */ 4062306a36Sopenharmony_ci u32 stats; /* Status register */ 4162306a36Sopenharmony_ci u32 order_hi; /* Next descriptor high address register */ 4262306a36Sopenharmony_ci u32 saddr_hi; /* High source address register */ 4362306a36Sopenharmony_ci u32 res[6]; /* Reserved */ 4462306a36Sopenharmony_ci} __packed; 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_cistruct loongson_runtime_data { 4762306a36Sopenharmony_ci struct loongson_dma_data *dma_data; 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_ci struct loongson_dma_desc *dma_desc_arr; 5062306a36Sopenharmony_ci dma_addr_t dma_desc_arr_phy; 5162306a36Sopenharmony_ci int dma_desc_arr_size; 5262306a36Sopenharmony_ci 5362306a36Sopenharmony_ci struct loongson_dma_desc *dma_pos_desc; 5462306a36Sopenharmony_ci dma_addr_t dma_pos_desc_phy; 5562306a36Sopenharmony_ci}; 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_cistatic const struct snd_pcm_hardware ls_pcm_hardware = { 5862306a36Sopenharmony_ci .info = SNDRV_PCM_INFO_MMAP | 5962306a36Sopenharmony_ci SNDRV_PCM_INFO_INTERLEAVED | 6062306a36Sopenharmony_ci SNDRV_PCM_INFO_MMAP_VALID | 6162306a36Sopenharmony_ci SNDRV_PCM_INFO_RESUME | 6262306a36Sopenharmony_ci SNDRV_PCM_INFO_PAUSE, 6362306a36Sopenharmony_ci .formats = (SNDRV_PCM_FMTBIT_S8 | 6462306a36Sopenharmony_ci SNDRV_PCM_FMTBIT_S16_LE | 6562306a36Sopenharmony_ci SNDRV_PCM_FMTBIT_S20_3LE | 6662306a36Sopenharmony_ci SNDRV_PCM_FMTBIT_S24_LE), 6762306a36Sopenharmony_ci .period_bytes_min = 128, 6862306a36Sopenharmony_ci .period_bytes_max = 128 * 1024, 6962306a36Sopenharmony_ci .periods_min = 1, 7062306a36Sopenharmony_ci .periods_max = PAGE_SIZE / sizeof(struct loongson_dma_desc), 7162306a36Sopenharmony_ci .buffer_bytes_max = 1024 * 1024, 7262306a36Sopenharmony_ci}; 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_cistatic struct 7562306a36Sopenharmony_ciloongson_dma_desc *dma_desc_save(struct loongson_runtime_data *prtd) 7662306a36Sopenharmony_ci{ 7762306a36Sopenharmony_ci void __iomem *order_reg = prtd->dma_data->order_addr; 7862306a36Sopenharmony_ci u64 val; 7962306a36Sopenharmony_ci 8062306a36Sopenharmony_ci val = (u64)prtd->dma_pos_desc_phy & DMA_ORDER_ASK_MASK; 8162306a36Sopenharmony_ci val |= (readq(order_reg) & DMA_ORDER_CTRL_MASK); 8262306a36Sopenharmony_ci val |= DMA_ORDER_ASK_VALID; 8362306a36Sopenharmony_ci writeq(val, order_reg); 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_ci while (readl(order_reg) & DMA_ORDER_ASK_VALID) 8662306a36Sopenharmony_ci udelay(2); 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci return prtd->dma_pos_desc; 8962306a36Sopenharmony_ci} 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_cistatic int loongson_pcm_trigger(struct snd_soc_component *component, 9262306a36Sopenharmony_ci struct snd_pcm_substream *substream, int cmd) 9362306a36Sopenharmony_ci{ 9462306a36Sopenharmony_ci struct loongson_runtime_data *prtd = substream->runtime->private_data; 9562306a36Sopenharmony_ci struct device *dev = substream->pcm->card->dev; 9662306a36Sopenharmony_ci void __iomem *order_reg = prtd->dma_data->order_addr; 9762306a36Sopenharmony_ci u64 val; 9862306a36Sopenharmony_ci int ret = 0; 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_ci switch (cmd) { 10162306a36Sopenharmony_ci case SNDRV_PCM_TRIGGER_START: 10262306a36Sopenharmony_ci case SNDRV_PCM_TRIGGER_RESUME: 10362306a36Sopenharmony_ci case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: 10462306a36Sopenharmony_ci val = prtd->dma_pos_desc_phy & DMA_ORDER_ASK_MASK; 10562306a36Sopenharmony_ci if (dev->coherent_dma_mask == DMA_BIT_MASK(64)) 10662306a36Sopenharmony_ci val |= DMA_ORDER_ADDR_64; 10762306a36Sopenharmony_ci else 10862306a36Sopenharmony_ci val &= ~DMA_ORDER_ADDR_64; 10962306a36Sopenharmony_ci val |= (readq(order_reg) & DMA_ORDER_CTRL_MASK); 11062306a36Sopenharmony_ci val |= DMA_ORDER_START; 11162306a36Sopenharmony_ci writeq(val, order_reg); 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_ci while ((readl(order_reg) & DMA_ORDER_START)) 11462306a36Sopenharmony_ci udelay(2); 11562306a36Sopenharmony_ci break; 11662306a36Sopenharmony_ci case SNDRV_PCM_TRIGGER_STOP: 11762306a36Sopenharmony_ci case SNDRV_PCM_TRIGGER_SUSPEND: 11862306a36Sopenharmony_ci case SNDRV_PCM_TRIGGER_PAUSE_PUSH: 11962306a36Sopenharmony_ci dma_desc_save(prtd); 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ci /* dma stop */ 12262306a36Sopenharmony_ci val = readq(order_reg) | DMA_ORDER_STOP; 12362306a36Sopenharmony_ci writeq(val, order_reg); 12462306a36Sopenharmony_ci udelay(1000); 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_ci break; 12762306a36Sopenharmony_ci default: 12862306a36Sopenharmony_ci dev_err(dev, "Invalid pcm trigger operation\n"); 12962306a36Sopenharmony_ci return -EINVAL; 13062306a36Sopenharmony_ci } 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci return ret; 13362306a36Sopenharmony_ci} 13462306a36Sopenharmony_ci 13562306a36Sopenharmony_cistatic int loongson_pcm_hw_params(struct snd_soc_component *component, 13662306a36Sopenharmony_ci struct snd_pcm_substream *substream, 13762306a36Sopenharmony_ci struct snd_pcm_hw_params *params) 13862306a36Sopenharmony_ci{ 13962306a36Sopenharmony_ci struct snd_pcm_runtime *runtime = substream->runtime; 14062306a36Sopenharmony_ci struct device *dev = substream->pcm->card->dev; 14162306a36Sopenharmony_ci struct loongson_runtime_data *prtd = runtime->private_data; 14262306a36Sopenharmony_ci size_t buf_len = params_buffer_bytes(params); 14362306a36Sopenharmony_ci size_t period_len = params_period_bytes(params); 14462306a36Sopenharmony_ci dma_addr_t order_addr, mem_addr; 14562306a36Sopenharmony_ci struct loongson_dma_desc *desc; 14662306a36Sopenharmony_ci u32 num_periods; 14762306a36Sopenharmony_ci int i; 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_ci if (buf_len % period_len) { 15062306a36Sopenharmony_ci dev_err(dev, "buf len not multiply of period len\n"); 15162306a36Sopenharmony_ci return -EINVAL; 15262306a36Sopenharmony_ci } 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_ci num_periods = buf_len / period_len; 15562306a36Sopenharmony_ci if (!num_periods || num_periods > prtd->dma_desc_arr_size) { 15662306a36Sopenharmony_ci dev_err(dev, "dma data too small or too big\n"); 15762306a36Sopenharmony_ci return -EINVAL; 15862306a36Sopenharmony_ci } 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_ci snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer); 16162306a36Sopenharmony_ci runtime->dma_bytes = buf_len; 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci /* initialize dma descriptor array */ 16462306a36Sopenharmony_ci mem_addr = runtime->dma_addr; 16562306a36Sopenharmony_ci order_addr = prtd->dma_desc_arr_phy; 16662306a36Sopenharmony_ci for (i = 0; i < num_periods; i++) { 16762306a36Sopenharmony_ci desc = &prtd->dma_desc_arr[i]; 16862306a36Sopenharmony_ci 16962306a36Sopenharmony_ci /* next descriptor physical address */ 17062306a36Sopenharmony_ci order_addr += sizeof(*desc); 17162306a36Sopenharmony_ci desc->order = lower_32_bits(order_addr | BIT(0)); 17262306a36Sopenharmony_ci desc->order_hi = upper_32_bits(order_addr); 17362306a36Sopenharmony_ci 17462306a36Sopenharmony_ci desc->saddr = lower_32_bits(mem_addr); 17562306a36Sopenharmony_ci desc->saddr_hi = upper_32_bits(mem_addr); 17662306a36Sopenharmony_ci desc->daddr = prtd->dma_data->dev_addr; 17762306a36Sopenharmony_ci 17862306a36Sopenharmony_ci desc->cmd = BIT(0); 17962306a36Sopenharmony_ci if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) 18062306a36Sopenharmony_ci desc->cmd |= BIT(12); 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_ci desc->length = period_len >> 2; 18362306a36Sopenharmony_ci desc->step_length = 0; 18462306a36Sopenharmony_ci desc->step_times = 1; 18562306a36Sopenharmony_ci 18662306a36Sopenharmony_ci mem_addr += period_len; 18762306a36Sopenharmony_ci } 18862306a36Sopenharmony_ci desc = &prtd->dma_desc_arr[num_periods - 1]; 18962306a36Sopenharmony_ci desc->order = lower_32_bits(prtd->dma_desc_arr_phy | BIT(0)); 19062306a36Sopenharmony_ci desc->order_hi = upper_32_bits(prtd->dma_desc_arr_phy); 19162306a36Sopenharmony_ci 19262306a36Sopenharmony_ci /* init position descriptor */ 19362306a36Sopenharmony_ci *prtd->dma_pos_desc = *prtd->dma_desc_arr; 19462306a36Sopenharmony_ci 19562306a36Sopenharmony_ci return 0; 19662306a36Sopenharmony_ci} 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_cistatic snd_pcm_uframes_t 19962306a36Sopenharmony_ciloongson_pcm_pointer(struct snd_soc_component *component, 20062306a36Sopenharmony_ci struct snd_pcm_substream *substream) 20162306a36Sopenharmony_ci{ 20262306a36Sopenharmony_ci struct snd_pcm_runtime *runtime = substream->runtime; 20362306a36Sopenharmony_ci struct loongson_runtime_data *prtd = runtime->private_data; 20462306a36Sopenharmony_ci struct loongson_dma_desc *desc; 20562306a36Sopenharmony_ci snd_pcm_uframes_t x; 20662306a36Sopenharmony_ci u64 addr; 20762306a36Sopenharmony_ci 20862306a36Sopenharmony_ci desc = dma_desc_save(prtd); 20962306a36Sopenharmony_ci addr = ((u64)desc->saddr_hi << 32) | desc->saddr; 21062306a36Sopenharmony_ci 21162306a36Sopenharmony_ci x = bytes_to_frames(runtime, addr - runtime->dma_addr); 21262306a36Sopenharmony_ci if (x == runtime->buffer_size) 21362306a36Sopenharmony_ci x = 0; 21462306a36Sopenharmony_ci return x; 21562306a36Sopenharmony_ci} 21662306a36Sopenharmony_ci 21762306a36Sopenharmony_cistatic irqreturn_t loongson_pcm_dma_irq(int irq, void *devid) 21862306a36Sopenharmony_ci{ 21962306a36Sopenharmony_ci struct snd_pcm_substream *substream = devid; 22062306a36Sopenharmony_ci 22162306a36Sopenharmony_ci snd_pcm_period_elapsed(substream); 22262306a36Sopenharmony_ci return IRQ_HANDLED; 22362306a36Sopenharmony_ci} 22462306a36Sopenharmony_ci 22562306a36Sopenharmony_cistatic int loongson_pcm_open(struct snd_soc_component *component, 22662306a36Sopenharmony_ci struct snd_pcm_substream *substream) 22762306a36Sopenharmony_ci{ 22862306a36Sopenharmony_ci struct snd_pcm_runtime *runtime = substream->runtime; 22962306a36Sopenharmony_ci struct snd_soc_pcm_runtime *rtd = substream->private_data; 23062306a36Sopenharmony_ci struct snd_card *card = substream->pcm->card; 23162306a36Sopenharmony_ci struct loongson_runtime_data *prtd; 23262306a36Sopenharmony_ci struct loongson_dma_data *dma_data; 23362306a36Sopenharmony_ci int ret; 23462306a36Sopenharmony_ci 23562306a36Sopenharmony_ci /* 23662306a36Sopenharmony_ci * For mysterious reasons (and despite what the manual says) 23762306a36Sopenharmony_ci * playback samples are lost if the DMA count is not a multiple 23862306a36Sopenharmony_ci * of the DMA burst size. Let's add a rule to enforce that. 23962306a36Sopenharmony_ci */ 24062306a36Sopenharmony_ci snd_pcm_hw_constraint_step(runtime, 0, 24162306a36Sopenharmony_ci SNDRV_PCM_HW_PARAM_PERIOD_BYTES, 128); 24262306a36Sopenharmony_ci snd_pcm_hw_constraint_step(runtime, 0, 24362306a36Sopenharmony_ci SNDRV_PCM_HW_PARAM_BUFFER_BYTES, 128); 24462306a36Sopenharmony_ci snd_pcm_hw_constraint_integer(substream->runtime, 24562306a36Sopenharmony_ci SNDRV_PCM_HW_PARAM_PERIODS); 24662306a36Sopenharmony_ci snd_soc_set_runtime_hwparams(substream, &ls_pcm_hardware); 24762306a36Sopenharmony_ci 24862306a36Sopenharmony_ci prtd = kzalloc(sizeof(*prtd), GFP_KERNEL); 24962306a36Sopenharmony_ci if (!prtd) 25062306a36Sopenharmony_ci return -ENOMEM; 25162306a36Sopenharmony_ci 25262306a36Sopenharmony_ci prtd->dma_desc_arr = dma_alloc_coherent(card->dev, PAGE_SIZE, 25362306a36Sopenharmony_ci &prtd->dma_desc_arr_phy, 25462306a36Sopenharmony_ci GFP_KERNEL); 25562306a36Sopenharmony_ci if (!prtd->dma_desc_arr) { 25662306a36Sopenharmony_ci ret = -ENOMEM; 25762306a36Sopenharmony_ci goto desc_err; 25862306a36Sopenharmony_ci } 25962306a36Sopenharmony_ci prtd->dma_desc_arr_size = PAGE_SIZE / sizeof(*prtd->dma_desc_arr); 26062306a36Sopenharmony_ci 26162306a36Sopenharmony_ci prtd->dma_pos_desc = dma_alloc_coherent(card->dev, 26262306a36Sopenharmony_ci sizeof(*prtd->dma_pos_desc), 26362306a36Sopenharmony_ci &prtd->dma_pos_desc_phy, 26462306a36Sopenharmony_ci GFP_KERNEL); 26562306a36Sopenharmony_ci if (!prtd->dma_pos_desc) { 26662306a36Sopenharmony_ci ret = -ENOMEM; 26762306a36Sopenharmony_ci goto pos_err; 26862306a36Sopenharmony_ci } 26962306a36Sopenharmony_ci 27062306a36Sopenharmony_ci dma_data = snd_soc_dai_get_dma_data(asoc_rtd_to_cpu(rtd, 0), substream); 27162306a36Sopenharmony_ci prtd->dma_data = dma_data; 27262306a36Sopenharmony_ci 27362306a36Sopenharmony_ci substream->runtime->private_data = prtd; 27462306a36Sopenharmony_ci 27562306a36Sopenharmony_ci return 0; 27662306a36Sopenharmony_cipos_err: 27762306a36Sopenharmony_ci dma_free_coherent(card->dev, PAGE_SIZE, prtd->dma_desc_arr, 27862306a36Sopenharmony_ci prtd->dma_desc_arr_phy); 27962306a36Sopenharmony_cidesc_err: 28062306a36Sopenharmony_ci kfree(prtd); 28162306a36Sopenharmony_ci 28262306a36Sopenharmony_ci return ret; 28362306a36Sopenharmony_ci} 28462306a36Sopenharmony_ci 28562306a36Sopenharmony_cistatic int loongson_pcm_close(struct snd_soc_component *component, 28662306a36Sopenharmony_ci struct snd_pcm_substream *substream) 28762306a36Sopenharmony_ci{ 28862306a36Sopenharmony_ci struct snd_card *card = substream->pcm->card; 28962306a36Sopenharmony_ci struct loongson_runtime_data *prtd = substream->runtime->private_data; 29062306a36Sopenharmony_ci 29162306a36Sopenharmony_ci dma_free_coherent(card->dev, PAGE_SIZE, prtd->dma_desc_arr, 29262306a36Sopenharmony_ci prtd->dma_desc_arr_phy); 29362306a36Sopenharmony_ci 29462306a36Sopenharmony_ci dma_free_coherent(card->dev, sizeof(*prtd->dma_pos_desc), 29562306a36Sopenharmony_ci prtd->dma_pos_desc, prtd->dma_pos_desc_phy); 29662306a36Sopenharmony_ci 29762306a36Sopenharmony_ci kfree(prtd); 29862306a36Sopenharmony_ci return 0; 29962306a36Sopenharmony_ci} 30062306a36Sopenharmony_ci 30162306a36Sopenharmony_cistatic int loongson_pcm_mmap(struct snd_soc_component *component, 30262306a36Sopenharmony_ci struct snd_pcm_substream *substream, 30362306a36Sopenharmony_ci struct vm_area_struct *vma) 30462306a36Sopenharmony_ci{ 30562306a36Sopenharmony_ci return remap_pfn_range(vma, vma->vm_start, 30662306a36Sopenharmony_ci substream->dma_buffer.addr >> PAGE_SHIFT, 30762306a36Sopenharmony_ci vma->vm_end - vma->vm_start, vma->vm_page_prot); 30862306a36Sopenharmony_ci} 30962306a36Sopenharmony_ci 31062306a36Sopenharmony_cistatic int loongson_pcm_new(struct snd_soc_component *component, 31162306a36Sopenharmony_ci struct snd_soc_pcm_runtime *rtd) 31262306a36Sopenharmony_ci{ 31362306a36Sopenharmony_ci struct snd_card *card = rtd->card->snd_card; 31462306a36Sopenharmony_ci struct snd_pcm_substream *substream; 31562306a36Sopenharmony_ci struct loongson_dma_data *dma_data; 31662306a36Sopenharmony_ci unsigned int i; 31762306a36Sopenharmony_ci int ret; 31862306a36Sopenharmony_ci 31962306a36Sopenharmony_ci for_each_pcm_streams(i) { 32062306a36Sopenharmony_ci substream = rtd->pcm->streams[i].substream; 32162306a36Sopenharmony_ci if (!substream) 32262306a36Sopenharmony_ci continue; 32362306a36Sopenharmony_ci 32462306a36Sopenharmony_ci dma_data = snd_soc_dai_get_dma_data(asoc_rtd_to_cpu(rtd, 0), 32562306a36Sopenharmony_ci substream); 32662306a36Sopenharmony_ci ret = devm_request_irq(card->dev, dma_data->irq, 32762306a36Sopenharmony_ci loongson_pcm_dma_irq, 32862306a36Sopenharmony_ci IRQF_TRIGGER_HIGH, LS_I2S_DRVNAME, 32962306a36Sopenharmony_ci substream); 33062306a36Sopenharmony_ci if (ret < 0) { 33162306a36Sopenharmony_ci dev_err(card->dev, "request irq for DMA failed\n"); 33262306a36Sopenharmony_ci return ret; 33362306a36Sopenharmony_ci } 33462306a36Sopenharmony_ci } 33562306a36Sopenharmony_ci 33662306a36Sopenharmony_ci return snd_pcm_set_fixed_buffer_all(rtd->pcm, SNDRV_DMA_TYPE_DEV, 33762306a36Sopenharmony_ci card->dev, 33862306a36Sopenharmony_ci ls_pcm_hardware.buffer_bytes_max); 33962306a36Sopenharmony_ci} 34062306a36Sopenharmony_ci 34162306a36Sopenharmony_ciconst struct snd_soc_component_driver loongson_i2s_component = { 34262306a36Sopenharmony_ci .name = LS_I2S_DRVNAME, 34362306a36Sopenharmony_ci .open = loongson_pcm_open, 34462306a36Sopenharmony_ci .close = loongson_pcm_close, 34562306a36Sopenharmony_ci .hw_params = loongson_pcm_hw_params, 34662306a36Sopenharmony_ci .trigger = loongson_pcm_trigger, 34762306a36Sopenharmony_ci .pointer = loongson_pcm_pointer, 34862306a36Sopenharmony_ci .mmap = loongson_pcm_mmap, 34962306a36Sopenharmony_ci .pcm_construct = loongson_pcm_new, 35062306a36Sopenharmony_ci}; 351