From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from bombadil.infradead.org (bombadil.infradead.org [198.137.202.133]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.lore.kernel.org (Postfix) with ESMTPS id D549CC43458 for ; Tue, 30 Jun 2026 09:37:55 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; d=lists.infradead.org; s=bombadil.20210309; h=Sender:List-Subscribe:List-Help :List-Post:List-Archive:List-Unsubscribe:List-Id:Content-Type: Content-Transfer-Encoding:MIME-Version:References:In-Reply-To:Message-ID:Date :Subject:To:From:Reply-To:Cc:Content-ID:Content-Description:Resent-Date: Resent-From:Resent-Sender:Resent-To:Resent-Cc:Resent-Message-ID:List-Owner; bh=wiSc3fsvkU6q3SUT9K9r0Ovpjm41u3R6Eb/8O4+S8H0=; b=XIiqAEE3oLTVb8lpPWYWy81RdC O9Q8w3kKTdL3zAaY7LhSHF3xXmEo95hPKW2OnCGlJY4PgfjzIzF62S0AqZMkg36HbhLUo8qk9yUVa tYH3EZNifAlIDKLrcy6M2oZ9jSaTkQnBEh6NR8G8nOUVNZhMKuiDyB/gliy5Tq/EOyfscWzgervUg YEecxQlxsqp0KYACUOqo6CHCp30GjoKkT+aMX9uWmC6db+ze3OMSMF9kA9cUG1LpR8l9c1oDOrrOr mC+S/jAmbMxtLWemen1k6cQ7CJjlxVUWRSzVlSUjcWEsoMgCtFCh3xBGwzcoIAoYg4t1RMB/Ne5ED QnwdC6qg==; Received: from localhost ([::1] helo=bombadil.infradead.org) by bombadil.infradead.org with esmtp (Exim 4.99.1 #2 (Red Hat Linux)) id 1weUuU-0000000GSkI-0KkW; Tue, 30 Jun 2026 09:37:42 +0000 Received: from esa.microchip.iphmx.com ([68.232.153.233]) by bombadil.infradead.org with esmtps (Exim 4.99.1 #2 (Red Hat Linux)) id 1weUuQ-0000000GSi3-0mN2 for linux-arm-kernel@lists.infradead.org; Tue, 30 Jun 2026 09:37:41 +0000 DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=microchip.com; i=@microchip.com; q=dns/txt; s=mchp; t=1782812258; x=1814348258; h=from:to:subject:date:message-id:in-reply-to:references: mime-version:content-transfer-encoding; bh=o7wv8iNpD0SrjN1YDK0Po2BIvdzTGQE50pX3AAbBwEU=; b=lf4VqCfwXfZAjWs3wGm8S3SCVtcITy7hjdDJ5ecwbYpJysczb0Itij1B 1fv3MYMfHH39bJ5BpHAb6PlekN3NeAzpObiuKxsVU2lVBQaDOINzsQixL VnjLX+JJSyA0U0bND5zL2tiAA25D61EiY+HoE9hXzy9i7Se0AGBkTRhzF fWoGjcAVbzBXzHpaQwnhCUTfvTczUc9ZrtoLw0EQiLwm93S7CiPc4JAOQ kpkhNc9q7WtnPhyFpHiR9Tddrd8VaJnXa+ApGNkR1Aoqr+7pnrK+jiVo9 p7zpZv5kvkewe61JuQ6thuNX3DFG7TEObTvhv7pXtK45hqvI40/ahz3CU w==; X-CSE-ConnectionGUID: rJT7i00AStqBvu9gGiugIw== X-CSE-MsgGUID: MNjLtfyGQ+yOIElKU1h07g== X-IronPort-AV: E=Sophos;i="6.24,233,1774335600"; d="scan'208";a="59880480" X-Amp-Result: SKIPPED(no attachment in message) Received: from unknown (HELO email.microchip.com) ([170.129.1.10]) by esa3.microchip.iphmx.com with ESMTP/TLS/ECDHE-RSA-AES128-GCM-SHA256; 30 Jun 2026 02:37:37 -0700 Received: from chn-vm-ex02.mchp-main.com (10.10.85.144) by chn-vm-ex04.mchp-main.com (10.10.85.152) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.1.2507.58; Tue, 30 Jun 2026 02:37:37 -0700 Received: from che-ll-i67070.microchip.com (10.10.85.11) by chn-vm-ex02.mchp-main.com (10.10.85.144) with Microsoft SMTP Server id 15.1.2507.58 via Frontend Transport; Tue, 30 Jun 2026 02:37:28 -0700 From: Varshini Rajendran To: , , , , , , , , , , , , , , , , , , , , , , , Subject: [PATCH v3 06/13] nvmem: microchip-otpc: add tag-based packet lookup Date: Tue, 30 Jun 2026 15:05:56 +0530 Message-ID: <20260630093603.38663-7-varshini.rajendran@microchip.com> X-Mailer: git-send-email 2.34.1 In-Reply-To: <20260630093603.38663-1-varshini.rajendran@microchip.com> References: <20260630093603.38663-1-varshini.rajendran@microchip.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Content-Type: text/plain X-CRM114-Version: 20100106-BlameMichelson ( TRE 0.9.0 (BSD) ) MR-646709E3 X-CRM114-CacheID: sfid-20260630_023739_592798_AA4C9637 X-CRM114-Status: GOOD ( 33.33 ) X-BeenThere: linux-arm-kernel@lists.infradead.org X-Mailman-Version: 2.1.34 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Sender: "linux-arm-kernel" Errors-To: linux-arm-kernel-bounces+linux-arm-kernel=archiver.kernel.org@lists.infradead.org Add support for accessing OTP packets by their 4-byte ASCII tag while preserving backward compatibility with the existing ID-based lookup. The OTP memory layout can vary across devices and may change over time, making the packet ID approach unreliable when the memory map is not known in advance. The packet tag provides a reliable way to identify and access packets without prior knowledge of the OTP memory layout. Two offset encoding are now supported: 1. Legacy ID-based: offset = OTP_PKT(id) = id * 4 Used in DT as: reg = ; 2. TAG-based: offset = 4-byte ASCII packet tag Used in DT as: reg = <0x41435354 0x4c>; (tag "ACST") The driver resolves offsets matching valid legacy selectors (multiples of 4 within the packet count) through ID lookup, falling back to tag lookup for other values. This ensures existing device trees continue to work while enabling new tag-based access. During probe, packet meta data including the tag is read and cached. The driver also validates OTP memory accessibility and emulation mode status. When the boot packet is not configured, emulation mode allows access to the other packets. When both are not available an informational message is logged. The stride of the nvmem memory is set to 1 in order to support tag based offsets, comment in the header file is updated accordingly. Signed-off-by: Varshini Rajendran --- drivers/nvmem/microchip-otpc.c | 143 ++++++++++++++++-- .../nvmem/microchip,sama7g5-otpc.h | 4 +- 2 files changed, 136 insertions(+), 11 deletions(-) diff --git a/drivers/nvmem/microchip-otpc.c b/drivers/nvmem/microchip-otpc.c index df979e8549fd..bf8589048e17 100644 --- a/drivers/nvmem/microchip-otpc.c +++ b/drivers/nvmem/microchip-otpc.c @@ -18,16 +18,20 @@ #define MCHP_OTPC_CR_READ BIT(6) #define MCHP_OTPC_MR (0x4) #define MCHP_OTPC_MR_ADDR GENMASK(31, 16) +#define MCHP_OTPC_MR_EMUL BIT(7) #define MCHP_OTPC_AR (0x8) #define MCHP_OTPC_SR (0xc) #define MCHP_OTPC_SR_READ BIT(6) #define MCHP_OTPC_HR (0x20) #define MCHP_OTPC_HR_SIZE GENMASK(15, 8) +#define MCHP_OTPC_HR_PACKET_TYPE GENMASK(2, 0) #define MCHP_OTPC_DR (0x24) #define MCHP_OTPC_NAME "mchp-otpc" #define MCHP_OTPC_SIZE (11 * 1024) +#define PACKET_TYPE_REGULAR 1 + /** * struct mchp_otpc - OTPC private data structure * @base: base address @@ -47,11 +51,15 @@ struct mchp_otpc { * @list: list head * @id: packet ID * @offset: packet offset (in words) in OTP memory + * @type: type of the packet + * @tag: 4-byte ASCII tag of the packet */ struct mchp_otpc_packet { struct list_head list; u32 id; u32 offset; + u32 type; + u32 tag; }; static struct mchp_otpc_packet *mchp_otpc_id_to_packet(struct mchp_otpc *otpc, @@ -70,6 +78,56 @@ static struct mchp_otpc_packet *mchp_otpc_id_to_packet(struct mchp_otpc *otpc, return NULL; } +/** + * mchp_otpc_tag_to_packet() - find packet by tag + * @otpc: OTPC private data + * @tag: 4-byte ASCII tag to search for + * + * Return: pointer to packet if found, NULL otherwise + */ +static struct mchp_otpc_packet *mchp_otpc_tag_to_packet(struct mchp_otpc *otpc, + u32 tag) +{ + struct mchp_otpc_packet *packet; + + list_for_each_entry(packet, &otpc->packets, list) { + if (packet->tag == tag) + return packet; + } + + return NULL; +} + +/** + * mchp_otpc_resolve_packet() - resolve offset to packet + * @otpc: OTPC private data + * @off: NVMEM offset (legacy ID-based or TAG-based) + * + * Legacy offsets (multiples of 4 within valid ID range) are resolved + * through ID lookup. Other offsets are treated as 4-byte ASCII tags. + * + * Return: pointer to packet if found, NULL otherwise + */ +static struct mchp_otpc_packet *mchp_otpc_resolve_packet(struct mchp_otpc *otpc, + u32 off) +{ + /* + * Legacy id based packet access: offset = id * 4 + * Inside the driver we use continuous unsigned integer numbers + * for packet id, thus divide off by 4 before passing it to + * mchp_otpc_id_to_packet(). + */ + u32 id = off / 4; + + if (!(off % 4) && id < otpc->npackets) + return mchp_otpc_id_to_packet(otpc, id); + + /* + * TAG-based packet access: offset is a 4-byte ASCII tag + */ + return mchp_otpc_tag_to_packet(otpc, off); +} + static int mchp_otpc_prepare_read(struct mchp_otpc *otpc, unsigned int offset) { @@ -140,8 +198,29 @@ static int mchp_otpc_prepare_read(struct mchp_otpc *otpc, * offset returned by hardware. * * For this, the read function will return the first requested bytes in the - * packet. The user will have to be aware of the memory footprint before doing - * the read request. + * packet. + * + * Two offset encoding are supported: + * + * 1. Legacy ID-based: offset = OTP_PKT(id) = id * 4 + * Used in DT as: reg = ; + * 2. TAG-based: offset = 4-byte ASCII packet tag + * Used in DT as: reg = <0x41435354 0x4c>; (tag "ACST") + * + * To use the legacy ID based packet lookup the user will have to be aware of + * the memory footprint before doing the read request. + * + * But by using the TAG based packet lookup, the user won't have to be aware + * of the memory footprint before doing the read request since this driver has + * it abstracted and taken care of. + * + * Practically, there is no way of knowing the mapping of the OTP memory table + * in advance for every device. But by using the packet tag - the identifier + * ASCII value, the packets can be recognized without being aware of the + * flashed OTP memory map table and the payload can be acquired reliably. + * + * While the legacy ID based lookup is still supported, TAG based approach is + * recommended. */ static int mchp_otpc_read(void *priv, unsigned int off, void *val, size_t bytes) @@ -154,12 +233,11 @@ static int mchp_otpc_read(void *priv, unsigned int off, void *val, int ret, payload_size; /* - * We reach this point with off being multiple of stride = 4 to - * be able to cross the subsystem. Inside the driver we use continuous - * unsigned integer numbers for packet id, thus divide off by 4 - * before passing it to mchp_otpc_id_to_packet(). + * From this point the offset has to be translated into the actual + * packet. For this we traverse the table of contents stored in a list + * "packet" based on the access type - packet id or tag. */ - packet = mchp_otpc_id_to_packet(otpc, off / 4); + packet = mchp_otpc_resolve_packet(otpc, off); if (!packet) return -EINVAL; offset = packet->offset; @@ -190,6 +268,29 @@ static int mchp_otpc_read(void *priv, unsigned int off, void *val, return 0; } +/** + * mchp_otpc_read_packet_tag() - read tag from packet payload + * @otpc: OTPC private data + * @offset: packet offset in OTP memory + * @val: pointer to store the tag value + * + * Return: 0 on success, negative errno on failure + */ +static int mchp_otpc_read_packet_tag(struct mchp_otpc *otpc, unsigned int offset, + unsigned int *val) +{ + int ret; + + ret = mchp_otpc_prepare_read(otpc, offset); + if (ret) + return ret; + + writel_relaxed(0, otpc->base + MCHP_OTPC_AR); + *val = readl_relaxed(otpc->base + MCHP_OTPC_DR); + + return 0; +} + static int mchp_otpc_init_packets_list(struct mchp_otpc *otpc, u32 *size) { struct mchp_otpc_packet *packet; @@ -215,6 +316,17 @@ static int mchp_otpc_init_packets_list(struct mchp_otpc *otpc, u32 *size) packet->id = id++; packet->offset = word_pos; + packet->type = FIELD_GET(MCHP_OTPC_HR_PACKET_TYPE, word); + + if (packet->type == PACKET_TYPE_REGULAR) { + ret = mchp_otpc_read_packet_tag(otpc, packet->offset, + &packet->tag); + if (ret) + return ret; + } else { + packet->tag = 0; + } + INIT_LIST_HEAD(&packet->list); list_add_tail(&packet->list, &otpc->packets); @@ -236,7 +348,7 @@ static struct nvmem_config mchp_nvmem_config = { .type = NVMEM_TYPE_OTP, .read_only = true, .word_size = 4, - .stride = 4, + .stride = 1, .reg_read = mchp_otpc_read, }; @@ -244,7 +356,8 @@ static int mchp_otpc_probe(struct platform_device *pdev) { struct nvmem_device *nvmem; struct mchp_otpc *otpc; - u32 size; + bool emul_enable; + u32 size, tmp; int ret; otpc = devm_kzalloc(&pdev->dev, sizeof(*otpc), GFP_KERNEL); @@ -256,10 +369,22 @@ static int mchp_otpc_probe(struct platform_device *pdev) return PTR_ERR(otpc->base); otpc->dev = &pdev->dev; + + tmp = readl_relaxed(otpc->base + MCHP_OTPC_MR); + emul_enable = tmp & MCHP_OTPC_MR_EMUL; + if (emul_enable) + dev_info(otpc->dev, "Emulation mode enabled\n"); + ret = mchp_otpc_init_packets_list(otpc, &size); if (ret) return ret; + if (!size) { + dev_warn(otpc->dev, "Cannot access OTP memory\n"); + if (!emul_enable) + dev_info(otpc->dev, "Boot packet not programmed and emulation mode disabled\n"); + } + mchp_nvmem_config.dev = otpc->dev; mchp_nvmem_config.add_legacy_fixed_of_cells = true; mchp_nvmem_config.size = size; diff --git a/include/dt-bindings/nvmem/microchip,sama7g5-otpc.h b/include/dt-bindings/nvmem/microchip,sama7g5-otpc.h index f570b23165a2..5f72e75ad091 100644 --- a/include/dt-bindings/nvmem/microchip,sama7g5-otpc.h +++ b/include/dt-bindings/nvmem/microchip,sama7g5-otpc.h @@ -4,8 +4,8 @@ #define _DT_BINDINGS_NVMEM_MICROCHIP_OTPC_H /* - * Need to have it as a multiple of 4 as NVMEM memory is registered with - * stride = 4. + * Need to have it as a multiple of 4 for the legacy id based packet + * access. */ #define OTP_PKT(id) ((id) * 4) -- 2.34.1