2011-12-26 21:35:57 +00:00
|
|
|
/*
|
|
|
|
htop - AvailableMetersPanel.c
|
|
|
|
(C) 2004-2011 Hisham H. Muhammad
|
2020-10-05 07:51:32 +00:00
|
|
|
Released under the GNU GPLv2, see the COPYING file
|
2011-12-26 21:35:57 +00:00
|
|
|
in the source distribution for its full text.
|
|
|
|
*/
|
2006-03-04 18:16:49 +00:00
|
|
|
|
2006-05-30 13:47:28 +00:00
|
|
|
#include "AvailableMetersPanel.h"
|
2020-09-19 11:55:23 +00:00
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdlib.h>
|
2006-03-04 18:16:49 +00:00
|
|
|
|
2011-12-26 21:35:57 +00:00
|
|
|
#include "CPUMeter.h"
|
2021-06-23 07:44:56 +00:00
|
|
|
#include "DynamicMeter.h"
|
2020-09-19 11:55:23 +00:00
|
|
|
#include "FunctionBar.h"
|
PCP: support for 'dynamic columns' added at runtime
Implements support for arbitrary Performance Co-Pilot
metrics with per-process instance domains to form new
htop columns. The column-to-metric mappings are setup
using configuration files which will be documented via
man pages as part of a follow-up commit.
We provide an initial set of column configurations so
as to provide new capabilities to pcp-htop: including
configs for containers, open fd counts, scheduler run
queue time, tcp/udp bytes/calls sent/recv, delay acct,
virtual machine guests, detailed virtual memory, swap.
Note there is a change to the configuration file path
resolution algorithm introduced for 'dynamic meters'.
First, look in any custom PCP_HTOP_DIR location. Then
iterate, in priority order, users home directory, then
local sysadmins files in /etc/pcp/htop, then readonly
configuration files below /usr/share/pcp/htop. This
final location becomes the preferred place for our own
shipped meter and column files.
The Settings file (htoprc) writing code is updated to
not using the numeric identifier for dynamic columns.
The same strategy used for dynamic meters is used here
where we write Dynamic(name) so the name can be setup
once more at start. Regular (static) columns writing
to htoprc - i.e. numerically indexed - is unchanged.
2021-07-11 01:11:29 +00:00
|
|
|
#include "Hashtable.h"
|
2011-12-26 21:35:57 +00:00
|
|
|
#include "Header.h"
|
|
|
|
#include "ListItem.h"
|
PCP: support for 'dynamic columns' added at runtime
Implements support for arbitrary Performance Co-Pilot
metrics with per-process instance domains to form new
htop columns. The column-to-metric mappings are setup
using configuration files which will be documented via
man pages as part of a follow-up commit.
We provide an initial set of column configurations so
as to provide new capabilities to pcp-htop: including
configs for containers, open fd counts, scheduler run
queue time, tcp/udp bytes/calls sent/recv, delay acct,
virtual machine guests, detailed virtual memory, swap.
Note there is a change to the configuration file path
resolution algorithm introduced for 'dynamic meters'.
First, look in any custom PCP_HTOP_DIR location. Then
iterate, in priority order, users home directory, then
local sysadmins files in /etc/pcp/htop, then readonly
configuration files below /usr/share/pcp/htop. This
final location becomes the preferred place for our own
shipped meter and column files.
The Settings file (htoprc) writing code is updated to
not using the numeric identifier for dynamic columns.
The same strategy used for dynamic meters is used here
where we write Dynamic(name) so the name can be setup
once more at start. Regular (static) columns writing
to htoprc - i.e. numerically indexed - is unchanged.
2021-07-11 01:11:29 +00:00
|
|
|
#include "Macros.h"
|
2020-09-19 11:55:23 +00:00
|
|
|
#include "Meter.h"
|
|
|
|
#include "MetersPanel.h"
|
|
|
|
#include "Object.h"
|
2014-11-27 21:18:14 +00:00
|
|
|
#include "Platform.h"
|
2020-09-19 11:55:23 +00:00
|
|
|
#include "ProvideCurses.h"
|
2020-10-14 18:21:09 +00:00
|
|
|
#include "XUtils.h"
|
2011-12-26 21:35:57 +00:00
|
|
|
|
2006-03-04 18:16:49 +00:00
|
|
|
|
2008-03-09 08:58:38 +00:00
|
|
|
static void AvailableMetersPanel_delete(Object* object) {
|
2006-05-30 13:47:28 +00:00
|
|
|
Panel* super = (Panel*) object;
|
|
|
|
AvailableMetersPanel* this = (AvailableMetersPanel*) object;
|
|
|
|
Panel_done(super);
|
2020-12-25 15:42:35 +00:00
|
|
|
free(this->meterPanels);
|
2006-03-04 18:16:49 +00:00
|
|
|
free(this);
|
|
|
|
}
|
|
|
|
|
2020-12-25 15:42:35 +00:00
|
|
|
static inline void AvailableMetersPanel_addMeter(Header* header, MetersPanel* panel, const MeterClass* type, unsigned int param, size_t column) {
|
2021-01-11 22:46:06 +00:00
|
|
|
const Meter* meter = Header_addMeterByClass(header, type, param, column);
|
2020-12-25 15:42:35 +00:00
|
|
|
Panel_add((Panel*)panel, (Object*) Meter_toListItem(meter, false));
|
|
|
|
Panel_setSelected((Panel*)panel, Panel_size((Panel*)panel) - 1);
|
|
|
|
MetersPanel_setMoving(panel, true);
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
|
|
|
|
2008-03-09 08:58:38 +00:00
|
|
|
static HandlerResult AvailableMetersPanel_eventHandler(Panel* super, int ch) {
|
2006-05-30 13:47:28 +00:00
|
|
|
AvailableMetersPanel* this = (AvailableMetersPanel*) super;
|
2015-01-22 01:27:31 +00:00
|
|
|
Header* header = this->header;
|
2019-10-31 16:39:12 +00:00
|
|
|
|
2020-10-05 16:27:55 +00:00
|
|
|
const ListItem* selected = (ListItem*) Panel_getSelected(super);
|
|
|
|
if (!selected)
|
|
|
|
return IGNORED;
|
|
|
|
|
2021-06-23 07:18:20 +00:00
|
|
|
unsigned int param = selected->key & 0xffff;
|
2006-04-10 20:40:38 +00:00
|
|
|
int type = selected->key >> 16;
|
2006-03-04 18:16:49 +00:00
|
|
|
HandlerResult result = IGNORED;
|
2015-02-03 21:32:07 +00:00
|
|
|
bool update = false;
|
2006-03-04 18:16:49 +00:00
|
|
|
|
|
|
|
switch(ch) {
|
|
|
|
case KEY_F(5):
|
|
|
|
case 'l':
|
|
|
|
case 'L':
|
|
|
|
{
|
2020-12-25 15:42:35 +00:00
|
|
|
AvailableMetersPanel_addMeter(header, this->meterPanels[0], Platform_meterTypes[type], param, 0);
|
2006-03-04 18:16:49 +00:00
|
|
|
result = HANDLED;
|
2015-02-03 21:32:07 +00:00
|
|
|
update = true;
|
2006-03-04 18:16:49 +00:00
|
|
|
break;
|
|
|
|
}
|
2015-02-03 21:32:07 +00:00
|
|
|
case 0x0a:
|
|
|
|
case 0x0d:
|
|
|
|
case KEY_ENTER:
|
2006-03-04 18:16:49 +00:00
|
|
|
case KEY_F(6):
|
|
|
|
case 'r':
|
|
|
|
case 'R':
|
|
|
|
{
|
2020-12-25 15:42:35 +00:00
|
|
|
AvailableMetersPanel_addMeter(header, this->meterPanels[this->columns - 1], Platform_meterTypes[type], param, this->columns - 1);
|
2015-02-03 21:32:07 +00:00
|
|
|
result = (KEY_LEFT << 16) | SYNTH_KEY;
|
|
|
|
update = true;
|
2006-03-04 18:16:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2015-02-03 21:32:07 +00:00
|
|
|
if (update) {
|
2006-03-04 18:16:49 +00:00
|
|
|
this->settings->changed = true;
|
|
|
|
Header_calculateHeight(header);
|
2021-02-08 15:38:49 +00:00
|
|
|
Header_updateData(header);
|
2006-03-04 18:16:49 +00:00
|
|
|
Header_draw(header);
|
2021-08-21 18:42:04 +00:00
|
|
|
ScreenManager_resize(this->scr);
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2008-03-09 08:58:38 +00:00
|
|
|
|
2020-10-05 11:19:50 +00:00
|
|
|
const PanelClass AvailableMetersPanel_class = {
|
2012-12-05 15:12:20 +00:00
|
|
|
.super = {
|
|
|
|
.extends = Class(Panel),
|
|
|
|
.delete = AvailableMetersPanel_delete
|
|
|
|
},
|
|
|
|
.eventHandler = AvailableMetersPanel_eventHandler
|
|
|
|
};
|
|
|
|
|
2021-06-23 07:44:56 +00:00
|
|
|
// Handle (&CPUMeter_class) entries in the AvailableMetersPanel
|
|
|
|
static void AvailableMetersPanel_addCPUMeters(Panel* super, const MeterClass* type, const ProcessList* pl) {
|
2021-06-12 16:17:28 +00:00
|
|
|
if (pl->existingCPUs > 1) {
|
2021-06-23 07:44:56 +00:00
|
|
|
Panel_add(super, (Object*) ListItem_new("CPU average", 0));
|
2021-06-12 16:17:28 +00:00
|
|
|
for (unsigned int i = 1; i <= pl->existingCPUs; i++) {
|
2021-06-23 07:44:56 +00:00
|
|
|
char buffer[50];
|
|
|
|
xSnprintf(buffer, sizeof(buffer), "%s %d", type->uiName, Settings_cpuId(pl->settings, i - 1));
|
|
|
|
Panel_add(super, (Object*) ListItem_new(buffer, i));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Panel_add(super, (Object*) ListItem_new(type->uiName, 1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
Panel* super;
|
|
|
|
unsigned int id;
|
|
|
|
unsigned int offset;
|
|
|
|
} DynamicIterator;
|
|
|
|
|
|
|
|
static void AvailableMetersPanel_addDynamicMeter(ATTR_UNUSED ht_key_t key, void* value, void* data) {
|
|
|
|
const DynamicMeter* meter = (const DynamicMeter*)value;
|
|
|
|
DynamicIterator* iter = (DynamicIterator*)data;
|
|
|
|
unsigned int identifier = (iter->offset << 16) | iter->id;
|
|
|
|
const char* label = meter->description ? meter->description : meter->caption;
|
2021-07-14 17:11:18 +00:00
|
|
|
if (!label)
|
|
|
|
label = meter->name; /* last fallback to name, guaranteed set */
|
2021-06-23 07:44:56 +00:00
|
|
|
Panel_add(iter->super, (Object*) ListItem_new(label, identifier));
|
|
|
|
iter->id++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle (&DynamicMeter_class) entries in the AvailableMetersPanel
|
2021-07-14 17:24:18 +00:00
|
|
|
static void AvailableMetersPanel_addDynamicMeters(Panel* super, const ProcessList* pl, unsigned int offset) {
|
2021-06-23 07:44:56 +00:00
|
|
|
DynamicIterator iter = { .super = super, .id = 1, .offset = offset };
|
|
|
|
assert(pl->dynamicMeters != NULL);
|
|
|
|
Hashtable_foreach(pl->dynamicMeters, AvailableMetersPanel_addDynamicMeter, &iter);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle remaining Platform Meter entries in the AvailableMetersPanel
|
|
|
|
static void AvailableMetersPanel_addPlatformMeter(Panel* super, const MeterClass* type, unsigned int offset) {
|
|
|
|
const char* label = type->description ? type->description : type->uiName;
|
|
|
|
Panel_add(super, (Object*) ListItem_new(label, offset << 16));
|
|
|
|
}
|
|
|
|
|
2020-12-25 15:42:35 +00:00
|
|
|
AvailableMetersPanel* AvailableMetersPanel_new(Settings* settings, Header* header, size_t columns, MetersPanel** meterPanels, ScreenManager* scr, const ProcessList* pl) {
|
2012-12-05 15:12:20 +00:00
|
|
|
AvailableMetersPanel* this = AllocThis(AvailableMetersPanel);
|
2008-03-09 08:58:38 +00:00
|
|
|
Panel* super = (Panel*) this;
|
2015-03-23 22:24:34 +00:00
|
|
|
FunctionBar* fuBar = FunctionBar_newEnterEsc("Add ", "Done ");
|
2015-03-23 18:26:56 +00:00
|
|
|
Panel_init(super, 1, 1, 1, 1, Class(ListItem), true, fuBar);
|
2019-10-31 16:39:12 +00:00
|
|
|
|
2008-03-09 08:58:38 +00:00
|
|
|
this->settings = settings;
|
2015-01-22 01:27:31 +00:00
|
|
|
this->header = header;
|
2020-12-25 15:42:35 +00:00
|
|
|
this->columns = columns;
|
|
|
|
this->meterPanels = meterPanels;
|
2008-03-09 08:58:38 +00:00
|
|
|
this->scr = scr;
|
|
|
|
|
|
|
|
Panel_setHeader(super, "Available meters");
|
2021-06-23 07:44:56 +00:00
|
|
|
// Platform_meterTypes[0] should be always (&CPUMeter_class) which we will
|
|
|
|
// handle separately in the code below. Likewise, identifiers for Dynamic
|
|
|
|
// Meters are handled separately - similar to CPUs, this allows generation
|
|
|
|
// of multiple different Meters (also using 'param' to distinguish them).
|
|
|
|
for (unsigned int i = 1; Platform_meterTypes[i]; i++) {
|
2020-10-05 11:19:50 +00:00
|
|
|
const MeterClass* type = Platform_meterTypes[i];
|
2016-03-19 07:01:13 +00:00
|
|
|
assert(type != &CPUMeter_class);
|
2021-06-23 07:44:56 +00:00
|
|
|
if (type == &DynamicMeter_class)
|
|
|
|
AvailableMetersPanel_addDynamicMeters(super, pl, i);
|
|
|
|
else
|
|
|
|
AvailableMetersPanel_addPlatformMeter(super, type, i);
|
2008-03-09 08:58:38 +00:00
|
|
|
}
|
2021-06-23 07:44:56 +00:00
|
|
|
AvailableMetersPanel_addCPUMeters(super, &CPUMeter_class, pl);
|
|
|
|
|
2008-03-09 08:58:38 +00:00
|
|
|
return this;
|
|
|
|
}
|