Re: [PATCH 1/4] block: Measure flush round-trip times and report average value

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Mon, 29 Nov 2010, Darrick J. Wong wrote:

> This patch adds to the block layer the ability to intercept flush requests for
> the purpose of measuring the round-trip times of the write-cache flush command
> on whatever hardware sits underneath the block layer.  The eventual intent of
> this patch is for higher-level clients (filesystems) to be able to measure
> flush times to tune their use of them.
> 
> Signed-off-by: Darrick J. Wong <djwong@xxxxxxxxxx>
> ---
>  block/blk-core.c          |   13 +++++++++++++
>  block/genhd.c             |   39 +++++++++++++++++++++++++++++++++++++++
>  fs/bio.c                  |   11 +++++++++++
>  include/linux/blk_types.h |    2 ++
>  include/linux/blkdev.h    |    2 ++
>  include/linux/genhd.h     |   23 +++++++++++++++++++++++
>  6 files changed, 90 insertions(+), 0 deletions(-)
> 
> 
> diff --git a/block/blk-core.c b/block/blk-core.c
> index 4ce953f..233573a 100644
> --- a/block/blk-core.c
> +++ b/block/blk-core.c
> @@ -1540,6 +1540,9 @@ static inline void __generic_make_request(struct bio *bio)
>  
>  		trace_block_bio_queue(q, bio);
>  
> +		if (bio->bi_rw & REQ_FLUSH)
> +			bio->flush_start_time_ns = ktime_to_ns(ktime_get());
> +
>  		ret = q->make_request_fn(q, bio);
>  	} while (ret);
>  
> @@ -1954,6 +1957,9 @@ void blk_start_request(struct request *req)
>  		req->next_rq->resid_len = blk_rq_bytes(req->next_rq);
>  
>  	blk_add_timer(req);
> +
> +	if (req->cmd_flags & REQ_FLUSH)
> +		req->flush_start_time_ns = ktime_to_ns(ktime_get());
>  }
>  EXPORT_SYMBOL(blk_start_request);
>  
> @@ -2182,6 +2188,13 @@ EXPORT_SYMBOL_GPL(blk_unprep_request);
>   */
>  static void blk_finish_request(struct request *req, int error)
>  {
> +	if (!error && req->cmd_flags & REQ_FLUSH) {
> +		u64 delta;
> +
> +		delta = ktime_to_ns(ktime_get()) - req->flush_start_time_ns;
> +		update_flush_times(req->rq_disk, delta);
> +	}
> +
>  	if (blk_rq_tagged(req))
>  		blk_queue_end_tag(req->q, req);
>  
> diff --git a/block/genhd.c b/block/genhd.c
> index 5fa2b44..465bd00 100644
> --- a/block/genhd.c
> +++ b/block/genhd.c
> @@ -538,6 +538,9 @@ void add_disk(struct gendisk *disk)
>  	 */
>  	disk->major = MAJOR(devt);
>  	disk->first_minor = MINOR(devt);
> +	spin_lock_init(&disk->flush_time_lock);
> +	disk->avg_flush_time_ns = 0;
> +	disk->flush_samples = 0;
>  
>  	/* Register BDI before referencing it from bdev */ 
>  	bdi = &disk->queue->backing_dev_info;
> @@ -824,6 +827,37 @@ static ssize_t disk_range_show(struct device *dev,
>  	return sprintf(buf, "%d\n", disk->minors);
>  }
>  
> +static ssize_t disk_flush_samples_show(struct device *dev,
> +				   struct device_attribute *attr, char *buf)
> +{
> +	struct gendisk *disk = dev_to_disk(dev);
> +
> +	return sprintf(buf, "%llu\n", disk->flush_samples);
> +}
> +
> +static ssize_t disk_avg_flush_time_ns_show(struct device *dev,
> +				   struct device_attribute *attr, char *buf)
> +{
> +	struct gendisk *disk = dev_to_disk(dev);
> +
> +	return sprintf(buf, "%llu\n", disk->avg_flush_time_ns);
> +}
> +
> +static ssize_t disk_avg_flush_time_ns_store(struct device *dev,
> +					    struct device_attribute *attr,
> +					    const char *buf, size_t count)

Hi,

This is not exactly helpful name for the function since it does not
store anything, but rather clear the counters. Would not be
disk_avg_flush_time_ns_clear a better name ?

Thanks!

-Lukas

> +{
> +	struct gendisk *disk = dev_to_disk(dev);
> +
> +	spin_lock(&disk->flush_time_lock);
> +	disk->avg_flush_time_ns = 0;
> +	disk->flush_samples = 0;
> +	spin_unlock(&disk->flush_time_lock);
> +
> +	return count;
> +}
> +
> +
>  static ssize_t disk_ext_range_show(struct device *dev,
>  				   struct device_attribute *attr, char *buf)
>  {
> @@ -876,6 +910,9 @@ static ssize_t disk_discard_alignment_show(struct device *dev,
>  }
>  
>  static DEVICE_ATTR(range, S_IRUGO, disk_range_show, NULL);
> +static DEVICE_ATTR(avg_flush_time_ns, S_IRUGO | S_IWUSR,
> +		   disk_avg_flush_time_ns_show, disk_avg_flush_time_ns_store);
> +static DEVICE_ATTR(flush_samples, S_IRUGO, disk_flush_samples_show, NULL);
>  static DEVICE_ATTR(ext_range, S_IRUGO, disk_ext_range_show, NULL);
>  static DEVICE_ATTR(removable, S_IRUGO, disk_removable_show, NULL);
>  static DEVICE_ATTR(ro, S_IRUGO, disk_ro_show, NULL);
> @@ -898,6 +935,8 @@ static struct device_attribute dev_attr_fail_timeout =
>  
>  static struct attribute *disk_attrs[] = {
>  	&dev_attr_range.attr,
> +	&dev_attr_avg_flush_time_ns.attr,
> +	&dev_attr_flush_samples.attr,
>  	&dev_attr_ext_range.attr,
>  	&dev_attr_removable.attr,
>  	&dev_attr_ro.attr,
> diff --git a/fs/bio.c b/fs/bio.c
> index 4bd454f..aab5f27 100644
> --- a/fs/bio.c
> +++ b/fs/bio.c
> @@ -1442,11 +1442,22 @@ EXPORT_SYMBOL(bio_flush_dcache_pages);
>   **/
>  void bio_endio(struct bio *bio, int error)
>  {
> +	struct request_queue *q = NULL;
> +
>  	if (error)
>  		clear_bit(BIO_UPTODATE, &bio->bi_flags);
>  	else if (!test_bit(BIO_UPTODATE, &bio->bi_flags))
>  		error = -EIO;
>  
> +	if (bio->bi_bdev)
> +		q = bdev_get_queue(bio->bi_bdev);
> +	if (!(q && q->prep_rq_fn) && !error && bio->bi_rw & REQ_FLUSH) {
> +		u64 delta;
> +
> +		delta = ktime_to_ns(ktime_get()) - bio->flush_start_time_ns;
> +		update_flush_times(bio->bi_bdev->bd_disk, delta);
> +	}
> +
>  	if (bio->bi_end_io)
>  		bio->bi_end_io(bio, error);
>  }
> diff --git a/include/linux/blk_types.h b/include/linux/blk_types.h
> index 46ad519..e8c29b9 100644
> --- a/include/linux/blk_types.h
> +++ b/include/linux/blk_types.h
> @@ -74,6 +74,8 @@ struct bio {
>  
>  	bio_destructor_t	*bi_destructor;	/* destructor */
>  
> +	u64			flush_start_time_ns;
> +
>  	/*
>  	 * We can inline a number of vecs at the end of the bio, to avoid
>  	 * double allocations for a small number of bio_vecs. This member
> diff --git a/include/linux/blkdev.h b/include/linux/blkdev.h
> index aae86fd..357d669 100644
> --- a/include/linux/blkdev.h
> +++ b/include/linux/blkdev.h
> @@ -163,6 +163,8 @@ struct request {
>  
>  	/* for bidi */
>  	struct request *next_rq;
> +
> +	u64 flush_start_time_ns;
>  };
>  
>  static inline unsigned short req_get_ioprio(struct request *req)
> diff --git a/include/linux/genhd.h b/include/linux/genhd.h
> index 7a7b9c1..7097396 100644
> --- a/include/linux/genhd.h
> +++ b/include/linux/genhd.h
> @@ -178,8 +178,31 @@ struct gendisk {
>  	struct blk_integrity *integrity;
>  #endif
>  	int node_id;
> +
> +	spinlock_t flush_time_lock;
> +	u64 avg_flush_time_ns;
> +	u64 flush_samples;
>  };
>  
> +static inline void update_flush_times(struct gendisk *disk, u64 delta)
> +{
> +	u64 data;
> +
> +	spin_lock(&disk->flush_time_lock);
> +	if (disk->flush_samples < 256)
> +		disk->flush_samples++;
> +	if (!disk->avg_flush_time_ns)
> +		disk->avg_flush_time_ns = delta;
> +	else {
> +		data = disk->avg_flush_time_ns;
> +		data *= 255;
> +		data += delta;
> +		data /= 256;
> +		disk->avg_flush_time_ns = data;
> +	}
> +	spin_unlock(&disk->flush_time_lock);
> +}
> +
>  static inline struct gendisk *part_to_disk(struct hd_struct *part)
>  {
>  	if (likely(part)) {
> 
> --
> To unsubscribe from this list: send the line "unsubscribe linux-ext4" in
> the body of a message to majordomo@xxxxxxxxxxxxxxx
> More majordomo info at  http://vger.kernel.org/majordomo-info.html
> 
--
To unsubscribe from this list: send the line "unsubscribe linux-ext4" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [Reiser Filesystem Development]     [Ceph FS]     [Kernel Newbies]     [Security]     [Netfilter]     [Bugtraq]     [Linux FS]     [Yosemite National Park]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Samba]     [Device Mapper]     [Linux Media]

  Powered by Linux