Received: by 2002:a05:6358:45e:b0:b5:b6eb:e1f9 with SMTP id 30csp4301648rwe; Tue, 30 Aug 2022 07:54:45 -0700 (PDT) X-Google-Smtp-Source: AA6agR6Nr/gjZcpU9BzBfrI32KT2iQJton+2tpDPHHuJZ7HD/8kGvbT9TU8TvhukMuOFVV+W5M9I X-Received: by 2002:a17:906:fc6:b0:72f:d080:416 with SMTP id c6-20020a1709060fc600b0072fd0800416mr17339207ejk.1.1661871285220; Tue, 30 Aug 2022 07:54:45 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1661871285; cv=none; d=google.com; s=arc-20160816; b=CXYWKkCwx3p9siIkeMAaDU0re3Kv/c4IqCH78i1dKWPOgVs321wIyj7t39dx+9bIcM UltJeA0L5e+nyJB+tVlicEDDANTkHwB4A9owxqtA6WYp9jMG0X9PWz1p2oz/08J+ZR7p swPSiub+vNLr2Mc/U1HOUdXj1ezz7Qt+2G7Y96HsOiJbLoq3MAo2j8VVDJ52AeATJXH6 XV1pM8VeF0SR1erpWtPZFZr9F4IxW7E50QeSx0qCw2NTQuVslFd914KXy6wOSgjmNOwz V75IplLT3AVfRbrbB0xNmwOoKn3pU59KkAksOXVIykEAzGm+G3+RXNgopOqRM6qO5XTh JWtw== 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:dkim-signature; bh=Zex3UDcuXi6on9TVwM9aa8TlelCGRWq6XcPdeEq0Btc=; b=lm1xIui9IeIKHDXbNNIJ6SNqC90BpRGS9eAVxLmg4+UqdUXlzKGWvxzysteebrOYyf 6hCt+xsniNpH+/r7BxdYYJ28ZZknS4EOLHpfaQN/X0AUb8rosyUdKH963jP8WC8pq1/4 ZNbXV89QlemOuXtXuecqGVaj3VgzgZHv+NrRsC5QIqK0njkVhcS4EuQk9H5zB9eHcTTP zFW7mxDmYyib+/CxMrFjngVcQALh/O0yQcvS7nvp/NiDOVXXybiJoT5LNgtLKWn7CCjj 0zL0VoXXF9VfThNdq74fAyCRPStNRp/dUZMbOt6TrCYS4W4GAhRb1ekUXTgVefV0sA8p LXnw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=AMX+w5tS; 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=NONE dis=NONE) header.from=linaro.org Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id m6-20020a17090679c600b00730aa3e3843si8398894ejo.131.2022.08.30.07.54.19; Tue, 30 Aug 2022 07:54:45 -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=@linaro.org header.s=google header.b=AMX+w5tS; 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=NONE dis=NONE) header.from=linaro.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S230312AbiH3OnI (ORCPT + 99 others); Tue, 30 Aug 2022 10:43:08 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:43216 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229916AbiH3OnF (ORCPT ); Tue, 30 Aug 2022 10:43:05 -0400 Received: from mail-wm1-x32f.google.com (mail-wm1-x32f.google.com [IPv6:2a00:1450:4864:20::32f]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id CDF55BD772 for ; Tue, 30 Aug 2022 07:43:03 -0700 (PDT) Received: by mail-wm1-x32f.google.com with SMTP id r204-20020a1c44d5000000b003a84b160addso1987723wma.2 for ; Tue, 30 Aug 2022 07:43:03 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; h=content-transfer-encoding:in-reply-to:from:references:cc:to :content-language:subject:user-agent:mime-version:date:message-id :from:to:cc; bh=Zex3UDcuXi6on9TVwM9aa8TlelCGRWq6XcPdeEq0Btc=; b=AMX+w5tS+SI7L6d39SCRygk3KPxJxH6BBJXRVl/gqrEFC7ivS/r5KP1jpHPBvKjH7T 7ahUimipG2N+l16nzOe8oRFqMFHhrcnGTJ3WdX/dxvyzr88HZsJ/jXXfyGPWE2p8L3Je YJ91tE1bbbyICoy1qqWHuiJoYdxyncXmQdf7wCOWwGobh39l0/ySfaoWUY9Zn8D/aDjo ruRm0PR5E2uIKDWlwBu46BWP7kcbz+vqSKVJFLFRpGJEMRKXebT19ht9G4870f5lp4vI AAs7d0UyZB4XR7iRMI0dSJ9o5HhxcsB7WhzshdgBVCbAExwgvk1ySEy/7zeWDeZKf9Jl dPJA== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=content-transfer-encoding:in-reply-to:from:references:cc:to :content-language:subject:user-agent:mime-version:date:message-id :x-gm-message-state:from:to:cc; bh=Zex3UDcuXi6on9TVwM9aa8TlelCGRWq6XcPdeEq0Btc=; b=GrouokuD54zLIErb9ZURzJ3SBWLfcJ/HwMNWWYaNoNCk24Dy8jmP8ti75xAmOH5ftc Y/9uJq5iVAlzuyBBHR8iE5jm9d8YTZIkvUHINmCofuy9rB7+mttwzzjSw5kiLciOOW7o gVvU4DSc58TurTUWmwpfaUT3oF1Kb3W/zJyoe/dqqIm9gt8M2d5CZof+oYCGc0hIWK3n 6Va4OAwhrxYLUN0RvVRqR8/iUN6cM1YW7uPBPmeucwikW8doxPqLMTu8nUVXtbsWl3DC s5RLo0b0LFigetzMKIzwArXeY9YL4OujCgodxyRRCgpTVV6TwdV2M25uHVfwU+bMmw29 TJDw== X-Gm-Message-State: ACgBeo2QSTMkALxGqhmzTzM6uGKXL0kztKW12itk1ko6mdyKA92rO3e1 zP9BwqKrivDx8HBpQbHXHVfVuQ== X-Received: by 2002:a05:600c:3844:b0:3a6:19a:d980 with SMTP id s4-20020a05600c384400b003a6019ad980mr9964395wmr.65.1661870582325; Tue, 30 Aug 2022 07:43:02 -0700 (PDT) Received: from [192.168.86.238] (cpc90716-aztw32-2-0-cust825.18-1.cable.virginm.net. [86.26.103.58]) by smtp.googlemail.com with ESMTPSA id a22-20020a05600c225600b003a3442f1229sm12891999wmm.29.2022.08.30.07.43.01 (version=TLS1_3 cipher=TLS_AES_128_GCM_SHA256 bits=128/128); Tue, 30 Aug 2022 07:43:01 -0700 (PDT) Message-ID: <815f8e22-3a23-ebdb-7476-14682d0b3287@linaro.org> Date: Tue, 30 Aug 2022 15:43:00 +0100 MIME-Version: 1.0 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:91.0) Gecko/20100101 Thunderbird/91.11.0 Subject: Re: [PATCH v1 06/14] nvmem: core: introduce NVMEM layouts Content-Language: en-US To: Michael Walle Cc: Miquel Raynal , Richard Weinberger , Vignesh Raghavendra , Rob Herring , Krzysztof Kozlowski , Shawn Guo , Li Yang , =?UTF-8?B?UmFmYcWCIE1pxYJlY2tp?= , "David S . Miller" , Eric Dumazet , Jakub Kicinski , Paolo Abeni , Frank Rowand , linux-mtd@lists.infradead.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, linux-arm-kernel@lists.infradead.org, netdev@vger.kernel.org, Ahmad Fatoum References: <20220825214423.903672-1-michael@walle.cc> <20220825214423.903672-7-michael@walle.cc> From: Srinivas Kandagatla In-Reply-To: Content-Type: text/plain; charset=UTF-8; format=flowed Content-Transfer-Encoding: 8bit X-Spam-Status: No, score=-2.1 required=5.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,NICE_REPLY_A,RCVD_IN_DNSWL_NONE, SPF_HELO_NONE,SPF_PASS,T_SCC_BODY_TEXT_LINE autolearn=unavailable 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 On 30/08/2022 15:24, Michael Walle wrote: > Am 2022-08-30 15:36, schrieb Srinivas Kandagatla: >> On 25/08/2022 22:44, Michael Walle wrote: >>> NVMEM layouts are used to generate NVMEM cells during runtime. Think of >>> an EEPROM with a well-defined conent. For now, the content can be >>> described by a device tree or a board file. But this only works if the >>> offsets and lengths are static and don't change. One could also argue >>> that putting the layout of the EEPROM in the device tree is the wrong >>> place. Instead, the device tree should just have a specific compatible >>> string. >>> >>> Right now there are two use cases: >>>   (1) The NVMEM cell needs special processing. E.g. if it only specifies >>>       a base MAC address offset and you need to add an offset, or it >>>       needs to parse a MAC from ASCII format or some proprietary format. >>>       (Post processing of cells is added in a later commit). >>>   (2) u-boot environment parsing. The cells don't have a particular >>>       offset but it needs parsing the content to determine the offsets >>>       and length. >>> >>> Signed-off-by: Michael Walle >>> --- >>>   drivers/nvmem/Kconfig          |  2 ++ >>>   drivers/nvmem/Makefile         |  1 + >>>   drivers/nvmem/core.c           | 57 ++++++++++++++++++++++++++++++++++ >>>   drivers/nvmem/layouts/Kconfig  |  5 +++ >>>   drivers/nvmem/layouts/Makefile |  4 +++ >>>   include/linux/nvmem-provider.h | 38 +++++++++++++++++++++++ >>>   6 files changed, 107 insertions(+) >>>   create mode 100644 drivers/nvmem/layouts/Kconfig >>>   create mode 100644 drivers/nvmem/layouts/Makefile >> >> update to ./Documentation/driver-api/nvmem.rst would help others. > > Sure. Didn't know about that one. > >>> diff --git a/drivers/nvmem/Kconfig b/drivers/nvmem/Kconfig >>> index bab8a29c9861..1416837b107b 100644 >>> --- a/drivers/nvmem/Kconfig >>> +++ b/drivers/nvmem/Kconfig >>> @@ -357,4 +357,6 @@ config NVMEM_U_BOOT_ENV >>>           If compiled as module it will be called nvmem_u-boot-env. >>>   +source "drivers/nvmem/layouts/Kconfig" >>> + >>>   endif >>> diff --git a/drivers/nvmem/Makefile b/drivers/nvmem/Makefile >>> index 399f9972d45b..cd5a5baa2f3a 100644 >>> --- a/drivers/nvmem/Makefile >>> +++ b/drivers/nvmem/Makefile >>> @@ -5,6 +5,7 @@ >>>     obj-$(CONFIG_NVMEM)        += nvmem_core.o >>>   nvmem_core-y            := core.o >>> +obj-y                += layouts/ >>>     # Devices >>>   obj-$(CONFIG_NVMEM_BCM_OCOTP)    += nvmem-bcm-ocotp.o >>> diff --git a/drivers/nvmem/core.c b/drivers/nvmem/core.c >>> index 3dfd149374a8..5357fc378700 100644 >>> --- a/drivers/nvmem/core.c >>> +++ b/drivers/nvmem/core.c >>> @@ -74,6 +74,9 @@ static LIST_HEAD(nvmem_lookup_list); >>>     static BLOCKING_NOTIFIER_HEAD(nvmem_notifier); >>>   +static DEFINE_SPINLOCK(nvmem_layout_lock); >>> +static LIST_HEAD(nvmem_layouts); >>> + >>>   static int __nvmem_reg_read(struct nvmem_device *nvmem, unsigned >>> int offset, >>>                   void *val, size_t bytes) >>>   { >>> @@ -744,6 +747,56 @@ static int nvmem_add_cells_from_of(struct >>> nvmem_device *nvmem) >>>       return 0; >>>   } >>>   +int nvmem_register_layout(struct nvmem_layout *layout) >>> +{ >>> +    spin_lock(&nvmem_layout_lock); >>> +    list_add(&layout->node, &nvmem_layouts); >>> +    spin_unlock(&nvmem_layout_lock); >>> + >>> +    return 0; >>> +} >>> +EXPORT_SYMBOL_GPL(nvmem_register_layout); >> >> we should provide nvmem_unregister_layout too, so that providers can >> add them if they can in there respective drivers. > > Actually, that was the idea; that you can have layouts outside of layouts/. > I also had a nvmem_unregister_layout() but removed it because it was dead > code. Will re-add it again. > >>> + >>> +static struct nvmem_layout *nvmem_get_compatible_layout(struct >>> device_node *np) >>> +{ >>> +    struct nvmem_layout *p, *ret = NULL; >>> + >>> +    spin_lock(&nvmem_layout_lock); >>> + >>> +    list_for_each_entry(p, &nvmem_layouts, node) { >>> +        if (of_match_node(p->of_match_table, np)) { >>> +            ret = p; >>> +            break; >>> +        } >>> +    } >>> + >>> +    spin_unlock(&nvmem_layout_lock); >>> + >>> +    return ret; >>> +} >>> + >>> +static int nvmem_add_cells_from_layout(struct nvmem_device *nvmem) >>> +{ >>> +    struct nvmem_layout *layout; >>> + >>> +    layout = nvmem_get_compatible_layout(nvmem->dev.of_node); >>> +    if (layout) >>> +        layout->add_cells(&nvmem->dev, nvmem, layout); >> >> access to add_cells can crash hear as we did not check it before >> adding in to list. >> Or >> we could relax add_cells callback for usecases like imx-octop. > > good catch, will use layout && layout->add_cells. > >>> + >>> +    return 0; >>> +} >>> + >>> +const void *nvmem_layout_get_match_data(struct nvmem_device *nvmem, >>> +                    struct nvmem_layout *layout) >>> +{ >>> +    const struct of_device_id *match; >>> + >>> +    match = of_match_node(layout->of_match_table, nvmem->dev.of_node); >>> + >>> +    return match ? match->data : NULL; >>> +} >>> +EXPORT_SYMBOL_GPL(nvmem_layout_get_match_data); >>> + >>>   /** >>>    * nvmem_register() - Register a nvmem device for given nvmem_config. >>>    * Also creates a binary entry in >>> /sys/bus/nvmem/devices/dev-name/nvmem >>> @@ -872,6 +925,10 @@ struct nvmem_device *nvmem_register(const struct >>> nvmem_config *config) >>>       if (rval) >>>           goto err_remove_cells; >>>   +    rval = nvmem_add_cells_from_layout(nvmem); >>> +    if (rval) >>> +        goto err_remove_cells; >>> + >>>       blocking_notifier_call_chain(&nvmem_notifier, NVMEM_ADD, nvmem); >>>         return nvmem; >>> diff --git a/drivers/nvmem/layouts/Kconfig >>> b/drivers/nvmem/layouts/Kconfig >>> new file mode 100644 >>> index 000000000000..9ad3911d1605 >>> --- /dev/null >>> +++ b/drivers/nvmem/layouts/Kconfig >>> @@ -0,0 +1,5 @@ >>> +# SPDX-License-Identifier: GPL-2.0 >>> + >>> +menu "Layout Types" >>> + >>> +endmenu >>> diff --git a/drivers/nvmem/layouts/Makefile >>> b/drivers/nvmem/layouts/Makefile >>> new file mode 100644 >>> index 000000000000..6fdb3c60a4fa >>> --- /dev/null >>> +++ b/drivers/nvmem/layouts/Makefile >>> @@ -0,0 +1,4 @@ >>> +# SPDX-License-Identifier: GPL-2.0 >>> +# >>> +# Makefile for nvmem layouts. >>> +# >>> diff --git a/include/linux/nvmem-provider.h >>> b/include/linux/nvmem-provider.h >>> index e710404959e7..323685841e9f 100644 >>> --- a/include/linux/nvmem-provider.h >>> +++ b/include/linux/nvmem-provider.h >>> @@ -127,6 +127,28 @@ struct nvmem_cell_table { >>>       struct list_head    node; >>>   }; >>>   +/** >>> + * struct nvmem_layout - NVMEM layout definitions >>> + * >>> + * @name:        Layout name. >>> + * @of_match_table:    Open firmware match table. >>> + * @add_cells:        Will be called if a nvmem device is found which >>> + *            has this layout. The function will add layout >>> + *            specific cells with nvmem_add_one_cell(). >>> + * @node:        List node. >>> + * >>> + * A nvmem device can hold a well defined structure which can just be >>> + * evaluated during runtime. For example a TLV list, or a list of >>> "name=val" >>> + * pairs. A nvmem layout can parse the nvmem device and add appropriate >>> + * cells. >>> + */ >>> +struct nvmem_layout { >>> +    const char *name; >>> +    const struct of_device_id *of_match_table; >> >> looking at this, I think its doable to convert the existing >> cell_post_process callback to layouts by adding a layout specific >> callback here. > > can you elaborate on that? If we relax add_cells + add nvmem_unregister_layout() and update struct nvmem_layout to include post_process callback like struct nvmem_layout { const char *name; const struct of_device_id *of_match_table; int (*add_cells)(struct nvmem_device *nvmem, struct nvmem_layout *layout); struct list_head node; /* default callback for every cell */ nvmem_cell_post_process_t post_process; }; then we can move imx-ocotp to this layout style without add_cell callback, and finally get rid of cell_process_callback from both nvmem_config and nvmem_device. If layout specific post_process callback is available and cell does not have a callback set then we can can be either updated cell post_process callback with this one or invoke layout specific callback directly. does that make sense? --srini > > -michael