Received: by 2002:a05:6a10:9848:0:0:0:0 with SMTP id x8csp4741503pxf; Tue, 30 Mar 2021 16:17:50 -0700 (PDT) X-Google-Smtp-Source: ABdhPJzciCXz1R3caZAZEzb8/7nxYfiw4riNXNW9y4EYWS3Ap+uvPdXxh4gQ9CgsrBGAEV13c2MT X-Received: by 2002:a17:906:86c1:: with SMTP id j1mr533920ejy.373.1617146270286; Tue, 30 Mar 2021 16:17:50 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1617146270; cv=none; d=google.com; s=arc-20160816; b=VGiudlgQgphzAT5354t7hQXNfnx+Dxj+27Fh1TOZyMpa8Mw1JDteXgglEjBWLr9bm4 UI3Cb+VViz3R8N4CRWQibcFk8XRd2tXAcJzeduX1jvX5N+hYYtKKpOu3yRnYXt8BcBBz 03NKLoyeHh6xbHzk6VECrMa4V/ilGgMH0iX04fkEabVNgNMNU6LT9aDCbYI+1BKWZKX6 6YokTYdcQ1utUYj2uIRV7ShJclfc8MVAKdiPyExekjBiDLxAB0zmKo4NgBTL4XIgw1Xi 9XRery96u1kAodrbnHI/gZGACXpQvNxrO5OrNZD1TpVTt+b9FkM0zq7Evk/x23nbn8ag r/Qg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:cc:to:from:subject:mime-version:message-id:date :dkim-signature; bh=0NQkAr5xUuyBWeb2zhIfxOf+Q+/q8yOlT371yO9KsBE=; b=nBGKckl+PzAz/B3F0yCKuFZ+GPk2gxuAmaAD8JKZB05DcAffcGDyBy8qHM2364zdqD o8i9KVZgIuXJaasePkCJqPuS4kHS+VMGSXloUCan9QGj1BRbG1CHynWz/qiD0BzKPO2J AIWg5lQrZdk1wO3NRcyUgN3jN95+I45ZjYGl8w2xf6tcKQ2WJu/aQFFIvVvTaACMf4GY lvreF4NraKm/1fxltSIsVNAozaXA61BzzAgQKLaLsAh4tBYh5UFjEpN0JZLx3t6WEcNs KEV3+sZ+BafmFenbAULpunCYJihSgyJWdbwV54aLutcr3Uv5zfv/yZn4Nh27eX2w4njA NpQA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=D49BgR0E; 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=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id e22si346459edu.575.2021.03.30.16.17.28; Tue, 30 Mar 2021 16:17:50 -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; dkim=pass header.i=@google.com header.s=20161025 header.b=D49BgR0E; 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=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S233279AbhC3XQe (ORCPT + 99 others); Tue, 30 Mar 2021 19:16:34 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:46570 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S233139AbhC3XQI (ORCPT ); Tue, 30 Mar 2021 19:16:08 -0400 Received: from mail-pj1-x1049.google.com (mail-pj1-x1049.google.com [IPv6:2607:f8b0:4864:20::1049]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 95254C061574 for ; Tue, 30 Mar 2021 16:16:07 -0700 (PDT) Received: by mail-pj1-x1049.google.com with SMTP id jp20so41879pjb.2 for ; Tue, 30 Mar 2021 16:16:07 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20161025; h=date:message-id:mime-version:subject:from:to:cc; bh=0NQkAr5xUuyBWeb2zhIfxOf+Q+/q8yOlT371yO9KsBE=; b=D49BgR0Efo2giRGhHiKQHIQoOQDI6FBtWWTXLhas/Iw21fFFAIz2k5mo8V5I7/ARN+ KM7nbiL5fnjGKs8zXXSWxdAr6oO+btPay2GwZ7u2qWZjSQM5dq9ryqXvU8W1WMIdbAWX /d769DmeW8FQjKjr/khi0Y8fXuVDFYoyS0CmMUvTiGfc4RBTGXel/t23SbQ/1EdvOaY4 vXYAWEH+a2RJigf5v1dcsedTBlUXMHwNQoVGZu0ItTSUmGaxYl4xR9d1eCFDWLVrk5xK tnAUKvMsVsuC0dt5WxAnGqzeD6/lPuBBFxx/frlFuD7BOvNuJyMrXNqEYWothfuQFyM5 IxvQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:message-id:mime-version:subject:from:to:cc; bh=0NQkAr5xUuyBWeb2zhIfxOf+Q+/q8yOlT371yO9KsBE=; b=ad2+dNN1Oe9yrdJVy9ICoC0JYHEzVqfTM7CfVMDa9WA5UnwCClHPDcL/sLN9nyKhvn NikH9k1/8cuRjJfFhpuTYhJ1d6DRInCnbnH1UtGp/5ROkZT5ycuYjmeReqRmAk+q95qz 8F8IPyW+mnBDuiCGlw27wkbWVWUmSn8gACGRQ/buIF5UGw41Si7Ye/esnbucX2kpUATJ Uy0cAhmiI4IdM2luNMfuyB5FtCiMjAdW5gTsuaYxbhZdbJZz9qMWa7ScrP5MnGfyX7S5 PHAkC1OzhsdBBoEOOelgP25TipuFk3dbaNWReBy6/fZz9w9ubr5T012k6ylyFlPh0sY0 d+yQ== X-Gm-Message-State: AOAM533VwlYhtsENr6ypV+O8cctCG/J2OGHpJ044XoEMG/gXOJg3M45W NWpV5M8NZ6FtETzA6dwBaEHeY9eoSlzyZg== X-Received: from egcloud.c.googlers.com ([fda3:e722:ac3:10:24:72f4:c0a8:35b5]) (user=egranata job=sendgmr) by 2002:a17:902:8303:b029:e6:4ef3:4f17 with SMTP id bd3-20020a1709028303b02900e64ef34f17mr400127plb.22.1617146167082; Tue, 30 Mar 2021 16:16:07 -0700 (PDT) Date: Tue, 30 Mar 2021 23:16:02 +0000 Message-Id: <20210330231602.1223216-1-egranata@google.com> Mime-Version: 1.0 X-Mailer: git-send-email 2.31.0.291.g576ba9dcdaf-goog Subject: [PATCH] virtio_blk: Add support for lifetime feature From: Enrico Granata To: mst@redhat.com, jasowang@redhat.com, pbonzini@redhat.com, stefanha@redhat.com, axboe@kernel.dk, virtualization@lists.linux-foundation.org, linux-block@vger.kernel.org, linux-kernel@vger.kernel.org Cc: egranata@google.com Content-Type: text/plain; charset="UTF-8" Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org The VirtIO TC has adopted a new feature in virtio-blk enabling discovery of lifetime information. This commit adds support for the VIRTIO_BLK_T_LIFETIME command to the virtio_blk driver, and adds two new attributes to the sysfs entry for virtio_blk: * pre_eol_info * life_time which are defined in the same manner as the files of the same name for the eMMC driver, in line with the VirtIO specification. Signed-off-by: Enrico Granata --- drivers/block/virtio_blk.c | 76 ++++++++++++++++++++++++++++++++- include/uapi/linux/virtio_blk.h | 11 +++++ 2 files changed, 86 insertions(+), 1 deletion(-) diff --git a/drivers/block/virtio_blk.c b/drivers/block/virtio_blk.c index b9fa3ef5b57c..1fc0ec000b4f 100644 --- a/drivers/block/virtio_blk.c +++ b/drivers/block/virtio_blk.c @@ -246,7 +246,7 @@ static blk_status_t virtio_queue_rq(struct blk_mq_hw_ctx *hctx, unmap = !(req->cmd_flags & REQ_NOUNMAP); break; case REQ_OP_DRV_IN: - type = VIRTIO_BLK_T_GET_ID; + type = vbr->out_hdr.type; break; default: WARN_ON_ONCE(1); @@ -310,11 +310,14 @@ static int virtblk_get_id(struct gendisk *disk, char *id_str) struct virtio_blk *vblk = disk->private_data; struct request_queue *q = vblk->disk->queue; struct request *req; + struct virtblk_req *vbreq; int err; req = blk_get_request(q, REQ_OP_DRV_IN, 0); if (IS_ERR(req)) return PTR_ERR(req); + vbreq = blk_mq_rq_to_pdu(req); + vbreq->out_hdr.type = VIRTIO_BLK_T_GET_ID; err = blk_rq_map_kern(q, req, id_str, VIRTIO_BLK_ID_BYTES, GFP_KERNEL); if (err) @@ -327,6 +330,34 @@ static int virtblk_get_id(struct gendisk *disk, char *id_str) return err; } +static int virtblk_get_lifetime(struct gendisk *disk, struct virtio_blk_lifetime *lifetime) +{ + struct virtio_blk *vblk = disk->private_data; + struct request_queue *q = vblk->disk->queue; + struct request *req; + struct virtblk_req *vbreq; + int err; + + if (!virtio_has_feature(vblk->vdev, VIRTIO_BLK_F_LIFETIME)) + return -EOPNOTSUPP; + + req = blk_get_request(q, REQ_OP_DRV_IN, 0); + if (IS_ERR(req)) + return PTR_ERR(req); + vbreq = blk_mq_rq_to_pdu(req); + vbreq->out_hdr.type = VIRTIO_BLK_T_GET_LIFETIME; + + err = blk_rq_map_kern(q, req, lifetime, sizeof(*lifetime), GFP_KERNEL); + if (err) + goto out; + + blk_execute_rq(vblk->disk, req, false); + err = blk_status_to_errno(virtblk_result(blk_mq_rq_to_pdu(req))); +out: + blk_put_request(req); + return err; +} + static void virtblk_get(struct virtio_blk *vblk) { refcount_inc(&vblk->refs); @@ -435,6 +466,46 @@ static ssize_t serial_show(struct device *dev, static DEVICE_ATTR_RO(serial); +static ssize_t pre_eol_info_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct gendisk *disk = dev_to_disk(dev); + struct virtio_blk_lifetime lft; + int err; + + /* sysfs gives us a PAGE_SIZE buffer */ + BUILD_BUG_ON(sizeof(lft) >= PAGE_SIZE); + + err = virtblk_get_lifetime(disk, &lft); + if (err) + return 0; + + return sprintf(buf, "0x%02x\n", le16_to_cpu(lft.pre_eol_info)); +} + +static DEVICE_ATTR_RO(pre_eol_info); + +static ssize_t life_time_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct gendisk *disk = dev_to_disk(dev); + struct virtio_blk_lifetime lft; + int err; + + /* sysfs gives us a PAGE_SIZE buffer */ + BUILD_BUG_ON(sizeof(lft) >= PAGE_SIZE); + + err = virtblk_get_lifetime(disk, &lft); + if (err) + return 0; + + return sprintf(buf, "0x%02x 0x%02x\n", + le16_to_cpu(lft.device_life_time_est_typ_a), + le16_to_cpu(lft.device_life_time_est_typ_b)); +} + +static DEVICE_ATTR_RO(life_time); + /* The queue's logical block size must be set before calling this */ static void virtblk_update_capacity(struct virtio_blk *vblk, bool resize) { @@ -638,6 +709,8 @@ static DEVICE_ATTR_RW(cache_type); static struct attribute *virtblk_attrs[] = { &dev_attr_serial.attr, + &dev_attr_pre_eol_info.attr, + &dev_attr_life_time.attr, &dev_attr_cache_type.attr, NULL, }; @@ -984,6 +1057,7 @@ static unsigned int features[] = { VIRTIO_BLK_F_RO, VIRTIO_BLK_F_BLK_SIZE, VIRTIO_BLK_F_FLUSH, VIRTIO_BLK_F_TOPOLOGY, VIRTIO_BLK_F_CONFIG_WCE, VIRTIO_BLK_F_MQ, VIRTIO_BLK_F_DISCARD, VIRTIO_BLK_F_WRITE_ZEROES, + VIRTIO_BLK_F_LIFETIME, }; static struct virtio_driver virtio_blk = { diff --git a/include/uapi/linux/virtio_blk.h b/include/uapi/linux/virtio_blk.h index d888f013d9ff..bbd3978b9d08 100644 --- a/include/uapi/linux/virtio_blk.h +++ b/include/uapi/linux/virtio_blk.h @@ -40,6 +40,7 @@ #define VIRTIO_BLK_F_MQ 12 /* support more than one vq */ #define VIRTIO_BLK_F_DISCARD 13 /* DISCARD is supported */ #define VIRTIO_BLK_F_WRITE_ZEROES 14 /* WRITE ZEROES is supported */ +#define VIRTIO_BLK_F_LIFETIME 15 /* LIFETIME is supported */ /* Legacy feature bits */ #ifndef VIRTIO_BLK_NO_LEGACY @@ -149,6 +150,9 @@ struct virtio_blk_config { /* Get device ID command */ #define VIRTIO_BLK_T_GET_ID 8 +/* Get device lifetime command */ +#define VIRTIO_BLK_T_GET_LIFETIME 10 + /* Discard command */ #define VIRTIO_BLK_T_DISCARD 11 @@ -196,6 +200,13 @@ struct virtio_scsi_inhdr { }; #endif /* !VIRTIO_BLK_NO_LEGACY */ +/* Lifetime information for virtio_blk device */ +struct virtio_blk_lifetime { + __le16 pre_eol_info; + __le16 device_life_time_est_typ_a; + __le16 device_life_time_est_typ_b; +}; + /* And this is the final byte of the write scatter-gather list. */ #define VIRTIO_BLK_S_OK 0 #define VIRTIO_BLK_S_IOERR 1 -- 2.31.0.291.g576ba9dcdaf-goog