Received: by 2002:a25:ad19:0:0:0:0:0 with SMTP id y25csp379278ybi; Mon, 15 Jul 2019 22:35:22 -0700 (PDT) X-Google-Smtp-Source: APXvYqxRRpMEg44HKE/cPkHyrl7CQfIQuZL0ltgFMBebBBDJ1ijbPKKMUBQEOqT6K++loMTo3fAs X-Received: by 2002:a63:30c6:: with SMTP id w189mr30173328pgw.398.1563255321803; Mon, 15 Jul 2019 22:35:21 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1563255321; cv=none; d=google.com; s=arc-20160816; b=oGLQvEqUd6Oo3nyWVDuE5ks1wgTMnG0TsO4mAeAYOgc96uRuAofC+50h4uOOeMLx/r nBm8Oq6B8xQvhNMT1eD/H6V0wkCI6bjP2kDoW3Up3Kkhm53N2B25NsxthO06KkWGEfMe 7rcOpuj+Lk4rpkEe/10ZIjsDpLRVGsYkSWltjtVLvZv+ZyrlDz9k1pZUBpDlri6lOt5D UDu9pnMih8BzJ82U2byt/Za9Z6PBxcYkDYfwIuV6k9FCZRex17jAvFNGR6Sw06ABZFLj msRHRgy+eDGa66F/6gOL8pyHNf3qI8ef1p+XR9LGzCwo3FrWhrQNuDxSV3RaU7c6MvdC 8FJA== 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:subject:cc:to:from:date :dkim-signature; bh=+S2/ceIBrlaZ7vN8q3YBC37ssOnRLmTnqeoExPiamY0=; b=EQOzOvw+ZWvSEgeZifxUKLsxZf+lYYUhBJgkEYLYwi47irADjd1eQSUtAMDZ8zrFle dzmjlAPkW0nr2YVVfeMQSu+nJ/7LOnLauWDCYuBJyUqFsTgVDZClGpm29EGODpQoIDmR K5uv6LJSPdrNedJTCqC4bi46jsRObgLxa3FgCmRUac6Iqc0q41BSy2eKHOIJ4gkskiWs s20DVhgty89Z4jlSysyCGFcCE881kqNck/7Hv5agQJUtZFuWo2cuvcOK7bLvjTVrE4LS H72XpTZQ7FMtyCLM4DmsGlFEmNIlwiBSbIWH+Hywx8H28BQuiOVmQYvvwC+fptmEWLi4 32pg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=uUpzrNOa; 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 q187si17813766pga.220.2019.07.15.22.35.05; Mon, 15 Jul 2019 22:35:21 -0700 (PDT) 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=uUpzrNOa; 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 S1729749AbfGPFdi (ORCPT + 99 others); Tue, 16 Jul 2019 01:33:38 -0400 Received: from mail-lj1-f195.google.com ([209.85.208.195]:36898 "EHLO mail-lj1-f195.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726564AbfGPFdi (ORCPT ); Tue, 16 Jul 2019 01:33:38 -0400 Received: by mail-lj1-f195.google.com with SMTP id z28so18647255ljn.4; Mon, 15 Jul 2019 22:33:35 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=date:from:to:cc:subject:message-id:in-reply-to:references :mime-version:content-transfer-encoding; bh=+S2/ceIBrlaZ7vN8q3YBC37ssOnRLmTnqeoExPiamY0=; b=uUpzrNOalcO1T1yF08A5NgF3Kf/3IUU5X867dbmDdur2zm4po/LVmIqRPsQ85mW5gi R3YJMXc+AIwjDFObumpHD2cOra6q5upi+kUlzuMIznyxoNwGwRyCsyHMntvrp7FW3ATD 9wEryAe+03eFy9ffURAAKAKo/429rJhdk8vnRKZARuBOhp35Tw1w+XlH3hNQK6ehC4vb b3jU6cCXioncHFJpQ758l6y0AKN25ZHHkY+g0odnR7FjNHxOw86TVT/pYlFA5x1XbzJx Est+47xkS7w3IquClTzqwARnwKtJ+YcSlAWalIznA2qOIoaQnssALj5efeeZgO85aJc/ Je0Q== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:from:to:cc:subject:message-id:in-reply-to :references:mime-version:content-transfer-encoding; bh=+S2/ceIBrlaZ7vN8q3YBC37ssOnRLmTnqeoExPiamY0=; b=DE8EEpb1lx4SiJiD6scdZe7qkulZZd6vLvcXLpBIRWlHLoLZlhXdLcJ2ZAlJ2V3ns4 1gEJfIENetzEXU1sp+igblqJ9l/unhXJ888+8/+DdUW1hPAAoYhNI752k/qvuiShc8OK BuYIh1BOJiEiH0GtdRy+UTXlHHPP/g+w0fCXDVibodOluwyNkm+gQvrLpVCvsBC0hV27 Iahndv38qa8gzu8tTE4NQ4VqzWQPbv9RgD/FqnoRMKwBiWVzs6w12jDQy/maiBA8BtOd 7mrTJNDtE9UGESz0laO+n5YsNmS1ooyHz1vwYx/+yMQCEEmDjNZgac5sxyVNx8OWSna2 qO5A== X-Gm-Message-State: APjAAAX679ir2Pg3P8pljO8zWR5gmRnw90OagkZWm+xek5P4O3XXOAXK 2L/Re0g2VhzWGt8pJoCzOx0= X-Received: by 2002:a2e:9dc1:: with SMTP id x1mr16453185ljj.0.1563255214437; Mon, 15 Jul 2019 22:33:34 -0700 (PDT) Received: from dimatab (ppp79-139-233-208.pppoe.spdop.ru. [79.139.233.208]) by smtp.gmail.com with ESMTPSA id k4sm3524471ljg.59.2019.07.15.22.33.32 (version=TLS1_3 cipher=AEAD-AES256-GCM-SHA384 bits=256/256); Mon, 15 Jul 2019 22:33:33 -0700 (PDT) Date: Tue, 16 Jul 2019 08:37:01 +0300 From: Dmitry Osipenko To: Sowjanya Komatineni Cc: , , , , , , , , , , , , , , , , , , , , , Subject: Re: [PATCH V5 11/18] clk: tegra210: Add support for Tegra210 clocks Message-ID: <20190716083701.225f0fd9@dimatab> In-Reply-To: <86fc07d5-ab2e-a52a-a570-b1dfff4c20fe@nvidia.com> References: <1561687972-19319-1-git-send-email-skomatineni@nvidia.com> <1561687972-19319-12-git-send-email-skomatineni@nvidia.com> <3938092a-bbc7-b304-641d-31677539598d@nvidia.com> <932d4d50-120c-9191-6a9a-23bf9c96633b@nvidia.com> <0ee055ad-d397-32e5-60ee-d62c14c6f77b@gmail.com> <86fc07d5-ab2e-a52a-a570-b1dfff4c20fe@nvidia.com> X-Mailer: Claws Mail 3.17.3 (GTK+ 2.24.32; arm-unknown-linux-gnueabihf) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: quoted-printable Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org =D0=92 Mon, 15 Jul 2019 21:37:09 -0700 Sowjanya Komatineni =D0=BF=D0=B8=D1=88=D0=B5=D1=82: > On 7/15/19 8:50 PM, Dmitry Osipenko wrote: > > 16.07.2019 6:00, Sowjanya Komatineni =D0=BF=D0=B8=D1=88=D0=B5=D1=82: =20 > >> On 7/15/19 5:35 PM, Sowjanya Komatineni wrote: =20 > >>> On 7/14/19 2:41 PM, Dmitry Osipenko wrote: =20 > >>>> 13.07.2019 8:54, Sowjanya Komatineni =D0=BF=D0=B8=D1=88=D0=B5=D1=82:= =20 > >>>>> On 6/29/19 8:10 AM, Dmitry Osipenko wrote: =20 > >>>>>> 28.06.2019 5:12, Sowjanya Komatineni =D0=BF=D0=B8=D1=88=D0=B5=D1= =82: =20 > >>>>>>> This patch adds system suspend and resume support for Tegra210 > >>>>>>> clocks. > >>>>>>> > >>>>>>> All the CAR controller settings are lost on suspend when core > >>>>>>> power goes off. > >>>>>>> > >>>>>>> This patch has implementation for saving and restoring all > >>>>>>> the PLLs and clocks context during system suspend and resume > >>>>>>> to have the clocks back to same state for normal operation. > >>>>>>> > >>>>>>> Acked-by: Thierry Reding > >>>>>>> Signed-off-by: Sowjanya Komatineni > >>>>>>> --- > >>>>>>> =C2=A0=C2=A0 drivers/clk/tegra/clk-tegra210.c | 115 > >>>>>>> ++++++++++++++++++++++++++++++++++++++- > >>>>>>> =C2=A0=C2=A0 drivers/clk/tegra/clk.c=C2=A0=C2=A0=C2=A0=C2=A0=C2= =A0=C2=A0=C2=A0=C2=A0=C2=A0 |=C2=A0 14 +++++ > >>>>>>> =C2=A0=C2=A0 drivers/clk/tegra/clk.h=C2=A0=C2=A0=C2=A0=C2=A0=C2= =A0=C2=A0=C2=A0=C2=A0=C2=A0 |=C2=A0=C2=A0 1 + > >>>>>>> =C2=A0=C2=A0 3 files changed, 127 insertions(+), 3 deletions(-) > >>>>>>> > >>>>>>> diff --git a/drivers/clk/tegra/clk-tegra210.c > >>>>>>> b/drivers/clk/tegra/clk-tegra210.c > >>>>>>> index 1c08c53482a5..1b839544e086 100644 > >>>>>>> --- a/drivers/clk/tegra/clk-tegra210.c > >>>>>>> +++ b/drivers/clk/tegra/clk-tegra210.c > >>>>>>> @@ -9,10 +9,12 @@ > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> +#include > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> +#include > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> @@ -20,6 +22,7 @@ > >>>>>>> =C2=A0=C2=A0 #include > >>>>>>> =C2=A0=C2=A0 =C2=A0 #include "clk.h" > >>>>>>> +#include "clk-dfll.h" > >>>>>>> =C2=A0=C2=A0 #include "clk-id.h" > >>>>>>> =C2=A0=C2=A0 =C2=A0 /* > >>>>>>> @@ -225,6 +228,7 @@ > >>>>>>> =C2=A0=C2=A0 =C2=A0 #define CLK_RST_CONTROLLER_RST_DEV_Y_SET 0x2= a8 > >>>>>>> =C2=A0=C2=A0 #define CLK_RST_CONTROLLER_RST_DEV_Y_CLR 0x2ac > >>>>>>> +#define CPU_SOFTRST_CTRL 0x380 > >>>>>>> =C2=A0=C2=A0 =C2=A0 #define LVL2_CLK_GATE_OVRA 0xf8 > >>>>>>> =C2=A0=C2=A0 #define LVL2_CLK_GATE_OVRC 0x3a0 > >>>>>>> @@ -2820,6 +2824,7 @@ static int tegra210_enable_pllu(void) > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 struct tegra_clk_pll_freq_t= able *fentry; > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 struct tegra_clk_pll pllu; > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 u32 reg; > >>>>>>> +=C2=A0=C2=A0=C2=A0 int ret; > >>>>>>> =C2=A0=C2=A0 =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 for (fentry =3D pll_= u_freq_table; fentry->input_rate; > >>>>>>> fentry++) { > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 if = (fentry->input_rate =3D=3D pll_ref_freq) > >>>>>>> @@ -2847,10 +2852,10 @@ static int tegra210_enable_pllu(void) > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 fence_udelay(1, clk_base); > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 reg |=3D PLL_ENABLE; > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 writel(reg, clk_base + PLLU= _BASE); > >>>>>>> +=C2=A0=C2=A0=C2=A0 fence_udelay(1, clk_base); > >>>>>>> =C2=A0=C2=A0 -=C2=A0=C2=A0=C2=A0 readl_relaxed_poll_timeout_atom= ic(clk_base + > >>>>>>> PLLU_BASE, reg, > >>>>>>> -=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2= =A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 reg & PLL_B= ASE_LOCK, 2, 1000); > >>>>>>> -=C2=A0=C2=A0=C2=A0 if (!(reg & PLL_BASE_LOCK)) { > >>>>>>> +=C2=A0=C2=A0=C2=A0 ret =3D tegra210_wait_for_mask(&pllu, PLLU_BA= SE, > >>>>>>> PLL_BASE_LOCK); > >>>>>>> +=C2=A0=C2=A0=C2=A0 if (ret) { > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 pr_= err("Timed out waiting for PLL_U to lock\n"); > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 ret= urn -ETIMEDOUT; > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 } > >>>>>>> @@ -3283,6 +3288,103 @@ static void > >>>>>>> tegra210_disable_cpu_clock(u32 cpu) > >>>>>>> =C2=A0=C2=A0 } > >>>>>>> =C2=A0=C2=A0 =C2=A0 #ifdef CONFIG_PM_SLEEP > >>>>>>> +static u32 cpu_softrst_ctx[3]; > >>>>>>> +static struct platform_device *dfll_pdev; > >>>>>>> +#define car_readl(_base, _off) readl_relaxed(clk_base + > >>>>>>> (_base) + ((_off) * 4)) > >>>>>>> +#define car_writel(_val, _base, _off) \ > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 writel_relaxed(_val, = clk_base + (_base) + ((_off) * > >>>>>>> 4)) + > >>>>>>> +static int tegra210_clk_suspend(void) > >>>>>>> +{ > >>>>>>> +=C2=A0=C2=A0=C2=A0 unsigned int i; > >>>>>>> +=C2=A0=C2=A0=C2=A0 struct device_node *node; > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 tegra_cclkg_burst_policy_save_context(); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 if (!dfll_pdev) { > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 node =3D of_find_comp= atible_node(NULL, NULL, > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2= =A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0= =C2=A0=C2=A0=C2=A0 "nvidia,tegra210-dfll"); > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 if (node) > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2= =A0 dfll_pdev =3D of_find_device_by_node(node); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 of_node_put(node); > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 if (!dfll_pdev) > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2= =A0 pr_err("dfll node not found. no suspend for > >>>>>>> dfll\n"); > >>>>>>> +=C2=A0=C2=A0=C2=A0 } > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 if (dfll_pdev) > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 tegra_dfll_suspend(df= ll_pdev); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 /* Enable PLLP_OUT_CPU after dfll suspend */ > >>>>>>> +=C2=A0=C2=A0=C2=A0 tegra_clk_set_pllp_out_cpu(true); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 tegra_sclk_cclklp_burst_policy_save_context(); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 clk_save_context(); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 for (i =3D 0; i < ARRAY_SIZE(cpu_softrst_ctx)= ; i++) > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 cpu_softrst_ctx[i] = =3D car_readl(CPU_SOFTRST_CTRL, i); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 return 0; > >>>>>>> +} > >>>>>>> + > >>>>>>> +static void tegra210_clk_resume(void) > >>>>>>> +{ > >>>>>>> +=C2=A0=C2=A0=C2=A0 unsigned int i; > >>>>>>> +=C2=A0=C2=A0=C2=A0 struct clk_hw *parent; > >>>>>>> +=C2=A0=C2=A0=C2=A0 struct clk *clk; > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 /* > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * clk_restore_context restores clocks a= s per the clock > >>>>>>> tree. > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * dfllCPU_out is first in the clock tre= e to get > >>>>>>> restored and it > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * involves programming DFLL controller = along with > >>>>>>> restoring CPUG > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * clock burst policy. > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * DFLL programming needs dfll_ref and d= fll_soc > >>>>>>> peripheral clocks > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * to be restores which are part ofthe p= eripheral > >>>>>>> clocks. =20 > >>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0= =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2= =A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0= =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 ^ white-space > >>>> > >>>> Please use spellchecker to avoid typos. > >>>> =20 > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * So, peripheral clocks restore should = happen prior to > >>>>>>> dfll clock > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * restore. > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 */ > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 tegra_clk_osc_resume(clk_base); > >>>>>>> +=C2=A0=C2=A0=C2=A0 for (i =3D 0; i < ARRAY_SIZE(cpu_softrst_ctx)= ; i++) > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 car_writel(cpu_softrs= t_ctx[i], CPU_SOFTRST_CTRL, i); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 /* restore all plls and peripheral clocks */ > >>>>>>> +=C2=A0=C2=A0=C2=A0 tegra210_init_pllu(); > >>>>>>> +=C2=A0=C2=A0=C2=A0 clk_restore_context(); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 fence_udelay(5, clk_base); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 /* resume SCLK and CPULP clocks */ > >>>>>>> +=C2=A0=C2=A0=C2=A0 tegra_sclk_cpulp_burst_policy_restore_context= (); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 /* > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * restore CPUG clocks: > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * - enable DFLL in open loop mode > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * - switch CPUG to DFLL clock source > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * - close DFLL loop > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 * - sync PLLX state > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0 */ > >>>>>>> +=C2=A0=C2=A0=C2=A0 if (dfll_pdev) > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 tegra_dfll_resume(dfl= l_pdev, false); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 tegra_cclkg_burst_policy_restore_context(); > >>>>>>> +=C2=A0=C2=A0=C2=A0 fence_udelay(2, clk_base); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 if (dfll_pdev) > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 tegra_dfll_resume(dfl= l_pdev, true); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 parent =3D > >>>>>>> clk_hw_get_parent(__clk_get_hw(clks[TEGRA210_CLK_CCLK_G])); > >>>>>>> +=C2=A0=C2=A0=C2=A0 clk =3D clks[TEGRA210_CLK_PLL_X]; > >>>>>>> +=C2=A0=C2=A0=C2=A0 if (parent !=3D __clk_get_hw(clk)) > >>>>>>> +=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 tegra_clk_sync_state_= pll(__clk_get_hw(clk)); > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 /* Disable PLL_OUT_CPU after DFLL resume */ > >>>>>>> +=C2=A0=C2=A0=C2=A0 tegra_clk_set_pllp_out_cpu(false); > >>>>>>> +} > >>>>>>> + > >>>>>>> =C2=A0=C2=A0 static void tegra210_cpu_clock_suspend(void) > >>>>>>> =C2=A0=C2=A0 { > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 /* switch coresite to clk_m= , save off original source > >>>>>>> */ @@ -3298,6 +3400,11 @@ static void > >>>>>>> tegra210_cpu_clock_resume(void) } > >>>>>>> =C2=A0=C2=A0 #endif > >>>>>>> =C2=A0=C2=A0 +static struct syscore_ops tegra_clk_syscore_ops = =3D { > >>>>>>> +=C2=A0=C2=A0=C2=A0 .suspend =3D tegra210_clk_suspend, > >>>>>>> +=C2=A0=C2=A0=C2=A0 .resume =3D tegra210_clk_resume, > >>>>>>> +}; > >>>>>>> + > >>>>>>> =C2=A0=C2=A0 static struct tegra_cpu_car_ops tegra210_cpu_car_op= s =3D { > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 .wait_for_reset=C2=A0=C2=A0= =C2=A0 =3D tegra210_wait_cpu_in_reset, > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 .disable_clock=C2=A0=C2=A0= =C2=A0 =3D tegra210_disable_cpu_clock, > >>>>>>> @@ -3583,5 +3690,7 @@ static void __init > >>>>>>> tegra210_clock_init(struct device_node *np) > >>>>>>> =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 tegra210_mbist_clk_init(); > >>>>>>> =C2=A0=C2=A0 =C2=A0=C2=A0=C2=A0=C2=A0=C2=A0 tegra_cpu_car_ops = =3D &tegra210_cpu_car_ops; > >>>>>>> + > >>>>>>> +=C2=A0=C2=A0=C2=A0 register_syscore_ops(&tegra_clk_syscore_ops); > >>>>>>> =C2=A0=C2=A0 } =20 > >>>>>> Is it really worthwhile to use syscore_ops for suspend/resume > >>>>>> given that drivers for > >>>>>> won't resume before the CLK driver anyway? Are there any other > >>>>>> options for CLK > >>>>>> suspend/resume? > >>>>>> > >>>>>> I'm also not sure whether PM runtime API could be used at all > >>>>>> in the context of > >>>>>> syscore_ops .. > >>>>>> > >>>>>> Secondly, what about to use generic clk_save_context() / > >>>>>> clk_restore_context() > >>>>>> helpers for the suspend-resume? It looks to me that some other > >>>>>> essential (and proper) > >>>>>> platform driver (soc/tegra/? PMC?) should suspend-resume the > >>>>>> clocks using the generic > >>>>>> CLK Framework API. =20 > >>>>> Clock resume should happen very early to restore peripheral and > >>>>> cpu clocks very early than peripheral drivers resume happens. =20 > >>>> If all peripheral drivers properly requested all of the > >>>> necessary clocks and CLK driver was a platform driver, then I > >>>> guess the probe should have been naturally ordered. But that's > >>>> not very achievable with the currently available infrastructure > >>>> in the kernel, so I'm not arguing that the clocks should be > >>>> explicitly resumed before the users.=20 > >>>>> this patch series uses clk_save_context and clk_restore_context > >>>>> for corresponding divider, pll, pllout.. save and restore > >>>>> context. =20 > >>>> Now I see that indeed this API is utilized in this patch, thank > >>>> you for the clarification. > >>>> =20 > >>>>> But as there is dependency on dfll resume and cpu and pllx > >>>>> clocks restore, couldnt use clk_save_context and > >>>>> clk_restore_context for dfll. > >>>>> > >>>>> So implemented recommended dfll resume sequence in main > >>>>> Tegra210 clock driver along with invoking > >>>>> clk_save_context/clk_restore_context where all other clocks > >>>>> save/restore happens as per clock tree traversal. =20 > >>>> Could you please clarify what part of peripherals clocks is > >>>> required for DFLL's restore? Couldn't DFLL driver be changed to > >>>> avoid that quirkness and thus to make DFLL driver suspend/resume > >>>> the clock? =20 > >>> DFLL source ref_clk and soc_clk need to be restored prior to dfll. > >>> > >>> I see dfllCPU_out parent to CCLK_G first in the clock tree and > >>> dfll_ref and dfll_soc peripheral clocks are not resumed by the > >>> time dfll resume happens first. > >>> > >>> ref_clk and soc_clk source is from pll_p and clock tree has these > >>> registered under pll_p which happens later. > >>> > >>> tegra210_clock_init registers in order plls, peripheral clocks, > >>> super_clk init for cclk_g during clock driver probe and dfll > >>> probe and register happens later. > >>> =20 > >> One more thing, CLDVFS peripheral clock enable is also needed to be > >> enabled to program DFLL Controller and all peripheral clock > >> context is restored only after their PLL sources are restored. > >> > >> DFLL restore involves dfll source clock resume along with CLDVFS > >> periheral clock enable and reset > >> =20 > > I don't quite see why you can't simply add suspend/resume callbacks > > to the CPUFreq driver to: > > > > On suspend: > > 1. Switch CPU to PLLP (or whatever "safe" parent) > > 2. Disable/teardown DFLL > > > > On resume: > > 1. Enable/restore DFLL > > 2. Switch CPU back to DFLL =20 >=20 > dfll runtime suspend/resume are already part of dfll_pm_ops. Don't we=20 > want to use it for suspend/resume as well? Looks like no. Seems runtime PM of that driver is intended solely for the DFLL's clk management. > currently no APIs are shared b/w clk/tegra driver and CPUFreq driver > to invoke dfll suspend/resume in CPUFreq driver >=20 Just add it. Also, please note that CPUFreq driver is optional and thus you may need to switch CPU to a safe parent on clk-core suspend as well in order to resume properly if CPU was running off unsafe parent during boot and CPUFreq driver is disabled in kernel build (or failed to load). The other thing that also need attention is that T124 CPUFreq driver implicitly relies on DFLL driver to be probed first, which is icky.