Received: by 2002:a05:6a10:6744:0:0:0:0 with SMTP id w4csp321848pxu; Thu, 15 Oct 2020 05:02:38 -0700 (PDT) X-Google-Smtp-Source: ABdhPJyj8J6t5lHEDIXsp4djWFoO141f/T87/snBekjfHdcmBUIhn98Ds5cdOOVTkyvWpeUzFt7d X-Received: by 2002:a19:dc5:: with SMTP id 188mr905007lfn.272.1602763249118; Thu, 15 Oct 2020 05:00:49 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1602763249; cv=none; d=google.com; s=arc-20160816; b=wKr/zJ+tgHDhrbzfiqMschb3/+ktDRk9ie6xv6S7bZ981Q4rxDWHnDCXaa2raFC+8Q GUlUkWmuwpvOtpikCIRBdHyI40hsBRhfr0mKYSvpNptwwWjFKPYy39Xt3D+H7DRMuta3 +X/NHEL5svG4P5hcgXr2Z050vU2BjgHO7xfMmGgF+3LQFEYyvL/5lx8oesOLRhHnSOLz 6528tZO3hHugMqXMdVJWfCc6A8iiA098xlwTWp67YwY5y2/l5iOdGmKNhF0jW20mGIeD e/f1Nj3RkXLVhZZCoYxvvKrfCFqhqqo0PAC5bMGVemVb1CtjuLl21lrqXYzzx9eMfFgN yQJQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:user-agent:in-reply-to:content-disposition :mime-version:references:message-id:subject:cc:to:from:date; bh=ZM7SNNrmm9GJM7zOZsSMV9DTsYNa2jZOpH7K5IvKu+k=; b=tDWAzLKvWSDVIXeNKIVKBeMBrkkTgrJWCzVLhEJvrdmu42gjJ2mXPIwCbsDqMp1eyk Ajeqsfk2ztkvCtO+Vdznq0C4IRUCE8tNTnUq+KD4II26cMITN/Dq4DDJP6gdumqpisnm 1ccjkAucYpXJvJ4GAelNbWZ31RZPMqVhUS3mMKx4CG7czfh28UT4ek5yfTl0zvf7KQHl Tjr0Wsor9EKVLf/cYo9mVbY8umaw1nEExgJ+W18ufGjDRh21G+wU/KSEYlJ028EU+tjm EoW53dJ6ey6lcF/5Y/WQs3vVgIwmQmhBNqroGtsyug5VDmQ2ZNwxTJKRlubO0GcMiTnF N52w== 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; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=arm.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id bx3si128367ejb.709.2020.10.15.05.00.16; Thu, 15 Oct 2020 05:00:49 -0700 (PDT) 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; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=arm.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2389095AbgJOJ6m (ORCPT + 99 others); Thu, 15 Oct 2020 05:58:42 -0400 Received: from foss.arm.com ([217.140.110.172]:53862 "EHLO foss.arm.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2389050AbgJOJ6k (ORCPT ); Thu, 15 Oct 2020 05:58:40 -0400 Received: from usa-sjc-imap-foss1.foss.arm.com (unknown [10.121.207.14]) by usa-sjc-mx-foss1.foss.arm.com (Postfix) with ESMTP id 879B8D6E; Thu, 15 Oct 2020 02:58:38 -0700 (PDT) Received: from e120937-lin (unknown [172.31.20.19]) by usa-sjc-imap-foss1.foss.arm.com (Postfix) with ESMTPSA id 0E89D3F66B; Thu, 15 Oct 2020 02:58:36 -0700 (PDT) Date: Thu, 15 Oct 2020 10:58:30 +0100 From: Cristian Marussi To: Vincent Guittot Cc: linux-kernel , LAK , Sudeep Holla , Lukasz Luba , james.quinlan@broadcom.com, Jonathan Cameron , f.fainelli@gmail.com, Etienne Carriere , Thara Gopinath , Souvik Chakravarty Subject: Re: [PATCH 01/11] firmware: arm_scmi: review protocol registration interface Message-ID: <20201015095830.GC34395@e120937-lin> References: <20201014150545.44807-1-cristian.marussi@arm.com> <20201014150545.44807-2-cristian.marussi@arm.com> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: User-Agent: Mutt/1.9.4 (2018-02-28) Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi Vincent, thanks for the review. On Thu, Oct 15, 2020 at 11:41:09AM +0200, Vincent Guittot wrote: > On Wed, 14 Oct 2020 at 17:06, Cristian Marussi wrote: > > > > Extend common protocol registration routines and provide some new generic > > protocols' init/deinit helpers that tracks protocols' users and automatically > > perform the proper initialization/de-initialization on demand. > > > > Convert all protocols to use new registration schema while modifying only Base > > protocol to use also the new initialization helpers. > > > > All other standard protocols' initialization is still umodified and bound to > > SCMI devices probing. > > > > Signed-off-by: Cristian Marussi > > --- > > [...] > > > > > +/** > > + * scmi_get_protocol_instance - Protocol initialization helper. > > + * @handle: A reference to the SCMI platform instance. > > + * @protocol_id: The protocol being requested. > > + * > > + * In case the required protocol has never been requested before for this > > + * instance, allocate and initialize all the needed structures while handling > > + * resource allocation with a dedicated per-protocol devres subgroup. > > + * > > + * Return: A reference to an initialized protocol instance or error on failure. > > + */ > > +static struct scmi_protocol_instance * __must_check > > +scmi_get_protocol_instance(struct scmi_handle *handle, u8 protocol_id) > > +{ > > + int ret = -ENOMEM; > > + void *gid; > > + struct scmi_protocol_instance *pi; > > + struct scmi_info *info = handle_to_scmi_info(handle); > > + > > + mutex_lock(&info->protocols_mtx); > > + /* Ensure protocols has been updated */ > > + smp_rmb(); > > Why do you need this smp_rmb and the smp_wmb below ? > > Isn't the mutex not enough ? > You're right, it's a leftover I think, since initially there was no mutex. I'll remove it in V2. Thanks Cristian > > + pi = info->protocols[protocol_id]; > > + > > + if (!pi) { > > + const struct scmi_protocol *proto; > > + > > + /* Fail if protocol not registered on bus */ > > + proto = scmi_get_protocol(protocol_id); > > + if (!proto) { > > + ret = -EINVAL; > > + goto out; > > + } > > + > > + /* Protocol specific devres group */ > > + gid = devres_open_group(handle->dev, NULL, GFP_KERNEL); > > + if (!gid) > > + goto out; > > + > > + pi = devm_kzalloc(handle->dev, sizeof(*pi), GFP_KERNEL); > > + if (!pi) > > + goto clean; > > + > > + pi->gid = gid; > > + pi->proto = proto; > > + refcount_set(&pi->users, 1); > > + /* proto->init is assured NON NULL by scmi_protocol_register */ > > + ret = pi->proto->init(handle); > > + if (ret) > > + goto clean; > > + > > + info->protocols[protocol_id] = pi; > > + /* Ensure initialized protocol is visible */ > > + smp_wmb(); > > + > > + devres_close_group(handle->dev, pi->gid); > > + dev_dbg(handle->dev, "Initialized protocol: 0x%X\n", > > + protocol_id); > > + } else { > > + refcount_inc(&pi->users); > > + } > > + mutex_unlock(&info->protocols_mtx); > > + > > + return pi; > > + > > +clean: > > + devres_release_group(handle->dev, gid); > > +out: > > + mutex_unlock(&info->protocols_mtx); > > + return ERR_PTR(ret); > > +} > > + > > +/** > > + * scmi_acquire_protocol - Protocol acquire > > + * @handle: A reference to the SCMI platform instance. > > + * @protocol_id: The protocol being requested. > > + * > > + * Register a new user for the requested protocol on the specified SCMI > > + * platform instance, possibly triggering its initialization on first user. > > + * > > + * Return: 0 if protocol was acquired successfully. > > + */ > > +int scmi_acquire_protocol(struct scmi_handle *handle, u8 protocol_id) > > +{ > > + return IS_ERR(scmi_get_protocol_instance(handle, protocol_id)); > > +} > > + > > +/** > > + * scmi_release_protocol - Protocol de-initialization helper. > > + * @handle: A reference to the SCMI platform instance. > > + * @protocol_id: The protocol being requested. > > + * > > + * Remove one user for the specified protocol and triggers de-initialization > > + * and resources de-allocation once the last user has gone. > > + */ > > +void scmi_release_protocol(struct scmi_handle *handle, u8 protocol_id) > > +{ > > + struct scmi_info *info = handle_to_scmi_info(handle); > > + struct scmi_protocol_instance *pi; > > + > > + mutex_lock(&info->protocols_mtx); > > + /* Ensure protocols has been updated */ > > + smp_rmb(); > > + pi = info->protocols[protocol_id]; > > + if (WARN_ON(!pi)) { > > + mutex_unlock(&info->protocols_mtx); > > + return; > > + } > > + > > + if (refcount_dec_and_test(&pi->users)) { > > + void *gid = pi->gid; > > + > > + if (pi->proto->deinit) > > + pi->proto->deinit(handle); > > + > > + info->protocols[protocol_id] = NULL; > > + /* Ensure deinitialized protocol is visible */ > > + smp_wmb(); > > + > > + devres_release_group(handle->dev, gid); > > + dev_dbg(handle->dev, "De-Initialized protocol: 0x%X\n", > > + protocol_id); > > + } > > + mutex_unlock(&info->protocols_mtx); > > +} > > + > > void scmi_setup_protocol_implemented(const struct scmi_handle *handle, > > u8 *prot_imp) > > { > > @@ -785,6 +934,7 @@ static int scmi_probe(struct platform_device *pdev) > > info->dev = dev; > > info->desc = desc; > > INIT_LIST_HEAD(&info->node); > > + mutex_init(&info->protocols_mtx); > > > > platform_set_drvdata(pdev, info); > > idr_init(&info->tx_idr); > > @@ -805,10 +955,14 @@ static int scmi_probe(struct platform_device *pdev) > > if (scmi_notification_init(handle)) > > dev_err(dev, "SCMI Notifications NOT available.\n"); > > > > - ret = scmi_base_protocol_init(handle); > > - if (ret) { > > - dev_err(dev, "unable to communicate with SCMI(%d)\n", ret); > > - return ret; > > + /* > > + * Trigger SCMI Base protocol initialization. > > + * It's mandatory and won't be ever released/deinit until the > > + * SCMI stack is shutdown/unloaded as a whole. > > + */ > > + if (scmi_acquire_protocol(handle, SCMI_PROTOCOL_BASE)) { > > + dev_err(dev, "unable to communicate with SCMI\n"); > > + return -ENODEV; > > } > > > > mutex_lock(&scmi_list_mutex); > > @@ -941,6 +1095,8 @@ static int __init scmi_driver_init(void) > > { > > scmi_bus_init(); > > > > + scmi_base_register(); > > + > > scmi_clock_register(); > > scmi_perf_register(); > > scmi_power_register(); > > @@ -954,7 +1110,7 @@ subsys_initcall(scmi_driver_init); > > > > static void __exit scmi_driver_exit(void) > > { > > - scmi_bus_exit(); > > + scmi_base_unregister(); > > > > scmi_clock_unregister(); > > scmi_perf_unregister(); > > @@ -963,6 +1119,8 @@ static void __exit scmi_driver_exit(void) > > scmi_sensors_unregister(); > > scmi_system_unregister(); > > > > + scmi_bus_exit(); > > + > > platform_driver_unregister(&scmi_driver); > > } > > module_exit(scmi_driver_exit); > > diff --git a/drivers/firmware/arm_scmi/notify.c b/drivers/firmware/arm_scmi/notify.c > > index c24e427dce0d..eae58b2a92cc 100644 > > --- a/drivers/firmware/arm_scmi/notify.c > > +++ b/drivers/firmware/arm_scmi/notify.c > > @@ -91,10 +91,9 @@ > > #include > > #include > > > > +#include "common.h" > > #include "notify.h" > > > > -#define SCMI_MAX_PROTO 256 > > - > > #define PROTO_ID_MASK GENMASK(31, 24) > > #define EVT_ID_MASK GENMASK(23, 16) > > #define SRC_ID_MASK GENMASK(15, 0) > > diff --git a/drivers/firmware/arm_scmi/perf.c b/drivers/firmware/arm_scmi/perf.c > > index 82fb3babff72..854460a051c2 100644 > > --- a/drivers/firmware/arm_scmi/perf.c > > +++ b/drivers/firmware/arm_scmi/perf.c > > @@ -2,7 +2,7 @@ > > /* > > * System Control and Management Interface (SCMI) Performance Protocol > > * > > - * Copyright (C) 2018 ARM Ltd. > > + * Copyright (C) 2018-2020 ARM Ltd. > > */ > > > > #define pr_fmt(fmt) "SCMI Notifications PERF - " fmt > > @@ -892,4 +892,10 @@ static int scmi_perf_protocol_init(struct scmi_handle *handle) > > return 0; > > } > > > > -DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(SCMI_PROTOCOL_PERF, perf) > > +static struct scmi_protocol scmi_perf = { > > + .id = SCMI_PROTOCOL_PERF, > > + .init = &scmi_perf_protocol_init, > > + .ops = &perf_ops, > > +}; > > + > > +DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(perf, scmi_perf) > > diff --git a/drivers/firmware/arm_scmi/power.c b/drivers/firmware/arm_scmi/power.c > > index 1f37258e9bee..42c9c88da07c 100644 > > --- a/drivers/firmware/arm_scmi/power.c > > +++ b/drivers/firmware/arm_scmi/power.c > > @@ -2,7 +2,7 @@ > > /* > > * System Control and Management Interface (SCMI) Power Protocol > > * > > - * Copyright (C) 2018 ARM Ltd. > > + * Copyright (C) 2018-2020 ARM Ltd. > > */ > > > > #define pr_fmt(fmt) "SCMI Notifications POWER - " fmt > > @@ -301,4 +301,10 @@ static int scmi_power_protocol_init(struct scmi_handle *handle) > > return 0; > > } > > > > -DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(SCMI_PROTOCOL_POWER, power) > > +static struct scmi_protocol scmi_power = { > > + .id = SCMI_PROTOCOL_POWER, > > + .init = &scmi_power_protocol_init, > > + .ops = &power_ops, > > +}; > > + > > +DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(power, scmi_power) > > diff --git a/drivers/firmware/arm_scmi/reset.c b/drivers/firmware/arm_scmi/reset.c > > index a981a22cfe89..2caf0bdb6fdc 100644 > > --- a/drivers/firmware/arm_scmi/reset.c > > +++ b/drivers/firmware/arm_scmi/reset.c > > @@ -2,7 +2,7 @@ > > /* > > * System Control and Management Interface (SCMI) Reset Protocol > > * > > - * Copyright (C) 2019 ARM Ltd. > > + * Copyright (C) 2019-2020 ARM Ltd. > > */ > > > > #define pr_fmt(fmt) "SCMI Notifications RESET - " fmt > > @@ -311,4 +311,10 @@ static int scmi_reset_protocol_init(struct scmi_handle *handle) > > return 0; > > } > > > > -DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(SCMI_PROTOCOL_RESET, reset) > > +static struct scmi_protocol scmi_reset = { > > + .id = SCMI_PROTOCOL_RESET, > > + .init = &scmi_reset_protocol_init, > > + .ops = &reset_ops, > > +}; > > + > > +DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(reset, scmi_reset) > > diff --git a/drivers/firmware/arm_scmi/sensors.c b/drivers/firmware/arm_scmi/sensors.c > > index b4232d611033..dfe3076d2093 100644 > > --- a/drivers/firmware/arm_scmi/sensors.c > > +++ b/drivers/firmware/arm_scmi/sensors.c > > @@ -2,7 +2,7 @@ > > /* > > * System Control and Management Interface (SCMI) Sensor Protocol > > * > > - * Copyright (C) 2018 ARM Ltd. > > + * Copyright (C) 2018-2020 ARM Ltd. > > */ > > > > #define pr_fmt(fmt) "SCMI Notifications SENSOR - " fmt > > @@ -367,4 +367,10 @@ static int scmi_sensors_protocol_init(struct scmi_handle *handle) > > return 0; > > } > > > > -DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(SCMI_PROTOCOL_SENSOR, sensors) > > +static struct scmi_protocol scmi_sensors = { > > + .id = SCMI_PROTOCOL_SENSOR, > > + .init = &scmi_sensors_protocol_init, > > + .ops = &sensor_ops, > > +}; > > + > > +DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(sensors, scmi_sensors) > > diff --git a/drivers/firmware/arm_scmi/system.c b/drivers/firmware/arm_scmi/system.c > > index 283e12d5f24b..bcea18bf54ab 100644 > > --- a/drivers/firmware/arm_scmi/system.c > > +++ b/drivers/firmware/arm_scmi/system.c > > @@ -128,4 +128,10 @@ static int scmi_system_protocol_init(struct scmi_handle *handle) > > return 0; > > } > > > > -DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(SCMI_PROTOCOL_SYSTEM, system) > > +static struct scmi_protocol scmi_system = { > > + .id = SCMI_PROTOCOL_SYSTEM, > > + .init = &scmi_system_protocol_init, > > + .ops = NULL, > > +}; > > + > > +DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(system, scmi_system) > > diff --git a/include/linux/scmi_protocol.h b/include/linux/scmi_protocol.h > > index 9cd312a1ff92..ca23d682941e 100644 > > --- a/include/linux/scmi_protocol.h > > +++ b/include/linux/scmi_protocol.h > > @@ -376,9 +376,9 @@ static inline void scmi_driver_unregister(struct scmi_driver *driver) {} > > #define module_scmi_driver(__scmi_driver) \ > > module_driver(__scmi_driver, scmi_register, scmi_unregister) > > > > -typedef int (*scmi_prot_init_fn_t)(struct scmi_handle *); > > -int scmi_protocol_register(int protocol_id, scmi_prot_init_fn_t fn); > > -void scmi_protocol_unregister(int protocol_id); > > +struct scmi_protocol; > > +int scmi_protocol_register(struct scmi_protocol *proto); > > +void scmi_protocol_unregister(const struct scmi_protocol *proto); > > > > /* SCMI Notification API - Custom Event Reports */ > > enum scmi_notification_events { > > -- > > 2.17.1 > >