Received: by 2002:a05:6a10:6d10:0:0:0:0 with SMTP id gq16csp272380pxb; Tue, 12 Apr 2022 01:25:24 -0700 (PDT) X-Google-Smtp-Source: ABdhPJzmZCmNxyqeasrPIlxl97NBxEauMwlZqst6Z1gUI/DFlFNAse0jO82RhTvk1qbsHpUICCTU X-Received: by 2002:a17:90b:4f43:b0:1c7:552b:7553 with SMTP id pj3-20020a17090b4f4300b001c7552b7553mr3663437pjb.117.1649751924589; Tue, 12 Apr 2022 01:25:24 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1649751924; cv=none; d=google.com; s=arc-20160816; b=IekCuerEMQtIwRqP0k51vrt/DK1hsBoNsgV97kJ+cq00P8jNRx/BnTwX0JBaaQb5ED 4WnYuNFlK6nXcwdQwJi2w7OymhwjrNFrB8d3B4x2rGLdV2s7e9OEGeNQrqpsLuMny3BT nBrjcebguW6CvBrxJDoXp2DLa9PLsU9eZYkukXHIa4NR5RF98CN0EbpL2ymkV7rm9R/Z xuWt9z5SIjJnfddalifOCLJG2wnR5ryLCy10Hpwbkon1jPF1SLk+ltE5/gi6tIhi0foU B8eos9h+n4wc+XXP3+HkDYLvRGbIxx9AK6IAwbG9rnwiiQKDBfuuDJf1gtCsJ53j20K8 q7eQ== 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:from :references:cc:to:content-language:subject:user-agent:mime-version :date:message-id; bh=bDurj4Ox10Oav2uXBuMAcZmSRqlBSuPk05pivaeu56U=; b=KqjPKAqclvdNwtTeOzdwuyETycw1Rsb8iLiOwGuSx6W1TWCWYbzfyjL6Va1t8r+DZT EgINZPhFIJe582Y9nPzsoczfb/KMOgHL+fANsurtsNKiylEzKnhhS93KuoMy8bTJAoGO r7EcH1fWEK3OKBoe031xWSAg6DHYOejCKVzkk14gialZTnsXrv095IEvxDNAxt60BPYy B+WAIL8k8FpRJKWIi719dHctBPOjQR9GJ4HVd38DLClqP1yYioAJgSg6FDFVpQ2d5MnD Dx465t5r6vQncrV+SCrmFp4JGCvyEEPY4Yf+79Rjd6ig3P8vmpbQJFRllOSc/BlLQfhZ lN2Q== ARC-Authentication-Results: i=1; mx.google.com; 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 Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id o4-20020a656a44000000b0039cf4948dabsi2469845pgu.314.2022.04.12.01.25.11; Tue, 12 Apr 2022 01:25:24 -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; 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 Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S237932AbiDKCqZ (ORCPT + 99 others); Sun, 10 Apr 2022 22:46:25 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:37144 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S244394AbiDKCqU (ORCPT ); Sun, 10 Apr 2022 22:46:20 -0400 Received: from mail-sz.amlogic.com (mail-sz.amlogic.com [211.162.65.117]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 6B02038BE; Sun, 10 Apr 2022 19:44:07 -0700 (PDT) Received: from [10.28.39.106] (10.28.39.106) by mail-sz.amlogic.com (10.28.11.5) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.1.2176.2; Mon, 11 Apr 2022 10:44:05 +0800 Message-ID: <0435d1f5-af28-2a3a-b53c-d7557e3fe75f@amlogic.com> Date: Mon, 11 Apr 2022 10:44:05 +0800 MIME-Version: 1.0 User-Agent: Mozilla/5.0 (Windows NT 6.1; rv:91.0) Gecko/20100101 Thunderbird/91.7.0 Subject: Re: [PATCH v4 1/2] mtd: rawnand: meson: discard the common MMC sub clock framework Content-Language: en-US To: Neil Armstrong , Miquel Raynal , CC: Rob Herring , Richard Weinberger , Vignesh Raghavendra , Jerome Brunet , Martin Blumenstingl , Kevin Hilman , Jianxin Pan , Victor Wan , XianWei Zhao , Kelvin Zhang , BiChao Zheng , YongHui Yu , , , , References: <20220402074921.13316-1-liang.yang@amlogic.com> <20220402074921.13316-2-liang.yang@amlogic.com> From: Liang Yang In-Reply-To: Content-Type: text/plain; charset="UTF-8"; format=flowed Content-Transfer-Encoding: 8bit X-Originating-IP: [10.28.39.106] X-ClientProxiedBy: mail-sz.amlogic.com (10.28.11.5) To mail-sz.amlogic.com (10.28.11.5) X-Spam-Status: No, score=-5.0 required=5.0 tests=BAYES_00,NICE_REPLY_A, SPF_HELO_NONE,SPF_PASS,T_SCC_BODY_TEXT_LINE 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 Hi Neil, On 2022/4/4 20:40, Neil Armstrong wrote: > [ EXTERNAL EMAIL ] > > Hi, > > On 02/04/2022 09:49, Liang Yang wrote: >> EMMC and NAND have the same clock control register named >> 'SD_EMMC_CLOCK' which is >> defined in EMMC port internally. bit0~5 of 'SD_EMMC_CLOCK' is the >> divider and >> bit6~7 is the mux for fix pll and xtal.A common MMC and NAND sub-clock >> has been >> implemented and can be used by the eMMC and NAND controller (which are >> mutually >> exclusive anyway). Let's use this new clock. >> >> Signed-off-by: Liang Yang >> --- >>   drivers/mtd/nand/raw/meson_nand.c | 89 +++++++++++++++---------------- >>   1 file changed, 42 insertions(+), 47 deletions(-) >> >> diff --git a/drivers/mtd/nand/raw/meson_nand.c >> b/drivers/mtd/nand/raw/meson_nand.c >> index ac3be92872d0..1b1a9407fb2f 100644 >> --- a/drivers/mtd/nand/raw/meson_nand.c >> +++ b/drivers/mtd/nand/raw/meson_nand.c >> @@ -10,6 +10,7 @@ >>   #include >>   #include >>   #include >> +#include >>   #include >>   #include >>   #include >> @@ -19,6 +20,7 @@ >>   #include >>   #include >>   #include >> +#include >>   #include >>   #define NFC_REG_CMD        0x00 >> @@ -104,6 +106,9 @@ >>   #define PER_INFO_BYTE        8 >> +#define CLK_DIV_SHIFT        0 >> +#define CLK_DIV_WIDTH        6 >> + >>   struct meson_nfc_nand_chip { >>       struct list_head node; >>       struct nand_chip nand; >> @@ -151,15 +156,15 @@ struct meson_nfc { >>       struct nand_controller controller; >>       struct clk *core_clk; >>       struct clk *device_clk; >> -    struct clk *phase_tx; >> -    struct clk *phase_rx; >> +    struct clk *nand_clk; >> +    struct clk_divider nand_divider; >>       unsigned long clk_rate; >>       u32 bus_timing; >>       struct device *dev; >>       void __iomem *reg_base; >> -    struct regmap *reg_clk; >> +    void __iomem *sd_emmc_clock; > > The name could still be reg_clk, even if not a regmap anymore. ok, i will do it. > >>       struct completion completion; >>       struct list_head chips; >>       const struct meson_nfc_data *data; >> @@ -235,7 +240,7 @@ static void meson_nfc_select_chip(struct nand_chip >> *nand, int chip) >>       nfc->timing.tbers_max = meson_chip->tbers_max; >>       if (nfc->clk_rate != meson_chip->clk_rate) { >> -        ret = clk_set_rate(nfc->device_clk, meson_chip->clk_rate); >> +        ret = clk_set_rate(nfc->nand_clk, meson_chip->clk_rate); >>           if (ret) { >>               dev_err(nfc->dev, "failed to set clock rate\n"); >>               return; >> @@ -406,7 +411,6 @@ static int meson_nfc_queue_rb(struct meson_nfc >> *nfc, int timeout_ms) >>       cmd = NFC_CMD_RB | NFC_CMD_RB_INT >>           | nfc->param.chip_select | nfc->timing.tbers_max; >>       writel(cmd, nfc->reg_base + NFC_REG_CMD); >> - >>       ret = wait_for_completion_timeout(&nfc->completion, >>                         msecs_to_jiffies(timeout_ms)); >>       if (ret == 0) >> @@ -985,9 +989,11 @@ static const struct mtd_ooblayout_ops >> meson_ooblayout_ops = { >>       .free = meson_ooblayout_free, >>   }; >> +struct clk_parent_data nfc_divider_parent_data[1]; > > This should be in the meson_nfc_clk_init() function, not global ok. > >>   static int meson_nfc_clk_init(struct meson_nfc *nfc) >>   { >>       int ret; >> +    struct clk_init_data init = {0}; >>       /* request core clock */ >>       nfc->core_clk = devm_clk_get(nfc->dev, "core"); >> @@ -1002,21 +1008,26 @@ static int meson_nfc_clk_init(struct meson_nfc >> *nfc) >>           return PTR_ERR(nfc->device_clk); >>       } >> -    nfc->phase_tx = devm_clk_get(nfc->dev, "tx"); >> -    if (IS_ERR(nfc->phase_tx)) { >> -        dev_err(nfc->dev, "failed to get TX clk\n"); >> -        return PTR_ERR(nfc->phase_tx); >> -    } >> - >> -    nfc->phase_rx = devm_clk_get(nfc->dev, "rx"); >> -    if (IS_ERR(nfc->phase_rx)) { >> -        dev_err(nfc->dev, "failed to get RX clk\n"); >> -        return PTR_ERR(nfc->phase_rx); >> -    } >> +    init.name = devm_kstrdup(nfc->dev, "nfc#div", GFP_KERNEL); > > name = devm_kasprintf(nfc->dev, GFP_KERNEL, "%s#div", dev_name(dev)); > > would be better. ok, thanks. > >> +    init.ops = &clk_divider_ops; >> +    nfc_divider_parent_data[0].fw_name = "device"; >> +    init.parent_data = nfc_divider_parent_data; >> +    init.num_parents = 1; >> +    nfc->nand_divider.reg = nfc->sd_emmc_clock; >> +    nfc->nand_divider.shift = CLK_DIV_SHIFT; >> +    nfc->nand_divider.width = CLK_DIV_WIDTH; >> +    nfc->nand_divider.hw.init = &init; >> +    nfc->nand_divider.flags = CLK_DIVIDER_ONE_BASED | >> +                  CLK_DIVIDER_ROUND_CLOSEST | >> +                  CLK_DIVIDER_ALLOW_ZERO; >> + >> +    nfc->nand_clk = devm_clk_register(nfc->dev, &nfc->nand_divider.hw); >> +    if (IS_ERR(nfc->nand_clk)) >> +        return PTR_ERR(nfc->nand_clk); >>       /* init SD_EMMC_CLOCK to sane defaults w/min clock rate */ >> -    regmap_update_bits(nfc->reg_clk, >> -               0, CLK_SELECT_NAND, CLK_SELECT_NAND); >> +    writel(CLK_SELECT_NAND | readl(nfc->sd_emmc_clock), >> +           nfc->sd_emmc_clock); >>       ret = clk_prepare_enable(nfc->core_clk); >>       if (ret) { >> @@ -1030,29 +1041,21 @@ static int meson_nfc_clk_init(struct meson_nfc >> *nfc) >>           goto err_device_clk; >>       } >> -    ret = clk_prepare_enable(nfc->phase_tx); >> +    ret = clk_prepare_enable(nfc->nand_clk); >>       if (ret) { >> -        dev_err(nfc->dev, "failed to enable TX clock\n"); >> -        goto err_phase_tx; >> +        dev_err(nfc->dev, "pre enable NFC divider fail\n"); >> +        goto err_nand_clk; >>       } >> -    ret = clk_prepare_enable(nfc->phase_rx); >> -    if (ret) { >> -        dev_err(nfc->dev, "failed to enable RX clock\n"); >> -        goto err_phase_rx; >> -    } >> - >> -    ret = clk_set_rate(nfc->device_clk, 24000000); >> +    ret = clk_set_rate(nfc->nand_clk, 24000000); >>       if (ret) >> -        goto err_disable_rx; >> +        goto err_disable_clk; >>       return 0; >> -err_disable_rx: >> -    clk_disable_unprepare(nfc->phase_rx); >> -err_phase_rx: >> -    clk_disable_unprepare(nfc->phase_tx); >> -err_phase_tx: >> +err_disable_clk: >> +    clk_disable_unprepare(nfc->nand_clk); >> +err_nand_clk: >>       clk_disable_unprepare(nfc->device_clk); >>   err_device_clk: >>       clk_disable_unprepare(nfc->core_clk); >> @@ -1061,8 +1064,7 @@ static int meson_nfc_clk_init(struct meson_nfc >> *nfc) >>   static void meson_nfc_disable_clk(struct meson_nfc *nfc) >>   { >> -    clk_disable_unprepare(nfc->phase_rx); >> -    clk_disable_unprepare(nfc->phase_tx); >> +    clk_disable_unprepare(nfc->nand_clk); >>       clk_disable_unprepare(nfc->device_clk); >>       clk_disable_unprepare(nfc->core_clk); >>   } >> @@ -1374,7 +1376,6 @@ static int meson_nfc_probe(struct >> platform_device *pdev) >>   { >>       struct device *dev = &pdev->dev; >>       struct meson_nfc *nfc; >> -    struct resource *res; >>       int ret, irq; >>       nfc = devm_kzalloc(dev, sizeof(*nfc), GFP_KERNEL); >> @@ -1388,21 +1389,15 @@ static int meson_nfc_probe(struct >> platform_device *pdev) >>       nand_controller_init(&nfc->controller); >>       INIT_LIST_HEAD(&nfc->chips); >>       init_completion(&nfc->completion); >> - >>       nfc->dev = dev; >> -    res = platform_get_resource(pdev, IORESOURCE_MEM, 0); >> -    nfc->reg_base = devm_ioremap_resource(dev, res); >> +    nfc->reg_base = devm_platform_ioremap_resource_byname(pdev, "nfc"); >>       if (IS_ERR(nfc->reg_base)) >>           return PTR_ERR(nfc->reg_base); >> -    nfc->reg_clk = >> -        syscon_regmap_lookup_by_phandle(dev->of_node, >> -                        "amlogic,mmc-syscon"); >> -    if (IS_ERR(nfc->reg_clk)) { >> -        dev_err(dev, "Failed to lookup clock base\n"); >> -        return PTR_ERR(nfc->reg_clk); >> -    } >> +    nfc->sd_emmc_clock = devm_platform_ioremap_resource_byname(pdev, >> "emmc"); >> +    if (IS_ERR(nfc->sd_emmc_clock)) >> +        return PTR_ERR(nfc->sd_emmc_clock); >>       irq = platform_get_irq(pdev, 0); >>       if (irq < 0) > > Thanks, > Neil > > .