-
Notifications
You must be signed in to change notification settings - Fork 84
/
Copy pathmetrics.c
151 lines (123 loc) · 5.15 KB
/
metrics.c
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
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
//! @file
//!
//! Copyright (c) Memfault, Inc.
//! See LICENSE for details
//! @brief Metrics for the application
#include "metrics.h"
#include <malloc.h>
#include "FreeRTOS.h"
#include "memfault/components.h"
#include "memfault/ports/freertos/metrics.h"
#include "memfault/ports/freertos/thread_metrics.h"
#include "task.h"
#define EXAMPLE_TASK_STACKS 300
#define DEBUG_ 0
#if DEBUG_
#include <stdio.h>
#define DEBUG_PRINTF(fmt, ...) printf(fmt, ##__VA_ARGS__)
#else
#define DEBUG_PRINTF(fmt, ...)
#endif
static StackType_t metrics_task_stack[EXAMPLE_TASK_STACKS];
static StaticTask_t metrics_task_tcb;
// This is incompatible with cstd=gnu11 and gcc < 5
#if (__STDC_VERSION__ < 201100L) || (__GNUC__ >= 5)
MEMFAULT_METRICS_DEFINE_THREAD_METRICS(
{
.thread_name = "IDLE",
.stack_usage_metric_key = MEMFAULT_METRICS_KEY(memory_idle_pct_max),
},
{
.thread_name = "Tmr Svc",
.stack_usage_metric_key = MEMFAULT_METRICS_KEY(memory_tmr_svc_pct_max),
},
{
.thread_name = "Console Input",
.stack_usage_metric_key = MEMFAULT_METRICS_KEY(memory_console_input_pct_max),
},
{
.thread_name = "📊 Metrics",
.stack_usage_metric_key = MEMFAULT_METRICS_KEY(memory_metrics_pct_max),
},
{
.thread_name = "Heap Task",
.stack_usage_metric_key = MEMFAULT_METRICS_KEY(memory_heap_task_pct_max),
});
#endif
static void prv_run_metrics_task(MEMFAULT_UNUSED void *pvParameters) {
while (true) {
HeapStats_t stats = { 0 };
vPortGetHeapStats(&stats);
MEMFAULT_METRIC_SET_UNSIGNED(FreeRTOS_HeapFreeBytes, stats.xAvailableHeapSpaceInBytes);
MEMFAULT_METRIC_SET_UNSIGNED(FreeRTOS_HeapMinFreeBytes, stats.xMinimumEverFreeBytesRemaining);
// per-mille (1/1000) percentage, scaled on ingestion
const uint32_t heap_pct_max =
(10000 * (configTOTAL_HEAP_SIZE - stats.xMinimumEverFreeBytesRemaining)) /
configTOTAL_HEAP_SIZE;
MEMFAULT_METRIC_SET_UNSIGNED(FreeRTOS_Heap_pct_max, heap_pct_max);
vTaskDelay((1000 * 10) / portTICK_PERIOD_MS);
}
}
void metrics_task_init(void) {
xTaskCreateStatic(
prv_run_metrics_task, /* Function that implements the task. */
"📊 Metrics", /* Text name for the task. */
MEMFAULT_ARRAY_SIZE(metrics_task_stack), /* Number of indexes in the xStack array. */
NULL, /* Parameter passed into the task. */
tskIDLE_PRIORITY, /* Priority at which the task is created. */
metrics_task_stack, /* Array to use as the task's stack. */
&metrics_task_tcb);
}
static void prv_collect_libc_heap_usage_metrics(void) {
// use mallinfo to compute libc heap utilization percentage
struct mallinfo info = mallinfo();
extern uint32_t _heap_bottom;
extern uint32_t _heap_top;
const uint32_t heap_size = (uint32_t)(&_heap_top - &_heap_bottom);
const uint32_t heap_used = info.uordblks;
const uint32_t heap_pct = (10000 * heap_used) / heap_size;
DEBUG_PRINTF("Heap Usage: %lu/%lu (%lu.%02lu%%)\n", heap_used, heap_size, heap_pct / 100,
heap_pct % 100);
MEMFAULT_METRIC_SET_UNSIGNED(memory_pct_max, heap_pct);
}
static void prv_daily_heartbeat(void) {
// This function is called once a day to stop + start a "daily_heartbeat"
// session.
// The first time this is called on boot will be a no-op, since the session
// has not been started yet
uint32_t uptime_s = memfault_platform_get_time_since_boot_ms() / 1000;
if (uptime_s > MEMFAULT_METRICS_HEARTBEAT_INTERVAL_SECS) {
MEMFAULT_LOG_INFO("📆 Triggering daily heartbeat");
}
// Record a sample metric into the daily session
MEMFAULT_METRIC_SESSION_SET_UNSIGNED(uptime_s, daily_heartbeat, uptime_s);
// End the session
MEMFAULT_METRICS_SESSION_END(daily_heartbeat);
// Start a new session for the next daily interval
MEMFAULT_METRICS_SESSION_START(daily_heartbeat);
}
void memfault_metrics_heartbeat_collect_data(void) {
MEMFAULT_LOG_INFO("💓 Heartbeat callback triggered");
MEMFAULT_STATIC_ASSERT(
MEMFAULT_METRICS_HEARTBEAT_INTERVAL_SECS <= (60 * 60),
"Heartbeat must be an hour or less for runtime metrics to mitigate counter overflow");
#if MEMFAULT_TEST_USE_PORT_TEMPLATE != 1
memfault_freertos_port_task_runtime_metrics();
#endif
prv_collect_libc_heap_usage_metrics();
// Call prv_daily_heartbeat() once a day, based on MEMFAULT_METRICS_HEARTBEAT_INTERVAL_SECS
// This is a simple way to track daily metrics
MEMFAULT_STATIC_ASSERT((24 * 60 * 60) % MEMFAULT_METRICS_HEARTBEAT_INTERVAL_SECS == 0,
"Heartbeat interval must be an even divisor of a day");
// For testing, set MEMFAULT_METRICS_HEARTBEAT_INTERVAL_SECS to a low value (1 second), and
// uncomment the below line
// #define DAILY_HEARTBEAT_INTERVAL_COUNT (24)
#define DAILY_HEARTBEAT_INTERVAL_COUNT ((24 * 60 * 60) / MEMFAULT_METRICS_HEARTBEAT_INTERVAL_SECS)
static uint64_t s_last_daily_heartbeat_interval = 0;
if (++s_last_daily_heartbeat_interval % DAILY_HEARTBEAT_INTERVAL_COUNT == 0) {
prv_daily_heartbeat();
}
// For demonstration purposes, print the current values. This is not
// recommended for production.
memfault_metrics_heartbeat_debug_print();
}