Received: by 2002:a05:6a10:16a7:0:0:0:0 with SMTP id gp39csp2033660pxb; Sat, 21 Nov 2020 06:37:29 -0800 (PST) X-Google-Smtp-Source: ABdhPJxApSPYtiQW5XZHxytrKdL/C8jSlbMHXFURDa7CQNDirXFhmEaUgD+V3IWr2N2aKmbFCtIk X-Received: by 2002:a17:907:2166:: with SMTP id rl6mr37514187ejb.61.1605969449699; Sat, 21 Nov 2020 06:37:29 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1605969449; cv=none; d=google.com; s=arc-20160816; b=ky2IT5q4Z137P/+jlThuKZuueXIEFycWFR3cE8YAck+DY9ScHvTmgMXnpc87rGvPdv W23U7NGm581y6t0ddhMyJgrvz/z2LFIKbwL22wwg3aBJFfuVh6fjJBbwu7cbGYig9ixc 6EQ7Y3OJ7D88/yUXD/n7Nj6k2pDrHrU67Pfvq8tUrR1ielPvj2472V3Ydw/c0NvFGqKS 3buzEKcRJFYr6Wuygpso0+9dsrCgJMGU3p4+ZBHBuuahyg5o3d7GB908pwWJv2B4HP9x 2ctyMgccQFW8LJwTQqmPiD0jRwnaBqCHcupEHlMyRBU/XIe2eV3RXah3uBOhtlMM06r0 9aAw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:user-agent:organization:in-reply-to :content-disposition:mime-version:references:mail-followup-to :message-id:subject:cc:to:from:date; bh=hIDxPAzQoHaBq6/UDdsc1wi4LHMSs7ohrYUlPlfZbN0=; b=KzDylAP4CCRjmYye65yISSRgEFUGZ0ZqJdP6aaCL1xjduoSYIjVmY32qGgtAB9JbnN lePMbaJ0aI3e2LjB0UccZFm1VWpRBRZv92iA38vDqZAeDG+RNdi+DZvPqyVwqQyLhyvf 0Sk83pSZ31QO3FBEXVV7JO56t7STzUf1bA8EQAo27nwRcfgEzW5Kcq5khxls9jNQBgB4 RLBjL14W/GbFOJzDa/8sEqb4vLQdU77Se9Ax3dMhlXo5UiFKLWJDYaxV9e72raYjTjlj zZB2QnmgJF6EsDz2oqsPx6VintxkqXftqUuAifWlXsN9gkyeOFshhdvHxazFHuI4umb2 Yh/g== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id hh13si3449356ejb.360.2020.11.21.06.37.06; Sat, 21 Nov 2020 06:37:29 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1728005AbgKUOep (ORCPT + 99 others); Sat, 21 Nov 2020 09:34:45 -0500 Received: from smtp2207-205.mail.aliyun.com ([121.197.207.205]:57306 "EHLO smtp2207-205.mail.aliyun.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727851AbgKUOeo (ORCPT ); Sat, 21 Nov 2020 09:34:44 -0500 X-Alimail-AntiSpam: AC=CONTINUE;BC=0.07436287|-1;CH=green;DM=|CONTINUE|false|;DS=CONTINUE|ham_system_inform|0.0754813-0.000384625-0.924134;FP=0|0|0|0|0|-1|-1|-1;HT=ay29a033018047206;MF=fuyao@allwinnertech.com;NM=1;PH=DS;RN=5;RT=5;SR=0;TI=SMTPD_---.IzafzYM_1605969186; Received: from localhost(mailfrom:fuyao@allwinnertech.com fp:SMTPD_---.IzafzYM_1605969186) by smtp.aliyun-inc.com(10.147.41.199); Sat, 21 Nov 2020 22:33:07 +0800 Date: Sat, 21 Nov 2020 22:33:06 +0800 From: fuyao To: Bjorn Andersson Cc: mripard@kernel.org, wens@csie.org, linux-remoteproc@vger.kernel.org, linux-kernel@vger.kernel.org Subject: Re: [PATCH 2/2] hwspinlock: add SUNXI implementation Message-ID: <20201121143306.GB23438@debian> Mail-Followup-To: Bjorn Andersson , mripard@kernel.org, wens@csie.org, linux-remoteproc@vger.kernel.org, linux-kernel@vger.kernel.org References: <20201121040104.GI9177@builder.lan> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: <20201121040104.GI9177@builder.lan> Organization: fuyao_love_xxt.Allwinnertech.Technology User-Agent: Mutt/1.12.1+6 (4c2f7c70) (2019-08-28) Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Fri, Nov 20, 2020 at 10:01:04PM -0600, Bjorn Andersson wrote: > On Thu 19 Nov 00:44 CST 2020, fuyao@allwinnertech.com wrote: > > > From: fuyao > > > > Add hwspinlock support for the SUNXI Hardware Spinlock device. > > > > The Hardware Spinlock device on SUNXI provides hardware assistance > > for synchronization between the multiple processors in the system > > (Cortex-A7, or1k, Xtensa DSP, Cortex-A53) > > > > Signed-off-by: fuyao > > --- > > MAINTAINERS | 6 + > > drivers/hwspinlock/Kconfig | 10 ++ > > drivers/hwspinlock/Makefile | 1 + > > drivers/hwspinlock/sunxi_hwspinlock.c | 205 ++++++++++++++++++++++++++ > > 4 files changed, 222 insertions(+) > > create mode 100644 drivers/hwspinlock/sunxi_hwspinlock.c > > > > diff --git a/MAINTAINERS b/MAINTAINERS > > index e451dcce054f0..68d25574432d0 100644 > > --- a/MAINTAINERS > > +++ b/MAINTAINERS > > @@ -737,6 +737,12 @@ L: linux-media@vger.kernel.org > > S: Maintained > > F: drivers/staging/media/sunxi/cedrus/ > > > > +ALLWINNER HWSPINLOCK DRIVER > > +M: fuyao > > +S: Maintained > > +F: drivers/hwspinlock/sunxi_hwspinlock.c > > +F: Documentation/devicetree/bindings/hwlock/sunxi,hwspinlock.yaml > > + > > ALPHA PORT > > M: Richard Henderson > > M: Ivan Kokshaysky > > diff --git a/drivers/hwspinlock/Kconfig b/drivers/hwspinlock/Kconfig > > index 32cd26352f381..4d0d516dcb544 100644 > > --- a/drivers/hwspinlock/Kconfig > > +++ b/drivers/hwspinlock/Kconfig > > @@ -55,6 +55,16 @@ config HWSPINLOCK_STM32 > > > > If unsure, say N. > > > > +config HWSPINLOCK_SUNXI > > + tristate "SUNXI Hardware Spinlock device" > > + depends on ARCH_SUNXI || COMPILE_TEST > > + help > > + Say y here to support the SUNXI Hardware Semaphore functionality, which > > + provides a synchronisation mechanism for the various processor on the > > + SoC. > > + > > + If unsure, say N. > > + > > config HSEM_U8500 > > tristate "STE Hardware Semaphore functionality" > > depends on ARCH_U8500 || COMPILE_TEST > > diff --git a/drivers/hwspinlock/Makefile b/drivers/hwspinlock/Makefile > > index ed053e3f02be4..839a053205f73 100644 > > --- a/drivers/hwspinlock/Makefile > > +++ b/drivers/hwspinlock/Makefile > > @@ -10,3 +10,4 @@ obj-$(CONFIG_HWSPINLOCK_SIRF) += sirf_hwspinlock.o > > obj-$(CONFIG_HWSPINLOCK_SPRD) += sprd_hwspinlock.o > > obj-$(CONFIG_HWSPINLOCK_STM32) += stm32_hwspinlock.o > > obj-$(CONFIG_HSEM_U8500) += u8500_hsem.o > > +obj-$(CONFIG_HWSPINLOCK_SUNXI) += sunxi_hwspinlock.o > > diff --git a/drivers/hwspinlock/sunxi_hwspinlock.c b/drivers/hwspinlock/sunxi_hwspinlock.c > > new file mode 100644 > > index 0000000000000..2c3dc148c9b72 > > --- /dev/null > > +++ b/drivers/hwspinlock/sunxi_hwspinlock.c > > @@ -0,0 +1,205 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * SUNXI hardware spinlock driver > > + * > > + * Copyright (C) 2020 Allwinnertech - http://www.allwinnertech.com > > + * > > Please remove the remainder of this comment, it's already covered by the > SPDX header above. > > > + * This program is free software; you can redistribute it and/or > > + * modify it under the terms of the GNU General Public License > > + * version 2 as published by the Free Software Foundation. > > + * > > + * This program is distributed in the hope that it will be useful, but > > + * WITHOUT ANY WARRANTY; without even the implied warranty of > > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU > > + * General Public License for more details. > > + */ > > + > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > +#include > > You don't need all of these. > > > + > > +#include "hwspinlock_internal.h" > > + > > +/* hardware spinlock register list */ > > +#define LOCK_SYS_STATUS_REG (0x0000) > > +#define LOCK_STATUS_REG (0x0010) > > +#define LOCK_BASE_OFFSET (0x0100) > > +#define LOCK_BASE_ID (0) > > No need for the parenthesis on these, please drop them. > > > + > > +/* Possible values of SPINLOCK_LOCK_REG */ > > +#define SPINLOCK_NOTTAKEN (0) /* free */ > > +#define SPINLOCK_TAKEN (1) /* locked */ > > + > > +struct sunxi_spinlock_config { > > + int nspin; > > +}; > > + > > +static int sunxi_hwspinlock_trylock(struct hwspinlock *lock) > > +{ > > + void __iomem *lock_addr = lock->priv; > > + > > + /* attempt to acquire the lock by reading its value */ > > + return (readl(lock_addr) == SPINLOCK_NOTTAKEN); > > Please drop the outer (). > > > +} > > + > > +static void sunxi_hwspinlock_unlock(struct hwspinlock *lock) > > +{ > > + void __iomem *lock_addr = lock->priv; > > + > > + /* release the lock by writing 0 to it */ > > + writel(SPINLOCK_NOTTAKEN, lock_addr); > > +} > > + > > +/* > > + * relax the SUNXI interconnect while spinning on it. > > + * > > + * The specs recommended that the retry delay time will be > > + * just over half of the time that a requester would be > > + * expected to hold the lock. > > + * > > + * in sunxi spinlock time less then 200 cycles > > + * > > + * The number below is taken from an hardware specs example, > > + * obviously it is somewhat arbitrary. > > Thank you for the good explanation. > > > + */ > > +static void sunxi_hwspinlock_relax(struct hwspinlock *lock) > > +{ > > + ndelay(50); > > +} > > + > > +static const struct hwspinlock_ops sunxi_hwspinlock_ops = { > > + .trylock = sunxi_hwspinlock_trylock, > > + .unlock = sunxi_hwspinlock_unlock, > > + .relax = sunxi_hwspinlock_relax, > > +}; > > + > > +struct sunxi_hwspinlock_device { > > + struct hwspinlock_device *bank; > > + struct clk *bus_clk; > > + struct reset_control *reset; > > +}; > > + > > +static void sunxi_hwspinlock_clk_init(struct platform_device *pdev, > > + struct sunxi_hwspinlock_device *private) > > +{ > > + private->bus_clk = devm_clk_get(&pdev->dev, NULL); > > + private->reset = devm_reset_control_get(&pdev->dev, NULL); > > You should check the return value of these, e.g. for EPROBE_DEFER and if > so return appropriately from sunxi_hwspinlock_probe(). > > So please move them to the probe function to make this easier and > cleaner. > > > + > > + if (private->reset) > > + reset_control_deassert(private->reset); > > + if (private->bus_clk) > > + clk_prepare_enable(private->bus_clk); > > Both of these apis start with > > if (!argument) > return; > > So there's no need for you to check for NULL before calling them. Also > to make it clear that you want these to be deassered and prepare_enabled > between probe and remvoe, move them into probe (and next function into > remove). > > > +} > > + > > +static void sunxi_hwspinlock_clk_dinit(struct sunxi_hwspinlock_device *private) > > +{ > > + if (private->reset) > > + reset_control_assert(private->reset); > > + if (private->bus_clk) > > + clk_disable(private->bus_clk); > > +} > > + > > +static const struct sunxi_spinlock_config spin_ver_1 = { > > + .nspin = 32, > > +}; > > + > > +static const struct of_device_id sunxi_hwspinlock_of_match[] = { > > + { > > + .compatible = "allwinner,h3-hwspinlock", > > + .data = &spin_ver_1, > > If all cases comes with the same "data", then please just put nspin in a > #define until you're going to support hardware that has some other > number of locks. > > > + }, > > + { > > + .compatible = "allwinner,h6-hwspinlock", > > + .data = &spin_ver_1, > > + }, > > + { /* Sentinel */ }, > > No need to spell out "/* Sentinel */", leave it emtpy and please drop the , at > the end. > > > +}; > > +MODULE_DEVICE_TABLE(of, sunxi_hwspinlock_of_match); > > Please move this down by sunxi_hwspinlock_driver and use > device_get_match_data() instead of of_match_device(). > > > + > > +static int sunxi_hwspinlock_probe(struct platform_device *pdev) > > +{ > > + struct sunxi_hwspinlock_device *private; > > + struct hwspinlock_device *bank; > > + struct hwspinlock *hwlock; > > + const struct sunxi_spinlock_config *config; > > + const struct of_device_id *match; > > + void __iomem *iobase; > > + int num_locks, i, ret; > > + > > + iobase = devm_platform_ioremap_resource(pdev, 0); > > + if (PTR_ERR(iobase)) > > + return PTR_ERR(iobase); > > + > > + match = of_match_device(of_match_ptr(sunxi_hwspinlock_of_match), > > + &pdev->dev); > > + if (!match) > > + return -ENODEV; > > + > > + config = match->data; > > + num_locks = config->nspin; > > + > > + private = devm_kzalloc(&pdev->dev, sizeof(*private), GFP_KERNEL); > > + if (!private) > > + return -ENOMEM; > > + > > + bank = devm_kzalloc(&pdev->dev, > > + sizeof(*bank) + num_locks * sizeof(*hwlock), > > + GFP_KERNEL); > > + if (!bank) > > + return -ENOMEM; > > + > > + private->bank = bank; > > + sunxi_hwspinlock_clk_init(pdev, private); > > + > > + platform_set_drvdata(pdev, private); > > + > > + for (i = 0, hwlock = &bank->lock[0]; i < num_locks; i++, hwlock++) > > + hwlock->priv = iobase + LOCK_BASE_OFFSET + sizeof(u32) * i; > > + > > + ret = devm_hwspin_lock_register(&pdev->dev, bank, &sunxi_hwspinlock_ops, > > + LOCK_BASE_ID, num_locks); > > This returns 0 or -errno, so rather than returning ret if ret otherwise > 0, just do: > > return devm_hwspin_lock_register(...) > > > + if (ret) > > + return ret; > > + > > + > > + return 0; > > +} > > + > > +static int sunxi_hwspinlock_remove(struct platform_device *pdev) > > +{ > > + struct sunxi_hwspinlock_device *private = platform_get_drvdata(pdev); > > + > > + sunxi_hwspinlock_clk_dinit(private); > > + > > + return 0; > > +} > > + > > +static struct platform_driver sunxi_hwspinlock_driver = { > > + .probe = sunxi_hwspinlock_probe, > > + .remove = sunxi_hwspinlock_remove, > > + .driver = { > > + .name = "sunxi-hwspinlock", > > + .owner = THIS_MODULE, > > module_platform_driver() fills out .owner for you, so please remove > this. > > > + .of_match_table = of_match_ptr(sunxi_hwspinlock_of_match), > > Please skip of_match_ptr(), it will just cause build warnings when > compile tested without CONFIG_OF. > > Thank you, > Bjorn > > > + }, > > +}; > > + > > +module_platform_driver(sunxi_hwspinlock_driver); > > + > > +MODULE_LICENSE("GPL v2"); > > +MODULE_DESCRIPTION("Hardware spinlock driver for SUNXI"); > > +MODULE_AUTHOR("fuyao "); Thanks for you review, I read it carefully, and learned a lot. Maxim tells that there is already the same submission, so this submission will be abandoned. thanks again. -- fuyao