Received: by 2002:a05:7412:b795:b0:e2:908c:2ebd with SMTP id iv21csp334714rdb; Thu, 2 Nov 2023 05:19:26 -0700 (PDT) X-Google-Smtp-Source: AGHT+IEBLCWywNInzRrO00Tfi0bh4zqH/JIOdpCQwAEQqFODtlnNq1NiPneQs36MUTqMTLunLI4r X-Received: by 2002:a17:90a:ca13:b0:27f:fce3:2266 with SMTP id x19-20020a17090aca1300b0027ffce32266mr16218845pjt.24.1698927566637; Thu, 02 Nov 2023 05:19:26 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1698927566; cv=none; d=google.com; s=arc-20160816; b=zBxaDDus5EBg63u4xaaOcqcQ+hIQ2D3H8uIg4HDVu1duQ60NrziPgaFNc+LUi+flXv Xi6Rs34uON+LAQVgHUwju27bzHvUrbQ9oVGxQVfwmAHO9ewDwqOqslf54dXSETNvO2NZ iBttnhaMbIq459v5SjcJ5Pe1pnLzBxNH7zWP70Eludah2Pw8IElBunyaNgCFAilb/a8f 0MUZhxkJgASDNAf45azhCBdmFbPmiQqvgP9Ggi+hqAOMRMlcqtVKOmjNzl+hdIvYwKSg wv8FZpF/90Prd8J7QDWRM76ddD4xRKzraY8hoUKmZdwSNLj72TW7436xzUpaQG/pdWAR k5Dw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:mime-version:message-id:date:subject:cc:to:from; bh=4yoCvNV4O3yFa03KuaAf7XXnWo6Se4fFGWaooCn77B8=; fh=zgzHV65DKnrUpgHiselshH9xv6nmPvca/3SIMVtE2OM=; b=sGDMSkxUEwpi9YiUiUlUjOjRCLDT4rIg7SFy3DWOD8z2w2jC08KsIlr8YPX1F08R0f uAwszhtxpxoZFlrdfOQIqWhG20zBB4u4rAMcSRv55gSrRrtSgC/tPgIXIXYzokdPcs+0 ZRb+EdlFiACLT3YZIauTQHngBuJeVb8isLaKyrh1ApY9bKVgzjoiDM4ea+3VD1uk+iiI pss+wAy9Mg2QGNjzkx1pNGWFgnoexqQbs3KlKurNOrDzG4at6haufhNDz54IJSRMFliX r56WwrR1ToW84vc3CX/HJljeMfaFVot6I3V9KEb5pZ9r2hjycIbgy5F2lpYG01qn1VG5 JOgA== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::3:2 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from agentk.vger.email (agentk.vger.email. [2620:137:e000::3:2]) by mx.google.com with ESMTPS id b15-20020a17090a12cf00b0028098007c8csi2777726pjg.84.2023.11.02.05.19.26 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Thu, 02 Nov 2023 05:19:26 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::3:2 as permitted sender) client-ip=2620:137:e000::3:2; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::3:2 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: from out1.vger.email (depot.vger.email [IPv6:2620:137:e000::3:0]) by agentk.vger.email (Postfix) with ESMTP id CF50580F696F; Thu, 2 Nov 2023 05:19:23 -0700 (PDT) X-Virus-Status: Clean X-Virus-Scanned: clamav-milter 0.103.10 at agentk.vger.email Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1347360AbjKBMSH (ORCPT + 99 others); Thu, 2 Nov 2023 08:18:07 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:34922 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1347434AbjKBMRu (ORCPT ); Thu, 2 Nov 2023 08:17:50 -0400 Received: from SHSQR01.spreadtrum.com (unknown [222.66.158.135]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id BDC3F193; Thu, 2 Nov 2023 05:17:46 -0700 (PDT) Received: from dlp.unisoc.com ([10.29.3.86]) by SHSQR01.spreadtrum.com with ESMTP id 3A2CHFCR024307; Thu, 2 Nov 2023 20:17:15 +0800 (+08) (envelope-from Kaiwei.Liu@unisoc.com) Received: from SHDLP.spreadtrum.com (shmbx07.spreadtrum.com [10.0.1.12]) by dlp.unisoc.com (SkyGuard) with ESMTPS id 4SLjS10k1Yz2M9fbK; Thu, 2 Nov 2023 20:12:33 +0800 (CST) Received: from xm9614pcu.spreadtrum.com (10.13.2.29) by shmbx07.spreadtrum.com (10.0.1.12) with Microsoft SMTP Server (TLS) id 15.0.1497.23; Thu, 2 Nov 2023 20:17:14 +0800 From: Kaiwei Liu To: Vinod Koul , Orson Zhai , Baolin Wang , Chunyan Zhang CC: , , kaiwei liu , Wenming Wu Subject: [PATCH 2/2] dmaengine: sprd: optimize two stage transfer function Date: Thu, 2 Nov 2023 20:17:08 +0800 Message-ID: <20231102121708.32279-1-kaiwei.liu@unisoc.com> X-Mailer: git-send-email 2.17.1 MIME-Version: 1.0 Content-Type: text/plain X-Originating-IP: [10.13.2.29] X-ClientProxiedBy: SHCAS03.spreadtrum.com (10.0.1.207) To shmbx07.spreadtrum.com (10.0.1.12) X-MAIL: SHSQR01.spreadtrum.com 3A2CHFCR024307 X-Spam-Status: No, score=-0.8 required=5.0 tests=HEADER_FROM_DIFFERENT_DOMAINS, MAILING_LIST_MULTI,SPF_HELO_NONE,SPF_PASS,T_SCC_BODY_TEXT_LINE autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on agentk.vger.email Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org X-Greylist: Sender passed SPF test, not delayed by milter-greylist-4.6.4 (agentk.vger.email [0.0.0.0]); Thu, 02 Nov 2023 05:19:24 -0700 (PDT) From: "kaiwei.liu" For SPRD DMA, it provides a function that one channel can start the second channel after completing the transmission, which we call two stage transfer mode. You can choose which channel can generate interrupt when finished. It can support up to two sets of such patterns. When configuring registers for two stage transfer mode, we need to set the mask bit to ensure that the setting are accurate. And we should clear the two stage transfer configuration when release DMA channel. The two stage transfer function is mainly used by SPRD audio, and now audio also requires that the data need to be accessed on the device side. So here use the src_port_window_size and dst_port_win- dow_size in the struct of dma_slave_config. Signed-off-by: kaiwei.liu --- drivers/dma/sprd-dma.c | 116 ++++++++++++++++++++++++----------------- 1 file changed, 69 insertions(+), 47 deletions(-) diff --git a/drivers/dma/sprd-dma.c b/drivers/dma/sprd-dma.c index 8ab5a9082fc5..f54f4ba1a8bd 100644 --- a/drivers/dma/sprd-dma.c +++ b/drivers/dma/sprd-dma.c @@ -68,6 +68,7 @@ #define SPRD_DMA_GLB_TRANS_DONE_TRG BIT(18) #define SPRD_DMA_GLB_BLOCK_DONE_TRG BIT(17) #define SPRD_DMA_GLB_FRAG_DONE_TRG BIT(16) +#define SPRD_DMA_GLB_TRG_MASK GENMASK(19, 16) #define SPRD_DMA_GLB_TRG_OFFSET 16 #define SPRD_DMA_GLB_DEST_CHN_MASK GENMASK(13, 8) #define SPRD_DMA_GLB_DEST_CHN_OFFSET 8 @@ -155,6 +156,13 @@ #define SPRD_DMA_SOFTWARE_UID 0 +#define SPRD_DMA_SRC_CHN0_INT 9 +#define SPRD_DMA_SRC_CHN1_INT 10 +#define SPRD_DMA_DST_CHN0_INT 11 +#define SPRD_DMA_DST_CHN1_INT 12 +#define SPRD_DMA_2STAGE_SET 1 +#define SPRD_DMA_2STAGE_CLEAR 0 + /* dma data width values */ enum sprd_dma_datawidth { SPRD_DMA_DATAWIDTH_1_BYTE, @@ -431,53 +439,57 @@ static enum sprd_dma_req_mode sprd_dma_get_req_type(struct sprd_dma_chn *schan) return (frag_reg >> SPRD_DMA_REQ_MODE_OFFSET) & SPRD_DMA_REQ_MODE_MASK; } -static int sprd_dma_set_2stage_config(struct sprd_dma_chn *schan) +static void sprd_dma_2stage_write(struct sprd_dma_chn *schan, + u32 config_type, u32 grp_offset) { struct sprd_dma_dev *sdev = to_sprd_dma_dev(&schan->vc.chan); - u32 val, chn = schan->chn_num + 1; - - switch (schan->chn_mode) { - case SPRD_DMA_SRC_CHN0: - val = chn & SPRD_DMA_GLB_SRC_CHN_MASK; - val |= BIT(schan->trg_mode - 1) << SPRD_DMA_GLB_TRG_OFFSET; - val |= SPRD_DMA_GLB_2STAGE_EN; - if (schan->int_type != SPRD_DMA_NO_INT) - val |= SPRD_DMA_GLB_SRC_INT; - - sprd_dma_glb_update(sdev, SPRD_DMA_GLB_2STAGE_GRP1, val, val); - break; - - case SPRD_DMA_SRC_CHN1: - val = chn & SPRD_DMA_GLB_SRC_CHN_MASK; - val |= BIT(schan->trg_mode - 1) << SPRD_DMA_GLB_TRG_OFFSET; - val |= SPRD_DMA_GLB_2STAGE_EN; - if (schan->int_type != SPRD_DMA_NO_INT) - val |= SPRD_DMA_GLB_SRC_INT; - - sprd_dma_glb_update(sdev, SPRD_DMA_GLB_2STAGE_GRP2, val, val); - break; - - case SPRD_DMA_DST_CHN0: - val = (chn << SPRD_DMA_GLB_DEST_CHN_OFFSET) & - SPRD_DMA_GLB_DEST_CHN_MASK; - val |= SPRD_DMA_GLB_2STAGE_EN; - if (schan->int_type != SPRD_DMA_NO_INT) - val |= SPRD_DMA_GLB_DEST_INT; - - sprd_dma_glb_update(sdev, SPRD_DMA_GLB_2STAGE_GRP1, val, val); - break; - - case SPRD_DMA_DST_CHN1: - val = (chn << SPRD_DMA_GLB_DEST_CHN_OFFSET) & - SPRD_DMA_GLB_DEST_CHN_MASK; - val |= SPRD_DMA_GLB_2STAGE_EN; - if (schan->int_type != SPRD_DMA_NO_INT) - val |= SPRD_DMA_GLB_DEST_INT; - - sprd_dma_glb_update(sdev, SPRD_DMA_GLB_2STAGE_GRP2, val, val); - break; + u32 mask_val; + u32 chn = schan->chn_num + 1; + u32 val = 0; + + if (config_type == SPRD_DMA_2STAGE_SET) { + if (schan->chn_mode == SPRD_DMA_SRC_CHN0 || + schan->chn_mode == SPRD_DMA_SRC_CHN1) { + val = chn & SPRD_DMA_GLB_SRC_CHN_MASK; + val |= BIT(schan->trg_mode - 1) << SPRD_DMA_GLB_TRG_OFFSET; + val |= SPRD_DMA_GLB_2STAGE_EN; + if (schan->int_type & SPRD_DMA_SRC_CHN0_INT || + schan->int_type & SPRD_DMA_SRC_CHN1_INT) + val |= SPRD_DMA_GLB_SRC_INT; + mask_val = SPRD_DMA_GLB_SRC_INT | SPRD_DMA_GLB_TRG_MASK | + SPRD_DMA_GLB_SRC_CHN_MASK; + } else { + val = (chn << SPRD_DMA_GLB_DEST_CHN_OFFSET) & + SPRD_DMA_GLB_DEST_CHN_MASK; + val |= SPRD_DMA_GLB_2STAGE_EN; + if (schan->int_type & SPRD_DMA_DST_CHN0_INT || + schan->int_type & SPRD_DMA_DST_CHN1_INT) + val |= SPRD_DMA_GLB_DEST_INT; + mask_val = SPRD_DMA_GLB_DEST_INT | SPRD_DMA_GLB_DEST_CHN_MASK; + } + } else { + if (schan->chn_mode == SPRD_DMA_SRC_CHN0 || + schan->chn_mode == SPRD_DMA_SRC_CHN1) + mask_val = SPRD_DMA_GLB_SRC_INT | SPRD_DMA_GLB_TRG_MASK | + SPRD_DMA_GLB_2STAGE_EN | SPRD_DMA_GLB_SRC_CHN_MASK; + else + mask_val = SPRD_DMA_GLB_DEST_INT | SPRD_DMA_GLB_2STAGE_EN | + SPRD_DMA_GLB_DEST_CHN_MASK; + } + sprd_dma_glb_update(sdev, grp_offset, mask_val, val); +} - default: +static int sprd_dma_2stage_config(struct sprd_dma_chn *schan, u32 config_type) +{ + struct sprd_dma_dev *sdev = to_sprd_dma_dev(&schan->vc.chan); + + if (schan->chn_mode == SPRD_DMA_SRC_CHN0 || + schan->chn_mode == SPRD_DMA_DST_CHN0) + sprd_dma_2stage_write(schan, config_type, SPRD_DMA_GLB_2STAGE_GRP1); + else if (schan->chn_mode == SPRD_DMA_SRC_CHN1 || + schan->chn_mode == SPRD_DMA_DST_CHN1) + sprd_dma_2stage_write(schan, config_type, SPRD_DMA_GLB_2STAGE_GRP2); + else { dev_err(sdev->dma_dev.dev, "invalid channel mode setting %d\n", schan->chn_mode); return -EINVAL; @@ -545,7 +557,7 @@ static void sprd_dma_start(struct sprd_dma_chn *schan) * Set 2-stage configuration if the channel starts one 2-stage * transfer. */ - if (schan->chn_mode && sprd_dma_set_2stage_config(schan)) + if (schan->chn_mode && sprd_dma_2stage_config(schan, SPRD_DMA_2STAGE_SET)) return; /* @@ -569,6 +581,12 @@ static void sprd_dma_stop(struct sprd_dma_chn *schan) sprd_dma_set_pending(schan, false); sprd_dma_unset_uid(schan); sprd_dma_clear_int(schan); + /* + * If 2-stage transfer is used, the configuration must be clear + * when release DMA channel. + */ + if (schan->chn_mode) + sprd_dma_2stage_config(schan, SPRD_DMA_2STAGE_CLEAR); schan->cur_desc = NULL; } @@ -757,7 +775,9 @@ static int sprd_dma_fill_desc(struct dma_chan *chan, phys_addr_t llist_ptr; if (dir == DMA_MEM_TO_DEV) { - src_step = sprd_dma_get_step(slave_cfg->src_addr_width); + src_step = slave_cfg->src_port_window_size ? + slave_cfg->src_port_window_size : + sprd_dma_get_step(slave_cfg->src_addr_width); if (src_step < 0) { dev_err(sdev->dma_dev.dev, "invalid source step\n"); return src_step; @@ -773,7 +793,9 @@ static int sprd_dma_fill_desc(struct dma_chan *chan, else dst_step = SPRD_DMA_NONE_STEP; } else { - dst_step = sprd_dma_get_step(slave_cfg->dst_addr_width); + dst_step = slave_cfg->dst_port_window_size ? + slave_cfg->dst_port_window_size : + sprd_dma_get_step(slave_cfg->dst_addr_width); if (dst_step < 0) { dev_err(sdev->dma_dev.dev, "invalid destination step\n"); return dst_step; -- 2.17.1