Disclaimer

This is notebook explains in more detail how to solve a problem with multi-trait BLUP of breeding values. The terms ‘multi-trait’ and ‘multi-variate’ are used as synonyms.

Data

The same dataset as in Problems 1 and 2 of Exercise 9 (https://charlotte-ngs.github.io/lbgfs2020/ex/lbgfs2020_w10_ex09.pdf) is used. The dataset is shown in the following table.

n_nr_trait <- 2
n_nr_founder <- 3
n_nr_animal <- 8
n_nr_observation <- n_nr_animal - n_nr_founder
tbl_data_sol12p01 <- tibble::tibble(Animal = c((n_nr_founder+1):n_nr_animal),
                                        Sex = c("Male", "Female","Female","Male","Male"),
                                        Sire = c(1,3,1,4,3),
                                        Dam = c(NA,2,2,5,6),
                                        WWG = c(4.5,2.9,3.9,3.5,5.0),
                                        PWG = c(6.8,5.0,6.8,6.0,7.5))
tbl_data_sol12p01

The variables defined before the data have the following meaning

Parameter

The parameters that are given consist of the genetic variance-covariance matrix \(G_0\) between the two traits WWG and PWG and the variance-covariance matrix \(R_0\) between the traits. These matrices are

(mat_G0 <- matrix(data = c(20,18,18,40), nrow = n_nr_trait, byrow = TRUE))
     [,1] [,2]
[1,]   20   18
[2,]   18   40

and

(mat_R0 <- matrix(data = c(40,11,11,30), nrow = n_nr_trait, byrow = TRUE))
     [,1] [,2]
[1,]   40   11
[2,]   11   30

Model

The multi-trait model for estimating fixed effects and for predicting breeding values looks the same as the model for only one trait.

\[y = X\beta + Zu + e\]

Although, the model looks the same, the components must be adapted to the situation of multiple traits. This adaptation is different for the vectors \(y\), \(\beta\), \(u\) and \(e\) in the model compared to the matrices \(X\) and \(Z\).

Vectors

For the vectors, the versions of the single traits are just appended to each other. For the vector \(y\) this means that

\[y = \begin{bmatrix} y_1 \\ y_2 \end{bmatrix}\] where \(y_1\) are all the observations for the trait ‘WWG’ and \(y_2\) are all the observations for the trait ‘PWG’. Inserting the numbers this means

(y1 <- tbl_data_sol12p01$WWG)
[1] 4.5 2.9 3.9 3.5 5.0

and

(y2 <- tbl_data_sol12p01$PWG)
[1] 6.8 5.0 6.8 6.0 7.5

and appending \(y_2\) at the end of \(y_1\) leads to

(y <- c(y1,y2))
 [1] 4.5 2.9 3.9 3.5 5.0 6.8 5.0 6.8 6.0 7.5

The same is done for all the other vectors. Because, ‘sex’ with two levels ‘female’ and ‘male’ is the only fixed effect, the vector \(\beta\) for the multi-trait model looks as follows

\[\beta = \begin{bmatrix} \beta_{M,WWG} \\\beta_{F,WWG} \\\beta_{M,PWG} \\\beta_{F,PWG}\end{bmatrix}\]

The vector \(u\) of breeding values contains as components all breeding values for all animals in the pedigree.

\[u = \begin{bmatrix} u_{WWG} \\ u_{PWG} \end{bmatrix}\] where \(u_{WWG}\) is the vector with breeding values for ‘WWG’ of all animals in the pedigree and \(u_{PWG}\) is the vector with breeding values for ‘PWG’ of all animals in the pedigree.

Matrices

The dimensions of the design matrices must match the number of observations and the length of their associated effects. Hence for the matrix \(X\) which has the vector \(\beta\) of fixed effects as the associated effect, it must have 4 columns as their are two fixed effect levels for each of the two traits.

The matrix \(X\) is the design matrix for the vector of fixed effects \(\beta\).

\[X = \left[ \begin{array}{lr} X_1 & 0 \\ 0 & X_2 \end{array} \right]\]

where \(X_1\) and \(X_2\) are the design matrix for the fixed effects for the single traits ‘WWG’ and ‘PWG’.

mat_X1 <- matrix(c(1, 0,
                   0, 1,
                   0, 1,
                   1, 0,
                   1, 0), ncol = n_nr_trait, byrow = TRUE)
mat_X2 <- mat_X1
mat_Xzero <- matrix(0, nrow = n_nr_observation, ncol = n_nr_trait)
(mat_X <- rbind(cbind(mat_X1, mat_Xzero), cbind(mat_Xzero, mat_X2)))
      [,1] [,2] [,3] [,4]
 [1,]    1    0    0    0
 [2,]    0    1    0    0
 [3,]    0    1    0    0
 [4,]    1    0    0    0
 [5,]    1    0    0    0
 [6,]    0    0    1    0
 [7,]    0    0    0    1
 [8,]    0    0    0    1
 [9,]    0    0    1    0
[10,]    0    0    1    0

The matrix \(Z\) links breeding values to observations.

mat_Zfounder <- matrix(0, nrow = n_nr_observation, ncol = n_nr_founder)
mat_Z1 <- cbind(mat_Zfounder, diag(1, nrow = n_nr_observation))
mat_Z2 <- mat_Z1
mat_Zzero <- matrix(0, nrow = n_nr_observation, ncol = n_nr_animal)
(mat_Z <- rbind(cbind(mat_Z1, mat_Zzero), cbind(mat_Zzero, mat_Z2)))
      [,1] [,2] [,3] [,4] [,5] [,6] [,7] [,8] [,9] [,10] [,11] [,12] [,13] [,14] [,15] [,16]
 [1,]    0    0    0    1    0    0    0    0    0     0     0     0     0     0     0     0
 [2,]    0    0    0    0    1    0    0    0    0     0     0     0     0     0     0     0
 [3,]    0    0    0    0    0    1    0    0    0     0     0     0     0     0     0     0
 [4,]    0    0    0    0    0    0    1    0    0     0     0     0     0     0     0     0
 [5,]    0    0    0    0    0    0    0    1    0     0     0     0     0     0     0     0
 [6,]    0    0    0    0    0    0    0    0    0     0     0     1     0     0     0     0
 [7,]    0    0    0    0    0    0    0    0    0     0     0     0     1     0     0     0
 [8,]    0    0    0    0    0    0    0    0    0     0     0     0     0     1     0     0
 [9,]    0    0    0    0    0    0    0    0    0     0     0     0     0     0     1     0
[10,]    0    0    0    0    0    0    0    0    0     0     0     0     0     0     0     1

Solutions

Solutions are obtained by mixed-model equations.

\[ \begin{bmatrix} X^TR^{-1}X & X^TR^{-1}Z \\ Z^TR^{-1}X & Z^TR^{-1}Z + G^{-1} \end{bmatrix} \begin{bmatrix} \hat{\beta} \\ \hat{u} \end{bmatrix} = \begin{bmatrix} X^TR^{-1}y \\ Z^TR^{-1}y \end{bmatrix} \]

In the multi-trait case, we have to use the general form of mixed-model equations, because the variance-covariance matrix of the random error terms does not have the simple form of \(R = I *\sigma_e^2\). In the multi-trait case the variance-covariance matrix \(R = R_0 \otimes I_n\) where \(n\) is the number of observations for a single trait. For the mixed model equations, the inverse \(R^{-1}\) of \(R\) is needed. This can be computed as

\[R^{-1} = R_0^{-1} \otimes I_n\]

mat_R0inv <- solve(mat_R0)
(mat_Rinv <- mat_R0inv %x% diag(1, nrow = n_nr_observation))
             [,1]        [,2]        [,3]        [,4]        [,5]        [,6]        [,7]        [,8]        [,9]       [,10]
 [1,]  0.02780352  0.00000000  0.00000000  0.00000000  0.00000000 -0.01019462  0.00000000  0.00000000  0.00000000  0.00000000
 [2,]  0.00000000  0.02780352  0.00000000  0.00000000  0.00000000  0.00000000 -0.01019462  0.00000000  0.00000000  0.00000000
 [3,]  0.00000000  0.00000000  0.02780352  0.00000000  0.00000000  0.00000000  0.00000000 -0.01019462  0.00000000  0.00000000
 [4,]  0.00000000  0.00000000  0.00000000  0.02780352  0.00000000  0.00000000  0.00000000  0.00000000 -0.01019462  0.00000000
 [5,]  0.00000000  0.00000000  0.00000000  0.00000000  0.02780352  0.00000000  0.00000000  0.00000000  0.00000000 -0.01019462
 [6,] -0.01019462  0.00000000  0.00000000  0.00000000  0.00000000  0.03707136  0.00000000  0.00000000  0.00000000  0.00000000
 [7,]  0.00000000 -0.01019462  0.00000000  0.00000000  0.00000000  0.00000000  0.03707136  0.00000000  0.00000000  0.00000000
 [8,]  0.00000000  0.00000000 -0.01019462  0.00000000  0.00000000  0.00000000  0.00000000  0.03707136  0.00000000  0.00000000
 [9,]  0.00000000  0.00000000  0.00000000 -0.01019462  0.00000000  0.00000000  0.00000000  0.00000000  0.03707136  0.00000000
[10,]  0.00000000  0.00000000  0.00000000  0.00000000 -0.01019462  0.00000000  0.00000000  0.00000000  0.00000000  0.03707136

The matrix \(G^{-1}\) in the mixed model equations corresponds to the inverse of the variance-covariance matrix of all breeding values and is computed as

\[G^{-1} = G_0^{-1} \otimes A^{-1}\]

mat_G0inv <- solve(mat_G0)
(ped <- pedigreemm::pedigree(sire = c(rep(NA, n_nr_founder), tbl_data_sol12p01$Sire), 
                             dam = c(rep(NA, n_nr_founder), tbl_data_sol12p01$Dam), 
                             label = as.character(1:n_nr_animal)))
mat_Ainv <- as.matrix(pedigreemm::getAInv(ped = ped))
mat_Ginv <- mat_G0inv %x% mat_Ainv
#round(mat_Ginv, digits = 4)

Coefficient Matrix \(C\)

The setup of the mixed model equations starts with the coefficient matrix \(C\)

\[ C = \begin{bmatrix} X^TR^{-1}X & X^TR^{-1}Z \\ Z^TR^{-1}X & Z^TR^{-1}Z + G^{-1} \end{bmatrix} = \begin{bmatrix} C_{11} & C_{12} \\ C_{21} & C_{22} \end{bmatrix} \] with

\[C_{11} = X^TR^{-1}X\] \[C_{12} = X^TR^{-1}Z\] \[C_{21} = Z^TR^{-1}X\] \[C_{22} = Z^TR^{-1}Z + G^{-1}\] Because the matrix C is symmetric, \(C_{21} = C_{12}^T\)

The coefficient matrix \(C\) can be constructed as

mat_C11 <- t(mat_X) %*% mat_Rinv %*% mat_X
mat_C12 <- t(mat_X) %*% mat_Rinv %*% mat_Z
mat_C21 <- t(mat_Z) %*% mat_Rinv %*% mat_X
mat_C22 <- t(mat_Z) %*% mat_Rinv %*% mat_Z + mat_Ginv
mat_C <- rbind(cbind(mat_C11, mat_C12), cbind(mat_C21, mat_C22))
#round(mat_C, digits = 3)

Right-Hand Side \(r\)

The vector \(r\) consists of the right-hand side of the mixed-model equations.

\[ r = \begin{bmatrix} r_1 \\ r_2 \end{bmatrix} = \begin{bmatrix} X^TR^{-1}y \\ Z^TR^{-1}y \end{bmatrix} \]

(vec_r <- rbind(t(mat_X) %*% mat_Rinv %*% y, t(mat_Z) %*% mat_Rinv %*% y))
            [,1]
 [1,] 0.15449490
 [2,] 0.06876738
 [3,] 0.62001854
 [4,] 0.36811863
 [5,] 0.00000000
 [6,] 0.00000000
 [7,] 0.00000000
 [8,] 0.05579240
 [9,] 0.02965709
[10,] 0.03911029
[11,] 0.03614458
[12,] 0.06255792
[13,] 0.00000000
[14,] 0.00000000
[15,] 0.00000000
[16,] 0.20620945
[17,] 0.15579240
[18,] 0.21232623
[19,] 0.18674699
[20,] 0.22706209

Solution Vector

The solution vector \(s\) is

\[s = \begin{bmatrix} \hat{\beta} \\ \hat{u} \end{bmatrix} = C^{-1}r \]

s <- solve(mat_C, vec_r)
s
              [,1]
 [1,]  4.360866999
 [2,]  3.397261592
 [3,]  6.799897620
 [4,]  5.880295937
 [5,]  0.150915567
 [6,] -0.015392510
 [7,] -0.078391896
 [8,] -0.010238959
 [9,] -0.270331441
[10,]  0.275808258
[11,] -0.316117562
[12,]  0.243755523
[13,]  0.279597973
[14,] -0.007610071
[15,] -0.170341439
[16,] -0.012670709
[17,] -0.477830262
[18,]  0.517238387
[19,] -0.478983695
[20,]  0.391961543

Reliabilities

The reliability \(B_i\) of a predicted breeding value \(\hat{u}_i\) of animal \(i\) for a given trait is computed according to formula (7.4) on page 80 of the course notes

\[ B_i = r_{u_i,\hat{u}_i}^2 = 1 - \frac{C_{ii}^{22}}{var(u_i)} \] where \(C_{ii}^{22}\) is the i-th diagnoal element of the matrix \(C^{22}\) and \(var(u_i)\) is the genetic additive variance of the trait for which we want to compute \(B_i\)

The matrix \(C^{22}\) is obtained from the inverse (\(C^{-1}\)) of the coeffient matrix \(C\) of the mixed model equations.

\[ C^{-1} = \begin{bmatrix} X^TR^{-1}X & X^TR^{-1}Z \\ Z^TR^{-1}X & Z^TR^{-1}Z + G^{-1} \end{bmatrix}^{-1} = \begin{bmatrix} C^{11} & C^{12} \\ C^{21} & C^{22} \end{bmatrix} \]

For our data the matrix \(C^{22}\) is extracted from \(C^{-1}\) by taking the sub-matrix in the lower right corner with dimensions \(16 \times 16\). The dimension is given by the number of traits (2) time the number of animals with predicted breeding values (8).

mat_Cinv <- solve(mat_C)
mat_Cinv22 <- mat_Cinv[5:20,5:20]
round(mat_Cinv22, digits = 3)
        [,1]   [,2]   [,3]   [,4]   [,5]   [,6]   [,7]   [,8]   [,9]  [,10]  [,11]  [,12]  [,13]  [,14]  [,15]  [,16]
 [1,] 18.605  0.324  1.579  8.540  2.163  8.577  5.632  5.446 16.090  0.433  2.165  7.012  2.953  7.041  5.339  5.113
 [2,]  0.324 19.596 -0.480  1.002  9.390  9.725  4.700  4.310  0.433 17.424 -0.639  1.389  8.165  8.580  4.037  3.559
 [3,]  1.579 -0.480 17.893  2.302  7.655  1.120  5.074  8.451  2.165 -0.639 15.110  3.128  5.796  1.564  4.633  6.888
 [4,]  8.540  1.002  2.302 16.505  2.270  5.155  9.706  5.412  7.012  1.389  3.128 13.212  3.106  4.743  8.656  6.216
 [5,]  2.163  9.390  7.655  2.270 16.541  7.148  8.548  7.038  2.953  8.165  5.796  3.106 13.278  7.427  7.025  6.109
 [6,]  8.577  9.725  1.120  5.155  7.148 17.151  6.206  8.531  7.041  8.580  1.564  4.743  7.427 14.036  6.035  7.010
 [7,]  5.632  4.700  5.074  9.706  8.548  6.206 17.115  7.053  5.339  4.037  4.633  8.656  7.025  6.035 13.970  7.278
 [8,]  5.446  4.310  8.451  5.412  7.038  8.531  7.053 16.284  5.113  3.559  6.888  6.216  6.109  7.010  7.278 12.951
 [9,] 16.090  0.433  2.165  7.012  2.953  7.041  5.339  5.113 35.902  0.931  4.646 15.733  6.339 15.797 11.804 11.316
[10,]  0.433 17.424 -0.639  1.389  8.165  8.580  4.037  3.559  0.931 38.768 -1.374  2.978 18.208 19.106  9.014  7.980
[11,]  2.165 -0.639 15.110  3.128  5.796  1.564  4.633  6.888  4.646 -1.374 33.799  6.717 13.122  3.352 10.281 15.466
[12,]  7.012  1.389  3.128 13.212  3.106  4.743  8.656  6.216 15.733  2.978  6.717 29.725  6.665 10.516 19.253 13.515
[13,]  2.953  8.165  5.796  3.106 13.278  7.427  7.025  6.109  6.339 18.208 13.122  6.665 29.865 16.283 15.759 13.625
[14,]  7.041  8.580  1.564  4.743  7.427 14.036  6.035  7.010 15.797 19.106  3.352 10.516 16.283 31.503 13.312 15.726
[15,]  5.339  4.037  4.633  8.656  7.025  6.035 13.970  7.278 11.804  9.014 10.281 19.253 15.759 13.312 31.364 15.967
[16,]  5.113  3.559  6.888  6.216  6.109  7.010  7.278 12.951 11.316  7.980 15.466 13.515 13.625 15.726 15.967 29.159

Let us assume that we want to compute the reliability of the predicted breeding value for animal 1 in the trait WWG. Then we have to extract the element \(C_{11}^{22}\) from the matrix \(C^{22}\). This element corresponds to the element shown in the diagram below.

Hence the computation of reliability \(B_1\) for animal 1 for the trait WWG is done as

B1_WWG <- 1 - mat_Cinv22[1,1] / mat_G0[1,1]
B1_WWG
[1] 0.06976326

where \(var(u_i)\) is the genetic additive variance of the trait WWG and is extracted from the matrix \(G0\).

The computation of the reliability \(B_1\) for the trait PWG depends on the element \(C_{88}^{22}\), with that \(B_1\) for PWG is

B_1_PWG <- 1 - mat_Cinv22[9,9] / mat_G0[2,2]
B_1_PWG
[1] 0.1024554

For all animals and all traits, this is obtained by

n_nr_sol <- dim(mat_Cinv)[1]
vec_c22_diag <- diag(mat_Cinv)[(n_nr_sol - n_nr_trait * n_nr_animal + 1):n_nr_sol]
vec_rel <- 1 - vec_c22_diag / c(rep(mat_G0[1,1], n_nr_animal), rep(mat_G0[2,2], n_nr_animal))
vec_rel
 [1] 0.06976326 0.02018124 0.10535569 0.17476719 0.17294668 0.14242750 0.14424801 0.18578089 0.10245535 0.03080923 0.15501891 0.25687708 0.25338456 0.21241863
[15] 0.21591115 0.27101269

Appendix

Proof: According to the mixed-product property: \((A \otimes B)(C \otimes D) = AC \otimes BD\) \[I = RR^{-1} = (R_0 \otimes I_n)(R_0^{-1} \otimes I_n) = R_0R_0^{-1} \otimes I_nI_n = I\]

LS0tCnRpdGxlOiAiTXVsdGktVHJhaXQgQkxVUCIKb3V0cHV0OiBodG1sX25vdGVib29rCi0tLQoKYGBge3Igc2V0dXAsIGluY2x1ZGU9RkFMU0V9CiMgcm1kaGVscDo6c2hvd19rbml0X2hvb2tfY2FsbCgpCmtuaXRyOjprbml0X2hvb2tzJHNldChob29rX2NvbnZlcnRfb2RnID0gcm1kaGVscDo6aG9va19jb252ZXJ0X29kZykKYGBgCgojIERpc2NsYWltZXIKVGhpcyBpcyBub3RlYm9vayBleHBsYWlucyBpbiBtb3JlIGRldGFpbCBob3cgdG8gc29sdmUgYSBwcm9ibGVtIHdpdGggbXVsdGktdHJhaXQgQkxVUCBvZiBicmVlZGluZyB2YWx1ZXMuIFRoZSB0ZXJtcyAnbXVsdGktdHJhaXQnIGFuZCAnbXVsdGktdmFyaWF0ZScgYXJlIHVzZWQgYXMgc3lub255bXMuCgoKIyBEYXRhClRoZSBzYW1lIGRhdGFzZXQgYXMgaW4gUHJvYmxlbXMgMSBhbmQgMiBvZiBFeGVyY2lzZSA5IChodHRwczovL2NoYXJsb3R0ZS1uZ3MuZ2l0aHViLmlvL2xiZ2ZzMjAyMC9leC9sYmdmczIwMjBfdzEwX2V4MDkucGRmKSBpcyB1c2VkLiBUaGUgZGF0YXNldCBpcyBzaG93biBpbiB0aGUgZm9sbG93aW5nIHRhYmxlLiAKCmBgYHtyfQpuX25yX3RyYWl0IDwtIDIKbl9ucl9mb3VuZGVyIDwtIDMKbl9ucl9hbmltYWwgPC0gOApuX25yX29ic2VydmF0aW9uIDwtIG5fbnJfYW5pbWFsIC0gbl9ucl9mb3VuZGVyCnRibF9kYXRhX3NvbDEycDAxIDwtIHRpYmJsZTo6dGliYmxlKEFuaW1hbCA9IGMoKG5fbnJfZm91bmRlcisxKTpuX25yX2FuaW1hbCksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTZXggPSBjKCJNYWxlIiwgIkZlbWFsZSIsIkZlbWFsZSIsIk1hbGUiLCJNYWxlIiksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTaXJlID0gYygxLDMsMSw0LDMpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRGFtID0gYyhOQSwyLDIsNSw2KSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFdXRyA9IGMoNC41LDIuOSwzLjksMy41LDUuMCksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQV0cgPSBjKDYuOCw1LjAsNi44LDYuMCw3LjUpKQp0YmxfZGF0YV9zb2wxMnAwMQpgYGAKClRoZSB2YXJpYWJsZXMgZGVmaW5lZCBiZWZvcmUgdGhlIGRhdGEgaGF2ZSB0aGUgZm9sbG93aW5nIG1lYW5pbmcKCiogYG5fbnJfdHJhaXRgOiBudW1iZXIgb2YgdHJhaXRzIC0gdHJhaXRzIGFyZSBXV0cgYW5kIFBXRywgaGVuY2UgaXQgaXMgMgoqIGBuX25yX2ZvdW5kZXJgOiBudW1iZXIgb2YgZm91bmRlcnMgLSBhbmltYWxzIHdpdGhvdXQgcGFyZW50cyAKKiBgbl9ucl9hbmltYWxgOiB0b3RhbCBudW1iZXIgb2YgYW5pbWFscyBpbiB0aGUgcGVkaWdyZWUKKiBgbl9ucl9vYnNlcnZhdGlvbmA6IG51bWJlciBvZiBvYnNlcnZhdGlvbnMgZm9yIGEgc2luZ2xlIHRyYWl0CgoKIyBQYXJhbWV0ZXIKVGhlIHBhcmFtZXRlcnMgdGhhdCBhcmUgZ2l2ZW4gY29uc2lzdCBvZiB0aGUgZ2VuZXRpYyB2YXJpYW5jZS1jb3ZhcmlhbmNlIG1hdHJpeCAkR18wJCBiZXR3ZWVuIHRoZSB0d28gdHJhaXRzIFdXRyBhbmQgUFdHIGFuZCB0aGUgdmFyaWFuY2UtY292YXJpYW5jZSBtYXRyaXggJFJfMCQgYmV0d2VlbiB0aGUgdHJhaXRzLiBUaGVzZSBtYXRyaWNlcyBhcmUKCmBgYHtyfQoobWF0X0cwIDwtIG1hdHJpeChkYXRhID0gYygyMCwxOCwxOCw0MCksIG5yb3cgPSBuX25yX3RyYWl0LCBieXJvdyA9IFRSVUUpKQpgYGAKCmFuZCAKCmBgYHtyfQoobWF0X1IwIDwtIG1hdHJpeChkYXRhID0gYyg0MCwxMSwxMSwzMCksIG5yb3cgPSBuX25yX3RyYWl0LCBieXJvdyA9IFRSVUUpKQpgYGAKCgojIE1vZGVsClRoZSBtdWx0aS10cmFpdCBtb2RlbCBmb3IgZXN0aW1hdGluZyBmaXhlZCBlZmZlY3RzIGFuZCBmb3IgcHJlZGljdGluZyBicmVlZGluZyB2YWx1ZXMgbG9va3MgdGhlIHNhbWUgYXMgdGhlIG1vZGVsIGZvciBvbmx5IG9uZSB0cmFpdC4KCiQkeSA9IFhcYmV0YSArIFp1ICsgZSQkCgpBbHRob3VnaCwgdGhlIG1vZGVsIGxvb2tzIHRoZSBzYW1lLCB0aGUgY29tcG9uZW50cyBtdXN0IGJlIGFkYXB0ZWQgdG8gdGhlIHNpdHVhdGlvbiBvZiBtdWx0aXBsZSB0cmFpdHMuIFRoaXMgYWRhcHRhdGlvbiBpcyBkaWZmZXJlbnQgZm9yIHRoZSB2ZWN0b3JzICR5JCwgJFxiZXRhJCwgJHUkIGFuZCAkZSQgaW4gdGhlIG1vZGVsIGNvbXBhcmVkIHRvIHRoZSBtYXRyaWNlcyAkWCQgYW5kICRaJC4gCgoKIyMgVmVjdG9ycwpGb3IgdGhlIHZlY3RvcnMsIHRoZSB2ZXJzaW9ucyBvZiB0aGUgc2luZ2xlIHRyYWl0cyBhcmUganVzdCBhcHBlbmRlZCB0byBlYWNoIG90aGVyLiBGb3IgdGhlIHZlY3RvciAkeSQgdGhpcyBtZWFucyB0aGF0CgokJHkgPSBcYmVnaW57Ym1hdHJpeH0geV8xIFxcIHlfMiBcZW5ke2JtYXRyaXh9JCQKd2hlcmUgJHlfMSQgYXJlIGFsbCB0aGUgb2JzZXJ2YXRpb25zIGZvciB0aGUgdHJhaXQgJ1dXRycgYW5kICR5XzIkIGFyZSBhbGwgdGhlIG9ic2VydmF0aW9ucyBmb3IgdGhlIHRyYWl0ICdQV0cnLiBJbnNlcnRpbmcgdGhlIG51bWJlcnMgdGhpcyBtZWFucwoKYGBge3J9Cih5MSA8LSB0YmxfZGF0YV9zb2wxMnAwMSRXV0cpCmBgYAoKYW5kIAoKYGBge3J9Cih5MiA8LSB0YmxfZGF0YV9zb2wxMnAwMSRQV0cpCmBgYAoKYW5kIGFwcGVuZGluZyAkeV8yJCBhdCB0aGUgZW5kIG9mICR5XzEkIGxlYWRzIHRvCgpgYGB7cn0KKHkgPC0gYyh5MSx5MikpCmBgYAoKVGhlIHNhbWUgaXMgZG9uZSBmb3IgYWxsIHRoZSBvdGhlciB2ZWN0b3JzLiBCZWNhdXNlLCAnc2V4JyB3aXRoIHR3byBsZXZlbHMgJ2ZlbWFsZScgYW5kICdtYWxlJyBpcyB0aGUgb25seSBmaXhlZCBlZmZlY3QsIHRoZSB2ZWN0b3IgJFxiZXRhJCBmb3IgdGhlIG11bHRpLXRyYWl0IG1vZGVsIGxvb2tzIGFzIGZvbGxvd3MKCiQkXGJldGEgPSBcYmVnaW57Ym1hdHJpeH0gXGJldGFfe00sV1dHfSBcXFxiZXRhX3tGLFdXR30gXFxcYmV0YV97TSxQV0d9IFxcXGJldGFfe0YsUFdHfVxlbmR7Ym1hdHJpeH0kJCAKClRoZSB2ZWN0b3IgJHUkIG9mIGJyZWVkaW5nIHZhbHVlcyBjb250YWlucyBhcyBjb21wb25lbnRzIGFsbCBicmVlZGluZyB2YWx1ZXMgZm9yIGFsbCBhbmltYWxzIGluIHRoZSBwZWRpZ3JlZS4gCgokJHUgPSBcYmVnaW57Ym1hdHJpeH0gdV97V1dHfSBcXCB1X3tQV0d9IFxlbmR7Ym1hdHJpeH0kJAp3aGVyZSAkdV97V1dHfSQgaXMgdGhlIHZlY3RvciB3aXRoIGJyZWVkaW5nIHZhbHVlcyBmb3IgJ1dXRycgb2YgYWxsIGFuaW1hbHMgaW4gdGhlIHBlZGlncmVlIGFuZCAkdV97UFdHfSQgaXMgdGhlIHZlY3RvciB3aXRoIGJyZWVkaW5nIHZhbHVlcyBmb3IgJ1BXRycgb2YgYWxsIGFuaW1hbHMgaW4gdGhlIHBlZGlncmVlLgoKCiMjIE1hdHJpY2VzClRoZSBkaW1lbnNpb25zIG9mIHRoZSBkZXNpZ24gbWF0cmljZXMgbXVzdCBtYXRjaCB0aGUgbnVtYmVyIG9mIG9ic2VydmF0aW9ucyBhbmQgdGhlIGxlbmd0aCBvZiB0aGVpciBhc3NvY2lhdGVkIGVmZmVjdHMuIEhlbmNlIGZvciB0aGUgbWF0cml4ICRYJCB3aGljaCBoYXMgdGhlIHZlY3RvciAkXGJldGEkIG9mIGZpeGVkIGVmZmVjdHMgYXMgdGhlIGFzc29jaWF0ZWQgZWZmZWN0LCBpdCBtdXN0IGhhdmUgNCBjb2x1bW5zIGFzIHRoZWlyIGFyZSB0d28gZml4ZWQgZWZmZWN0IGxldmVscyBmb3IgZWFjaCBvZiB0aGUgdHdvIHRyYWl0cy4gCgpUaGUgbWF0cml4ICRYJCBpcyB0aGUgZGVzaWduIG1hdHJpeCBmb3IgdGhlIHZlY3RvciBvZiBmaXhlZCBlZmZlY3RzICRcYmV0YSQuIAoKJCRYID0gXGxlZnRbCiAgXGJlZ2lue2FycmF5fXtscn0KICBYXzEgICYgIDAgXFwKICAwICAgICYgIFhfMgogIFxlbmR7YXJyYXl9ClxyaWdodF0kJAoKd2hlcmUgJFhfMSQgYW5kICRYXzIkIGFyZSB0aGUgZGVzaWduIG1hdHJpeCBmb3IgdGhlIGZpeGVkIGVmZmVjdHMgZm9yIHRoZSBzaW5nbGUgdHJhaXRzICdXV0cnIGFuZCAnUFdHJy4gCgpgYGB7cn0KbWF0X1gxIDwtIG1hdHJpeChjKDEsIDAsCiAgICAgICAgICAgICAgICAgICAwLCAxLAogICAgICAgICAgICAgICAgICAgMCwgMSwKICAgICAgICAgICAgICAgICAgIDEsIDAsCiAgICAgICAgICAgICAgICAgICAxLCAwKSwgbmNvbCA9IG5fbnJfdHJhaXQsIGJ5cm93ID0gVFJVRSkKbWF0X1gyIDwtIG1hdF9YMQptYXRfWHplcm8gPC0gbWF0cml4KDAsIG5yb3cgPSBuX25yX29ic2VydmF0aW9uLCBuY29sID0gbl9ucl90cmFpdCkKKG1hdF9YIDwtIHJiaW5kKGNiaW5kKG1hdF9YMSwgbWF0X1h6ZXJvKSwgY2JpbmQobWF0X1h6ZXJvLCBtYXRfWDIpKSkKYGBgCgpUaGUgbWF0cml4ICRaJCBsaW5rcyBicmVlZGluZyB2YWx1ZXMgdG8gb2JzZXJ2YXRpb25zLiAKCmBgYHtyfQptYXRfWmZvdW5kZXIgPC0gbWF0cml4KDAsIG5yb3cgPSBuX25yX29ic2VydmF0aW9uLCBuY29sID0gbl9ucl9mb3VuZGVyKQptYXRfWjEgPC0gY2JpbmQobWF0X1pmb3VuZGVyLCBkaWFnKDEsIG5yb3cgPSBuX25yX29ic2VydmF0aW9uKSkKbWF0X1oyIDwtIG1hdF9aMQptYXRfWnplcm8gPC0gbWF0cml4KDAsIG5yb3cgPSBuX25yX29ic2VydmF0aW9uLCBuY29sID0gbl9ucl9hbmltYWwpCihtYXRfWiA8LSByYmluZChjYmluZChtYXRfWjEsIG1hdF9aemVybyksIGNiaW5kKG1hdF9aemVybywgbWF0X1oyKSkpCmBgYAoKCiMgU29sdXRpb25zClNvbHV0aW9ucyBhcmUgb2J0YWluZWQgYnkgbWl4ZWQtbW9kZWwgZXF1YXRpb25zLiAKCiQkClxiZWdpbntibWF0cml4fQpYXlRSXnstMX1YICAmICBYXlRSXnstMX1aIFxcClpeVFJeey0xfVggICYgIFpeVFJeey0xfVogKyBHXnstMX0KXGVuZHtibWF0cml4fQpcYmVnaW57Ym1hdHJpeH0KXGhhdHtcYmV0YX0gXFwKXGhhdHt1fQpcZW5ke2JtYXRyaXh9Cj0KXGJlZ2lue2JtYXRyaXh9ClheVFJeey0xfXkgXFwKWl5UUl57LTF9eQpcZW5ke2JtYXRyaXh9CiQkCgpJbiB0aGUgbXVsdGktdHJhaXQgY2FzZSwgd2UgaGF2ZSB0byB1c2UgdGhlIGdlbmVyYWwgZm9ybSBvZiBtaXhlZC1tb2RlbCBlcXVhdGlvbnMsIGJlY2F1c2UgdGhlIHZhcmlhbmNlLWNvdmFyaWFuY2UgbWF0cml4IG9mIHRoZSByYW5kb20gZXJyb3IgdGVybXMgZG9lcyBub3QgaGF2ZSB0aGUgc2ltcGxlIGZvcm0gb2YgJFIgPSBJICpcc2lnbWFfZV4yJC4gSW4gdGhlIG11bHRpLXRyYWl0IGNhc2UgdGhlIHZhcmlhbmNlLWNvdmFyaWFuY2UgbWF0cml4ICRSID0gUl8wIFxvdGltZXMgSV9uJCB3aGVyZSAkbiQgaXMgdGhlIG51bWJlciBvZiBvYnNlcnZhdGlvbnMgZm9yIGEgc2luZ2xlIHRyYWl0LiBGb3IgdGhlIG1peGVkIG1vZGVsIGVxdWF0aW9ucywgdGhlIGludmVyc2UgJFJeey0xfSQgb2YgJFIkIGlzIG5lZWRlZC4gVGhpcyBjYW4gYmUgY29tcHV0ZWQgYXMgCgokJFJeey0xfSA9IFJfMF57LTF9IFxvdGltZXMgSV9uJCQKCgpgYGB7cn0KbWF0X1IwaW52IDwtIHNvbHZlKG1hdF9SMCkKKG1hdF9SaW52IDwtIG1hdF9SMGludiAleCUgZGlhZygxLCBucm93ID0gbl9ucl9vYnNlcnZhdGlvbikpCmBgYAoKVGhlIG1hdHJpeCAkR157LTF9JCBpbiB0aGUgbWl4ZWQgbW9kZWwgZXF1YXRpb25zIGNvcnJlc3BvbmRzIHRvIHRoZSBpbnZlcnNlIG9mIHRoZSB2YXJpYW5jZS1jb3ZhcmlhbmNlIG1hdHJpeCBvZiBhbGwgYnJlZWRpbmcgdmFsdWVzIGFuZCBpcyBjb21wdXRlZCBhcyAKCiQkR157LTF9ID0gR18wXnstMX0gXG90aW1lcyBBXnstMX0kJApgYGB7cn0KbWF0X0cwaW52IDwtIHNvbHZlKG1hdF9HMCkKKHBlZCA8LSBwZWRpZ3JlZW1tOjpwZWRpZ3JlZShzaXJlID0gYyhyZXAoTkEsIG5fbnJfZm91bmRlciksIHRibF9kYXRhX3NvbDEycDAxJFNpcmUpLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkYW0gPSBjKHJlcChOQSwgbl9ucl9mb3VuZGVyKSwgdGJsX2RhdGFfc29sMTJwMDEkRGFtKSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbGFiZWwgPSBhcy5jaGFyYWN0ZXIoMTpuX25yX2FuaW1hbCkpKQptYXRfQWludiA8LSBhcy5tYXRyaXgocGVkaWdyZWVtbTo6Z2V0QUludihwZWQgPSBwZWQpKQptYXRfR2ludiA8LSBtYXRfRzBpbnYgJXglIG1hdF9BaW52CiNyb3VuZChtYXRfR2ludiwgZGlnaXRzID0gNCkKYGBgCgoKIyMgQ29lZmZpY2llbnQgTWF0cml4ICRDJApUaGUgc2V0dXAgb2YgdGhlIG1peGVkIG1vZGVsIGVxdWF0aW9ucyBzdGFydHMgd2l0aCB0aGUgY29lZmZpY2llbnQgbWF0cml4ICRDJAoKJCQKQyA9IFxiZWdpbntibWF0cml4fQpYXlRSXnstMX1YICAmICBYXlRSXnstMX1aIFxcClpeVFJeey0xfVggICYgIFpeVFJeey0xfVogKyBHXnstMX0KXGVuZHtibWF0cml4fQo9ClxiZWdpbntibWF0cml4fQpDX3sxMX0gICYgIENfezEyfSAgXFwKQ197MjF9ICAmICBDX3syMn0KXGVuZHtibWF0cml4fQokJAp3aXRoIAoKJCRDX3sxMX0gPSAgWF5UUl57LTF9WCQkCiQkQ197MTJ9ID0gWF5UUl57LTF9WiQkCiQkQ197MjF9ID0gWl5UUl57LTF9WCQkCiQkQ197MjJ9ID0gWl5UUl57LTF9WiArIEdeey0xfSQkCkJlY2F1c2UgdGhlIG1hdHJpeCBDIGlzIHN5bW1ldHJpYywgJENfezIxfSA9IENfezEyfV5UJAoKVGhlIGNvZWZmaWNpZW50IG1hdHJpeCAkQyQgY2FuIGJlIGNvbnN0cnVjdGVkIGFzCgpgYGB7cn0KbWF0X0MxMSA8LSB0KG1hdF9YKSAlKiUgbWF0X1JpbnYgJSolIG1hdF9YCm1hdF9DMTIgPC0gdChtYXRfWCkgJSolIG1hdF9SaW52ICUqJSBtYXRfWgptYXRfQzIxIDwtIHQobWF0X1opICUqJSBtYXRfUmludiAlKiUgbWF0X1gKbWF0X0MyMiA8LSB0KG1hdF9aKSAlKiUgbWF0X1JpbnYgJSolIG1hdF9aICsgbWF0X0dpbnYKbWF0X0MgPC0gcmJpbmQoY2JpbmQobWF0X0MxMSwgbWF0X0MxMiksIGNiaW5kKG1hdF9DMjEsIG1hdF9DMjIpKQojcm91bmQobWF0X0MsIGRpZ2l0cyA9IDMpCmBgYAoKCiMjIFJpZ2h0LUhhbmQgU2lkZSAkciQKVGhlIHZlY3RvciAkciQgY29uc2lzdHMgb2YgdGhlIHJpZ2h0LWhhbmQgc2lkZSBvZiB0aGUgbWl4ZWQtbW9kZWwgZXF1YXRpb25zLiAKCiQkCnIgPSBcYmVnaW57Ym1hdHJpeH0Kcl8xIFxcIApyXzIKXGVuZHtibWF0cml4fQo9ClxiZWdpbntibWF0cml4fQpYXlRSXnstMX15IFxcClpeVFJeey0xfXkKXGVuZHtibWF0cml4fQokJApgYGB7cn0KKHZlY19yIDwtIHJiaW5kKHQobWF0X1gpICUqJSBtYXRfUmludiAlKiUgeSwgdChtYXRfWikgJSolIG1hdF9SaW52ICUqJSB5KSkKYGBgCgoKIyMgU29sdXRpb24gVmVjdG9yClRoZSBzb2x1dGlvbiB2ZWN0b3IgJHMkIGlzCgokJHMgPSBcYmVnaW57Ym1hdHJpeH0KXGhhdHtcYmV0YX0gXFwKXGhhdHt1fQpcZW5ke2JtYXRyaXh9ID0gQ157LTF9ciAKJCQKCmBgYHtyfQpzIDwtIHNvbHZlKG1hdF9DLCB2ZWNfcikKcwpgYGAKCgojIFJlbGlhYmlsaXRpZXMKVGhlIHJlbGlhYmlsaXR5ICRCX2kkIG9mIGEgcHJlZGljdGVkIGJyZWVkaW5nIHZhbHVlICRcaGF0e3V9X2kkIG9mIGFuaW1hbCAkaSQgZm9yIGEgZ2l2ZW4gdHJhaXQgaXMgY29tcHV0ZWQgYWNjb3JkaW5nIHRvIGZvcm11bGEgKDcuNCkgb24gcGFnZSA4MCBvZiB0aGUgY291cnNlIG5vdGVzCgokJApCX2kgPSByX3t1X2ksXGhhdHt1fV9pfV4yID0gMSAtIFxmcmFje0Nfe2lpfV57MjJ9fXt2YXIodV9pKX0KJCQKd2hlcmUgJENfe2lpfV57MjJ9JCBpcyB0aGUgaS10aCBkaWFnbm9hbCBlbGVtZW50IG9mIHRoZSBtYXRyaXggJENeezIyfSQgYW5kICR2YXIodV9pKSQgaXMgdGhlIGdlbmV0aWMgYWRkaXRpdmUgdmFyaWFuY2Ugb2YgdGhlIHRyYWl0IGZvciB3aGljaCB3ZSB3YW50IHRvIGNvbXB1dGUgJEJfaSQKClRoZSBtYXRyaXggJENeezIyfSQgaXMgb2J0YWluZWQgZnJvbSB0aGUgaW52ZXJzZSAoJENeey0xfSQpIG9mIHRoZSBjb2VmZmllbnQgbWF0cml4ICRDJCBvZiB0aGUgbWl4ZWQgbW9kZWwgZXF1YXRpb25zLiAKCiQkCkNeey0xfSAKPSAKXGJlZ2lue2JtYXRyaXh9ClheVFJeey0xfVggICYgIFheVFJeey0xfVogXFwKWl5UUl57LTF9WCAgJiAgWl5UUl57LTF9WiArIEdeey0xfQpcZW5ke2JtYXRyaXh9XnstMX0KPSAKXGJlZ2lue2JtYXRyaXh9CkNeezExfSAgJiAgQ157MTJ9IFxcCkNeezIxfSAgJiAgQ157MjJ9ClxlbmR7Ym1hdHJpeH0KJCQKCkZvciBvdXIgZGF0YSB0aGUgbWF0cml4ICRDXnsyMn0kIGlzIGV4dHJhY3RlZCBmcm9tICRDXnstMX0kIGJ5IHRha2luZyB0aGUgc3ViLW1hdHJpeCBpbiB0aGUgbG93ZXIgcmlnaHQgY29ybmVyIHdpdGggZGltZW5zaW9ucyAkMTYgXHRpbWVzIDE2JC4gVGhlIGRpbWVuc2lvbiBpcyBnaXZlbiBieSB0aGUgbnVtYmVyIG9mIHRyYWl0cyAoMikgdGltZSB0aGUgbnVtYmVyIG9mIGFuaW1hbHMgd2l0aCBwcmVkaWN0ZWQgYnJlZWRpbmcgdmFsdWVzICg4KS4KCmBgYHtyfQptYXRfQ2ludiA8LSBzb2x2ZShtYXRfQykKbWF0X0NpbnYyMiA8LSBtYXRfQ2ludls1OjIwLDU6MjBdCnJvdW5kKG1hdF9DaW52MjIsIGRpZ2l0cyA9IDMpCmBgYAoKTGV0IHVzIGFzc3VtZSB0aGF0IHdlIHdhbnQgdG8gY29tcHV0ZSB0aGUgcmVsaWFiaWxpdHkgb2YgdGhlIHByZWRpY3RlZCBicmVlZGluZyB2YWx1ZSBmb3IgYW5pbWFsIDEgaW4gdGhlIHRyYWl0IFdXRy4gVGhlbiB3ZSBoYXZlIHRvIGV4dHJhY3QgdGhlIGVsZW1lbnQgJENfezExfV57MjJ9JCBmcm9tIHRoZSBtYXRyaXggJENeezIyfSQuIFRoaXMgZWxlbWVudCBjb3JyZXNwb25kcyB0byB0aGUgZWxlbWVudCBzaG93biBpbiB0aGUgZGlhZ3JhbSBiZWxvdy4gCgpgYGB7ciByZWwtY29tcCwgZWNobz1GQUxTRSwgaG9va19jb252ZXJ0X29kZz1UUlVFLCBmaWdfcGF0aD0ib2RnIn0KI3JtZGhlbHA6OnVzZV9vZGdfZ3JhcGhpYyhwc19wYXRoID0gJ29kZy9yZWwtY29tcC5vZGcnKQprbml0cjo6aW5jbHVkZV9ncmFwaGljcyhwYXRoID0gIm9kZy9yZWwtY29tcC5wbmciKQoKYGBgCgpIZW5jZSB0aGUgY29tcHV0YXRpb24gb2YgcmVsaWFiaWxpdHkgJEJfMSQgZm9yIGFuaW1hbCAxIGZvciB0aGUgdHJhaXQgV1dHIGlzIGRvbmUgYXMgCgpgYGB7cn0KQjFfV1dHIDwtIDEgLSBtYXRfQ2ludjIyWzEsMV0gLyBtYXRfRzBbMSwxXQpCMV9XV0cKYGBgCgp3aGVyZSAkdmFyKHVfaSkkIGlzIHRoZSBnZW5ldGljIGFkZGl0aXZlIHZhcmlhbmNlIG9mIHRoZSB0cmFpdCBXV0cgYW5kIGlzIGV4dHJhY3RlZCBmcm9tIHRoZSBtYXRyaXggJEcwJC4KClRoZSBjb21wdXRhdGlvbiBvZiB0aGUgcmVsaWFiaWxpdHkgJEJfMSQgZm9yIHRoZSB0cmFpdCBQV0cgZGVwZW5kcyBvbiB0aGUgZWxlbWVudCAkQ197ODh9XnsyMn0kLCB3aXRoIHRoYXQgJEJfMSQgZm9yIFBXRyBpcwoKYGBge3J9CkJfMV9QV0cgPC0gMSAtIG1hdF9DaW52MjJbOSw5XSAvIG1hdF9HMFsyLDJdCkJfMV9QV0cKYGBgCgpGb3IgYWxsIGFuaW1hbHMgYW5kIGFsbCB0cmFpdHMsIHRoaXMgaXMgb2J0YWluZWQgYnkKCmBgYHtyfQpuX25yX3NvbCA8LSBkaW0obWF0X0NpbnYpWzFdCnZlY19jMjJfZGlhZyA8LSBkaWFnKG1hdF9DaW52KVsobl9ucl9zb2wgLSBuX25yX3RyYWl0ICogbl9ucl9hbmltYWwgKyAxKTpuX25yX3NvbF0KdmVjX3JlbCA8LSAxIC0gdmVjX2MyMl9kaWFnIC8gYyhyZXAobWF0X0cwWzEsMV0sIG5fbnJfYW5pbWFsKSwgcmVwKG1hdF9HMFsyLDJdLCBuX25yX2FuaW1hbCkpCnZlY19yZWwKYGBgCgoKIyBBcHBlbmRpeApQcm9vZjoKQWNjb3JkaW5nIHRvIHRoZSBtaXhlZC1wcm9kdWN0IHByb3BlcnR5OiAkKEEgXG90aW1lcyBCKShDIFxvdGltZXMgRCkgPSBBQyBcb3RpbWVzIEJEJAokJEkgPSBSUl57LTF9ID0gKFJfMCBcb3RpbWVzIElfbikoUl8wXnstMX0gXG90aW1lcyBJX24pID0gUl8wUl8wXnstMX0gXG90aW1lcyBJX25JX24gPSBJJCQKIAo=