Line data Source code
1 : use std::marker::PhantomData;
2 :
3 : use measured::label::NoLabels;
4 : use measured::metric::gauge::GaugeState;
5 : use measured::metric::group::Encoding;
6 : use measured::metric::name::MetricNameEncoder;
7 : use measured::metric::{MetricEncoding, MetricFamilyEncoding, MetricType};
8 : use measured::text::TextEncoder;
9 : use measured::{LabelGroup, MetricGroup};
10 : use tikv_jemalloc_ctl::{config, epoch, epoch_mib, stats, version};
11 :
12 : pub struct MetricRecorder {
13 : epoch: epoch_mib,
14 : inner: Metrics,
15 : }
16 :
17 : #[derive(MetricGroup)]
18 : struct Metrics {
19 : active_bytes: JemallocGaugeFamily<stats::active_mib>,
20 : allocated_bytes: JemallocGaugeFamily<stats::allocated_mib>,
21 : mapped_bytes: JemallocGaugeFamily<stats::mapped_mib>,
22 : metadata_bytes: JemallocGaugeFamily<stats::metadata_mib>,
23 : resident_bytes: JemallocGaugeFamily<stats::resident_mib>,
24 : retained_bytes: JemallocGaugeFamily<stats::retained_mib>,
25 : }
26 :
27 : impl<Enc: Encoding> MetricGroup<Enc> for MetricRecorder
28 : where
29 : Metrics: MetricGroup<Enc>,
30 : {
31 0 : fn collect_group_into(&self, enc: &mut Enc) -> Result<(), Enc::Err> {
32 0 : if self.epoch.advance().is_ok() {
33 0 : self.inner.collect_group_into(enc)?;
34 0 : }
35 0 : Ok(())
36 0 : }
37 : }
38 :
39 : impl MetricRecorder {
40 0 : pub fn new() -> Result<Self, anyhow::Error> {
41 0 : tracing::info!(
42 0 : config = config::malloc_conf::read()?,
43 0 : version = version::read()?,
44 0 : "starting jemalloc recorder"
45 : );
46 :
47 : Ok(Self {
48 0 : epoch: epoch::mib()?,
49 : inner: Metrics {
50 0 : active_bytes: JemallocGaugeFamily(stats::active::mib()?),
51 0 : allocated_bytes: JemallocGaugeFamily(stats::allocated::mib()?),
52 0 : mapped_bytes: JemallocGaugeFamily(stats::mapped::mib()?),
53 0 : metadata_bytes: JemallocGaugeFamily(stats::metadata::mib()?),
54 0 : resident_bytes: JemallocGaugeFamily(stats::resident::mib()?),
55 0 : retained_bytes: JemallocGaugeFamily(stats::retained::mib()?),
56 : },
57 : })
58 0 : }
59 : }
60 :
61 : struct JemallocGauge<T>(PhantomData<T>);
62 :
63 : impl<T> Default for JemallocGauge<T> {
64 0 : fn default() -> Self {
65 0 : JemallocGauge(PhantomData)
66 0 : }
67 : }
68 : impl<T> MetricType for JemallocGauge<T> {
69 : type Metadata = T;
70 : }
71 :
72 : struct JemallocGaugeFamily<T>(T);
73 : impl<M, T: Encoding> MetricFamilyEncoding<T> for JemallocGaugeFamily<M>
74 : where
75 : JemallocGauge<M>: MetricEncoding<T, Metadata = M>,
76 : {
77 0 : fn collect_family_into(&self, name: impl MetricNameEncoder, enc: &mut T) -> Result<(), T::Err> {
78 0 : JemallocGauge::write_type(&name, enc)?;
79 0 : JemallocGauge(PhantomData).collect_into(&self.0, NoLabels, name, enc)
80 0 : }
81 : }
82 :
83 : macro_rules! jemalloc_gauge {
84 : ($stat:ident, $mib:ident) => {
85 : impl<W: std::io::Write> MetricEncoding<TextEncoder<W>> for JemallocGauge<stats::$mib> {
86 0 : fn write_type(
87 0 : name: impl MetricNameEncoder,
88 0 : enc: &mut TextEncoder<W>,
89 0 : ) -> Result<(), std::io::Error> {
90 0 : GaugeState::write_type(name, enc)
91 0 : }
92 :
93 0 : fn collect_into(
94 0 : &self,
95 0 : mib: &stats::$mib,
96 0 : labels: impl LabelGroup,
97 0 : name: impl MetricNameEncoder,
98 0 : enc: &mut TextEncoder<W>,
99 0 : ) -> Result<(), std::io::Error> {
100 0 : if let Ok(v) = mib.read() {
101 0 : GaugeState::new(v as i64).collect_into(&(), labels, name, enc)?;
102 0 : }
103 0 : Ok(())
104 0 : }
105 : }
106 : };
107 : }
108 :
109 : jemalloc_gauge!(active, active_mib);
110 : jemalloc_gauge!(allocated, allocated_mib);
111 : jemalloc_gauge!(mapped, mapped_mib);
112 : jemalloc_gauge!(metadata, metadata_mib);
113 : jemalloc_gauge!(resident, resident_mib);
114 : jemalloc_gauge!(retained, retained_mib);
|