2014-09-05 21:46:16 +08:00
|
|
|
/*
|
|
|
|
* QEMU System Emulator block accounting
|
|
|
|
*
|
|
|
|
* Copyright (c) 2011 Christoph Hellwig
|
2015-10-28 23:33:18 +08:00
|
|
|
* Copyright (c) 2015 Igalia, S.L.
|
2014-09-05 21:46:16 +08:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
#ifndef BLOCK_ACCOUNTING_H
|
|
|
|
#define BLOCK_ACCOUNTING_H
|
|
|
|
|
2015-10-28 23:33:05 +08:00
|
|
|
#include "qemu/timed-average.h"
|
2017-06-05 20:39:08 +08:00
|
|
|
#include "qemu/thread.h"
|
2018-03-10 00:52:11 +08:00
|
|
|
#include "qapi/qapi-builtin-types.h"
|
2015-10-28 23:33:05 +08:00
|
|
|
|
|
|
|
typedef struct BlockAcctTimedStats BlockAcctTimedStats;
|
2017-06-05 20:39:08 +08:00
|
|
|
typedef struct BlockAcctStats BlockAcctStats;
|
2014-09-05 21:46:16 +08:00
|
|
|
|
|
|
|
enum BlockAcctType {
|
2014-09-05 21:46:17 +08:00
|
|
|
BLOCK_ACCT_READ,
|
|
|
|
BLOCK_ACCT_WRITE,
|
|
|
|
BLOCK_ACCT_FLUSH,
|
|
|
|
BLOCK_MAX_IOTYPE,
|
2014-09-05 21:46:16 +08:00
|
|
|
};
|
|
|
|
|
2015-10-28 23:33:05 +08:00
|
|
|
struct BlockAcctTimedStats {
|
2017-06-05 20:39:08 +08:00
|
|
|
BlockAcctStats *stats;
|
2015-10-28 23:33:05 +08:00
|
|
|
TimedAverage latency[BLOCK_MAX_IOTYPE];
|
|
|
|
unsigned interval_length; /* in seconds */
|
|
|
|
QSLIST_ENTRY(BlockAcctTimedStats) entries;
|
|
|
|
};
|
|
|
|
|
2018-03-10 00:52:11 +08:00
|
|
|
typedef struct BlockLatencyHistogram {
|
|
|
|
/* The following histogram is represented like this:
|
|
|
|
*
|
|
|
|
* 5| *
|
|
|
|
* 4| *
|
|
|
|
* 3| * *
|
|
|
|
* 2| * * *
|
|
|
|
* 1| * * * *
|
|
|
|
* +------------------
|
|
|
|
* 10 50 100
|
|
|
|
*
|
|
|
|
* BlockLatencyHistogram histogram = {
|
|
|
|
* .nbins = 4,
|
|
|
|
* .boundaries = {10, 50, 100},
|
|
|
|
* .bins = {3, 1, 5, 2},
|
|
|
|
* };
|
|
|
|
*
|
|
|
|
* @boundaries array define histogram intervals as follows:
|
|
|
|
* [0, boundaries[0]), [boundaries[0], boundaries[1]), ...
|
|
|
|
* [boundaries[nbins-2], +inf)
|
|
|
|
*
|
|
|
|
* So, for example above, histogram intervals are:
|
|
|
|
* [0, 10), [10, 50), [50, 100), [100, +inf)
|
|
|
|
*/
|
|
|
|
int nbins;
|
|
|
|
uint64_t *boundaries; /* @nbins-1 numbers here
|
|
|
|
(all boundaries, except 0 and +inf) */
|
|
|
|
uint64_t *bins;
|
|
|
|
} BlockLatencyHistogram;
|
|
|
|
|
2017-06-05 20:39:08 +08:00
|
|
|
struct BlockAcctStats {
|
|
|
|
QemuMutex lock;
|
2014-09-05 21:46:17 +08:00
|
|
|
uint64_t nr_bytes[BLOCK_MAX_IOTYPE];
|
|
|
|
uint64_t nr_ops[BLOCK_MAX_IOTYPE];
|
2015-10-28 23:33:03 +08:00
|
|
|
uint64_t invalid_ops[BLOCK_MAX_IOTYPE];
|
|
|
|
uint64_t failed_ops[BLOCK_MAX_IOTYPE];
|
2014-09-05 21:46:17 +08:00
|
|
|
uint64_t total_time_ns[BLOCK_MAX_IOTYPE];
|
2015-02-02 21:52:18 +08:00
|
|
|
uint64_t merged[BLOCK_MAX_IOTYPE];
|
2015-10-28 23:33:02 +08:00
|
|
|
int64_t last_access_time_ns;
|
2015-10-28 23:33:05 +08:00
|
|
|
QSLIST_HEAD(, BlockAcctTimedStats) intervals;
|
2015-10-28 23:33:04 +08:00
|
|
|
bool account_invalid;
|
|
|
|
bool account_failed;
|
2018-03-10 00:52:11 +08:00
|
|
|
BlockLatencyHistogram latency_histogram[BLOCK_MAX_IOTYPE];
|
2017-06-05 20:39:08 +08:00
|
|
|
};
|
2014-09-05 21:46:16 +08:00
|
|
|
|
|
|
|
typedef struct BlockAcctCookie {
|
|
|
|
int64_t bytes;
|
|
|
|
int64_t start_time_ns;
|
|
|
|
enum BlockAcctType type;
|
|
|
|
} BlockAcctCookie;
|
|
|
|
|
2017-06-05 20:39:07 +08:00
|
|
|
void block_acct_init(BlockAcctStats *stats);
|
|
|
|
void block_acct_setup(BlockAcctStats *stats, bool account_invalid,
|
2015-10-28 23:33:04 +08:00
|
|
|
bool account_failed);
|
2015-10-28 23:33:05 +08:00
|
|
|
void block_acct_cleanup(BlockAcctStats *stats);
|
|
|
|
void block_acct_add_interval(BlockAcctStats *stats, unsigned interval_length);
|
|
|
|
BlockAcctTimedStats *block_acct_interval_next(BlockAcctStats *stats,
|
|
|
|
BlockAcctTimedStats *s);
|
2014-09-05 21:46:18 +08:00
|
|
|
void block_acct_start(BlockAcctStats *stats, BlockAcctCookie *cookie,
|
|
|
|
int64_t bytes, enum BlockAcctType type);
|
|
|
|
void block_acct_done(BlockAcctStats *stats, BlockAcctCookie *cookie);
|
2015-10-28 23:33:03 +08:00
|
|
|
void block_acct_failed(BlockAcctStats *stats, BlockAcctCookie *cookie);
|
|
|
|
void block_acct_invalid(BlockAcctStats *stats, enum BlockAcctType type);
|
2015-02-02 21:52:18 +08:00
|
|
|
void block_acct_merge_done(BlockAcctStats *stats, enum BlockAcctType type,
|
|
|
|
int num_requests);
|
2015-10-28 23:33:02 +08:00
|
|
|
int64_t block_acct_idle_time_ns(BlockAcctStats *stats);
|
2015-10-28 23:33:06 +08:00
|
|
|
double block_acct_queue_depth(BlockAcctTimedStats *stats,
|
|
|
|
enum BlockAcctType type);
|
2018-03-10 00:52:11 +08:00
|
|
|
int block_latency_histogram_set(BlockAcctStats *stats, enum BlockAcctType type,
|
|
|
|
uint64List *boundaries);
|
|
|
|
void block_latency_histograms_clear(BlockAcctStats *stats);
|
2014-09-05 21:46:16 +08:00
|
|
|
|
|
|
|
#endif
|