Received: by 2002:ad5:474a:0:0:0:0:0 with SMTP id i10csp6910860imu; Mon, 3 Dec 2018 04:58:35 -0800 (PST) X-Google-Smtp-Source: AFSGD/XIApmozB4Lho7o0nZci1MeWDHPbcxtRySwK3IVynoA539tys2IWfc377WCBbYdiFBv9wst X-Received: by 2002:a17:902:1102:: with SMTP id d2mr11674505pla.138.1543841915297; Mon, 03 Dec 2018 04:58:35 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1543841915; cv=none; d=google.com; s=arc-20160816; b=Xhdn9Pgg8Gls1YDZwJ7Oi7A3uPSzPz0nRvMVMtSMmbpJ9b8HKdaZNyD9wBQykUWCSa I2SqtiLKwegvQ6zpQS3FfwQ0w8cN4vZ7AdZc6lzLCHmFi2URvpiShWNcybdS5phj8eKW NRlN+SU9Ape3RJGqrBhEJ+XaPD/kSHAzUMgiIQXK6VJxbhNg+WX7EWgPT1pyx1q+x/f5 tHMk22Sjce4OxfsluWhyBDo1ZNq9mXcAa7YSEKFbLWTqJuubK5OpOI9cc4a9Keho5nbx 3hMdrwP43lojGNN5dNzOkRuYCmqoqgXymaYCewCBmWqpalwf4BYcD48OSb2w7fPjUnXd Jf3A== 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:in-reply-to :mime-version:user-agent:date:message-id:from:cc:references:to :subject; bh=hRToAQdaS/6Vu6ja9OISTy3CN57I/j8ChE/bsFOin/U=; b=YoTmFzgs0bBU+rzuEpKk6ttOv8RUyKdq8vT+qTRA7OKtTMmInfVvuf2qDuV0s0cx5v qPeQqN3elb08LLeKHWdtIJbGnsC0/8UovVlUykV2Nv62vdvHVXqN1+n6WVNJFdl3srqt IANLLwAD0y+Nm11DCqhPkSeSp20wIj33T8DO55KDwFk5QE2HtTFUiAYbGPDYKtPrtFrh /mGCJt9P9dR04FpQCFo+m/z43Jnvt/P0wLAn8LmMWE9FGR2MvV942qskZzdeJqBXqWIp u7nmdw1DJ/rnSTkQ+RnRa9XlduHZPKok/N4IE6xXYcuHfzbcGuE3yfK0KW3ktcX0ANrR GDqQ== ARC-Authentication-Results: i=1; mx.google.com; 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 Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id p64si13900201pfa.94.2018.12.03.04.58.20; Mon, 03 Dec 2018 04:58:35 -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; 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 Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726569AbeLCM6e (ORCPT + 99 others); Mon, 3 Dec 2018 07:58:34 -0500 Received: from relay1.mentorg.com ([192.94.38.131]:43097 "EHLO relay1.mentorg.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726386AbeLCM6e (ORCPT ); Mon, 3 Dec 2018 07:58:34 -0500 Received: from nat-ies.mentorg.com ([192.94.31.2] helo=SVR-IES-MBX-04.mgc.mentorg.com) by relay1.mentorg.com with esmtps (TLSv1.2:ECDHE-RSA-AES256-SHA384:256) id 1gTnjY-0004u4-VG from Vladimir_Zapolskiy@mentor.com ; Mon, 03 Dec 2018 04:53:57 -0800 Received: from [137.202.108.125] (137.202.0.90) by SVR-IES-MBX-04.mgc.mentorg.com (139.181.222.4) with Microsoft SMTP Server (TLS) id 15.0.1320.4; Mon, 3 Dec 2018 12:53:52 +0000 Subject: Re: [PATCH linux-next v2 6/6] ASoC: rsnd: add avb clocks To: , , , , References: <20181203112427.18324-1-jiada_wang@mentor.com> CC: , From: Vladimir Zapolskiy Message-ID: Date: Mon, 3 Dec 2018 14:53:52 +0200 User-Agent: Mozilla/5.0 (X11; Linux i686; rv:45.0) Gecko/20100101 Icedove/45.2.0 MIME-Version: 1.0 In-Reply-To: <20181203112427.18324-1-jiada_wang@mentor.com> Content-Type: text/plain; charset="utf-8" Content-Transfer-Encoding: 8bit X-Originating-IP: [137.202.0.90] X-ClientProxiedBy: SVR-IES-MBX-08.mgc.mentorg.com (139.181.222.8) To SVR-IES-MBX-04.mgc.mentorg.com (139.181.222.4) Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi Jiada, On 12/03/2018 01:24 PM, jiada_wang@mentor.com wrote: > From: Jiada Wang > > There are AVB Counter Clocks in ADG, each clock has 12bits integral > and 8 bits fractional dividers which operates with S0D1ϕ clock. > > This patch registers 8 AVB Counter Clocks when clock-cells of > rcar_sound node is 2, > > Signed-off-by: Jiada Wang > --- > sound/soc/sh/rcar/adg.c | 306 +++++++++++++++++++++++++++++++++++++-- > sound/soc/sh/rcar/gen.c | 9 ++ > sound/soc/sh/rcar/rsnd.h | 9 ++ > 3 files changed, 315 insertions(+), 9 deletions(-) > > diff --git a/sound/soc/sh/rcar/adg.c b/sound/soc/sh/rcar/adg.c > index 6768a66588eb..2c03d420ae76 100644 > --- a/sound/soc/sh/rcar/adg.c > +++ b/sound/soc/sh/rcar/adg.c > @@ -5,6 +5,8 @@ > // Copyright (C) 2013 Kuninori Morimoto > > #include > +#include > +#include Drop the inclusion of the header above, see my comment to patch 5/6. > #include "rsnd.h" > > #define CLKA 0 > @@ -21,13 +23,33 @@ > > #define BRRx_MASK(x) (0x3FF & x) > > +#define ADG_CLK_NAME "adg" > +#define AVB_CLK_NAME "avb" Can you please remove two macro above and replace their usage by values in clk_register_avb() function? Also I don't think that it is good to hardcode parent clock name here, likely you should get it in runtime, see __clk_get_name(). > +#define AVB_CLK_NUM 8 > +#define AVB_CLK_NAME_SIZE 10 The one macro above also can be removed in my opinion. > +#define AVB_MAX_RATE 25000000 > +#define AVB_DIV_EN_COM BIT(31) > +#define AVB_DIV_MASK 0x3ffff > +#define AVB_MAX_DIV 0x3ffc0 > + > static struct rsnd_mod_ops adg_ops = { > .name = "adg", > }; > > +struct clk_avb { > + struct clk_hw hw; > + unsigned int idx; > + struct rsnd_mod *mod; > + /* lock reg access */ > + spinlock_t *lock; > +}; > + > +#define to_clk_avb(_hw) container_of(_hw, struct clk_avb, hw) > + > struct rsnd_adg { > struct clk *clk[CLKMAX]; > struct clk *clkout[CLKOUTMAX]; > + struct clk *clkavb[AVB_CLK_NUM]; > struct clk_onecell_data onecell; > struct rsnd_mod mod; > u32 flags; > @@ -37,6 +59,7 @@ struct rsnd_adg { > > int rbga_rate_for_441khz; /* RBGA */ > int rbgb_rate_for_48khz; /* RBGB */ > + spinlock_t avb_lock; > }; > > #define LRCLK_ASYNC (1 << 0) > @@ -408,6 +431,239 @@ void rsnd_adg_clk_control(struct rsnd_priv *priv, int enable) > } > } > > +static struct clk *rsnd_adg_clk_src_twocell_get(struct of_phandle_args *clkspec, > + void *data) > +{ > + unsigned int clkidx = clkspec->args[1]; > + struct rsnd_adg *adg = data; > + const char *type; > + struct clk *clk; > + > + switch (clkspec->args[0]) { > + case ADG_FIX: > + type = "fixed"; Apparently you need 'type' local variable just to print an error message. Please remove the variable and update format strings accordingly. > + if (clkidx >= CLKOUTMAX) { > + pr_err("Invalid %s clock index %u\n", type, > + clkidx); > + return ERR_PTR(-EINVAL); > + } > + clk = adg->clkout[clkidx]; > + break; > + case ADG_AVB: > + type = "avb"; > + if (clkidx >= AVB_CLK_NUM) { > + pr_err("Invalid %s clock index %u\n", type, > + clkidx); > + return ERR_PTR(-EINVAL); > + } > + clk = adg->clkavb[clkidx]; > + break; > + default: > + pr_err("Invalid ADG clock type %u\n", clkspec->args[0]); > + return ERR_PTR(-EINVAL); > + } > + > + return clk; > +} > + > +static void clk_avb_div_write(struct rsnd_mod *mod, u32 data, int idx) unsigned int idx to match a type of 'struct clk_avb' field. > +{ > + switch (idx) { > + case 0: > + rsnd_mod_write(mod, AVB_CLK_DIV0, data); > + break; > + case 1: > + rsnd_mod_write(mod, AVB_CLK_DIV1, data); > + break; > + case 2: > + rsnd_mod_write(mod, AVB_CLK_DIV2, data); > + break; > + case 3: > + rsnd_mod_write(mod, AVB_CLK_DIV3, data); > + break; > + case 4: > + rsnd_mod_write(mod, AVB_CLK_DIV4, data); > + break; > + case 5: > + rsnd_mod_write(mod, AVB_CLK_DIV5, data); > + break; > + case 6: > + rsnd_mod_write(mod, AVB_CLK_DIV6, data); > + break; > + case 7: > + rsnd_mod_write(mod, AVB_CLK_DIV7, data); > + break; > + } > +} > + > +static u32 clk_avb_div_read(struct rsnd_mod *mod, int idx) unsigned int idx to match a type of 'struct clk_avb' field. > +{ > + u32 val = 0; > + > + switch (idx) { > + case 0: > + val = rsnd_mod_read(mod, AVB_CLK_DIV0); > + break; > + case 1: > + val = rsnd_mod_read(mod, AVB_CLK_DIV1); > + break; > + case 2: > + val = rsnd_mod_read(mod, AVB_CLK_DIV2); > + break; > + case 3: > + val = rsnd_mod_read(mod, AVB_CLK_DIV3); > + break; > + case 4: > + val = rsnd_mod_read(mod, AVB_CLK_DIV4); > + break; > + case 5: > + val = rsnd_mod_read(mod, AVB_CLK_DIV5); > + break; > + case 6: > + val = rsnd_mod_read(mod, AVB_CLK_DIV6); > + break; > + case 7: > + val = rsnd_mod_read(mod, AVB_CLK_DIV7); > + break; > + } > + > + return val; > +} Apparently the macro nature of rsnd_mod_read() and rsnd_mod_write() does not allow to define a helper mapping function from index into RSND_REG_AVB_CLK_DIVx, okay... > + > +static int clk_avb_is_enabled(struct clk_hw *hw) > +{ > + struct clk_avb *avb = to_clk_avb(hw); > + > + return rsnd_mod_read(avb->mod, AVB_CLK_CONFIG) & BIT(avb->idx); > +} > + > +static int clk_avb_enabledisable(struct clk_hw *hw, int enable) > +{ > + struct clk_avb *avb = to_clk_avb(hw); > + u32 val; > + > + spin_lock(avb->lock); > + > + val = rsnd_mod_read(avb->mod, AVB_CLK_CONFIG); > + if (enable) > + val |= BIT(avb->idx); > + else > + val &= ~BIT(avb->idx); > + rsnd_mod_write(avb->mod, AVB_CLK_CONFIG, val); > + > + spin_unlock(avb->lock); > + > + return 0; > +} > + > +static int clk_avb_enable(struct clk_hw *hw) > +{ > + return clk_avb_enabledisable(hw, 1); > +} > + > +static void clk_avb_disable(struct clk_hw *hw) > +{ > + clk_avb_enabledisable(hw, 0); > +} > + > +static unsigned long clk_avb_recalc_rate(struct clk_hw *hw, > + unsigned long parent_rate) > +{ > + struct clk_avb *avb = to_clk_avb(hw); > + u32 div; > + > + div = clk_avb_div_read(avb->mod, avb->idx) & AVB_DIV_MASK; > + if (!div) > + return parent_rate; > + > + return parent_rate * 32 / div; > +} > + > +static unsigned int clk_avb_calc_div(unsigned long rate, > + unsigned long parent_rate) > +{ > + unsigned int div; > + > + if (!rate) > + rate = 1; > + > + if (rate > AVB_MAX_RATE) > + rate = AVB_MAX_RATE; > + > + div = DIV_ROUND_CLOSEST(parent_rate * 32, rate); > + > + if (div > AVB_MAX_DIV) > + div = AVB_MAX_DIV; > + > + return div; > +} > + > +static long clk_avb_round_rate(struct clk_hw *hw, unsigned long rate, > + unsigned long *parent_rate) > +{ > + unsigned int div = clk_avb_calc_div(rate, *parent_rate); > + > + return (*parent_rate * 32) / div; > +} > + > +static int clk_avb_set_rate(struct clk_hw *hw, unsigned long rate, > + unsigned long parent_rate) > +{ > + struct clk_avb *avb = to_clk_avb(hw); > + unsigned int div = clk_avb_calc_div(rate, parent_rate); > + u32 val; > + > + val = clk_avb_div_read(avb->mod, avb->idx) & ~AVB_DIV_MASK; > + clk_avb_div_write(avb->mod, val | div, avb->idx); > + > + return 0; > +} > + > +static const struct clk_ops clk_avb_ops = { > + .enable = clk_avb_enable, > + .disable = clk_avb_disable, > + .is_enabled = clk_avb_is_enabled, > + .recalc_rate = clk_avb_recalc_rate, > + .round_rate = clk_avb_round_rate, > + .set_rate = clk_avb_set_rate, > +}; > + > +static struct clk *clk_register_avb(struct device *dev, struct rsnd_mod *mod, > + unsigned int id, spinlock_t *lock) > +{ > + struct clk_init_data init; > + struct clk_avb *avb; > + struct clk *clk; > + char name[AVB_CLK_NAME_SIZE]; > + const char *parent_name = ADG_CLK_NAME; > + > + avb = devm_kzalloc(dev, sizeof(*avb), GFP_KERNEL); > + if (!avb) > + return ERR_PTR(-ENOMEM); > + > + snprintf(name, AVB_CLK_NAME_SIZE, "%s%u", AVB_CLK_NAME, id); > + > + avb->idx = id; > + avb->lock = lock; > + avb->mod = mod; > + > + /* Register the clock. */ > + init.name = name; > + init.ops = &clk_avb_ops; > + init.flags = CLK_IS_BASIC; > + init.parent_names = &parent_name; > + init.num_parents = 1; > + > + avb->hw.init = &init; > + > + /* init DIV to a valid state */ > + clk_avb_div_write(avb->mod, avb->idx, AVB_MAX_DIV); > + > + clk = devm_clk_register(dev, &avb->hw); > + > + return clk; > +} > + > static void rsnd_adg_get_clkin(struct rsnd_priv *priv, > struct rsnd_adg *adg) > { > @@ -436,6 +692,7 @@ static void rsnd_adg_get_clkout(struct rsnd_priv *priv, > unsigned long req_48kHz_rate, req_441kHz_rate; > int i, req_size; > const char *parent_clk_name = NULL; > + struct rsnd_mod *mod = rsnd_mod_get(adg); > static const char * const clkout_name[] = { > [CLKOUT] = "audio_clkout", > [CLKOUT1] = "audio_clkout1", > @@ -540,21 +797,23 @@ static void rsnd_adg_get_clkout(struct rsnd_priv *priv, > */ > > of_property_read_u32(np, "#clock-cells", &count); > - /* > - * for clkout > - */ > - if (!count) { > + > + switch (count) { > + case 0: > + /* > + * for clkout > + */ > clk = clk_register_fixed_rate(dev, clkout_name[CLKOUT], > parent_clk_name, 0, req_rate[0]); > if (!IS_ERR(clk)) { > adg->clkout[CLKOUT] = clk; > of_clk_add_provider(np, of_clk_src_simple_get, clk); > } > - } > - /* > - * for clkout0/1/2/3 > - */ > - else { > + break; > + case 1: > + /* > + * for clkout0/1/2/3 > + */ > for (i = 0; i < CLKOUTMAX; i++) { > clk = clk_register_fixed_rate(dev, clkout_name[i], > parent_clk_name, 0, > @@ -566,6 +825,33 @@ static void rsnd_adg_get_clkout(struct rsnd_priv *priv, > adg->onecell.clk_num = CLKOUTMAX; > of_clk_add_provider(np, of_clk_src_onecell_get, > &adg->onecell); > + break; > + case 2: > + /* > + * for clkout0/1/2/3 and avb clocks > + */ > + for (i = 0; i < CLKOUTMAX; i++) { > + clk = clk_register_fixed_rate(dev, clkout_name[i], > + parent_clk_name, 0, > + req_rate[0]); > + if (!IS_ERR(clk)) > + adg->clkout[i] = clk; > + } > + > + for (i = 0; i < AVB_CLK_NUM; i++) { > + clk = clk_register_avb(dev, mod, i, &adg->avb_lock); > + if (!IS_ERR(clk)) > + adg->clkavb[i] = clk; > + } > + > + of_clk_add_provider(np, rsnd_adg_clk_src_twocell_get, adg); > + > + rsnd_mod_write(mod, AVB_CLK_CONFIG, AVB_DIV_EN_COM); > + > + break; > + default: > + dev_err(dev, "Invalid clock-cell %d\n", count); > + break; > } > > rsnd_adg_get_clkout_end: > @@ -612,6 +898,8 @@ int rsnd_adg_probe(struct rsnd_priv *priv) > if (!adg) > return -ENOMEM; > > + spin_lock_init(&adg->avb_lock); > + > ret = rsnd_mod_init(priv, &adg->mod, &adg_ops, > NULL, 0, 0); > if (ret) -- Best wishes, Vladimir