Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp1175684ybl; Thu, 23 Jan 2020 15:12:46 -0800 (PST) X-Google-Smtp-Source: APXvYqw/04z7GtFYTKMAzWM4+m3KLlNlpvctptdHWdpS6Arm84YmnCKpUzPFRqXKHOw8ol0Nfj1q X-Received: by 2002:a05:6830:16c4:: with SMTP id l4mr609778otr.3.1579821166750; Thu, 23 Jan 2020 15:12:46 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1579821166; cv=none; d=google.com; s=arc-20160816; b=pE4qHrjM1gXbOypMPfvTXkt/4GvSjSdHTh4DSy4g+eR0Hoo7T0Pw8nCHngAcAwQkf7 dSoJO+BSD+1xQPL+QCWmW4DshkliUVtLTn0o8goWKqIVd+b/MiGK9zUnCC1pEl+6Lzop Hg78kcEU2iJNpIxPszGyAPoIfu0F0KE0jeSZa5Wc0HeOtNLEJKsEqh3p18Rc31PkDjC/ U3LSdulu80bnq89rlfs4zgR6B4CvoiQWzwECxaVwxpgl1DWW37ZTcQ+xJZXNQdDhnb4u 9ueyt6ejS4Fiye7oAs+Mb1wSFsXtf/p358exiGysIGxv7Nl20EyhXQv5C/MdXY+cXXWB PjBA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from :dkim-signature; bh=qyH+MurQ1K+JP11j8j9cz8CEs+bqnCzxDejqE3pavzg=; b=aUJXUoT/ozBl2FX03euPCC7SmErzAd/N76BOn0jBt6eevyTGpNdKLGBCuHL4MgjDeh lYZZQCl6BGf2I4bAU3+VEa8vAlJ+K2GS+g0DHJ1d/dLcG4tyxeWvxkNZ7q+e8AJz3GHL QyffUwhipj1AMHKKZM71tp9QaDiG7w9AGqy2YqqJ2flkYNQCRf86r6AeaF06g30iVqi3 uylzWpiltyOAKvhaiflu74wnkiLQzqXLtITXz6qY+wMEDiyUgc+Z6i0jlpVnuk60XHbD JrBHGqHt2bqD3A6ip/xP0VBPfwsMwVMuTyzRH/+GOCXBHh545D5yQZDQ3b2eao61Io+o xBjw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=udBXxgBs; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id m5si1742852ote.187.2020.01.23.15.12.34; Thu, 23 Jan 2020 15:12:46 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=udBXxgBs; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1729999AbgAWXLS (ORCPT + 99 others); Thu, 23 Jan 2020 18:11:18 -0500 Received: from mail-wm1-f67.google.com ([209.85.128.67]:36212 "EHLO mail-wm1-f67.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1729898AbgAWXLA (ORCPT ); Thu, 23 Jan 2020 18:11:00 -0500 Received: by mail-wm1-f67.google.com with SMTP id p17so4380721wma.1; Thu, 23 Jan 2020 15:10:58 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=from:to:cc:subject:date:message-id:in-reply-to:references :mime-version:content-transfer-encoding; bh=qyH+MurQ1K+JP11j8j9cz8CEs+bqnCzxDejqE3pavzg=; b=udBXxgBs4qAB9zBc/8NGftiGgBuHXk0vbsv5vqjAO/Y2H4CVcsAqbJUnRkaAt2sAFa bxMxWEEDjBIlTIcefoGrqM6YNrwCfQ1ZDVByg9e99k41n6h+515kigc3P4zIWo8wm43H altwffGEreat6+jY3zcRvUvsozvtwWmvKssZt1AdGsFN1qg4preaF6Fzl+il8/EZ3ao3 E7vO2nkq7cGG3P7DGLEnJw3pNkJNLqxHDIvXdyeSD/+NQOpwYNiJGcNeWfjqcUH0O96u /4vfZqtltDAEbyEoZEJ+Iw3/2X2cQFOsR61hlz6ecl6YwIL3Y3op1nGT4vfqD6WC4p2h nkhw== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references:mime-version:content-transfer-encoding; bh=qyH+MurQ1K+JP11j8j9cz8CEs+bqnCzxDejqE3pavzg=; b=KiH56GrOIKJ0LJSqNc7DNfvCw0gfQy9KBSazrMH/Q+DfP6qfrLRRTzsKvYjYcg1GPk ca2zTZ2paUmlT9Hq4CbNAqTI8NtihQORjNhoA2tL7hLfHcOTgmLeNgymnqAZRBZGeE/G N+/kXnITqtgDlHK+fkBRC5+yH1uRUeO47IaYG8qU7Ss0Gd5eM2IREEZ6KEhCz6FXpk9O u2c3ClLxixzVmsijRQ4TuDQQAjXnqArLyfj+2uuKvEBvBi5M6+GmlFE5MrnXa09Wc+aK RV3+lUHzkg0gLS2mu2+EN0s8+d9wLkXrA7ClSv+22CphIEFUitsbkwnV0AdRovMzVpOa pW4w== X-Gm-Message-State: APjAAAXQuMEa2rYLr0AIlGOHLTNJK0VO/AoIFsQO1D9txagBwgkOZMWf 7Bv+mWruG3rFRq6ybIdDDSk= X-Received: by 2002:a05:600c:145:: with SMTP id w5mr221972wmm.157.1579821057838; Thu, 23 Jan 2020 15:10:57 -0800 (PST) Received: from localhost.localdomain (79-139-233-37.dynamic.spd-mgts.ru. [79.139.233.37]) by smtp.gmail.com with ESMTPSA id z6sm5105552wrw.36.2020.01.23.15.10.56 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Thu, 23 Jan 2020 15:10:57 -0800 (PST) From: Dmitry Osipenko To: Laxman Dewangan , Vinod Koul , Dan Williams , Thierry Reding , Jonathan Hunter , =?UTF-8?q?Micha=C5=82=20Miros=C5=82aw?= Cc: dmaengine@vger.kernel.org, linux-tegra@vger.kernel.org, linux-kernel@vger.kernel.org Subject: [PATCH v5 11/14] dmaengine: tegra-apb: Clean up suspend-resume Date: Fri, 24 Jan 2020 02:03:22 +0300 Message-Id: <20200123230325.3037-12-digetx@gmail.com> X-Mailer: git-send-email 2.24.0 In-Reply-To: <20200123230325.3037-1-digetx@gmail.com> References: <20200123230325.3037-1-digetx@gmail.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org It is enough to check whether hardware is busy on suspend and to reset it across of suspend-resume because channel's configuration is fully re-programmed on each DMA transaction anyways and because save-restore of an active channel won't end up well without pausing transfer prior to saving of the state (note that all channels shall be idling at the time of suspend, so save-restore is not needed at all). Signed-off-by: Dmitry Osipenko --- drivers/dma/tegra20-apb-dma.c | 133 ++++++++++++++++++---------------- 1 file changed, 69 insertions(+), 64 deletions(-) diff --git a/drivers/dma/tegra20-apb-dma.c b/drivers/dma/tegra20-apb-dma.c index 0ee28d8e3c96..4a371891b7c4 100644 --- a/drivers/dma/tegra20-apb-dma.c +++ b/drivers/dma/tegra20-apb-dma.c @@ -1401,6 +1401,36 @@ static const struct tegra_dma_chip_data tegra148_dma_chip_data = { .support_separate_wcount_reg = true, }; +static int tegra_dma_init_hw(struct tegra_dma *tdma) +{ + int err; + + err = reset_control_assert(tdma->rst); + if (err) { + dev_err(tdma->dev, "failed to assert reset: %d\n", err); + return err; + } + + err = clk_enable(tdma->dma_clk); + if (err) { + dev_err(tdma->dev, "failed to enable clk: %d\n", err); + return err; + } + + /* reset DMA controller */ + udelay(2); + reset_control_deassert(tdma->rst); + + /* enable global DMA registers */ + tdma_write(tdma, TEGRA_APBDMA_GENERAL, TEGRA_APBDMA_GENERAL_ENABLE); + tdma_write(tdma, TEGRA_APBDMA_CONTROL, 0); + tdma_write(tdma, TEGRA_APBDMA_IRQ_MASK_SET, 0xFFFFFFFF); + + clk_disable(tdma->dma_clk); + + return 0; +} + static int tegra_dma_probe(struct platform_device *pdev) { const struct tegra_dma_chip_data *cdata; @@ -1442,25 +1472,13 @@ static int tegra_dma_probe(struct platform_device *pdev) if (ret) return ret; + ret = tegra_dma_init_hw(tdma); + if (ret) + goto err_clk_unprepare; + pm_runtime_irq_safe(&pdev->dev); pm_runtime_enable(&pdev->dev); - ret = pm_runtime_get_sync(&pdev->dev); - if (ret < 0) - goto err_pm_disable; - - /* Reset DMA controller */ - reset_control_assert(tdma->rst); - udelay(2); - reset_control_deassert(tdma->rst); - - /* Enable global DMA registers */ - tdma_write(tdma, TEGRA_APBDMA_GENERAL, TEGRA_APBDMA_GENERAL_ENABLE); - tdma_write(tdma, TEGRA_APBDMA_CONTROL, 0); - tdma_write(tdma, TEGRA_APBDMA_IRQ_MASK_SET, 0xFFFFFFFFul); - - pm_runtime_put(&pdev->dev); - INIT_LIST_HEAD(&tdma->dma_dev.channels); for (i = 0; i < cdata->nr_channels; i++) { struct tegra_dma_channel *tdc = &tdma->channels[i]; @@ -1558,6 +1576,8 @@ static int tegra_dma_probe(struct platform_device *pdev) err_pm_disable: pm_runtime_disable(&pdev->dev); + +err_clk_unprepare: clk_unprepare(tdma->dma_clk); return ret; @@ -1577,26 +1597,6 @@ static int tegra_dma_remove(struct platform_device *pdev) static int tegra_dma_runtime_suspend(struct device *dev) { struct tegra_dma *tdma = dev_get_drvdata(dev); - unsigned int i; - - tdma->reg_gen = tdma_read(tdma, TEGRA_APBDMA_GENERAL); - for (i = 0; i < tdma->chip_data->nr_channels; i++) { - struct tegra_dma_channel *tdc = &tdma->channels[i]; - struct tegra_dma_channel_regs *ch_reg = &tdc->channel_reg; - - /* Only save the state of DMA channels that are in use */ - if (!tdc->config_init) - continue; - - ch_reg->csr = tdc_read(tdc, TEGRA_APBDMA_CHAN_CSR); - ch_reg->ahb_ptr = tdc_read(tdc, TEGRA_APBDMA_CHAN_AHBPTR); - ch_reg->apb_ptr = tdc_read(tdc, TEGRA_APBDMA_CHAN_APBPTR); - ch_reg->ahb_seq = tdc_read(tdc, TEGRA_APBDMA_CHAN_AHBSEQ); - ch_reg->apb_seq = tdc_read(tdc, TEGRA_APBDMA_CHAN_APBSEQ); - if (tdma->chip_data->support_separate_wcount_reg) - ch_reg->wcount = tdc_read(tdc, - TEGRA_APBDMA_CHAN_WCOUNT); - } clk_disable(tdma->dma_clk); @@ -1606,46 +1606,51 @@ static int tegra_dma_runtime_suspend(struct device *dev) static int tegra_dma_runtime_resume(struct device *dev) { struct tegra_dma *tdma = dev_get_drvdata(dev); - unsigned int i; - int ret; - ret = clk_enable(tdma->dma_clk); - if (ret < 0) { - dev_err(dev, "clk_enable failed: %d\n", ret); - return ret; - } + return clk_enable(tdma->dma_clk); +} - tdma_write(tdma, TEGRA_APBDMA_GENERAL, tdma->reg_gen); - tdma_write(tdma, TEGRA_APBDMA_CONTROL, 0); - tdma_write(tdma, TEGRA_APBDMA_IRQ_MASK_SET, 0xFFFFFFFFul); +static int __maybe_unused tegra_dma_dev_suspend(struct device *dev) +{ + struct tegra_dma *tdma = dev_get_drvdata(dev); + unsigned long flags; + unsigned int i; + bool busy; for (i = 0; i < tdma->chip_data->nr_channels; i++) { struct tegra_dma_channel *tdc = &tdma->channels[i]; - struct tegra_dma_channel_regs *ch_reg = &tdc->channel_reg; - - /* Only restore the state of DMA channels that are in use */ - if (!tdc->config_init) - continue; - - if (tdma->chip_data->support_separate_wcount_reg) - tdc_write(tdc, TEGRA_APBDMA_CHAN_WCOUNT, - ch_reg->wcount); - tdc_write(tdc, TEGRA_APBDMA_CHAN_APBSEQ, ch_reg->apb_seq); - tdc_write(tdc, TEGRA_APBDMA_CHAN_APBPTR, ch_reg->apb_ptr); - tdc_write(tdc, TEGRA_APBDMA_CHAN_AHBSEQ, ch_reg->ahb_seq); - tdc_write(tdc, TEGRA_APBDMA_CHAN_AHBPTR, ch_reg->ahb_ptr); - tdc_write(tdc, TEGRA_APBDMA_CHAN_CSR, - ch_reg->csr & ~TEGRA_APBDMA_CSR_ENB); + + tasklet_kill(&tdc->tasklet); + + spin_lock_irqsave(&tdc->lock, flags); + busy = tdc->busy; + spin_unlock_irqrestore(&tdc->lock, flags); + + if (busy) { + dev_err(tdma->dev, "channel %u busy\n", i); + return -EBUSY; + } } - return 0; + return pm_runtime_force_suspend(dev); +} + +static int __maybe_unused tegra_dma_dev_resume(struct device *dev) +{ + struct tegra_dma *tdma = dev_get_drvdata(dev); + int err; + + err = tegra_dma_init_hw(tdma); + if (err) + return err; + + return pm_runtime_force_resume(dev); } static const struct dev_pm_ops tegra_dma_dev_pm_ops = { SET_RUNTIME_PM_OPS(tegra_dma_runtime_suspend, tegra_dma_runtime_resume, NULL) - SET_SYSTEM_SLEEP_PM_OPS(pm_runtime_force_suspend, - pm_runtime_force_resume) + SET_SYSTEM_SLEEP_PM_OPS(tegra_dma_dev_suspend, tegra_dma_dev_resume) }; static const struct of_device_id tegra_dma_of_match[] = { -- 2.24.0