# Load the tidyverse and mosaic packages
library(tidyverse)
library(mosaic)

Scenario: Ambiguous Prose

In 1972, 57 high school students were randomly assigned to one of three groups:

The students were asked to read the following passage and remember details from it:

Passage

If the balloons popped, the sound wouldn’t be able to carry since everything would be too far away from the correct floor. A closed window would also prevent the sound from carrying, since most buildings tend to be well insulated. Since the whole operation depends on a steady flow of electricity, a break in the middle of the wire would also cause problems. Of course, the fellow could shout, but the human voice is not loud enough to carry that far. An additional problem is that a string could break on the instrument. Then there could be no accompaniment to the message. It is clear that the best situation would involve less distance. Then there would be fewer potential problems. With face to face contact, the least number of things could go wrong.

Students were asked to:

  • Rate how easy it was to comprehend the passage on a 1-7 scale (1 = very difficult; 7 = very easy)
  • Recall the passage by writing out as many of the ideas as you remember.

Does seeing the picture influence a student’s ability to comprehend and/or recall the passage?


Picture

Data Exploration


For now, let’s focus on the comprehension scores. The code shows how the data were imported –>

# We'll download the data from my website
ambiguous <- read.csv("http://www.bradthiessen.com/html5/data/ambiguousprose.csv")

# Look at the first several rows to get a sense of the data
head(ambiguous)

# Get summary statistics
ambiguous %>%
  group_by(Condition) %>%
  summarize(n = n(),
            mean = mean(Comprehension),
            sd = sd(Comprehension))

ambiguous %>%
  summarize(n = n(),
            mean = mean(Comprehension),
            sd = sd(Comprehension))
Condition n mean sd
After 19 3.211 1.398
Before 19 4.947 1.311
None 19 3.368 1.257
Combined N = 57 M = 3.842 s = 1.521


  1. What do M and s represent? From these summary statistics, do you think we’ll conclude the picture influenced student comprehension?

Dotplot


ambiguous %>%
  ggplot(aes(x = Condition, y = Comprehension)) +
  geom_dotplot(aes(fill=Condition), binaxis = "y", binpositions="all", color = "white") +
  scale_y_continuous(breaks=seq(0, 7, 1), minor_breaks=NULL) +
  theme(axis.title.x = element_text(size = 12, color = "#777777")) +
  theme(axis.text.x = element_text(size = 12)) +
  theme(axis.title.y = element_text(size = 12, color="#777777")) +
  theme(axis.text.y = element_text(size = 12)) +
  theme(legend.position="none") +
  labs(
    title = "Comprehension by Condition"
  ) +
  coord_flip(xlim = NULL, ylim = NULL, expand = TRUE)

Boxplot


ambiguous %>%
  ggplot(aes(x = Condition, y = Comprehension)) +
  geom_dotplot(aes(fill=Condition), binaxis = "y", binpositions="all", color = "white", alpha=0.5) +
  geom_boxplot(fill = "white", color = "black", alpha = 0.6) +
  scale_y_continuous(breaks=seq(0, 7, 1), minor_breaks=NULL) +
  theme(axis.title.x = element_text(size = 12, color = "#777777")) +
  theme(axis.text.x = element_text(size = 12)) +
  theme(axis.title.y = element_text(size = 12, color="#777777")) +
  theme(axis.text.y = element_text(size = 12)) +
  theme(legend.position="none") +
  labs(
    title = "Comprehension by Condition"
  ) +
  coord_flip(xlim = NULL, ylim = NULL, expand = TRUE)

Means with standard errors


ambiguous %>%
  ggplot(aes(x = Condition, y = Comprehension)) +
  stat_summary(fun.y = mean, geom = "point") + 
  stat_summary(fun.y = mean, geom = "line", aes(group = 1),
               colour = "Blue", linetype = "dashed") +
  stat_summary(fun.data = mean_cl_boot, geom = "errorbar", width = 0.2) +
  labs(x = "Treatment", y = "Mean Comprehension Score")

Hypotheses

  1. Write out null and alternative hypotheses to compare the means of all three groups. How could we determine the likelihood of this null hypothesis?

Idea #1: multiple t-tests

If an independent samples t-test compares two group means, could we use multiple t-tests to compare all possible pairs of our three group means?


  1. How many t-tests would we need to test all possible pairs of three group means? How many t-tests would we need if we had G groups?
  1. Suppose we conduct 3 t-tests. If we set \(\alpha =0.05\) for each test, how likely would we be to make at least one alpha error in our 3 tests? In other words, what’s \(P(\mathrm{at\ least\ one\ }\alpha\ \mathrm{error\ in\ 3\ tests})\)?
  1. Suppose we conduct t-tests to compare all possible pairs of means from G groups. If we have a Type I error rate of \(\alpha\) for each test, what is the probability we make at least one alpha error across all our tests?

Problems with multiple t-tests

  1. Explain why we should not conduct multiple t-tests to compare pairs of group means from a single dataset.

Despite this problem, let’s conduct multiple t-tests using the pairwise.t.test() command to run all possible t-tests for a dataset.

pairwise.t.test(ambiguous$Comprehension, ambiguous$Condition, p.adjust.method="none")

    Pairwise comparisons using t tests with pooled SD 

data:  ambiguous$Comprehension and ambiguous$Condition 

       After   Before 
Before 0.00017 -      
None   0.71444 0.00054

P value adjustment method: none 


  1. The output states the t-tests used a “pooled SD.” What does that mean?
  1. What could we conclude from the p-values listed in the output? Why are we uncertain about our conclusions? Is there anything we could do to reduce this uncertainty?

Idea #2: get MAD (or SAD)

If running multiple t-tests is problematic, could we conduct a single test of our three group means using randomization-based methods?

Sure! If we can identify a single test statistic that compares all three group means, we can use randomization- or theory-based methods.


  1. What could we conclude from the p-values listed in the output? Why are we uncertain about our conclusions? Is there anything we could do to reduce this uncertainty?

Test statistic

To compare two means, we use: \(\overline{X}_{1}-\overline{X}_{2}\). If two means are similar, the test statistic is near zero. As the means differ by a greater amount, the value of the test statistic increases.


  1. Propose a single test statistic to measure the difference among 3 group means. What happens to the value of that test statistic as the group means get closer together or further apart?

Using the means of our three groups:

  • \(\overline{X}_{after}=\overline{X}_{a}=3.211\)
  • \(\overline{X}_{before}=\overline{X}_{b}=4.947\)
  • \(\overline{X}_{none}=\overline{X}_{n}=3.368\)


We can calculate:

\(SAD = |\overline{X}_{a}-\overline{X}_{b}|+|\overline{X}_{a}-\overline{X}_{n}|+|\overline{X}_{b}-\overline{X}_{n}|=3.474\)

\(MAD = \frac{|\overline{X}_{a}-\overline{X}_{b}|+|\overline{X}_{a}-\overline{X}_{n}|+|\overline{X}_{b}-\overline{X}_{n}|}{3}=1.158\)


Expand the code to see how to calculate this in R –>

SAD( mean(Comprehension ~ Condition, data=ambiguous) )
MAD( mean(Comprehension ~ Condition, data=ambiguous) )

test_stat <- SAD( mean(Comprehension ~ Condition, data=ambiguous) )

We’ll store SAD = 8.2151648 as our test_stat.


Randomization-based SAD test

  1. Assume your null hypothesis is true. In our sample data, the first subject in the after group rated the passage a 6 in comprehension. If we went back in time and randomly assigned this subject to the none group, what rating would the subject give to the passage?
  1. How many ways could 57 subjects be assigned to 3 groups (with 19 in each group)? Expand the code to see this calculation.
# Eliminate scientific notation
options(scipen=999)

# Calculate number of randomizations (group ID does not matter)
ways <- ( choose(n=57, k=19) * choose(38, 19) * choose(19, 19) ) / factorial(3)

# Put comma separators into the number
format(ways, big.mark=",", scientific=FALSE)

We’re never going to list all those randomizations, but we can get a random, representative sample of them. For each randomization, we’ll calculate SAD. We’ll then plot our distribution of randomized SADs and determine the likelihood that our test statistic came from this distribution.

# Randomize our data 10,000 times (shuffling condition)
SADrand <- Do(10000) * SAD( mean(Comprehension ~ shuffle(Condition), data=ambiguous) )

# Here's one way to do this in dplyr
# SADrand <- Do(10000) * ambiguous %>%
#   mutate(shuffled = sample(Comprehension)) %>%
#   group_by(Condition) %>%
#   summarize(shuffled_means = mean(shuffled)) %>%
#   mutate(shuffled_SAD = SAD(shuffled_means)) %>%
#   filter(Condition == "After") %>%   # Eliminates triplicate results
#   select(shuffled_SAD)

# Plot the distribution

# Here's the most straight-forward way to plot this
# histogram(~SAD, data=SADrand,
#           xlim=c(0,5),
#           width=.2,
#           xlab="Possible mean differences assuming null hypothesis is true")
# ladd(panel.abline(v=test_stat))   # Add vertical line at test statistic

# Using ggplot2 with a density plot
ggplot(data = SADrand, aes(x = SAD)) +
  geom_density(fill="lightblue", color="white", alpha = 0.8) +
  annotate("segment", x = test_stat, xend = test_stat, y = 0, yend = .4, color = "red") +
  annotate("text", x = test_stat, y = .45, label = "observed SAD = 3.474", color = "red") +
  labs(
      title = "Randomized SAD",
      x = "SAD"
      ) +
  scale_x_continuous(limits = c(0,5), breaks=seq(0, 5, 1), minor_breaks=NULL) +
    theme(
    axis.text.x = element_text(size = 11, color="grey10"),
    legend.position = "none",
    panel.grid.major.y = element_line(colour = "white"),
    panel.grid.major.x = element_line(colour = "white", size=.15),
    panel.grid.minor = element_blank(),
    panel.background = element_rect(fill = "grey93")
  )


# Estimate p-value
pvalue <- prop(SADrand$SAD >= test_stat)
pvalue
  1. When we compared two means, the randomized distribution of mean differences was centered near zero with a symmetric, unimodal appearance.

    When we compared two variances, the randomized distribution of variance ratios was centered near 1.0 with a positive skew.

    Explain why the randomized distribution of SAD is not centered at zero or 1.0. Explain why it has a positive skew.
  1. Explain, one last time, the process of randomization, what the plot represents, and how the p-value was estimated. Then, interpret the p-value of 0.0003.

Problems with randomized SAD (or MAD) test

  1. Three scenarios are plotted in the dotplots displayed below. Scenario B (the middle column) represents the actual data from this study. Data for scenarios A and C were simulated with (roughly) the same means as the actual data. This means that in all three scenarios, the value of SAD would be the same.

    Which scenario (A, B, C) provides the strongest evidence against the null hypothesis? In other words, in which scenario does the condition group matter the most?

    Identify a limitation of SAD as a test statistic to compare group means.
# A = Simulate data with same means, smaller standard deviations
dataA <- tibble(
  Condition = as.factor(c(rep("After", 19), rep("Before", 19), rep("None",19))),
  Comprehension = c(rnorm(19,3.211,0.5), rnorm(19,4.947,0.5), rnorm(19,3.368,0.5)),
  dataset = c(rep("A", 57))
)

# B = actual data from study
dataB <- ambiguous
dataB$dataset = c(rep("B", 57))

  
# C = simulate data with same means, larger standard deviations
dataC <- tibble(
  Condition = as.factor(c(rep("After", 19), rep("Before", 19), rep("None",19))),
  Comprehension = c(rnorm(19,3.211,3), rnorm(19,4.947,3), rnorm(19,3.368,3)),
  dataset = c(rep("C", 57))
)

# merge into a single data frame
data_sim <- bind_rows(dataA, dataB, dataC)

# Plot with dataset as facet
data_sim %>%
  ggplot(aes(x = Condition, y = Comprehension)) +
  geom_dotplot(aes(fill=Condition), binwidth=.5, binaxis = "y", binpositions="all", color = "white", alpha=0.9) +
  scale_y_continuous(limits = c(-5, 9), breaks=c(3, 5), minor_breaks=NULL) +
  stat_summary(fun.y = mean, geom = "point", size=3, aes(group=1)) +
  theme(axis.title.x = element_text(size = 12, color = "#777777")) +
  theme(axis.text.x = element_text(size = 12)) +
  theme(axis.title.y = element_text(size = 12, color="#777777")) +
  theme(axis.text.y = element_text(size = 12)) +
  theme(legend.position="none") +
  coord_flip(xlim = NULL, ylim = NULL, expand = FALSE) +
  facet_grid(. ~ dataset)

To address the limitations of SAD, our test statistic should consider the variability within the groups along with the variability among the group means.

  1. Assuming we can do this, how will we compare the variance among groups to the variance within groups? What distribution will this follow?

Idea #3: ANOVA (using the mean square ratio)


Our test statistic, then, will be: \(F=\frac{\textrm{variance between group means}}{\textrm{variance within groups}}\)


We’ll call this process analysis of variance or ANOVA. Despite its name, realize the goal is to compare means (by analyzing the variance between and within the groups).


  1. What conditions are necessary for a ratio of variances to follow an F-distribution? This is a drawback to ANOVA – we’ll need to make sure some conditions are met.
  1. Below, I’ve sketched hypothetical population distributions for each group in our study. List the assumptions I made in creating the plot on the left and the plot on the right.
# Null hypothesis is true
ggplot(data.frame(x = c(0, 8.5)), aes(x)) +
  stat_function(fun = dnorm, args = list(mean = 3.6, sd = 1.25), color="red", size=1) +
  stat_function(fun = dnorm, args = list(mean = 3.7, sd = 1.25), color="skyblue", size=1) +
  stat_function(fun = dnorm, args = list(mean = 3.8, sd = 1.25), color="267", size=1) +
  geom_dotplot(data = ambiguous, aes(x = Comprehension, y = 1, fill=Condition, color=Condition), 
               binwidth=.1, binaxis="x", stackgroups=TRUE, binpositions="all") +
  scale_y_continuous(limits = c(0, .4), breaks=NULL, minor_breaks=NULL) +
  scale_x_continuous(limits = c(-.5, 9), breaks=seq(1,7,1), minor_breaks=NULL) +
  theme(axis.title.x = element_text(size = 12, color = "#777777")) +
  theme(axis.text.x = element_text(size = 12)) +
  theme(legend.position="none")  +
    labs(
      title = "Null Hypothesis",
      x = "comprehension score",
      y="")

# Alternative hypothesis is true
ggplot(data.frame(x = c(0, 8.5)), aes(x)) +
  stat_function(fun = dnorm, args = list(mean = 2, sd = .6), color="red", size=1) +
  stat_function(fun = dnorm, args = list(mean = 3.7, sd = .6), color="skyblue", size=1) +
  stat_function(fun = dnorm, args = list(mean = 5.4, sd = .6), color="267", size=1) +
  geom_dotplot(data = ambiguous, aes(x = Comprehension, y = 1, fill=Condition, color=Condition), 
               binwidth=.1, binaxis="x", stackgroups=TRUE, binpositions="all") +
  scale_y_continuous(limits = c(0, .75), breaks=NULL, minor_breaks=NULL) +
  scale_x_continuous(limits = c(-.5, 9), breaks=seq(1,7,1), minor_breaks=NULL) +
  theme(axis.title.x = element_text(size = 12, color = "#777777")) +
  theme(axis.text.x = element_text(size = 12)) +
  theme(legend.position="none")  +
    labs(
      title = "Alternative Hypothesis",
      x = "comprehension score",
      y="") +
    annotate("text", x = 2.0, y = .7, label = "after", color = "red") +
    annotate("text", x = 3.7, y = .7, label = "none", color = "steelblue") +
    annotate("text", x = 5.4, y = .7, label = "before", color = "267")

Model

Let’s construct a model for this ambiguous prose study. Think about 3 subjects in this study:

  • One randomly assigned to the after group who rated the passage a 6
  • One assigned to the before group who rated the passage a 3
  • One assigned to the none group who rated the passage a 1
  1. Pick one of those individuals and list at least three reasons why that person assigned that comprehension rating to the passage.
  1. Our model is:

    \(y_{ig}=\mu +\alpha _{g}+\epsilon_{ig}=\mu+(\mu _{g}-\mu )+(y_{ig}-\mu _{g})\).

    Identify the components of this model and explain why the model must be true. What assumptions will we make regarding the error term?

With ANOVA, we partition the total variation in our data (the differences among comprehension ratings from all subjects in the study) into two components:

  • Treatment or between-groups = variation in comprehension due to the photo
  • Error or within-groups = variation in comprehension among subjects in the same group

We can (roughly) visualize these sources of variation as:

# Null hypothesis is true
ggplot(data.frame(x = c(0, 8.5)), aes(x)) +
  stat_function(fun = dnorm, args = list(mean = 3.6, sd = 1.25), color="red", size=1, alpha=0.5) +
  stat_function(fun = dnorm, args = list(mean = 3.7, sd = 1.25), color="skyblue", size=1, alpha=0.5) +
  stat_function(fun = dnorm, args = list(mean = 3.8, sd = 1.25), color="267", size=1, alpha=0.5) +
  scale_y_continuous(limits = c(0, .4), breaks=NULL, minor_breaks=NULL) +
  scale_x_continuous(limits = c(-.5, 9), breaks=seq(1,7,1), minor_breaks=NULL) +
  theme(axis.title.x = element_text(size = 12, color = "#777777")) +
  theme(axis.text.x = element_text(size = 12)) +
  theme(legend.position="none")  +
    labs(
      title = "Null Hypothesis",
      x = "comprehension score",
      y="") +
    annotate("segment", x = 3.6, xend = 3.8, y = .333, yend = .333, color = "black", size=1.5) +
    annotate("text", x = 3.7, y = .35, label = "between groups variation", color = "black") +
    annotate("segment", x = 2.1, xend = 5.3, y = .12, yend = .12, color = "black", size=1.5) +
    annotate("text", x = 3.7, y = .14, label = "within groups variation", color = "black") +
    annotate("segment", x = 0, xend = 8, y = .0, yend = .0, color = "black", size=1.5) +
    annotate("text", x = 3.7, y = .02, label = "total variation", color = "black") 


# Alternative hypothesis is true
ggplot(data.frame(x = c(0, 8.5)), aes(x)) +
  stat_function(fun = dnorm, args = list(mean = 2, sd = .6), color="red", size=1, alpha=0.5) +
  stat_function(fun = dnorm, args = list(mean = 3.7, sd = .6), color="skyblue", size=1, alpha=0.5) +
  stat_function(fun = dnorm, args = list(mean = 5.4, sd = .6), color="267", size=1, alpha=0.5) +
  scale_y_continuous(limits = c(0, .75), breaks=NULL, minor_breaks=NULL) +
  scale_x_continuous(limits = c(-.5, 9), breaks=seq(1,7,1), minor_breaks=NULL) +
  theme(axis.title.x = element_text(size = 12, color = "#777777")) +
  theme(axis.text.x = element_text(size = 12)) +
  theme(legend.position="none")  +
    labs(
      title = "Alternative Hypothesis",
      x = "comprehension score",
      y="") +
    annotate("segment", x = 2, xend = 5.4, y = .69, yend = .69, color = "black", size=1.5) +
    annotate("text", x = 3.7, y = .73, label = "between groups variation", color = "black") +
    annotate("segment", x = 3, xend = 4.4, y = .3, yend = .3, color = "black", size=1.5) +
    annotate("text", x = 3.7, y = .33, label = "within", color = "black") +
    annotate("segment", x = 0, xend = 8, y = .0, yend = .0, color = "black", size=1.5) +
    annotate("text", x = 3.7, y = .02, label = "total variation", color = "black") 


  1. We’re going to compare the between groups variation to the within groups variation. If that ratio is large, what does that say about our null hypothesis? What does it say if the ratio is relatively small? How are you defining large and small?

Partitioning (calculating) the sources of variation

Recall the formula for the unbiased estimate of the population variance:

\(s^{2}=\frac{\sum_{i=1}^{n}(x_{i}-\overline{X})^{2}}{n-1}=\frac{\textrm{sum of squared deviations from the mean}}{\textrm{degrees of freedom}}=\frac{SS}{df}=\textrm{mean square}\)


  1. What do SS and MS represent? What are degrees of freedom?

Total variation = MStotal

  1. Fill-in-the-blanks to derive the formula to calculate the total variation in comprehension scores. Then, considering what MStotal represents, rewrite the numerator (SStotal). Finally, calculate MStotal for this ambiguous prose dataset and explain what it represents.


\(\mathbf{\textrm{MS}_{\textrm{Total}}=\frac{\textrm{SS}_{\textrm{total}}}{\textrm{df}_{\textrm{total}}}=\frac{\sum_{i=1}^{n}(\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ )^{2}}{(\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ )}}=\)

Calculation:

SStotal <- sum( ( ambiguous$Comprehension - mean(ambiguous$Comprehension) )^2 )
SStotal

dftotal <- length(ambiguous$Comprehension) - 1
dftotal

MStotal <- SStotal / dftotal
MStotal

# Notice MStotal is just the variance of all the data
var(ambiguous$Comprehension)

Within-group variation = MSerror

  1. SStotal = 129.579. Let’s partition that into the within- and between-groups source.

    MSerror represents the variation within each group (variation in comprehension that is not due to seeing the photo).

    Fill-in-the-blanks to derive formulas for \(SS_{error}\), \(df_{error}\), and \(MS_{error}\). Explain what MSerror represents and write a simpler formula to calculate SSerror. Finally, calculate MSerror for our data.


\(\mathbf{\textrm{MS}_{\textrm{Error}}=\frac{\textrm{SS}_{\textrm{error}}}{\textrm{df}_{\textrm{error}}}=\frac{\sum(\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ )^{2}}{(\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ )}}=\)

Calculation:

SSerror <- sum( (19 - 1) * var(Comprehension ~ Condition, data=ambiguous) )
SSerror

dferror <- length(ambiguous$Comprehension) - 3
dferror

MSerror <- SSerror / dferror
MSerror

Between-group variation = MSphoto

  1. If we know SStotal = 129.579 and SSerror = 94.526, how can we easily calculate SSphoto?

    Fill-in-the-blanks to derive formulas for \(SS_{photo}\), \(df_{photo}\), and \(MS_{photo}\). Explain what they represent and calculate MSphoto for our data.


\(\mathbf{\textrm{MS}_{\textrm{A}}=\mathbf{\textrm{MS}_{\textrm{treatment}}}=\frac{\textrm{SS}_{\textrm{A}}}{\textrm{df}_{\textrm{A}}}=\frac{\sum\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ (\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ )^{2}}{(\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ )}}\)

Calculation:

# This is not at all the easiest way to calculate this...
SSphoto <- as.numeric(
  ambiguous %>%
  group_by(Condition) %>%
  summarize(squaredmeandiff = (mean(Comprehension)-mean(ambiguous$Comprehension))^2) %>%
  mutate(ssphoto = 19 * sum(squaredmeandiff)) %>%
  filter(Condition=="After") %>%
  select(ssphoto)
)

SSphoto

# Check to see if it's the same as SStotal - SSerror
SStotal - SSerror

dfphoto <- n_distinct(ambiguous$Condition) - 1
dfphoto

MSphoto <- SSphoto / dfphoto
MSphoto

ANOVA summary of calculations


For our data:


Source SS df MS MSR
Treatment 35.053 2 17.526 ?
Error 94.526 54 1.75 ?
Total 129.579 56 2.314 ?
  1. Will \(\mathbf{\textrm{SS}_{\textrm{total}}}\) always equal \(\mathbf{\textrm{SS}_{\textrm{A}}}+\mathbf{\textrm{SS}_{\textrm{E}}}\)? Explain what is happening at each step of the following derivation:


\(\mathbf{\textrm{SS}_{\textrm{total}}}=\sum \sum (x_{i}-M)^{2}=\sum \sum \left [ \left ( x_{i}- \overline{X}_{a}\right )-\left ( \overline{X}_{a}-M \right ) \right ]^{2}=\sum \sum \left [ \left ( \overline{X}_{a}- M\right )-\left ( x_{i}-\overline{X}_{a} \right ) \right ]^{2}=\)


\(\mathbf{\textrm{SS}_{\textrm{total}}}=\sum \sum\left ( \overline{X}_{a}- M\right )^{2}+\sum \sum\left ( x_{i}-\overline{X}_{a} \right )^{2}+2\sum \sum\left ( \overline{X}_{a}- M\right )\left ( x_{i}-\overline{X}_{a} \right )\)


Since \(\sum \left ( x_{i}-\overline{X}_{a} \right )=0\)


\(\mathbf{\textrm{SS}_{\textrm{total}}}=\sum n_{a}\left ( \overline{X}_{a}- M\right )^{2}+\sum \sum\left ( x_{i}-\overline{X}_{a} \right )^{2}=\)


\(\mathbf{\textrm{SS}_{\textrm{total}}}=\mathbf{\textrm{SS}_{\textrm{A}}}+\mathbf{\textrm{SS}_{\textrm{E}}}\)

One more look at MSerror

If mean square is a fancy term for variance, MSerror must represent the variance within a single group.


If we have 3 groups in our dataset, how can MSE represent the variance of a single group? Which group do we “choose” when we calculate MSE?


If we assume our data come from groups with equal population variances, we don’t have to choose a group. Instead, we can calculate a pooled variance of all our groups. In this sense, MSE represents the average variance of each of our groups.


The concept of an average, or pooled, variance should sound familiar. We’ve used Spooled to conduct independent samples t-tests (assuming equal variances). If we take the formula for Spooled and extend it to 3 groups, we have:


\(s_{\textrm{pooled}}^{2}=\frac{\left ( n_{1}-1 \right )s_{1}^{2}+\left ( n_{2}-1 \right )s_{2}^{2}+\left ( n_{3}-1 \right )s_{3}^{2}}{\left ( n_{1}-1 \right )+\left ( n_{2}-1 \right )+\left ( n_{3}-1 \right )}\)


\(s_{\textrm{pooled}}^{2}=\frac{\left ( n_{1}-1 \right )\frac{\sum\left ( x_{i1}-\overline{X}_{1} \right )^{2}}{n_{1}-1} + \left ( n_{2}-1 \right )\frac{\sum\left ( x_{i2}-\overline{X}_{2} \right )^{2}}{n_{2}-1} + \left ( n_{3}-1 \right )\frac{\sum\left ( x_{i3}-\overline{X}_{3} \right )^{2}}{n_{3}-1}}{\left ( n_{1}-1 \right )+\left ( n_{2}-1 \right )+\left ( n_{3}-1 \right )}\)


\(s_{\textrm{pooled}}^{2}=\frac{\sum \left ( x_{i1}-\overline{X}_{1} \right )^{2} + \sum \left ( x_{i2}-\overline{X}_{2} \right )^{2} + \sum \left ( x_{i3}-\overline{X}_{3} \right )^{2}}{\left ( n_{1}-1 \right )+\left ( n_{2}-1 \right )+\left ( n_{3}-1 \right )}\)


\(s_{\textrm{pooled}}^{2}=\frac{\sum \left ( x_{ia}-\overline{X}_{a} \right )^{2}}{N-a}=\textrm{MS}_{\textrm{E}}\)


  1. Under a true null hypothesis, what’s the expected value of MSerror? Does that expectation change if the null hypothesis were false?
  1. Under a true null hypothesis, what’s the expected value of MSphoto? Does that expectation change if the null hypothesis were false?

This is the logic behind ANOVA. Under a true null hypothesis, both MSphoto and MSerror represent unbiased estimates of the variance within a group. When the null hypothesis is false, MSphoto becomes larger.


  1. How will we compare MSphoto to MSerror? What sampling distribution will we use? How many degrees of freedom will we have? What’s the expected value of our test statistic if the null hypothesis were true?

ANOVA summary table

Complete the following ANOVA summary table and estimate the p-value. We can estimate the p-value directly in R or use the Statkey F-distribution applet.


Source SS df MS MSR
Treatment 35.053 2 17.53 MSR =
Error 94.526 54 1.75 p =
Total 129.579 56 2.31 \(\eta^2=\)


A quick look at an F-distribution table will give you a sense of what values of the MSR will yield low p-values.


  1. What does that p-value tell us? Does it inform us about the magnitude of the differences among the group means?

ANOVA in R

Conducting ANOVA in R is simple. Expand the code to see –>

# Our data frame is "ambiguous"
# Our model is:  Comprehension ~ Condition
# We use aov() to conduct the analysis of variance
# I'll store the results in a list called "model"
model <- aov( Comprehension ~ Condition, data = ambiguous )

# The default output isn't too helpful
# It only shows SS and df, with a residual standard error
model

# We can use the anova() function to produce more useful output
anova(model)

# Verify that these values match our calculations

# The aov() function produces much more output than we're seeing
# We can glance at some of this output with the broom package
# The following code will install the broom package (if necessary)
list.of.packages <- c("broom")
new.packages <- list.of.packages[!(list.of.packages %in% installed.packages()[,"Package"])]
if(length(new.packages)) install.packages(new.packages)

library(broom)
tidy(model)
glance(model)

# By the end of this course, you'll understand what all those
# terms mean: r.squared, logLik, AIC, deviance

Effect size: \(\eta^2\) and \(\omega^2\)

We can define an effect size (eta-squared) as: \(\eta ^{2}=\frac{\textrm{SS}_\textrm{A}}{\textrm{SS}_\textrm{T}}=\frac{35.05}{129.58}=0.27\)

  1. Interpret eta-squared in this scenario.

Eta-squared is based entirely on sums of squares from the sample and no adjustment is made to estimate the effect size for the population. As we’ll learn later in this course, our models typically overfit our sample data.

As an adjusted effect size, we can calculate omega-squared: \(\omega ^{2}=\frac{\textrm{SS}_\textrm{A}-\left (\textrm{df}_{\textrm{A}} \right )\left ( \textrm{MS}_\textrm{E} \right )}{\textrm{SS}_\textrm{T}+\textrm{MS}_\textrm{E}}\)


For our ambiguous prose data: \(\omega ^{2}=\frac{35.05- \left (2 \right )\left ( 1.75 \right )}{129.58+1.75}=0.25\)


  1. Why is \(\omega^2 < \eta^2\)? Based on the p-value and effect size estimates, what conclusions are you willing to make?

Evaluating ANOVA conditions (assumptions)

  1. List the conditions necessary to get valid results from an ANOVA. Identify why we need each condition.

To assess homogeneity of variances, we can use:

F-max test

The Fmax test, which should only be applied if data for each group are normally distributed, is something you can quickly estimate in your head. To do so, you calculate the following statistic and compare it to the Fmax distribution.

\(F_{max}=\frac{s^2_\mathrm{{biggest }}}{s^2_\mathrm{{smallest }}}=\frac{1.398^2}{1.257^2}=1.237\sim F_{max}\)


Levene’s test

Levene’s test, which is more appropriate if your data do not follow a normal distribution, is calculated by first converting every score to:

\(z_{ai}=\left |x_{ai}-\overline{X}_a \right |\)


The test statistic is calculated as:

\(W=\frac{\left ( N-a \right )}{a-1}\frac{\sum n_{a}\left ( \overline{z}_{a}-\overline{\overline{z}} \, \right )^2}{\sum \sum \left ( z_{ia}- \overline{z}_{a}\right )^2}\sim{F}^{a-1}_{N-a}\)


Expand the code to see how to conduct Levene’s test in R –>

# We need the 'car' package
list.of.packages <- c("car")
new.packages <- list.of.packages[!(list.of.packages %in% installed.packages()[,"Package"])]
if(length(new.packages)) install.packages(new.packages)
library(car)

# Levene Test
leveneTest(Comprehension ~ Condition, data = ambiguous, center = mean)

# Brown-Forsythe Test (uses medians instead of means)
leveneTest(Comprehension ~ Condition, data = ambiguous, center = median)


  1. Based on the output from Levene’s test, can we conclude our group variances are equal? Explain.

Evaluating normality assumption

There are several ways to test for normality:

  • Histomancy: The art of divining likelihood functions from empirical histograms. This sorcery is used, for example, when testing for normality before deciding whether or not to use a non-parametric procedure. Histomancy is a false god, because even perfectly good Gaussian varriables may not look Gaussian when displayed as a histogram. (McElreath, p. 282)
  • Q-Q plots (which have the same issues as looking at histograms)
  • Back-of-the-envelope test: Convert the maximum and minimum scores to z-scores. You shouldn’t have extreme z-scores in small datasets (\(z>4\) should only occur with at least 15,000 observations in the group)
  • Shapiro-Wilk test (seems to be the most powerful)
  • Kolmogorov-Smirnov test
  • Székely and Rizzo’s energy test

These methods (except for the energy test) are easy to understand. Expand the code to see how to conduct the Shapiro-Wilk test in R.

# The function is called 'shapiro.test`.  It would test for normality
# of all the data in our data frame (ignoring the groups)
shapiro.test(ambiguous$Comprehension)

# If we want to apply (tapply) the shapiro.test to all our groups, we use:
tapply(ambiguous$Comprehension, ambiguous$Condition, shapiro.test)

# We could then look at a Q-Q plot for that "before" group
# qqnorm(ambiguous$Comprehension[ambiguous$Condition=="Before"], ylab = "Essay scores"); qqline(ambiguous$Comprehension[ambiguous$Condition=="Before"])


Dealing with violations of conditions

What can we do if the conditions aren’t satisfied (or if the assumptions aren’t reasonable)? We’ll deal with this issue all semester. Some options include:

  • Transform your data to force it to better approximate a normal distribution (perhaps by using logarithms or Box-Cox transformation).
  • Use a nonparametric test (randomization-based methods or a test like the Kruskal-Wallis test)
  • Use robust ANOVA procedures (which we’ll investigate later in the course)
  • Use Bayesian methods (which we’ll investigate a bit in this course and in-detail in STAT 305)


Recall from the first lesson that if we want to conduct a t-test without a normality assumption, we could use the Welch-Satterthwaite approximation. We can also do this in ANOVA.

Expand the code to see how –>

# Notice the degrees of freedom differ from our "regular" ANOVA
oneway.test(Comprehension ~ Condition, data=ambiguous, var.equal=FALSE)

Randomized mean-square-ratios

We’ve seen how to conduct randomization-based tests of the SAD statistic (which do not account for variance within each group). We’ve also seen how to conduct an ANOVA (which requires conditions of normality and equal variances). Can we get the best of both worlds?

Let’s see if we can conduct a randomization-based test using the F-statistic (mean square ratio) as our test statistic. Expand the code to see how this is done.

# We'll store our observed mean square ratio of 10.012 as MSRobserved
# I could have used:  MSRobserved <- 10.012
MSRobserved <- tidy(aov(Comprehension~Condition, data=ambiguous))$statistic[1]

# 10,000 replications, shuffling the Condition each time
MSRrandomized <- Do(10000) * tidy(aov(Comprehension ~ shuffle(Condition), data=ambiguous))$statistic[1]

# Plot results
ggplot(data = MSRrandomized, aes(x = result)) +
  geom_density(fill="lightblue", color="white", alpha = 0.8) +
  annotate("segment", x = MSRobserved, xend = MSRobserved, y = 0, yend = .2, color = "red") +
  annotate("text", x = MSRobserved, y = .25, label = "observed MSR = 10.012", color = "red") +
  labs(
      title = "Randomized mean square ratios",
      x = "mean square ratios"
      ) +
  scale_x_continuous(limits = c(0,12), breaks=seq(0, 12, 2), minor_breaks=NULL) +
    theme(
    axis.text.x = element_text(size = 11, color="grey10"),
    legend.position = "none",
    panel.grid.major.y = element_line(colour = "white"),
    panel.grid.major.x = element_line(colour = "white", size=.15),
    panel.grid.minor = element_blank(),
    panel.background = element_rect(fill = "grey93")
  )

# Estimate p-value
pvalue <- prop(MSRrandomized$result >= MSRobserved)
pvalue

ANOVA vs. t-test

Let’s (finally) finish things up by looking at the relationship between the F-statistic (mean square ratio) in an ANOVA and the t-statistic in a t-test:


\(t_{\textrm{n}_1+\textrm{n}_2-2}=\frac{\overline{X}_{1}-\overline{X}_{2}}{\sqrt{\frac{1}{\textrm{n}_1}+\frac{1}{\textrm{n}_2}}\sqrt{\frac{\left ( \textrm{n}_1-1 \right )s_{1}^{2}+\left ( \textrm{n}_2-1 \right )s_{2}^{2}}{\textrm{n}_1+\textrm{n}_2-2}}}=\frac{\overline{X}_{1}-\overline{X}_{2}}{\sqrt{\frac{1}{\textrm{n}_1}+\frac{1}{\textrm{n}_2}}\sqrt{s_{pooled}^{2}}}\)


\(t_{{\textrm{n}_1+\textrm{n}_2-2}}^{2}=\frac{\left (\overline{X}_{1}-\overline{X}_{2} \right )^{2}}{\left ( \frac{1}{n_{1}}+ \frac{1}{n_{2}}\right )}s_{pooled}^{2}=\frac{\left ( n_{1}+n_{2} \right )\left (\overline{X}_{1}-\overline{X}_{2} \right )^{2}}{s_{pooled}^{2}}=\frac{n_{1}\left (\overline{X}_{1}-\overline{X}_{2} \right )^{2}+n_{2}\left (\overline{X}_{1}-\overline{X}_{2} \right )^{2}}{{s_{pooled}^{2}}}\)


\(t_{{\textrm{n}_1+\textrm{n}_2-2}}^{2}=\frac{\sum n_{a}\left (\overline{X}_{a}-M \right )^{2}/1}{\textrm{MS}_{\textrm{E}}}=\frac{\textrm{SS}_{\textrm{A}}/\textrm{df}_{\textrm{A}}}{\textrm{MS}_{\textrm{E}}}=\frac{\textrm{MS}_{\textrm{A}}}{\textrm{MS}_{\textrm{E}}}=F\)


  1. Explain the consequences of what was just derived. Also, verify this result with the following output.
# Recreate the spider dataset from lesson #1
spider <- tibble(
  group = c( rep("photo", 12), rep("real", 12) ),
  anxiety = c(30, 35, 45, 40, 50, 35, 55, 25, 30, 45, 40, 50, 
              40, 35, 50, 55, 65, 55, 50, 35, 30, 50, 60, 39))

# Run a t-test
t.test(anxiety ~ group, data=spider, alternative = c("less"), var.equal = TRUE, conf.level = 0.95)
# Store the value of the t-statistic (-1.68)
tstat <- t.test(anxiety ~ group, data=spider, alternative = c("less"), var.equal = TRUE, conf.level = 0.95)$statistic

# Run an ANOVA on the same data
anova(aov(anxiety ~ group, data=spider))
# Notice the F-statistic = 2.8269
# Is that the same as the tstat squared?  Let's see...
tstat^2
Condition n mean sd
After 19 3.211 1.398
Before 19 4.947 1.311
None 19 3.368 1.257
Combined N = 57 M = 3.842 s = 1.521

aov( anova(Comprehension ~ Condition, data = ambiguous) )


Your turn

  1. Are some diets more effective than others? 93 subjects were randomly assigned to one of four diets:

      (a) Atkins, a low-carb diet,   (b) Zone, a 4-4-3 ratio of carbs-protein-fat,
      (c) Ornish, a low-fat diet, or (d) Weight Watchers.

    Subjects were educated on their assigned diet and were monitored as they stayed on the diet for one year. At the end of the year, researchers calculated the changes in weight for each subject.

    (a) Suppose used t-tests (with \(\alpha=0.05\) fir each test) to compare all possible pairs of group means. Calculate the overall probability of making at least one Type I error across all the t-tests.

    (b) Check the conditions necessary to conduct an ANOVA. Choose your methods and interpret the results.

    (c) Conduct an ANOVA on this data (with or without the equal variance assumption) and write out any conclusions you can draw.

    (d) Calculate eta-squared and omega-squared.

    (e) Conduct a randomized test of the SAD (or MAD) statistic for this data. Estimate the p-value.

    The diet dataframe has been loaded.
    The variables are Diet (a factor variable identifying which diet each subject is assigned to) and Weightloss (the number of pounds lost by each subject by the end of the study).

    A summary of the data is displayed below. Notice the variables begin with Capital letters and that negative values of the Weightloss variable indicate subjects who gained weight.

Publishing Your Turn solutions


Sources

This document is released under a Creative Commons Attribution-ShareAlike 3.0 Unported license.

LS0tCnRpdGxlOiAiQ29tcGFyaW5nIDIrIG1lYW5zOiBBTk9WQSIKb3V0cHV0OgogIGh0bWxfZG9jdW1lbnQ6CiAgICBjb2RlX2ZvbGRpbmc6IGhpZGUKICAgIGNzczogaHR0cDovL3d3dy5icmFkdGhpZXNzZW4uY29tL2JhdGxhYjMuY3NzCiAgICBkZl9wcmludDogdGliYmxlCiAgICBmaWdfaGVpZ2h0OiAzLjkKICAgIGZpZ193aWR0aDogNi4zCiAgICBoaWdobGlnaHQ6IHB5Z21lbnRzCiAgICB0aGVtZTogc3BhY2VsYWIKICAgIHRvYzogeWVzCiAgICB0b2NfZmxvYXQ6IHllcwogIGh0bWxfbm90ZWJvb2s6CiAgICBjb2RlX2ZvbGRpbmc6IGhpZGUKICAgIGNzczogaHR0cDovL3d3dy5icmFkdGhpZXNzZW4uY29tL2JhdGxhYjMuY3NzCiAgICBmaWdfaGVpZ2h0OiAzLjkKICAgIGZpZ193aWR0aDogNi4zCiAgICBoaWdobGlnaHQ6IHB5Z21lbnRzCiAgICB0aGVtZTogc3BhY2VsYWIKICAgIHRvYzogeWVzCi0tLQoKYGBge3IgJ2dsb2JhbCBvcHRpb25zJywgZWNobz1GQUxTRSwgbWVzc2FnZT1GQUxTRSwgcmVzdWx0cz0naGlkZSd9CmtuaXRyOjpvcHRzX2NodW5rJHNldCgKICBjb21tZW50ID0gIiMgICAiLAogIGNvbGxhcHNlID0gVFJVRSwKICBmaWcuaGVpZ2h0ID0gMy45LAogIGZpZy53aWR0aCA9IDYuMwopCmBgYAoKYGBge3IgJ3ByZXJlcXMnLCBtZXNzYWdlPUZBTFNFfQojIExvYWQgdGhlIHRpZHl2ZXJzZSBhbmQgbW9zYWljIHBhY2thZ2VzCmxpYnJhcnkodGlkeXZlcnNlKQpsaWJyYXJ5KG1vc2FpYykKYGBgCgoqKioqKgoKIyBTY2VuYXJpbzogIEFtYmlndW91cyBQcm9zZQoKSW4gWzE5NzJdKGh0dHA6Ly9tZW1sYWIueWFsZS5lZHUvc2l0ZXMvZGVmYXVsdC9maWxlcy9maWxlcy8xOTcyX0JyYW5zZm9yZF9Kb2huc29uX0pWTFZCLnBkZiksIDU3IGhpZ2ggc2Nob29sIHN0dWRlbnRzIHdlcmUgcmFuZG9tbHkgYXNzaWduZWQgdG8gb25lIG9mIHRocmVlIGdyb3VwczoKCgotIDE5IHN0dWRlbnRzIHdlcmUgc2hvd24gYSBwaWN0dXJlICoqYmVmb3JlKiogcmVhZGluZyBhIHBhc3NhZ2UKLSAxOSBzdHVkZW50cyB3ZXJlIHNob3duIGEgcGljdHVyZSAqKmFmdGVyKiogcmVhZGluZyBhIHBhc3NhZ2UKLSAxOSBzdHVkZW50cyB3ZXJlICoqbm90IHNob3duIHRoZSBwaWN0dXJlIGF0IGFsbCoqCgpUaGUgc3R1ZGVudHMgd2VyZSBhc2tlZCB0byByZWFkIHRoZSBmb2xsb3dpbmcgcGFzc2FnZSBhbmQgcmVtZW1iZXIgZGV0YWlscyBmcm9tIGl0OgoKIyMjIyB7LnRhYnNldH0KCiMjIyMjIFBhc3NhZ2UKCj4gSWYgdGhlIGJhbGxvb25zIHBvcHBlZCwgdGhlIHNvdW5kIHdvdWxkbid0IGJlIGFibGUgdG8gY2Fycnkgc2luY2UgZXZlcnl0aGluZyB3b3VsZCBiZSB0b28gZmFyIGF3YXkgZnJvbSB0aGUgY29ycmVjdCBmbG9vci4gQSBjbG9zZWQgd2luZG93IHdvdWxkIGFsc28gcHJldmVudCB0aGUgc291bmQgZnJvbSBjYXJyeWluZywgc2luY2UgbW9zdCBidWlsZGluZ3MgdGVuZCB0byBiZSB3ZWxsIGluc3VsYXRlZC4gU2luY2UgdGhlIHdob2xlIG9wZXJhdGlvbiBkZXBlbmRzIG9uIGEgc3RlYWR5IGZsb3cgb2YgZWxlY3RyaWNpdHksIGEgYnJlYWsgaW4gdGhlIG1pZGRsZSBvZiB0aGUgd2lyZSB3b3VsZCBhbHNvIGNhdXNlIHByb2JsZW1zLiBPZiBjb3Vyc2UsIHRoZSBmZWxsb3cgY291bGQgc2hvdXQsIGJ1dCB0aGUgaHVtYW4gdm9pY2UgaXMgbm90IGxvdWQgZW5vdWdoIHRvIGNhcnJ5IHRoYXQgZmFyLiBBbiBhZGRpdGlvbmFsIHByb2JsZW0gaXMgdGhhdCBhIHN0cmluZyBjb3VsZCBicmVhayBvbiB0aGUgaW5zdHJ1bWVudC4gVGhlbiB0aGVyZSBjb3VsZCBiZSBubyBhY2NvbXBhbmltZW50IHRvIHRoZSBtZXNzYWdlLiBJdCBpcyBjbGVhciB0aGF0IHRoZSBiZXN0IHNpdHVhdGlvbiB3b3VsZCBpbnZvbHZlIGxlc3MgZGlzdGFuY2UuIFRoZW4gdGhlcmUgd291bGQgYmUgZmV3ZXIgcG90ZW50aWFsIHByb2JsZW1zLiBXaXRoIGZhY2UgdG8gZmFjZSBjb250YWN0LCB0aGUgbGVhc3QgbnVtYmVyIG9mIHRoaW5ncyBjb3VsZCBnbyB3cm9uZy4KClN0dWRlbnRzIHdlcmUgYXNrZWQgdG86CgotIFJhdGUgaG93IGVhc3kgaXQgd2FzIHRvICoqY29tcHJlaGVuZCoqIHRoZSBwYXNzYWdlIG9uIGEgMS03IHNjYWxlICgxID0gKnZlcnkgZGlmZmljdWx0KjsgNyA9ICp2ZXJ5IGVhc3kqKQotICoqUmVjYWxsKiogdGhlIHBhc3NhZ2UgYnkgd3JpdGluZyBvdXQgYXMgbWFueSBvZiB0aGUgaWRlYXMgYXMgeW91IHJlbWVtYmVyLgoKKipEb2VzIHNlZWluZyB0aGUgcGljdHVyZSBpbmZsdWVuY2UgYSBzdHVkZW50J3MgYWJpbGl0eSB0byBjb21wcmVoZW5kIGFuZC9vciByZWNhbGwgdGhlIHBhc3NhZ2U/KioKCjxiciAvPgoKIyMjIyMgUGljdHVyZQoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDo0MDBweCI+IVtdKGx1bHdvdC5wbmcpPC9kaXY+CgoKIyMgRGF0YSBFeHBsb3JhdGlvbgoJCjxiciAvPgoKRm9yIG5vdywgbGV0J3MgZm9jdXMgb24gdGhlICoqY29tcHJlaGVuc2lvbioqIHNjb3Jlcy4gIFRoZSBgY29kZWAgc2hvd3MgaG93IHRoZSBkYXRhIHdlcmUgaW1wb3J0ZWQgLS0+CgpgYGB7ciAnZGF0YS1zaW11bGF0aW9uJ30KIyBXZSdsbCBkb3dubG9hZCB0aGUgZGF0YSBmcm9tIG15IHdlYnNpdGUKYW1iaWd1b3VzIDwtIHJlYWQuY3N2KCJodHRwOi8vd3d3LmJyYWR0aGllc3Nlbi5jb20vaHRtbDUvZGF0YS9hbWJpZ3VvdXNwcm9zZS5jc3YiKQoKIyBMb29rIGF0IHRoZSBmaXJzdCBzZXZlcmFsIHJvd3MgdG8gZ2V0IGEgc2Vuc2Ugb2YgdGhlIGRhdGEKaGVhZChhbWJpZ3VvdXMpCgojIEdldCBzdW1tYXJ5IHN0YXRpc3RpY3MKYW1iaWd1b3VzICU+JQogIGdyb3VwX2J5KENvbmRpdGlvbikgJT4lCiAgc3VtbWFyaXplKG4gPSBuKCksCiAgICAgICAgICAgIG1lYW4gPSBtZWFuKENvbXByZWhlbnNpb24pLAogICAgICAgICAgICBzZCA9IHNkKENvbXByZWhlbnNpb24pKQoKYW1iaWd1b3VzICU+JQogIHN1bW1hcml6ZShuID0gbigpLAogICAgICAgICAgICBtZWFuID0gbWVhbihDb21wcmVoZW5zaW9uKSwKICAgICAgICAgICAgc2QgPSBzZChDb21wcmVoZW5zaW9uKSkKCmBgYAoKfCBDb25kaXRpb24gICAgIHwgbiAgICAgICAgICB8IG1lYW4gICAgICAgICAgfCBzZCAgICAgICAgICAgIHwKfC0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLTp8LS0tLS0tLS0tLS0tLS06fC0tLS0tLS0tLS0tLS0tOnwKfCBBZnRlciAgICAgICAgIHwgMTkgICAgICAgICB8IDMuMjExICAgICAgICAgfCAxLjM5OCAgICAgICAgIHwKfCBCZWZvcmUgICAgICAgIHwgMTkgICAgICAgICB8IDQuOTQ3ICAgICAgICAgfCAxLjMxMSAgICAgICAgIHwKfCBOb25lICAgICAgICAgIHwgMTkgICAgICAgICB8IDMuMzY4ICAgICAgICAgfCAxLjI1NyAgICAgICAgIHwKfCAqKkNvbWJpbmVkKiogIHwgKipOID0gNTcqKiB8ICoqTSA9IDMuODQyKiogfCAqKnMgPSAxLjUyMSoqIHwKCjxiciAvPgoKCjEuIFdoYXQgZG8gKipNKiogYW5kICoqcyoqIHJlcHJlc2VudD8gIEZyb20gdGhlc2Ugc3VtbWFyeSBzdGF0aXN0aWNzLCBkbyB5b3UgdGhpbmsgd2UnbGwgY29uY2x1ZGUgdGhlIHBpY3R1cmUgaW5mbHVlbmNlZCBzdHVkZW50IGNvbXByZWhlbnNpb24/Cgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0Ojc1cHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoKIyMjIHsudGFic2V0IC50YWJzZXQtZmFkZX0KCiMjIyMgRG90cGxvdAoKYGBge3IgJ2RvdHBsb3QnLCBtZXNzYWdlPUZBTFNFLCBmaWcuaGVpZ2h0ID0gMy45LCBmaWcud2lkdGggPSA0LjV9CgphbWJpZ3VvdXMgJT4lCiAgZ2dwbG90KGFlcyh4ID0gQ29uZGl0aW9uLCB5ID0gQ29tcHJlaGVuc2lvbikpICsKICBnZW9tX2RvdHBsb3QoYWVzKGZpbGw9Q29uZGl0aW9uKSwgYmluYXhpcyA9ICJ5IiwgYmlucG9zaXRpb25zPSJhbGwiLCBjb2xvciA9ICJ3aGl0ZSIpICsKICBzY2FsZV95X2NvbnRpbnVvdXMoYnJlYWtzPXNlcSgwLCA3LCAxKSwgbWlub3JfYnJlYWtzPU5VTEwpICsKICB0aGVtZShheGlzLnRpdGxlLnggPSBlbGVtZW50X3RleHQoc2l6ZSA9IDEyLCBjb2xvciA9ICIjNzc3Nzc3IikpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIpKSArCiAgdGhlbWUoYXhpcy50aXRsZS55ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMiwgY29sb3I9IiM3Nzc3NzciKSkgKwogIHRoZW1lKGF4aXMudGV4dC55ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMikpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb249Im5vbmUiKSArCiAgbGFicygKICAgIHRpdGxlID0gIkNvbXByZWhlbnNpb24gYnkgQ29uZGl0aW9uIgogICkgKwogIGNvb3JkX2ZsaXAoeGxpbSA9IE5VTEwsIHlsaW0gPSBOVUxMLCBleHBhbmQgPSBUUlVFKQpgYGAKCgojIyMjIEJveHBsb3QKCmBgYHtyICdkZW5zaXR5JywgbWVzc2FnZT1GQUxTRSwgZmlnLmhlaWdodCA9IDMuOSwgZmlnLndpZHRoID0gNC41fQoKYW1iaWd1b3VzICU+JQogIGdncGxvdChhZXMoeCA9IENvbmRpdGlvbiwgeSA9IENvbXByZWhlbnNpb24pKSArCiAgZ2VvbV9kb3RwbG90KGFlcyhmaWxsPUNvbmRpdGlvbiksIGJpbmF4aXMgPSAieSIsIGJpbnBvc2l0aW9ucz0iYWxsIiwgY29sb3IgPSAid2hpdGUiLCBhbHBoYT0wLjUpICsKICBnZW9tX2JveHBsb3QoZmlsbCA9ICJ3aGl0ZSIsIGNvbG9yID0gImJsYWNrIiwgYWxwaGEgPSAwLjYpICsKICBzY2FsZV95X2NvbnRpbnVvdXMoYnJlYWtzPXNlcSgwLCA3LCAxKSwgbWlub3JfYnJlYWtzPU5VTEwpICsKICB0aGVtZShheGlzLnRpdGxlLnggPSBlbGVtZW50X3RleHQoc2l6ZSA9IDEyLCBjb2xvciA9ICIjNzc3Nzc3IikpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIpKSArCiAgdGhlbWUoYXhpcy50aXRsZS55ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMiwgY29sb3I9IiM3Nzc3NzciKSkgKwogIHRoZW1lKGF4aXMudGV4dC55ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMikpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb249Im5vbmUiKSArCiAgbGFicygKICAgIHRpdGxlID0gIkNvbXByZWhlbnNpb24gYnkgQ29uZGl0aW9uIgogICkgKwogIGNvb3JkX2ZsaXAoeGxpbSA9IE5VTEwsIHlsaW0gPSBOVUxMLCBleHBhbmQgPSBUUlVFKQoKYGBgCgoKIyMjIyBNZWFucyB3aXRoIHN0YW5kYXJkIGVycm9ycwoKYGBge3IgJ2NvbmRpdGlvbmFsLW1lYW5zLXBsb3QnLCBtZXNzYWdlPUZBTFNFLCBmaWcuaGVpZ2h0ID0gMy45LCBmaWcud2lkdGggPSA0LjV9CgphbWJpZ3VvdXMgJT4lCiAgZ2dwbG90KGFlcyh4ID0gQ29uZGl0aW9uLCB5ID0gQ29tcHJlaGVuc2lvbikpICsKICBzdGF0X3N1bW1hcnkoZnVuLnkgPSBtZWFuLCBnZW9tID0gInBvaW50IikgKyAKICBzdGF0X3N1bW1hcnkoZnVuLnkgPSBtZWFuLCBnZW9tID0gImxpbmUiLCBhZXMoZ3JvdXAgPSAxKSwKICAgICAgICAgICAgICAgY29sb3VyID0gIkJsdWUiLCBsaW5ldHlwZSA9ICJkYXNoZWQiKSArCiAgc3RhdF9zdW1tYXJ5KGZ1bi5kYXRhID0gbWVhbl9jbF9ib290LCBnZW9tID0gImVycm9yYmFyIiwgd2lkdGggPSAwLjIpICsKICBsYWJzKHggPSAiVHJlYXRtZW50IiwgeSA9ICJNZWFuIENvbXByZWhlbnNpb24gU2NvcmUiKQpgYGAKCgojIyBIeXBvdGhlc2VzCgoyLiBXcml0ZSBvdXQgbnVsbCBhbmQgYWx0ZXJuYXRpdmUgaHlwb3RoZXNlcyB0byBjb21wYXJlIHRoZSBtZWFucyBvZiBhbGwgdGhyZWUgZ3JvdXBzLiAgSG93IGNvdWxkIHdlIGRldGVybWluZSB0aGUgbGlrZWxpaG9vZCBvZiB0aGlzIG51bGwgaHlwb3RoZXNpcz8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoKKioqKioKCiMjIElkZWEgIzE6IG11bHRpcGxlIHQtdGVzdHMKCklmIGFuIGluZGVwZW5kZW50IHNhbXBsZXMgdC10ZXN0IGNvbXBhcmVzIHR3byBncm91cCBtZWFucywgY291bGQgd2UgdXNlIG11bHRpcGxlIHQtdGVzdHMgdG8gY29tcGFyZSBhbGwgcG9zc2libGUgcGFpcnMgb2Ygb3VyIHRocmVlIGdyb3VwIG1lYW5zPwoKPGJyIC8+CgozLiBIb3cgbWFueSB0LXRlc3RzIHdvdWxkIHdlIG5lZWQgdG8gdGVzdCBhbGwgcG9zc2libGUgcGFpcnMgb2YgdGhyZWUgZ3JvdXAgbWVhbnM/ICBIb3cgbWFueSB0LXRlc3RzIHdvdWxkIHdlIG5lZWQgaWYgd2UgaGFkICoqRyoqIGdyb3Vwcz8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+Cgo0LiBTdXBwb3NlIHdlIGNvbmR1Y3QgMyB0LXRlc3RzLiAgSWYgd2Ugc2V0ICRcYWxwaGEgPTAuMDUkIGZvciBlYWNoIHRlc3QsIGhvdyBsaWtlbHkgd291bGQgd2UgYmUgdG8gbWFrZSAqKmF0IGxlYXN0IG9uZSoqIGFscGhhIGVycm9yIGluIG91ciAzIHRlc3RzPyAgSW4gb3RoZXIgd29yZHMsIHdoYXQncyAkUChcbWF0aHJte2F0XCAgbGVhc3RcIG9uZVwgfVxhbHBoYVwgXG1hdGhybXtlcnJvclwgaW5cIDNcIHRlc3RzfSkkPwoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoxNTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCjUuIFN1cHBvc2Ugd2UgY29uZHVjdCB0LXRlc3RzIHRvIGNvbXBhcmUgYWxsIHBvc3NpYmxlIHBhaXJzIG9mIG1lYW5zIGZyb20gKipHKiogZ3JvdXBzLiAgSWYgd2UgaGF2ZSBhIFR5cGUgSSBlcnJvciByYXRlIG9mICRcYWxwaGEkIGZvciBlYWNoIHRlc3QsIHdoYXQgaXMgdGhlIHByb2JhYmlsaXR5IHdlIG1ha2UgKiphdCBsZWFzdCBvbmUqKiBhbHBoYSBlcnJvciBhY3Jvc3MgYWxsIG91ciB0ZXN0cz8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgojIyMgUHJvYmxlbXMgd2l0aCBtdWx0aXBsZSB0LXRlc3RzCgo2LiBFeHBsYWluIHdoeSB3ZSBzaG91bGQgKipub3QqKiBjb25kdWN0IG11bHRpcGxlIHQtdGVzdHMgdG8gY29tcGFyZSBwYWlycyBvZiBncm91cCBtZWFucyBmcm9tIGEgc2luZ2xlIGRhdGFzZXQuCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjE1MHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKCkRlc3BpdGUgdGhpcyBwcm9ibGVtLCBsZXQncyBjb25kdWN0IG11bHRpcGxlIHQtdGVzdHMgdXNpbmcgdGhlIGBwYWlyd2lzZS50LnRlc3QoKWAgY29tbWFuZCB0byBydW4gYWxsIHBvc3NpYmxlIHQtdGVzdHMgZm9yIGEgZGF0YXNldC4KCmBgYHtyfQojIFdhaXQhISEhISEhISEhCiMgV2h5IHdvdWxkIFIgaGF2ZSBhIHBhaXJ3aXNlLnQudGVzdCBmdW5jdGlvbiBpZiB3ZSdyZSBOT1QKIyBzdXBwb3NlZCB0byBjb25kdWN0IG11bHRpcGxlIHQtdGVzdHM/ICBHb29kIHF1ZXN0aW9uLgojIFdlJ2xsIGNvbWUgYmFjayB0byB0aGlzIHdoZW4gd2UgbGVhcm4gYWJvdXQgcG9zdCBob2MgdGVzdHMuCgojIFRvIHVzZSB0aGlzIGZ1bmN0aW9uLCB3ZSBuZWVkIHRvIGlkZW50aWZ5IHRoZSB2YXJpYWJsZXMgd2l0aGluCiMgdGhlIGRhdGFmcmFtZSB1c2luZyB0aGUgJCBpZGVudGlmaWVyLgojIHAuYWRqdXN0Lm1ldGhvZD0ibm9uZSIganVzdCBtZWFucyBJIGRvbid0IHdhbnQgUiB0byBtb2RpZnkgdGhlIHAtdmFsdWUKIyBJbiB0aGUgbmV4dCBsZXNzb24sIHdlJ2xsIHNlZSB3aHkgd2UgbWlnaHQgd2FudCB0byBtb2RpZnkgdGhlIHAtdmFsdWUKcGFpcndpc2UudC50ZXN0KGFtYmlndW91cyRDb21wcmVoZW5zaW9uLCBhbWJpZ3VvdXMkQ29uZGl0aW9uLCBwLmFkanVzdC5tZXRob2Q9Im5vbmUiKQoKCmBgYAoKPGJyIC8+Cgo3LiBUaGUgb3V0cHV0IHN0YXRlcyB0aGUgdC10ZXN0cyB1c2VkIGEgInBvb2xlZCBTRC4iICBXaGF0IGRvZXMgdGhhdCBtZWFuPwoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDo1MHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKCjguIFdoYXQgY291bGQgd2UgY29uY2x1ZGUgZnJvbSB0aGUgcC12YWx1ZXMgbGlzdGVkIGluIHRoZSBvdXRwdXQ/ICBXaHkgYXJlIHdlIHVuY2VydGFpbiBhYm91dCBvdXIgY29uY2x1c2lvbnM/ICBJcyB0aGVyZSBhbnl0aGluZyB3ZSBjb3VsZCBkbyB0byByZWR1Y2UgdGhpcyB1bmNlcnRhaW50eT8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoqKioqKgoKIyMgSWRlYSAjMjogZ2V0IE1BRCAob3IgU0FEKQoKSWYgcnVubmluZyBtdWx0aXBsZSB0LXRlc3RzIGlzIHByb2JsZW1hdGljLCBjb3VsZCB3ZSBjb25kdWN0IGEgc2luZ2xlIHRlc3Qgb2Ygb3VyIHRocmVlIGdyb3VwIG1lYW5zIHVzaW5nIHJhbmRvbWl6YXRpb24tYmFzZWQgbWV0aG9kcz8KClN1cmUhICBJZiB3ZSBjYW4gaWRlbnRpZnkgYSBzaW5nbGUgdGVzdCBzdGF0aXN0aWMgdGhhdCBjb21wYXJlcyBhbGwgdGhyZWUgZ3JvdXAgbWVhbnMsIHdlIGNhbiB1c2UgcmFuZG9taXphdGlvbi0gb3IgdGhlb3J5LWJhc2VkIG1ldGhvZHMuCgo8YnIgLz4KCjkuIFdoYXQgY291bGQgd2UgY29uY2x1ZGUgZnJvbSB0aGUgcC12YWx1ZXMgbGlzdGVkIGluIHRoZSBvdXRwdXQ/ICBXaHkgYXJlIHdlIHVuY2VydGFpbiBhYm91dCBvdXIgY29uY2x1c2lvbnM/ICBJcyB0aGVyZSBhbnl0aGluZyB3ZSBjb3VsZCBkbyB0byByZWR1Y2UgdGhpcyB1bmNlcnRhaW50eT8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgojIyMgVGVzdCBzdGF0aXN0aWMKClRvIGNvbXBhcmUgdHdvIG1lYW5zLCB3ZSB1c2U6ICAkXG92ZXJsaW5le1h9X3sxfS1cb3ZlcmxpbmV7WH1fezJ9JC4gIElmIHR3byBtZWFucyBhcmUgc2ltaWxhciwgdGhlIHRlc3Qgc3RhdGlzdGljIGlzIG5lYXIgemVyby4gIEFzIHRoZSBtZWFucyBkaWZmZXIgYnkgYSBncmVhdGVyIGFtb3VudCwgdGhlIHZhbHVlIG9mIHRoZSB0ZXN0IHN0YXRpc3RpYyBpbmNyZWFzZXMuCgo8YnIgLz4KCjEwLiBQcm9wb3NlIGEgc2luZ2xlIHRlc3Qgc3RhdGlzdGljIHRvIG1lYXN1cmUgdGhlIGRpZmZlcmVuY2UgYW1vbmcgMyBncm91cCBtZWFucy4gIFdoYXQgaGFwcGVucyB0byB0aGUgdmFsdWUgb2YgdGhhdCB0ZXN0IHN0YXRpc3RpYyBhcyB0aGUgZ3JvdXAgbWVhbnMgZ2V0IGNsb3NlciB0b2dldGhlciBvciBmdXJ0aGVyIGFwYXJ0PwoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoxNTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KClVzaW5nIHRoZSBtZWFucyBvZiBvdXIgdGhyZWUgZ3JvdXBzOgoKLSAkXG92ZXJsaW5le1h9X3thZnRlcn09XG92ZXJsaW5le1h9X3thfT0zLjIxMSQKLSAkXG92ZXJsaW5le1h9X3tiZWZvcmV9PVxvdmVybGluZXtYfV97Yn09NC45NDckCi0gJFxvdmVybGluZXtYfV97bm9uZX09XG92ZXJsaW5le1h9X3tufT0zLjM2OCQKCjxiciAvPgoKV2UgY2FuIGNhbGN1bGF0ZToKCiRTQUQgPSB8XG92ZXJsaW5le1h9X3thfS1cb3ZlcmxpbmV7WH1fe2J9fCt8XG92ZXJsaW5le1h9X3thfS1cb3ZlcmxpbmV7WH1fe259fCt8XG92ZXJsaW5le1h9X3tifS1cb3ZlcmxpbmV7WH1fe259fD0zLjQ3NCQKCiRNQUQgPSBcZnJhY3t8XG92ZXJsaW5le1h9X3thfS1cb3ZlcmxpbmV7WH1fe2J9fCt8XG92ZXJsaW5le1h9X3thfS1cb3ZlcmxpbmV7WH1fe259fCt8XG92ZXJsaW5le1h9X3tifS1cb3ZlcmxpbmV7WH1fe259fH17M309MS4xNTgkCgo8YnIgLz4KCkV4cGFuZCB0aGUgY29kZSB0byBzZWUgaG93IHRvIGNhbGN1bGF0ZSB0aGlzIGluIFIgLS0+CgpgYGB7ciAnY2FsY3VsYXRlLVNBRC1NQUQnfQpTQUQoIG1lYW4oQ29tcHJlaGVuc2lvbiB+IENvbmRpdGlvbiwgZGF0YT1hbWJpZ3VvdXMpICkKTUFEKCBtZWFuKENvbXByZWhlbnNpb24gfiBDb25kaXRpb24sIGRhdGE9YW1iaWd1b3VzKSApCgp0ZXN0X3N0YXQgPC0gU0FEKCBtZWFuKENvbXByZWhlbnNpb24gfiBDb25kaXRpb24sIGRhdGE9YW1iaWd1b3VzKSApCmBgYAoKV2UnbGwgc3RvcmUgYFNBRCA9IGByIHRlc3Rfc3RhdGBgIGFzIG91ciBgdGVzdF9zdGF0YC4KCjxiciAvPgoKIyMjIFJhbmRvbWl6YXRpb24tYmFzZWQgU0FEIHRlc3QKCjExLiBBc3N1bWUgeW91ciBudWxsIGh5cG90aGVzaXMgaXMgdHJ1ZS4gIEluIG91ciBzYW1wbGUgZGF0YSwgdGhlIGZpcnN0IHN1YmplY3QgaW4gdGhlICoqYWZ0ZXIqKiBncm91cCByYXRlZCB0aGUgcGFzc2FnZSBhICoqNioqIGluIGNvbXByZWhlbnNpb24uICBJZiB3ZSB3ZW50IGJhY2sgaW4gdGltZSBhbmQgcmFuZG9tbHkgYXNzaWduZWQgdGhpcyBzdWJqZWN0IHRvIHRoZSAqKm5vbmUqKiBncm91cCwgd2hhdCByYXRpbmcgd291bGQgdGhlIHN1YmplY3QgZ2l2ZSB0byB0aGUgcGFzc2FnZT8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6NTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCjEyLiBIb3cgbWFueSB3YXlzIGNvdWxkIDU3IHN1YmplY3RzIGJlIGFzc2lnbmVkIHRvIDMgZ3JvdXBzICh3aXRoIDE5IGluIGVhY2ggZ3JvdXApPyAgRXhwYW5kIHRoZSBjb2RlIHRvIHNlZSB0aGlzIGNhbGN1bGF0aW9uLgoKCmBgYHtyICdudW1iZXItb2YtcmFuZG9taXphdGlvbnMnfQojIEVsaW1pbmF0ZSBzY2llbnRpZmljIG5vdGF0aW9uCm9wdGlvbnMoc2NpcGVuPTk5OSkKCiMgQ2FsY3VsYXRlIG51bWJlciBvZiByYW5kb21pemF0aW9ucyAoZ3JvdXAgSUQgZG9lcyBub3QgbWF0dGVyKQp3YXlzIDwtICggY2hvb3NlKG49NTcsIGs9MTkpICogY2hvb3NlKDM4LCAxOSkgKiBjaG9vc2UoMTksIDE5KSApIC8gZmFjdG9yaWFsKDMpCgojIFB1dCBjb21tYSBzZXBhcmF0b3JzIGludG8gdGhlIG51bWJlcgpmb3JtYXQod2F5cywgYmlnLm1hcms9IiwiLCBzY2llbnRpZmljPUZBTFNFKQoKYGBgCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjEwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKV2UncmUgbmV2ZXIgZ29pbmcgdG8gbGlzdCBhbGwgdGhvc2UgcmFuZG9taXphdGlvbnMsIGJ1dCB3ZSBjYW4gZ2V0IGEgcmFuZG9tLCByZXByZXNlbnRhdGl2ZSBzYW1wbGUgb2YgdGhlbS4gIEZvciBlYWNoIHJhbmRvbWl6YXRpb24sIHdlJ2xsIGNhbGN1bGF0ZSBTQUQuICBXZSdsbCB0aGVuIHBsb3Qgb3VyIGRpc3RyaWJ1dGlvbiBvZiByYW5kb21pemVkIFNBRHMgYW5kIGRldGVybWluZSB0aGUgbGlrZWxpaG9vZCB0aGF0IG91ciB0ZXN0IHN0YXRpc3RpYyBjYW1lIGZyb20gdGhpcyBkaXN0cmlidXRpb24uCgpgYGB7ciAncmFuZG9taXplZC1TQUQtdGVzdCd9CiMgUmFuZG9taXplIG91ciBkYXRhIDEwLDAwMCB0aW1lcyAoc2h1ZmZsaW5nIGNvbmRpdGlvbikKU0FEcmFuZCA8LSBEbygxMDAwMCkgKiBTQUQoIG1lYW4oQ29tcHJlaGVuc2lvbiB+IHNodWZmbGUoQ29uZGl0aW9uKSwgZGF0YT1hbWJpZ3VvdXMpICkKCiMgSGVyZSdzIG9uZSB3YXkgdG8gZG8gdGhpcyBpbiBkcGx5cgojIFNBRHJhbmQgPC0gRG8oMTAwMDApICogYW1iaWd1b3VzICU+JQojICAgbXV0YXRlKHNodWZmbGVkID0gc2FtcGxlKENvbXByZWhlbnNpb24pKSAlPiUKIyAgIGdyb3VwX2J5KENvbmRpdGlvbikgJT4lCiMgICBzdW1tYXJpemUoc2h1ZmZsZWRfbWVhbnMgPSBtZWFuKHNodWZmbGVkKSkgJT4lCiMgICBtdXRhdGUoc2h1ZmZsZWRfU0FEID0gU0FEKHNodWZmbGVkX21lYW5zKSkgJT4lCiMgICBmaWx0ZXIoQ29uZGl0aW9uID09ICJBZnRlciIpICU+JSAgICMgRWxpbWluYXRlcyB0cmlwbGljYXRlIHJlc3VsdHMKIyAgIHNlbGVjdChzaHVmZmxlZF9TQUQpCgojIFBsb3QgdGhlIGRpc3RyaWJ1dGlvbgoKIyBIZXJlJ3MgdGhlIG1vc3Qgc3RyYWlnaHQtZm9yd2FyZCB3YXkgdG8gcGxvdCB0aGlzCiMgaGlzdG9ncmFtKH5TQUQsIGRhdGE9U0FEcmFuZCwKIyAgICAgICAgICAgeGxpbT1jKDAsNSksCiMgICAgICAgICAgIHdpZHRoPS4yLAojICAgICAgICAgICB4bGFiPSJQb3NzaWJsZSBtZWFuIGRpZmZlcmVuY2VzIGFzc3VtaW5nIG51bGwgaHlwb3RoZXNpcyBpcyB0cnVlIikKIyBsYWRkKHBhbmVsLmFibGluZSh2PXRlc3Rfc3RhdCkpICAgIyBBZGQgdmVydGljYWwgbGluZSBhdCB0ZXN0IHN0YXRpc3RpYwoKIyBVc2luZyBnZ3Bsb3QyIHdpdGggYSBkZW5zaXR5IHBsb3QKZ2dwbG90KGRhdGEgPSBTQURyYW5kLCBhZXMoeCA9IFNBRCkpICsKICBnZW9tX2RlbnNpdHkoZmlsbD0ibGlnaHRibHVlIiwgY29sb3I9IndoaXRlIiwgYWxwaGEgPSAwLjgpICsKICBhbm5vdGF0ZSgic2VnbWVudCIsIHggPSB0ZXN0X3N0YXQsIHhlbmQgPSB0ZXN0X3N0YXQsIHkgPSAwLCB5ZW5kID0gLjQsIGNvbG9yID0gInJlZCIpICsKICBhbm5vdGF0ZSgidGV4dCIsIHggPSB0ZXN0X3N0YXQsIHkgPSAuNDUsIGxhYmVsID0gIm9ic2VydmVkIFNBRCA9IDMuNDc0IiwgY29sb3IgPSAicmVkIikgKwogIGxhYnMoCiAgICAgIHRpdGxlID0gIlJhbmRvbWl6ZWQgU0FEIiwKICAgICAgeCA9ICJTQUQiCiAgICAgICkgKwogIHNjYWxlX3hfY29udGludW91cyhsaW1pdHMgPSBjKDAsNSksIGJyZWFrcz1zZXEoMCwgNSwgMSksIG1pbm9yX2JyZWFrcz1OVUxMKSArCiAgICB0aGVtZSgKICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMSwgY29sb3I9ImdyZXkxMCIpLAogICAgbGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiLAogICAgcGFuZWwuZ3JpZC5tYWpvci55ID0gZWxlbWVudF9saW5lKGNvbG91ciA9ICJ3aGl0ZSIpLAogICAgcGFuZWwuZ3JpZC5tYWpvci54ID0gZWxlbWVudF9saW5lKGNvbG91ciA9ICJ3aGl0ZSIsIHNpemU9LjE1KSwKICAgIHBhbmVsLmdyaWQubWlub3IgPSBlbGVtZW50X2JsYW5rKCksCiAgICBwYW5lbC5iYWNrZ3JvdW5kID0gZWxlbWVudF9yZWN0KGZpbGwgPSAiZ3JleTkzIikKICApCgoKIyBFc3RpbWF0ZSBwLXZhbHVlCnB2YWx1ZSA8LSBwcm9wKFNBRHJhbmQkU0FEID49IHRlc3Rfc3RhdCkKcHZhbHVlCmBgYAoKCjEzLiBXaGVuIHdlIGNvbXBhcmVkIHR3byBtZWFucywgdGhlICpyYW5kb21pemVkIGRpc3RyaWJ1dGlvbiBvZiBtZWFuIGRpZmZlcmVuY2VzKiB3YXMgY2VudGVyZWQgbmVhciB6ZXJvIHdpdGggYSBzeW1tZXRyaWMsIHVuaW1vZGFsIGFwcGVhcmFuY2UuPGJyIC8+PGJyIC8+V2hlbiB3ZSBjb21wYXJlZCB0d28gdmFyaWFuY2VzLCB0aGUgKnJhbmRvbWl6ZWQgZGlzdHJpYnV0aW9uIG9mIHZhcmlhbmNlIHJhdGlvcyogd2FzIGNlbnRlcmVkIG5lYXIgMS4wIHdpdGggYSBwb3NpdGl2ZSBza2V3LjxiciAvPjxiciAvPkV4cGxhaW4gd2h5IHRoZSByYW5kb21pemVkIGRpc3RyaWJ1dGlvbiBvZiBTQUQgaXMgKipub3QqKiBjZW50ZXJlZCBhdCB6ZXJvIG9yIDEuMC4gIEV4cGxhaW4gd2h5IGl0IGhhcyBhIHBvc2l0aXZlIHNrZXcuCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjEwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKCjE0LiBFeHBsYWluLCBvbmUgbGFzdCB0aW1lLCB0aGUgcHJvY2VzcyBvZiByYW5kb21pemF0aW9uLCB3aGF0IHRoZSBwbG90IHJlcHJlc2VudHMsIGFuZCBob3cgdGhlIHAtdmFsdWUgd2FzIGVzdGltYXRlZC4gIFRoZW4sIGludGVycHJldCB0aGUgcC12YWx1ZSBvZiBgciByb3VuZChwdmFsdWUsNClgLgoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoxMDBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCiMjIyBQcm9ibGVtcyB3aXRoIHJhbmRvbWl6ZWQgU0FEIChvciBNQUQpIHRlc3QKCjE1LiBUaHJlZSBzY2VuYXJpb3MgYXJlIHBsb3R0ZWQgaW4gdGhlIGRvdHBsb3RzIGRpc3BsYXllZCBiZWxvdy4gIFNjZW5hcmlvIEIgKHRoZSBtaWRkbGUgY29sdW1uKSByZXByZXNlbnRzIHRoZSBhY3R1YWwgZGF0YSBmcm9tIHRoaXMgc3R1ZHkuICBEYXRhIGZvciBzY2VuYXJpb3MgQSBhbmQgQyB3ZXJlIHNpbXVsYXRlZCB3aXRoIChyb3VnaGx5KSB0aGUgc2FtZSBtZWFucyBhcyB0aGUgYWN0dWFsIGRhdGEuICBUaGlzIG1lYW5zIHRoYXQgKippbiBhbGwgdGhyZWUgc2NlbmFyaW9zLCB0aGUgdmFsdWUgb2YgU0FEIHdvdWxkIGJlIHRoZSBzYW1lKiouPGJyIC8+PGJyIC8+V2hpY2ggc2NlbmFyaW8gKEEsIEIsIEMpIHByb3ZpZGVzIHRoZSBzdHJvbmdlc3QgZXZpZGVuY2UgYWdhaW5zdCB0aGUgbnVsbCBoeXBvdGhlc2lzPyAgSW4gb3RoZXIgd29yZHMsIGluIHdoaWNoIHNjZW5hcmlvIGRvZXMgdGhlICpjb25kaXRpb24qIGdyb3VwIG1hdHRlciB0aGUgbW9zdD88YnIgLz48YnIgLz5JZGVudGlmeSBhIGxpbWl0YXRpb24gb2YgU0FEIGFzIGEgdGVzdCBzdGF0aXN0aWMgdG8gY29tcGFyZSBncm91cCBtZWFucy4KCmBgYHtyICdzaW11bGF0ZWQtdGhyZWUtZ3JvdXBzJ30KIyBBID0gU2ltdWxhdGUgZGF0YSB3aXRoIHNhbWUgbWVhbnMsIHNtYWxsZXIgc3RhbmRhcmQgZGV2aWF0aW9ucwpkYXRhQSA8LSB0aWJibGUoCiAgQ29uZGl0aW9uID0gYXMuZmFjdG9yKGMocmVwKCJBZnRlciIsIDE5KSwgcmVwKCJCZWZvcmUiLCAxOSksIHJlcCgiTm9uZSIsMTkpKSksCiAgQ29tcHJlaGVuc2lvbiA9IGMocm5vcm0oMTksMy4yMTEsMC41KSwgcm5vcm0oMTksNC45NDcsMC41KSwgcm5vcm0oMTksMy4zNjgsMC41KSksCiAgZGF0YXNldCA9IGMocmVwKCJBIiwgNTcpKQopCgojIEIgPSBhY3R1YWwgZGF0YSBmcm9tIHN0dWR5CmRhdGFCIDwtIGFtYmlndW91cwpkYXRhQiRkYXRhc2V0ID0gYyhyZXAoIkIiLCA1NykpCgogIAojIEMgPSBzaW11bGF0ZSBkYXRhIHdpdGggc2FtZSBtZWFucywgbGFyZ2VyIHN0YW5kYXJkIGRldmlhdGlvbnMKZGF0YUMgPC0gdGliYmxlKAogIENvbmRpdGlvbiA9IGFzLmZhY3RvcihjKHJlcCgiQWZ0ZXIiLCAxOSksIHJlcCgiQmVmb3JlIiwgMTkpLCByZXAoIk5vbmUiLDE5KSkpLAogIENvbXByZWhlbnNpb24gPSBjKHJub3JtKDE5LDMuMjExLDMpLCBybm9ybSgxOSw0Ljk0NywzKSwgcm5vcm0oMTksMy4zNjgsMykpLAogIGRhdGFzZXQgPSBjKHJlcCgiQyIsIDU3KSkKKQoKIyBtZXJnZSBpbnRvIGEgc2luZ2xlIGRhdGEgZnJhbWUKZGF0YV9zaW0gPC0gYmluZF9yb3dzKGRhdGFBLCBkYXRhQiwgZGF0YUMpCgojIFBsb3Qgd2l0aCBkYXRhc2V0IGFzIGZhY2V0CmRhdGFfc2ltICU+JQogIGdncGxvdChhZXMoeCA9IENvbmRpdGlvbiwgeSA9IENvbXByZWhlbnNpb24pKSArCiAgZ2VvbV9kb3RwbG90KGFlcyhmaWxsPUNvbmRpdGlvbiksIGJpbndpZHRoPS41LCBiaW5heGlzID0gInkiLCBiaW5wb3NpdGlvbnM9ImFsbCIsIGNvbG9yID0gIndoaXRlIiwgYWxwaGE9MC45KSArCiAgc2NhbGVfeV9jb250aW51b3VzKGxpbWl0cyA9IGMoLTUsIDkpLCBicmVha3M9YygzLCA1KSwgbWlub3JfYnJlYWtzPU5VTEwpICsKICBzdGF0X3N1bW1hcnkoZnVuLnkgPSBtZWFuLCBnZW9tID0gInBvaW50Iiwgc2l6ZT0zLCBhZXMoZ3JvdXA9MSkpICsKICB0aGVtZShheGlzLnRpdGxlLnggPSBlbGVtZW50X3RleHQoc2l6ZSA9IDEyLCBjb2xvciA9ICIjNzc3Nzc3IikpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIpKSArCiAgdGhlbWUoYXhpcy50aXRsZS55ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMiwgY29sb3I9IiM3Nzc3NzciKSkgKwogIHRoZW1lKGF4aXMudGV4dC55ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMikpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb249Im5vbmUiKSArCiAgY29vcmRfZmxpcCh4bGltID0gTlVMTCwgeWxpbSA9IE5VTEwsIGV4cGFuZCA9IEZBTFNFKSArCiAgZmFjZXRfZ3JpZCguIH4gZGF0YXNldCkKCgoKYGBgCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjEwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKVG8gYWRkcmVzcyB0aGUgbGltaXRhdGlvbnMgb2YgU0FELCBvdXIgdGVzdCBzdGF0aXN0aWMgc2hvdWxkIGNvbnNpZGVyIHRoZSAqKnZhcmlhYmlsaXR5ICp3aXRoaW4qIHRoZSBncm91cHMqKiBhbG9uZyB3aXRoIHRoZSAqKnZhcmlhYmlsaXR5ICphbW9uZyogdGhlIGdyb3VwIG1lYW5zKiouCgoxNi4gQXNzdW1pbmcgd2UgY2FuIGRvIHRoaXMsIGhvdyB3aWxsIHdlICoqY29tcGFyZSoqIHRoZSB2YXJpYW5jZSAqYW1vbmcqIGdyb3VwcyB0byB0aGUgdmFyaWFuY2UgKndpdGhpbiogZ3JvdXBzPyAgV2hhdCBkaXN0cmlidXRpb24gd2lsbCB0aGlzIGZvbGxvdz8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6NTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCioqKioqCgojIyBJZGVhICMzOiBBTk9WQSAodXNpbmcgdGhlIG1lYW4gc3F1YXJlIHJhdGlvKQoKPGJyIC8+CgpPdXIgdGVzdCBzdGF0aXN0aWMsIHRoZW4sIHdpbGwgYmU6ICAkRj1cZnJhY3tcdGV4dHJte3ZhcmlhbmNlIGJldHdlZW4gZ3JvdXAgbWVhbnN9fXtcdGV4dHJte3ZhcmlhbmNlIHdpdGhpbiBncm91cHN9fSQKCjxiciAvPgoKV2UnbGwgY2FsbCB0aGlzIHByb2Nlc3MgKiphbmFseXNpcyBvZiB2YXJpYW5jZSoqIG9yICoqQU5PVkEqKi4gIERlc3BpdGUgaXRzIG5hbWUsIHJlYWxpemUgdGhlIGdvYWwgaXMgdG8gY29tcGFyZSAqKm1lYW5zKiogKGJ5IGFuYWx5emluZyB0aGUgdmFyaWFuY2UgYmV0d2VlbiBhbmQgd2l0aGluIHRoZSBncm91cHMpLgoKPGJyIC8+CgoxNy4gV2hhdCBjb25kaXRpb25zIGFyZSBuZWNlc3NhcnkgZm9yIGEgcmF0aW8gb2YgdmFyaWFuY2VzIHRvIGZvbGxvdyBhbiBGLWRpc3RyaWJ1dGlvbj8gIFRoaXMgaXMgYSBkcmF3YmFjayB0byBBTk9WQSAtLSB3ZSdsbCBuZWVkIHRvIG1ha2Ugc3VyZSBzb21lIGNvbmRpdGlvbnMgYXJlIG1ldC4KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoxOC4gQmVsb3csIEkndmUgc2tldGNoZWQgaHlwb3RoZXRpY2FsIHBvcHVsYXRpb24gZGlzdHJpYnV0aW9ucyBmb3IgZWFjaCBncm91cCBpbiBvdXIgc3R1ZHkuICBMaXN0IHRoZSBhc3N1bXB0aW9ucyBJIG1hZGUgaW4gY3JlYXRpbmcgdGhlIHBsb3Qgb24gdGhlIGxlZnQgYW5kIHRoZSBwbG90IG9uIHRoZSByaWdodC4KCgpgYGB7ciAncG9zc2libGUtbnVsbC1vdXRjb21lcycsIGV2YWw9RkFMU0V9CgojIE51bGwgaHlwb3RoZXNpcyBpcyB0cnVlCmdncGxvdChkYXRhLmZyYW1lKHggPSBjKDAsIDguNSkpLCBhZXMoeCkpICsKICBzdGF0X2Z1bmN0aW9uKGZ1biA9IGRub3JtLCBhcmdzID0gbGlzdChtZWFuID0gMy42LCBzZCA9IDEuMjUpLCBjb2xvcj0icmVkIiwgc2l6ZT0xKSArCiAgc3RhdF9mdW5jdGlvbihmdW4gPSBkbm9ybSwgYXJncyA9IGxpc3QobWVhbiA9IDMuNywgc2QgPSAxLjI1KSwgY29sb3I9InNreWJsdWUiLCBzaXplPTEpICsKICBzdGF0X2Z1bmN0aW9uKGZ1biA9IGRub3JtLCBhcmdzID0gbGlzdChtZWFuID0gMy44LCBzZCA9IDEuMjUpLCBjb2xvcj0iMjY3Iiwgc2l6ZT0xKSArCiAgZ2VvbV9kb3RwbG90KGRhdGEgPSBhbWJpZ3VvdXMsIGFlcyh4ID0gQ29tcHJlaGVuc2lvbiwgeSA9IDEsIGZpbGw9Q29uZGl0aW9uLCBjb2xvcj1Db25kaXRpb24pLCAKICAgICAgICAgICAgICAgYmlud2lkdGg9LjEsIGJpbmF4aXM9IngiLCBzdGFja2dyb3Vwcz1UUlVFLCBiaW5wb3NpdGlvbnM9ImFsbCIpICsKICBzY2FsZV95X2NvbnRpbnVvdXMobGltaXRzID0gYygwLCAuNCksIGJyZWFrcz1OVUxMLCBtaW5vcl9icmVha3M9TlVMTCkgKwogIHNjYWxlX3hfY29udGludW91cyhsaW1pdHMgPSBjKC0uNSwgOSksIGJyZWFrcz1zZXEoMSw3LDEpLCBtaW5vcl9icmVha3M9TlVMTCkgKwogIHRoZW1lKGF4aXMudGl0bGUueCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIsIGNvbG9yID0gIiM3Nzc3NzciKSkgKwogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMikpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb249Im5vbmUiKSAgKwogICAgbGFicygKICAgICAgdGl0bGUgPSAiTnVsbCBIeXBvdGhlc2lzIiwKICAgICAgeCA9ICJjb21wcmVoZW5zaW9uIHNjb3JlIiwKICAgICAgeT0iIikKCiMgQWx0ZXJuYXRpdmUgaHlwb3RoZXNpcyBpcyB0cnVlCmdncGxvdChkYXRhLmZyYW1lKHggPSBjKDAsIDguNSkpLCBhZXMoeCkpICsKICBzdGF0X2Z1bmN0aW9uKGZ1biA9IGRub3JtLCBhcmdzID0gbGlzdChtZWFuID0gMiwgc2QgPSAuNiksIGNvbG9yPSJyZWQiLCBzaXplPTEpICsKICBzdGF0X2Z1bmN0aW9uKGZ1biA9IGRub3JtLCBhcmdzID0gbGlzdChtZWFuID0gMy43LCBzZCA9IC42KSwgY29sb3I9InNreWJsdWUiLCBzaXplPTEpICsKICBzdGF0X2Z1bmN0aW9uKGZ1biA9IGRub3JtLCBhcmdzID0gbGlzdChtZWFuID0gNS40LCBzZCA9IC42KSwgY29sb3I9IjI2NyIsIHNpemU9MSkgKwogIGdlb21fZG90cGxvdChkYXRhID0gYW1iaWd1b3VzLCBhZXMoeCA9IENvbXByZWhlbnNpb24sIHkgPSAxLCBmaWxsPUNvbmRpdGlvbiwgY29sb3I9Q29uZGl0aW9uKSwgCiAgICAgICAgICAgICAgIGJpbndpZHRoPS4xLCBiaW5heGlzPSJ4Iiwgc3RhY2tncm91cHM9VFJVRSwgYmlucG9zaXRpb25zPSJhbGwiKSArCiAgc2NhbGVfeV9jb250aW51b3VzKGxpbWl0cyA9IGMoMCwgLjc1KSwgYnJlYWtzPU5VTEwsIG1pbm9yX2JyZWFrcz1OVUxMKSArCiAgc2NhbGVfeF9jb250aW51b3VzKGxpbWl0cyA9IGMoLS41LCA5KSwgYnJlYWtzPXNlcSgxLDcsMSksIG1pbm9yX2JyZWFrcz1OVUxMKSArCiAgdGhlbWUoYXhpcy50aXRsZS54ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMiwgY29sb3IgPSAiIzc3Nzc3NyIpKSArCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoc2l6ZSA9IDEyKSkgKwogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbj0ibm9uZSIpICArCiAgICBsYWJzKAogICAgICB0aXRsZSA9ICJBbHRlcm5hdGl2ZSBIeXBvdGhlc2lzIiwKICAgICAgeCA9ICJjb21wcmVoZW5zaW9uIHNjb3JlIiwKICAgICAgeT0iIikgKwogICAgYW5ub3RhdGUoInRleHQiLCB4ID0gMi4wLCB5ID0gLjcsIGxhYmVsID0gImFmdGVyIiwgY29sb3IgPSAicmVkIikgKwogICAgYW5ub3RhdGUoInRleHQiLCB4ID0gMy43LCB5ID0gLjcsIGxhYmVsID0gIm5vbmUiLCBjb2xvciA9ICJzdGVlbGJsdWUiKSArCiAgICBhbm5vdGF0ZSgidGV4dCIsIHggPSA1LjQsIHkgPSAuNywgbGFiZWwgPSAiYmVmb3JlIiwgY29sb3IgPSAiMjY3IikKCmBgYAoKIVtdKHRocmVlcGxvdHMyLnBuZykKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MjAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgojIyMgTW9kZWwKCkxldCdzIGNvbnN0cnVjdCBhIG1vZGVsIGZvciB0aGlzIGFtYmlndW91cyBwcm9zZSBzdHVkeS4gIFRoaW5rIGFib3V0IDMgc3ViamVjdHMgaW4gdGhpcyBzdHVkeToKCi0gT25lIHJhbmRvbWx5IGFzc2lnbmVkIHRvIHRoZSAqKmFmdGVyKiogZ3JvdXAgd2hvIHJhdGVkIHRoZSBwYXNzYWdlIGEgKio2KioKLSBPbmUgYXNzaWduZWQgdG8gdGhlICoqYmVmb3JlKiogZ3JvdXAgd2hvIHJhdGVkIHRoZSBwYXNzYWdlIGEgKiozKioKLSBPbmUgYXNzaWduZWQgdG8gdGhlICoqbm9uZSoqIGdyb3VwIHdobyByYXRlZCB0aGUgcGFzc2FnZSBhICoqMSoqCgoxOS4gUGljayBvbmUgb2YgdGhvc2UgaW5kaXZpZHVhbHMgYW5kIGxpc3QgYXQgbGVhc3QgdGhyZWUgcmVhc29ucyB3aHkgdGhhdCBwZXJzb24gYXNzaWduZWQgdGhhdCBjb21wcmVoZW5zaW9uIHJhdGluZyB0byB0aGUgcGFzc2FnZS4KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoyMC4gT3VyIG1vZGVsIGlzOjxiciAvPjxiciAvPiR5X3tpZ309XG11ICtcYWxwaGEgX3tnfStcZXBzaWxvbl97aWd9PVxtdSsoXG11IF97Z30tXG11ICkrKHlfe2lnfS1cbXUgX3tnfSkkLjxiciAvPjxiciAvPklkZW50aWZ5IHRoZSBjb21wb25lbnRzIG9mIHRoaXMgbW9kZWwgYW5kIGV4cGxhaW4gd2h5IHRoZSBtb2RlbCBtdXN0IGJlIHRydWUuICBXaGF0IGFzc3VtcHRpb25zIHdpbGwgd2UgbWFrZSByZWdhcmRpbmcgdGhlIGVycm9yIHRlcm0/Cgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjIwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKV2l0aCBBTk9WQSwgd2UgKnBhcnRpdGlvbiogdGhlIHRvdGFsIHZhcmlhdGlvbiBpbiBvdXIgZGF0YSAodGhlIGRpZmZlcmVuY2VzIGFtb25nIGNvbXByZWhlbnNpb24gcmF0aW5ncyBmcm9tIGFsbCBzdWJqZWN0cyBpbiB0aGUgc3R1ZHkpIGludG8gdHdvIGNvbXBvbmVudHM6CgotICoqVHJlYXRtZW50Kiogb3IgKipiZXR3ZWVuLWdyb3VwcyoqID0gdmFyaWF0aW9uIGluIGNvbXByZWhlbnNpb24gZHVlIHRvIHRoZSBwaG90bwotICoqRXJyb3IqKiBvciAqKndpdGhpbi1ncm91cHMqKiA9IHZhcmlhdGlvbiBpbiBjb21wcmVoZW5zaW9uIGFtb25nIHN1YmplY3RzIGluIHRoZSBzYW1lIGdyb3VwCgpXZSBjYW4gKHJvdWdobHkpIHZpc3VhbGl6ZSB0aGVzZSBzb3VyY2VzIG9mIHZhcmlhdGlvbiBhczoKCmBgYHtyICdwb3NzaWJsZS1udWxsLW91dGNvbWVzLXdpdGgtbGFiZWxzJywgZXZhbD1GQUxTRX0KCiMgTnVsbCBoeXBvdGhlc2lzIGlzIHRydWUKZ2dwbG90KGRhdGEuZnJhbWUoeCA9IGMoMCwgOC41KSksIGFlcyh4KSkgKwogIHN0YXRfZnVuY3Rpb24oZnVuID0gZG5vcm0sIGFyZ3MgPSBsaXN0KG1lYW4gPSAzLjYsIHNkID0gMS4yNSksIGNvbG9yPSJyZWQiLCBzaXplPTEsIGFscGhhPTAuNSkgKwogIHN0YXRfZnVuY3Rpb24oZnVuID0gZG5vcm0sIGFyZ3MgPSBsaXN0KG1lYW4gPSAzLjcsIHNkID0gMS4yNSksIGNvbG9yPSJza3libHVlIiwgc2l6ZT0xLCBhbHBoYT0wLjUpICsKICBzdGF0X2Z1bmN0aW9uKGZ1biA9IGRub3JtLCBhcmdzID0gbGlzdChtZWFuID0gMy44LCBzZCA9IDEuMjUpLCBjb2xvcj0iMjY3Iiwgc2l6ZT0xLCBhbHBoYT0wLjUpICsKICBzY2FsZV95X2NvbnRpbnVvdXMobGltaXRzID0gYygwLCAuNCksIGJyZWFrcz1OVUxMLCBtaW5vcl9icmVha3M9TlVMTCkgKwogIHNjYWxlX3hfY29udGludW91cyhsaW1pdHMgPSBjKC0uNSwgOSksIGJyZWFrcz1zZXEoMSw3LDEpLCBtaW5vcl9icmVha3M9TlVMTCkgKwogIHRoZW1lKGF4aXMudGl0bGUueCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIsIGNvbG9yID0gIiM3Nzc3NzciKSkgKwogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMikpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb249Im5vbmUiKSAgKwogICAgbGFicygKICAgICAgdGl0bGUgPSAiTnVsbCBIeXBvdGhlc2lzIiwKICAgICAgeCA9ICJjb21wcmVoZW5zaW9uIHNjb3JlIiwKICAgICAgeT0iIikgKwogICAgYW5ub3RhdGUoInNlZ21lbnQiLCB4ID0gMy42LCB4ZW5kID0gMy44LCB5ID0gLjMzMywgeWVuZCA9IC4zMzMsIGNvbG9yID0gImJsYWNrIiwgc2l6ZT0xLjUpICsKICAgIGFubm90YXRlKCJ0ZXh0IiwgeCA9IDMuNywgeSA9IC4zNSwgbGFiZWwgPSAiYmV0d2VlbiBncm91cHMgdmFyaWF0aW9uIiwgY29sb3IgPSAiYmxhY2siKSArCiAgICBhbm5vdGF0ZSgic2VnbWVudCIsIHggPSAyLjEsIHhlbmQgPSA1LjMsIHkgPSAuMTIsIHllbmQgPSAuMTIsIGNvbG9yID0gImJsYWNrIiwgc2l6ZT0xLjUpICsKICAgIGFubm90YXRlKCJ0ZXh0IiwgeCA9IDMuNywgeSA9IC4xNCwgbGFiZWwgPSAid2l0aGluIGdyb3VwcyB2YXJpYXRpb24iLCBjb2xvciA9ICJibGFjayIpICsKICAgIGFubm90YXRlKCJzZWdtZW50IiwgeCA9IDAsIHhlbmQgPSA4LCB5ID0gLjAsIHllbmQgPSAuMCwgY29sb3IgPSAiYmxhY2siLCBzaXplPTEuNSkgKwogICAgYW5ub3RhdGUoInRleHQiLCB4ID0gMy43LCB5ID0gLjAyLCBsYWJlbCA9ICJ0b3RhbCB2YXJpYXRpb24iLCBjb2xvciA9ICJibGFjayIpIAoKCiMgQWx0ZXJuYXRpdmUgaHlwb3RoZXNpcyBpcyB0cnVlCmdncGxvdChkYXRhLmZyYW1lKHggPSBjKDAsIDguNSkpLCBhZXMoeCkpICsKICBzdGF0X2Z1bmN0aW9uKGZ1biA9IGRub3JtLCBhcmdzID0gbGlzdChtZWFuID0gMiwgc2QgPSAuNiksIGNvbG9yPSJyZWQiLCBzaXplPTEsIGFscGhhPTAuNSkgKwogIHN0YXRfZnVuY3Rpb24oZnVuID0gZG5vcm0sIGFyZ3MgPSBsaXN0KG1lYW4gPSAzLjcsIHNkID0gLjYpLCBjb2xvcj0ic2t5Ymx1ZSIsIHNpemU9MSwgYWxwaGE9MC41KSArCiAgc3RhdF9mdW5jdGlvbihmdW4gPSBkbm9ybSwgYXJncyA9IGxpc3QobWVhbiA9IDUuNCwgc2QgPSAuNiksIGNvbG9yPSIyNjciLCBzaXplPTEsIGFscGhhPTAuNSkgKwogIHNjYWxlX3lfY29udGludW91cyhsaW1pdHMgPSBjKDAsIC43NSksIGJyZWFrcz1OVUxMLCBtaW5vcl9icmVha3M9TlVMTCkgKwogIHNjYWxlX3hfY29udGludW91cyhsaW1pdHMgPSBjKC0uNSwgOSksIGJyZWFrcz1zZXEoMSw3LDEpLCBtaW5vcl9icmVha3M9TlVMTCkgKwogIHRoZW1lKGF4aXMudGl0bGUueCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIsIGNvbG9yID0gIiM3Nzc3NzciKSkgKwogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMikpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb249Im5vbmUiKSAgKwogICAgbGFicygKICAgICAgdGl0bGUgPSAiQWx0ZXJuYXRpdmUgSHlwb3RoZXNpcyIsCiAgICAgIHggPSAiY29tcHJlaGVuc2lvbiBzY29yZSIsCiAgICAgIHk9IiIpICsKICAgIGFubm90YXRlKCJzZWdtZW50IiwgeCA9IDIsIHhlbmQgPSA1LjQsIHkgPSAuNjksIHllbmQgPSAuNjksIGNvbG9yID0gImJsYWNrIiwgc2l6ZT0xLjUpICsKICAgIGFubm90YXRlKCJ0ZXh0IiwgeCA9IDMuNywgeSA9IC43MywgbGFiZWwgPSAiYmV0d2VlbiBncm91cHMgdmFyaWF0aW9uIiwgY29sb3IgPSAiYmxhY2siKSArCiAgICBhbm5vdGF0ZSgic2VnbWVudCIsIHggPSAzLCB4ZW5kID0gNC40LCB5ID0gLjMsIHllbmQgPSAuMywgY29sb3IgPSAiYmxhY2siLCBzaXplPTEuNSkgKwogICAgYW5ub3RhdGUoInRleHQiLCB4ID0gMy43LCB5ID0gLjMzLCBsYWJlbCA9ICJ3aXRoaW4iLCBjb2xvciA9ICJibGFjayIpICsKICAgIGFubm90YXRlKCJzZWdtZW50IiwgeCA9IDAsIHhlbmQgPSA4LCB5ID0gLjAsIHllbmQgPSAuMCwgY29sb3IgPSAiYmxhY2siLCBzaXplPTEuNSkgKwogICAgYW5ub3RhdGUoInRleHQiLCB4ID0gMy43LCB5ID0gLjAyLCBsYWJlbCA9ICJ0b3RhbCB2YXJpYXRpb24iLCBjb2xvciA9ICJibGFjayIpIAoKYGBgCgohW10odGhyZWVwbG90czMucG5nKQoKPGJyIC8+CgoyMS4gV2UncmUgZ29pbmcgdG8gY29tcGFyZSB0aGUgKipiZXR3ZWVuKiogZ3JvdXBzIHZhcmlhdGlvbiB0byB0aGUgKip3aXRoaW4qKiBncm91cHMgdmFyaWF0aW9uLiAgSWYgdGhhdCByYXRpbyBpcyBsYXJnZSwgd2hhdCBkb2VzIHRoYXQgc2F5IGFib3V0IG91ciBudWxsIGh5cG90aGVzaXM/ICBXaGF0IGRvZXMgaXQgc2F5IGlmIHRoZSByYXRpbyBpcyByZWxhdGl2ZWx5IHNtYWxsPyAgSG93IGFyZSB5b3UgZGVmaW5pbmcgKmxhcmdlKiBhbmQgKnNtYWxsKj8gCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjE1MHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKCiMjIyBQYXJ0aXRpb25pbmcgKGNhbGN1bGF0aW5nKSB0aGUgc291cmNlcyBvZiB2YXJpYXRpb24KClJlY2FsbCB0aGUgZm9ybXVsYSBmb3IgdGhlIHVuYmlhc2VkIGVzdGltYXRlIG9mIHRoZSBwb3B1bGF0aW9uIHZhcmlhbmNlOgoKJHNeezJ9PVxmcmFje1xzdW1fe2k9MX1ee259KHhfe2l9LVxvdmVybGluZXtYfSleezJ9fXtuLTF9PVxmcmFje1x0ZXh0cm17c3VtIG9mIHNxdWFyZWQgZGV2aWF0aW9ucyBmcm9tIHRoZSBtZWFufX17XHRleHRybXtkZWdyZWVzIG9mIGZyZWVkb219fT1cZnJhY3tTU317ZGZ9PVx0ZXh0cm17bWVhbiBzcXVhcmV9JAoKPGJyIC8+CgoyMi4gV2hhdCBkbyAqKlNTKiogYW5kICoqTVMqKiByZXByZXNlbnQ/IFdoYXQgYXJlICoqZGVncmVlcyBvZiBmcmVlZG9tKio/Cgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjEwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKIyMjIyBUb3RhbCB2YXJpYXRpb24gPSBNU3RvdGFsCgoyMy4gRmlsbC1pbi10aGUtYmxhbmtzIHRvIGRlcml2ZSB0aGUgZm9ybXVsYSB0byBjYWxjdWxhdGUgdGhlIHRvdGFsIHZhcmlhdGlvbiBpbiBjb21wcmVoZW5zaW9uIHNjb3Jlcy4gIFRoZW4sIGNvbnNpZGVyaW5nIHdoYXQgKk1TdG90YWwqIHJlcHJlc2VudHMsIHJld3JpdGUgdGhlIG51bWVyYXRvciAoU1N0b3RhbCkuICBGaW5hbGx5LCBjYWxjdWxhdGUgTVN0b3RhbCBmb3IgdGhpcyBhbWJpZ3VvdXMgcHJvc2UgZGF0YXNldCBhbmQgZXhwbGFpbiB3aGF0IGl0IHJlcHJlc2VudHMuCgo8YnIgLz4KCiRcbWF0aGJme1x0ZXh0cm17TVN9X3tcdGV4dHJte1RvdGFsfX09XGZyYWN7XHRleHRybXtTU31fe1x0ZXh0cm17dG90YWx9fX17XHRleHRybXtkZn1fe1x0ZXh0cm17dG90YWx9fX09XGZyYWN7XHN1bV97aT0xfV57bn0oXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCAtXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCApXnsyfX17KFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCAtXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcICl9fT0kCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoqKkNhbGN1bGF0aW9uKio6CgpgYGB7ciAnY2FsY3VsYXRlLU1TdG90YWwnfQpTU3RvdGFsIDwtIHN1bSggKCBhbWJpZ3VvdXMkQ29tcHJlaGVuc2lvbiAtIG1lYW4oYW1iaWd1b3VzJENvbXByZWhlbnNpb24pICleMiApClNTdG90YWwKCmRmdG90YWwgPC0gbGVuZ3RoKGFtYmlndW91cyRDb21wcmVoZW5zaW9uKSAtIDEKZGZ0b3RhbAoKTVN0b3RhbCA8LSBTU3RvdGFsIC8gZGZ0b3RhbApNU3RvdGFsCgojIE5vdGljZSBNU3RvdGFsIGlzIGp1c3QgdGhlIHZhcmlhbmNlIG9mIGFsbCB0aGUgZGF0YQp2YXIoYW1iaWd1b3VzJENvbXByZWhlbnNpb24pCmBgYAoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoyMDBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCgojIyMjIFdpdGhpbi1ncm91cCB2YXJpYXRpb24gPSBNU2Vycm9yCgoyNC4gU1N0b3RhbCA9IGByIHJvdW5kKFNTdG90YWwsMylgLiAgTGV0J3MgcGFydGl0aW9uIHRoYXQgaW50byB0aGUgd2l0aGluLSBhbmQgYmV0d2Vlbi1ncm91cHMgc291cmNlLjxiciAvPjxiciAvPioqTVNlcnJvcioqIHJlcHJlc2VudHMgdGhlIHZhcmlhdGlvbiAqKndpdGhpbioqIGVhY2ggZ3JvdXAgKHZhcmlhdGlvbiBpbiBjb21wcmVoZW5zaW9uIHRoYXQgaXMgKipub3QqKiBkdWUgdG8gc2VlaW5nIHRoZSBwaG90bykuPGJyIC8+PGJyIC8+RmlsbC1pbi10aGUtYmxhbmtzIHRvIGRlcml2ZSBmb3JtdWxhcyBmb3IgJFNTX3tlcnJvcn0kLCAkZGZfe2Vycm9yfSQsIGFuZCAkTVNfe2Vycm9yfSQuICBFeHBsYWluIHdoYXQgTVNlcnJvciByZXByZXNlbnRzIGFuZCB3cml0ZSBhIHNpbXBsZXIgZm9ybXVsYSB0byBjYWxjdWxhdGUgU1NlcnJvci4gIEZpbmFsbHksIGNhbGN1bGF0ZSBNU2Vycm9yIGZvciBvdXIgZGF0YS4KCjxiciAvPgoKJFxtYXRoYmZ7XHRleHRybXtNU31fe1x0ZXh0cm17RXJyb3J9fT1cZnJhY3tcdGV4dHJte1NTfV97XHRleHRybXtlcnJvcn19fXtcdGV4dHJte2RmfV97XHRleHRybXtlcnJvcn19fT1cZnJhY3tcc3VtKFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIC1cIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCApXnsyfX17KFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgLVwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgKX19PSQKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6NTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCioqQ2FsY3VsYXRpb24qKjoKCmBgYHtyICdjYWxjdWxhdGUtTVNlcnJvcid9ClNTZXJyb3IgPC0gc3VtKCAoMTkgLSAxKSAqIHZhcihDb21wcmVoZW5zaW9uIH4gQ29uZGl0aW9uLCBkYXRhPWFtYmlndW91cykgKQpTU2Vycm9yCgpkZmVycm9yIDwtIGxlbmd0aChhbWJpZ3VvdXMkQ29tcHJlaGVuc2lvbikgLSAzCmRmZXJyb3IKCk1TZXJyb3IgPC0gU1NlcnJvciAvIGRmZXJyb3IKTVNlcnJvcgoKYGBgCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjIwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKCiMjIyMgQmV0d2Vlbi1ncm91cCB2YXJpYXRpb24gPSBNU3Bob3RvCgoyNS4gSWYgd2Uga25vdyBTU3RvdGFsID0gYHIgcm91bmQoU1N0b3RhbCwzKWAgYW5kIFNTZXJyb3IgPSBgciByb3VuZChTU2Vycm9yLDMpYCwgaG93IGNhbiB3ZSBlYXNpbHkgY2FsY3VsYXRlIFNTcGhvdG8/PGJyIC8+PGJyIC8+RmlsbC1pbi10aGUtYmxhbmtzIHRvIGRlcml2ZSBmb3JtdWxhcyBmb3IgJFNTX3twaG90b30kLCAkZGZfe3Bob3RvfSQsIGFuZCAkTVNfe3Bob3RvfSQuICBFeHBsYWluIHdoYXQgdGhleSByZXByZXNlbnQgYW5kIGNhbGN1bGF0ZSBNU3Bob3RvIGZvciBvdXIgZGF0YS4KCjxiciAvPgoKJFxtYXRoYmZ7XHRleHRybXtNU31fe1x0ZXh0cm17QX19PVxtYXRoYmZ7XHRleHRybXtNU31fe1x0ZXh0cm17dHJlYXRtZW50fX19PVxmcmFje1x0ZXh0cm17U1N9X3tcdGV4dHJte0F9fX17XHRleHRybXtkZn1fe1x0ZXh0cm17QX19fT1cZnJhY3tcc3VtXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCAoXCBcIFwgXCBcIFwgXCBcICBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgLVwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgKV57Mn19eyhcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcICBcIFwgXCBcIFwgXCBcIC1cIFwgXCBcIFwgXCBcIFwgXCBcICBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcIFwgXCBcICl9fSQKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6NTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCioqQ2FsY3VsYXRpb24qKjoKCmBgYHtyICdjYWxjdWxhdGUtTVNwaG90byd9CiMgVGhpcyBpcyBub3QgYXQgYWxsIHRoZSBlYXNpZXN0IHdheSB0byBjYWxjdWxhdGUgdGhpcy4uLgpTU3Bob3RvIDwtIGFzLm51bWVyaWMoCiAgYW1iaWd1b3VzICU+JQogIGdyb3VwX2J5KENvbmRpdGlvbikgJT4lCiAgc3VtbWFyaXplKHNxdWFyZWRtZWFuZGlmZiA9IChtZWFuKENvbXByZWhlbnNpb24pLW1lYW4oYW1iaWd1b3VzJENvbXByZWhlbnNpb24pKV4yKSAlPiUKICBtdXRhdGUoc3NwaG90byA9IDE5ICogc3VtKHNxdWFyZWRtZWFuZGlmZikpICU+JQogIGZpbHRlcihDb25kaXRpb249PSJBZnRlciIpICU+JQogIHNlbGVjdChzc3Bob3RvKQopCgpTU3Bob3RvCgojIENoZWNrIHRvIHNlZSBpZiBpdCdzIHRoZSBzYW1lIGFzIFNTdG90YWwgLSBTU2Vycm9yClNTdG90YWwgLSBTU2Vycm9yCgpkZnBob3RvIDwtIG5fZGlzdGluY3QoYW1iaWd1b3VzJENvbmRpdGlvbikgLSAxCmRmcGhvdG8KCk1TcGhvdG8gPC0gU1NwaG90byAvIGRmcGhvdG8KTVNwaG90bwoKYGBgCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjIwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKKioqKioKCiMjIyBBTk9WQSBzdW1tYXJ5IG9mIGNhbGN1bGF0aW9ucwoKPGJyIC8+CgohW10oYW5vdmFjYWxjLnBuZykKCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoqKkZvciBvdXIgZGF0YSoqOgoKPGJyIC8+CgpTb3VyY2UgICAgfCBTUyAgICAgICAgICAgICAgICAgICB8IGRmICAgICAgICAgIHwgTVMgICAgICAgICAgICAgICAgICAgIHwgTVNSCi0tLS0tLS0tLTp8IC0tLS0tLS0tLS0tLS0tLS0tLS0tOnwtLS0tLS0tLS0tLS06fCAtLS0tLS0tLS0tLS0tLS0tLS0tLS06fCAtLS0tLS0tLToKVHJlYXRtZW50IHwgYHIgcm91bmQoU1NwaG90bywzKWAgfCBgciBkZnBob3RvYCB8ICBgciByb3VuZChNU3Bob3RvLDMpYCB8ID8KRXJyb3IgICAgIHwgYHIgcm91bmQoU1NlcnJvciwzKWAgfCBgciBkZmVycm9yYCB8ICBgciByb3VuZChNU2Vycm9yLDMpYCB8ID8KVG90YWwgICAgIHwgYHIgcm91bmQoU1N0b3RhbCwzKWAgfCBgciBkZnRvdGFsYCB8ICBgciByb3VuZChNU3RvdGFsLDMpYCB8ID8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6NTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCgoyNS4gV2lsbCAkXG1hdGhiZntcdGV4dHJte1NTfV97XHRleHRybXt0b3RhbH19fSQgKiphbHdheXMqKiBlcXVhbCAkXG1hdGhiZntcdGV4dHJte1NTfV97XHRleHRybXtBfX19K1xtYXRoYmZ7XHRleHRybXtTU31fe1x0ZXh0cm17RX19fSQ/ICBFeHBsYWluIHdoYXQgaXMgaGFwcGVuaW5nIGF0IGVhY2ggc3RlcCBvZiB0aGUgZm9sbG93aW5nIGRlcml2YXRpb246CgoKPGJyIC8+CgokXG1hdGhiZntcdGV4dHJte1NTfV97XHRleHRybXt0b3RhbH19fT1cc3VtIFxzdW0gKHhfe2l9LU0pXnsyfT1cc3VtIFxzdW0gXGxlZnQgWyBcbGVmdCAoIHhfe2l9LSBcb3ZlcmxpbmV7WH1fe2F9XHJpZ2h0ICktXGxlZnQgKCBcb3ZlcmxpbmV7WH1fe2F9LU0gXHJpZ2h0ICkgXHJpZ2h0IF1eezJ9PVxzdW0gXHN1bSBcbGVmdCBbIFxsZWZ0ICggXG92ZXJsaW5le1h9X3thfS0gTVxyaWdodCApLVxsZWZ0ICggeF97aX0tXG92ZXJsaW5le1h9X3thfSBccmlnaHQgKSBccmlnaHQgXV57Mn09JAoKPGJyIC8+CgokXG1hdGhiZntcdGV4dHJte1NTfV97XHRleHRybXt0b3RhbH19fT1cc3VtIFxzdW1cbGVmdCAoIFxvdmVybGluZXtYfV97YX0tIE1ccmlnaHQgKV57Mn0rXHN1bSBcc3VtXGxlZnQgKCB4X3tpfS1cb3ZlcmxpbmV7WH1fe2F9IFxyaWdodCApXnsyfSsyXHN1bSBcc3VtXGxlZnQgKCBcb3ZlcmxpbmV7WH1fe2F9LSBNXHJpZ2h0IClcbGVmdCAoIHhfe2l9LVxvdmVybGluZXtYfV97YX0gXHJpZ2h0ICkkCgo8YnIgLz4KClNpbmNlICRcc3VtIFxsZWZ0ICggeF97aX0tXG92ZXJsaW5le1h9X3thfSBccmlnaHQgKT0wJAoKPGJyIC8+CiRcbWF0aGJme1x0ZXh0cm17U1N9X3tcdGV4dHJte3RvdGFsfX19PVxzdW0gbl97YX1cbGVmdCAoIFxvdmVybGluZXtYfV97YX0tIE1ccmlnaHQgKV57Mn0rXHN1bSBcc3VtXGxlZnQgKCB4X3tpfS1cb3ZlcmxpbmV7WH1fe2F9IFxyaWdodCApXnsyfT0kCgo8YnIgLz4KCiRcbWF0aGJme1x0ZXh0cm17U1N9X3tcdGV4dHJte3RvdGFsfX19PVxtYXRoYmZ7XHRleHRybXtTU31fe1x0ZXh0cm17QX19fStcbWF0aGJme1x0ZXh0cm17U1N9X3tcdGV4dHJte0V9fX0kCgoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDo1MHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKIyMjIyBPbmUgbW9yZSBsb29rIGF0IE1TZXJyb3IKCklmICptZWFuIHNxdWFyZSogaXMgYSBmYW5jeSB0ZXJtIGZvciAqdmFyaWFuY2UqLCBNU2Vycm9yIG11c3QgcmVwcmVzZW50IHRoZSB2YXJpYW5jZSB3aXRoaW4gYSBzaW5nbGUgZ3JvdXAuCgo8YnIgLz4KCklmIHdlIGhhdmUgMyBncm91cHMgaW4gb3VyIGRhdGFzZXQsIGhvdyBjYW4gTVNFIHJlcHJlc2VudCB0aGUgdmFyaWFuY2Ugb2YgYSBzaW5nbGUgZ3JvdXA/ICBXaGljaCBncm91cCBkbyB3ZSAiY2hvb3NlIiB3aGVuIHdlIGNhbGN1bGF0ZSBNU0U/Cgo8YnIgLz4KCklmIHdlIGFzc3VtZSBvdXIgZGF0YSBjb21lIGZyb20gZ3JvdXBzIHdpdGggZXF1YWwgcG9wdWxhdGlvbiB2YXJpYW5jZXMsIHdlIGRvbid0IGhhdmUgdG8gY2hvb3NlIGEgZ3JvdXAuICBJbnN0ZWFkLCB3ZSBjYW4gY2FsY3VsYXRlIGEgKipwb29sZWQgdmFyaWFuY2Ugb2YgKmFsbCogb3VyIGdyb3VwcyoqLiAgSW4gdGhpcyBzZW5zZSwgTVNFIHJlcHJlc2VudHMgdGhlICoqYXZlcmFnZSB2YXJpYW5jZSoqIG9mIGVhY2ggb2Ygb3VyIGdyb3Vwcy4KCjxiciAvPgoKVGhlIGNvbmNlcHQgb2YgYW4gYXZlcmFnZSwgb3IgKnBvb2xlZCosIHZhcmlhbmNlIHNob3VsZCBzb3VuZCBmYW1pbGlhci4gIFdlJ3ZlIHVzZWQgU3Bvb2xlZCB0byBjb25kdWN0IGluZGVwZW5kZW50IHNhbXBsZXMgdC10ZXN0cyAoYXNzdW1pbmcgZXF1YWwgdmFyaWFuY2VzKS4gIElmIHdlIHRha2UgdGhlIGZvcm11bGEgZm9yIFNwb29sZWQgYW5kIGV4dGVuZCBpdCB0byAzIGdyb3Vwcywgd2UgaGF2ZToKCjxiciAvPgoKJHNfe1x0ZXh0cm17cG9vbGVkfX1eezJ9PVxmcmFje1xsZWZ0ICggbl97MX0tMSBccmlnaHQgKXNfezF9XnsyfStcbGVmdCAoIG5fezJ9LTEgXHJpZ2h0IClzX3syfV57Mn0rXGxlZnQgKCBuX3szfS0xIFxyaWdodCApc197M31eezJ9fXtcbGVmdCAoIG5fezF9LTEgXHJpZ2h0ICkrXGxlZnQgKCBuX3syfS0xIFxyaWdodCApK1xsZWZ0ICggbl97M30tMSBccmlnaHQgKX0kCgo8YnIgLz4KCiRzX3tcdGV4dHJte3Bvb2xlZH19XnsyfT1cZnJhY3tcbGVmdCAoIG5fezF9LTEgXHJpZ2h0IClcZnJhY3tcc3VtXGxlZnQgKCB4X3tpMX0tXG92ZXJsaW5le1h9X3sxfSBccmlnaHQgKV57Mn19e25fezF9LTF9ICArICBcbGVmdCAoIG5fezJ9LTEgXHJpZ2h0IClcZnJhY3tcc3VtXGxlZnQgKCB4X3tpMn0tXG92ZXJsaW5le1h9X3syfSBccmlnaHQgKV57Mn19e25fezJ9LTF9ICsgXGxlZnQgKCBuX3szfS0xIFxyaWdodCApXGZyYWN7XHN1bVxsZWZ0ICggeF97aTN9LVxvdmVybGluZXtYfV97M30gXHJpZ2h0ICleezJ9fXtuX3szfS0xfX17XGxlZnQgKCBuX3sxfS0xIFxyaWdodCApK1xsZWZ0ICggbl97Mn0tMSBccmlnaHQgKStcbGVmdCAoIG5fezN9LTEgXHJpZ2h0ICl9JAoKPGJyIC8+Cgokc197XHRleHRybXtwb29sZWR9fV57Mn09XGZyYWN7XHN1bSBcbGVmdCAoIHhfe2kxfS1cb3ZlcmxpbmV7WH1fezF9IFxyaWdodCApXnsyfSArIFxzdW0gXGxlZnQgKCB4X3tpMn0tXG92ZXJsaW5le1h9X3syfSBccmlnaHQgKV57Mn0gKyBcc3VtIFxsZWZ0ICggeF97aTN9LVxvdmVybGluZXtYfV97M30gXHJpZ2h0ICleezJ9fXtcbGVmdCAoIG5fezF9LTEgXHJpZ2h0ICkrXGxlZnQgKCBuX3syfS0xIFxyaWdodCApK1xsZWZ0ICggbl97M30tMSBccmlnaHQgKX0kCgo8YnIgLz4KCiRzX3tcdGV4dHJte3Bvb2xlZH19XnsyfT1cZnJhY3tcc3VtIFxsZWZ0ICggeF97aWF9LVxvdmVybGluZXtYfV97YX0gXHJpZ2h0ICleezJ9fXtOLWF9PVx0ZXh0cm17TVN9X3tcdGV4dHJte0V9fSQKCjxiciAvPgoKCjI2LiBVbmRlciBhIHRydWUgbnVsbCBoeXBvdGhlc2lzLCB3aGF0J3MgdGhlIGV4cGVjdGVkIHZhbHVlIG9mICoqTVNlcnJvcioqPyAgRG9lcyB0aGF0IGV4cGVjdGF0aW9uIGNoYW5nZSBpZiB0aGUgbnVsbCBoeXBvdGhlc2lzIHdlcmUgZmFsc2U/Cgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjEwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKMjYuIFVuZGVyIGEgdHJ1ZSBudWxsIGh5cG90aGVzaXMsIHdoYXQncyB0aGUgZXhwZWN0ZWQgdmFsdWUgb2YgKipNU3Bob3RvKio/ICBEb2VzIHRoYXQgZXhwZWN0YXRpb24gY2hhbmdlIGlmIHRoZSBudWxsIGh5cG90aGVzaXMgd2VyZSBmYWxzZT8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgpUaGlzIGlzIHRoZSBsb2dpYyBiZWhpbmQgQU5PVkEuICBVbmRlciBhIHRydWUgbnVsbCBoeXBvdGhlc2lzLCBib3RoIE1TcGhvdG8gYW5kIE1TZXJyb3IgcmVwcmVzZW50IHVuYmlhc2VkIGVzdGltYXRlcyBvZiB0aGUgdmFyaWFuY2Ugd2l0aGluIGEgZ3JvdXAuICBXaGVuIHRoZSBudWxsIGh5cG90aGVzaXMgaXMgZmFsc2UsIE1TcGhvdG8gYmVjb21lcyBsYXJnZXIuCgo8YnIgLz4KCjI4LiBIb3cgd2lsbCB3ZSBjb21wYXJlIE1TcGhvdG8gdG8gTVNlcnJvcj8gIFdoYXQgc2FtcGxpbmcgZGlzdHJpYnV0aW9uIHdpbGwgd2UgdXNlPyAgSG93IG1hbnkgZGVncmVlcyBvZiBmcmVlZG9tIHdpbGwgd2UgaGF2ZT8gIFdoYXQncyB0aGUgZXhwZWN0ZWQgdmFsdWUgb2Ygb3VyIHRlc3Qgc3RhdGlzdGljIGlmIHRoZSBudWxsIGh5cG90aGVzaXMgd2VyZSB0cnVlPwoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoxNTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCiMjIyBBTk9WQSBzdW1tYXJ5IHRhYmxlCgpDb21wbGV0ZSB0aGUgZm9sbG93aW5nIEFOT1ZBIHN1bW1hcnkgdGFibGUgYW5kIGVzdGltYXRlIHRoZSBwLXZhbHVlLiAgV2UgY2FuIGVzdGltYXRlIHRoZSBwLXZhbHVlIGRpcmVjdGx5IGluIFIgb3IgdXNlIHRoZSBbU3RhdGtleSBGLWRpc3RyaWJ1dGlvbiBhcHBsZXRdKGh0dHA6Ly9sb2NrNXN0YXQuY29tL3N0YXRrZXkvdGhlb3JldGljYWxfZGlzdHJpYnV0aW9uL3RoZW9yZXRpY2FsX2Rpc3RyaWJ1dGlvbi5odG1sI0YpLgoKPGJyIC8+CgpTb3VyY2UgICAgfCBTUyAgICAgICAgICAgICAgICAgICB8IGRmICAgICAgICAgIHwgTVMgICAgICAgICAgICAgICAgICAgIHwgTVNSCi0tLS0tLS0tLTp8IC0tLS0tLS0tLS0tLS0tLS0tLS0tOnwtLS0tLS0tLS0tLS06fCAtLS0tLS0tLS0tLS0tLS0tLS0tLS06fDotLS0tLS0tLQpUcmVhdG1lbnQgfCBgciByb3VuZChTU3Bob3RvLDMpYCB8IGByIGRmcGhvdG9gIHwgIGByIHJvdW5kKE1TcGhvdG8sMilgIHwgTVNSID0gCkVycm9yICAgICB8IGByIHJvdW5kKFNTZXJyb3IsMylgIHwgYHIgZGZlcnJvcmAgfCAgYHIgcm91bmQoTVNlcnJvciwyKWAgfCBwID0gClRvdGFsICAgICB8IGByIHJvdW5kKFNTdG90YWwsMylgIHwgYHIgZGZ0b3RhbGAgfCAgYHIgcm91bmQoTVN0b3RhbCwyKWAgfCAkXGV0YV4yPSQKCjxiciAvPgoKQSBxdWljayBsb29rIGF0IGFuIFtGLWRpc3RyaWJ1dGlvbiB0YWJsZV0oaHR0cDovL2JyYWR0aGllc3Nlbi5jb20vaHRtbDUvc3RhdHMvbTMwMS9mdGFibGUucGRmKSB3aWxsIGdpdmUgeW91IGEgc2Vuc2Ugb2Ygd2hhdCB2YWx1ZXMgb2YgdGhlIE1TUiB3aWxsIHlpZWxkIGxvdyBwLXZhbHVlcy4KCjxiciAvPgoKMjkuIFdoYXQgZG9lcyB0aGF0IHAtdmFsdWUgdGVsbCB1cz8gIERvZXMgaXQgaW5mb3JtIHVzIGFib3V0IHRoZSBtYWduaXR1ZGUgb2YgdGhlIGRpZmZlcmVuY2VzIGFtb25nIHRoZSBncm91cCBtZWFucz8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoqKioqKgoKIyMgQU5PVkEgaW4gUgoKQ29uZHVjdGluZyBBTk9WQSBpbiBSIGlzIHNpbXBsZS4gIEV4cGFuZCB0aGUgY29kZSB0byBzZWUgLS0+CgpgYGB7ciAnQU5PVkEtaW4tUid9CiMgT3VyIGRhdGEgZnJhbWUgaXMgImFtYmlndW91cyIKIyBPdXIgbW9kZWwgaXM6ICBDb21wcmVoZW5zaW9uIH4gQ29uZGl0aW9uCiMgV2UgdXNlIGFvdigpIHRvIGNvbmR1Y3QgdGhlIGFuYWx5c2lzIG9mIHZhcmlhbmNlCiMgSSdsbCBzdG9yZSB0aGUgcmVzdWx0cyBpbiBhIGxpc3QgY2FsbGVkICJtb2RlbCIKbW9kZWwgPC0gYW92KCBDb21wcmVoZW5zaW9uIH4gQ29uZGl0aW9uLCBkYXRhID0gYW1iaWd1b3VzICkKCiMgVGhlIGRlZmF1bHQgb3V0cHV0IGlzbid0IHRvbyBoZWxwZnVsCiMgSXQgb25seSBzaG93cyBTUyBhbmQgZGYsIHdpdGggYSByZXNpZHVhbCBzdGFuZGFyZCBlcnJvcgptb2RlbAoKIyBXZSBjYW4gdXNlIHRoZSBhbm92YSgpIGZ1bmN0aW9uIHRvIHByb2R1Y2UgbW9yZSB1c2VmdWwgb3V0cHV0CmFub3ZhKG1vZGVsKQoKIyBWZXJpZnkgdGhhdCB0aGVzZSB2YWx1ZXMgbWF0Y2ggb3VyIGNhbGN1bGF0aW9ucwoKIyBUaGUgYW92KCkgZnVuY3Rpb24gcHJvZHVjZXMgbXVjaCBtb3JlIG91dHB1dCB0aGFuIHdlJ3JlIHNlZWluZwojIFdlIGNhbiBnbGFuY2UgYXQgc29tZSBvZiB0aGlzIG91dHB1dCB3aXRoIHRoZSBicm9vbSBwYWNrYWdlCiMgVGhlIGZvbGxvd2luZyBjb2RlIHdpbGwgaW5zdGFsbCB0aGUgYnJvb20gcGFja2FnZSAoaWYgbmVjZXNzYXJ5KQpsaXN0Lm9mLnBhY2thZ2VzIDwtIGMoImJyb29tIikKbmV3LnBhY2thZ2VzIDwtIGxpc3Qub2YucGFja2FnZXNbIShsaXN0Lm9mLnBhY2thZ2VzICVpbiUgaW5zdGFsbGVkLnBhY2thZ2VzKClbLCJQYWNrYWdlIl0pXQppZihsZW5ndGgobmV3LnBhY2thZ2VzKSkgaW5zdGFsbC5wYWNrYWdlcyhuZXcucGFja2FnZXMpCgpsaWJyYXJ5KGJyb29tKQp0aWR5KG1vZGVsKQpnbGFuY2UobW9kZWwpCgojIEJ5IHRoZSBlbmQgb2YgdGhpcyBjb3Vyc2UsIHlvdSdsbCB1bmRlcnN0YW5kIHdoYXQgYWxsIHRob3NlCiMgdGVybXMgbWVhbjogci5zcXVhcmVkLCBsb2dMaWssIEFJQywgZGV2aWFuY2UKYGBgCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoKKioqKioKCiMjIEVmZmVjdCBzaXplOiAgJFxldGFeMiQgYW5kICRcb21lZ2FeMiQKCldlIGNhbiBkZWZpbmUgYW4gZWZmZWN0IHNpemUgKGV0YS1zcXVhcmVkKSBhczogJFxldGEgXnsyfT1cZnJhY3tcdGV4dHJte1NTfV9cdGV4dHJte0F9fXtcdGV4dHJte1NTfV9cdGV4dHJte1R9fT1cZnJhY3szNS4wNX17MTI5LjU4fT0wLjI3JAoKCjMwLiBJbnRlcnByZXQgZXRhLXNxdWFyZWQgaW4gdGhpcyBzY2VuYXJpby4KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgpFdGEtc3F1YXJlZCBpcyBiYXNlZCBlbnRpcmVseSBvbiBzdW1zIG9mIHNxdWFyZXMgZnJvbSB0aGUgKnNhbXBsZSogYW5kIG5vIGFkanVzdG1lbnQgaXMgbWFkZSB0byBlc3RpbWF0ZSB0aGUgZWZmZWN0IHNpemUgZm9yIHRoZSBwb3B1bGF0aW9uLiAgQXMgd2UnbGwgbGVhcm4gbGF0ZXIgaW4gdGhpcyBjb3Vyc2UsIG91ciBtb2RlbHMgdHlwaWNhbGx5ICoqb3ZlcmZpdCoqIG91ciBzYW1wbGUgZGF0YS4KCkFzIGFuIGFkanVzdGVkIGVmZmVjdCBzaXplLCB3ZSBjYW4gY2FsY3VsYXRlICoqb21lZ2Etc3F1YXJlZCoqOiAgJFxvbWVnYSBeezJ9PVxmcmFje1x0ZXh0cm17U1N9X1x0ZXh0cm17QX0tXGxlZnQgKFx0ZXh0cm17ZGZ9X3tcdGV4dHJte0F9fSAgXHJpZ2h0IClcbGVmdCAoIFx0ZXh0cm17TVN9X1x0ZXh0cm17RX0gXHJpZ2h0ICl9e1x0ZXh0cm17U1N9X1x0ZXh0cm17VH0rXHRleHRybXtNU31fXHRleHRybXtFfX0kCgo8YnIgLz4KCkZvciBvdXIgYW1iaWd1b3VzIHByb3NlIGRhdGE6ICAkXG9tZWdhIF57Mn09XGZyYWN7MzUuMDUtIFxsZWZ0ICgyIFxyaWdodCApXGxlZnQgKCAxLjc1IFxyaWdodCApfXsxMjkuNTgrMS43NX09MC4yNSQKCjxiciAvPgoKMzEuIFdoeSBpcyAkXG9tZWdhXjIgPCBcZXRhXjIkPyAgQmFzZWQgb24gdGhlIHAtdmFsdWUgYW5kIGVmZmVjdCBzaXplIGVzdGltYXRlcywgd2hhdCBjb25jbHVzaW9ucyBhcmUgeW91IHdpbGxpbmcgdG8gbWFrZT8KCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoqKioqKgoKIyMgRXZhbHVhdGluZyBBTk9WQSBjb25kaXRpb25zIChhc3N1bXB0aW9ucykKCjMyLiBMaXN0IHRoZSBjb25kaXRpb25zIG5lY2Vzc2FyeSB0byBnZXQgdmFsaWQgcmVzdWx0cyBmcm9tIGFuIEFOT1ZBLiAgSWRlbnRpZnkgKip3aHkqKiB3ZSBuZWVkIGVhY2ggY29uZGl0aW9uLiAKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MjAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgpUbyBhc3Nlc3MgKipob21vZ2VuZWl0eSBvZiB2YXJpYW5jZXMqKiwgd2UgY2FuIHVzZToKCi0gW0hhcnRsZXkncyBGLW1heCB0ZXN0XShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9CYXJ0bGV0dCUyN3NfdGVzdCkgKHNlbnNpdGl2ZSB0byB2aW9sYXRpb25zIG9mIG5vcm1hbGl0eSkKLSBbQmFydGxldHQncyB0ZXN0XShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9CYXJ0bGV0dCUyN3NfdGVzdCkgKHNlbnNpdGl2ZSB0byB2aW9sYXRpb25zIG9mIG5vcm1hbGl0eSkKLSBbTGV2ZW5lJ3MgdGVzdF0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvTGV2ZW5lJTI3c190ZXN0KSAocm9idXN0IGFnYWluc3QgdmlvbGF0aW9ucyBvZiBub3JtYWxpdHkpCi0gW0Jyb3duLUZvcnN5dGhlIHRlc3RdKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0Jyb3duw6LigqzigJxGb3JzeXRoZV90ZXN0KSAocm9idXN0IGFnYWluc3QgdmlvbGF0aW9ucyBvZiBub3JtYWxpdHkpCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjI1cHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgojIyMgRi1tYXggdGVzdAoKVGhlIEZtYXggdGVzdCwgd2hpY2ggc2hvdWxkIG9ubHkgYmUgYXBwbGllZCBpZiBkYXRhIGZvciBlYWNoIGdyb3VwIGFyZSBub3JtYWxseSBkaXN0cmlidXRlZCwgaXMgc29tZXRoaW5nIHlvdSBjYW4gcXVpY2tseSBlc3RpbWF0ZSBpbiB5b3VyIGhlYWQuICBUbyBkbyBzbywgeW91IGNhbGN1bGF0ZSB0aGUgZm9sbG93aW5nIHN0YXRpc3RpYyBhbmQgY29tcGFyZSBpdCB0byB0aGUgW0ZtYXggZGlzdHJpYnV0aW9uXShodHRwOi8vYnJhZHRoaWVzc2VuLmNvbS9odG1sNS9zdGF0cy9tMzAxLzRjLnBkZikuCgokRl97bWF4fT1cZnJhY3tzXjJfXG1hdGhybXt7YmlnZ2VzdCB9fX17c14yX1xtYXRocm17e3NtYWxsZXN0IH19fT1cZnJhY3sxLjM5OF4yfXsxLjI1N14yfT0xLjIzN1xzaW0gRl97bWF4fSQKCjxiciAvPgoKIyMjIExldmVuZSdzIHRlc3QKCkxldmVuZSdzIHRlc3QsIHdoaWNoIGlzIG1vcmUgYXBwcm9wcmlhdGUgaWYgeW91ciBkYXRhIGRvIG5vdCBmb2xsb3cgYSBub3JtYWwgZGlzdHJpYnV0aW9uLCBpcyBjYWxjdWxhdGVkIGJ5IGZpcnN0IGNvbnZlcnRpbmcgZXZlcnkgc2NvcmUgdG86Cgokel97YWl9PVxsZWZ0IHx4X3thaX0tXG92ZXJsaW5le1h9X2EgIFxyaWdodCB8JAoKPGJyIC8+CgpUaGUgdGVzdCBzdGF0aXN0aWMgaXMgY2FsY3VsYXRlZCBhczoKCiRXPVxmcmFje1xsZWZ0ICggTi1hIFxyaWdodCApfXthLTF9XGZyYWN7XHN1bSBuX3thfVxsZWZ0ICggXG92ZXJsaW5le3p9X3thfS1cb3ZlcmxpbmV7XG92ZXJsaW5le3p9fSBcLCAgXHJpZ2h0ICleMn17XHN1bSBcc3VtIFxsZWZ0ICggel97aWF9LSBcb3ZlcmxpbmV7en1fe2F9XHJpZ2h0ICleMn1cc2lte0Z9XnthLTF9X3tOLWF9JAoKPGJyIC8+CgpFeHBhbmQgdGhlIGNvZGUgdG8gc2VlIGhvdyB0byBjb25kdWN0IExldmVuZSdzIHRlc3QgaW4gUiAtLT4KCmBgYHtyICdsZXZlbmVzLXRlc3QnLCBtZXNzYWdlPUZBTFNFfQojIFdlIG5lZWQgdGhlICdjYXInIHBhY2thZ2UKbGlzdC5vZi5wYWNrYWdlcyA8LSBjKCJjYXIiKQpuZXcucGFja2FnZXMgPC0gbGlzdC5vZi5wYWNrYWdlc1shKGxpc3Qub2YucGFja2FnZXMgJWluJSBpbnN0YWxsZWQucGFja2FnZXMoKVssIlBhY2thZ2UiXSldCmlmKGxlbmd0aChuZXcucGFja2FnZXMpKSBpbnN0YWxsLnBhY2thZ2VzKG5ldy5wYWNrYWdlcykKbGlicmFyeShjYXIpCgojIExldmVuZSBUZXN0CmxldmVuZVRlc3QoQ29tcHJlaGVuc2lvbiB+IENvbmRpdGlvbiwgZGF0YSA9IGFtYmlndW91cywgY2VudGVyID0gbWVhbikKCiMgQnJvd24tRm9yc3l0aGUgVGVzdCAodXNlcyBtZWRpYW5zIGluc3RlYWQgb2YgbWVhbnMpCmxldmVuZVRlc3QoQ29tcHJlaGVuc2lvbiB+IENvbmRpdGlvbiwgZGF0YSA9IGFtYmlndW91cywgY2VudGVyID0gbWVkaWFuKQoKCmBgYAoKPGJyIC8+CgozMy4gQmFzZWQgb24gdGhlIG91dHB1dCBmcm9tIExldmVuZSdzIHRlc3QsIGNhbiB3ZSBjb25jbHVkZSBvdXIgZ3JvdXAgdmFyaWFuY2VzIGFyZSBlcXVhbD8gIEV4cGxhaW4uIAoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoxMDBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCgojIyMgRXZhbHVhdGluZyBub3JtYWxpdHkgYXNzdW1wdGlvbgoKVGhlcmUgYXJlIHNldmVyYWwgd2F5cyB0byBbdGVzdCBmb3Igbm9ybWFsaXR5XShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9Ob3JtYWxpdHlfdGVzdCk6CgotIFtIaXN0b21hbmN5XShodHRwczovL2Jvb2tzLmdvb2dsZS5jb20vYm9va3M/aWQ9bURvMEN3QUFRQkFKJnBnPVBBMjgyJmxwZz1QQTI4MiZkcT1oaXN0b21hbmN5JnNvdXJjZT1ibCZvdHM9ZkRhYzZvRUszbCZzaWc9WmNNY3BwVjh5MUhuUFltQWxEUzBSOXkyVDdFJmhsPWVuJnNhPVgmdmVkPTBhaFVLRXdqLXplRHN4T3JUQWhWQlpDWUtIWEZrQjRJUTZBRUlNVEFDI3Y9b25lcGFnZSZxPWhpc3RvbWFuY3kmZj1mYWxzZSk6ICAqVGhlIGFydCBvZiBkaXZpbmluZyBsaWtlbGlob29kIGZ1bmN0aW9ucyBmcm9tIGVtcGlyaWNhbCBoaXN0b2dyYW1zLiAgVGhpcyBzb3JjZXJ5IGlzIHVzZWQsIGZvciBleGFtcGxlLCB3aGVuIHRlc3RpbmcgZm9yIG5vcm1hbGl0eSBiZWZvcmUgZGVjaWRpbmcgd2hldGhlciBvciBub3QgdG8gdXNlIGEgbm9uLXBhcmFtZXRyaWMgcHJvY2VkdXJlLiAgSGlzdG9tYW5jeSBpcyBhIGZhbHNlIGdvZCwgYmVjYXVzZSBldmVuIHBlcmZlY3RseSBnb29kIEdhdXNzaWFuIHZhcnJpYWJsZXMgbWF5IG5vdCBsb29rIEdhdXNzaWFuIHdoZW4gZGlzcGxheWVkIGFzIGEgaGlzdG9ncmFtLiogKE1jRWxyZWF0aCwgcC4gMjgyKQotIFEtUSBwbG90cyAod2hpY2ggaGF2ZSB0aGUgc2FtZSBpc3N1ZXMgYXMgbG9va2luZyBhdCBoaXN0b2dyYW1zKQotIEJhY2stb2YtdGhlLWVudmVsb3BlIHRlc3Q6ICBDb252ZXJ0IHRoZSBtYXhpbXVtIGFuZCBtaW5pbXVtIHNjb3JlcyB0byB6LXNjb3Jlcy4gIFlvdSBzaG91bGRuJ3QgaGF2ZSBleHRyZW1lIHotc2NvcmVzIGluIHNtYWxsIGRhdGFzZXRzICgkej40JCBzaG91bGQgb25seSBvY2N1ciB3aXRoIGF0IGxlYXN0IDE1LDAwMCBvYnNlcnZhdGlvbnMgaW4gdGhlIGdyb3VwKQotIFtTaGFwaXJvLVdpbGtdKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL1NoYXBpcm/igJNXaWxrX3Rlc3QpIHRlc3QgKHNlZW1zIHRvIGJlIHRoZSBtb3N0IHBvd2VyZnVsKQotIFtLb2xtb2dvcm92LVNtaXJub3ZdKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0tvbG1vZ29yb3bigJNTbWlybm92X3Rlc3QpIHRlc3QKLSBbU3rDqWtlbHkgYW5kIFJpenpvJ3MgZW5lcmd5IHRlc3RdKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0VuZXJneV9kaXN0YW5jZSkKClRoZXNlIG1ldGhvZHMgKGV4Y2VwdCBmb3IgdGhlIGVuZXJneSB0ZXN0KSBhcmUgZWFzeSB0byB1bmRlcnN0YW5kLiBFeHBhbmQgdGhlIGNvZGUgdG8gc2VlIGhvdyB0byBjb25kdWN0IHRoZSBTaGFwaXJvLVdpbGsgdGVzdCBpbiBSLiAgCgpgYGB7ciAnc2hhcGlybyd9CiMgVGhlIGZ1bmN0aW9uIGlzIGNhbGxlZCAnc2hhcGlyby50ZXN0YC4gIEl0IHdvdWxkIHRlc3QgZm9yIG5vcm1hbGl0eQojIG9mIGFsbCB0aGUgZGF0YSBpbiBvdXIgZGF0YSBmcmFtZSAoaWdub3JpbmcgdGhlIGdyb3VwcykKc2hhcGlyby50ZXN0KGFtYmlndW91cyRDb21wcmVoZW5zaW9uKQoKIyBJZiB3ZSB3YW50IHRvIGFwcGx5ICh0YXBwbHkpIHRoZSBzaGFwaXJvLnRlc3QgdG8gYWxsIG91ciBncm91cHMsIHdlIHVzZToKdGFwcGx5KGFtYmlndW91cyRDb21wcmVoZW5zaW9uLCBhbWJpZ3VvdXMkQ29uZGl0aW9uLCBzaGFwaXJvLnRlc3QpCgojIFdlIGNvdWxkIHRoZW4gbG9vayBhdCBhIFEtUSBwbG90IGZvciB0aGF0ICJiZWZvcmUiIGdyb3VwCiMgcXFub3JtKGFtYmlndW91cyRDb21wcmVoZW5zaW9uW2FtYmlndW91cyRDb25kaXRpb249PSJCZWZvcmUiXSwgeWxhYiA9ICJFc3NheSBzY29yZXMiKTsgcXFsaW5lKGFtYmlndW91cyRDb21wcmVoZW5zaW9uW2FtYmlndW91cyRDb25kaXRpb249PSJCZWZvcmUiXSkKCmBgYAoKPGJyIC8+CgojIyMgRGVhbGluZyB3aXRoIHZpb2xhdGlvbnMgb2YgY29uZGl0aW9ucwoKKipXaGF0IGNhbiB3ZSBkbyBpZiB0aGUgY29uZGl0aW9ucyBhcmVuJ3Qgc2F0aXNmaWVkIChvciBpZiB0aGUgYXNzdW1wdGlvbnMgYXJlbid0IHJlYXNvbmFibGUpPyoqICBXZSdsbCBkZWFsIHdpdGggdGhpcyBpc3N1ZSBhbGwgc2VtZXN0ZXIuICBTb21lIG9wdGlvbnMgaW5jbHVkZToKCi0gVHJhbnNmb3JtIHlvdXIgZGF0YSB0byBmb3JjZSBpdCB0byBiZXR0ZXIgYXBwcm94aW1hdGUgYSBub3JtYWwgZGlzdHJpYnV0aW9uIChwZXJoYXBzIGJ5IHVzaW5nIGxvZ2FyaXRobXMgb3IgW0JveC1Db3ggdHJhbnNmb3JtYXRpb25dKGh0dHA6Ly9vbmxpbmVzdGF0Ym9vay5jb20vMi90cmFuc2Zvcm1hdGlvbnMvYm94LWNveC5odG1sKSkuCi0gVXNlIGEgbm9ucGFyYW1ldHJpYyB0ZXN0IChyYW5kb21pemF0aW9uLWJhc2VkIG1ldGhvZHMgb3IgYSB0ZXN0IGxpa2UgdGhlIFtLcnVza2FsLVdhbGxpc10oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvS3J1c2thbOKAk1dhbGxpc19vbmUtd2F5X2FuYWx5c2lzX29mX3ZhcmlhbmNlKSB0ZXN0KQotIFVzZSByb2J1c3QgQU5PVkEgcHJvY2VkdXJlcyAod2hpY2ggd2UnbGwgaW52ZXN0aWdhdGUgbGF0ZXIgaW4gdGhlIGNvdXJzZSkKLSBVc2UgQmF5ZXNpYW4gbWV0aG9kcyAod2hpY2ggd2UnbGwgaW52ZXN0aWdhdGUgYSBiaXQgaW4gdGhpcyBjb3Vyc2UgYW5kIGluLWRldGFpbCBpbiBTVEFUIDMwNSkKCjxiciAvPgoKUmVjYWxsIGZyb20gdGhlIGZpcnN0IGxlc3NvbiB0aGF0IGlmIHdlIHdhbnQgdG8gY29uZHVjdCBhIHQtdGVzdCB3aXRob3V0IGEgbm9ybWFsaXR5IGFzc3VtcHRpb24sIHdlIGNvdWxkIHVzZSB0aGUgW1dlbGNoLVNhdHRlcnRod2FpdGVdKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL1dlbGNo4oCTU2F0dGVydGh3YWl0ZV9lcXVhdGlvbikgYXBwcm94aW1hdGlvbi4gIFdlIGNhbiBhbHNvIGRvIHRoaXMgaW4gQU5PVkEuICAKCkV4cGFuZCB0aGUgY29kZSB0byBzZWUgaG93IC0tPgoKYGBge3IgJ0FOT1ZBLXdlbGNoJ30KIyBOb3RpY2UgdGhlIGRlZ3JlZXMgb2YgZnJlZWRvbSBkaWZmZXIgZnJvbSBvdXIgInJlZ3VsYXIiIEFOT1ZBCm9uZXdheS50ZXN0KENvbXByZWhlbnNpb24gfiBDb25kaXRpb24sIGRhdGE9YW1iaWd1b3VzLCB2YXIuZXF1YWw9RkFMU0UpCgpgYGAKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6NTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCioqKioqCgojIyBSYW5kb21pemVkIG1lYW4tc3F1YXJlLXJhdGlvcwoKV2UndmUgc2VlbiBob3cgdG8gY29uZHVjdCByYW5kb21pemF0aW9uLWJhc2VkIHRlc3RzIG9mIHRoZSBTQUQgc3RhdGlzdGljICh3aGljaCBkbyBub3QgYWNjb3VudCBmb3IgdmFyaWFuY2Ugd2l0aGluIGVhY2ggZ3JvdXApLiAgV2UndmUgYWxzbyBzZWVuIGhvdyB0byBjb25kdWN0IGFuIEFOT1ZBICh3aGljaCByZXF1aXJlcyBjb25kaXRpb25zIG9mIG5vcm1hbGl0eSBhbmQgZXF1YWwgdmFyaWFuY2VzKS4gIENhbiB3ZSBnZXQgdGhlIGJlc3Qgb2YgYm90aCB3b3JsZHM/ICAKCkxldCdzIHNlZSBpZiB3ZSBjYW4gY29uZHVjdCBhIHJhbmRvbWl6YXRpb24tYmFzZWQgdGVzdCB1c2luZyB0aGUgRi1zdGF0aXN0aWMgKG1lYW4gc3F1YXJlIHJhdGlvKSBhcyBvdXIgdGVzdCBzdGF0aXN0aWMuICBFeHBhbmQgdGhlIGNvZGUgdG8gc2VlIGhvdyB0aGlzIGlzIGRvbmUuCgpgYGB7ciAncmFuZG9taXplZC1GLXRlc3QnfQojIFdlJ2xsIHN0b3JlIG91ciBvYnNlcnZlZCBtZWFuIHNxdWFyZSByYXRpbyBvZiAxMC4wMTIgYXMgTVNSb2JzZXJ2ZWQKIyBJIGNvdWxkIGhhdmUgdXNlZDogIE1TUm9ic2VydmVkIDwtIDEwLjAxMgpNU1JvYnNlcnZlZCA8LSB0aWR5KGFvdihDb21wcmVoZW5zaW9ufkNvbmRpdGlvbiwgZGF0YT1hbWJpZ3VvdXMpKSRzdGF0aXN0aWNbMV0KCiMgMTAsMDAwIHJlcGxpY2F0aW9ucywgc2h1ZmZsaW5nIHRoZSBDb25kaXRpb24gZWFjaCB0aW1lCk1TUnJhbmRvbWl6ZWQgPC0gRG8oMTAwMDApICogdGlkeShhb3YoQ29tcHJlaGVuc2lvbiB+IHNodWZmbGUoQ29uZGl0aW9uKSwgZGF0YT1hbWJpZ3VvdXMpKSRzdGF0aXN0aWNbMV0KCiMgUGxvdCByZXN1bHRzCmdncGxvdChkYXRhID0gTVNScmFuZG9taXplZCwgYWVzKHggPSByZXN1bHQpKSArCiAgZ2VvbV9kZW5zaXR5KGZpbGw9ImxpZ2h0Ymx1ZSIsIGNvbG9yPSJ3aGl0ZSIsIGFscGhhID0gMC44KSArCiAgYW5ub3RhdGUoInNlZ21lbnQiLCB4ID0gTVNSb2JzZXJ2ZWQsIHhlbmQgPSBNU1JvYnNlcnZlZCwgeSA9IDAsIHllbmQgPSAuMiwgY29sb3IgPSAicmVkIikgKwogIGFubm90YXRlKCJ0ZXh0IiwgeCA9IE1TUm9ic2VydmVkLCB5ID0gLjI1LCBsYWJlbCA9ICJvYnNlcnZlZCBNU1IgPSAxMC4wMTIiLCBjb2xvciA9ICJyZWQiKSArCiAgbGFicygKICAgICAgdGl0bGUgPSAiUmFuZG9taXplZCBtZWFuIHNxdWFyZSByYXRpb3MiLAogICAgICB4ID0gIm1lYW4gc3F1YXJlIHJhdGlvcyIKICAgICAgKSArCiAgc2NhbGVfeF9jb250aW51b3VzKGxpbWl0cyA9IGMoMCwxMiksIGJyZWFrcz1zZXEoMCwgMTIsIDIpLCBtaW5vcl9icmVha3M9TlVMTCkgKwogICAgdGhlbWUoCiAgICBheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTEsIGNvbG9yPSJncmV5MTAiKSwKICAgIGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIiwKICAgIHBhbmVsLmdyaWQubWFqb3IueSA9IGVsZW1lbnRfbGluZShjb2xvdXIgPSAid2hpdGUiKSwKICAgIHBhbmVsLmdyaWQubWFqb3IueCA9IGVsZW1lbnRfbGluZShjb2xvdXIgPSAid2hpdGUiLCBzaXplPS4xNSksCiAgICBwYW5lbC5ncmlkLm1pbm9yID0gZWxlbWVudF9ibGFuaygpLAogICAgcGFuZWwuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsID0gImdyZXk5MyIpCiAgKQoKIyBFc3RpbWF0ZSBwLXZhbHVlCnB2YWx1ZSA8LSBwcm9wKE1TUnJhbmRvbWl6ZWQkcmVzdWx0ID49IE1TUm9ic2VydmVkKQpwdmFsdWUKCgpgYGAKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6NTBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCioqKioqCiMjIEFOT1ZBIHZzLiB0LXRlc3QKCkxldCdzIChmaW5hbGx5KSBmaW5pc2ggdGhpbmdzIHVwIGJ5IGxvb2tpbmcgYXQgdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIHRoZSBGLXN0YXRpc3RpYyAobWVhbiBzcXVhcmUgcmF0aW8pIGluIGFuIEFOT1ZBIGFuZCB0aGUgdC1zdGF0aXN0aWMgaW4gYSB0LXRlc3Q6Cgo8YnIgLz4KCiR0X3tcdGV4dHJte259XzErXHRleHRybXtufV8yLTJ9PVxmcmFje1xvdmVybGluZXtYfV97MX0tXG92ZXJsaW5le1h9X3syfX17XHNxcnR7XGZyYWN7MX17XHRleHRybXtufV8xfStcZnJhY3sxfXtcdGV4dHJte259XzJ9fVxzcXJ0e1xmcmFje1xsZWZ0ICggXHRleHRybXtufV8xLTEgXHJpZ2h0IClzX3sxfV57Mn0rXGxlZnQgKCBcdGV4dHJte259XzItMSBccmlnaHQgKXNfezJ9XnsyfX17XHRleHRybXtufV8xK1x0ZXh0cm17bn1fMi0yfX19PVxmcmFje1xvdmVybGluZXtYfV97MX0tXG92ZXJsaW5le1h9X3syfX17XHNxcnR7XGZyYWN7MX17XHRleHRybXtufV8xfStcZnJhY3sxfXtcdGV4dHJte259XzJ9fVxzcXJ0e3Nfe3Bvb2xlZH1eezJ9fX0kCgo8YnIgLz4KCiR0X3t7XHRleHRybXtufV8xK1x0ZXh0cm17bn1fMi0yfX1eezJ9PVxmcmFje1xsZWZ0IChcb3ZlcmxpbmV7WH1fezF9LVxvdmVybGluZXtYfV97Mn0gIFxyaWdodCApXnsyfX17XGxlZnQgKCBcZnJhY3sxfXtuX3sxfX0rIFxmcmFjezF9e25fezJ9fVxyaWdodCApfXNfe3Bvb2xlZH1eezJ9PVxmcmFje1xsZWZ0ICggbl97MX0rbl97Mn0gXHJpZ2h0IClcbGVmdCAoXG92ZXJsaW5le1h9X3sxfS1cb3ZlcmxpbmV7WH1fezJ9ICBccmlnaHQgKV57Mn19e3Nfe3Bvb2xlZH1eezJ9fT1cZnJhY3tuX3sxfVxsZWZ0IChcb3ZlcmxpbmV7WH1fezF9LVxvdmVybGluZXtYfV97Mn0gIFxyaWdodCApXnsyfStuX3syfVxsZWZ0IChcb3ZlcmxpbmV7WH1fezF9LVxvdmVybGluZXtYfV97Mn0gIFxyaWdodCApXnsyfX17e3Nfe3Bvb2xlZH1eezJ9fX0kCgo8YnIgLz4KCiR0X3t7XHRleHRybXtufV8xK1x0ZXh0cm17bn1fMi0yfX1eezJ9PVxmcmFje1xzdW0gbl97YX1cbGVmdCAoXG92ZXJsaW5le1h9X3thfS1NICBccmlnaHQgKV57Mn0vMX17XHRleHRybXtNU31fe1x0ZXh0cm17RX19fT1cZnJhY3tcdGV4dHJte1NTfV97XHRleHRybXtBfX0vXHRleHRybXtkZn1fe1x0ZXh0cm17QX19fXtcdGV4dHJte01TfV97XHRleHRybXtFfX19PVxmcmFje1x0ZXh0cm17TVN9X3tcdGV4dHJte0F9fX17XHRleHRybXtNU31fe1x0ZXh0cm17RX19fT1GJAoKPGJyIC8+CgozNC4gRXhwbGFpbiB0aGUgY29uc2VxdWVuY2VzIG9mIHdoYXQgd2FzIGp1c3QgZGVyaXZlZC4gIEFsc28sIHZlcmlmeSB0aGlzIHJlc3VsdCB3aXRoIHRoZSBmb2xsb3dpbmcgb3V0cHV0LgoKYGBge3IgJ3R0ZXN0LWFub3ZhJ30KIyBSZWNyZWF0ZSB0aGUgc3BpZGVyIGRhdGFzZXQgZnJvbSBsZXNzb24gIzEKc3BpZGVyIDwtIHRpYmJsZSgKICBncm91cCA9IGMoIHJlcCgicGhvdG8iLCAxMiksIHJlcCgicmVhbCIsIDEyKSApLAogIGFueGlldHkgPSBjKDMwLCAzNSwgNDUsIDQwLCA1MCwgMzUsIDU1LCAyNSwgMzAsIDQ1LCA0MCwgNTAsIAogICAgICAgICAgICAgIDQwLCAzNSwgNTAsIDU1LCA2NSwgNTUsIDUwLCAzNSwgMzAsIDUwLCA2MCwgMzkpKQoKIyBSdW4gYSB0LXRlc3QKdC50ZXN0KGFueGlldHkgfiBncm91cCwgZGF0YT1zcGlkZXIsIGFsdGVybmF0aXZlID0gYygibGVzcyIpLCB2YXIuZXF1YWwgPSBUUlVFLCBjb25mLmxldmVsID0gMC45NSkKIyBTdG9yZSB0aGUgdmFsdWUgb2YgdGhlIHQtc3RhdGlzdGljICgtMS42OCkKdHN0YXQgPC0gdC50ZXN0KGFueGlldHkgfiBncm91cCwgZGF0YT1zcGlkZXIsIGFsdGVybmF0aXZlID0gYygibGVzcyIpLCB2YXIuZXF1YWwgPSBUUlVFLCBjb25mLmxldmVsID0gMC45NSkkc3RhdGlzdGljCgojIFJ1biBhbiBBTk9WQSBvbiB0aGUgc2FtZSBkYXRhCmFub3ZhKGFvdihhbnhpZXR5IH4gZ3JvdXAsIGRhdGE9c3BpZGVyKSkKIyBOb3RpY2UgdGhlIEYtc3RhdGlzdGljID0gMi44MjY5CiMgSXMgdGhhdCB0aGUgc2FtZSBhcyB0aGUgdHN0YXQgc3F1YXJlZD8gIExldCdzIHNlZS4uLgp0c3RhdF4yCgpgYGAKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoKCgoKCgoKfCBDb25kaXRpb24gICAgIHwgbiAgICAgICAgICB8IG1lYW4gICAgICAgICAgfCBzZCAgICAgICAgICAgIHwKfC0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLTp8LS0tLS0tLS0tLS0tLS06fC0tLS0tLS0tLS0tLS0tOnwKfCBBZnRlciAgICAgICAgIHwgMTkgICAgICAgICB8IDMuMjExICAgICAgICAgfCAxLjM5OCAgICAgICAgIHwKfCBCZWZvcmUgICAgICAgIHwgMTkgICAgICAgICB8IDQuOTQ3ICAgICAgICAgfCAxLjMxMSAgICAgICAgIHwKfCBOb25lICAgICAgICAgIHwgMTkgICAgICAgICB8IDMuMzY4ICAgICAgICAgfCAxLjI1NyAgICAgICAgIHwKfCAqKkNvbWJpbmVkKiogIHwgKipOID0gNTcqKiB8ICoqTSA9IDMuODQyKiogfCAqKnMgPSAxLjUyMSoqIHwKCgphb3YoIGFub3ZhKENvbXByZWhlbnNpb24gfiBDb25kaXRpb24sIGRhdGEgPSBhbWJpZ3VvdXMpICkKCgoqKioqKgoKIyBZb3VyIHR1cm4KCjM1LiBBcmUgc29tZSBkaWV0cyBtb3JlIGVmZmVjdGl2ZSB0aGFuIG90aGVycz8gIDkzIHN1YmplY3RzIHdlcmUgcmFuZG9tbHkgYXNzaWduZWQgdG8gb25lIG9mIGZvdXIgZGlldHM6PGJyIC8+PGJyIC8+ICZuYnNwOyAoYSkgKipBdGtpbnMqKiwgYSBsb3ctY2FyYiBkaWV0LCAmbmJzcDsgKGIpICoqWm9uZSoqLCBhIDQtNC0zIHJhdGlvIG9mIGNhcmJzLXByb3RlaW4tZmF0LCA8YnIgLz4gJm5ic3A7IChjKSAqKk9ybmlzaCoqLCBhIGxvdy1mYXQgZGlldCwgb3IgKGQpICoqV2VpZ2h0IFdhdGNoZXJzKiouIDxiciAvPjxiciAvPiBTdWJqZWN0cyB3ZXJlIGVkdWNhdGVkIG9uIHRoZWlyIGFzc2lnbmVkIGRpZXQgYW5kICB3ZXJlIG1vbml0b3JlZCBhcyB0aGV5IHN0YXllZCBvbiB0aGUgZGlldCBmb3Igb25lIHllYXIuIEF0IHRoZSBlbmQgb2YgdGhlIHllYXIsIHJlc2VhcmNoZXJzIGNhbGN1bGF0ZWQgdGhlICoqY2hhbmdlcyBpbiB3ZWlnaHQqKiBmb3IgZWFjaCBzdWJqZWN0LiA8YnIgLz48YnIgLz4oYSkgU3VwcG9zZSB1c2VkIHQtdGVzdHMgKHdpdGggJFxhbHBoYT0wLjA1JCBmaXIgZWFjaCB0ZXN0KSB0byBjb21wYXJlIGFsbCBwb3NzaWJsZSBwYWlycyBvZiBncm91cCBtZWFucy4gIENhbGN1bGF0ZSB0aGUgb3ZlcmFsbCBwcm9iYWJpbGl0eSBvZiBtYWtpbmcgYXQgbGVhc3Qgb25lIFR5cGUgSSBlcnJvciBhY3Jvc3MgYWxsIHRoZSB0LXRlc3RzLjxiciAvPjxiciAvPihiKSBDaGVjayB0aGUgY29uZGl0aW9ucyBuZWNlc3NhcnkgdG8gY29uZHVjdCBhbiBBTk9WQS4gIENob29zZSB5b3VyIG1ldGhvZHMgYW5kIGludGVycHJldCB0aGUgcmVzdWx0cy4gPGJyIC8+PGJyIC8+KGMpIENvbmR1Y3QgYW4gQU5PVkEgb24gdGhpcyBkYXRhICh3aXRoIG9yIHdpdGhvdXQgdGhlIGVxdWFsIHZhcmlhbmNlIGFzc3VtcHRpb24pIGFuZCB3cml0ZSBvdXQgYW55IGNvbmNsdXNpb25zIHlvdSBjYW4gZHJhdy4gPGJyIC8+PGJyIC8+KGQpIENhbGN1bGF0ZSBldGEtc3F1YXJlZCBhbmQgb21lZ2Etc3F1YXJlZC48YnIgLz48YnIgLz4oZSkgQ29uZHVjdCBhIHJhbmRvbWl6ZWQgdGVzdCBvZiB0aGUgU0FEIChvciBNQUQpIHN0YXRpc3RpYyBmb3IgdGhpcyBkYXRhLiAgRXN0aW1hdGUgdGhlIHAtdmFsdWUuPGJyIC8+PGJyIC8+VGhlIGBkaWV0YCBkYXRhZnJhbWUgaGFzIGJlZW4gbG9hZGVkLjxiciAvPlRoZSB2YXJpYWJsZXMgYXJlIGBEaWV0YCAoYSBmYWN0b3IgdmFyaWFibGUgaWRlbnRpZnlpbmcgd2hpY2ggZGlldCBlYWNoIHN1YmplY3QgaXMgYXNzaWduZWQgdG8pIGFuZCBgV2VpZ2h0bG9zc2AgKHRoZSBudW1iZXIgb2YgcG91bmRzIGxvc3QgYnkgZWFjaCBzdWJqZWN0IGJ5IHRoZSBlbmQgb2YgdGhlIHN0dWR5KS48YnIgLz48YnIgLz5BIHN1bW1hcnkgb2YgdGhlIGRhdGEgaXMgZGlzcGxheWVkIGJlbG93LiAgTm90aWNlIHRoZSB2YXJpYWJsZXMgYmVnaW4gd2l0aCAqKkNhcGl0YWwqKiBsZXR0ZXJzIGFuZCB0aGF0IG5lZ2F0aXZlIHZhbHVlcyBvZiB0aGUgKipXZWlnaHRsb3NzKiogdmFyaWFibGUgaW5kaWNhdGUgc3ViamVjdHMgd2hvICpnYWluZWQqIHdlaWdodC4KCiFbXShkaWV0ZGF0YS5wbmcpCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjUwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoKKioqKioKCiMgUHVibGlzaGluZyBZb3VyIFR1cm4gc29sdXRpb25zCgotIERvd25sb2FkIHRoZSBbWW91ciBUdXJuIFNvbHV0aW9ucyB0ZW1wbGF0ZV0oaHR0cDovL2JyYWR0aGllc3Nlbi5jb20vaHRtbDUvc3RhdHMvbTMwMS8zL1lUVDMuUm1kKQogICAgIC0gUlN0dWRpbyB3aWxsIGF1dG9tYXRpY2FsbHkgb3BlbiBpdCBpZiBpdCBkb3dubG9hZHMgYXMgYSAqLlJtZCogZmlsZS4KICAgICAtIElmIGl0IGRvd25sb2FkcyBhcyBhICoudHh0KiBmaWxlLCB0aGVuIHlvdSBjYW46CiAgICAgICAgLSBvcGVuIHRoZSBmaWxlIGluIGEgdGV4dCBlZGl0b3IgYW5kIGNvcHkgaXRzIGNvbnRlbnRzCiAgICAgICAgLSBvcGVuIFJTdHVkaW8gYW5kIGNyZWF0ZSBhIG5ldyAqKlIgTWFya2Rvd24qKiBmaWxlIChodG1sIGZpbGUpCiAgICAgICAgLSBkZWxldGUgZXZlcnl0aGluZyBhbmQgcGFzdGUgdGhlIGNvbnRlbnRzIG9mIHRoZSB0ZW1wbGF0ZSAKCi0gVHlwZSB5b3VyIHNvbHV0aW9ucyBpbnRvIHRoZSBbWW91ciBUdXJuIFNvbHV0aW9ucyB0ZW1wbGF0ZV0oaHR0cDovL2JyYWR0aGllc3Nlbi5jb20vaHRtbDUvc3RhdHMvbTMwMS8zL1lUVDMuUm1kKQogICAgIC0gSSd2ZSB0cmllZCB0byBzaG93IHlvdSAqd2hlcmUqIHRvIHR5cGUgZWFjaCBvZiB5b3VyIHNvbHV0aW9ucy9hbnN3ZXJzCiAgICAgLSBZb3UgY2FuIHJ1biB0aGUgY29kZSBhcyB5b3UgdHlwZSBpdC4KCi0gV2hlbiB5b3UndmUgYW5zd2VyZWQgZXZlcnkgcXVlc3Rpb24sIGNsaWNrIHRoZSAqKktuaXQgSFRNTCoqIGJ1dHRvbiBsb2NhdGVkIGF0IHRoZSB0b3Agb2YgUlN0dWRpbzogPGltZyBzcmM9Imh0dHA6Ly9icmFkdGhpZXNzZW4uY29tL2h0bWw1L3N0YXRzL20zMDAva25pdHIucG5nIiBhbHQ9IkRyYXdpbmciLz4KICAgICAtIFJTdHVkaW8gd2lsbCBiZWdpbiB3b3JraW5nIHRvIGNyZWF0ZSBhIC5odG1sIGZpbGUgb2YgeW91ciBzb2x1dGlvbnMuCiAgICAgLSBJdCBtYXkgdGFrZSBhIGZldyBtaW51dGVzIHRvIGNvbXBpbGUgZXZlcnl0aGluZyAoaWYgeW91J3ZlIGNvbmR1Y3RlZCBhbnkgc2ltdWxhdGlvbnMpCiAgICAgLSBXaGlsZSB0aGUgZmlsZSBpcyBjb21waWxpbmcsIHlvdSBtYXkgc2VlIHNvbWUgcmVkIHRleHQgaW4gdGhlIGNvbnNvbGUuICAKICAgICAtIFdoZW4gdGhlIGZpbGUgaXMgZmluaXNoZWQsIGl0IHdpbGwgb3BlbiBhdXRvbWF0aWNhbGx5IChvciBpdCB3aWxsIGdpdmUgeW91IGFuIGVycm9yIG1lc3NhZ2UpLgogICAgICAgIC0gSWYgdGhlIGZpbGUgZG9lcyBub3Qgb3BlbiBhdXRvbWF0aWNhbGx5LCB5b3UnbGwgc2VlIGFuIGVycm9yIG1lc3NhZ2UgaW4gdGhlIGNvbnNvbGUuCgotIE9uY2UgeW91J3JlIHNhdGlzZmllZCB3aXRoIHlvdXIgc29sdXRpb25zLCBlbWFpbCB0aGF0IGh0bWwgZmlsZSB0byBbdGhpZXNzZW5icmFkbGV5YUBzYXUuZWR1XShtYWlsdG86dGhpZXNzZW5icmFkbGV5YUBzYXUuZWR1KSBvciBnaXZlIG1lIGEgcHJpbnRlZCBjb3B5LgoKLSBJZiB5b3UgcnVuIGludG8gYW55IHByb2JsZW1zLCBbZW1haWwgbWVdKG1haWx0bzp0aGllc3NlbmJyYWRsZXlhQHNhdS5lZHUpIG9yIHdvcmsgd2l0aCBvdGhlciBzdHVkZW50cyBpbiB0aGUgY2xhc3MuCgo8YnIgLz4KCjxkaXYgaWQ9ImxpY2Vuc2UiPgoqKlNvdXJjZXMqKgoKVGhpcyBkb2N1bWVudCBpcyByZWxlYXNlZCB1bmRlciBhIFtDcmVhdGl2ZSBDb21tb25zIEF0dHJpYnV0aW9uLVNoYXJlQWxpa2UgMy4wIFVucG9ydGVkXShodHRwOi8vY3JlYXRpdmVjb21tb25zLm9yZy9saWNlbnNlcy9ieS1zYS8zLjApIGxpY2Vuc2UuCjwvZGl2Pgo=