Replication comment: Running the notebook within the replication docker ensures that results perfectly replicate. Otherwise, results might differ depending on which package versions you use.

This notebook runs the application described in Section 5.2 of Knaus (2024). The first part replicates the results presented in the paper, the second part provides supplementary information

1. Replication of paper results

Getting started

First, load packages and set the seed:

if (!require("OutcomeWeights")) install.packages("OutcomeWeights", dependencies = TRUE); library(OutcomeWeights)
if (!require("hdm")) install.packages("hdm", dependencies = TRUE); library(hdm)
if (!require("grf")) install.packages("grf", dependencies = TRUE); library(grf)
if (!require("tidyverse")) install.packages("tidyverse", dependencies = TRUE); library(tidyverse)
if (!require("viridis")) install.packages("viridis", dependencies = TRUE); library(viridis)
if (!require("reshape2")) install.packages("reshape2", dependencies = TRUE); library(reshape2)
if (!require("ggridges")) install.packages("ggridges", dependencies = TRUE); library(ggridges)

set.seed(1234)

Next, load the data. Here we use the 401(k) data of the hdm package. However, you can adapt the following code chunk to load any suitable data of your choice. Just make sure to call the treatment D, covariates X, and instrument Z. The rest of the notebook should run without further modifications.

data(pension) # Find variable description if you type ?pension in console

# Treatment
D = pension$p401
# Instrument
Z = pension$e401
# Outcome
Y = pension$net_tfa
# Controls
X = model.matrix(~ 0 + age + db + educ + fsize + hown + inc + male + marr + pira + twoearn, data = pension)
var_nm = c("Age","Benefit pension","Education","Family size","Home owner","Income","Male","Married","IRA","Two earners")
colnames(X) = var_nm

Get outcome model and smoother matrix

The grf package does not save the nuisance parameter models. Thus, we could not retrieve the required smoother matrices after running causal_forest and instrumental_forest below. Therefore, we estimate the outcome nuisance model externally to pass the nuisance parameters later to the functions. As described in Section 5.2 of the paper, we run a default and a tuned version:

### Externally calculate outcome nuisance
rf_Y.hat_default = regression_forest(X,Y)
rf_Y.hat_tuned = regression_forest(X,Y,tune.parameters = "all")
Y.hat_default = predict(rf_Y.hat_default)$predictions
Y.hat_tuned = predict(rf_Y.hat_tuned)$predictions

Then, we extract the smoother matrix using get_forest_weights():

# And get smoother matrices
S_default = get_forest_weights(rf_Y.hat_default)
S_tuned = get_forest_weights(rf_Y.hat_tuned)

For illustration, check that random forest is an affine smoother by summing all smoother vectors and checking whether they sum to one:

cat("RF affine smoother?", 
    all.equal(rowSums(as.matrix(S_default)),
      rep(1,length(Y))
    ))
RF affine smoother? TRUE

Causal forest

Run causal forest with the externally estimated outcome nuisance:

# Run CF with the pre-specified outcome nuisance 
cf_default = causal_forest(X,Y,D,Y.hat=Y.hat_default)
cf_tuned = causal_forest(X,Y,D,Y.hat=Y.hat_tuned,tune.parameters = "all")

Get the out-of-bag CATEs:

cates_default = predict(cf_default)$predictions
cates_tuned = predict(cf_tuned)$predictions

Use the new get_outcome_weights() method, which requires to pass the externally estimated outcome smoother matrix:

omega_cf_default = get_outcome_weights(cf_default, S = S_default)
omega_cf_tuned = get_outcome_weights(cf_tuned, S = S_tuned)

Observe that the outcome weights recover the grf package output:

cat("ω'Y replicates CATE point estimates (default)?", 
    all.equal(as.numeric(omega_cf_default$omega %*% Y),
      as.numeric(cates_default)
    ))
ω'Y replicates CATE point estimates (default)? TRUE
cat("\nω'Y replicates CATE point estimates (tuned)?", 
    all.equal(as.numeric(omega_cf_tuned$omega %*% Y),
      as.numeric(cates_tuned)
    ))

ω'Y replicates CATE point estimates (tuned)? TRUE

Now calculate the absolute standardized mean differences and plot them for each CATE and variables (Figure 3 in paper):

cb_cate_default = standardized_mean_differences(X,D,omega_cf_default$omega,X)
cb_cate_tuned = standardized_mean_differences(X,D,omega_cf_tuned$omega,X)

smd_default = t(abs(cb_cate_default[,3,]))
smd_tuned = t(abs(cb_cate_tuned[,3,]))

# Melt the smd_default matrix to long format
df_default_long = melt(smd_default)
df_default_long$Group = "smd_default"  # Add a group identifier

# Melt the smd_tuned matrix to long format
df_tuned_long = melt(smd_tuned)
df_tuned_long$Group = "smd_tuned"  # Add a group identifier

# Combine the two data frames
df_long = rbind(df_default_long, df_tuned_long)

# Rename the columns for clarity
colnames(df_long) = c("Row", "Variable", "Value", "Group")

# Create the ggplot
figure3 = ggplot(df_long, aes(x = factor(Variable, levels = rev(unique(Variable))), y = Value, fill = Group)) +
  geom_boxplot(position = position_dodge(width = 0.8)) +
  labs(x = element_blank(), y = "Absolute Standardized Mean Differences") +
  scale_fill_manual(values = viridis(2),
                    name = element_blank(),
                    labels = c("default", "tuned")) +
  theme_minimal() +
  geom_hline(yintercept = 0, linetype = "solid", color = "black") + 
  coord_flip()
figure3

2. Supplementary results

Instrumental forest

Run instrumental forest with the externally estimated outcome nuisance:

# Run IF with the pre-specified outcome nuisance
ivf_default = instrumental_forest(X,Y,D,Z,Y.hat=Y.hat_default)
ivf_tuned = instrumental_forest(X,Y,D,Z,Y.hat=Y.hat_tuned,tune.parameters = "all")

Get the out-of-bag CLATEs:

clates_default = predict(ivf_default)$predictions
clates_tuned = predict(ivf_tuned)$predictions

Use the new get_outcome_weights() method, which requires to pass the externally estimated outcome smoother matrix:

omega_if_default = get_outcome_weights(ivf_default, S = S_default)
omega_if_tuned = get_outcome_weights(ivf_tuned, S = S_tuned)

Observe that the outcome weights recover the grf package output:

cat("ω'Y replicates CLATE point estimates (default)?", 
    all.equal(as.numeric(omega_if_default$omega %*% Y),
      as.numeric(clates_default)
    ))
ω'Y replicates CLATE point estimates (default)? TRUE
cat("\nω'Y replicates CLATE point estimates (tuned)?", 
    all.equal(as.numeric(omega_if_tuned$omega %*% Y),
      as.numeric(clates_tuned)
    ))

ω'Y replicates CLATE point estimates (tuned)? TRUE

Now calculate the absolute standardized mean differences and plot them for each CLATE and variables:

cb_clate_default = standardized_mean_differences(X,D,omega_if_default$omega,X)
cb_clate_tuned = standardized_mean_differences(X,D,omega_if_tuned$omega,X)

smd_default = t(abs(cb_clate_default[,3,]))
smd_tuned = t(abs(cb_clate_tuned[,3,]))

# Melt the smd_default matrix to long format
df_default_long = melt(smd_default)
df_default_long$Group = "smd_default"  # Add a group identifier

# Melt the smd_tuned matrix to long format
df_tuned_long = melt(smd_tuned)
df_tuned_long$Group = "smd_tuned"  # Add a group identifier

# Combine the two data frames
df_long = rbind(df_default_long, df_tuned_long)

# Rename the columns for clarity
colnames(df_long) = c("Row", "Variable", "Value", "Group")

# Create the ggplot
ggplot(df_long, aes(x = factor(Variable, levels = rev(unique(Variable))), y = Value, fill = Group)) +
  geom_boxplot(position = position_dodge(width = 0.8)) +
  labs(x = element_blank(), y = "Absolute Standardized Mean Differences") +
  scale_fill_manual(values = viridis(2),
                    name = element_blank(),
                    labels = c("default", "tuned")) +
  theme_minimal() +
  geom_hline(yintercept = 0, linetype = "solid", color = "black") + 
  coord_flip()

Plot results

Here we observe that not only the default effects show much worse balancing, but they also lead to implausibly high variance in their estimates.

data = data.frame(
  value = c(cates_default, cates_tuned,clates_default, clates_tuned),
  category = rep(c("grf CATEs default", "grf CATEs tuned","grf CLATEs default", "grf CLATEs tuned"), each = length(cates_default))
)

ggplot(data, aes(y = category, x = value, fill = category)) +
  geom_boxplot(alpha = 0.7) +
  geom_vline(xintercept = 0, color = "black", linetype = "solid") +
  labs(
    x = "Estimate",
    y = "Estimator/Implementation"
  ) +
  theme_minimal() +
  theme(legend.position = "none")

# Create the ridge plot
ggplot(data, aes(x = value, y = category, fill = category)) +
  geom_density_ridges(alpha = 0.7, scale = 1) +
  labs(
    x = "Estimate",
    y = "Estimator/Implementation"
  ) +
  theme_minimal() +
  theme(legend.position = "none")

Summarize weights

Finally, we summarize the weights vectors for each C(L)ATE plotting different descriptives:

  • Minimum weight

  • Maximum weight

  • % Negative

  • Sum largest 10%

  • Sum of weights

  • Sum of absolute weights

Especially the sum of weights is interesting in light of the paper. It shows that tuning also produces weight sums closer to one in this data set. However, also negative weights are fewer and less pronounced for the tuned version:

summary_weights_cf_default = summary(omega_cf_default, quiet = TRUE)
summary_weights_cf_tuned = summary(omega_cf_tuned, quiet = TRUE)
summary_weights_if_default = summary(omega_if_default, quiet = TRUE)
summary_weights_if_tuned = summary(omega_if_tuned, quiet = TRUE)

for (i in 1:dim(summary_weights_if_tuned)[3]) {
  # Extract untreated and treated weights for each group
  sum_weights_cf_default_untreated = summary_weights_cf_default[1,,i]
  sum_weights_cf_default_treated = summary_weights_cf_default[2,,i]
  
  sum_weights_cf_tuned_untreated = summary_weights_cf_tuned[1,,i]
  sum_weights_cf_tuned_treated = summary_weights_cf_tuned[2,,i]
  
  sum_weights_if_default_untreated = summary_weights_if_default[1,,i]
  sum_weights_if_default_treated = summary_weights_if_default[2,,i]
  
  sum_weights_if_tuned_untreated = summary_weights_if_tuned[1,,i]
  sum_weights_if_tuned_treated = summary_weights_if_tuned[2,,i]
  
  # Combine all vectors into a single data frame, with a new 'Treatment' column
  df_weights <- data.frame(
    Value = c(
      sum_weights_cf_default_untreated, sum_weights_cf_default_treated,
      sum_weights_cf_tuned_untreated, sum_weights_cf_tuned_treated,
      sum_weights_if_default_untreated, sum_weights_if_default_treated,
      sum_weights_if_tuned_untreated, sum_weights_if_tuned_treated
    ),
    Group = factor(c(
      rep("CF default", length(sum_weights_cf_default_untreated) + length(sum_weights_cf_default_treated)),
      rep("CF tuned", length(sum_weights_cf_tuned_untreated) + length(sum_weights_cf_tuned_treated)),
      rep("IF default", length(sum_weights_if_default_untreated) + length(sum_weights_if_default_treated)),
      rep("IF tuned", length(sum_weights_if_tuned_untreated) + length(sum_weights_if_tuned_treated))
    )),
    Treatment = factor(c(
      rep("Untreated", length(sum_weights_cf_default_untreated)),
      rep("Treated", length(sum_weights_cf_default_treated)),
      rep("Untreated", length(sum_weights_cf_tuned_untreated)),
      rep("Treated", length(sum_weights_cf_tuned_treated)),
      rep("Untreated", length(sum_weights_if_default_untreated)),
      rep("Treated", length(sum_weights_if_default_treated)),
      rep("Untreated", length(sum_weights_if_tuned_untreated)),
      rep("Treated", length(sum_weights_if_tuned_treated))
    ))
  )
  
  # Plot with ggplot and ggridges
  g <- ggplot(df_weights, aes(x = Value, y = Group, fill = Treatment)) +
    geom_boxplot(position = position_dodge(width = 0.75)) +
    labs(x = dimnames(summary_weights_cf_tuned)[[3]][i], y = NULL) +
    theme_minimal() +
    theme(legend.position = "bottom")  # Place legend at the bottom
  print(g)
}

LS0tCnRpdGxlOiAiVHJlYXRtZW50IEVmZmVjdCBFc3RpbWF0b3JzIGFzIFdlaWdodGVkIE91dGNvbWVzIgpzdWJ0aXRsZTogIkFwcGxpY2F0aW9uIDQwMShrKSAtIGhldGVyb2dlbmVvdXMgZWZmZWN0cyIKYXV0aG9yOiAiTWljaGFlbCBDLiBLbmF1cyIKZGF0ZTogIjExLzI0IgpvdXRwdXQ6IAogIGh0bWxfbm90ZWJvb2s6CiAgICB0b2M6IHRydWUKICAgIHRvY19mbG9hdDogdHJ1ZQogICAgY29kZV9mb2xkaW5nOiBzaG93Ci0tLQoKKlJlcGxpY2F0aW9uIGNvbW1lbnQ6IFJ1bm5pbmcgdGhlIG5vdGVib29rIHdpdGhpbiB0aGUgcmVwbGljYXRpb24gZG9ja2VyIGVuc3VyZXMgdGhhdCByZXN1bHRzIHBlcmZlY3RseSByZXBsaWNhdGUuIE90aGVyd2lzZSwgcmVzdWx0cyBtaWdodCBkaWZmZXIgZGVwZW5kaW5nIG9uIHdoaWNoIHBhY2thZ2UgdmVyc2lvbnMgeW91IHVzZS4qCgoKVGhpcyBub3RlYm9vayBydW5zIHRoZSBhcHBsaWNhdGlvbiBkZXNjcmliZWQgaW4gU2VjdGlvbiA1LjIgb2YgW0tuYXVzICgyMDI0KV0oaHR0cHM6Ly9hcnhpdi5vcmcvYWJzLzI0MTEuMTE1NTkpLiBUaGUgZmlyc3QgcGFydCByZXBsaWNhdGVzIHRoZSByZXN1bHRzIHByZXNlbnRlZCBpbiB0aGUgcGFwZXIsIHRoZSBzZWNvbmQgcGFydCBwcm92aWRlcyBzdXBwbGVtZW50YXJ5IGluZm9ybWF0aW9uCgojIDEuIFJlcGxpY2F0aW9uIG9mIHBhcGVyIHJlc3VsdHMKCiMjIEdldHRpbmcgc3RhcnRlZAoKRmlyc3QsIGxvYWQgcGFja2FnZXMgYW5kIHNldCB0aGUgc2VlZDoKCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmc9RkFMU0V9CmlmICghcmVxdWlyZSgiT3V0Y29tZVdlaWdodHMiKSkgaW5zdGFsbC5wYWNrYWdlcygiT3V0Y29tZVdlaWdodHMiLCBkZXBlbmRlbmNpZXMgPSBUUlVFKTsgbGlicmFyeShPdXRjb21lV2VpZ2h0cykKaWYgKCFyZXF1aXJlKCJoZG0iKSkgaW5zdGFsbC5wYWNrYWdlcygiaGRtIiwgZGVwZW5kZW5jaWVzID0gVFJVRSk7IGxpYnJhcnkoaGRtKQppZiAoIXJlcXVpcmUoImdyZiIpKSBpbnN0YWxsLnBhY2thZ2VzKCJncmYiLCBkZXBlbmRlbmNpZXMgPSBUUlVFKTsgbGlicmFyeShncmYpCmlmICghcmVxdWlyZSgidGlkeXZlcnNlIikpIGluc3RhbGwucGFja2FnZXMoInRpZHl2ZXJzZSIsIGRlcGVuZGVuY2llcyA9IFRSVUUpOyBsaWJyYXJ5KHRpZHl2ZXJzZSkKaWYgKCFyZXF1aXJlKCJ2aXJpZGlzIikpIGluc3RhbGwucGFja2FnZXMoInZpcmlkaXMiLCBkZXBlbmRlbmNpZXMgPSBUUlVFKTsgbGlicmFyeSh2aXJpZGlzKQppZiAoIXJlcXVpcmUoInJlc2hhcGUyIikpIGluc3RhbGwucGFja2FnZXMoInJlc2hhcGUyIiwgZGVwZW5kZW5jaWVzID0gVFJVRSk7IGxpYnJhcnkocmVzaGFwZTIpCmlmICghcmVxdWlyZSgiZ2dyaWRnZXMiKSkgaW5zdGFsbC5wYWNrYWdlcygiZ2dyaWRnZXMiLCBkZXBlbmRlbmNpZXMgPSBUUlVFKTsgbGlicmFyeShnZ3JpZGdlcykKCnNldC5zZWVkKDEyMzQpCmBgYAoKTmV4dCwgbG9hZCB0aGUgZGF0YS4gSGVyZSB3ZSB1c2UgdGhlIDQwMShrKSBkYXRhIG9mIHRoZSBgaGRtYCBwYWNrYWdlLiBIb3dldmVyLCB5b3UgY2FuIGFkYXB0IHRoZSBmb2xsb3dpbmcgY29kZSBjaHVuayB0byBsb2FkIGFueSBzdWl0YWJsZSBkYXRhIG9mIHlvdXIgY2hvaWNlLiBKdXN0IG1ha2Ugc3VyZSB0byBjYWxsIHRoZSB0cmVhdG1lbnQgYERgLCBjb3ZhcmlhdGVzIGBYYCwgYW5kIGluc3RydW1lbnQgYFpgLiBUaGUgcmVzdCBvZiB0aGUgbm90ZWJvb2sgc2hvdWxkIHJ1biB3aXRob3V0IGZ1cnRoZXIgbW9kaWZpY2F0aW9ucy4KCmBgYHtyfQpkYXRhKHBlbnNpb24pICMgRmluZCB2YXJpYWJsZSBkZXNjcmlwdGlvbiBpZiB5b3UgdHlwZSA/cGVuc2lvbiBpbiBjb25zb2xlCgojIFRyZWF0bWVudApEID0gcGVuc2lvbiRwNDAxCiMgSW5zdHJ1bWVudApaID0gcGVuc2lvbiRlNDAxCiMgT3V0Y29tZQpZID0gcGVuc2lvbiRuZXRfdGZhCiMgQ29udHJvbHMKWCA9IG1vZGVsLm1hdHJpeCh+IDAgKyBhZ2UgKyBkYiArIGVkdWMgKyBmc2l6ZSArIGhvd24gKyBpbmMgKyBtYWxlICsgbWFyciArIHBpcmEgKyB0d29lYXJuLCBkYXRhID0gcGVuc2lvbikKdmFyX25tID0gYygiQWdlIiwiQmVuZWZpdCBwZW5zaW9uIiwiRWR1Y2F0aW9uIiwiRmFtaWx5IHNpemUiLCJIb21lIG93bmVyIiwiSW5jb21lIiwiTWFsZSIsIk1hcnJpZWQiLCJJUkEiLCJUd28gZWFybmVycyIpCmNvbG5hbWVzKFgpID0gdmFyX25tCmBgYAoKCiMjIEdldCBvdXRjb21lIG1vZGVsIGFuZCBzbW9vdGhlciBtYXRyaXgKClRoZSBgZ3JmYCBwYWNrYWdlIGRvZXMgbm90IHNhdmUgdGhlIG51aXNhbmNlIHBhcmFtZXRlciBtb2RlbHMuIFRodXMsIHdlIGNvdWxkIG5vdCByZXRyaWV2ZSB0aGUgcmVxdWlyZWQgc21vb3RoZXIgbWF0cmljZXMgYWZ0ZXIgcnVubmluZyBgY2F1c2FsX2ZvcmVzdGAgYW5kIGBpbnN0cnVtZW50YWxfZm9yZXN0YCBiZWxvdy4gVGhlcmVmb3JlLCB3ZSBlc3RpbWF0ZSB0aGUgb3V0Y29tZSBudWlzYW5jZSBtb2RlbCBleHRlcm5hbGx5IHRvIHBhc3MgdGhlIG51aXNhbmNlIHBhcmFtZXRlcnMgbGF0ZXIgdG8gdGhlIGZ1bmN0aW9ucy4gQXMgZGVzY3JpYmVkIGluIFNlY3Rpb24gNS4yIG9mIHRoZSBwYXBlciwgd2UgcnVuIGEgZGVmYXVsdCBhbmQgYSB0dW5lZCB2ZXJzaW9uOgoKYGBge3J9CiMjIyBFeHRlcm5hbGx5IGNhbGN1bGF0ZSBvdXRjb21lIG51aXNhbmNlCnJmX1kuaGF0X2RlZmF1bHQgPSByZWdyZXNzaW9uX2ZvcmVzdChYLFkpCnJmX1kuaGF0X3R1bmVkID0gcmVncmVzc2lvbl9mb3Jlc3QoWCxZLHR1bmUucGFyYW1ldGVycyA9ICJhbGwiKQpZLmhhdF9kZWZhdWx0ID0gcHJlZGljdChyZl9ZLmhhdF9kZWZhdWx0KSRwcmVkaWN0aW9ucwpZLmhhdF90dW5lZCA9IHByZWRpY3QocmZfWS5oYXRfdHVuZWQpJHByZWRpY3Rpb25zCmBgYAoKVGhlbiwgd2UgZXh0cmFjdCB0aGUgc21vb3RoZXIgbWF0cml4IHVzaW5nIGBnZXRfZm9yZXN0X3dlaWdodHMoKWA6CgpgYGB7cn0KIyBBbmQgZ2V0IHNtb290aGVyIG1hdHJpY2VzClNfZGVmYXVsdCA9IGdldF9mb3Jlc3Rfd2VpZ2h0cyhyZl9ZLmhhdF9kZWZhdWx0KQpTX3R1bmVkID0gZ2V0X2ZvcmVzdF93ZWlnaHRzKHJmX1kuaGF0X3R1bmVkKQpgYGAKCkZvciBpbGx1c3RyYXRpb24sIGNoZWNrIHRoYXQgcmFuZG9tIGZvcmVzdCBpcyBhbiBhZmZpbmUgc21vb3RoZXIgYnkgc3VtbWluZyBhbGwgc21vb3RoZXIgdmVjdG9ycyBhbmQgY2hlY2tpbmcgd2hldGhlciB0aGV5IHN1bSB0byBvbmU6CgpgYGB7cn0KY2F0KCJSRiBhZmZpbmUgc21vb3RoZXI/IiwgCiAgICBhbGwuZXF1YWwocm93U3Vtcyhhcy5tYXRyaXgoU19kZWZhdWx0KSksCiAgICAgIHJlcCgxLGxlbmd0aChZKSkKICAgICkpCmBgYAoKCiMjIENhdXNhbCBmb3Jlc3QKClJ1biBjYXVzYWwgZm9yZXN0IHdpdGggdGhlIGV4dGVybmFsbHkgZXN0aW1hdGVkIG91dGNvbWUgbnVpc2FuY2U6CgpgYGB7cn0KIyBSdW4gQ0Ygd2l0aCB0aGUgcHJlLXNwZWNpZmllZCBvdXRjb21lIG51aXNhbmNlIApjZl9kZWZhdWx0ID0gY2F1c2FsX2ZvcmVzdChYLFksRCxZLmhhdD1ZLmhhdF9kZWZhdWx0KQpjZl90dW5lZCA9IGNhdXNhbF9mb3Jlc3QoWCxZLEQsWS5oYXQ9WS5oYXRfdHVuZWQsdHVuZS5wYXJhbWV0ZXJzID0gImFsbCIpCmBgYAoKR2V0IHRoZSBvdXQtb2YtYmFnIENBVEVzOgoKYGBge3J9CmNhdGVzX2RlZmF1bHQgPSBwcmVkaWN0KGNmX2RlZmF1bHQpJHByZWRpY3Rpb25zCmNhdGVzX3R1bmVkID0gcHJlZGljdChjZl90dW5lZCkkcHJlZGljdGlvbnMKYGBgCgpVc2UgdGhlIG5ldyBgZ2V0X291dGNvbWVfd2VpZ2h0cygpYCBtZXRob2QsIHdoaWNoIHJlcXVpcmVzIHRvIHBhc3MgdGhlIGV4dGVybmFsbHkgZXN0aW1hdGVkIG91dGNvbWUgc21vb3RoZXIgbWF0cml4OgoKYGBge3J9Cm9tZWdhX2NmX2RlZmF1bHQgPSBnZXRfb3V0Y29tZV93ZWlnaHRzKGNmX2RlZmF1bHQsIFMgPSBTX2RlZmF1bHQpCm9tZWdhX2NmX3R1bmVkID0gZ2V0X291dGNvbWVfd2VpZ2h0cyhjZl90dW5lZCwgUyA9IFNfdHVuZWQpCmBgYAoKT2JzZXJ2ZSB0aGF0IHRoZSBvdXRjb21lIHdlaWdodHMgcmVjb3ZlciB0aGUgYGdyZmAgcGFja2FnZSBvdXRwdXQ6CgpgYGB7cn0KY2F0KCLPiSdZIHJlcGxpY2F0ZXMgQ0FURSBwb2ludCBlc3RpbWF0ZXMgKGRlZmF1bHQpPyIsIAogICAgYWxsLmVxdWFsKGFzLm51bWVyaWMob21lZ2FfY2ZfZGVmYXVsdCRvbWVnYSAlKiUgWSksCiAgICAgIGFzLm51bWVyaWMoY2F0ZXNfZGVmYXVsdCkKICAgICkpCmNhdCgiXG7PiSdZIHJlcGxpY2F0ZXMgQ0FURSBwb2ludCBlc3RpbWF0ZXMgKHR1bmVkKT8iLCAKICAgIGFsbC5lcXVhbChhcy5udW1lcmljKG9tZWdhX2NmX3R1bmVkJG9tZWdhICUqJSBZKSwKICAgICAgYXMubnVtZXJpYyhjYXRlc190dW5lZCkKICAgICkpCmBgYAoKTm93IGNhbGN1bGF0ZSB0aGUgYWJzb2x1dGUgc3RhbmRhcmRpemVkIG1lYW4gZGlmZmVyZW5jZXMgYW5kIHBsb3QgdGhlbSBmb3IgZWFjaCBDQVRFIGFuZCB2YXJpYWJsZXMgKEZpZ3VyZSAzIGluIHBhcGVyKToKCmBgYHtyfQpjYl9jYXRlX2RlZmF1bHQgPSBzdGFuZGFyZGl6ZWRfbWVhbl9kaWZmZXJlbmNlcyhYLEQsb21lZ2FfY2ZfZGVmYXVsdCRvbWVnYSxYKQpjYl9jYXRlX3R1bmVkID0gc3RhbmRhcmRpemVkX21lYW5fZGlmZmVyZW5jZXMoWCxELG9tZWdhX2NmX3R1bmVkJG9tZWdhLFgpCgpzbWRfZGVmYXVsdCA9IHQoYWJzKGNiX2NhdGVfZGVmYXVsdFssMyxdKSkKc21kX3R1bmVkID0gdChhYnMoY2JfY2F0ZV90dW5lZFssMyxdKSkKCiMgTWVsdCB0aGUgc21kX2RlZmF1bHQgbWF0cml4IHRvIGxvbmcgZm9ybWF0CmRmX2RlZmF1bHRfbG9uZyA9IG1lbHQoc21kX2RlZmF1bHQpCmRmX2RlZmF1bHRfbG9uZyRHcm91cCA9ICJzbWRfZGVmYXVsdCIgICMgQWRkIGEgZ3JvdXAgaWRlbnRpZmllcgoKIyBNZWx0IHRoZSBzbWRfdHVuZWQgbWF0cml4IHRvIGxvbmcgZm9ybWF0CmRmX3R1bmVkX2xvbmcgPSBtZWx0KHNtZF90dW5lZCkKZGZfdHVuZWRfbG9uZyRHcm91cCA9ICJzbWRfdHVuZWQiICAjIEFkZCBhIGdyb3VwIGlkZW50aWZpZXIKCiMgQ29tYmluZSB0aGUgdHdvIGRhdGEgZnJhbWVzCmRmX2xvbmcgPSByYmluZChkZl9kZWZhdWx0X2xvbmcsIGRmX3R1bmVkX2xvbmcpCgojIFJlbmFtZSB0aGUgY29sdW1ucyBmb3IgY2xhcml0eQpjb2xuYW1lcyhkZl9sb25nKSA9IGMoIlJvdyIsICJWYXJpYWJsZSIsICJWYWx1ZSIsICJHcm91cCIpCgojIENyZWF0ZSB0aGUgZ2dwbG90CmZpZ3VyZTMgPSBnZ3Bsb3QoZGZfbG9uZywgYWVzKHggPSBmYWN0b3IoVmFyaWFibGUsIGxldmVscyA9IHJldih1bmlxdWUoVmFyaWFibGUpKSksIHkgPSBWYWx1ZSwgZmlsbCA9IEdyb3VwKSkgKwogIGdlb21fYm94cGxvdChwb3NpdGlvbiA9IHBvc2l0aW9uX2RvZGdlKHdpZHRoID0gMC44KSkgKwogIGxhYnMoeCA9IGVsZW1lbnRfYmxhbmsoKSwgeSA9ICJBYnNvbHV0ZSBTdGFuZGFyZGl6ZWQgTWVhbiBEaWZmZXJlbmNlcyIpICsKICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSB2aXJpZGlzKDIpLAogICAgICAgICAgICAgICAgICAgIG5hbWUgPSBlbGVtZW50X2JsYW5rKCksCiAgICAgICAgICAgICAgICAgICAgbGFiZWxzID0gYygiZGVmYXVsdCIsICJ0dW5lZCIpKSArCiAgdGhlbWVfbWluaW1hbCgpICsKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSAwLCBsaW5ldHlwZSA9ICJzb2xpZCIsIGNvbG9yID0gImJsYWNrIikgKyAKICBjb29yZF9mbGlwKCkKZmlndXJlMwpgYGAKCgojIDIuIFN1cHBsZW1lbnRhcnkgcmVzdWx0cwoKIyMgSW5zdHJ1bWVudGFsIGZvcmVzdAoKUnVuIGluc3RydW1lbnRhbCBmb3Jlc3Qgd2l0aCB0aGUgZXh0ZXJuYWxseSBlc3RpbWF0ZWQgb3V0Y29tZSBudWlzYW5jZToKCmBgYHtyfQojIFJ1biBJRiB3aXRoIHRoZSBwcmUtc3BlY2lmaWVkIG91dGNvbWUgbnVpc2FuY2UKaXZmX2RlZmF1bHQgPSBpbnN0cnVtZW50YWxfZm9yZXN0KFgsWSxELFosWS5oYXQ9WS5oYXRfZGVmYXVsdCkKaXZmX3R1bmVkID0gaW5zdHJ1bWVudGFsX2ZvcmVzdChYLFksRCxaLFkuaGF0PVkuaGF0X3R1bmVkLHR1bmUucGFyYW1ldGVycyA9ICJhbGwiKQpgYGAKCkdldCB0aGUgb3V0LW9mLWJhZyBDTEFURXM6CgpgYGB7cn0KY2xhdGVzX2RlZmF1bHQgPSBwcmVkaWN0KGl2Zl9kZWZhdWx0KSRwcmVkaWN0aW9ucwpjbGF0ZXNfdHVuZWQgPSBwcmVkaWN0KGl2Zl90dW5lZCkkcHJlZGljdGlvbnMKYGBgCgpVc2UgdGhlIG5ldyBgZ2V0X291dGNvbWVfd2VpZ2h0cygpYCBtZXRob2QsIHdoaWNoIHJlcXVpcmVzIHRvIHBhc3MgdGhlIGV4dGVybmFsbHkgZXN0aW1hdGVkIG91dGNvbWUgc21vb3RoZXIgbWF0cml4OgoKYGBge3J9Cm9tZWdhX2lmX2RlZmF1bHQgPSBnZXRfb3V0Y29tZV93ZWlnaHRzKGl2Zl9kZWZhdWx0LCBTID0gU19kZWZhdWx0KQpvbWVnYV9pZl90dW5lZCA9IGdldF9vdXRjb21lX3dlaWdodHMoaXZmX3R1bmVkLCBTID0gU190dW5lZCkKYGBgCgpPYnNlcnZlIHRoYXQgdGhlIG91dGNvbWUgd2VpZ2h0cyByZWNvdmVyIHRoZSBgZ3JmYCBwYWNrYWdlIG91dHB1dDoKCmBgYHtyfQpjYXQoIs+JJ1kgcmVwbGljYXRlcyBDTEFURSBwb2ludCBlc3RpbWF0ZXMgKGRlZmF1bHQpPyIsIAogICAgYWxsLmVxdWFsKGFzLm51bWVyaWMob21lZ2FfaWZfZGVmYXVsdCRvbWVnYSAlKiUgWSksCiAgICAgIGFzLm51bWVyaWMoY2xhdGVzX2RlZmF1bHQpCiAgICApKQpjYXQoIlxuz4knWSByZXBsaWNhdGVzIENMQVRFIHBvaW50IGVzdGltYXRlcyAodHVuZWQpPyIsIAogICAgYWxsLmVxdWFsKGFzLm51bWVyaWMob21lZ2FfaWZfdHVuZWQkb21lZ2EgJSolIFkpLAogICAgICBhcy5udW1lcmljKGNsYXRlc190dW5lZCkKICAgICkpCmBgYAoKTm93IGNhbGN1bGF0ZSB0aGUgYWJzb2x1dGUgc3RhbmRhcmRpemVkIG1lYW4gZGlmZmVyZW5jZXMgYW5kIHBsb3QgdGhlbSBmb3IgZWFjaCBDTEFURSBhbmQgdmFyaWFibGVzOgoKYGBge3J9CmNiX2NsYXRlX2RlZmF1bHQgPSBzdGFuZGFyZGl6ZWRfbWVhbl9kaWZmZXJlbmNlcyhYLEQsb21lZ2FfaWZfZGVmYXVsdCRvbWVnYSxYKQpjYl9jbGF0ZV90dW5lZCA9IHN0YW5kYXJkaXplZF9tZWFuX2RpZmZlcmVuY2VzKFgsRCxvbWVnYV9pZl90dW5lZCRvbWVnYSxYKQoKc21kX2RlZmF1bHQgPSB0KGFicyhjYl9jbGF0ZV9kZWZhdWx0WywzLF0pKQpzbWRfdHVuZWQgPSB0KGFicyhjYl9jbGF0ZV90dW5lZFssMyxdKSkKCiMgTWVsdCB0aGUgc21kX2RlZmF1bHQgbWF0cml4IHRvIGxvbmcgZm9ybWF0CmRmX2RlZmF1bHRfbG9uZyA9IG1lbHQoc21kX2RlZmF1bHQpCmRmX2RlZmF1bHRfbG9uZyRHcm91cCA9ICJzbWRfZGVmYXVsdCIgICMgQWRkIGEgZ3JvdXAgaWRlbnRpZmllcgoKIyBNZWx0IHRoZSBzbWRfdHVuZWQgbWF0cml4IHRvIGxvbmcgZm9ybWF0CmRmX3R1bmVkX2xvbmcgPSBtZWx0KHNtZF90dW5lZCkKZGZfdHVuZWRfbG9uZyRHcm91cCA9ICJzbWRfdHVuZWQiICAjIEFkZCBhIGdyb3VwIGlkZW50aWZpZXIKCiMgQ29tYmluZSB0aGUgdHdvIGRhdGEgZnJhbWVzCmRmX2xvbmcgPSByYmluZChkZl9kZWZhdWx0X2xvbmcsIGRmX3R1bmVkX2xvbmcpCgojIFJlbmFtZSB0aGUgY29sdW1ucyBmb3IgY2xhcml0eQpjb2xuYW1lcyhkZl9sb25nKSA9IGMoIlJvdyIsICJWYXJpYWJsZSIsICJWYWx1ZSIsICJHcm91cCIpCgojIENyZWF0ZSB0aGUgZ2dwbG90CmdncGxvdChkZl9sb25nLCBhZXMoeCA9IGZhY3RvcihWYXJpYWJsZSwgbGV2ZWxzID0gcmV2KHVuaXF1ZShWYXJpYWJsZSkpKSwgeSA9IFZhbHVlLCBmaWxsID0gR3JvdXApKSArCiAgZ2VvbV9ib3hwbG90KHBvc2l0aW9uID0gcG9zaXRpb25fZG9kZ2Uod2lkdGggPSAwLjgpKSArCiAgbGFicyh4ID0gZWxlbWVudF9ibGFuaygpLCB5ID0gIkFic29sdXRlIFN0YW5kYXJkaXplZCBNZWFuIERpZmZlcmVuY2VzIikgKwogIHNjYWxlX2ZpbGxfbWFudWFsKHZhbHVlcyA9IHZpcmlkaXMoMiksCiAgICAgICAgICAgICAgICAgICAgbmFtZSA9IGVsZW1lbnRfYmxhbmsoKSwKICAgICAgICAgICAgICAgICAgICBsYWJlbHMgPSBjKCJkZWZhdWx0IiwgInR1bmVkIikpICsKICB0aGVtZV9taW5pbWFsKCkgKwogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IDAsIGxpbmV0eXBlID0gInNvbGlkIiwgY29sb3IgPSAiYmxhY2siKSArIAogIGNvb3JkX2ZsaXAoKQpgYGAKCgpgYGB7ciwgZWNobz1GfQojIFRoaXMgcGFydCBpcyByZWxldmFudCBpZiB5b3UgcnVuIHRoZSBub3RlYm9va3MgaW5zaWRlIHRoZSBkb2NrZXIgYW5kIHdhbnQgdG8gc2F2ZSBncmFwaHMgYW5kIGltYWdlIGluIGEgc2hhcmVkIGhvc3Qgdm9sdW1lIGNhbGxlZCBzaGFyZWRfZmlsZXMgKHVuY29tbWVudCBhbmQvb3IgYWRqdXN0IG9uIGRlbWFuZCk6CgojIGdnc2F2ZSgiL2hvbWUvcnN0dWRpby9zaGFyZWRfZmlsZXMvRmlndXJlMy5wZGYiLCBwbG90ID0gZmlndXJlMywgd2lkdGggPSA3LCBoZWlnaHQgPSAzLjI1LGRwaT0zMDApCiMgZ2dzYXZlKCIvaG9tZS9yc3R1ZGlvL3NoYXJlZF9maWxlcy9GaWd1cmUzLnBuZyIsIHBsb3QgPSBmaWd1cmUzLCB3aWR0aCA9IDcsIGhlaWdodCA9IDMuMjUsZHBpPTgwMCkKIyBzYXZlLmltYWdlKGZpbGUgPSAiL2hvbWUvcnN0dWRpby9zaGFyZWRfZmlsZXMvQXBwbGljYXRpb25fNDAxa19oZXRlcm9nZW5lb3VzLlJEYXRhIikKYGBgCgoKIyMgUGxvdCByZXN1bHRzCgpIZXJlIHdlIG9ic2VydmUgdGhhdCBub3Qgb25seSB0aGUgZGVmYXVsdCBlZmZlY3RzIHNob3cgbXVjaCB3b3JzZSBiYWxhbmNpbmcsIGJ1dCB0aGV5IGFsc28gbGVhZCB0byBpbXBsYXVzaWJseSBoaWdoIHZhcmlhbmNlIGluIHRoZWlyIGVzdGltYXRlcy4KCmBgYHtyfQpkYXRhID0gZGF0YS5mcmFtZSgKICB2YWx1ZSA9IGMoY2F0ZXNfZGVmYXVsdCwgY2F0ZXNfdHVuZWQsY2xhdGVzX2RlZmF1bHQsIGNsYXRlc190dW5lZCksCiAgY2F0ZWdvcnkgPSByZXAoYygiZ3JmIENBVEVzIGRlZmF1bHQiLCAiZ3JmIENBVEVzIHR1bmVkIiwiZ3JmIENMQVRFcyBkZWZhdWx0IiwgImdyZiBDTEFURXMgdHVuZWQiKSwgZWFjaCA9IGxlbmd0aChjYXRlc19kZWZhdWx0KSkKKQoKZ2dwbG90KGRhdGEsIGFlcyh5ID0gY2F0ZWdvcnksIHggPSB2YWx1ZSwgZmlsbCA9IGNhdGVnb3J5KSkgKwogIGdlb21fYm94cGxvdChhbHBoYSA9IDAuNykgKwogIGdlb21fdmxpbmUoeGludGVyY2VwdCA9IDAsIGNvbG9yID0gImJsYWNrIiwgbGluZXR5cGUgPSAic29saWQiKSArCiAgbGFicygKICAgIHggPSAiRXN0aW1hdGUiLAogICAgeSA9ICJFc3RpbWF0b3IvSW1wbGVtZW50YXRpb24iCiAgKSArCiAgdGhlbWVfbWluaW1hbCgpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCmBgYAoKCmBgYHtyfQojIENyZWF0ZSB0aGUgcmlkZ2UgcGxvdApnZ3Bsb3QoZGF0YSwgYWVzKHggPSB2YWx1ZSwgeSA9IGNhdGVnb3J5LCBmaWxsID0gY2F0ZWdvcnkpKSArCiAgZ2VvbV9kZW5zaXR5X3JpZGdlcyhhbHBoYSA9IDAuNywgc2NhbGUgPSAxKSArCiAgbGFicygKICAgIHggPSAiRXN0aW1hdGUiLAogICAgeSA9ICJFc3RpbWF0b3IvSW1wbGVtZW50YXRpb24iCiAgKSArCiAgdGhlbWVfbWluaW1hbCgpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCmBgYAoKCiMjIFN1bW1hcml6ZSB3ZWlnaHRzCgpGaW5hbGx5LCB3ZSBzdW1tYXJpemUgdGhlIHdlaWdodHMgdmVjdG9ycyBmb3IgZWFjaCBDKEwpQVRFIHBsb3R0aW5nIGRpZmZlcmVudCBkZXNjcmlwdGl2ZXM6CgotIE1pbmltdW0gd2VpZ2h0CgotIE1heGltdW0gd2VpZ2h0CgotICUgTmVnYXRpdmUKCi0gU3VtIGxhcmdlc3QgMTAlCgotIFN1bSBvZiB3ZWlnaHRzCgotIFN1bSBvZiBhYnNvbHV0ZSB3ZWlnaHRzCgpFc3BlY2lhbGx5IHRoZSBzdW0gb2Ygd2VpZ2h0cyBpcyBpbnRlcmVzdGluZyBpbiBsaWdodCBvZiB0aGUgcGFwZXIuIEl0IHNob3dzIHRoYXQgdHVuaW5nIGFsc28gcHJvZHVjZXMgd2VpZ2h0IHN1bXMgY2xvc2VyIHRvIG9uZSBpbiB0aGlzIGRhdGEgc2V0LiBIb3dldmVyLCBhbHNvIG5lZ2F0aXZlIHdlaWdodHMgYXJlIGZld2VyIGFuZCBsZXNzIHByb25vdW5jZWQgZm9yIHRoZSB0dW5lZCB2ZXJzaW9uOgoKYGBge3J9CnN1bW1hcnlfd2VpZ2h0c19jZl9kZWZhdWx0ID0gc3VtbWFyeShvbWVnYV9jZl9kZWZhdWx0LCBxdWlldCA9IFRSVUUpCnN1bW1hcnlfd2VpZ2h0c19jZl90dW5lZCA9IHN1bW1hcnkob21lZ2FfY2ZfdHVuZWQsIHF1aWV0ID0gVFJVRSkKc3VtbWFyeV93ZWlnaHRzX2lmX2RlZmF1bHQgPSBzdW1tYXJ5KG9tZWdhX2lmX2RlZmF1bHQsIHF1aWV0ID0gVFJVRSkKc3VtbWFyeV93ZWlnaHRzX2lmX3R1bmVkID0gc3VtbWFyeShvbWVnYV9pZl90dW5lZCwgcXVpZXQgPSBUUlVFKQoKZm9yIChpIGluIDE6ZGltKHN1bW1hcnlfd2VpZ2h0c19pZl90dW5lZClbM10pIHsKICAjIEV4dHJhY3QgdW50cmVhdGVkIGFuZCB0cmVhdGVkIHdlaWdodHMgZm9yIGVhY2ggZ3JvdXAKICBzdW1fd2VpZ2h0c19jZl9kZWZhdWx0X3VudHJlYXRlZCA9IHN1bW1hcnlfd2VpZ2h0c19jZl9kZWZhdWx0WzEsLGldCiAgc3VtX3dlaWdodHNfY2ZfZGVmYXVsdF90cmVhdGVkID0gc3VtbWFyeV93ZWlnaHRzX2NmX2RlZmF1bHRbMiwsaV0KICAKICBzdW1fd2VpZ2h0c19jZl90dW5lZF91bnRyZWF0ZWQgPSBzdW1tYXJ5X3dlaWdodHNfY2ZfdHVuZWRbMSwsaV0KICBzdW1fd2VpZ2h0c19jZl90dW5lZF90cmVhdGVkID0gc3VtbWFyeV93ZWlnaHRzX2NmX3R1bmVkWzIsLGldCiAgCiAgc3VtX3dlaWdodHNfaWZfZGVmYXVsdF91bnRyZWF0ZWQgPSBzdW1tYXJ5X3dlaWdodHNfaWZfZGVmYXVsdFsxLCxpXQogIHN1bV93ZWlnaHRzX2lmX2RlZmF1bHRfdHJlYXRlZCA9IHN1bW1hcnlfd2VpZ2h0c19pZl9kZWZhdWx0WzIsLGldCiAgCiAgc3VtX3dlaWdodHNfaWZfdHVuZWRfdW50cmVhdGVkID0gc3VtbWFyeV93ZWlnaHRzX2lmX3R1bmVkWzEsLGldCiAgc3VtX3dlaWdodHNfaWZfdHVuZWRfdHJlYXRlZCA9IHN1bW1hcnlfd2VpZ2h0c19pZl90dW5lZFsyLCxpXQogIAogICMgQ29tYmluZSBhbGwgdmVjdG9ycyBpbnRvIGEgc2luZ2xlIGRhdGEgZnJhbWUsIHdpdGggYSBuZXcgJ1RyZWF0bWVudCcgY29sdW1uCiAgZGZfd2VpZ2h0cyA8LSBkYXRhLmZyYW1lKAogICAgVmFsdWUgPSBjKAogICAgICBzdW1fd2VpZ2h0c19jZl9kZWZhdWx0X3VudHJlYXRlZCwgc3VtX3dlaWdodHNfY2ZfZGVmYXVsdF90cmVhdGVkLAogICAgICBzdW1fd2VpZ2h0c19jZl90dW5lZF91bnRyZWF0ZWQsIHN1bV93ZWlnaHRzX2NmX3R1bmVkX3RyZWF0ZWQsCiAgICAgIHN1bV93ZWlnaHRzX2lmX2RlZmF1bHRfdW50cmVhdGVkLCBzdW1fd2VpZ2h0c19pZl9kZWZhdWx0X3RyZWF0ZWQsCiAgICAgIHN1bV93ZWlnaHRzX2lmX3R1bmVkX3VudHJlYXRlZCwgc3VtX3dlaWdodHNfaWZfdHVuZWRfdHJlYXRlZAogICAgKSwKICAgIEdyb3VwID0gZmFjdG9yKGMoCiAgICAgIHJlcCgiQ0YgZGVmYXVsdCIsIGxlbmd0aChzdW1fd2VpZ2h0c19jZl9kZWZhdWx0X3VudHJlYXRlZCkgKyBsZW5ndGgoc3VtX3dlaWdodHNfY2ZfZGVmYXVsdF90cmVhdGVkKSksCiAgICAgIHJlcCgiQ0YgdHVuZWQiLCBsZW5ndGgoc3VtX3dlaWdodHNfY2ZfdHVuZWRfdW50cmVhdGVkKSArIGxlbmd0aChzdW1fd2VpZ2h0c19jZl90dW5lZF90cmVhdGVkKSksCiAgICAgIHJlcCgiSUYgZGVmYXVsdCIsIGxlbmd0aChzdW1fd2VpZ2h0c19pZl9kZWZhdWx0X3VudHJlYXRlZCkgKyBsZW5ndGgoc3VtX3dlaWdodHNfaWZfZGVmYXVsdF90cmVhdGVkKSksCiAgICAgIHJlcCgiSUYgdHVuZWQiLCBsZW5ndGgoc3VtX3dlaWdodHNfaWZfdHVuZWRfdW50cmVhdGVkKSArIGxlbmd0aChzdW1fd2VpZ2h0c19pZl90dW5lZF90cmVhdGVkKSkKICAgICkpLAogICAgVHJlYXRtZW50ID0gZmFjdG9yKGMoCiAgICAgIHJlcCgiVW50cmVhdGVkIiwgbGVuZ3RoKHN1bV93ZWlnaHRzX2NmX2RlZmF1bHRfdW50cmVhdGVkKSksCiAgICAgIHJlcCgiVHJlYXRlZCIsIGxlbmd0aChzdW1fd2VpZ2h0c19jZl9kZWZhdWx0X3RyZWF0ZWQpKSwKICAgICAgcmVwKCJVbnRyZWF0ZWQiLCBsZW5ndGgoc3VtX3dlaWdodHNfY2ZfdHVuZWRfdW50cmVhdGVkKSksCiAgICAgIHJlcCgiVHJlYXRlZCIsIGxlbmd0aChzdW1fd2VpZ2h0c19jZl90dW5lZF90cmVhdGVkKSksCiAgICAgIHJlcCgiVW50cmVhdGVkIiwgbGVuZ3RoKHN1bV93ZWlnaHRzX2lmX2RlZmF1bHRfdW50cmVhdGVkKSksCiAgICAgIHJlcCgiVHJlYXRlZCIsIGxlbmd0aChzdW1fd2VpZ2h0c19pZl9kZWZhdWx0X3RyZWF0ZWQpKSwKICAgICAgcmVwKCJVbnRyZWF0ZWQiLCBsZW5ndGgoc3VtX3dlaWdodHNfaWZfdHVuZWRfdW50cmVhdGVkKSksCiAgICAgIHJlcCgiVHJlYXRlZCIsIGxlbmd0aChzdW1fd2VpZ2h0c19pZl90dW5lZF90cmVhdGVkKSkKICAgICkpCiAgKQogIAogICMgUGxvdCB3aXRoIGdncGxvdCBhbmQgZ2dyaWRnZXMKICBnIDwtIGdncGxvdChkZl93ZWlnaHRzLCBhZXMoeCA9IFZhbHVlLCB5ID0gR3JvdXAsIGZpbGwgPSBUcmVhdG1lbnQpKSArCiAgICBnZW9tX2JveHBsb3QocG9zaXRpb24gPSBwb3NpdGlvbl9kb2RnZSh3aWR0aCA9IDAuNzUpKSArCiAgICBsYWJzKHggPSBkaW1uYW1lcyhzdW1tYXJ5X3dlaWdodHNfY2ZfdHVuZWQpW1szXV1baV0sIHkgPSBOVUxMKSArCiAgICB0aGVtZV9taW5pbWFsKCkgKwogICAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gImJvdHRvbSIpICAjIFBsYWNlIGxlZ2VuZCBhdCB0aGUgYm90dG9tCiAgcHJpbnQoZykKfQoKYGBgCgo=