Many open source projects have moved to using SPDX identifiers to specify license information, reducing the amount of boilerplate code in every source file. This patch replaces the bulk of SPDK .c, .cpp and Makefiles with the BSD-3-Clause identifier. Almost all of these files share the exact same license text, and this patch only modifies the files that contain the most common license text. There can be slight variations because the third clause contains company names - most say "Intel Corporation", but there are instances for Nvidia, Samsung, Eideticom and even "the copyright holder". Used a bash script to automate replacement of the license text with SPDX identifier which is checked into scripts/spdx.sh. Signed-off-by: Jim Harris <james.r.harris@intel.com> Change-Id: Iaa88ab5e92ea471691dc298cfe41ebfb5d169780 Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/12904 Community-CI: Broadcom CI <spdk-ci.pdl@broadcom.com> Community-CI: Mellanox Build Bot Tested-by: SPDK CI Jenkins <sys_sgci@intel.com> Reviewed-by: Aleksey Marchuk <alexeymar@nvidia.com> Reviewed-by: Changpeng Liu <changpeng.liu@intel.com> Reviewed-by: Dong Yi <dongx.yi@intel.com> Reviewed-by: Konrad Sztyber <konrad.sztyber@intel.com> Reviewed-by: Paul Luse <paul.e.luse@intel.com> Reviewed-by: <qun.wan@intel.com>
235 lines
6.5 KiB
C
235 lines
6.5 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright (c) Intel Corporation.
|
|
* All rights reserved.
|
|
*/
|
|
|
|
/**
|
|
* \file
|
|
* Generic histogram library
|
|
*/
|
|
|
|
#ifndef _SPDK_HISTOGRAM_DATA_H_
|
|
#define _SPDK_HISTOGRAM_DATA_H_
|
|
|
|
#include "spdk/stdinc.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
#define SPDK_HISTOGRAM_BUCKET_SHIFT_DEFAULT 7
|
|
#define SPDK_HISTOGRAM_BUCKET_SHIFT(h) h->bucket_shift
|
|
#define SPDK_HISTOGRAM_BUCKET_LSB(h) (64 - SPDK_HISTOGRAM_BUCKET_SHIFT(h))
|
|
#define SPDK_HISTOGRAM_NUM_BUCKETS_PER_RANGE(h) (1ULL << SPDK_HISTOGRAM_BUCKET_SHIFT(h))
|
|
#define SPDK_HISTOGRAM_BUCKET_MASK(h) (SPDK_HISTOGRAM_NUM_BUCKETS_PER_RANGE(h) - 1)
|
|
#define SPDK_HISTOGRAM_NUM_BUCKET_RANGES(h) (SPDK_HISTOGRAM_BUCKET_LSB(h) + 1)
|
|
#define SPDK_HISTOGRAM_NUM_BUCKETS(h) (SPDK_HISTOGRAM_NUM_BUCKETS_PER_RANGE(h) * \
|
|
SPDK_HISTOGRAM_NUM_BUCKET_RANGES(h))
|
|
|
|
/*
|
|
* SPDK histograms are implemented using ranges of bucket arrays. The most common usage
|
|
* model is using TSC datapoints to capture an I/O latency histogram. For this usage model,
|
|
* the histogram tracks only TSC deltas - any translation to microseconds is done by the
|
|
* histogram user calling spdk_histogram_data_iterate() to iterate over the buckets to perform
|
|
* the translations.
|
|
*
|
|
* Each range has a number of buckets determined by SPDK_HISTOGRAM_NUM_BUCKETS_PER_RANGE
|
|
* which is 128. The buckets in ranges 0 and 1 each map to one specific datapoint value.
|
|
* The buckets in subsequent ranges each map to twice as many datapoint values as buckets
|
|
* in the range before it:
|
|
*
|
|
* Range 0: 1 value each - 128 buckets cover 0 to 127 (2^7-1)
|
|
* Range 1: 1 value each - 128 buckets cover 128 to 255 (2^8-1)
|
|
* Range 2: 2 values each - 128 buckets cover 256 to 511 (2^9-1)
|
|
* Range 3: 4 values each - 128 buckets cover 512 to 1023 (2^10-1)
|
|
* Range 4: 8 values each - 128 buckets cover 1024 to 2047 (2^11-1)
|
|
* Range 5: 16 values each - 128 buckets cover 2048 to 4095 (2^12-1)
|
|
* ...
|
|
* Range 55: 2^54 values each - 128 buckets cover 2^61 to 2^62-1
|
|
* Range 56: 2^55 values each - 128 buckets cover 2^62 to 2^63-1
|
|
* Range 57: 2^56 values each - 128 buckets cover 2^63 to 2^64-1
|
|
*
|
|
* On a 2.3GHz processor, this strategy results in 50ns buckets in the 7-14us range (sweet
|
|
* spot for Intel Optane SSD latency testing).
|
|
*
|
|
* Buckets can be made more granular by increasing SPDK_HISTOGRAM_BUCKET_SHIFT. This
|
|
* comes at the cost of additional storage per namespace context to store the bucket data.
|
|
*/
|
|
|
|
struct spdk_histogram_data {
|
|
|
|
uint32_t bucket_shift;
|
|
uint64_t *bucket;
|
|
|
|
};
|
|
|
|
static inline void
|
|
__spdk_histogram_increment(struct spdk_histogram_data *h, uint32_t range, uint32_t index)
|
|
{
|
|
uint64_t *count;
|
|
|
|
count = &h->bucket[(range << SPDK_HISTOGRAM_BUCKET_SHIFT(h)) + index];
|
|
(*count)++;
|
|
}
|
|
|
|
static inline uint64_t
|
|
__spdk_histogram_get_count(const struct spdk_histogram_data *h, uint32_t range, uint32_t index)
|
|
{
|
|
return h->bucket[(range << SPDK_HISTOGRAM_BUCKET_SHIFT(h)) + index];
|
|
}
|
|
|
|
static inline uint64_t *
|
|
__spdk_histogram_get_bucket(const struct spdk_histogram_data *h, uint32_t range, uint32_t index)
|
|
{
|
|
return &h->bucket[(range << SPDK_HISTOGRAM_BUCKET_SHIFT(h)) + index];
|
|
}
|
|
|
|
static inline void
|
|
spdk_histogram_data_reset(struct spdk_histogram_data *histogram)
|
|
{
|
|
memset(histogram->bucket, 0, SPDK_HISTOGRAM_NUM_BUCKETS(histogram) * sizeof(uint64_t));
|
|
}
|
|
|
|
static inline uint32_t
|
|
__spdk_histogram_data_get_bucket_range(struct spdk_histogram_data *h, uint64_t datapoint)
|
|
{
|
|
uint32_t clz, range;
|
|
|
|
clz = datapoint > 0 ? __builtin_clzll(datapoint) : 64;
|
|
|
|
if (clz <= SPDK_HISTOGRAM_BUCKET_LSB(h)) {
|
|
range = SPDK_HISTOGRAM_BUCKET_LSB(h) - clz;
|
|
} else {
|
|
range = 0;
|
|
}
|
|
|
|
return range;
|
|
}
|
|
|
|
static inline uint32_t
|
|
__spdk_histogram_data_get_bucket_index(struct spdk_histogram_data *h, uint64_t datapoint,
|
|
uint32_t range)
|
|
{
|
|
uint32_t shift;
|
|
|
|
if (range == 0) {
|
|
shift = 0;
|
|
} else {
|
|
shift = range - 1;
|
|
}
|
|
|
|
return (datapoint >> shift) & SPDK_HISTOGRAM_BUCKET_MASK(h);
|
|
}
|
|
|
|
static inline void
|
|
spdk_histogram_data_tally(struct spdk_histogram_data *histogram, uint64_t datapoint)
|
|
{
|
|
uint32_t range = __spdk_histogram_data_get_bucket_range(histogram, datapoint);
|
|
uint32_t index = __spdk_histogram_data_get_bucket_index(histogram, datapoint, range);
|
|
|
|
__spdk_histogram_increment(histogram, range, index);
|
|
}
|
|
|
|
static inline uint64_t
|
|
__spdk_histogram_data_get_bucket_start(const struct spdk_histogram_data *h, uint32_t range,
|
|
uint32_t index)
|
|
{
|
|
uint64_t bucket;
|
|
|
|
index += 1;
|
|
if (range > 0) {
|
|
bucket = 1ULL << (range + SPDK_HISTOGRAM_BUCKET_SHIFT(h) - 1);
|
|
bucket += (uint64_t)index << (range - 1);
|
|
} else {
|
|
bucket = index;
|
|
}
|
|
|
|
return bucket;
|
|
}
|
|
|
|
typedef void (*spdk_histogram_data_fn)(void *ctx, uint64_t start, uint64_t end, uint64_t count,
|
|
uint64_t total, uint64_t so_far);
|
|
|
|
static inline void
|
|
spdk_histogram_data_iterate(const struct spdk_histogram_data *histogram,
|
|
spdk_histogram_data_fn fn, void *ctx)
|
|
{
|
|
uint64_t i, j, count, so_far, total;
|
|
uint64_t bucket, last_bucket;
|
|
|
|
total = 0;
|
|
|
|
for (i = 0; i < SPDK_HISTOGRAM_NUM_BUCKET_RANGES(histogram); i++) {
|
|
for (j = 0; j < SPDK_HISTOGRAM_NUM_BUCKETS_PER_RANGE(histogram); j++) {
|
|
total += __spdk_histogram_get_count(histogram, i, j);
|
|
}
|
|
}
|
|
|
|
so_far = 0;
|
|
bucket = 0;
|
|
|
|
for (i = 0; i < SPDK_HISTOGRAM_NUM_BUCKET_RANGES(histogram); i++) {
|
|
for (j = 0; j < SPDK_HISTOGRAM_NUM_BUCKETS_PER_RANGE(histogram); j++) {
|
|
count = __spdk_histogram_get_count(histogram, i, j);
|
|
so_far += count;
|
|
last_bucket = bucket;
|
|
bucket = __spdk_histogram_data_get_bucket_start(histogram, i, j);
|
|
fn(ctx, last_bucket, bucket, count, total, so_far);
|
|
}
|
|
}
|
|
}
|
|
|
|
static inline void
|
|
spdk_histogram_data_merge(const struct spdk_histogram_data *dst,
|
|
const struct spdk_histogram_data *src)
|
|
{
|
|
uint64_t i;
|
|
|
|
for (i = 0; i < SPDK_HISTOGRAM_NUM_BUCKETS(dst); i++) {
|
|
dst->bucket[i] += src->bucket[i];
|
|
}
|
|
}
|
|
|
|
static inline struct spdk_histogram_data *
|
|
spdk_histogram_data_alloc_sized(uint32_t bucket_shift)
|
|
{
|
|
struct spdk_histogram_data *h;
|
|
|
|
h = (struct spdk_histogram_data *)calloc(1, sizeof(*h));
|
|
if (h == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
h->bucket_shift = bucket_shift;
|
|
h->bucket = (uint64_t *)calloc(SPDK_HISTOGRAM_NUM_BUCKETS(h), sizeof(uint64_t));
|
|
if (h->bucket == NULL) {
|
|
free(h);
|
|
return NULL;
|
|
}
|
|
|
|
return h;
|
|
}
|
|
|
|
static inline struct spdk_histogram_data *
|
|
spdk_histogram_data_alloc(void)
|
|
{
|
|
return spdk_histogram_data_alloc_sized(SPDK_HISTOGRAM_BUCKET_SHIFT_DEFAULT);
|
|
}
|
|
|
|
static inline void
|
|
spdk_histogram_data_free(struct spdk_histogram_data *h)
|
|
{
|
|
if (h == NULL) {
|
|
return;
|
|
}
|
|
|
|
free(h->bucket);
|
|
free(h);
|
|
}
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif
|