inspect_counters/
lib.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

//! Inspectable types for implementing performance counters.

use inspect::Inspect;
use inspect::Value;
use std::sync::atomic::AtomicU64;
use std::sync::atomic::Ordering;

/// A simple 64-bit counter.
#[derive(Debug, Default, Clone)]
pub struct Counter(u64);

impl Counter {
    /// Returns an empty counter.
    pub fn new() -> Self {
        Self::default()
    }

    /// Increments the counter by one.
    pub fn increment(&mut self) {
        self.add(1);
    }

    /// Adds `n` to the counter, wrapping on overflow.
    pub fn add(&mut self, n: u64) {
        self.0 = self.0.wrapping_add(n);
    }

    /// Gets the current counter value.
    pub fn get(&self) -> u64 {
        self.0
    }
}

impl Inspect for Counter {
    fn inspect(&self, req: inspect::Request<'_>) {
        req.value(Value::counter(self.0))
    }
}

/// A 64-bit counter that can be concurrently accessed by multiple threads.
///
/// Prefer [`Counter`] for counters that are not accessed concurrently.
#[derive(Debug, Default)]
pub struct SharedCounter(AtomicU64);

impl SharedCounter {
    /// Returns an empty counter.
    pub fn new() -> Self {
        Self::default()
    }

    /// Increments the counter by one.
    pub fn increment(&self) {
        self.add(1);
    }

    /// Adds `n` to the counter, wrapping on overflow.
    pub fn add(&self, n: u64) {
        self.0.fetch_add(n, Ordering::Relaxed);
    }

    /// Gets the current counter value.
    pub fn get(&self) -> u64 {
        self.0.load(Ordering::Relaxed)
    }
}

impl Inspect for SharedCounter {
    fn inspect(&self, req: inspect::Request<'_>) {
        req.value(Value::counter(self.0.load(Ordering::Relaxed)))
    }
}

/// A power-of-two histogram with `N` buckets.
#[derive(Clone, Debug)]
pub struct Histogram<const N: usize>([u64; N]);

impl<const N: usize> Default for Histogram<N> {
    fn default() -> Self {
        Self::new()
    }
}

impl<const N: usize> Histogram<N> {
    /// Returns an empty histogram.
    pub fn new() -> Self {
        assert!(N > 2);
        assert!(N < BUCKETS.len());
        Self([0; N])
    }

    /// Adds a sample to the histogram.
    pub fn add_sample(&mut self, n: impl Into<u64>) {
        self.0[(64 - n.into().leading_zeros() as usize).min(N - 1)] += 1;
    }
}

static BUCKETS: &[&str] = &[
    "0",
    "1",
    "2-3",
    "4-7",
    "8-15",
    "16-31",
    "32-63",
    "64-127",
    "128-255",
    "256-511",
    "512-1023",
    "1024-2047",
    "2048-4195",
    "4196-8191",
    "8192-16383",
    "16384-32767",
    "32768-65535",
];

static WIDTH: &[usize] = &[1, 1, 1, 1, 1, 2, 2, 2, 3, 3, 3, 4, 4, 4, 4, 5, 5];

impl<const N: usize> Inspect for Histogram<N> {
    fn inspect(&self, req: inspect::Request<'_>) {
        let mut resp = req.respond();
        for (i, &n) in self.0[..N - 1].iter().enumerate() {
            resp.counter(BUCKETS[i], n);
        }
        resp.counter(&BUCKETS[N - 1][..WIDTH[N - 1] + 1], self.0[N - 1]);
    }
}