841 lines
24 KiB
C
841 lines
24 KiB
C
/*
|
|
* Copyright(c) 2012-2021 Intel Corporation
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <errno.h>
|
|
#include <assert.h>
|
|
#include <fcntl.h>
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <inttypes.h>
|
|
#include <fstab.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/types.h>
|
|
#include <linux/major.h>
|
|
#include <mntent.h>
|
|
#include <sys/mman.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/types.h>
|
|
#include <time.h>
|
|
#include "cas_lib.h"
|
|
#include "extended_err_msg.h"
|
|
#include "cas_lib_utils.h"
|
|
#include <pthread.h>
|
|
#include <stdbool.h>
|
|
#include <cas_ioctl_codes.h>
|
|
|
|
#include "csvparse.h"
|
|
#include "statistics_view.h"
|
|
#include "safeclib/safe_str_lib.h"
|
|
#include "ocf/ocf_cache.h"
|
|
|
|
#define IOCLASS_UNCLASSIFIED (0)
|
|
|
|
#define UNIT_REQUESTS "Requests"
|
|
#define UNIT_BLOCKS "4KiB Blocks"
|
|
|
|
static inline float fraction(uint64_t numerator, uint64_t denominator)
|
|
{
|
|
float result;
|
|
if (denominator) {
|
|
result = 10000.0 * numerator / denominator;
|
|
} else {
|
|
result = 0;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static inline long unsigned int cache_line_in_4k(uint64_t size,
|
|
ocf_cache_line_size_t cache_line_size)
|
|
{
|
|
long unsigned int result;
|
|
|
|
result = size * (cache_line_size / 4);
|
|
|
|
return result;
|
|
}
|
|
|
|
static inline unsigned long bytes_to_4k(uint64_t size)
|
|
{
|
|
return (size + 4095UL) >> 12;
|
|
}
|
|
|
|
static float calc_gb(uint64_t clines)
|
|
{
|
|
return (float) clines * 4 * KiB / GiB;
|
|
}
|
|
|
|
static void print_dirty_for_time(uint64_t t, FILE *outfile)
|
|
{
|
|
uint32_t d, h, m, s;
|
|
|
|
fprintf(outfile, "%lu,[s],", t);
|
|
|
|
if (!t) {
|
|
fprintf(outfile, "Cache clean");
|
|
return;
|
|
}
|
|
|
|
d = t / (24 * 3600);
|
|
h = (t % (24 * 3600)) / 3600;
|
|
m = (t % 3600) / 60;
|
|
s = (t % 60);
|
|
|
|
if (d) {
|
|
fprintf(outfile, "%u [d] ", d);
|
|
}
|
|
if (h) {
|
|
fprintf(outfile, "%u [h] ", h);
|
|
}
|
|
if (m) {
|
|
fprintf(outfile, "%u [m] ", m);
|
|
}
|
|
if (s) {
|
|
fprintf(outfile, "%u [s] ", s);
|
|
}
|
|
}
|
|
|
|
__attribute__((format(printf, 3, 4)))
|
|
static void print_kv_pair(FILE *outfile, const char *title, const char *fmt, ...)
|
|
{
|
|
va_list ap;
|
|
|
|
fprintf(outfile, TAG(KV_PAIR) "\"%s\",", title);
|
|
va_start(ap, fmt);
|
|
vfprintf(outfile, fmt, ap);
|
|
va_end(ap);
|
|
fprintf(outfile, "\n");
|
|
}
|
|
|
|
static void print_kv_pair_time(FILE *outfile, const char *title, uint64_t time)
|
|
{
|
|
fprintf(outfile, TAG(KV_PAIR) "\"%s\",", title);
|
|
print_dirty_for_time(time, outfile);
|
|
fprintf(outfile, "\n");
|
|
}
|
|
|
|
static void begin_record(FILE *outfile)
|
|
{
|
|
fprintf(outfile, TAG(RECORD) "\n");
|
|
}
|
|
|
|
static void print_table_header(FILE *outfile, uint32_t ncols, ...)
|
|
{
|
|
va_list ap;
|
|
const char *s;
|
|
|
|
fprintf(outfile, TAG(TABLE_HEADER));
|
|
va_start(ap, ncols);
|
|
while (ncols--) {
|
|
s = va_arg(ap, const char *);
|
|
fprintf(outfile, "\"%s\"%s", s, ncols ? "," : "\n");
|
|
}
|
|
va_end(ap);
|
|
}
|
|
|
|
static void print_val_perc_table_elem(FILE *outfile, const char *tag,
|
|
const char *title, const char *unit,
|
|
uint64_t percent, const char * fmt,
|
|
va_list ap)
|
|
{
|
|
float percent_val = (percent % 10 >= 5 ? percent+5 : percent) / 100.f;
|
|
fprintf(outfile, "%s\"%s\",", tag, title);
|
|
vfprintf(outfile, fmt, ap);
|
|
fprintf(outfile, ",%.1f", percent_val);
|
|
if (unit) {
|
|
fprintf(outfile, ",\"[%s]\"", unit);
|
|
}
|
|
fprintf(outfile, "\n");
|
|
}
|
|
|
|
__attribute__((format(printf, 5, 6)))
|
|
static inline void print_val_perc_table_row(FILE *outfile, const char *title,
|
|
const char *unit, float percent,
|
|
const char *fmt, ...)
|
|
{
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
print_val_perc_table_elem(outfile, TAG(TABLE_ROW), title, unit,
|
|
percent, fmt, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
__attribute__((format(printf, 5, 6)))
|
|
static inline void print_val_perc_table_section(FILE *outfile, const char *title,
|
|
const char *unit, uint64_t percent,
|
|
const char *fmt, ...)
|
|
{
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
print_val_perc_table_elem(outfile, TAG(TABLE_SECTION), title, unit,
|
|
percent, fmt, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
static inline const char *make_row_title(const char *s1, const char *s2)
|
|
{
|
|
static char buffer[64];
|
|
snprintf(buffer, sizeof(buffer), "%s %s", s1, s2);
|
|
return buffer;
|
|
}
|
|
|
|
static void print_core_conf(const struct kcas_core_info *info, FILE *outfile)
|
|
{
|
|
uint64_t core_size;
|
|
float core_size_gb;
|
|
char exp_obj[32];
|
|
|
|
core_size = info->info.core_size_bytes / KiB / 4;
|
|
core_size_gb = calc_gb(core_size);
|
|
|
|
snprintf(exp_obj, sizeof(exp_obj), "/dev/cas%d-%d",
|
|
info->cache_id, info->core_id);
|
|
|
|
print_kv_pair(outfile, "Core Id", "%i", info->core_id);
|
|
print_kv_pair(outfile, "Core Device", "%s",
|
|
info->core_path_name);
|
|
print_kv_pair(outfile, "Exported Object", "%s",
|
|
info->exp_obj_exists ? exp_obj : "-");
|
|
print_kv_pair(outfile, "Core Size", "%lu, [" UNIT_BLOCKS "], %.2f, [GiB]",
|
|
core_size, core_size_gb);
|
|
print_kv_pair_time(outfile, "Dirty for", info->info.dirty_for);
|
|
|
|
print_kv_pair(outfile, "Status", "%s",
|
|
get_core_state_name(info->state));
|
|
|
|
print_kv_pair(outfile, "Seq cutoff threshold", "%llu, [KiB]",
|
|
info->info.seq_cutoff_threshold / KiB);
|
|
|
|
print_kv_pair(outfile, "Seq cutoff policy", "%s",
|
|
seq_cutoff_policy_to_name(info->info.seq_cutoff_policy));
|
|
}
|
|
|
|
static void print_usage_header(FILE* outfile)
|
|
{
|
|
print_table_header(outfile, 4, "Usage statistics", "Count",
|
|
"%", "[Units]");
|
|
}
|
|
|
|
static void print_usage_stats(struct ocf_stats_usage *stats, FILE* outfile)
|
|
{
|
|
print_usage_header(outfile);
|
|
|
|
print_val_perc_table_row(outfile, "Occupancy", UNIT_BLOCKS,
|
|
stats->occupancy.fraction, "%lu", stats->occupancy.value);
|
|
print_val_perc_table_row(outfile, "Free", UNIT_BLOCKS,
|
|
stats->free.fraction, "%lu", stats->free.value);
|
|
print_val_perc_table_row(outfile, "Clean", UNIT_BLOCKS,
|
|
stats->clean.fraction, "%lu", stats->clean.value);
|
|
print_val_perc_table_row(outfile, "Dirty", UNIT_BLOCKS,
|
|
stats->dirty.fraction, "%lu", stats->dirty.value);
|
|
}
|
|
|
|
static void print_ioclass_usage_stats(struct ocf_stats_usage *stats, FILE* outfile)
|
|
{
|
|
print_usage_header(outfile);
|
|
|
|
print_val_perc_table_row(outfile, "Occupancy", UNIT_BLOCKS,
|
|
stats->occupancy.fraction, "%lu", stats->occupancy.value);
|
|
print_val_perc_table_row(outfile, "Clean", UNIT_BLOCKS,
|
|
stats->clean.fraction, "%lu", stats->clean.value);
|
|
print_val_perc_table_row(outfile, "Dirty", UNIT_BLOCKS,
|
|
stats->dirty.fraction, "%lu", stats->dirty.value);
|
|
}
|
|
|
|
static void print_req_stats(const struct ocf_stats_requests *stats,
|
|
FILE *outfile)
|
|
{
|
|
print_table_header(outfile, 4, "Request statistics", "Count",
|
|
"%", "[Units]");
|
|
|
|
print_val_perc_table_section(outfile, "Read hits",
|
|
UNIT_REQUESTS, stats->rd_hits.fraction, "%lu",
|
|
stats->rd_hits.value);
|
|
print_val_perc_table_row(outfile, "Read partial misses",
|
|
UNIT_REQUESTS, stats->rd_partial_misses.fraction, "%lu",
|
|
stats->rd_partial_misses.value);
|
|
print_val_perc_table_row(outfile, "Read full misses",
|
|
UNIT_REQUESTS, stats->rd_full_misses.fraction, "%lu",
|
|
stats->rd_full_misses.value);
|
|
print_val_perc_table_row(outfile, "Read total",
|
|
UNIT_REQUESTS, stats->rd_total.fraction, "%lu",
|
|
stats->rd_total.value);
|
|
|
|
print_val_perc_table_section(outfile, "Write hits",
|
|
UNIT_REQUESTS, stats->wr_hits.fraction, "%lu",
|
|
stats->wr_hits.value);
|
|
print_val_perc_table_row(outfile, "Write partial misses",
|
|
UNIT_REQUESTS, stats->wr_partial_misses.fraction, "%lu",
|
|
stats->wr_partial_misses.value);
|
|
print_val_perc_table_row(outfile, "Write full misses",
|
|
UNIT_REQUESTS, stats->wr_full_misses.fraction, "%lu",
|
|
stats->wr_full_misses.value);
|
|
print_val_perc_table_row(outfile, "Write total",
|
|
UNIT_REQUESTS, stats->wr_total.fraction, "%lu",
|
|
stats->wr_total.value);
|
|
|
|
print_val_perc_table_section(outfile, "Pass-Through reads",
|
|
UNIT_REQUESTS, stats->rd_pt.fraction, "%lu",
|
|
stats->rd_pt.value);
|
|
print_val_perc_table_row(outfile, "Pass-Through writes",
|
|
UNIT_REQUESTS, stats->wr_pt.fraction, "%lu",
|
|
stats->wr_pt.value);
|
|
print_val_perc_table_row(outfile, "Serviced requests",
|
|
UNIT_REQUESTS, stats->serviced.fraction, "%lu",
|
|
stats->serviced.value);
|
|
|
|
print_val_perc_table_section(outfile, "Total requests",
|
|
UNIT_REQUESTS, stats->total.fraction, "%lu",
|
|
stats->total.value);
|
|
}
|
|
|
|
#define get_stat_name(__dst, __len, __name, __postfix) \
|
|
memset(__dst, 0, __len); \
|
|
snprintf(__dst, __len, "%s%s", __name, __postfix);
|
|
|
|
|
|
static void print_blk_stats(const struct ocf_stats_blocks *stats,
|
|
bool cache_stats, FILE *outfile)
|
|
{
|
|
print_table_header(outfile, 4, "Block statistics", "Count",
|
|
"%", "[Units]");
|
|
|
|
char *postfix = (cache_stats ? "(s)" : "");
|
|
size_t max_stat_len = 128;
|
|
char stat_name[max_stat_len];
|
|
|
|
get_stat_name(stat_name, max_stat_len, "Reads from core", postfix);
|
|
print_val_perc_table_section(outfile, stat_name,
|
|
UNIT_BLOCKS, stats->core_volume_rd.fraction, "%lu",
|
|
stats->core_volume_rd.value);
|
|
|
|
get_stat_name(stat_name, max_stat_len, "Writes to core", postfix);
|
|
print_val_perc_table_row(outfile, stat_name,
|
|
UNIT_BLOCKS, stats->core_volume_wr.fraction, "%lu",
|
|
stats->core_volume_wr.value);
|
|
|
|
get_stat_name(stat_name, max_stat_len, "Total to/from core", postfix);
|
|
print_val_perc_table_row(outfile, stat_name,
|
|
UNIT_BLOCKS, stats->core_volume_total.fraction, "%lu",
|
|
stats->core_volume_total.value);
|
|
|
|
print_val_perc_table_section(outfile, "Reads from cache",
|
|
UNIT_BLOCKS, stats->cache_volume_rd.fraction, "%lu",
|
|
stats->cache_volume_rd.value);
|
|
|
|
print_val_perc_table_row(outfile, "Writes to cache",
|
|
UNIT_BLOCKS, stats->cache_volume_wr.fraction, "%lu",
|
|
stats->cache_volume_wr.value);
|
|
|
|
print_val_perc_table_row(outfile, "Total to/from cache",
|
|
UNIT_BLOCKS, stats->cache_volume_total.fraction, "%lu",
|
|
stats->cache_volume_total.value);
|
|
|
|
get_stat_name(stat_name, max_stat_len, "Reads from exported object",
|
|
postfix);
|
|
print_val_perc_table_section(outfile, stat_name,
|
|
UNIT_BLOCKS, stats->volume_rd.fraction, "%lu",
|
|
stats->volume_rd.value);
|
|
|
|
get_stat_name(stat_name, max_stat_len, "Writes to exported object",
|
|
postfix);
|
|
print_val_perc_table_row(outfile, stat_name,
|
|
UNIT_BLOCKS, stats->volume_wr.fraction, "%lu",
|
|
stats->volume_wr.value);
|
|
|
|
get_stat_name(stat_name, max_stat_len, "Total to/from exported object",
|
|
postfix);
|
|
print_val_perc_table_row(outfile, stat_name,
|
|
UNIT_BLOCKS, stats->volume_total.fraction, "%lu",
|
|
stats->volume_total.value);
|
|
}
|
|
|
|
static void print_err_stats(const struct ocf_stats_errors *stats,
|
|
FILE *outfile)
|
|
{
|
|
print_table_header(outfile, 4, "Error statistics", "Count", "%",
|
|
"[Units]");
|
|
|
|
print_val_perc_table_section(outfile, "Cache read errors",
|
|
UNIT_REQUESTS, stats->cache_volume_rd.fraction, "%lu",
|
|
stats->cache_volume_rd.value);
|
|
print_val_perc_table_row(outfile, "Cache write errors",
|
|
UNIT_REQUESTS, stats->cache_volume_wr.fraction, "%lu",
|
|
stats->cache_volume_wr.value);
|
|
print_val_perc_table_row(outfile, "Cache total errors",
|
|
UNIT_REQUESTS, stats->cache_volume_total.fraction, "%lu",
|
|
stats->cache_volume_total.value);
|
|
|
|
print_val_perc_table_section(outfile, "Core read errors",
|
|
UNIT_REQUESTS, stats->core_volume_rd.fraction, "%lu",
|
|
stats->core_volume_rd.value);
|
|
print_val_perc_table_row(outfile, "Core write errors",
|
|
UNIT_REQUESTS, stats->core_volume_wr.fraction, "%lu",
|
|
stats->core_volume_wr.value);
|
|
print_val_perc_table_row(outfile, "Core total errors",
|
|
UNIT_REQUESTS, stats->core_volume_total.fraction, "%lu",
|
|
stats->core_volume_total.value);
|
|
|
|
print_val_perc_table_section(outfile, "Total errors",
|
|
UNIT_REQUESTS, stats->total.fraction, "%lu",
|
|
stats->total.value);
|
|
}
|
|
|
|
void cache_stats_core_counters(const struct kcas_core_info *info,
|
|
struct kcas_get_stats *stats,
|
|
unsigned int stats_filters, FILE *outfile)
|
|
{
|
|
begin_record(outfile);
|
|
|
|
if (stats_filters & STATS_FILTER_CONF)
|
|
print_core_conf(info, outfile);
|
|
|
|
if (stats_filters & STATS_FILTER_USAGE)
|
|
print_usage_stats(&stats->usage, outfile);
|
|
|
|
if (stats_filters & STATS_FILTER_REQ)
|
|
print_req_stats(&stats->req, outfile);
|
|
|
|
if (stats_filters & STATS_FILTER_BLK)
|
|
print_blk_stats(&stats->blocks, false, outfile);
|
|
|
|
if (stats_filters & STATS_FILTER_ERR)
|
|
print_err_stats(&stats->errors, outfile);
|
|
}
|
|
|
|
static void print_stats_ioclass_conf(const struct kcas_io_class* io_class,
|
|
FILE* outfile)
|
|
{
|
|
print_kv_pair(outfile, "IO class ID", "%d", io_class->class_id);
|
|
print_kv_pair(outfile, "IO class name", "%s", io_class->info.name);
|
|
if (-1 == io_class->info.priority) {
|
|
print_kv_pair(outfile, "Eviction priority", "Pinned");
|
|
} else {
|
|
print_kv_pair(outfile, "Eviction priority", "%d",
|
|
io_class->info.priority);
|
|
}
|
|
print_kv_pair(outfile, "Max size", "%u%%", io_class->info.max_size);
|
|
}
|
|
|
|
|
|
void cache_stats_inactive_usage(int ctrl_fd, const struct kcas_cache_info *cache_info,
|
|
unsigned int cache_id, FILE* outfile)
|
|
{
|
|
print_table_header(outfile, 4, "Inactive usage statistics", "Count",
|
|
"%", "[Units]");
|
|
|
|
print_val_perc_table_row(outfile, "Inactive Occupancy", UNIT_BLOCKS,
|
|
cache_info->info.inactive.occupancy.fraction, "%lu",
|
|
cache_info->info.inactive.occupancy.value);
|
|
print_val_perc_table_row(outfile, "Inactive Clean", UNIT_BLOCKS,
|
|
cache_info->info.inactive.clean.fraction, "%lu",
|
|
cache_info->info.inactive.clean.value);
|
|
print_val_perc_table_row(outfile, "Inactive Dirty", UNIT_BLOCKS,
|
|
cache_info->info.inactive.dirty.fraction, "%lu",
|
|
cache_info->info.inactive.dirty.value);
|
|
}
|
|
|
|
/**
|
|
* print statistics regarding single io class (partition)
|
|
*/
|
|
void print_stats_ioclass(struct kcas_io_class *io_class,
|
|
struct kcas_get_stats *stats, bool cache_stats,
|
|
FILE *outfile, unsigned int stats_filters)
|
|
{
|
|
if (stats_filters & STATS_FILTER_CONF)
|
|
print_stats_ioclass_conf(io_class, outfile);
|
|
|
|
if (stats_filters & STATS_FILTER_USAGE)
|
|
print_ioclass_usage_stats(&stats->usage, outfile);
|
|
|
|
if (stats_filters & STATS_FILTER_REQ)
|
|
print_req_stats(&stats->req, outfile);
|
|
|
|
if (stats_filters & STATS_FILTER_BLK)
|
|
print_blk_stats(&stats->blocks, cache_stats, outfile);
|
|
}
|
|
|
|
/**
|
|
* @brief print per-io-class statistics for all configured io classes
|
|
*
|
|
*/
|
|
int cache_stats_ioclasses(int ctrl_fd, const struct kcas_cache_info *cache_info,
|
|
unsigned int cache_id, unsigned int core_id,
|
|
int io_class_id, FILE *outfile,
|
|
unsigned int stats_filters)
|
|
{
|
|
struct kcas_io_class info = {};
|
|
struct kcas_get_stats stats = {};
|
|
int part_iter_id;
|
|
bool cache_stats = (core_id == OCF_CORE_ID_INVALID);
|
|
int ret;
|
|
|
|
if (io_class_id != OCF_IO_CLASS_INVALID) {
|
|
info.cache_id = cache_id;
|
|
info.class_id = io_class_id;
|
|
stats.cache_id = cache_id;
|
|
stats.core_id = core_id;
|
|
stats.part_id = io_class_id;
|
|
|
|
ret = ioctl(ctrl_fd, KCAS_IOCTL_PARTITION_INFO, &info);
|
|
if (info.ext_err_code == OCF_ERR_IO_CLASS_NOT_EXIST) {
|
|
cas_printf(LOG_ERR, "IO class %d is not configured.\n",
|
|
io_class_id);
|
|
return FAILURE;
|
|
} else if (ret) {
|
|
print_err(info.ext_err_code);
|
|
return FAILURE;
|
|
}
|
|
|
|
if (ioctl(ctrl_fd, KCAS_IOCTL_GET_STATS, &stats) < 0)
|
|
return FAILURE;
|
|
|
|
begin_record(outfile);
|
|
|
|
print_stats_ioclass(&info, &stats, cache_stats,
|
|
outfile, stats_filters);
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
for (part_iter_id = 0; part_iter_id < OCF_USER_IO_CLASS_MAX; part_iter_id++) {
|
|
info.cache_id = cache_id;
|
|
info.class_id = part_iter_id;
|
|
stats.cache_id = cache_id;
|
|
stats.core_id = core_id;
|
|
stats.part_id = part_iter_id;
|
|
|
|
ret = ioctl(ctrl_fd, KCAS_IOCTL_PARTITION_INFO, &info);
|
|
if (info.ext_err_code == OCF_ERR_IO_CLASS_NOT_EXIST)
|
|
continue;
|
|
else if (ret) {
|
|
print_err(info.ext_err_code);
|
|
return FAILURE;
|
|
}
|
|
|
|
ret = ioctl(ctrl_fd, KCAS_IOCTL_GET_STATS, &stats);
|
|
if (ret)
|
|
return FAILURE;
|
|
|
|
begin_record(outfile);
|
|
|
|
print_stats_ioclass(&info, &stats, cache_stats,
|
|
outfile, stats_filters);
|
|
|
|
memset(&stats, 0, sizeof(stats));
|
|
memset(&info, 0, sizeof(info));
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
int cache_stats_conf(int ctrl_fd, const struct kcas_cache_info *cache_info,
|
|
unsigned int cache_id, FILE *outfile, bool by_id_path)
|
|
{
|
|
float flush_progress = 0;
|
|
float value;
|
|
const char *units;
|
|
long unsigned int cache_size;
|
|
const char *cache_path;
|
|
char dev_path[MAX_STR_LEN];
|
|
int inactive_cores;
|
|
bool standby = (cache_info->info.state & (1 << ocf_cache_state_standby));
|
|
bool standby_detached = cache_info->info.standby_detached;
|
|
bool cache_exported_obj_exists = (standby && !standby_detached);
|
|
|
|
if (strnlen(cache_info->cache_path_name, sizeof(cache_info->cache_path_name)) == 0) {
|
|
cache_path = "-";
|
|
} else if (!by_id_path && get_dev_path(cache_info->cache_path_name, dev_path, sizeof(dev_path)) == SUCCESS) {
|
|
cache_path = dev_path;
|
|
} else {
|
|
cache_path = cache_info->cache_path_name;
|
|
}
|
|
|
|
flush_progress = calculate_flush_progress(cache_info->info.dirty,
|
|
cache_info->info.flushed);
|
|
|
|
print_kv_pair(outfile, "Cache Id", "%d",
|
|
cache_info->cache_id);
|
|
|
|
cache_size = cache_line_in_4k(cache_info->info.size,
|
|
cache_info->info.cache_line_size / KiB);
|
|
|
|
print_kv_pair(outfile, "Cache Size", "%lu, [" UNIT_BLOCKS "], %.2f, [GiB]",
|
|
cache_size,
|
|
(float) cache_size * (4 * KiB) / GiB);
|
|
|
|
print_kv_pair(outfile, "Cache Device", "%s",
|
|
cache_path);
|
|
if (cache_exported_obj_exists) {
|
|
print_kv_pair(outfile, "Exported Object", "/dev/cas-cache-%d",
|
|
cache_info->cache_id);
|
|
} else {
|
|
print_kv_pair(outfile, "Exported Object", "-");
|
|
}
|
|
print_kv_pair(outfile, "Core Devices", "%d",
|
|
cache_info->info.core_count);
|
|
inactive_cores = get_inactive_core_count(cache_info);
|
|
if (inactive_cores < 0)
|
|
return FAILURE;
|
|
print_kv_pair(outfile, "Inactive Core Devices", "%d", inactive_cores);
|
|
|
|
print_kv_pair(outfile, "Write Policy", "%s", standby ? "-" :
|
|
cache_mode_to_name(cache_info->info.cache_mode));
|
|
print_kv_pair(outfile, "Cleaning Policy", "%s", standby ? "-" :
|
|
cleaning_policy_to_name(cache_info->info.cleaning_policy));
|
|
print_kv_pair(outfile, "Promotion Policy", "%s", standby ? "-" :
|
|
promotion_policy_to_name(cache_info->info.promotion_policy));
|
|
print_kv_pair(outfile, "Cache line size", "%llu, [KiB]",
|
|
cache_info->info.cache_line_size / KiB);
|
|
|
|
metadata_memory_footprint(cache_info->info.metadata_footprint,
|
|
&value, &units);
|
|
print_kv_pair(outfile, "Metadata Memory Footprint", "%.1f, [%s]",
|
|
value, units);
|
|
|
|
print_kv_pair_time(outfile, "Dirty for", cache_info->info.dirty_for);
|
|
|
|
if (flush_progress) {
|
|
print_kv_pair(outfile, "Status", "%s (%3.1f %%)",
|
|
"Flushing", flush_progress);
|
|
} else {
|
|
print_kv_pair(outfile, "Status", "%s",
|
|
get_cache_state_name(cache_info->info.state, cache_info->info.standby_detached));
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
void cache_stats_counters(struct kcas_get_stats *cache_stats, FILE *outfile,
|
|
unsigned int stats_filters)
|
|
{
|
|
/* Totals for requests stats. */
|
|
if (stats_filters & STATS_FILTER_REQ)
|
|
print_req_stats(&cache_stats->req, outfile);
|
|
|
|
/* Totals for blocks stats. */
|
|
if (stats_filters & STATS_FILTER_BLK)
|
|
print_blk_stats(&cache_stats->blocks, true, outfile);
|
|
|
|
/* Totals for error stats. */
|
|
if (stats_filters & STATS_FILTER_ERR)
|
|
print_err_stats(&cache_stats->errors, outfile);
|
|
}
|
|
|
|
static int cache_stats(int ctrl_fd, const struct kcas_cache_info *cache_info,
|
|
unsigned int cache_id, FILE *outfile, unsigned int stats_filters,
|
|
bool by_id_path)
|
|
{
|
|
struct kcas_get_stats cache_stats = {};
|
|
bool standby;
|
|
|
|
cache_stats.cache_id = cache_id;
|
|
cache_stats.core_id = OCF_CORE_ID_INVALID;
|
|
cache_stats.part_id = OCF_IO_CLASS_INVALID;
|
|
|
|
standby = !!(cache_info->info.state & (1 << ocf_cache_state_standby));
|
|
|
|
if (!standby) {
|
|
if (ioctl(ctrl_fd, KCAS_IOCTL_GET_STATS, &cache_stats)) {
|
|
print_err(cache_stats.ext_err_code);
|
|
return FAILURE;
|
|
}
|
|
}
|
|
|
|
begin_record(outfile);
|
|
|
|
if (stats_filters & STATS_FILTER_CONF)
|
|
cache_stats_conf(ctrl_fd, cache_info, cache_id, outfile, by_id_path);
|
|
|
|
/* Don't print stats for a cache in standby state */
|
|
if (standby)
|
|
return SUCCESS;
|
|
|
|
if (stats_filters & STATS_FILTER_USAGE)
|
|
print_usage_stats(&cache_stats.usage, outfile);
|
|
|
|
if ((cache_info->info.state & (1 << ocf_cache_state_incomplete))
|
|
&& (stats_filters & STATS_FILTER_USAGE)) {
|
|
cache_stats_inactive_usage(ctrl_fd, cache_info, cache_id, outfile);
|
|
}
|
|
|
|
if (stats_filters & STATS_FILTER_COUNTERS)
|
|
cache_stats_counters(&cache_stats, outfile, stats_filters);
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
int cache_stats_cores(int ctrl_fd, const struct kcas_cache_info *cache_info,
|
|
unsigned int cache_id, unsigned int core_id, int io_class_id,
|
|
FILE *outfile, unsigned int stats_filters, bool by_id_path)
|
|
{
|
|
struct kcas_core_info core_info;
|
|
struct kcas_get_stats stats;
|
|
|
|
if (get_core_info(ctrl_fd, cache_id, core_id, &core_info, by_id_path)) {
|
|
cas_printf(LOG_ERR, "Error while retrieving stats for core %d\n", core_id);
|
|
print_err(core_info.ext_err_code);
|
|
return FAILURE;
|
|
}
|
|
|
|
stats.cache_id = cache_id;
|
|
stats.core_id = core_id;
|
|
stats.part_id = OCF_IO_CLASS_INVALID;
|
|
|
|
if (ioctl(ctrl_fd, KCAS_IOCTL_GET_STATS, &stats) < 0) {
|
|
cas_printf(LOG_ERR, "Error while retrieving stats for core %d\n", core_id);
|
|
print_err(core_info.ext_err_code);
|
|
return FAILURE;
|
|
}
|
|
|
|
cache_stats_core_counters(&core_info, &stats, stats_filters, outfile);
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
struct stats_printout_ctx
|
|
{
|
|
FILE *intermediate;
|
|
FILE *out;
|
|
int type;
|
|
int result;
|
|
};
|
|
void *stats_printout(void *ctx)
|
|
{
|
|
struct stats_printout_ctx *spc = ctx;
|
|
if (stat_format_output(spc->intermediate,
|
|
spc->out, spc->type)) {
|
|
cas_printf(LOG_ERR, "An error occured during statistics formatting.\n");
|
|
spc->result = FAILURE;
|
|
} else {
|
|
spc->result = SUCCESS;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @brief print cache statistics in various variants
|
|
*
|
|
* this routine implements -P (--stats) subcommand of casadm.
|
|
* @param cache_id id of a cache, to which stats query pertains
|
|
* @param stats_filters subset of statistics to be displayed. If filters are not
|
|
* specified STATS_FILTER_DEFAULT are displayd.
|
|
* @param fpath path to an output CSV file to which statistics shall be printed. single "-"
|
|
* can be passed as a path, to generate CSV to stdout. Henceforth non-NULL value of
|
|
* fpath is a sign that stats shall be printed in CSV-format, and NULL value will]
|
|
* cause stats to be printed in pretty tables.
|
|
*
|
|
* @return SUCCESS upon successful printing of statistic. FAILURE if any error happens
|
|
*/
|
|
int cache_status(unsigned int cache_id, unsigned int core_id, int io_class_id,
|
|
unsigned int stats_filters, unsigned int output_format, bool by_id_path)
|
|
{
|
|
int ctrl_fd, i;
|
|
int ret = SUCCESS;
|
|
struct kcas_cache_info cache_info;
|
|
|
|
ctrl_fd = open_ctrl_device();
|
|
|
|
if (ctrl_fd < 0) {
|
|
print_err(KCAS_ERR_SYSTEM);
|
|
return FAILURE;
|
|
}
|
|
|
|
/* 1 is writing end, 0 is reading end of a pipe */
|
|
FILE *intermediate_file[2];
|
|
|
|
if (create_pipe_pair(intermediate_file)) {
|
|
cas_printf(LOG_ERR,"Failed to create unidirectional pipe.\n");
|
|
close(ctrl_fd);
|
|
return FAILURE;
|
|
}
|
|
|
|
/**
|
|
* printing in statistics will be performed in separate
|
|
* thread, so that we can interleave statistics collecting
|
|
* and formatting tables
|
|
*/
|
|
struct stats_printout_ctx printout_ctx;
|
|
printout_ctx.intermediate = intermediate_file[0];
|
|
printout_ctx.out = stdout;
|
|
printout_ctx.type = (OUTPUT_FORMAT_CSV == output_format ? CSV : TEXT);
|
|
pthread_t thread;
|
|
pthread_create(&thread, 0, stats_printout, &printout_ctx);
|
|
|
|
memset(&cache_info, 0, sizeof(cache_info));
|
|
|
|
cache_info.cache_id = cache_id;
|
|
|
|
if (ioctl(ctrl_fd, KCAS_IOCTL_CACHE_INFO, &cache_info) < 0) {
|
|
cas_printf(LOG_ERR, "Cache Id %d not running\n", cache_id);
|
|
ret = FAILURE;
|
|
goto cleanup;
|
|
}
|
|
|
|
if ((cache_info.info.state & (1 << ocf_cache_state_standby)) &&
|
|
core_id != OCF_CORE_ID_INVALID) {
|
|
/* Explicitly fail due to standby mode rather than
|
|
* bouncing off the fact that there are 0 cores in the
|
|
* cache and saying "no such core device"
|
|
*/
|
|
print_err(OCF_ERR_CACHE_STANDBY);
|
|
ret = FAILURE;
|
|
goto cleanup;
|
|
}
|
|
|
|
/* Check if core exists in cache */
|
|
if (core_id != OCF_CORE_ID_INVALID) {
|
|
for (i = 0; i < cache_info.info.core_count; ++i) {
|
|
if (core_id == cache_info.core_id[i]) {
|
|
break;
|
|
}
|
|
}
|
|
if (i == cache_info.info.core_count) {
|
|
cas_printf(LOG_ERR, "No such core device in cache.\n");
|
|
ret = FAILURE;
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
if (stats_filters & STATS_FILTER_IOCLASS) {
|
|
if (cache_stats_ioclasses(ctrl_fd, &cache_info, cache_id,
|
|
core_id, io_class_id,
|
|
intermediate_file[1],
|
|
stats_filters)) {
|
|
ret = FAILURE;
|
|
goto cleanup;
|
|
}
|
|
} else if (core_id == OCF_CORE_ID_INVALID) {
|
|
if (cache_stats(ctrl_fd, &cache_info, cache_id, intermediate_file[1],
|
|
stats_filters, by_id_path)) {
|
|
ret = FAILURE;
|
|
goto cleanup;
|
|
}
|
|
} else {
|
|
if (cache_stats_cores(ctrl_fd, &cache_info, cache_id, core_id,
|
|
io_class_id, intermediate_file[1],
|
|
stats_filters, by_id_path)) {
|
|
ret = FAILURE;
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
cleanup:
|
|
close(ctrl_fd);
|
|
fclose(intermediate_file[1]);
|
|
pthread_join(thread, 0);
|
|
if (printout_ctx.result) {
|
|
ret = 1;
|
|
}
|
|
|
|
fclose(intermediate_file[0]);
|
|
|
|
return ret;
|
|
}
|