Skip to contents

Create a dynamic ODE-based model object suitably for translation into fast C code

Usage

rxode2(
  model,
  modName = basename(wd),
  wd = getwd(),
  filename = NULL,
  extraC = NULL,
  debug = FALSE,
  calcJac = NULL,
  calcSens = NULL,
  collapseModel = FALSE,
  package = NULL,
  ...,
  linCmtSens = c("linCmtA", "linCmtB", "linCmtC"),
  indLin = FALSE,
  verbose = FALSE,
  fullPrint = getOption("rxode2.fullPrint", FALSE),
  envir = parent.frame()
)

RxODE(
  model,
  modName = basename(wd),
  wd = getwd(),
  filename = NULL,
  extraC = NULL,
  debug = FALSE,
  calcJac = NULL,
  calcSens = NULL,
  collapseModel = FALSE,
  package = NULL,
  ...,
  linCmtSens = c("linCmtA", "linCmtB", "linCmtC"),
  indLin = FALSE,
  verbose = FALSE,
  fullPrint = getOption("rxode2.fullPrint", FALSE),
  envir = parent.frame()
)

rxode(
  model,
  modName = basename(wd),
  wd = getwd(),
  filename = NULL,
  extraC = NULL,
  debug = FALSE,
  calcJac = NULL,
  calcSens = NULL,
  collapseModel = FALSE,
  package = NULL,
  ...,
  linCmtSens = c("linCmtA", "linCmtB", "linCmtC"),
  indLin = FALSE,
  verbose = FALSE,
  fullPrint = getOption("rxode2.fullPrint", FALSE),
  envir = parent.frame()
)

Arguments

model

This is the ODE model specification. It can be:

  • a string containing the set of ordinary differential equations (ODE) and other expressions defining the changes in the dynamic system.

  • a file name where the ODE system equation is contained

An ODE expression enclosed in \{\}

(see also the filename argument). For details, see the sections “Details” and rxode2 Syntax below.

modName

a string to be used as the model name. This string is used for naming various aspects of the computations, including generating C symbol names, dynamic libraries, etc. Therefore, it is necessary that modName consists of simple ASCII alphanumeric characters starting with a letter.

wd

character string with a working directory where to create a subdirectory according to modName. When specified, a subdirectory named after the “modName.d” will be created and populated with a C file, a dynamic loading library, plus various other working files. If missing, the files are created (and removed) in the temporary directory, and the rxode2 DLL for the model is created in the current directory named rx_????_platform, for example rx_129f8f97fb94a87ca49ca8dafe691e1e_i386.dll

filename

A file name or connection object where the ODE-based model specification resides. Only one of model or filename may be specified.

extraC

Extra c code to include in the model. This can be useful to specify functions in the model. These C functions should usually take double precision arguments, and return double precision values.

debug

is a boolean indicating if the executable should be compiled with verbose debugging information turned on.

calcJac

boolean indicating if rxode2 will calculate the Jacobain according to the specified ODEs.

calcSens

boolean indicating if rxode2 will calculate the sensitivities according to the specified ODEs.

collapseModel

boolean indicating if rxode2 will remove all LHS variables when calculating sensitivities.

package

Package name for pre-compiled binaries.

...

ignored arguments.

linCmtSens

The method to calculate the linCmt() solutions

indLin

Calculate inductive linearization matrices and compile with inductive linearization support.

verbose

When TRUE be verbose with the linear compartmental model

fullPrint

When using printf within the model, if TRUE print on every step (except ME/indLin), otherwise when FALSE print only when calculating the d/dt

envir

is the environment to look for R user functions (defaults to parent environment)

Value

An object (environment) of class rxode2 (see Chambers and Temple Lang (2001)) consisting of the following list of strings and functions:

* `model` a character string holding the source model specification.
* `get.modelVars`a function that returns a list with 3 character
    vectors, `params`, `state`, and `lhs` of variable names used in the model
    specification. These will be output when the model is computed (i.e., the ODE solved by integration).

  * `solve`{this function solves (integrates) the ODE. This
      is done by passing the code to [rxSolve()].
      This is as if you called `rxSolve(rxode2object, ...)`,
      but returns a matrix instead of a rxSolve object.

      `params`: a numeric named vector with values for every parameter
      in the ODE system; the names must correspond to the parameter
      identifiers used in the ODE specification;

      `events`: an `eventTable` object describing the
      input (e.g., doses) to the dynamic system and observation
      sampling time points (see  [eventTable()]);

      `inits`: a vector of initial values of the state variables
      (e.g., amounts in each compartment), and the order in this vector
      must be the same as the state variables (e.g., PK/PD compartments);


      `stiff`: a logical (`TRUE` by default) indicating whether
      the ODE system is stiff or not.

      For stiff ODE systems (`stiff = TRUE`), `rxode2` uses
      the LSODA (Livermore Solver for Ordinary Differential Equations)
      Fortran package, which implements an automatic method switching
      for stiff and non-stiff problems along the integration interval,
      authored by Hindmarsh and Petzold (2003).

      For non-stiff systems (`stiff = FALSE`), `rxode2` uses `DOP853`,
      an explicit Runge-Kutta method of order 8(5, 3) of Dormand and Prince
      as implemented in C by Hairer and Wanner (1993).

      `trans_abs`: a logical (`FALSE` by default) indicating
      whether to fit a transit absorption term
      (TODO: need further documentation and example);

      `atol`: a numeric absolute tolerance (1e-08 by default);

      `rtol`: a numeric relative tolerance (1e-06 by default).

      The output of \dQuote{solve} is a matrix with as many rows as there
      are sampled time points and as many columns as system variables
      (as defined by the ODEs and additional assignments in the rxode2 model
          code).}

  * `isValid` a function that (naively) checks for model validity,
      namely that the C object code reflects the latest model
      specification.
  * `version` a string with the version of the `rxode2`
      object (not the package).
  * `dynLoad` a function with one `force = FALSE` argument
      that dynamically loads the object code if needed.
  * `dynUnload` a function with no argument that unloads
      the model object code.
  * `delete` removes all created model files, including C and DLL files.
      The model object is no longer valid and should be removed, e.g.,
      `rm(m1)`.
  * `run` deprecated, use `solve`.
  * `get.index` deprecated.
  * `getObj` internal (not user callable) function.

Details

The Rx in the name rxode2 is meant to suggest the abbreviation Rx for a medical prescription, and thus to suggest the package emphasis on pharmacometrics modeling, including pharmacokinetics (PK), pharmacodynamics (PD), disease progression, drug-disease modeling, etc.

The ODE-based model specification may be coded inside four places:

  • Inside a rxode2({}) block statements:

library(rxode2)
mod <- rxode2({
  # simple assignment
  C2 <- centr/V2

  # time-derivative assignment
  d/dt(centr) <- F*KA*depot - CL*C2 - Q*C2 + Q*C3;
})

## using C compiler: ‘gcc (Ubuntu 11.4.0-1ubuntu1~22.04) 11.4.0’
## In file included from /usr/share/R/include/R.h:71,
##                  from /home/matt/R/x86_64-pc-linux-gnu-library/4.3/rxode2parse/include/rxode2parse.h:33,
##                  from /home/matt/src/rxode2/inst/include/rxode2.h:9,
##                  from /home/matt/R/x86_64-pc-linux-gnu-library/4.3/rxode2parse/include/rxode2_model_shared.h:3,
##                  from rx_36fd6c4312c660ddf5390102e91a4bb3_.c:117:
## /usr/share/R/include/R_ext/Complex.h:80:6: warning: ISO C99 doesn’t support unnamed structs/unions [-Wpedantic]
##    80 |     };
##       |      ^

  • Inside a rxode2("") string statement:

mod <- rxode2("
  # simple assignment
  C2 <- centr/V2

  # time-derivative assignment
  d/dt(centr) <- F*KA*depot - CL*C2 - Q*C2 + Q*C3;
")

## using C compiler: ‘gcc (Ubuntu 11.4.0-1ubuntu1~22.04) 11.4.0’
## In file included from /usr/share/R/include/R.h:71,
##                  from /home/matt/R/x86_64-pc-linux-gnu-library/4.3/rxode2parse/include/rxode2parse.h:33,
##                  from /home/matt/src/rxode2/inst/include/rxode2.h:9,
##                  from /home/matt/R/x86_64-pc-linux-gnu-library/4.3/rxode2parse/include/rxode2_model_shared.h:3,
##                  from rx_4fdd80e5dd5fd33d1951b83e06f7ad84_.c:117:
## /usr/share/R/include/R_ext/Complex.h:80:6: warning: ISO C99 doesn’t support unnamed structs/unions [-Wpedantic]
##    80 |     };
##       |      ^

  • In a file name to be loaded by rxode2:

writeLines("
  # simple assignment
  C2 <- centr/V2

  # time-derivative assignment
  d/dt(centr) <- F*KA*depot - CL*C2 - Q*C2 + Q*C3;
", "modelFile.rxode2")
mod <- rxode2(filename='modelFile.rxode2')
unlink("modelFile.rxode2")

  • In a model function which can be parsed by rxode2:

mod <- function() {
  model({
    # simple assignment
    C2 <- centr/V2

    # time-derivative assignment
    d/dt(centr) <- F*KA*depot - CL*C2 - Q*C2 + Q*C3;
  })
}

mod <- rxode2(mod) # or simply mod() if the model is at the end of the function

# These model functions often have residual components and initial
# (`ini({})`) conditions attached as well. For example the
# theophylline model can be written as:

one.compartment <- function() {
  ini({
    tka <- 0.45 # Log Ka
    tcl <- 1 # Log Cl
    tv <- 3.45    # Log V
    eta.ka ~ 0.6
    eta.cl ~ 0.3
    eta.v ~ 0.1
    add.sd <- 0.7
  })
  model({
    ka <- exp(tka + eta.ka)
    cl <- exp(tcl + eta.cl)
    v <- exp(tv + eta.v)
    d/dt(depot) = -ka * depot
    d/dt(center) = ka * depot - cl / v * center
    cp = center / v
    cp ~ add(add.sd)
  })
}

# after parsing the model
mod <- one.compartment()

For the block statement, character string or text file an internal rxode2 compilation manager translates the ODE system into C, compiles it and loads it into the R session. The call to rxode2 produces an object of class rxode2 which consists of a list-like structure (environment) with various member functions.

For the last type of model (a model function), a call to rxode2 creates a parsed rxode2 ui that can be translated to the rxode2 compilation model.

mod$simulationModel

## using C compiler: ‘gcc (Ubuntu 11.4.0-1ubuntu1~22.04) 11.4.0’
## In file included from /usr/share/R/include/R.h:71,
##                  from /home/matt/R/x86_64-pc-linux-gnu-library/4.3/rxode2parse/include/rxode2parse.h:33,
##                  from /home/matt/src/rxode2/inst/include/rxode2.h:9,
##                  from /home/matt/R/x86_64-pc-linux-gnu-library/4.3/rxode2parse/include/rxode2_model_shared.h:3,
##                  from rx_bbee652505e8bddf28c7e688baef12d9_.c:117:
## /usr/share/R/include/R_ext/Complex.h:80:6: warning: ISO C99 doesn’t support unnamed structs/unions [-Wpedantic]
##    80 |     };
##       |      ^

## rxode2 2.1.2.9000 model named rx_bbee652505e8bddf28c7e688baef12d9 model (ready). 
## x$state: depot, center
## x$stateExtra: cp
## x$params: tka, tcl, tv, add.sd, eta.ka, eta.cl, eta.v, rxerr.cp
## x$lhs: ka, cl, v, cp, ipredSim, sim

# or 
mod$simulationIniModel

## using C compiler: ‘gcc (Ubuntu 11.4.0-1ubuntu1~22.04) 11.4.0’
## In file included from /usr/share/R/include/R.h:71,
##                  from /home/matt/R/x86_64-pc-linux-gnu-library/4.3/rxode2parse/include/rxode2parse.h:33,
##                  from /home/matt/src/rxode2/inst/include/rxode2.h:9,
##                  from /home/matt/R/x86_64-pc-linux-gnu-library/4.3/rxode2parse/include/rxode2_model_shared.h:3,
##                  from rx_fe01e26500a95dd1ba4d8972bc71bc7a_.c:117:
## /usr/share/R/include/R_ext/Complex.h:80:6: warning: ISO C99 doesn’t support unnamed structs/unions [-Wpedantic]
##    80 |     };
##       |      ^

## rxode2 2.1.2.9000 model named rx_fe01e26500a95dd1ba4d8972bc71bc7a model (ready). 
## x$state: depot, center
## x$stateExtra: cp
## x$params: tka, tcl, tv, add.sd, eta.ka, eta.cl, eta.v, rxerr.cp
## x$lhs: ka, cl, v, cp, ipredSim, sim

This is the same type of function required for nlmixr2 estimation and can be extended and modified by model piping. For this reason will be focused on in the documentation.

This basic model specification consists of one or more statements optionally terminated by semi-colons ; and optional comments (comments are delimited by # and an end-of-line).

A block of statements is a set of statements delimited by curly braces, { ... }.

Statements can be either assignments, conditional if/else if/else, while loops (can be exited by break), special statements, or printing statements (for debugging/testing).

Assignment statements can be:

  • simple assignments, where the left hand is an identifier (i.e., variable)

  • special time-derivative assignments, where the left hand specifies the change of the amount in the corresponding state variable (compartment) with respect to time e.g., d/dt(depot):

  • special initial-condition assignments where the left hand specifies the compartment of the initial condition being specified, e.g. depot(0) = 0

  • special model event changes including bioavailability (f(depot)=1), lag time (alag(depot)=0), modeled rate (rate(depot)=2) and modeled duration (dur(depot)=2). An example of these model features and the event specification for the modeled infusions the rxode2 data specification is found in rxode2 events vignette.

  • special change point syntax, or model times. These model times are specified by mtime(var)=time

  • special Jacobian-derivative assignments, where the left hand specifies the change in the compartment ode with respect to a variable. For example, if d/dt(y) = dy, then a Jacobian for this compartment can be specified as df(y)/dy(dy) = 1. There may be some advantage to obtaining the solution or specifying the Jacobian for very stiff ODE systems. However, for the few stiff systems we tried with LSODA, this actually slightly slowed down the solving.

Note that assignment can be done by =, <- or ~.

When assigning with the ~ operator, the simple assignments and time-derivative assignments will not be output. Note that with the rxode2 model functions assignment with ~ can also be overloaded with a residual distribution specification.

Special statements can be:

  • Compartment declaration statements, which can change the default dosing compartment and the assumed compartment number(s) as well as add extra compartment names at the end (useful for multiple-endpoint nlmixr models); These are specified by cmt(compartmentName)

  • Parameter declaration statements, which can make sure the input parameters are in a certain order instead of ordering the parameters by the order they are parsed. This is useful for keeping the parameter order the same when using 2 different ODE models. These are specified by param(par1, par2,...)

An example model is shown below:

   # simple assignment
   C2 <- centr/V2

   # time-derivative assignment
   d/dt(centr) <- F*KA*depot - CL*C2 - Q*C2 + Q*C3;

Expressions in assignment and if statements can be numeric or logical.

Numeric expressions can include the following numeric operators +, -, *, /, ^ and those mathematical functions defined in the C or the R math libraries (e.g., fabs, exp, log, sin, abs).

You may also access the R’s functions in the R math libraries, like lgammafn for the log gamma function.

The rxode2 syntax is case-sensitive, i.e., ABC is different than abc, Abc, ABc, etc.

Identifiers

Like R, Identifiers (variable names) may consist of one or more alphanumeric, underscore _ or period . characters, but the first character cannot be a digit or underscore _.

Identifiers in a model specification can refer to:

  • State variables in the dynamic system (e.g., compartments in a pharmacokinetics model).

  • Implied input variable, t (time), tlast (last time point), and podo (oral dose, in the undocumented case of absorption transit models).

  • Special constants like pi or R’s predefined constants.

  • Model parameters (e.g., ka rate of absorption, CL clearance, etc.)

  • Others, as created by assignments as part of the model specification; these are referred as LHS (left-hand side) variable.

Currently, the rxode2 modeling language only recognizes system state variables and “parameters”, thus, any values that need to be passed from R to the ODE model (e.g., age) should be either passed in the params argument of the integrator function rxSolve() or be in the supplied event data-set.

There are certain variable names that are in the rxode2 event tables. To avoid confusion, the following event table-related items cannot be assigned, or used as a state but can be accessed in the rxode2 code:

  • cmt

  • dvid

  • addl

  • ss

  • rate

  • id

However the following variables are cannot be used in a model specification:

  • evid

  • ii

Sometimes rxode2 generates variables that are fed back to rxode2. Similarly, nlmixr2 generates some variables that are used in nlmixr estimation and simulation. These variables start with the either the rx or nlmixr prefixes. To avoid any problems, it is suggested to not use these variables starting with either the rx or nlmixr prefixes.

Logical Operators

Logical operators support the standard R operators ==, != >= <= > and <. Like R these can be in if() or while() statements, ifelse() expressions. Additionally they can be in a standard assignment. For instance, the following is valid:

cov1 = covm*(sexf == "female") + covm*(sexf != "female")

Notice that you can also use character expressions in comparisons. This convenience comes at a cost since character comparisons are slower than numeric expressions. Unlike R, as.numeric or as.integer for these logical statements is not only not needed, but will cause an syntax error if you try to use the function.

Supported functions

All the supported functions in rxode2 can be seen with the rxSupportedFuns().

A brief description of the built-in functions are in the following table:

FunctionDescriptionAliases
gamma(x)The Gamma functiongammafn
lgamma(x)Natural logarithm of absolute value of gamma functiondigamma
digamma(x)First derivative of lgamma
trigamma(x)Second derivative of lgamma
tetragamma(x)Third derivative of lgamma
pentagamma(x)Fourth derivative of lgamma
psigamma(x, deriv)n-th derivative of Psi, the digamma function, which is the derivative of lgammafn. In other words, digamma(x) is the same as psigamma(x,0), trigamma(x) == psigamma(x,1), etc.
cospi(x)cos(pi*x)
sinpi(x)sin(pi*x)
tanpi(x)tan(pi*x)
beta(a, b)Beta function
lbeta(a, b)log Beta function
bessel_i(x, nu, expo)Bessel function type I with index nuexpo==1 is unscaled expo==2 is scaled by exp(-x)
bessel_j(x, nu)Bessel function type J with index nu
bessel_k(x, ku, expo)Bessel function type K with index nuexpo==1 is unscaled expo==2 is scaled by exp(x)
bessel_y(x, nu)Bessel function type Y with index nu
R_pow(x, y)x^y
R_pow_di(x, I)x^yy is an integer
log1pmxlog(1+x) - x
log1pexplog(1+exp(x))
expm1(x)exp(x)-1
lgamma1p(x)log(gamma(x+1))
sign(x)Compute the signum function where sign(x) is 1, 0 -1
fsign(x, y)abs(x)*sign(y)
fprec(x, digits)x rounded to digits (after the decimal point, used by signif()
fround(x, digits)Round, used by R’s round()
ftrunc(x)Truncated towards zero
abs(x)absolute value of xfabs
sin(x)sine of x
cos(x)cos of x
tan(x)tan of x
factorial(x)factorial of x
lfactorial(x)log(factorial(x))
log10(x)log base 10
log2(x)log base 2
pnorm(x)Normal CDF of xnormcdf, phi
qnorm(x)Normal pdf of xnorminv
probit(x, low=0, hi=1)Probit (normal pdf) of x transforming into a range
probitInv(q, low=0, hi=1)Inverse probit of x transforming into a range
acos(x)Inverse cosine
asin(x)Inverse sine
atan(x)Inverse tangent
atan2(a, b)Four quadrant inverse tangent
sinh(x)Hyperbolic sine
cosh(x)Hyperbolic cosine
tanh(x)Hyperbolic tangent
floor(x)Downward rounding
ceil(x)Upward rounding
logit(x, low=0, hi=1)Logit transformation of x transforming into a range
expit(x, low=0, hi=1)expit transofmration in rangeinvLogit, logitInv
gammaq(a, z)Normalized incomplete gamma from boost
gammaqInv(a, q)Normalized incomplete gamma inverse from boost
ifelse(cond, trueValue, falseValue)if else function
gammap(a, z)Normalized lower incomplete gamma from boost
gammapInv(a, p)Inverse of Normalized lower incomplete gamma from boost
gammapInva(x, p)Inverse of Normalized lower incomplete gamma from boost
rxnorm(x)Generate one deviate of from a normal distribution for each observation scale
rxnormV(x)Generate one deviate from low discrepancy normal for each observation
rxcauchyGenerate one deviate from the cauchy distribution for each observation
rxchisqGenerate one deviate from the chisq distribution for each observation
rxexpGenerate one deviate from the exponential distribution for each observation
rxfGenerate one deviate from low discrepancy normal for each observation
rxgammaGenerate one deviate from the gamma distribution for each observation
rxbetaGenerate one deviate from the beta distribution for each observation
rxgeomGenerate one deviate from the geometric distribution for each observation
rxpoisGenerate one deviate from the poission distribution for each observation
rxtGenerate one deviate from the t distribtuion for each observation
tad() or tad(x)Time after dose (tad()) or time after dose for a compartment tad(cmt)
tafd() or tafd(x)Time after first dose (tafd()) or time after first dose for a compartment tafd(cmt)
dosenum()Dose Number
tlast() or tlast(cmt)Time of Last dose; This takes into consideration any lag time, so if there is a dose at time 3 and a lag of 1, the time of last dose would be 4. tlast(cmt) calculates the time since last dose of a compartment
tfirst() or tfirst(cmt)Time since first dose or time since first dose of a compartment
prod(…)product of terms; This uses PreciseSums so the product will not have as much floating point errors (though it will take longer)
sum(…)sum of terms; This uses PreciseSums so the product will not have as much floating point errors (though it will take longer)
max(…)maximum of a group of numbers
min(…)Min of a group of numbers
lag(parameter, number=1)Get the lag of an input parameter; You can specify a number of lagged observations
lead(parameter, number=2)Get the lead of an input parameter; You can specify a number of lead observation
diff(par, number=1)Get the difference between the current parameter and the last parameter; Can change the parameter number
first(par)Get the first value of an input parameter
last(par)Get the last value of an input parameter
transit()The transit compartment psuedo function
is.na()Determine if a value is NA
is.nan()Determine if a value is NaN
is.infinite()Check to see if the value is infinite
rinorm(x)Generate one deviate of from a normal distribution for each individual
rinormV(x)Generate one deviate from low discrepancy normal for each individual
ricauchyGenerate one deviate from the cauchy distribution for each individual
richisqGenerate one deviate from the chisq distribution for each individual
riexpGenerate one deviate from the exponential distribution for each individual
rifGenerate one deviate from low discrepancy normal for each individual
rigammaGenerate one deviate from the gamma distribution for each individual
ribetaGenerate one deviate from the beta distribution for each individual
rigeomGenerate one deviate from the geometric distribution for each individual
ropoisGenerate one deviate from the poission distribution for each individual
ritGenerate one deviate from the t distribtuion for each individual
simepsSimulate EPS from possibly truncated sigma matrix. Will take sigma matrix from the current study. Simulated at the very last moment.
simetaSimulate ETA from possibly truncated omega matrix. Will take the omega matrix from the current study. Simulated at the initilization of the ODE system or the intialization of lhs

Note that lag(cmt) = is equivalent to alag(cmt) = and not the same as = lag(wt)

Reserved keywords

There are a few reserved keywords in a rxode2 model. They are in the following table:

Reserved NameMeaningAlias
timesolver timet
podoIn Transit compartment models, last dose amount
tlastTime of Last dose
M_EExp(1)
M_LOG2Elog2(e)
M_LOG10Elog10(e)
M_LN2log(2)
M_LN10log(10)
M_PIpi
M_PI_2pi/2
M_PI_4pi/4
M_1_PI1/pi
M_2_PI2/pi
M_2_SQRTPI2/sqrt(pi)
M_SQRT2sqrt(2)
M_SQRT1_21/sqrt(2)
M_SQRT_3sqrt(3)
M_SQRT_32sqrt(32)
M_LOG10_2Log10(2)
M_2PI2*pi
M_SQRT_PIsqrt(pi)
M_1_SQRT_2PI1/(sqrt(2*pi))
M_LN_SQRT_PIlog(sqrt(pi))
M_LN_SQRT_2PIlog(sqrt(2*pi))
M_LN_SQRT_PId2log(sqrt(pi/2))
pipi
NAR’s NA value
NaNNot a Number Value
InfInfinite Value
newind1: First record of individual; 2: Subsequent record of individualNEWIND
rxFlagFlag for what part of the rxode2 model is being run; 1: ddt; 2: jac; 3: ini; 4: F; 5: lag; 6: rate; 7: dur; 8: mtime; 9: matrix exponential; 10: inductive linearization; 11: lhs

Note that rxFlag will always output 11 or calc_lhs since that is where the final variables are calculated, though you can tweak or test certain parts of rxode2 by using this flag.

Residual functions when using rxode2 functions

In addition to ~ hiding output for certain types of output, it also is used to specify a residual output or endpoint when the input is an rxode2 model function (that includes the residual in the model({}) block).

These specifications are of the form:

var ~ add(add.sd)

Indicating the variable var is the variable that represents the individual central tendencies of the model and it also represents the compartment specification in the data-set.

You can also change the compartment name using the | syntax, that is:

var ~ add(add.sd) | cmt

In the above case var represents the central tendency and cmt represents the compartment or dvid specification.

Transformations

For normal and related distributions, you can apply the transformation on both sides by using some keywords/functions to apply these transformations.

Transformationrxode2/nlmixr2 code
Box-Cox+boxCox(lambda)
Yeo-Johnson+yeoJohnson(lambda)
logit-normal+logitNorm(logit.sd, low, hi)
probit-normal+probitNorm(probid.sd, low, hi)
log-normal+lnorm(lnorm.sd)

By default for the likelihood for all of these transformations is calculated on the untransformed scale.

For bounded variables like logit-normal or probit-normal the low and high values are defaulted to 0 and 1 if missing.

For models where you wish to have a proportional model on one of these transformation you can replace the standard deviation with NA

To allow for more transformations, lnorm(), probitNorm() and logitNorm() can be combined the variance stabilizing yeoJohnson() transformation.

For the normal and t-related distributions, we wanted to keep the ability to use skewed distributions additive and proportional in the t/cauchy-space, so these distributions are specified differently in comparison to the other supported distributions within nlmixr2:

DistributionHow to AddExample
Normal (log-likelihood)+dnorm()cc ~ add(add.sd) + dnorm()
T-distribution+dt(df)cc ~a dd(add.sd) + dt(df)
Cauchy (t with df=1)+dcauchy()cc ~ add(add.sd) + dcauchy()

Note that with the normal and t-related distributions nlmixr2 will calculate cwres and npde under the normal assumption to help assess the goodness of the fit of the model.

Also note that the +dnorm() is mostly for testing purposes and will slow down the estimation procedure in nlmixr2. We suggest not adding it (except for explicit testing). When there are multiple endpoint models that mix non-normal and normal distributions, the whole problem is shifted to a log-likelihood method for estimation in nlmixr2.

Notes on additive + proportional models

There are two different ways to specify additive and proportional models, which we will call combined1 and combined2, the same way that Monolix calls the two distributions (to avoid between software differences in naming).

The first, combined1, assumes that the additive and proportional differences are on the standard deviation scale, or:

y=f+(a+b* f^c)*err

The second, combined2, assumes that the additive and proportional differences are combined on a variance scale:

y=f+[sqrt(a^2+b^2 *f^(2c))]*err

The default in nlmixr2/rxode2 if not otherwise specified is combined2 since it mirrors how adding 2 normal distributions in statistics will add their variances (not the standard deviations). However, the combined1 can describe the data possibly even better than combined2 so both are possible options in rxode2/nlmixr2.

Distributions of known likelihoods

For residuals that are not related to normal, t-distribution or cauchy, often the residual specification is of the form:

cmt ~ dbeta(alpha, beta)

Where the compartment specification is on the left handed side of the specification.

For generalized likelihood you can specify:

ll(cmt) ~ llik specification

Ordinal likelihoods

Finally, ordinal likelihoods/simulations can be specified in 2 ways. The first is:

err ~ c(p0, p1, p2)

Here err represents the compartment and p0 is the probability of being in a specific category:

CategoryProbability
1p0
2p1
3p2
41-p0-p1-p2

It is up to the model to ensure that the sum of the p values are less than 1. Additionally you can write an arbitrary number of categories in the ordinal model described above.

It seems a little off that p0 is the probability for category 1 and sometimes scores are in non-whole numbers. This can be modeled as follows:

err ~ c(p0=0, p1=1, p2=2, 3)

Here the numeric categories are specified explicitly, and the probabilities remain the same:

CategoryProbability
0p0
1p1
2p2
31-p0-p1-p2

General table of supported residual distributions

In general all the that are supported are in the following table (available in rxode2::rxResidualError)

Error modelFunctional FormTransformationcodeaddProplhs
constantNonevar ~ add(add.sd)response variable
proportionalNonevar ~ prop(prop.sd)response variable
powerNonevar ~ pow(pow.sd, exponent)response variable
additive+proportionalcombined1Nonevar ~ add(add.sd) + prop(prop.sd) + combined1()addProp=1response variable
additive+proportionalcombined2Nonevar ~ add(add.sd) + prop(prop.sd) + combined2()addProp=2response variable
additive+powercombined1Nonevar ~ add(add.sd) + pow(pow.sd, exponent) + combined1()addProp=1response variable
additive+powercombined2Nonevar ~ add(add.sd) + pow(pow.sd, exponent) + combined2()addProp=2response variable
constantlogvar ~ lnorm(add.sd)response variable
proportionallogvar ~ lnorm(NA) + prop(prop.sd)response variable
powerlogvar ~ lnorm(NA) + pow(pow.sd, exponent)response variable
additive+proportionalcombined1logvar ~ lnorm(add.sd) + prop(prop.sd) + combined1()addProp=1response variable
additive+proportionalcombined2logvar ~ lnorm(add.sd) + prop(prop.sd) + combined2()addProp=2response variable
additive+powercombined1logvar ~ lnorm(add.sd) + pow(pow.sd, exponent) + combined1()addProp=1response variable
additive+powercombined2logvar ~ lnorm(add.sd) + pow(pow.sd, exponent) + combined2()addProp=2response variable
constantboxCoxvar ~ boxCox(lambda) + add(add.sd)response variable
proportionalboxCoxvar ~ boxCox(lambda) + prop(prop.sd)response variable
powerboxCoxvar ~ boxCox(lambda) + pow(pow.sd, exponent)response variable
additive+proportionalcombined1boxCoxvar ~ boxCox(lambda) + add(add.sd) + prop(prop.sd) + combined1()addProp=1response variable
additive+proportionalcombined2boxCoxvar ~ boxCox(lambda) + add(add.sd) + prop(prop.sd) + combined2()addProp=2response variable
additive+powercombined1boxCoxvar ~ boxCox(lambda) + add(add.sd) + pow(pop.sd, exponent) + combined1()addProp=1response variable
additive+powercombined2boxCoxvar ~ boxCox(lambda) + add(add.sd) + pow(pop.sd, exponent) + combined2()addProp=2response variable
constantyeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd)response variable
proportionalyeoJohnsonvar ~ yeoJohnson(lambda) + prop(prop.sd)response variable
poweryeoJohnsonvar ~ yeoJohnson(lambda) + pow(pow.sd, exponent)response variable
additive+proportionalcombined1yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + prop(prop.sd) + combined1()addProp=1response variable
additive+proportionalcombined2yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + prop(prop.sd) + combined2()addProp=2response variable
additive+powercombined1yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + pow(pop.sd, exponent) + combined1()addProp=1response variable
additive+powercombined2yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + pow(pop.sd, exponent) + combined2()addProp=2response variable
constantlogitvar ~ logitNorm(logit.sd)response variable
proportionallogitvar ~ logitNorm(NA) + prop(prop.sd)response variable
powerlogitvar ~ logitNorm(NA) + pow(pow.sd, exponent)response variable
additive+proportionalcombined1logitvar ~ logitNorm(logit.sd) + prop(prop.sd)addProp=1response variable
additive+proportionalcombined2logitvar ~ logitNorm(logit.sd) + prop(prop.sd)addProp=2response variable
additive+powercombined1logitvar ~ logitNorm(logit.sd) + pow(pow.sd, exponent)addProp=1response variable
additive+powercombined2logitvar ~ logitNorm(logit.sd) + pow(pow.sd, exponent)addProp=2response variable
additiveyeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd)response variable
proportionalyeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(NA) + prop(prop.sd)response variable
poweryeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(NA) + pow(pow.sd, exponent)response variable
additive+proportionalcombined1yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + prop(prop.sd)addProp=1response variable
additive+proportionalcombined2yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + prop(prop.sd)addProp=2response variable
additive+powercombined1yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + pow(pow.sd, exponent)addProp=1response variable
additive+powercombined2yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + pow(pow.sd, exponent)addProp=2response variable
constantlogitvar ~ probitNorm(probit.sd)response variable
proportionalprobitvar ~ probitNorm(NA) + prop(prop.sd)response variable
powerprobitvar ~ probitNorm(NA) + pow(pow.sd, exponent)response variable
additive+proportionalcombined1probitvar ~ probitNorm(probit.sd) + prop(prop.sd) + combined1()addProp=1response variable
additive+proportionalcombined2probitvar ~ probitNorm(probit.sd) + prop(prop.sd) + combined2()addProp=2response variable
additive+powercombined1probitvar ~ probitNorm(probit.sd) + pow(pow.sd, exponent) + combined1()addProp=1response variable
additive+powercombined2probitvar ~ probitNorm(probit.sd) + pow(pow.sd, exponent) + combined2()addProp=2response variable
additiveyeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd)response variable
proportionalyeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(NA) + prop(prop.sd)response variable
poweryeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(NA) + pow(pow.sd, exponent)response variable
additive+proportionalcombined1yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + prop(prop.sd) + combined1()addProp=1response variable
additive+proportionalcombined2yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + prop(prop.sd) + combined2()addProp=2response variable
additive+powercombined1yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + pow(pow.sd, exponent) + combined1()addProp=1response variable
additive+powercombined2yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + pow(pow.sd, exponent) + combined2()addProp=2response variable
constant+tNonevar ~ add(add.sd) + dt(df)response variable
proportional+tNonevar ~ prop(prop.sd) + dt(df)response variable
power+tNonevar ~ pow(pow.sd, exponent) + dt(df)response variable
additive+proportional+tcombined1Nonevar ~ add(add.sd) + prop(prop.sd) + dt(df) + combined1()addProp=1response variable
additive+proportional+tcombined2Nonevar ~ add(add.sd) + prop(prop.sd) + dt(df) + combined2()addProp=2response variable
additive+power+tcombined1Nonevar ~ add(add.sd) + pow(pow.sd, exponent) + dt(df) +combined1()addProp=1response variable
additive+power+tcombined2Nonevar ~ add(add.sd) + pow(pow.sd, exponent) + dt(df) +combined2()addProp=2response variable
constant+tlogvar ~ lnorm(add.sd) + dt(df)response variable
proportional+tlogvar ~ lnorm(NA) + prop(prop.sd) + dt(df)response variable
power+tlogvar ~ lnorm(NA) + pow(pow.sd, exponent) + dt(df)response variable
additive+proportional+tcombined1logvar ~ lnorm(add.sd) + prop(prop.sd) + dt(df) +combined1()addProp=1response variable
additive+proportional+tcombined2logvar ~ lnorm(add.sd) + prop(prop.sd) + dt(df) + combined2()addProp=2response variable
additive+power+tcombined1logvar ~ lnorm(add.sd) + pow(pow.sd, exponent) + dt(df) + combined1()addProp=1response variable
additive+power+tcombined2logvar ~ lnorm(add.sd) + pow(pow.sd, exponent) + dt(df) + combined2()addProp=2response variable
constant+tboxCoxvar ~ boxCox(lambda) + add(add.sd)+dt(df)response variable
proportional+tboxCoxvar ~ boxCox(lambda) + prop(prop.sd)+dt(df)response variable
power+tboxCoxvar ~ boxCox(lambda) + pow(pow.sd, exponent)+dt(df)response variable
additive+proportional+tcombined1boxCoxvar ~ boxCox(lambda) + add(add.sd) + prop(prop.sd) + dt(df) + combined1()addProp=1response variable
additive+proportional+tcombined2boxCoxvar ~ boxCox(lambda) + add(add.sd) + prop(prop.sd) + dt(df) + combined2()addProp=2response variable
additive+power+tcombined1boxCoxvar ~ boxCox(lambda) + add(add.sd) + pow(pop.sd, exponent) + dt(df) + combined1()addProp=1response variable
additive+power+tcombined2boxCoxvar ~ boxCox(lambda) + add(add.sd) + pow(pop.sd, exponent) + dt(df) + combined2()addProp=2response variable
constant+tyeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + dt(df)response variable
proportional+tyeoJohnsonvar ~ yeoJohnson(lambda) + prop(prop.sd) + dt(df)response variable
power+tyeoJohnsonvar ~ yeoJohnson(lambda) + pow(pow.sd, exponent) + dt(df)response variable
additive+proportional+tcombined1yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + prop(prop.sd) + dt(df) + combined1()addProp=1response variable
additive+proportional+tcombined2yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + prop(prop.sd) + dt(df) + combined2()addProp=2response variable
additive+power+tcombined1yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + pow(pop.sd, exponent) + dt(df) + combined1()addProp=1response variable
additive+power+tcombined2yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + pow(pop.sd, exponent) + dt(df) + combined2()addProp=2response variable
constant+tlogitvar ~ logitNorm(logit.sd)+dt(df)response variable
proportional+tlogitvar ~ logitNorm(NA) + prop(prop.sd)+dt(df)response variable
power+tlogitvar ~ logitNorm(NA) + pow(pow.sd, exponent) + dt(df)response variable
additive+proportional+tcombined1logitvar ~ logitNorm(logit.sd) + prop(prop.sd) + dt(df) + combined1()addProp=1response variable
additive+proportional+tcombined2logitvar ~ logitNorm(logit.sd) + prop(prop.sd) + dt(df) + combined2()addProp=2response variable
additive+power+tcombined1logitvar ~ logitNorm(logit.sd) + pow(pow.sd, exponent) + dt(df) + combined1()addProp=1response variable
additive+power+tcombined2logitvar ~ logitNorm(logit.sd) + pow(pow.sd, exponent) + dt(df) + combined2()addProp=2response variable
additive+tyeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + dt(df)response variable
proportional+tyeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(NA) + prop(prop.sd) + dt(df)response variable
power+tyeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(NA) + pow(pow.sd, exponent) + dt(df)response variable
additive+proportional+tcombined1yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + prop(prop.sd) + dt(df) + combined1()addProp=1response variable
additive+proportional+tcombined2yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + prop(prop.sd) + dt(df) + combined2()addProp=2response variable
additive+power+tcombined1yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + pow(pow.sd, exponent) + dt(df) + combined1()addProp=1response variable
additive+power+tcombined2yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + pow(pow.sd, exponent) + dt(df) + combined2()addProp=2response variable
constant+tlogitvar ~ probitNorm(probit.sd) + dt(df)response variable
proportional+tprobitvar ~ probitNorm(NA) + prop(prop.sd) + dt(df)response variable
power+tprobitvar ~ probitNorm(NA) + pow(pow.sd, exponent) + dt(df)response variable
additive+proportional+tcombined1probitvar ~ probitNorm(probit.sd) + prop(prop.sd) + dt(df) + combined1()addProp=1response variable
additive+proportional+tcombined2probitvar ~ probitNorm(probit.sd) + prop(prop.sd) + dt(df) + combined2()addProp=2response variable
additive+power+tcombined1probitvar ~ probitNorm(probit.sd) + pow(pow.sd, exponent) + dt(df) + combined1()addProp=1response variable
additive+power+tcombined2probitvar ~ probitNorm(probit.sd) + pow(pow.sd, exponent) + dt(df) + combined2()addProp=2response variable
additive+tyeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + dt(df)response variable
proportional+tyeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(NA) + prop(prop.sd) + dt(df)response variable
power+tyeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(NA) + pow(pow.sd, exponent) + dt(df)response variable
additive+proportional+tcombined1yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + prop(prop.sd) + dt(df) + combined1()addProp=1response variable
additive+proportional+tcombined2yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + prop(prop.sd) + dt(df) + combined2()addProp=2response variable
additive+power+tcombined1yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + pow(pow.sd, exponent) + dt(df) + combined1()addProp=1response variable
additive+power+tcombined2yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + pow(pow.sd, exponent) + dt(df) +combined2()addProp=2response variable
constant+cauchyNonevar ~ add(add.sd) + dcauchy()response variable
proportional+cauchyNonevar ~ prop(prop.sd) + dcauchy()response variable
power+cauchyNonevar ~ pow(pow.sd, exponent) + dcauchy()response variable
additive+proportional+cauchycombined1Nonevar ~ add(add.sd) + prop(prop.sd) + dcauchy() + combined1()addProp=1response variable
additive+proportional+cauchycombined2Nonevar ~ add(add.sd) + prop(prop.sd) + dcauchy() + combined2()addProp=2response variable
additive+power+cauchycombined1Nonevar ~ add(add.sd) + pow(pow.sd, exponent) + dcauchy() +combined1()addProp=1response variable
additive+power+cauchycombined2Nonevar ~ add(add.sd) + pow(pow.sd, exponent) + dcauchy() +combined2()addProp=2response variable
constant+cauchylogvar ~ lnorm(add.sd) + dcauchy()response variable
proportional+cauchylogvar ~ lnorm(NA) + prop(prop.sd) + dcauchy()response variable
power+cauchylogvar ~ lnorm(NA) + pow(pow.sd, exponent) + dcauchy()response variable
additive+proportional+cauchycombined1logvar ~ lnorm(add.sd) + prop(prop.sd) + dcauchy() +combined1()addProp=1response variable
additive+proportional+cauchycombined2logvar ~ lnorm(add.sd) + prop(prop.sd) + dcauchy() + combined2()addProp=2response variable
additive+power+cauchycombined1logvar ~ lnorm(add.sd) + pow(pow.sd, exponent) + dcauchy() + combined1()addProp=1response variable
additive+power+cauchycombined2logvar ~ lnorm(add.sd) + pow(pow.sd, exponent) + dcauchy() + combined2()addProp=2response variable
constant+cauchyboxCoxvar ~ boxCox(lambda) + add(add.sd)+dcauchy()response variable
proportional+cauchyboxCoxvar ~ boxCox(lambda) + prop(prop.sd)+dcauchy()response variable
power+cauchyboxCoxvar ~ boxCox(lambda) + pow(pow.sd, exponent)+dcauchy()response variable
additive+proportional+cauchycombined1boxCoxvar ~ boxCox(lambda) + add(add.sd) + prop(prop.sd) + dcauchy() + combined1()addProp=1response variable
additive+proportional+cauchycombined2boxCoxvar ~ boxCox(lambda) + add(add.sd) + prop(prop.sd) + dcauchy() + combined2()addProp=2response variable
additive+power+cauchycombined1boxCoxvar ~ boxCox(lambda) + add(add.sd) + pow(pop.sd, exponent) + dcauchy() + combined1()addProp=1response variable
additive+power+cauchycombined2boxCoxvar ~ boxCox(lambda) + add(add.sd) + pow(pop.sd, exponent) + dcauchy() + combined2()addProp=2response variable
constant+cauchyyeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + dcauchy()response variable
proportional+cauchyyeoJohnsonvar ~ yeoJohnson(lambda) + prop(prop.sd) + dcauchy()response variable
power+cauchyyeoJohnsonvar ~ yeoJohnson(lambda) + pow(pow.sd, exponent) + dcauchy()response variable
additive+proportional+cauchycombined1yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + prop(prop.sd) + dcauchy() + combined1()addProp=1response variable
additive+proportional+cauchycombined2yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + prop(prop.sd) + dcauchy() + combined2()addProp=2response variable
additive+power+cauchycombined1yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + pow(pop.sd, exponent) + dcauchy() + combined1()addProp=1response variable
additive+power+cauchycombined2yeoJohnsonvar ~ yeoJohnson(lambda) + add(add.sd) + pow(pop.sd, exponent) + dcauchy() + combined2()addProp=2response variable
constant+cauchylogitvar ~ logitNorm(logit.sd)+dcauchy()response variable
proportional+cauchylogitvar ~ logitNorm(NA) + prop(prop.sd)+dcauchy()response variable
power+cauchylogitvar ~ logitNorm(NA) + pow(pow.sd, exponent) + dcauchy()response variable
additive+proportional+cauchycombined1logitvar ~ logitNorm(logit.sd) + prop(prop.sd) + dcauchy() + combined1()addProp=1response variable
additive+proportional+cauchycombined2logitvar ~ logitNorm(logit.sd) + prop(prop.sd) + dcauchy() + combined2()addProp=2response variable
additive+power+cauchycombined1logitvar ~ logitNorm(logit.sd) + pow(pow.sd, exponent) + dcauchy() + combined1()addProp=1response variable
additive+power+cauchycombined2logitvar ~ logitNorm(logit.sd) + pow(pow.sd, exponent) + dcauchy() + combined2()addProp=2response variable
additive+cauchyyeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + dcauchy()response variable
proportional+cauchyyeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(NA) + prop(prop.sd) + dcauchy()response variable
power+cauchyyeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(NA) + pow(pow.sd, exponent) + dcauchy()response variable
additive+proportional+cauchycombined1yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + prop(prop.sd) + dcauchy() + combined1()addProp=1response variable
additive+proportional+cauchycombined2yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + prop(prop.sd) + dcauchy() + combined2()addProp=2response variable
additive+power+cauchycombined1yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + pow(pow.sd, exponent) + dcauchy() + combined1()addProp=1response variable
additive+power+cauchycombined2yeoJohnson(logit())var ~ yeoJohnson(lambda) + logitNorm(logit.sd) + pow(pow.sd, exponent) + dcauchy() + combined2()addProp=2response variable
constant+cauchylogitvar ~ probitNorm(probit.sd) + dcauchy()response variable
proportional+cauchyprobitvar ~ probitNorm(NA) + prop(prop.sd) + dcauchy()response variable
power+cauchyprobitvar ~ probitNorm(NA) + pow(pow.sd, exponent) + dcauchy()response variable
additive+proportional+cauchycombined1probitvar ~ probitNorm(probit.sd) + prop(prop.sd) + dcauchy() + combined1()addProp=1response variable
additive+proportional+cauchycombined2probitvar ~ probitNorm(probit.sd) + prop(prop.sd) + dcauchy() + combined2()addProp=2response variable
additive+power+cauchycombined1probitvar ~ probitNorm(probit.sd) + pow(pow.sd, exponent) + dcauchy() + combined1()addProp=1response variable
additive+power+cauchycombined2probitvar ~ probitNorm(probit.sd) + pow(pow.sd, exponent) + dcauchy() + combined2()addProp=2response variable
additive+cauchyyeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + dcauchy()response variable
proportional+cauchyyeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(NA) + prop(prop.sd) + dcauchy()response variable
power+cauchyyeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(NA) + pow(pow.sd, exponent) + dcauchy()response variable
additive+proportional+cauchycombined1yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + prop(prop.sd) + dcauchy() + combined1()addProp=1response variable
additive+proportional+cauchycombined2yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + prop(prop.sd) + dcauchy() + combined2()addProp=2response variable
additive+power+cauchycombined1yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + pow(pow.sd, exponent) + dcauchy() + combined1()addProp=1response variable
additive+power+cauchycombined2yeoJohnson(probit())var ~ yeoJohnson(lambda) + probitNorm(probit.sd) + pow(pow.sd, exponent) + dcauchy() +combined2()addProp=2response variable
poissionnonecmt ~ dpois(lamba)compartment specification
binomialnonecmt ~ dbinom(n, p)compartment specification
betanonecmt ~ dbeta(alpha, beta)compartment specification
chisqnonecmt ~ dchisq(nu)compartment specification
exponentialnonecmt ~ dexp(r)compartment specification
uniformnonecmt ~ dunif(a, b)compartment specification
weibullnonecmt ~ dweibull(a, b)compartment specification
gammanonecmt ~ dgamma(a, b)compartment specification
geometricnonecmt ~ dgeom(a)compartment specification
negative binomial form #1nonecmt ~ dnbinom(n, p)compartment specification
negative binomial form #2nonecmt ~ dnbinomMu(size, mu)compartment specification
ordinal probabilitynonecmt ~ c(p0=0, p1=1, p2=2, 3)compartment specification
log-likelihoodnonell(cmt) ~ log likelihood expressionlikelihood + compartment expression

Creating rxode2 models

NA

References

Chamber, J. M. and Temple Lang, D. (2001) Object Oriented Programming in R. R News, Vol. 1, No. 3, September 2001. https://cran.r-project.org/doc/Rnews/Rnews_2001-3.pdf.

Hindmarsh, A. C. ODEPACK, A Systematized Collection of ODE Solvers. Scientific Computing, R. S. Stepleman et al. (Eds.), North-Holland, Amsterdam, 1983, pp. 55-64.

Petzold, L. R. Automatic Selection of Methods for Solving Stiff and Nonstiff Systems of Ordinary Differential Equations. Siam J. Sci. Stat. Comput. 4 (1983), pp. 136-148.

Hairer, E., Norsett, S. P., and Wanner, G. Solving ordinary differential equations I, nonstiff problems. 2nd edition, Springer Series in Computational Mathematics, Springer-Verlag (1993).

Plevyak, J. dparser, https://dparser.sourceforge.net/. Web. 12 Oct. 2015.

Author

Melissa Hallow, Wenping Wang and Matthew Fidler

Examples

# \donttest{

mod <- function() {
  ini({
    KA   <- .291
    CL   <- 18.6
    V2   <- 40.2
    Q    <- 10.5
    V3   <- 297.0
    Kin  <- 1.0
    Kout <- 1.0
    EC50 <- 200.0
  })
  model({
    # A 4-compartment model, 3 PK and a PD (effect) compartment
    # (notice state variable names 'depot', 'centr', 'peri', 'eff')
    C2 <- centr/V2
    C3 <- peri/V3
    d/dt(depot) <- -KA*depot;
    d/dt(centr) <- KA*depot - CL*C2 - Q*C2 + Q*C3;
    d/dt(peri)  <-                    Q*C2 - Q*C3;
    d/dt(eff)   <- Kin - Kout*(1-C2/(EC50+C2))*eff;
    eff(0)      <- 1
  })
}

m1 <- rxode2(mod)
#>  
#>  
print(m1)
#>  ── rxode2-based free-form 4-cmt ODE model ────────────────────────────────────── 
#>  ── Initalization: ──  
#> Fixed Effects ($theta): 
#>      KA      CL      V2       Q      V3     Kin    Kout    EC50 
#>   0.291  18.600  40.200  10.500 297.000   1.000   1.000 200.000 
#> 
#> States ($state or $stateDf): 
#>   Compartment Number Compartment Name
#> 1                  1            depot
#> 2                  2            centr
#> 3                  3             peri
#> 4                  4              eff
#>  ── Model (Normalized Syntax): ── 
#> function() {
#>     ini({
#>         KA <- 0.291
#>         CL <- 18.6
#>         V2 <- 40.2
#>         Q <- 10.5
#>         V3 <- 297
#>         Kin <- 1
#>         Kout <- 1
#>         EC50 <- 200
#>     })
#>     model({
#>         C2 <- centr/V2
#>         C3 <- peri/V3
#>         d/dt(depot) <- -KA * depot
#>         d/dt(centr) <- KA * depot - CL * C2 - Q * C2 + Q * C3
#>         d/dt(peri) <- Q * C2 - Q * C3
#>         d/dt(eff) <- Kin - Kout * (1 - C2/(EC50 + C2)) * eff
#>         eff(0) <- 1
#>     })
#> }

# Step 2 - Create the model input as an EventTable,
# including dosing and observation (sampling) events

# QD (once daily) dosing for 5 days.

qd <- et(amountUnits = "ug", timeUnits = "hours") %>%
  et(amt = 10000, addl = 4, ii = 24)

# Sample the system hourly during the first day, every 8 hours
# then after
qd <- qd %>% et(0:24) %>%
  et(from = 24 + 8, to = 5 * 24, by = 8)

# Step 3 - solve the system

qd.cp <- rxSolve(m1, qd)
#>  
#>  
#> using C compiler: ‘gcc (Ubuntu 11.4.0-1ubuntu1~22.04) 11.4.0’

head(qd.cp)
#>    time       C2        C3     depot    centr      peri      eff
#> 1 0 [h]  0.00000 0.0000000 10000.000    0.000    0.0000 1.000000
#> 2 1 [h] 43.99334 0.9113641  7475.157 1768.532  270.6751 1.083968
#> 3 2 [h] 54.50866 2.6510696  5587.797 2191.248  787.3677 1.179529
#> 4 3 [h] 51.65163 4.4243597  4176.966 2076.396 1314.0348 1.227523
#> 5 4 [h] 44.37513 5.9432612  3122.347 1783.880 1765.1486 1.233503
#> 6 5 [h] 36.46382 7.1389804  2334.004 1465.845 2120.2772 1.214084

# }