Received: by 2002:a05:6358:9144:b0:117:f937:c515 with SMTP id r4csp5097314rwr; Mon, 8 May 2023 18:20:39 -0700 (PDT) X-Google-Smtp-Source: ACHHUZ5vRVbdnrt8WnWu08NVF1ct8IJnFkIP7aUHs4I/i4VQZEO9PuBiAzMWsTBmJ1EYi8Sd81cN X-Received: by 2002:a17:902:ea01:b0:1ac:6d4c:c24b with SMTP id s1-20020a170902ea0100b001ac6d4cc24bmr8368356plg.3.1683595239606; Mon, 08 May 2023 18:20:39 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1683595239; cv=none; d=google.com; s=arc-20160816; b=g1NTR6lbk+mUXjawjMKCMFERY1FyUNDoVrh3QQLhulbBNPtD5minOETEsFXXgHKc0C a/EJwNSeQ9XuWAiVANIDCOngG5Bdj1kOofOL7Ovo9901a39nTW8rYXtRCc7mZUG608R7 XD/0OXWeD4/VtGCiYMZDWlF80lu/ykXDpBthJIScl7Rg1v8o4+htDURwbRijU913pirl kH1E/6323vPi3NAAbrplFANzgD0RsdEyB9n295ka5B8SCHsy+4WykFaUdVs/K4IWVVsO y9YJEsfX0wDzzqZ7PtD2lsLoXVCcQCYcmxauA6XEq2U4GYM87SWGwgp7+topKqV5Pt7l ZjYg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:in-reply-to :content-language:references:cc:to:subject:from:user-agent :mime-version:date:message-id:dkim-signature; bh=0WhWKe5Ry9BFBJzgo91hGa8wb9flJdaliFeOwksUl9Q=; b=JwaC4lgZzpqurPyBTInMefWdo0hHxxkadnfjG7fGit7dvN+OmDnnA7KeUy3VZtYCcO iKbNMsG77TLg2yiOw6r/mcB0qorKGBKztEpJ7eErD9mHEgvj3lH1pcf5MIXF+9+SMOaa XhaYwZBCKl7vn1364qSZBdmHTA4tcAWIdai1fUQMAfcAAOFKVcg65DTBJ5JzLw31vzXp vBkpFndbbFJKkA4vzf+RCT3zNxBxDT1D2DnYGk2u8W1GCLjhpg+nqJM20NJEKiaKPDqV 9lnknjScMu+3tvNYAXIoAp5U+CLP1CuwnyVK732nWxsAZjINe9MvSHQstozqTIi+xwVD J+Nw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20221208 header.b=HPuEVTh3; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 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 out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id m2-20020a170902db0200b001a97e24eb34si314839plx.201.2023.05.08.18.20.24; Mon, 08 May 2023 18:20:39 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; dkim=pass header.i=@gmail.com header.s=20221208 header.b=HPuEVTh3; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 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 S229539AbjEIA65 (ORCPT + 99 others); Mon, 8 May 2023 20:58:57 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:51084 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232812AbjEIA6z (ORCPT ); Mon, 8 May 2023 20:58:55 -0400 Received: from mail-pf1-x430.google.com (mail-pf1-x430.google.com [IPv6:2607:f8b0:4864:20::430]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 02BB259E8; Mon, 8 May 2023 17:58:46 -0700 (PDT) Received: by mail-pf1-x430.google.com with SMTP id d2e1a72fcca58-644d9bf05b7so1764334b3a.3; Mon, 08 May 2023 17:58:45 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20221208; t=1683593925; x=1686185925; h=content-transfer-encoding:in-reply-to:content-language:references :cc:to:subject:from:user-agent:mime-version:date:message-id:from:to :cc:subject:date:message-id:reply-to; bh=0WhWKe5Ry9BFBJzgo91hGa8wb9flJdaliFeOwksUl9Q=; b=HPuEVTh3BQ4mQ9/WHxjCA/BEhfhpnovyS7vh52JNXg3eMu6TB9jt2vkLAWJSMzA6MH 5CL2tkpII2im4hF4MpjwCQpRyhZyvE6OJ+/Yvu05G0zb3JFgedXYuSlGx9rc6JYTgXf6 6JQenRmgCnbhbx0SL/GuxOMxE9U0Yil33znelXg+hG6PN6VVogma4HLiJQu+bZZjeoqE RIcyRdHQD4LXVg75fFMXDvC2wgTVG6BUfUA4SrWhkInqtVSezdo6zAygokq4zx8PTvA2 lNroIayegSbgWGjRj4LazFJTXjsZYqZ4bPdAIJS8Z2jr9uplj0BYlNve54OZYO3lDc6L iWYg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20221208; t=1683593925; x=1686185925; h=content-transfer-encoding:in-reply-to:content-language:references :cc:to:subject:from:user-agent:mime-version:date:message-id :x-gm-message-state:from:to:cc:subject:date:message-id:reply-to; bh=0WhWKe5Ry9BFBJzgo91hGa8wb9flJdaliFeOwksUl9Q=; b=ZHIEPakg3R8HDvDvWPc/fuZKrZzhCBXelyvxVWMBzsAzcpPWnWVWEiJZ06qQfZgOi5 O+wodHMzVds7B6nYAdUEHjqCHbDpLaixOTKmVBaBEez/XXQB7D1VmVKRkzWxakBW+aX5 7T1FMLdgzoLHx4r0aUCn82e7uP2c7Nnju2fwvsrNqQx/JzDVs3OtNPr8aS06SiSvz2R+ weWBA1fT91JkpbkhfPtw4021oAENGoN+C9u5bHL/dWv7GOY79JmdSeBvvxEYL8cSJrOh Rg2FtS1evcqjMeXk05NnRvDgL8+5y8ULFR16PSq3hRzWZHUWXPXxJR6r+dYz+negLK65 prJQ== X-Gm-Message-State: AC+VfDyh3noJybi0amNzn/R6VExLtArihC80a/CADUj8IFBUUCnxCpht a+ftxLxZTxKwUkv1HutGH5I= X-Received: by 2002:a05:6a00:cca:b0:643:7fcf:836d with SMTP id b10-20020a056a000cca00b006437fcf836dmr17455413pfv.25.1683593925198; Mon, 08 May 2023 17:58:45 -0700 (PDT) Received: from [172.19.1.47] (60-250-192-107.hinet-ip.hinet.net. [60.250.192.107]) by smtp.gmail.com with ESMTPSA id j10-20020a62e90a000000b0063f16daf7dbsm551288pfh.55.2023.05.08.17.58.41 (version=TLS1_3 cipher=TLS_AES_128_GCM_SHA256 bits=128/128); Mon, 08 May 2023 17:58:44 -0700 (PDT) Message-ID: <0d715170-e9c9-ce1f-3105-5ba5d505b655@gmail.com> Date: Tue, 9 May 2023 08:58:39 +0800 MIME-Version: 1.0 User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:102.0) Gecko/20100101 Thunderbird/102.10.1 From: Jacky Huang Subject: Re: [PATCH v10 08/10] clk: nuvoton: Add clock driver for ma35d1 clock controller To: =?UTF-8?Q?Ilpo_J=c3=a4rvinen?= Cc: robh+dt@kernel.org, krzysztof.kozlowski+dt@linaro.org, lee@kernel.org, mturquette@baylibre.com, sboyd@kernel.org, p.zabel@pengutronix.de, Greg Kroah-Hartman , Jiri Slaby , tmaimon77@gmail.com, catalin.marinas@arm.com, will@kernel.org, devicetree@vger.kernel.org, linux-clk@vger.kernel.org, LKML , linux-arm-kernel@lists.infradead.org, linux-serial , arnd@arndb.de, schung@nuvoton.com, mjchen@nuvoton.com, Jacky Huang , Krzysztof Kozlowski References: <20230508025936.36776-1-ychuang570808@gmail.com> <20230508025936.36776-9-ychuang570808@gmail.com> Content-Language: en-US In-Reply-To: Content-Type: text/plain; charset=UTF-8; format=flowed Content-Transfer-Encoding: 8bit X-Spam-Status: No, score=-3.7 required=5.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,FREEMAIL_ENVFROM_END_DIGIT, FREEMAIL_FROM,NICE_REPLY_A,RCVD_IN_DNSWL_NONE,SPF_HELO_NONE,SPF_PASS, T_SCC_BODY_TEXT_LINE,URIBL_BLOCKED autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Dear Ilpo, Thank you for the advice. On 2023/5/8 下午 06:52, Ilpo Järvinen wrote: > On Mon, 8 May 2023, Jacky Huang wrote: > >> From: Jacky Huang >> >> The clock controller generates clocks for the whole chip, including >> system clocks and all peripheral clocks. This driver support ma35d1 >> clock gating, divider, and individual PLL configuration. >> >> There are 6 PLLs in ma35d1 SoC: >> - CA-PLL for the two Cortex-A35 CPU clock >> - SYS-PLL for system bus, which comes from the companion MCU >> and cannot be programmed by clock controller. >> - DDR-PLL for DDR >> - EPLL for GMAC and GFX, Display, and VDEC IPs. >> - VPLL for video output pixel clock >> - APLL for SDHC, I2S audio, and other IPs. >> CA-PLL has only one operation mode. >> DDR-PLL, EPLL, VPLL, and APLL are advanced PLLs which have 3 >> operation modes: integer mode, fraction mode, and spread specturm mode. >> >> Signed-off-by: Jacky Huang >> Acked-by: Krzysztof Kozlowski >> --- > Hi, > > A few small things still below. > >> diff --git a/drivers/clk/nuvoton/clk-ma35d1-divider.c b/drivers/clk/nuvoton/clk-ma35d1-divider.c >> new file mode 100644 >> index 000000000000..0d4d8186a85c >> --- /dev/null >> +++ b/drivers/clk/nuvoton/clk-ma35d1-divider.c >> @@ -0,0 +1,140 @@ >> +// SPDX-License-Identifier: GPL-2.0-only >> +/* >> + * Copyright (C) 2023 Nuvoton Technology Corp. >> + * Author: Chi-Fang Li >> + */ >> + >> +#include >> +#include >> +#include >> +#include >> + >> +struct ma35d1_adc_clk_div { >> + struct clk_hw hw; >> + void __iomem *reg; >> + u8 shift; >> + u8 width; >> + u32 mask; >> + const struct clk_div_table *table; >> + /* protects concurrent access to clock divider registers */ >> + spinlock_t *lock; >> +}; >> + >> +struct clk_hw *ma35d1_reg_adc_clkdiv(struct device *dev, const char *name, >> + struct clk_hw *parent_hw, spinlock_t *lock, >> + unsigned long flags, void __iomem *reg, >> + u8 shift, u8 width, u32 mask_bit); >> + >> +static inline struct ma35d1_adc_clk_div *to_ma35d1_adc_clk_div(struct clk_hw *_hw) >> +{ >> + return container_of(_hw, struct ma35d1_adc_clk_div, hw); >> +} >> + >> +static inline unsigned long ma35d1_clkdiv_recalc_rate(struct clk_hw *hw, >> + unsigned long parent_rate) >> +{ >> + unsigned int val; >> + struct ma35d1_adc_clk_div *dclk = to_ma35d1_adc_clk_div(hw); >> + >> + val = readl_relaxed(dclk->reg) >> dclk->shift; >> + val &= clk_div_mask(dclk->width); >> + val += 1; >> + return divider_recalc_rate(hw, parent_rate, val, dclk->table, >> + CLK_DIVIDER_ROUND_CLOSEST, dclk->width); >> +} >> + >> +static inline long ma35d1_clkdiv_round_rate(struct clk_hw *hw, >> + unsigned long rate, >> + unsigned long *prate) >> +{ >> + struct ma35d1_adc_clk_div *dclk = to_ma35d1_adc_clk_div(hw); >> + >> + return divider_round_rate(hw, rate, prate, dclk->table, >> + dclk->width, CLK_DIVIDER_ROUND_CLOSEST); >> +} >> + >> +static inline int ma35d1_clkdiv_set_rate(struct clk_hw *hw, >> + unsigned long rate, >> + unsigned long parent_rate) >> +{ >> + int value; >> + unsigned long flags = 0; >> + u32 data; >> + struct ma35d1_adc_clk_div *dclk = to_ma35d1_adc_clk_div(hw); >> + >> + value = divider_get_val(rate, parent_rate, dclk->table, >> + dclk->width, CLK_DIVIDER_ROUND_CLOSEST); >> + >> + spin_lock_irqsave(dclk->lock, flags); >> + >> + data = readl_relaxed(dclk->reg); >> + data &= ~(clk_div_mask(dclk->width) << dclk->shift); >> + data |= (value - 1) << dclk->shift; >> + data |= dclk->mask; >> + writel_relaxed(data, dclk->reg); >> + >> + spin_unlock_irqrestore(dclk->lock, flags); >> + return 0; >> +} >> + >> +static const struct clk_ops ma35d1_adc_clkdiv_ops = { >> + .recalc_rate = ma35d1_clkdiv_recalc_rate, >> + .round_rate = ma35d1_clkdiv_round_rate, >> + .set_rate = ma35d1_clkdiv_set_rate, >> +}; >> + >> +struct clk_hw *ma35d1_reg_adc_clkdiv(struct device *dev, const char *name, >> + struct clk_hw *parent_hw, spinlock_t *lock, >> + unsigned long flags, void __iomem *reg, >> + u8 shift, u8 width, u32 mask_bit) >> +{ >> + struct ma35d1_adc_clk_div *div; >> + struct clk_init_data init; >> + struct clk_div_table *table; >> + struct clk_parent_data pdata = { .index = 0 }; >> + u32 max_div, min_div; >> + struct clk_hw *hw; >> + int ret; >> + int i; >> + >> + div = devm_kzalloc(dev, sizeof(*div), GFP_KERNEL); >> + if (!div) >> + return ERR_PTR(-ENOMEM); >> + >> + max_div = clk_div_mask(width) + 1; >> + min_div = 1; >> + >> + table = devm_kcalloc(dev, max_div + 1, sizeof(*table), GFP_KERNEL); >> + if (!table) >> + return ERR_PTR(-ENOMEM); >> + >> + for (i = 0; i < max_div; i++) { >> + table[i].val = min_div + i; >> + table[i].div = 2 * table[i].val; >> + } >> + table[max_div].val = 0; >> + table[max_div].div = 0; >> + >> + memset(&init, 0, sizeof(init)); >> + init.name = name; >> + init.ops = &ma35d1_adc_clkdiv_ops; >> + init.flags |= flags; >> + pdata.hw = parent_hw; >> + init.parent_data = &pdata; >> + init.num_parents = 1; >> + >> + div->reg = reg; >> + div->shift = shift; >> + div->width = width; >> + div->mask = mask_bit ? BIT(mask_bit) : 0; >> + div->lock = lock; >> + div->hw.init = &init; >> + div->table = table; >> + >> + hw = &div->hw; >> + ret = devm_clk_hw_register(dev, hw); >> + if (ret) >> + return ERR_PTR(ret); >> + return hw; >> +} >> +EXPORT_SYMBOL_GPL(ma35d1_reg_adc_clkdiv); >> diff --git a/drivers/clk/nuvoton/clk-ma35d1-pll.c b/drivers/clk/nuvoton/clk-ma35d1-pll.c >> new file mode 100644 >> index 000000000000..a9981db0a54a >> --- /dev/null >> +++ b/drivers/clk/nuvoton/clk-ma35d1-pll.c >> @@ -0,0 +1,365 @@ >> +// SPDX-License-Identifier: GPL-2.0-only >> +/* >> + * Copyright (C) 2023 Nuvoton Technology Corp. >> + * Author: Chi-Fang Li >> + */ >> + >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> + >> +/* PLL frequency limits */ >> +#define PLL_FREF_MAX_FREQ (200 * HZ_PER_MHZ) >> +#define PLL_FREF_MIN_FREQ (1 * HZ_PER_MHZ) >> +#define PLL_FREF_M_MAX_FREQ (40 * HZ_PER_MHZ) >> +#define PLL_FREF_M_MIN_FREQ (10 * HZ_PER_MHZ) >> +#define PLL_FCLK_MAX_FREQ (2400 * HZ_PER_MHZ) >> +#define PLL_FCLK_MIN_FREQ (600 * HZ_PER_MHZ) >> +#define PLL_FCLKO_MAX_FREQ (2400 * HZ_PER_MHZ) >> +#define PLL_FCLKO_MIN_FREQ (85700 * HZ_PER_KHZ) >> +#define PLL_SS_RATE 0x77 >> +#define PLL_SLOPE 0x58CFA >> + >> +#define REG_PLL_CTL0_OFFSET 0x0 >> +#define REG_PLL_CTL1_OFFSET 0x4 >> +#define REG_PLL_CTL2_OFFSET 0x8 >> + >> +/* bit fields for REG_CLK_PLL0CTL0, which is SMIC PLL design */ >> +#define SPLL0_CTL0_FBDIV GENMASK(7, 0) >> +#define SPLL0_CTL0_INDIV GENMASK(11, 8) >> +#define SPLL0_CTL0_OUTDIV GENMASK(13, 12) >> +#define SPLL0_CTL0_PD BIT(16) >> +#define SPLL0_CTL0_BP BIT(17) >> + >> +/* bit fields for REG_CLK_PLLxCTL0 ~ REG_CLK_PLLxCTL2, where x = 2 ~ 5 */ >> +#define PLL_CTL0_FBDIV GENMASK(10, 0) >> +#define PLL_CTL0_INDIV GENMASK(17, 12) >> +#define PLL_CTL0_MODE GENMASK(19, 18) >> +#define PLL_CTL0_SSRATE GENMASK(30, 20) >> +#define PLL_CTL1_PD BIT(0) >> +#define PLL_CTL1_BP BIT(1) >> +#define PLL_CTL1_OUTDIV GENMASK(6, 4) >> +#define PLL_CTL1_FRAC GENMASK(31, 24) >> +#define PLL_CTL2_SLOPE GENMASK(23, 0) >> + >> +#define INDIV_MIN 1 >> +#define INDIV_MAX 63 >> +#define FBDIV_MIN 16 >> +#define FBDIV_MAX 2047 >> +#define FBDIV_FRAC_MIN 1600 >> +#define FBDIV_FRAC_MAX 204700 >> +#define OUTDIV_MIN 1 >> +#define OUTDIV_MAX 7 >> + >> +#define PLL_MODE_INT 0 >> +#define PLL_MODE_FRAC 1 >> +#define PLL_MODE_SS 2 >> + >> +struct ma35d1_clk_pll { >> + struct clk_hw hw; >> + u32 id; >> + u8 mode; >> + void __iomem *ctl0_base; >> + void __iomem *ctl1_base; >> + void __iomem *ctl2_base; >> +}; >> + >> +struct clk_hw *ma35d1_reg_clk_pll(struct device *dev, u32 id, u8 u8mode, const char *name, >> + struct clk_hw *parent_hw, void __iomem *base); >> + >> +static inline struct ma35d1_clk_pll *to_ma35d1_clk_pll(struct clk_hw *_hw) >> +{ >> + return container_of(_hw, struct ma35d1_clk_pll, hw); >> +} >> + >> +static unsigned long ma35d1_calc_smic_pll_freq(u32 pll0_ctl0, >> + unsigned long parent_rate) >> +{ >> + const u32 clk_div_table[] = { 1, 2, 4, 8 }; > This is just 1 << n so the table doesn't seem necessary. > >> + u32 m, n, p, outdiv; >> + u64 pll_freq; >> + >> + if (pll0_ctl0 & SPLL0_CTL0_BP) >> + return parent_rate; >> + >> + n = FIELD_GET(SPLL0_CTL0_FBDIV, pll0_ctl0); >> + m = FIELD_GET(SPLL0_CTL0_INDIV, pll0_ctl0); >> + p = FIELD_GET(SPLL0_CTL0_OUTDIV, pll0_ctl0); >> + outdiv = clk_div_table[p]; >> + pll_freq = (u64)parent_rate * n; >> + div_u64(pll_freq, m * outdiv); >> + return pll_freq; >> +} >> + >> +static unsigned long ma35d1_calc_pll_freq(u8 mode, u32 *reg_ctl, unsigned long parent_rate) >> +{ >> + unsigned long pll_freq, x; >> + u32 m, n, p; >> + >> + if (reg_ctl[1] & PLL_CTL1_BP) >> + return parent_rate; >> + >> + n = FIELD_GET(PLL_CTL0_FBDIV, reg_ctl[0]); >> + m = FIELD_GET(PLL_CTL0_INDIV, reg_ctl[0]); >> + p = FIELD_GET(PLL_CTL1_OUTDIV, reg_ctl[1]); >> + >> + if (mode == PLL_MODE_INT) { >> + pll_freq = (u64)parent_rate * n; >> + div_u64(pll_freq, m * p); >> + } else { >> + x = FIELD_GET(PLL_CTL1_FRAC, reg_ctl[1]); >> + /* 2 decimal places floating to integer (ex. 1.23 to 123) */ >> + n = n * 100 + ((x * 100) / FIELD_MAX(PLL_CTL1_FRAC)); >> + >> + /* pll_freq = parent_rate * n / 100 / m / p */ > Repeats what the code does, drop. I will remove it. >> + pll_freq = div_u64(parent_rate * n, 100 * m * p); >> + } >> + return pll_freq; >> +} >> + >> +static int ma35d1_pll_find_closest(struct ma35d1_clk_pll *pll, unsigned long rate, >> + unsigned long parent_rate, u32 *reg_ctl, >> + unsigned long *freq) >> +{ >> + unsigned long min_diff = ULONG_MAX; >> + int fbdiv_min, fbdiv_max; >> + int p, m, n; >> + >> + *freq = 0; >> + if (rate < PLL_FCLKO_MIN_FREQ || rate > PLL_FCLKO_MAX_FREQ) >> + return -EINVAL; >> + >> + if (pll->mode == PLL_MODE_INT) { >> + fbdiv_min = FBDIV_MIN; >> + fbdiv_max = FBDIV_MAX; >> + } else { >> + fbdiv_min = FBDIV_FRAC_MIN; >> + fbdiv_max = FBDIV_FRAC_MAX; >> + } >> + >> + for (m = INDIV_MIN; m <= INDIV_MAX; m++) { >> + for (n = fbdiv_min; n <= fbdiv_max; n++) { >> + for (p = OUTDIV_MIN; p <= OUTDIV_MAX; p++) { >> + unsigned long tmp, fout, fclk, diff; >> + >> + tmp = div_u64(parent_rate, m); >> + if (tmp < PLL_FREF_M_MIN_FREQ || >> + tmp > PLL_FREF_M_MAX_FREQ) >> + continue; /* constrain */ >> + >> + fclk = div_u64(parent_rate * n, m); >> + /* for 2 decimal places */ >> + if (pll->mode != PLL_MODE_INT) >> + fclk = div_u64(fclk, 100); >> + >> + if (fclk < PLL_FCLK_MIN_FREQ || >> + fclk > PLL_FCLK_MAX_FREQ) >> + continue; /* constrain */ >> + >> + fout = div_u64(fclk, p); >> + if (fout < PLL_FCLKO_MIN_FREQ || >> + fout > PLL_FCLKO_MAX_FREQ) >> + continue; /* constrain */ >> + >> + diff = abs(rate - fout); >> + if (diff < min_diff) { >> + reg_ctl[0] = FIELD_PREP(PLL_CTL0_INDIV, m) | >> + FIELD_PREP(PLL_CTL0_FBDIV, n); >> + reg_ctl[1] = FIELD_PREP(PLL_CTL1_OUTDIV, p); >> + *freq = fout; >> + min_diff = diff; >> + if (min_diff == 0) >> + break; >> + } >> + } >> + } >> + } >> + if (*freq == 0) >> + return -EINVAL; /* cannot find even one valid setting */ >> + return 0; >> +} >> + >> +static int ma35d1_clk_pll_set_rate(struct clk_hw *hw, unsigned long rate, >> + unsigned long parent_rate) >> +{ >> + struct ma35d1_clk_pll *pll = to_ma35d1_clk_pll(hw); >> + u32 reg_ctl[3] = { 0 }; >> + unsigned long pll_freq; >> + int ret; >> + >> + if (parent_rate < PLL_FREF_MIN_FREQ || >> + parent_rate > PLL_FREF_MAX_FREQ) > One line. I will fix it. >> + return -EINVAL; >> + >> + ret = ma35d1_pll_find_closest(pll, rate, parent_rate, reg_ctl, &pll_freq); >> + if (ret != 0) >> + return ret; >> + >> + switch (pll->mode) { >> + case PLL_MODE_INT: >> + reg_ctl[0] |= FIELD_PREP(PLL_CTL0_MODE, PLL_MODE_INT); >> + break; >> + case PLL_MODE_FRAC: >> + reg_ctl[0] |= FIELD_PREP(PLL_CTL0_MODE, PLL_MODE_FRAC); >> + break; >> + case PLL_MODE_SS: >> + reg_ctl[0] |= FIELD_PREP(PLL_CTL0_MODE, PLL_MODE_SS) | >> + FIELD_PREP(PLL_CTL0_SSRATE, PLL_SS_RATE); >> + reg_ctl[2] = FIELD_PREP(PLL_CTL2_SLOPE, PLL_SLOPE); >> + break; >> + } >> + reg_ctl[1] |= PLL_CTL1_PD; >> + >> + writel_relaxed(reg_ctl[0], pll->ctl0_base); >> + writel_relaxed(reg_ctl[1], pll->ctl1_base); >> + writel_relaxed(reg_ctl[2], pll->ctl2_base); >> + return 0; >> +} >> + >> +static unsigned long ma35d1_clk_pll_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) >> +{ >> + struct ma35d1_clk_pll *pll = to_ma35d1_clk_pll(hw); >> + u32 reg_ctl[3]; >> + unsigned long pll_freq; >> + >> + if (parent_rate < PLL_FREF_MIN_FREQ || parent_rate > PLL_FREF_MAX_FREQ) >> + return 0; >> + >> + switch (pll->id) { >> + case CAPLL: >> + reg_ctl[0] = readl_relaxed(pll->ctl0_base); >> + pll_freq = ma35d1_calc_smic_pll_freq(reg_ctl[0], parent_rate); >> + return pll_freq; >> + case DDRPLL: >> + case APLL: >> + case EPLL: >> + case VPLL: >> + reg_ctl[0] = readl_relaxed(pll->ctl0_base); >> + reg_ctl[1] = readl_relaxed(pll->ctl1_base); >> + pll_freq = ma35d1_calc_pll_freq(pll->mode, reg_ctl, parent_rate); >> + return pll_freq; >> + } >> + return 0; >> +} >> + >> +static long ma35d1_clk_pll_round_rate(struct clk_hw *hw, unsigned long rate, >> + unsigned long *parent_rate) >> +{ >> + struct ma35d1_clk_pll *pll = to_ma35d1_clk_pll(hw); >> + u32 reg_ctl[3] = { 0 }; >> + unsigned long pll_freq; >> + long ret; >> + >> + if (*parent_rate < PLL_FREF_MIN_FREQ || *parent_rate > PLL_FREF_MAX_FREQ) >> + return -EINVAL; >> + >> + ret = ma35d1_pll_find_closest(pll, rate, *parent_rate, reg_ctl, &pll_freq); >> + if (ret != 0) > < 0 I will fix it. >> + return ret; >> + >> + switch (pll->id) { >> + case CAPLL: >> + reg_ctl[0] = readl_relaxed(pll->ctl0_base); >> + pll_freq = ma35d1_calc_smic_pll_freq(reg_ctl[0], *parent_rate); >> + return pll_freq; >> + case DDRPLL: >> + case APLL: >> + case EPLL: >> + case VPLL: >> + reg_ctl[0] = readl_relaxed(pll->ctl0_base); >> + reg_ctl[1] = readl_relaxed(pll->ctl1_base); >> + pll_freq = ma35d1_calc_pll_freq(pll->mode, reg_ctl, *parent_rate); >> + return pll_freq; >> + } >> + return 0; >> +} >> + >> +static int ma35d1_clk_pll_is_prepared(struct clk_hw *hw) >> +{ >> + struct ma35d1_clk_pll *pll = to_ma35d1_clk_pll(hw); >> + u32 val = readl_relaxed(pll->ctl1_base); >> + >> + return !(val & PLL_CTL1_PD); >> +} >> + >> +static int ma35d1_clk_pll_prepare(struct clk_hw *hw) >> +{ >> + struct ma35d1_clk_pll *pll = to_ma35d1_clk_pll(hw); >> + u32 val; >> + >> + val = readl_relaxed(pll->ctl1_base); >> + val &= ~PLL_CTL1_PD; >> + writel_relaxed(val, pll->ctl1_base); >> + return 0; >> +} >> + >> +static void ma35d1_clk_pll_unprepare(struct clk_hw *hw) >> +{ >> + struct ma35d1_clk_pll *pll = to_ma35d1_clk_pll(hw); >> + u32 val; >> + >> + val = readl_relaxed(pll->ctl1_base); >> + val |= PLL_CTL1_PD; >> + writel_relaxed(val, pll->ctl1_base); >> +} >> + >> +static const struct clk_ops ma35d1_clk_pll_ops = { >> + .is_prepared = ma35d1_clk_pll_is_prepared, >> + .prepare = ma35d1_clk_pll_prepare, >> + .unprepare = ma35d1_clk_pll_unprepare, >> + .set_rate = ma35d1_clk_pll_set_rate, >> + .recalc_rate = ma35d1_clk_pll_recalc_rate, >> + .round_rate = ma35d1_clk_pll_round_rate, >> +}; >> + >> +static const struct clk_ops ma35d1_clk_fixed_pll_ops = { >> + .recalc_rate = ma35d1_clk_pll_recalc_rate, >> + .round_rate = ma35d1_clk_pll_round_rate, >> +}; >> + >> +struct clk_hw *ma35d1_reg_clk_pll(struct device *dev, u32 id, u8 u8mode, const char *name, >> + struct clk_hw *parent_hw, void __iomem *base) >> +{ >> + struct clk_parent_data pdata = { .index = 0 }; >> + struct clk_init_data init = {}; >> + struct ma35d1_clk_pll *pll; >> + struct clk_hw *hw; >> + int ret; >> + >> + pll = devm_kzalloc(dev, sizeof(*pll), GFP_KERNEL); >> + if (!pll) >> + return ERR_PTR(-ENOMEM); >> + >> + pll->id = id; >> + pll->mode = u8mode; >> + pll->ctl0_base = base + REG_PLL_CTL0_OFFSET; >> + pll->ctl1_base = base + REG_PLL_CTL1_OFFSET; >> + pll->ctl2_base = base + REG_PLL_CTL2_OFFSET; >> + >> + init.name = name; >> + init.flags = 0; >> + pdata.hw = parent_hw; >> + init.parent_data = &pdata; >> + init.num_parents = 1; >> + >> + if (id == CAPLL || id == DDRPLL) >> + init.ops = &ma35d1_clk_fixed_pll_ops; >> + else >> + init.ops = &ma35d1_clk_pll_ops; >> + >> + pll->hw.init = &init; >> + hw = &pll->hw; >> + >> + ret = devm_clk_hw_register(dev, hw); >> + if (ret) >> + return ERR_PTR(ret); >> + return hw; >> +} >> +EXPORT_SYMBOL_GPL(ma35d1_reg_clk_pll); >> diff --git a/drivers/clk/nuvoton/clk-ma35d1.c b/drivers/clk/nuvoton/clk-ma35d1.c >> new file mode 100644 >> index 000000000000..68fbaf2f4945 >> --- /dev/null >> +++ b/drivers/clk/nuvoton/clk-ma35d1.c >> @@ -0,0 +1,948 @@ >> +static inline struct clk_hw *ma35d1_clk_fixed(const char *name, int rate) >> +{ >> + return clk_hw_register_fixed_rate(NULL, name, NULL, 0, rate); >> +} >> + >> +static inline struct clk_hw *ma35d1_clk_mux_parent(struct device *dev, >> + const char *name, >> + void __iomem *reg, >> + u8 shift, u8 width, >> + const struct clk_parent_data *pdata, >> + int num_pdata) >> +{ >> + return clk_hw_register_mux_parent_data(dev, name, pdata, num_pdata, >> + CLK_SET_RATE_NO_REPARENT, reg, shift, >> + width, 0, &ma35d1_lock); >> +} >> + >> +static inline struct clk_hw *ma35d1_clk_mux(struct device *dev, >> + const char *name, >> + void __iomem *reg, >> + u8 shift, u8 width, >> + const char *const *parents, >> + int num_parents) >> +{ >> + return devm_clk_hw_register_mux(dev, name, parents, num_parents, >> + CLK_SET_RATE_NO_REPARENT, reg, shift, >> + width, 0, &ma35d1_lock); >> +} >> + >> +static inline struct clk_hw *ma35d1_clk_divider(struct device *dev, >> + const char *name, >> + const char *parent, >> + void __iomem *reg, u8 shift, >> + u8 width) >> +{ >> + return devm_clk_hw_register_divider(dev, name, parent, CLK_SET_RATE_PARENT, >> + reg, shift, width, 0, &ma35d1_lock); >> +} >> + >> +static inline struct clk_hw *ma35d1_clk_divider_pow2(struct device *dev, >> + const char *name, >> + const char *parent, >> + void __iomem *reg, >> + u8 shift, u8 width) >> +{ >> + return devm_clk_hw_register_divider(dev, name, parent, >> + CLK_DIVIDER_POWER_OF_TWO, reg, shift, >> + width, 0, &ma35d1_lock); >> +} >> + >> +static inline struct clk_hw *ma35d1_clk_divider_table(struct device *dev, >> + const char *name, >> + const char *parent, >> + void __iomem *reg, >> + u8 shift, u8 width, >> + const struct clk_div_table *table) >> +{ >> + return devm_clk_hw_register_divider_table(dev, name, parent, 0, >> + reg, shift, width, 0, >> + table, &ma35d1_lock); >> +} >> + >> +static inline struct clk_hw *ma35d1_clk_fixed_factor(struct device *dev, >> + const char *name, >> + const char *parent, >> + unsigned int mult, >> + unsigned int div) >> +{ >> + return devm_clk_hw_register_fixed_factor(dev, name, parent, >> + CLK_SET_RATE_PARENT, mult, div); >> +} >> + >> +static inline struct clk_hw *ma35d1_clk_gate(struct device *dev, >> + const char *name, >> + const char *parent, >> + void __iomem *reg, u8 shift) >> +{ >> + return devm_clk_hw_register_gate(dev, name, parent, CLK_SET_RATE_PARENT, >> + reg, shift, 0, &ma35d1_lock); >> +} > Inline is not something that should be used that much in .c files, just > leave it up to the compiler to decide whether to inline or not. There were > some other inlines in the other file I noted later but I didn't mark them > but please check their inline usage too. Okay, I will check these .c files and remove the inline usage. >> +static int ma35d1_get_pll_setting(struct device_node *clk_node, u32 *pllmode) >> +{ >> + const char *of_str; >> + int i; >> + >> + for (i = 0; i < PLL_MAX_NUM; i++) { >> + if (of_property_read_string_index(clk_node, "nuvoton,pll-mode", i, &of_str)) >> + return -EINVAL; >> + if (!strcmp(of_str, "integer")) >> + pllmode[i] = PLL_MODE_INT; >> + else if (!strcmp(of_str, "fractional")) >> + pllmode[i] = PLL_MODE_FRAC; >> + else if (!strcmp(of_str, "spread-spectrum")) >> + pllmode[i] = PLL_MODE_SS; >> + else >> + return -EINVAL; >> + } >> + return 0; >> +} >> + >> +static int ma35d1_clocks_probe(struct platform_device *pdev) >> +{ >> + struct device *dev = &pdev->dev; >> + struct device_node *clk_node = pdev->dev.of_node; >> + struct resource *res = platform_get_resource(pdev, IORESOURCE_MEM, 0); >> + void __iomem *clk_base; >> + static struct clk_hw **hws; >> + static struct clk_hw_onecell_data *ma35d1_hw_data; >> + u32 pllmode[PLL_MAX_NUM]; >> + int ret; >> + >> + ma35d1_hw_data = devm_kzalloc(dev, >> + struct_size(ma35d1_hw_data, hws, CLK_MAX_IDX), >> + GFP_KERNEL); >> + if (WARN_ON(!ma35d1_hw_data)) > Alloc error should not be WARN_ON()'ed, it's not a bug. If you want to > notify about normal (non-programming) errors, use dev_err() instead. Yes, I will use dev_err() instead. >> + return -ENOMEM; >> + >> + ma35d1_hw_data->num = CLK_MAX_IDX; >> + hws = ma35d1_hw_data->hws; >> + >> + clk_base = devm_ioremap_resource(dev, res); >> + if (IS_ERR(clk_base)) >> + return PTR_ERR(clk_base); >> + >> + ret = ma35d1_get_pll_setting(clk_node, pllmode); >> + if (ret < 0) { >> + dev_err(dev, "Invalid PLL setting!\n"); >> + return -EINVAL; >> + } Best Regards, Jacky Huang