The PRocess MONitor is a small stand alone program that can monitor the resource consumption of a process and its children. This is useful in the context of the WLCG/HSF working group to evaluate the costs and performance of HEP workflows in WLCG. In a previous incarnation (MemoryMonitor) it has been used by ATLAS for some time to gather data on resource consumption by production jobs. One of its most useful features is to use smaps to correctly calculate the Proportional Set Size in the group of processes monitored, which is a much better indication of the true memory consumption of a group of processes where children share many pages.
prmon currently runs on Linux machines as it requires access to the
/proc
interface to process statistics.
As prmon has dependencies on submodules, clone the project as
git clone --recurse-submodules https://github.com/HSF/prmon.git
Building prmon requires a C++ compiler that fully supports C++11, and CMake version 3.3 or higher. It also has dependencies on:
- Niels Lohmann JSON libraries
nlohmann-json-dev
in Ubuntu 18,nlohmann-json3-dev
in Ubuntu 20
- spdlog: Fast C++ logging library
libspdlog-dev
in Ubuntu
and can use either external system-supplied versions or internal copies provided by submodules.
Building is usually as simple as:
mkdir build
cd build
cmake -DCMAKE_INSTALL_PREFIX=<installdir> -S .. -B .
make -j<number of cores on your machine>
make install
Unless otherwise specified, the default behavior for dependencies is to first try to find an external version and fall back to the internal submodule copy if not found. To explicitly force the use of either add any of the following configure options:
-DUSE_EXTERNAL_NLOHMANN_JSON={TRUE,FALSE,
AUTO
}
ON
,TRUE
: Force an external version and fail if not found.OFF
,FALSE
: Require the internal copy be used.AUTO
: Search for an external version and fall back to the internal copy if not found.
-Dnlohmann_json_DIR=/path/to/config
- The path to the directory containing
nlohmann_jsonConfig.cmake
. Necessary if nlohmann_json is not installed into CMake's search path.
- The path to the directory containing
-DUSE_EXTERNAL_SPDLOG={TRUE,FALSE,
AUTO
}
ON
,TRUE
: Force an external version and fail if not found.OFF
,FALSE
: Require the internal copy be used.AUTO
: Search for an external version and fall back to the internal copy if not found.
-Dspdlog_DIR=/path/to/config
- The path to the directory containing
spdlogConfig.cmake
. Necessary if spdlog is not installed into CMake's search path.
- The path to the directory containing
The option -DCMAKE_BUILD_TYPE
can switch between all of the standard
build types. The default is Release
; use RelWithDebInfo
if you want
debug symbols.
To build a statically linked version of prmon
, set the BUILD_STATIC
CMake variable to ON
(e.g., adding -DBUILD_STATIC=ON
to the
command line).
Note that in a build environment with CVMFS available the C++ compiler and CMake can be taken by setting up a recent LCG release.
To enable pulling and building the gtest framework as well as tests dependent on gtest,
build with -DBUILD_GTESTS=ON
.
A cpack based package can be created by invoking
make package
To run the tests of the project, first build it and then invoke
make test
Running the tests requires Python version 3.6 or higher.
The prmon
binary is invoked with the following arguments:
prmon [--pid PPP] [--filename prmon.txt] [--json-summary prmon.json] \
[--log-filename prmon.log] [--interval 30] \
[--suppress-hw-info] [--units] [--netdev DEV] \
[--disable MON1] [--level LEV] [--level MON:LEV] \
[--fast-memmon] \
[-- prog arg arg ...]
--pid
the 'mother' PID to monitor (all children in the same process tree are monitored as well)--filename
output file for time-stamped monitored values--json-summmary
output file for summary data written in JSON format--log-filename
output file for log messages--interval
time, in seconds, between monitoring snapshots--suppress-hw-info
flag that turns-off hardware information collection--units
add information on units for each metric to JSON file--netdev
restricts network statistics to one (or more) network devices--disable
is used to disable specific monitors (and can be specified multiple times); the default is thatprmon
monitors everything that it can- Note that the
wallmon
monitor is the only monitor that cannot be disabled
- Note that the
--level
is used to set the logging level for monitors--level LEV
sets the level for all monitors to LEV--level MON:LEV
sets the level for monitor MON to LEV- The valid levels are
trace
,debug
,info
,warn
,error
,critical
--fast-memmon
toggles on fast memory monitoring usingsmaps_rollup
--
after this argument the following arguments are treated as a program to invoke and remaining arguments are passed to it;prmon
will then monitor this process instead of being given a PID via--pid
prmon
will exit with 1
if there is a problem with inconsistent or
incomplete arguments. If prmon
starts a program itself (using --
) then
prmon
will exit with the exit code of the child process.
When invoked with -h
or --help
usage information is printed, as well as a
list of all available monitoring components.
When invoked with --fast-memmon
prmon
uses the smaps_rollup
files
that contain pre-summed memory information for each monitored process.
This is a faster approach compared to the default behavior,
where prmon
aggregates the results itself by going over each of the monitored
processes' mappings one by one.
If the current kernel doesn't support smaps_rollup
then the default
approach is used. Users should also note that fast memory monitoring
might not contain all metrics that the default approach supports, e.g.,
vmem
. In that case, the missing metric will be omitted in the output.
If any of these issues are encountered, a relevant message is printed
to notify the user.
The PRMON_DISABLE_MONITOR
environment variable can be used to specify a comma
separated list of monitor names that will be disabled. This is useful when
prmon
is being invoked by some other part of a job or workflow, so the user
does not have direct access to the command line options used. e.g.
export PRMON_DISABLE_MONITOR=nvidiamon
other_code_that_invokes_prmon
...
Disables the nvidiamon
monitor.
In the filename
output file, plain text with statistics written every
interval
seconds are written. The first line gives the column names.
In the json-summary
file values for the maximum and average statistics
are given in JSON format. This file is rewritten every interval
seconds
with the current summary values. Use the --units
option to see exactly
which units are used for each metric (the value of 1
for a unit means
it is a pure number).
In the log-filename
output file, log messages (e.g., errors, warnings etc.)
are written.
Monitoring of CPU, I/O and memory is reliably accurate, at least to within the sampling time. Monitoring of network I/O is not reliable unless the monitored process is isolated from other processes performing network I/O (it gives an upper bound on the network activity, but the monitoring is per network device as Linux does not give per-process network data by default).
The prmon_plot.py
script (Python3) can be used to plot the outputs of prmon from the
timestamped output file (usually prmon.txt
). Some examples include:
- Memory usage as a function of wall-time:
prmon_plot.py --input prmon.txt --xvar wtime --yvar vmem,pss,rss,swap --yunit GB
- Rate of change in memory usage as a function of wall-time:
prmon_plot.py --input prmon.txt --xvar wtime --yvar vmem,pss,rss,swap --diff --yunit MB
- Rate of change in CPU usage as a function of wall-time with stacked user and system utilizations:
prmon_plot.py --input prmon.txt --xvar wtime --yvar utime,stime --yunit SEC --diff --stacked
The plots above, as well as the input prmon.txt
file that is used
to produce them, can be found under the example-plots
folder.
The script allows the user to specify variables, their units, plotting
style (stacked vs overlaid), as well as the format of the output image.
Use -h
for more information.
The prmon_compress_output.py
script (Python3) can be used to compress the output file
while keeping the most relevant information.
The compression algorithm works as follows:
- For the number of processes, threads, and GPUs, only the measurements that are different with respect to the previous ones are kept.
- For all other metrics, only the measurements that satisfy an interpolation condition are kept.
This latter condition can be summarized as:
- For any three neighboring (and time-ordered) measurements, A, B, and C, B is deleted if the linear interpolation between A and C is consistent with B Β± threshold. Otherwise, it's retained. The threshold can be configured via the
--precision
parameter (default: 0.05, i.e. 5%)
The time index of the final output will be the union of the algorithm outputs of the single
time series. Each series will have NA values where a point was deleted at a kept index and, unless otherwise
specified by the --skip-interpolate
parameter, will be linearly interpolated to maintain a consistent number of data points
and the result will be rounded to the nearest integer for consistency with the original input.
If the --skip-interpolate
parameter is passed, deleted values will be written as empty strings in the output file, and will be interpreted
as NA
values when imported into Pandas.
Example:
prmon_compress_output.py --input prmon.txt --precision 0.3 --skip-interpolate
We're very happy to get feedback on prmon as well as suggestions for future development. Please have a look at our Contribution Guide.
To build prmon with profiling, set one of the CMake variables
PROFILE_GPROF
or PROFILE_GPERFTOOLS
to ON
. This enables
GNU prof profiling or gperftools profiling, respectively.
If your gperftools are in a non-standard place, pass a hint
to CMake using Gperftools_ROOT_DIR
.
Copyright (c) 2018-2025 CERN.
Thanks goes to these wonderful people (emoji key):
This project follows the all-contributors specification. Contributions of any kind welcome!