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
2020-10-05 07:51:32 +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 + + ) {
if ( this - > hColumns [ i ] . names ) {
for ( uint8_t j = 0 ; j < this - > hColumns [ i ] . len ; j + + )
free ( this - > hColumns [ i ] . names [ j ] ) ;
free ( this - > hColumns [ i ] . names ) ;
}
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-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-02-02 09:32:11 +00:00
if ( initialCpuCount > 4 & & initialCpuCount < = 128 ) {
2015-01-22 01:27:31 +00:00
sizes [ 1 ] + + ;
}
for ( int 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
}
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.
char * end , dynamic [ 32 ] = { 0 } ;
if ( sscanf ( ids [ i ] , " Dynamic(%30s) " , dynamic ) ) {
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
fprintf ( stderr , " WARNING: %s specifies configuration format version v%d, but this %s binary supports up to v%d. " , fileName , this - > config_version , PACKAGE , CONFIG_READER_MIN_VERSION ) ;
fprintf ( stderr , " The configuration version 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 ) ;
2018-02-18 23:35:23 +00:00
if ( ! didReadMeters ) {
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 = " " ;
2020-12-25 15:42:35 +00:00
for ( uint8_t i = 0 ; i < this - > hColumns [ column ] . len ; i + + ) {
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 = " " ;
2020-12-25 15:42:35 +00:00
for ( uint8_t i = 0 ; i < this - > hColumns [ column ] . len ; i + + ) {
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_) \
fprintf ( fd , setting_ " =%d%c " , ( int ) value_ , separator ) ;
# define printSettingString(setting_, value_) \
fprintf ( fd , setting_ " =%s%c " , value_ , separator ) ;
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-08-02 15:33:34 +00:00
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 ) {
for ( uint8_t j = 0 ; j < this - > hColumns [ i ] . len ; j + + )
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 ;
}