|  | /* SPDX-License-Identifier: GPL-2.0 */ | 
|  | #ifndef DM_STATS_H | 
|  | #define DM_STATS_H | 
|  |  | 
|  | #include <linux/types.h> | 
|  | #include <linux/mutex.h> | 
|  | #include <linux/list.h> | 
|  |  | 
|  | int dm_statistics_init(void); | 
|  | void dm_statistics_exit(void); | 
|  |  | 
|  | struct dm_stats { | 
|  | struct mutex mutex; | 
|  | struct list_head list;	/* list of struct dm_stat */ | 
|  | struct dm_stats_last_position __percpu *last; | 
|  | bool precise_timestamps; | 
|  | }; | 
|  |  | 
|  | struct dm_stats_aux { | 
|  | bool merged; | 
|  | unsigned long long duration_ns; | 
|  | }; | 
|  |  | 
|  | int dm_stats_init(struct dm_stats *st); | 
|  | void dm_stats_cleanup(struct dm_stats *st); | 
|  |  | 
|  | struct mapped_device; | 
|  |  | 
|  | int dm_stats_message(struct mapped_device *md, unsigned int argc, char **argv, | 
|  | char *result, unsigned int maxlen); | 
|  |  | 
|  | void dm_stats_account_io(struct dm_stats *stats, unsigned long bi_rw, | 
|  | sector_t bi_sector, unsigned int bi_sectors, bool end, | 
|  | unsigned long start_time, | 
|  | struct dm_stats_aux *aux); | 
|  |  | 
|  | static inline bool dm_stats_used(struct dm_stats *st) | 
|  | { | 
|  | return !list_empty(&st->list); | 
|  | } | 
|  |  | 
|  | static inline void dm_stats_record_start(struct dm_stats *stats, struct dm_stats_aux *aux) | 
|  | { | 
|  | if (unlikely(stats->precise_timestamps)) | 
|  | aux->duration_ns = ktime_to_ns(ktime_get()); | 
|  | } | 
|  |  | 
|  | #endif |