2006-03-04 18:16:49 +00:00
|
|
|
/*
|
|
|
|
htop - Settings.c
|
2011-05-26 16:35:07 +00:00
|
|
|
(C) 2004-2011 Hisham H. Muhammad
|
2021-09-22 09:33:00 +00:00
|
|
|
Released under the GNU GPLv2+, see the COPYING file
|
2006-03-04 18:16:49 +00:00
|
|
|
in the source distribution for its full text.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Settings.h"
|
|
|
|
|
2021-08-24 15:40:22 +00:00
|
|
|
#include <ctype.h>
|
2021-03-12 15:56:06 +00:00
|
|
|
#include <errno.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 <limits.h>
|
2020-09-19 11:55:23 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2021-08-24 15:27:43 +00:00
|
|
|
#include <string.h>
|
2020-09-19 11:55:23 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
2015-04-09 18:19:31 +00:00
|
|
|
#include "CRT.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 "DynamicColumn.h"
|
2020-09-19 11:55:23 +00:00
|
|
|
#include "Macros.h"
|
|
|
|
#include "Meter.h"
|
2020-10-14 18:21:09 +00:00
|
|
|
#include "Platform.h"
|
|
|
|
#include "XUtils.h"
|
2006-03-04 18:16:49 +00:00
|
|
|
|
2011-12-26 21:35:57 +00:00
|
|
|
|
2006-03-04 18:16:49 +00:00
|
|
|
void Settings_delete(Settings* this) {
|
2015-01-22 01:27:31 +00:00
|
|
|
free(this->filename);
|
|
|
|
free(this->fields);
|
2020-12-25 15:42:35 +00:00
|
|
|
for (unsigned int i = 0; i < HeaderLayout_getColumns(this->hLayout); i++) {
|
2021-10-05 22:00:13 +00:00
|
|
|
String_freeArray(this->hColumns[i].names);
|
2020-12-25 15:42:35 +00:00
|
|
|
free(this->hColumns[i].modes);
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
2020-12-25 15:42:35 +00:00
|
|
|
free(this->hColumns);
|
2006-03-04 18:16:49 +00:00
|
|
|
free(this);
|
|
|
|
}
|
|
|
|
|
2020-12-25 15:42:35 +00:00
|
|
|
static void Settings_readMeters(Settings* this, const char* line, unsigned int column) {
|
2006-03-04 18:16:49 +00:00
|
|
|
char* trim = String_trim(line);
|
2020-10-03 19:20:43 +00:00
|
|
|
char** ids = String_split(trim, ' ', NULL);
|
2006-03-04 18:16:49 +00:00
|
|
|
free(trim);
|
2020-12-25 15:42:35 +00:00
|
|
|
column = MINIMUM(column, HeaderLayout_getColumns(this->hLayout) - 1);
|
|
|
|
this->hColumns[column].names = ids;
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
|
|
|
|
2020-12-25 15:42:35 +00:00
|
|
|
static void Settings_readMeterModes(Settings* this, const char* line, unsigned int column) {
|
2006-03-04 18:16:49 +00:00
|
|
|
char* trim = String_trim(line);
|
2020-10-03 19:20:43 +00:00
|
|
|
char** ids = String_split(trim, ' ', NULL);
|
2006-03-04 18:16:49 +00:00
|
|
|
free(trim);
|
2015-01-22 01:27:31 +00:00
|
|
|
int len = 0;
|
2011-08-29 20:45:29 +00:00
|
|
|
for (int i = 0; ids[i]; i++) {
|
2015-01-22 01:27:31 +00:00
|
|
|
len++;
|
|
|
|
}
|
2020-12-25 15:42:35 +00:00
|
|
|
column = MINIMUM(column, HeaderLayout_getColumns(this->hLayout) - 1);
|
|
|
|
this->hColumns[column].len = len;
|
2021-01-25 16:31:43 +00:00
|
|
|
int* modes = len ? xCalloc(len, sizeof(int)) : NULL;
|
2015-01-22 01:27:31 +00:00
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
modes[i] = atoi(ids[i]);
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
|
|
|
String_freeArray(ids);
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hColumns[column].modes = modes;
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
|
|
|
|
2021-10-05 22:00:55 +00:00
|
|
|
static bool Settings_validateMeters(Settings* this) {
|
|
|
|
const size_t colCount = HeaderLayout_getColumns(this->hLayout);
|
|
|
|
|
|
|
|
for (size_t column = 0; column < colCount; column++) {
|
|
|
|
char** names = this->hColumns[column].names;
|
|
|
|
const int* modes = this->hColumns[column].modes;
|
|
|
|
const size_t len = this->hColumns[column].len;
|
|
|
|
|
|
|
|
if (!names || !modes || !len)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check for each mode there is an entry with a non-NULL name
|
|
|
|
for (size_t meterIdx = 0; meterIdx < len; meterIdx++)
|
|
|
|
if (!names[meterIdx])
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (names[len])
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-02-17 16:38:35 +00:00
|
|
|
static void Settings_defaultMeters(Settings* this, unsigned int initialCpuCount) {
|
2015-01-22 01:27:31 +00:00
|
|
|
int sizes[] = { 3, 3 };
|
2021-10-05 22:00:13 +00:00
|
|
|
|
2021-02-02 09:32:11 +00:00
|
|
|
if (initialCpuCount > 4 && initialCpuCount <= 128) {
|
2015-01-22 01:27:31 +00:00
|
|
|
sizes[1]++;
|
|
|
|
}
|
2021-10-05 22:00:13 +00:00
|
|
|
|
|
|
|
// Release any previously allocated memory
|
|
|
|
for (size_t i = 0; i < HeaderLayout_getColumns(this->hLayout); i++) {
|
|
|
|
String_freeArray(this->hColumns[i].names);
|
|
|
|
free(this->hColumns[i].modes);
|
|
|
|
}
|
|
|
|
free(this->hColumns);
|
|
|
|
|
|
|
|
this->hLayout = HF_TWO_50_50;
|
|
|
|
this->hColumns = xCalloc(HeaderLayout_getColumns(this->hLayout), sizeof(MeterColumnSetting));
|
|
|
|
for (size_t i = 0; i < 2; i++) {
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hColumns[i].names = xCalloc(sizes[i] + 1, sizeof(char*));
|
|
|
|
this->hColumns[i].modes = xCalloc(sizes[i], sizeof(int));
|
|
|
|
this->hColumns[i].len = sizes[i];
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
2021-10-05 22:00:13 +00:00
|
|
|
|
2015-01-22 01:27:31 +00:00
|
|
|
int r = 0;
|
2021-02-02 09:32:11 +00:00
|
|
|
|
|
|
|
if (initialCpuCount > 128) {
|
|
|
|
// Just show the average, ricers need to config for impressive screenshots
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hColumns[0].names[0] = xStrdup("CPU");
|
|
|
|
this->hColumns[0].modes[0] = BAR_METERMODE;
|
2021-02-02 09:32:11 +00:00
|
|
|
} else if (initialCpuCount > 32) {
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hColumns[0].names[0] = xStrdup("LeftCPUs8");
|
|
|
|
this->hColumns[0].modes[0] = BAR_METERMODE;
|
|
|
|
this->hColumns[1].names[r] = xStrdup("RightCPUs8");
|
|
|
|
this->hColumns[1].modes[r++] = BAR_METERMODE;
|
2021-02-02 09:32:11 +00:00
|
|
|
} else if (initialCpuCount > 16) {
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hColumns[0].names[0] = xStrdup("LeftCPUs4");
|
|
|
|
this->hColumns[0].modes[0] = BAR_METERMODE;
|
|
|
|
this->hColumns[1].names[r] = xStrdup("RightCPUs4");
|
|
|
|
this->hColumns[1].modes[r++] = BAR_METERMODE;
|
2021-02-02 09:32:11 +00:00
|
|
|
} else if (initialCpuCount > 8) {
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hColumns[0].names[0] = xStrdup("LeftCPUs2");
|
|
|
|
this->hColumns[0].modes[0] = BAR_METERMODE;
|
|
|
|
this->hColumns[1].names[r] = xStrdup("RightCPUs2");
|
|
|
|
this->hColumns[1].modes[r++] = BAR_METERMODE;
|
2020-09-23 09:52:57 +00:00
|
|
|
} else if (initialCpuCount > 4) {
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hColumns[0].names[0] = xStrdup("LeftCPUs");
|
|
|
|
this->hColumns[0].modes[0] = BAR_METERMODE;
|
|
|
|
this->hColumns[1].names[r] = xStrdup("RightCPUs");
|
|
|
|
this->hColumns[1].modes[r++] = BAR_METERMODE;
|
2014-11-27 20:38:52 +00:00
|
|
|
} else {
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hColumns[0].names[0] = xStrdup("AllCPUs");
|
|
|
|
this->hColumns[0].modes[0] = BAR_METERMODE;
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hColumns[0].names[1] = xStrdup("Memory");
|
|
|
|
this->hColumns[0].modes[1] = BAR_METERMODE;
|
|
|
|
this->hColumns[0].names[2] = xStrdup("Swap");
|
|
|
|
this->hColumns[0].modes[2] = BAR_METERMODE;
|
|
|
|
this->hColumns[1].names[r] = xStrdup("Tasks");
|
|
|
|
this->hColumns[1].modes[r++] = TEXT_METERMODE;
|
|
|
|
this->hColumns[1].names[r] = xStrdup("LoadAverage");
|
|
|
|
this->hColumns[1].modes[r++] = TEXT_METERMODE;
|
|
|
|
this->hColumns[1].names[r] = xStrdup("Uptime");
|
|
|
|
this->hColumns[1].modes[r++] = TEXT_METERMODE;
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
|
|
|
|
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
|
|
|
static void Settings_readFields(Settings* settings, const char* line) {
|
2015-01-22 01:27:31 +00:00
|
|
|
char* trim = String_trim(line);
|
2020-10-03 19:20:43 +00:00
|
|
|
char** ids = String_split(trim, ' ', NULL);
|
2015-01-22 01:27:31 +00:00
|
|
|
free(trim);
|
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
|
|
|
|
|
|
|
settings->flags = 0;
|
|
|
|
|
|
|
|
unsigned int i, j;
|
|
|
|
for (j = 0, i = 0; ids[i]; i++) {
|
|
|
|
if (j >= UINT_MAX / sizeof(ProcessField))
|
|
|
|
continue;
|
|
|
|
if (j >= LAST_PROCESSFIELD) {
|
|
|
|
settings->fields = xRealloc(settings->fields, j * sizeof(ProcessField));
|
|
|
|
memset(&settings->fields[j], 0, sizeof(ProcessField));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Dynamically-defined columns are always stored by-name.
|
2021-09-02 20:32:40 +00:00
|
|
|
char dynamic[32] = {0};
|
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
|
|
|
if (sscanf(ids[i], "Dynamic(%30s)", dynamic)) {
|
2021-09-02 20:32:40 +00:00
|
|
|
char* end;
|
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
|
|
|
if ((end = strrchr(dynamic, ')')) == NULL)
|
|
|
|
continue;
|
|
|
|
*end = '\0';
|
|
|
|
unsigned int key;
|
|
|
|
if (!DynamicColumn_search(settings->dynamicColumns, dynamic, &key))
|
|
|
|
continue;
|
|
|
|
settings->fields[j++] = key;
|
|
|
|
continue;
|
|
|
|
}
|
2015-01-22 01:27:31 +00:00
|
|
|
// This "+1" is for compatibility with the older enum format.
|
|
|
|
int id = atoi(ids[i]) + 1;
|
2020-12-15 18:44:48 +00:00
|
|
|
if (id > 0 && id < LAST_PROCESSFIELD && Process_fields[id].name) {
|
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
|
|
|
settings->flags |= Process_fields[id].flags;
|
|
|
|
settings->fields[j++] = id;
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
2014-11-27 20:38:52 +00:00
|
|
|
}
|
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
|
|
|
settings->fields[j] = NULL_PROCESSFIELD;
|
2015-01-22 01:27:31 +00:00
|
|
|
String_freeArray(ids);
|
2014-11-27 20:38:52 +00:00
|
|
|
}
|
|
|
|
|
2021-02-17 16:38:35 +00:00
|
|
|
static bool Settings_read(Settings* this, const char* fileName, unsigned int initialCpuCount) {
|
2021-02-16 18:44:59 +00:00
|
|
|
FILE* fd = fopen(fileName, "r");
|
2011-12-25 20:22:41 +00:00
|
|
|
if (!fd)
|
2006-03-04 18:16:49 +00:00
|
|
|
return false;
|
2020-11-01 00:09:51 +00:00
|
|
|
|
2018-02-18 23:35:23 +00:00
|
|
|
bool didReadMeters = false;
|
2021-08-02 15:33:34 +00:00
|
|
|
bool didReadAny = false;
|
2016-06-19 21:55:35 +00:00
|
|
|
for (;;) {
|
|
|
|
char* line = String_readLine(fd);
|
|
|
|
if (!line) {
|
|
|
|
break;
|
|
|
|
}
|
2021-08-02 15:33:34 +00:00
|
|
|
didReadAny = true;
|
2020-10-03 19:20:43 +00:00
|
|
|
size_t nOptions;
|
2016-06-19 21:55:35 +00:00
|
|
|
char** option = String_split(line, '=', &nOptions);
|
|
|
|
free (line);
|
2011-08-29 20:45:29 +00:00
|
|
|
if (nOptions < 2) {
|
|
|
|
String_freeArray(option);
|
|
|
|
continue;
|
|
|
|
}
|
2021-07-16 15:04:23 +00:00
|
|
|
if (String_eq(option[0], "config_reader_min_version")) {
|
|
|
|
this->config_version = atoi(option[1]);
|
|
|
|
if (this->config_version > CONFIG_READER_MIN_VERSION) {
|
2021-08-23 06:58:14 +00:00
|
|
|
// the version of the config file on disk is newer than what we can read
|
2021-09-27 05:13:01 +00:00
|
|
|
fprintf(stderr, "WARNING: %s specifies configuration format\n", fileName);
|
|
|
|
fprintf(stderr, " version v%d, but this %s binary only supports up to version v%d.\n", this->config_version, PACKAGE, CONFIG_READER_MIN_VERSION);
|
|
|
|
fprintf(stderr, " The configuration file will be downgraded to v%d when %s exits.\n", CONFIG_READER_MIN_VERSION, PACKAGE);
|
2021-08-31 05:55:27 +00:00
|
|
|
String_freeArray(option);
|
|
|
|
fclose(fd);
|
2021-08-23 06:58:14 +00:00
|
|
|
return false;
|
2021-07-16 15:04:23 +00:00
|
|
|
}
|
|
|
|
} else if (String_eq(option[0], "fields")) {
|
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
|
|
|
Settings_readFields(this, option[1]);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "sort_key")) {
|
|
|
|
// This "+1" is for compatibility with the older enum format.
|
2015-01-22 01:27:31 +00:00
|
|
|
this->sortKey = atoi(option[1]) + 1;
|
2020-12-18 14:03:31 +00:00
|
|
|
} else if (String_eq(option[0], "tree_sort_key")) {
|
|
|
|
// This "+1" is for compatibility with the older enum format.
|
|
|
|
this->treeSortKey = atoi(option[1]) + 1;
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "sort_direction")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->direction = atoi(option[1]);
|
2020-12-18 14:03:31 +00:00
|
|
|
} else if (String_eq(option[0], "tree_sort_direction")) {
|
|
|
|
this->treeDirection = atoi(option[1]);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "tree_view")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->treeView = atoi(option[1]);
|
2020-12-17 22:08:56 +00:00
|
|
|
} else if (String_eq(option[0], "tree_view_always_by_pid")) {
|
|
|
|
this->treeViewAlwaysByPID = atoi(option[1]);
|
2021-02-12 17:48:09 +00:00
|
|
|
} else if (String_eq(option[0], "all_branches_collapsed")) {
|
|
|
|
this->allBranchesCollapsed = atoi(option[1]);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "hide_kernel_threads")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->hideKernelThreads = atoi(option[1]);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "hide_userland_threads")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->hideUserlandThreads = atoi(option[1]);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "shadow_other_users")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->shadowOtherUsers = atoi(option[1]);
|
2010-02-25 01:37:31 +00:00
|
|
|
} else if (String_eq(option[0], "show_thread_names")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->showThreadNames = atoi(option[1]);
|
2015-07-29 19:14:29 +00:00
|
|
|
} else if (String_eq(option[0], "show_program_path")) {
|
|
|
|
this->showProgramPath = atoi(option[1]);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "highlight_base_name")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->highlightBaseName = atoi(option[1]);
|
2020-12-19 15:46:00 +00:00
|
|
|
} else if (String_eq(option[0], "highlight_deleted_exe")) {
|
|
|
|
this->highlightDeletedExe = atoi(option[1]);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "highlight_megabytes")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->highlightMegabytes = atoi(option[1]);
|
2008-03-08 23:39:48 +00:00
|
|
|
} else if (String_eq(option[0], "highlight_threads")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->highlightThreads = atoi(option[1]);
|
2020-10-31 01:56:16 +00:00
|
|
|
} else if (String_eq(option[0], "highlight_changes")) {
|
|
|
|
this->highlightChanges = atoi(option[1]);
|
|
|
|
} else if (String_eq(option[0], "highlight_changes_delay_secs")) {
|
2021-07-14 17:18:27 +00:00
|
|
|
this->highlightDelaySecs = CLAMP(atoi(option[1]), 1, 24 * 60 * 60);
|
2020-10-17 10:54:45 +00:00
|
|
|
} else if (String_eq(option[0], "find_comm_in_cmdline")) {
|
|
|
|
this->findCommInCmdline = atoi(option[1]);
|
|
|
|
} else if (String_eq(option[0], "strip_exe_from_cmdline")) {
|
|
|
|
this->stripExeFromCmdline = atoi(option[1]);
|
|
|
|
} else if (String_eq(option[0], "show_merged_command")) {
|
|
|
|
this->showMergedCommand = atoi(option[1]);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "header_margin")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->headerMargin = atoi(option[1]);
|
2006-10-04 14:21:27 +00:00
|
|
|
} else if (String_eq(option[0], "expand_system_time")) {
|
2007-11-09 00:40:59 +00:00
|
|
|
// Compatibility option.
|
2015-01-22 01:27:31 +00:00
|
|
|
this->detailedCPUTime = atoi(option[1]);
|
2007-11-09 00:40:59 +00:00
|
|
|
} else if (String_eq(option[0], "detailed_cpu_time")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->detailedCPUTime = atoi(option[1]);
|
2019-12-19 22:30:45 +00:00
|
|
|
} else if (String_eq(option[0], "cpu_count_from_one")) {
|
|
|
|
this->countCPUsFromOne = atoi(option[1]);
|
2011-03-22 20:37:08 +00:00
|
|
|
} else if (String_eq(option[0], "cpu_count_from_zero")) {
|
2019-12-19 22:30:45 +00:00
|
|
|
// old (inverted) naming also supported for backwards compatibility
|
|
|
|
this->countCPUsFromOne = !atoi(option[1]);
|
2019-08-10 18:20:21 +00:00
|
|
|
} else if (String_eq(option[0], "show_cpu_usage")) {
|
|
|
|
this->showCPUUsage = atoi(option[1]);
|
2019-08-10 04:34:48 +00:00
|
|
|
} else if (String_eq(option[0], "show_cpu_frequency")) {
|
|
|
|
this->showCPUFrequency = atoi(option[1]);
|
2020-12-22 19:02:01 +00:00
|
|
|
#ifdef BUILD_WITH_CPU_TEMP
|
2020-09-10 17:56:33 +00:00
|
|
|
} else if (String_eq(option[0], "show_cpu_temperature")) {
|
|
|
|
this->showCPUTemperature = atoi(option[1]);
|
|
|
|
} else if (String_eq(option[0], "degree_fahrenheit")) {
|
|
|
|
this->degreeFahrenheit = atoi(option[1]);
|
|
|
|
#endif
|
2012-10-20 00:43:25 +00:00
|
|
|
} else if (String_eq(option[0], "update_process_names")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->updateProcessNames = atoi(option[1]);
|
2013-12-18 02:58:34 +00:00
|
|
|
} else if (String_eq(option[0], "account_guest_in_cpu_meter")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->accountGuestInCPUMeter = atoi(option[1]);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "delay")) {
|
2020-11-21 20:40:08 +00:00
|
|
|
this->delay = CLAMP(atoi(option[1]), 1, 255);
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "color_scheme")) {
|
|
|
|
this->colorScheme = atoi(option[1]);
|
2020-11-01 00:09:51 +00:00
|
|
|
if (this->colorScheme < 0 || this->colorScheme >= LAST_COLORSCHEME) {
|
2020-10-31 21:14:27 +00:00
|
|
|
this->colorScheme = 0;
|
2020-11-01 00:09:51 +00:00
|
|
|
}
|
2021-07-14 19:07:43 +00:00
|
|
|
#ifdef HAVE_GETMOUSE
|
2020-10-31 21:14:27 +00:00
|
|
|
} else if (String_eq(option[0], "enable_mouse")) {
|
2019-07-12 19:41:09 +00:00
|
|
|
this->enableMouse = atoi(option[1]);
|
2021-07-14 19:07:43 +00:00
|
|
|
#endif
|
2020-12-25 15:42:35 +00:00
|
|
|
} else if (String_eq(option[0], "header_layout")) {
|
2021-08-24 15:40:22 +00:00
|
|
|
this->hLayout = isdigit((unsigned char)option[1][0]) ? ((HeaderLayout) atoi(option[1])) : HeaderLayout_fromName(option[1]);
|
2020-12-25 15:42:35 +00:00
|
|
|
if (this->hLayout < 0 || this->hLayout >= LAST_HEADER_LAYOUT)
|
|
|
|
this->hLayout = HF_TWO_50_50;
|
|
|
|
free(this->hColumns);
|
|
|
|
this->hColumns = xCalloc(HeaderLayout_getColumns(this->hLayout), sizeof(MeterColumnSetting));
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "left_meters")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
Settings_readMeters(this, option[1], 0);
|
2018-02-18 23:35:23 +00:00
|
|
|
didReadMeters = true;
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "right_meters")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
Settings_readMeters(this, option[1], 1);
|
2018-02-18 23:35:23 +00:00
|
|
|
didReadMeters = true;
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "left_meter_modes")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
Settings_readMeterModes(this, option[1], 0);
|
2018-02-18 23:35:23 +00:00
|
|
|
didReadMeters = true;
|
2006-03-04 18:16:49 +00:00
|
|
|
} else if (String_eq(option[0], "right_meter_modes")) {
|
2015-01-22 01:27:31 +00:00
|
|
|
Settings_readMeterModes(this, option[1], 1);
|
2018-02-18 23:35:23 +00:00
|
|
|
didReadMeters = true;
|
2020-12-25 15:42:35 +00:00
|
|
|
} else if (String_startsWith(option[0], "column_meters_")) {
|
|
|
|
Settings_readMeters(this, option[1], atoi(option[0] + strlen("column_meters_")));
|
|
|
|
didReadMeters = true;
|
|
|
|
} else if (String_startsWith(option[0], "column_meter_modes_")) {
|
|
|
|
Settings_readMeterModes(this, option[1], atoi(option[0] + strlen("column_meter_modes_")));
|
|
|
|
didReadMeters = true;
|
2020-12-28 22:26:14 +00:00
|
|
|
} else if (String_eq(option[0], "hide_function_bar")) {
|
|
|
|
this->hideFunctionBar = atoi(option[1]);
|
2020-08-26 00:15:00 +00:00
|
|
|
#ifdef HAVE_LIBHWLOC
|
|
|
|
} else if (String_eq(option[0], "topology_affinity")) {
|
|
|
|
this->topologyAffinity = !!atoi(option[1]);
|
|
|
|
#endif
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
|
|
|
String_freeArray(option);
|
|
|
|
}
|
|
|
|
fclose(fd);
|
2021-10-05 22:00:55 +00:00
|
|
|
if (!didReadMeters || !Settings_validateMeters(this)) {
|
2020-09-23 09:52:57 +00:00
|
|
|
Settings_defaultMeters(this, initialCpuCount);
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
2021-08-02 15:33:34 +00:00
|
|
|
return didReadAny;
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
|
|
|
|
2021-08-08 17:47:17 +00:00
|
|
|
static void writeFields(FILE* fd, const ProcessField* fields, Hashtable* columns, const char* name, char separator) {
|
2015-01-22 01:27:31 +00:00
|
|
|
fprintf(fd, "%s=", name);
|
2017-07-24 23:36:27 +00:00
|
|
|
const char* sep = "";
|
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
|
|
|
for (unsigned int i = 0; fields[i]; i++) {
|
|
|
|
if (fields[i] >= LAST_PROCESSFIELD) {
|
|
|
|
const DynamicColumn* column = DynamicColumn_lookup(columns, fields[i]);
|
|
|
|
fprintf(fd, "%sDynamic(%s)", sep, column->name);
|
|
|
|
} else {
|
|
|
|
// This "-1" is for compatibility with the older enum format.
|
|
|
|
fprintf(fd, "%s%d", sep, (int) fields[i] - 1);
|
|
|
|
}
|
2017-07-24 23:36:27 +00:00
|
|
|
sep = " ";
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
2021-08-08 17:47:17 +00:00
|
|
|
fputc(separator, fd);
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
|
|
|
|
2021-08-08 17:47:17 +00:00
|
|
|
static void writeMeters(const Settings* this, FILE* fd, char separator, unsigned int column) {
|
2017-07-24 23:36:27 +00:00
|
|
|
const char* sep = "";
|
2021-09-04 12:15:47 +00:00
|
|
|
for (size_t i = 0; i < this->hColumns[column].len; i++) {
|
2020-12-25 15:42:35 +00:00
|
|
|
fprintf(fd, "%s%s", sep, this->hColumns[column].names[i]);
|
2017-07-24 23:36:27 +00:00
|
|
|
sep = " ";
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
2021-08-08 17:47:17 +00:00
|
|
|
fputc(separator, fd);
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
|
|
|
|
2021-08-08 17:47:17 +00:00
|
|
|
static void writeMeterModes(const Settings* this, FILE* fd, char separator, unsigned int column) {
|
2017-07-24 23:36:27 +00:00
|
|
|
const char* sep = "";
|
2021-09-04 12:15:47 +00:00
|
|
|
for (size_t i = 0; i < this->hColumns[column].len; i++) {
|
2020-12-25 15:42:35 +00:00
|
|
|
fprintf(fd, "%s%d", sep, this->hColumns[column].modes[i]);
|
2017-07-24 23:36:27 +00:00
|
|
|
sep = " ";
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
2021-08-08 17:47:17 +00:00
|
|
|
fputc(separator, fd);
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
|
|
|
|
2021-05-16 17:55:31 +00:00
|
|
|
int Settings_write(const Settings* this, bool onCrash) {
|
|
|
|
FILE* fd;
|
2021-08-08 17:47:17 +00:00
|
|
|
char separator;
|
2021-05-16 17:55:31 +00:00
|
|
|
if (onCrash) {
|
|
|
|
fd = stderr;
|
2021-08-08 17:47:17 +00:00
|
|
|
separator = ';';
|
2021-05-16 17:55:31 +00:00
|
|
|
} else {
|
|
|
|
fd = fopen(this->filename, "w");
|
|
|
|
if (fd == NULL)
|
|
|
|
return -errno;
|
2021-08-08 17:47:17 +00:00
|
|
|
separator = '\n';
|
2021-05-16 17:55:31 +00:00
|
|
|
}
|
2021-02-16 18:44:59 +00:00
|
|
|
|
2021-08-08 17:47:17 +00:00
|
|
|
#define printSettingInteger(setting_, value_) \
|
2021-09-04 12:14:57 +00:00
|
|
|
fprintf(fd, setting_ "=%d%c", (int) (value_), separator)
|
2021-08-08 17:47:17 +00:00
|
|
|
#define printSettingString(setting_, value_) \
|
2021-09-04 12:14:57 +00:00
|
|
|
fprintf(fd, setting_ "=%s%c", value_, separator)
|
2021-08-08 17:47:17 +00:00
|
|
|
|
2021-05-16 17:55:31 +00:00
|
|
|
if (!onCrash) {
|
|
|
|
fprintf(fd, "# Beware! This file is rewritten by htop when settings are changed in the interface.\n");
|
|
|
|
fprintf(fd, "# The parser is also very primitive, and not human-friendly.\n");
|
|
|
|
}
|
2021-08-08 17:47:17 +00:00
|
|
|
printSettingString("htop_version", VERSION);
|
|
|
|
printSettingInteger("config_reader_min_version", CONFIG_READER_MIN_VERSION);
|
|
|
|
writeFields(fd, this->fields, this->dynamicColumns, "fields", separator);
|
2006-03-04 18:16:49 +00:00
|
|
|
// This "-1" is for compatibility with the older enum format.
|
2021-08-08 17:47:17 +00:00
|
|
|
printSettingInteger("sort_key", this->sortKey - 1);
|
|
|
|
printSettingInteger("sort_direction", this->direction);
|
|
|
|
printSettingInteger("tree_sort_key", this->treeSortKey - 1);
|
|
|
|
printSettingInteger("tree_sort_direction", this->treeDirection);
|
|
|
|
printSettingInteger("hide_kernel_threads", this->hideKernelThreads);
|
|
|
|
printSettingInteger("hide_userland_threads", this->hideUserlandThreads);
|
|
|
|
printSettingInteger("shadow_other_users", this->shadowOtherUsers);
|
|
|
|
printSettingInteger("show_thread_names", this->showThreadNames);
|
|
|
|
printSettingInteger("show_program_path", this->showProgramPath);
|
|
|
|
printSettingInteger("highlight_base_name", this->highlightBaseName);
|
|
|
|
printSettingInteger("highlight_deleted_exe", this->highlightDeletedExe);
|
|
|
|
printSettingInteger("highlight_megabytes", this->highlightMegabytes);
|
|
|
|
printSettingInteger("highlight_threads", this->highlightThreads);
|
|
|
|
printSettingInteger("highlight_changes", this->highlightChanges);
|
|
|
|
printSettingInteger("highlight_changes_delay_secs", this->highlightDelaySecs);
|
|
|
|
printSettingInteger("find_comm_in_cmdline", this->findCommInCmdline);
|
|
|
|
printSettingInteger("strip_exe_from_cmdline", this->stripExeFromCmdline);
|
|
|
|
printSettingInteger("show_merged_command", this->showMergedCommand);
|
|
|
|
printSettingInteger("tree_view", this->treeView);
|
|
|
|
printSettingInteger("tree_view_always_by_pid", this->treeViewAlwaysByPID);
|
|
|
|
printSettingInteger("all_branches_collapsed", this->allBranchesCollapsed);
|
|
|
|
printSettingInteger("header_margin", this->headerMargin);
|
|
|
|
printSettingInteger("detailed_cpu_time", this->detailedCPUTime);
|
|
|
|
printSettingInteger("cpu_count_from_one", this->countCPUsFromOne);
|
|
|
|
printSettingInteger("show_cpu_usage", this->showCPUUsage);
|
|
|
|
printSettingInteger("show_cpu_frequency", this->showCPUFrequency);
|
2020-12-22 19:02:01 +00:00
|
|
|
#ifdef BUILD_WITH_CPU_TEMP
|
2021-08-08 17:47:17 +00:00
|
|
|
printSettingInteger("show_cpu_temperature", this->showCPUTemperature);
|
|
|
|
printSettingInteger("degree_fahrenheit", this->degreeFahrenheit);
|
2020-09-10 17:56:33 +00:00
|
|
|
#endif
|
2021-08-08 17:47:17 +00:00
|
|
|
printSettingInteger("update_process_names", this->updateProcessNames);
|
|
|
|
printSettingInteger("account_guest_in_cpu_meter", this->accountGuestInCPUMeter);
|
|
|
|
printSettingInteger("color_scheme", this->colorScheme);
|
2021-07-14 19:07:43 +00:00
|
|
|
#ifdef HAVE_GETMOUSE
|
2021-08-08 17:47:17 +00:00
|
|
|
printSettingInteger("enable_mouse", this->enableMouse);
|
|
|
|
#endif
|
|
|
|
printSettingInteger("delay", (int) this->delay);
|
|
|
|
printSettingInteger("hide_function_bar", (int) this->hideFunctionBar);
|
|
|
|
#ifdef HAVE_LIBHWLOC
|
|
|
|
printSettingInteger("topology_affinity", this->topologyAffinity);
|
2021-07-14 19:07:43 +00:00
|
|
|
#endif
|
2021-08-08 17:47:17 +00:00
|
|
|
|
|
|
|
printSettingString("header_layout", HeaderLayout_getName(this->hLayout));
|
2020-12-25 15:42:35 +00:00
|
|
|
for (unsigned int i = 0; i < HeaderLayout_getColumns(this->hLayout); i++) {
|
|
|
|
fprintf(fd, "column_meters_%u=", i);
|
2021-08-08 17:47:17 +00:00
|
|
|
writeMeters(this, fd, separator, i);
|
2020-12-25 15:42:35 +00:00
|
|
|
fprintf(fd, "column_meter_modes_%u=", i);
|
2021-08-08 17:47:17 +00:00
|
|
|
writeMeterModes(this, fd, separator, i);
|
2020-12-25 15:42:35 +00:00
|
|
|
}
|
2021-08-08 17:47:17 +00:00
|
|
|
|
|
|
|
#undef printSettingString
|
|
|
|
#undef printSettingInteger
|
2021-03-12 15:56:06 +00:00
|
|
|
|
2021-05-16 17:55:31 +00:00
|
|
|
if (onCrash)
|
|
|
|
return 0;
|
|
|
|
|
2021-03-13 17:15:20 +00:00
|
|
|
int r = 0;
|
|
|
|
|
|
|
|
if (ferror(fd) != 0)
|
|
|
|
r = (errno != 0) ? -errno : -EBADF;
|
|
|
|
|
|
|
|
if (fclose(fd) != 0)
|
|
|
|
r = r ? r : -errno;
|
|
|
|
|
|
|
|
return r;
|
2006-03-04 18:16:49 +00:00
|
|
|
}
|
2008-03-09 08:58:38 +00:00
|
|
|
|
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
|
|
|
Settings* Settings_new(unsigned int initialCpuCount, Hashtable* dynamicColumns) {
|
2016-02-02 14:53:02 +00:00
|
|
|
Settings* this = xCalloc(1, sizeof(Settings));
|
2015-01-22 01:27:31 +00:00
|
|
|
|
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
|
|
|
this->dynamicColumns = dynamicColumns;
|
2020-12-25 15:42:35 +00:00
|
|
|
this->hLayout = HF_TWO_50_50;
|
|
|
|
this->hColumns = xCalloc(HeaderLayout_getColumns(this->hLayout), sizeof(MeterColumnSetting));
|
2015-01-22 01:27:31 +00:00
|
|
|
this->sortKey = PERCENT_CPU;
|
2020-12-18 14:03:31 +00:00
|
|
|
this->treeSortKey = PID;
|
2021-01-21 13:27:23 +00:00
|
|
|
this->direction = -1;
|
2020-12-18 14:03:31 +00:00
|
|
|
this->treeDirection = 1;
|
2015-01-22 01:27:31 +00:00
|
|
|
this->shadowOtherUsers = false;
|
|
|
|
this->showThreadNames = false;
|
2021-08-02 15:33:34 +00:00
|
|
|
this->hideKernelThreads = true;
|
2015-01-22 01:27:31 +00:00
|
|
|
this->hideUserlandThreads = false;
|
|
|
|
this->treeView = false;
|
2021-02-12 17:48:09 +00:00
|
|
|
this->allBranchesCollapsed = false;
|
2015-01-22 01:27:31 +00:00
|
|
|
this->highlightBaseName = false;
|
2020-12-19 15:46:00 +00:00
|
|
|
this->highlightDeletedExe = true;
|
2021-08-02 15:33:34 +00:00
|
|
|
this->highlightMegabytes = true;
|
2015-01-22 01:27:31 +00:00
|
|
|
this->detailedCPUTime = false;
|
2019-12-19 22:30:45 +00:00
|
|
|
this->countCPUsFromOne = false;
|
2019-08-10 18:20:21 +00:00
|
|
|
this->showCPUUsage = true;
|
2019-08-10 04:34:48 +00:00
|
|
|
this->showCPUFrequency = false;
|
2020-12-22 19:02:01 +00:00
|
|
|
#ifdef BUILD_WITH_CPU_TEMP
|
2020-09-10 17:56:33 +00:00
|
|
|
this->showCPUTemperature = false;
|
|
|
|
this->degreeFahrenheit = false;
|
|
|
|
#endif
|
2015-01-22 01:27:31 +00:00
|
|
|
this->updateProcessNames = false;
|
2015-08-12 20:24:41 +00:00
|
|
|
this->showProgramPath = true;
|
2016-01-31 11:07:48 +00:00
|
|
|
this->highlightThreads = true;
|
2020-11-01 00:36:53 +00:00
|
|
|
this->highlightChanges = false;
|
2020-10-31 01:56:16 +00:00
|
|
|
this->highlightDelaySecs = DEFAULT_HIGHLIGHT_SECS;
|
2020-10-17 10:54:45 +00:00
|
|
|
this->findCommInCmdline = true;
|
|
|
|
this->stripExeFromCmdline = true;
|
|
|
|
this->showMergedCommand = false;
|
2020-12-28 22:26:14 +00:00
|
|
|
this->hideFunctionBar = 0;
|
2021-08-02 15:33:34 +00:00
|
|
|
this->headerMargin = true;
|
2020-08-26 00:15:00 +00:00
|
|
|
#ifdef HAVE_LIBHWLOC
|
|
|
|
this->topologyAffinity = false;
|
|
|
|
#endif
|
2020-12-15 18:44:48 +00:00
|
|
|
this->fields = xCalloc(LAST_PROCESSFIELD + 1, sizeof(ProcessField));
|
2015-01-22 01:27:31 +00:00
|
|
|
// TODO: turn 'fields' into a Vector,
|
|
|
|
// (and ProcessFields into proper objects).
|
|
|
|
this->flags = 0;
|
2020-12-19 17:10:03 +00:00
|
|
|
const ProcessField* defaults = Platform_defaultFields;
|
2015-01-22 01:27:31 +00:00
|
|
|
for (int i = 0; defaults[i]; i++) {
|
|
|
|
this->fields[i] = defaults[i];
|
|
|
|
this->flags |= Process_fields[defaults[i]].flags;
|
|
|
|
}
|
|
|
|
|
2011-12-25 20:22:41 +00:00
|
|
|
char* legacyDotfile = NULL;
|
2021-01-05 22:42:55 +00:00
|
|
|
const char* rcfile = getenv("HTOPRC");
|
2011-12-25 20:22:41 +00:00
|
|
|
if (rcfile) {
|
2016-02-02 14:53:02 +00:00
|
|
|
this->filename = xStrdup(rcfile);
|
2011-12-25 20:22:41 +00:00
|
|
|
} else {
|
|
|
|
const char* home = getenv("HOME");
|
2020-11-01 00:09:51 +00:00
|
|
|
if (!home)
|
|
|
|
home = "";
|
|
|
|
|
2011-12-25 20:22:41 +00:00
|
|
|
const char* xdgConfigHome = getenv("XDG_CONFIG_HOME");
|
|
|
|
char* configDir = NULL;
|
|
|
|
char* htopDir = NULL;
|
|
|
|
if (xdgConfigHome) {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->filename = String_cat(xdgConfigHome, "/htop/htoprc");
|
2016-02-02 14:53:02 +00:00
|
|
|
configDir = xStrdup(xdgConfigHome);
|
2011-12-25 20:22:41 +00:00
|
|
|
htopDir = String_cat(xdgConfigHome, "/htop");
|
|
|
|
} else {
|
2015-01-22 01:27:31 +00:00
|
|
|
this->filename = String_cat(home, "/.config/htop/htoprc");
|
2011-12-25 20:22:41 +00:00
|
|
|
configDir = String_cat(home, "/.config");
|
|
|
|
htopDir = String_cat(home, "/.config/htop");
|
|
|
|
}
|
|
|
|
legacyDotfile = String_cat(home, "/.htoprc");
|
2014-04-22 23:35:57 +00:00
|
|
|
(void) mkdir(configDir, 0700);
|
|
|
|
(void) mkdir(htopDir, 0700);
|
2011-12-25 20:22:41 +00:00
|
|
|
free(htopDir);
|
|
|
|
free(configDir);
|
2012-03-05 11:12:58 +00:00
|
|
|
struct stat st;
|
2018-02-18 23:42:17 +00:00
|
|
|
int err = lstat(legacyDotfile, &st);
|
|
|
|
if (err || S_ISLNK(st.st_mode)) {
|
2011-12-25 20:22:41 +00:00
|
|
|
free(legacyDotfile);
|
|
|
|
legacyDotfile = NULL;
|
|
|
|
}
|
|
|
|
}
|
2008-03-09 08:58:38 +00:00
|
|
|
this->colorScheme = 0;
|
2021-07-14 19:07:43 +00:00
|
|
|
#ifdef HAVE_GETMOUSE
|
2019-07-12 19:41:09 +00:00
|
|
|
this->enableMouse = true;
|
2021-07-14 19:07:43 +00:00
|
|
|
#endif
|
2008-03-09 08:58:38 +00:00
|
|
|
this->changed = false;
|
|
|
|
this->delay = DEFAULT_DELAY;
|
2018-02-18 23:42:17 +00:00
|
|
|
bool ok = false;
|
|
|
|
if (legacyDotfile) {
|
2020-09-23 09:52:57 +00:00
|
|
|
ok = Settings_read(this, legacyDotfile, initialCpuCount);
|
2018-02-18 23:42:17 +00:00
|
|
|
if (ok) {
|
2011-12-25 20:22:41 +00:00
|
|
|
// Transition to new location and delete old configuration file
|
2021-05-16 17:55:31 +00:00
|
|
|
if (Settings_write(this, false) == 0) {
|
2011-12-25 20:22:41 +00:00
|
|
|
unlink(legacyDotfile);
|
2020-11-01 00:09:51 +00:00
|
|
|
}
|
2011-12-25 20:22:41 +00:00
|
|
|
}
|
2018-02-18 23:42:17 +00:00
|
|
|
free(legacyDotfile);
|
|
|
|
}
|
|
|
|
if (!ok) {
|
2020-09-23 09:52:57 +00:00
|
|
|
ok = Settings_read(this, this->filename, initialCpuCount);
|
2018-02-18 23:42:17 +00:00
|
|
|
}
|
|
|
|
if (!ok) {
|
2008-03-09 08:58:38 +00:00
|
|
|
this->changed = true;
|
2021-09-09 07:04:41 +00:00
|
|
|
ok = Settings_read(this, SYSCONFDIR "/htoprc", initialCpuCount);
|
2018-02-18 23:42:17 +00:00
|
|
|
}
|
2021-08-28 15:57:51 +00:00
|
|
|
if (!ok) {
|
|
|
|
Settings_defaultMeters(this, initialCpuCount);
|
|
|
|
}
|
2008-03-09 08:58:38 +00:00
|
|
|
return this;
|
|
|
|
}
|
2015-01-22 01:27:31 +00:00
|
|
|
|
|
|
|
void Settings_invertSortOrder(Settings* this) {
|
2020-12-18 14:03:31 +00:00
|
|
|
int* attr = (this->treeView) ? &(this->treeDirection) : &(this->direction);
|
|
|
|
*attr = (*attr == 1) ? -1 : 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Settings_setSortKey(Settings* this, ProcessField sortKey) {
|
|
|
|
if (this->treeViewAlwaysByPID || !this->treeView) {
|
|
|
|
this->sortKey = sortKey;
|
2021-01-21 13:27:23 +00:00
|
|
|
this->direction = (Process_fields[sortKey].defaultSortDesc) ? -1 : 1;
|
2020-12-18 14:03:31 +00:00
|
|
|
this->treeView = false;
|
|
|
|
} else {
|
|
|
|
this->treeSortKey = sortKey;
|
2021-01-21 13:27:23 +00:00
|
|
|
this->treeDirection = (Process_fields[sortKey].defaultSortDesc) ? -1 : 1;
|
2020-11-01 00:09:51 +00:00
|
|
|
}
|
2015-01-22 01:27:31 +00:00
|
|
|
}
|
2021-01-21 19:27:37 +00:00
|
|
|
|
|
|
|
static bool readonly = false;
|
|
|
|
|
|
|
|
void Settings_enableReadonly(void) {
|
|
|
|
readonly = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Settings_isReadonly(void) {
|
|
|
|
return readonly;
|
|
|
|
}
|
2020-12-25 15:42:35 +00:00
|
|
|
|
|
|
|
void Settings_setHeaderLayout(Settings* this, HeaderLayout hLayout) {
|
|
|
|
unsigned int oldColumns = HeaderLayout_getColumns(this->hLayout);
|
|
|
|
unsigned int newColumns = HeaderLayout_getColumns(hLayout);
|
|
|
|
|
|
|
|
if (newColumns > oldColumns) {
|
|
|
|
this->hColumns = xReallocArray(this->hColumns, newColumns, sizeof(MeterColumnSetting));
|
|
|
|
memset(this->hColumns + oldColumns, 0, (newColumns - oldColumns) * sizeof(MeterColumnSetting));
|
|
|
|
} else if (newColumns < oldColumns) {
|
|
|
|
for (unsigned int i = newColumns; i < oldColumns; i++) {
|
|
|
|
if (this->hColumns[i].names) {
|
2021-09-04 12:15:47 +00:00
|
|
|
for (size_t j = 0; j < this->hColumns[i].len; j++)
|
2020-12-25 15:42:35 +00:00
|
|
|
free(this->hColumns[i].names[j]);
|
|
|
|
free(this->hColumns[i].names);
|
|
|
|
}
|
|
|
|
free(this->hColumns[i].modes);
|
|
|
|
}
|
|
|
|
this->hColumns = xReallocArray(this->hColumns, newColumns, sizeof(MeterColumnSetting));
|
|
|
|
}
|
|
|
|
|
|
|
|
this->hLayout = hLayout;
|
|
|
|
this->changed = true;
|
|
|
|
}
|