Provided by: libpcp3-dev_3.10.8build1_amd64 bug

NAME

       pmRegisterDerived - register a derived metric name and definition

C SYNOPSIS

       #include <pcp/pmapi.h>

       char *pmRegisterDerived(char *name, char *expr);

       cc ... -lpcp

DESCRIPTION

       Derived  metrics  provide  a  way of extending the Performance Metrics Name Space (PMNS) with new metrics
       defined at the PCP client-side using arithmetic expressions over the existing performance metrics.

       Typical uses would be to aggregate a number of similar metrics to provide a higher-level  summary  metric
       or  to  support  the  ``delta  V  over  delta V'' class of metrics that are not possible in the base data
       semantics of PCP.  An example of the latter class would be the average I/O size, defined as
                                  delta(disk.dev.total_bytes) / delta(disk.dev.total)
       where both of the disk.dev metrics are counters, and what is required  is  to  to  sample  both  metrics,
       compute  the  difference  between  the  current and previous values and then calculate the ratio of these
       differences.

       The arguments to pmRegisterDerived are the name of the new derived  metric  and  expr  is  an  arithmetic
       expression defining how the values of name should be computed.

       name  should  follow  the  syntactic  rules  for  the  names  of  performance metrics, namely one or more
       components separated with a dot (``.''), and each component must begin with  an  alphabetic  followed  by
       zero  or  more characters drawn from the alphabetics, numerics and underscore (``_'').  For more details,
       refer to PCPIntro(1) and pmns(5).

       name must be unique across all derived metrics and should not match the name of any regular metric in the
       PMNS.  It is acceptable for name to share some part of its prefix with an existing subtree of  the  PMNS,
       e.g.  the  average  I/O  size metric above could be named disk.dev.avgsz which would place it amongst the
       other disk.dev metrics in the PMNS.  Alternatively, derived metrics could populate their own  subtree  of
       the PMNS, e.g. the average I/O size metric above could be named my.summary.disk.avgsz.

       The expression expr follows these syntactic rules:

       * Terminal elements are either names of existing metrics or integer constants.  Recursive definitions are
         not  allowed,  so  only  the  names of regular metrics (not other derived metrics) may be used. Integer
         constants are constrained to the precision of 32-bit unsigned integers.

       * The usual binary arithmetic operators are supported, namely - addition  (``+''),  subtraction  (``-''),
         multiplication  (``*'')  and division (``/'') with the normal precedence rules where multiplication and
         division have higher precedence than addition and subtraction, so a+b*c is evaluated as a+(b*c)

       * Parenthesis may be used for grouping.

       * The following unary functions operate on a single performance metric and return  one  or  more  values.
         For  all  functions  (except  count() and instant()), the type of the operand metric must be arithmetic
         (integer of various sizes and signedness, float or double).
                          ┌────────────┬────────────────────────────────────────────────────┐
                          │  Function  │                       Value                        │
                          ├────────────┼────────────────────────────────────────────────────┤
                          │ avg(x)     │ A singular instance being the average value across │
                          │            │ all instances for the metric x.                    │
                          ├────────────┼────────────────────────────────────────────────────┤
                          │ count(x)   │ A singular instance being the count of the  number │
                          │            │ of instances for the metric x.                     │
                          ├────────────┼────────────────────────────────────────────────────┤
                          │ delta(x)   │ Returns  the difference in values for the metric x │
                          │            │ between one call to pmFetch(3) and the next. There │
                          │            │ is one value in the result for each instance  that │
                          │            │ appears  in  both  the  current  and  the previous │
                          │            │ sample.                                            │
                          ├────────────┼────────────────────────────────────────────────────┤
                          │ rate(x)    │ Returns the difference in values for the metric  x │
                          │            │ between  one  call  to  pmFetch(3)  and  the  next │
                          │            │ divided by the elapsed time between the  calls  to │
                          │            │ pmFetch(3).   The  semantics of the derived metric │
                          │            │ are based on the semantics of the operand (x) with │
                          │            │ the dimension in the time domain decreased by  one │
                          │            │ and  scaling  if  required in the time utilization │
                          │            │ case where the operand is in units  of  time,  and │
                          │            │ the  derived  metric is unitless.  This mimics the │
                          │            │ rate conversion  applied  to  counter  metrics  by │
                          │            │ tools  such  as  pmval(1), pmie(1) and pmchart(1). │
                          │            │ There is one value in the result for each instance │
                          │            │ that appears in both the current and the  previous │
                          │            │ sample.                                            │
                          ├────────────┼────────────────────────────────────────────────────┤
                          │ instant(x) │ Returns the current value of the metric x, even it │
                          │            │ has    the    semantics   of   a   counter,   i.e. │
                          │            │ PM_SEM_COUNTER.   The  semantics  of  the  derived │
                          │            │ metric  are  based on the semantics of the operand │
                          │            │ (x);  if  x  has  semantics  PM_SEM_COUNTER,   the │
                          │            │ semantics   of   instant(x)   is   PM_SEM_INSTANT, │
                          │            │ otherwise the semantics of the derived  metric  is │
                          │            │ the same as the semantics of the metric x.         │
                          ├────────────┼────────────────────────────────────────────────────┤
                          │ max(x)     │ A singular instance being the maximum value across │
                          │            │ all instances for the metric x.                    │
                          ├────────────┼────────────────────────────────────────────────────┤
                          │ min(x)     │ A singular instance being the minimum value across │
                          │            │ all instances for the metric x.                    │
                          ├────────────┼────────────────────────────────────────────────────┤
                          │ sum(x)     │ A  singular  instance  being the sum of the values │
                          │            │ across all instances for the metric x.             │
                          └────────────┴────────────────────────────────────────────────────┘

       * White space is ignored.

       Syntactic checking is performed at the  time  pmRegisterDerived  is  called,  but  semantic  checking  is
       deferred   until   each   new   context   is   created   with   pmNewContext(3)  or  re-established  with
       pmReconnectContext(3), at which time the PMNS and metadata is available to allow  semantic  checking  and
       the metadata of the derived metrics to be established.

SEMANTIC CHECKS AND RULES

       There  are  a number of conversions required to determine the metadata for a derived metric and to ensure
       the semantics of the expressions are sound.

       In a binary expression, if the semantics of both operands  is  not  a  counter  (i.e.  PM_SEM_INSTANT  or
       PM_SEM_DISCRETE)   then   the  result  will  have  semantics  PM_SEM_INSTANT  unless  both  operands  are
       PM_SEM_DISCRETE in which case the result is also PM_SEM_DISCRETE.

       The mapping of the pmUnits of the metadata uses the following rules:

       * If both operands have a dimension of COUNT and the scales are not the same, use the  larger  scale  and
         convert the values of the operand with the smaller scale.

       * If  both  operands  have  a dimension of TIME and the scales are not the same, use the larger scale and
         convert the values of the operand with the smaller scale.

       * If both operands have a dimension of SPACE and the scales are not the same, use the  larger  scale  and
         convert the values of the operand with the smaller scale.

       * For addition and subtraction all dimensions for each of the operands and result are identical.

       * For multiplication, the dimensions of the result are the sum of the dimensions of the operands.

       * For division, the dimensions of the result are the difference of the dimensions of the operands.

       Scale  conversion  involves division if the dimension is positive else multiplication if the dimension is
       negative. If scale conversion is applied to either of the  operands,  the  result  is  promoted  to  type
       PM_TYPE_DOUBLE.

       Putting all of this together in an example, consider the derived metric defined as follows:
             x = network.interface.speed - delta(network.interface.in.bytes) / delta(sample.milliseconds)
       The type, dimension and scale settings would propagate up the expression tree as follows.
                    ┌─────────────────────────┬────────┬───────────────────┬────────────────────┐
                    │       Expression        │  Type  │ Dimension & Scale │  Scale Factor(s)   │
                    ├─────────────────────────┼────────┼───────────────────┼────────────────────┤
                    │ sample.milliseconds     │ DOUBLE │ millisec          │                    │
                    │ delta(...)              │ DOUBLE │ millisec          │                    │
                    │ network...bytes         │ U64    │ byte              │                    │
                    │ delta(...)              │ U64    │ byte              │                    │
                    │ delta(...) / delta(...) │ DOUBLE │ byte/millisec     │ /1048576 and *1000 │
                    │ network...speed         │ FLOAT  │ Mbyte/sec         │                    │
                    │ x                       │ DOUBLE │ Mbyte/sec         │                    │
                    └─────────────────────────┴────────┴───────────────────┴────────────────────┘

       Because semantic checking cannot be done at the time pmRegisterDerived is called, errors found during
       semantic checking are reported using pmprintf(3).  These include:

       Error: derived metric <name1>: operand: <name2>: <reason>
              There  was  a  problem  calling pmLookupName(3) to identify the operand metric <name2> used in the
              definition of the derived metric <name1>.

       Error: derived metric <name1>: operand (<name2> [<pmid2>]): <reason>
              There was a problem calling pmLookupDesc(3) to identify  the  operand  metric  <name2>  with  PMID
              <pmid2> used in the definition of the derived metric <name1>.

       Semantic error: derived metric <name>: <operand> <op> <operand>: Illegal operator for counters
              If  both  operands  have  the  semantics  of  counter, only addition or subtraction make sense, so
              multiplication and division are not allowed.

       Semantic error: derived metric <name>: <operand> <op> <operand>: Illegal operator for counter and non-
       counter
              Only multiplication or division are allowed if the left operand has the semantics of a counter and
              the right operand is not a counter.

       Semantic error: derived metric <name>: <operand> <op> <operand>: Illegal operator for non-counter and
       counter
              Only multiplication is allowed if the right operand has the semantics of a counter  and  the  left
              operand is not a counter.

       Semantic error: derived metric <name>: <operand> <op> <operand>: Non-arithmetic type for <left-or-right>
       operand
              The binary arithmetic operators are only allowed with operands with an arithmetic type (integer of
              various sizes and signedness, float or double).

       Semantic error: derived metric <name>: <function>(<operand>): Non-arithmetic operand for function
              The unary functions are only defined if the operand has arithmetic type.

       Semantic error: derived metric <name>: Incorrect time dimension for operand
              Rate  conversion  using  the  rate()  function  is  only  possible for operand metrics with a Time
              dimension of 0 or 1 (see pmLookupDesc(3)).  If the operand metric's Time dimension is 0, then  the
              derived  metrics  has  a  value "per second" (Time dimension of -1).  If the operand metric's Time
              dimension is 1, then the derived metrics has a value of time utilization (Time dimension of 0).

EXPRESSION EVALUATION

       For the binary arithmetic operators, if either operand must be scaled (e.g. convert bytes to Kbytes) then
       the result is promoted to PM_TYPE_DOUBLE.  Otherwise the type of the result is determined by the types of
       the operands, as per the following table which is evaluated from top to bottom until a match is found.
                             ┌──────────────────────────────┬──────────┬────────────────┐
                             │        Operand Types         │ Operator │  Result Type   │
                             ├──────────────────────────────┼──────────┼────────────────┤
                             │ either is PM_TYPE_DOUBLE     │ any      │ PM_TYPE_DOUBLE │
                             ├──────────────────────────────┼──────────┼────────────────┤
                             │ any                          │ division │ PM_TYPE_DOUBLE │
                             ├──────────────────────────────┼──────────┼────────────────┤
                             │ either is PM_TYPE_FLOAT      │ any      │ PM_TYPE_FLOAT  │
                             ├──────────────────────────────┼──────────┼────────────────┤
                             │ either is PM_TYPE_U64        │ any      │ PM_TYPE_U64    │
                             ├──────────────────────────────┼──────────┼────────────────┤
                             │ either is PM_TYPE_64         │ any      │ PM_TYPE_64     │
                             ├──────────────────────────────┼──────────┼────────────────┤
                             │ either is PM_TYPE_U32        │ any      │ PM_TYPE_U32    │
                             ├──────────────────────────────┼──────────┼────────────────┤
                             │ otherwise     (both      are │ any      │ PM_TYPE_32     │
                             │ PM_TYPE_32)                  │          │                │
                             └──────────────────────────────┴──────────┴────────────────┘

CAVEATS

       Unary  negation  is  not supported, so the following expressions would be syntactically incorrect, -3*abc
       and -this.number

       Derived metrics are not available when using pmFetchArchive(3) as this routine does not use a target list
       of PMIDs that could be remapped (as is done for pmFetch(3)).

       pmRegisterDerived does not apply retrospectively to any open contexts, so the normal use would be to make
       all calls to pmRegisterDerived (possibly via pmLoadDerivedConfig(3)) and then call pmNewContext(3).

       There is no pmUnregisterDerived method, so once registered a derived metric persists for the life of  the
       application.

DIAGNOSTICS

       On success, pmRegisterDerived returns NULL.

       If  a  syntactic error is found at the time of registration, the value returned by pmRegisterDerived is a
       pointer into expr indicating where the error was found.  To identify what the error was, the  application
       should call pmDerivedErrStr(3) to retrieve the corresponding parser error message.

SEE ALSO

       PCPIntro(1),   PMAPI(3),  pmDerivedErrStr(3),  pmFetch(3),  pmLoadDerivedConfig(3),  pmNewContext(3)  and
       pmReconnectContext(3).

Performance Co-Pilot                                                                        PMREGISTERDERIVED(3)