Received: by 2002:a25:f815:0:0:0:0:0 with SMTP id u21csp2830037ybd; Mon, 24 Jun 2019 13:31:41 -0700 (PDT) X-Google-Smtp-Source: APXvYqyHxJYQi8bNaPVQPgbbr8Gd8aNFHJgcFeOE04BrEMWeTHJqBvffBVyfYu1on629Vpl0Xkzv X-Received: by 2002:a17:90a:3225:: with SMTP id k34mr26655635pjb.31.1561408301673; Mon, 24 Jun 2019 13:31:41 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1561408301; cv=none; d=google.com; s=arc-20160816; b=QFqRhzoMJaf5aOosGzLs1nR/HJIqcegD6z+Hpv78NPEcSOpIF3/4JS64ENsb+Z6/Br FQD9JYQCZDGUhnuUUe6HH1XzUjZU+5qQ+AoFeaCWrE0Rtms16p9Slu4raC1jXhti/YNR +KRuPpck8gT4mdksDxAkO4oGpqXd3x8cx/rvbSOtGc/c2RzRgq36EW4yZYCTCbRZbg2D tGcHXZ7H/XQEZBVKELtEWiyl+9Nz5WXaRubnjSRVGg7nv7Knrkg5vS+CxSsZQtL+zlC4 S7zVN/fgduUNMFiDv7cc6SczNRUzdP2mHBvKDUsuXxhCqZzE9VZwNnJnq3l4PiSwii8m i2Ag== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-language :content-transfer-encoding:in-reply-to:mime-version:user-agent:date :message-id:from:references:cc:to:subject:dkim-signature; bh=rk32yi4TS8gcxPCzX8v7QY6cOglFHE8dF0I29LG2lnc=; b=YQQM2L9SY9M+As+r/FWeR0fCDDqhW4uIbq99hvEGy/eY9fH0er7frI8mYfjCYkv/6L aJnsqaHAhYrMVodsqL9lljlug/L9ZpVNv4Gx84MWvLwmgOjAm/s95vyubQ6jmAwZOQiD gTh7wO2VHj880MbpWevmGNd8AfzBgR4oYJrAgnCkPGtVlJkR733As3YeA61QIIGdnTTq ZROUBM9OC5yLKS3bRMp/FeHZOcfA9TGO07hLHgDG8BL9jG93DmwQThGn7euwzJqFn2R5 WILivl0EkAm1xI/FhlWKKGDf9DcLj0H24bxQSgK6TdxBLoO9yS7MHgHaHxvs7tYMztbd Bgmw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b="THWg/XKi"; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id w17si10729739pgj.190.2019.06.24.13.31.25; Mon, 24 Jun 2019 13:31:41 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b="THWg/XKi"; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1732196AbfFXQqV (ORCPT + 99 others); Mon, 24 Jun 2019 12:46:21 -0400 Received: from mail-pf1-f196.google.com ([209.85.210.196]:45256 "EHLO mail-pf1-f196.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727847AbfFXQqU (ORCPT ); Mon, 24 Jun 2019 12:46:20 -0400 Received: by mail-pf1-f196.google.com with SMTP id r1so7828763pfq.12; Mon, 24 Jun 2019 09:46:19 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=subject:to:cc:references:from:message-id:date:user-agent :mime-version:in-reply-to:content-transfer-encoding:content-language; bh=rk32yi4TS8gcxPCzX8v7QY6cOglFHE8dF0I29LG2lnc=; b=THWg/XKi4hT+eXURO6zCxBWh5cSpNTHmniF8SOoFsAEqFw3MYLE75X73Ad7B0Uoa/u Js0H1vA9IBcQT7Y2zHy34pz1g5j7v/pxeblkUMN1Rk1Xz1mPZ6gmwgJcNTFcmTXPsF08 SrSoJKUdi3xus6naqN3TllR7GRAsM4YE9Be/qcoLjROxw2xVWemXxpRXJ/fx3Yf90u1U DhvS26rMMyGG74u6F2J/dT609qVBy/gKV5dk5hA9Rsm0gJVfmuByPHgw2qcLYBZUnE1x 3KNGBWUu1sus3xLGGP5KtbBKJQ8PZnNvr48Gb77b8xwc8IJUi+KjhZ0uIdZ9udXpVBls p3QQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:subject:to:cc:references:from:message-id:date :user-agent:mime-version:in-reply-to:content-transfer-encoding :content-language; bh=rk32yi4TS8gcxPCzX8v7QY6cOglFHE8dF0I29LG2lnc=; b=obBZZvmdXJkTWnoEI5vC0QNde4bIrk/KL7tpxd1f7ZhmhbHVzr33bTefDNSJHW1RqN 5QDr1uPMqZwNEZ5qxlSzBhdXfyFy013Y7szBKcww8rhxYJd7eoq9I7cXyEVArprWSYGC T//S9JMH8G0oyo6flSdkhO8iAfRp+2Gk4siS2bE6bSi/CXOV+Esh2cWbBxl5S7y4XqSA 9BJ9TZcfZ9Hgc/fLSeH2G+fsJW7M77hCAlgdaWJa3qFZYyJVWHEA0M4ariia9zt7n33V FlZ+GITTpFomfyrd1PzvDbccRYwHX8BnBM1TJwLMB5jMOqgjy+aeSK+xOgAYSe2gWyAv mJSg== X-Gm-Message-State: APjAAAUlzW+cCmqheGilj+A/RlRZ2IbH+oJy8CLyBr8lSWCBvkrgUvmX M5EIGLv2iRzZIEOHR4Kiks0= X-Received: by 2002:a17:90a:c504:: with SMTP id k4mr25698395pjt.104.1561394778617; Mon, 24 Jun 2019 09:46:18 -0700 (PDT) Received: from ?IPv6:240b:10:2720:5510:a182:288:3ffa:432a? ([240b:10:2720:5510:a182:288:3ffa:432a]) by smtp.gmail.com with ESMTPSA id y22sm15041975pgj.38.2019.06.24.09.46.15 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Mon, 24 Jun 2019 09:46:17 -0700 (PDT) Subject: Re: [PATCH v7 1/5] mtd: cfi_cmdset_0002: Add support for polling status register To: Vignesh Raghavendra , Boris Brezillon , Marek Vasut , Richard Weinberger , Rob Herring Cc: devicetree@vger.kernel.org, Sergei Shtylyov , linux-kernel@vger.kernel.org, linux-mtd@lists.infradead.org, Miquel Raynal , Mason Yang , linux-arm-kernel@lists.infradead.org References: <20190620172250.9102-1-vigneshr@ti.com> <20190620172250.9102-2-vigneshr@ti.com> From: Tokunori Ikegami Message-ID: <571484c7-0cf4-6a7d-6d7f-375cfb13ce8b@gmail.com> Date: Tue, 25 Jun 2019 01:46:13 +0900 User-Agent: Mozilla/5.0 (Windows NT 10.0; WOW64; rv:60.0) Gecko/20100101 Thunderbird/60.7.2 MIME-Version: 1.0 In-Reply-To: <20190620172250.9102-2-vigneshr@ti.com> Content-Type: text/plain; charset=utf-8; format=flowed Content-Transfer-Encoding: 7bit Content-Language: en-US Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 2019/06/21 2:22, Vignesh Raghavendra wrote: > HyperFlash devices are compliant with CFI AMD/Fujitsu Extended Command > Set (0x0002) for flash operations, therefore > drivers/mtd/chips/cfi_cmdset_0002.c can be used as is. But these devices > do not support DQ polling method of determining chip ready/good status. > These flashes provide Status Register whose bits can be polled to know > status of flash operation. > > Cypress HyperFlash datasheet here[1], talks about CFI Amd/Fujitsu > Extended Query version 1.5. Bit 0 of "Software Features supported" field > of CFI Primary Vendor-Specific Extended Query table indicates > presence/absence of status register and Bit 1 indicates whether or not > DQ polling is supported. Using these bits, its possible to determine > whether flash supports DQ polling or need to use Status Register. > > Add support for polling Status Register to know device ready/status of > erase/write operations when DQ polling is not supported. > Print error messages on erase/program failure by looking at related > Status Register bits. > > [1] https://www.cypress.com/file/213346/download > > Signed-off-by: Vignesh Raghavendra > --- > v7: No change > > drivers/mtd/chips/cfi_cmdset_0002.c | 90 +++++++++++++++++++++++++++++ > include/linux/mtd/cfi.h | 5 ++ > 2 files changed, 95 insertions(+) > > diff --git a/drivers/mtd/chips/cfi_cmdset_0002.c b/drivers/mtd/chips/cfi_cmdset_0002.c > index c8fa5906bdf9..0f571f162e3b 100644 > --- a/drivers/mtd/chips/cfi_cmdset_0002.c > +++ b/drivers/mtd/chips/cfi_cmdset_0002.c > @@ -49,6 +49,16 @@ > #define SST49LF008A 0x005a > #define AT49BV6416 0x00d6 > > +/* > + * Status Register bit description. Used by flash devices that don't > + * support DQ polling (e.g. HyperFlash) > + */ > +#define CFI_SR_DRB BIT(7) > +#define CFI_SR_ESB BIT(5) > +#define CFI_SR_PSB BIT(4) > +#define CFI_SR_WBASB BIT(3) > +#define CFI_SR_SLSB BIT(1) > + > static int cfi_amdstd_read (struct mtd_info *, loff_t, size_t, size_t *, u_char *); > static int cfi_amdstd_write_words(struct mtd_info *, loff_t, size_t, size_t *, const u_char *); > static int cfi_amdstd_write_buffers(struct mtd_info *, loff_t, size_t, size_t *, const u_char *); > @@ -97,6 +107,48 @@ static struct mtd_chip_driver cfi_amdstd_chipdrv = { > .module = THIS_MODULE > }; > > +/* > + * Use status register to poll for Erase/write completion when DQ is not > + * supported. This is indicated by Bit[1:0] of SoftwareFeatures field in > + * CFI Primary Vendor-Specific Extended Query table 1.5 > + */ > +static int cfi_use_status_reg(struct cfi_private *cfi) > +{ > + struct cfi_pri_amdstd *extp = cfi->cmdset_priv; > + > + return extp->MinorVersion >= '5' && > + (extp->SoftwareFeatures & 0x3) == 0x1; Seems to be better to use defined values instead of 0x3 and 0x1 hard coded values. > +} > + > +static void cfi_check_err_status(struct map_info *map, unsigned long adr) > +{ > + struct cfi_private *cfi = map->fldrv_priv; > + map_word status; > + > + if (!cfi_use_status_reg(cfi)) > + return; > + > + cfi_send_gen_cmd(0x70, cfi->addr_unlock1, 0, map, cfi, Is it not necessary to set chip->start as the base parameter for cfi_send_gen_cmd()? > + cfi->device_type, NULL); > + status = map_read(map, adr); > + > + if (map_word_bitsset(map, status, CMD(0x3a))) { > + unsigned long chipstatus = MERGESTATUS(status); > + > + if (chipstatus & CFI_SR_ESB) > + pr_err("%s erase operation failed, status %lx\n", > + map->name, chipstatus); > + if (chipstatus & CFI_SR_PSB) > + pr_err("%s program operation failed, status %lx\n", > + map->name, chipstatus); > + if (chipstatus & CFI_SR_WBASB) > + pr_err("%s buffer program command aborted, status %lx\n", > + map->name, chipstatus); > + if (chipstatus & CFI_SR_SLSB) > + pr_err("%s sector write protected, status %lx\n", > + map->name, chipstatus); > + } > +} > > /* #define DEBUG_CFI_FEATURES */ > > @@ -744,8 +796,22 @@ static struct mtd_info *cfi_amdstd_setup(struct mtd_info *mtd) > */ > static int __xipram chip_ready(struct map_info *map, unsigned long addr) > { > + struct cfi_private *cfi = map->fldrv_priv; > map_word d, t; > > + if (cfi_use_status_reg(cfi)) { > + map_word ready = CMD(CFI_SR_DRB); > + /* > + * For chips that support status register, check device > + * ready bit > + */ > + cfi_send_gen_cmd(0x70, cfi->addr_unlock1, 0, map, cfi, Same comment as cfi_check_err_status() about the base address. > + cfi->device_type, NULL); > + d = map_read(map, addr); > + > + return map_word_andequal(map, d, ready, ready); > + } > + > d = map_read(map, addr); > t = map_read(map, addr); > > @@ -769,8 +835,27 @@ static int __xipram chip_ready(struct map_info *map, unsigned long addr) > */ > static int __xipram chip_good(struct map_info *map, unsigned long addr, map_word expected) > { > + struct cfi_private *cfi = map->fldrv_priv; > map_word oldd, curd; > > + if (cfi_use_status_reg(cfi)) { > + map_word ready = CMD(CFI_SR_DRB); > + map_word err = CMD(CFI_SR_PSB | CFI_SR_ESB); Is it not necessary to check CFI_SR_WBASB and CFI_SR_SLSB that are checked by cfi_check_err_status()? > + /* > + * For chips that support status register, check device > + * ready bit and Erase/Program status bit to know if > + * operation succeeded. > + */ > + cfi_send_gen_cmd(0x70, cfi->addr_unlock1, 0, map, cfi, Same as cfi_check_err_status() and chip_ready() about the base address. > + cfi->device_type, NULL); > + curd = map_read(map, addr); > + > + if (map_word_andequal(map, curd, ready, ready)) > + return !map_word_bitsset(map, curd, err); > + > + return 0; > + } > + > oldd = map_read(map, addr); > curd = map_read(map, addr); > > @@ -1644,6 +1729,7 @@ static int __xipram do_write_oneword(struct map_info *map, struct flchip *chip, > /* Did we succeed? */ > if (!chip_good(map, adr, datum)) { > /* reset on all failures. */ > + cfi_check_err_status(map, adr); > map_write(map, CMD(0xF0), chip->start); > /* FIXME - should have reset delay before continuing */ > > @@ -1901,6 +1987,7 @@ static int __xipram do_write_buffer(struct map_info *map, struct flchip *chip, > * See e.g. > * http://www.spansion.com/Support/Application%20Notes/MirrorBit_Write_Buffer_Prog_Page_Buffer_Read_AN.pdf > */ > + cfi_check_err_status(map, adr); > cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, > cfi->device_type, NULL); > cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, > @@ -2107,6 +2194,7 @@ static int do_panic_write_oneword(struct map_info *map, struct flchip *chip, > > if (!chip_good(map, adr, datum)) { > /* reset on all failures. */ > + cfi_check_err_status(map, adr); > map_write(map, CMD(0xF0), chip->start); > /* FIXME - should have reset delay before continuing */ > > @@ -2316,6 +2404,7 @@ static int __xipram do_erase_chip(struct map_info *map, struct flchip *chip) > /* Did we succeed? */ > if (ret) { > /* reset on all failures. */ > + cfi_check_err_status(map, adr); > map_write(map, CMD(0xF0), chip->start); > /* FIXME - should have reset delay before continuing */ > > @@ -2412,6 +2501,7 @@ static int __xipram do_erase_oneblock(struct map_info *map, struct flchip *chip, > /* Did we succeed? */ > if (ret) { > /* reset on all failures. */ > + cfi_check_err_status(map, adr); > map_write(map, CMD(0xF0), chip->start); > /* FIXME - should have reset delay before continuing */ > > diff --git a/include/linux/mtd/cfi.h b/include/linux/mtd/cfi.h > index 208c87cf2e3e..b50416169049 100644 > --- a/include/linux/mtd/cfi.h > +++ b/include/linux/mtd/cfi.h > @@ -219,6 +219,11 @@ struct cfi_pri_amdstd { > uint8_t VppMin; > uint8_t VppMax; > uint8_t TopBottom; > + /* Below field are added from version 1.5 */ > + uint8_t ProgramSuspend; > + uint8_t UnlockBypass; > + uint8_t SecureSiliconSector; > + uint8_t SoftwareFeatures; > } __packed; > > /* Vendor-Specific PRI for Atmel chips (command set 0x0002) */