# Install and load all necessary packages
list.of.packages <- c("tidyverse", "mosaic", "energy", "ggExtra")
new.packages <- list.of.packages[!(list.of.packages %in% installed.packages()[,"Package"])]
if(length(new.packages)) install.packages(new.packages)
library(tidyverse)
library(mosaic)
library(energy)
library(ggExtra)

Scenario: Villains wear black

Do NFL teams with more malevolent-looking uniforms play more aggressively than other teams?


In 1988 25 volunteers were paid $2 each to rate the perceived malevolence of each NFL team’s uniform and logo. The total yards each team was penalized that year were then recorded as z-scores (to represent standardized penalty yards for each team).

Here’s a sample of the data:



Covariance

  1. Describe the relationship between these two variables. Are larger values of malevolence associated with larger values of penalty yards? When one variable deviates from its mean, does the other variable deviate from its mean in a similar way?

Recall the concept of variance: \(s_{x}^{2}=\frac{\sum \left ( x_{i}-\bar{X} \right )^{2}}{n-1}=\frac{\sum \left ( x_{i}-\bar{X} \right )\left ( x_{i}-\bar{X} \right )}{n-1}=s_{xx}\)


We can visualize the covariance of our data in a couple ways:

diagram



scatterplot
nfl_sample <- tibble(
  malevolence = c(5.1, 4.68, 4, 3.9, 3.38, 2.8),
  penalty = c(1.19, 0.29, -0.73, -0.81, 0.04, -1.6))

ggplot(nfl_sample, aes(x = malevolence, y=penalty)) +
  geom_point(color="steelblue", size=3)

.

To calculate the covariance, we can modify the formula for variance:

\(\textrm{cov}_{xy}=s_{xy}=\frac{\sum \left ( x_{i}-\bar{X} \right )\left ( y_{i}-\bar{Y} \right )}{n-1}=\frac{(5.1-3.977)(1.19- -0.27)+...+(2.8-3.977)(-1.60- -0.27)}{6-1}=0.6889\).

# This calculates the variance-covariance matrix
cov(nfl_sample)


  1. Under what conditions would the covariance be negative? positive? zero?
  1. If we multiplied every value of X and Y by 10. What would happen to the covariance? What are the lowest and highest possible values for a covariance?
# Multiply everything by 10
ten_times <- nfl_sample * 10
cov(ten_times)

That’s one limitation of using covariance as a measure of the strength of the relationship between two variables: it depends on the scale of measurement used.

For example, the penalty yards variable in our data are reported as z-scores. If we were to use (unstandardized) penalty yards, the covariance increases to 4.822. If we used penalty feet (instead of yards), the covariance would again increase to 14.467. These examples show that larger values of covariance do not necessarily indicate stronger relationships between variables. Because of this, we shouldn’t directly compare covariances.



Correlation: standardized covariance

To address this scale dependence, we need to use a unit of measurement (e.g., meters, pounds, decibels) into which any scale of measurement (e.g., penalty distance, uniform malevolence)could be converted.

If we use the standard deviation as that standard unit of measurement, we can standardize the covariance by:

  • Standardizing our variables and calculating the covariance: \(\frac{\sum \left ( z_{xi}-\bar{Z_{x}} \right )\left ( z_{yi}-\bar{Z_{y}} \right )}{n-1}=\frac{\sum z_{x}z_{y}}{n-1}=r_{xy}\)

or

  • Calculating the covariance and standardizing it: \(\frac{\sum \left ( x_{i}-\bar{X} \right )\left ( y_{i}-\bar{Y} \right )}{(n-1)s_{x}s_{y}}=r_{xy}\)


This is called Pearson's product-moment correlation coefficient (or r).


  1. What are the smallest and largest values of r?

Calculating the correlation coefficient

  1. Expand the code to see how to calculate the correlation coefficient in R. Interpret this correlation.
# Using cor()
cor(nfl_sample$malevolence, nfl_sample$penalty)

# Using cor() within dplyr
nfl_sample %>%
  summarize(correlation = cor(malevolence, penalty))

Let’s now calculate the correlation for our full dataset:

# Load data
nfl <- read.csv("http://www.bradthiessen.com/html5/data/NFLuniforms.csv")

# Rename variables
names(nfl) <- c("team", "malevolence", "penalty")

# Plot and annotate with correlation
ggplot(nfl, aes(x = malevolence, y=penalty)) +
  geom_point(color="steelblue", size=3) +
  annotate("text", x=4.5, y=-1.2, label=paste("r = ", round(cor(nfl$malevolence, nfl$penalty),4)))

Characteristics of the Person product-moment correlation

Scale invariance

The magnitude of the correlation does not change under any linear transformation of the variables. Expand the code to see an example.

# Let's convert malevolence to centi-malevolence
nfl$centi_malevolence <- nfl$malevolence / 100

# The correlation does not change
cor(nfl$centi_malevolence, nfl$penalty)


Strength of linear relationship

Pearson’s r only measures the strength of a linear relationship. If we want to measure the strength of nonlinear relationships, we’ll need another statistic. Here are some values of r for various scatterplots:


Impact of outliers

  1. Below, I’ve highlighted the two most extreme observations in our data. We know the correlation between malevolence and penalty yards for all the data is \(r_{xy}=0.43\). What would happen to the value of this correlation if those two highlighted values were removed?
# Add a variable to highlight the two outliers
nfl$outlier <- c(1, rep(0,26), 1)

# Create scatterplot with the outliers highlighted
ggplot(data=nfl, aes(x = malevolence, y = penalty, color=outlier)) +
  geom_point(size=3) + theme(legend.position="none")

# Calculate correlation without highlighted observations
nfl %>%
  filter(outlier==0) %>%
  summarize(correlation = cor(malevolence, penalty))

Impact of range restriction

Let’s simulate a large dataset with a strong, positive correlation:

# Simulate data
sim_data <- tibble(
  x = c(1:1000),
  y = 2*x + rnorm(1000, 3, 300))

# Plot
ggplot(data = sim_data, aes(x, y)) +
  geom_point(color="steelblue", size=2, alpha=0.5) +
  annotate("text", x = 125, y = 2222, label=paste("r = ",round(cor(sim_data$x, sim_data$y),3))) +
  geom_vline(xintercept=900, color="red")


  1. Look at that sample data. Suppose we only restrict the range to only include data in which \(x>900\) (the data to the right of the red line). What will happen to the magnitude of the correlation? Why?
# Calculate correlation when only considering x>900
sim_data %>%
  filter(x>900) %>%
  summarize(correlation = cor(x, y))

Correlation does not imply causation

Everyone thinks they know correlation is not causation. A correlation does not imply a causal relationship, and a lack of correlation does not mean there is no relationship between two variables.

There are many factors that contribute to this. We’ve already seen that outliers and range restriction can influence the magnitude (and even sign) of Pearson’s r. Likewise, if two variables have a nonlinear relationship, Pearson’s r will misrepresent that relationship.

Yet another reason is the third-variable problem.

  1. The number of heart attacks in a given month is positively correlated with ice cream sales. Explain why this does not provide evidence that ice cream causes heart attacks.

A 2004 article questioned research into the relationship between hormone replacement therapy (HRT) and coronary heart disease (CHD). Numerous studies had shown negative correlations between HRT and CHD, leading doctors to propose that HRT was protective against CHD.

Randomized controlled trials showed, in fact, that HRT caused a small but significant increase in the risk of CHD.

Re-analysis of the data from the original studies showed that women undertaking HRT were more likely to be from higher socioeconomic groups. These women, therefore, were more likely to have better-than-average diets and exercise regimens.

The third variable linking hormone replacement therapy to coronary heart disease was socioeconomic status.


  1. In 2002, a letter in Diabetes Care detailed the calculation behind the correlation of r = 0.54 between diabetes rates and pollution levels across all 50 states.

    How would you interpret this correlation?
    The author tried to be careful in concluding, “… the correlation between air emissions and the prevalence of diabetes does not prove a cause-and-effect relationship; the significance of the relationship demands attention.”.

    In response, Mark Nicolich questioned if the relationship even demanded attention. Using the same diabetes data, Nicolich calculated the correlation between diabetes rates and the alphabetized rank of each state to be r = 0.49. He also found the correlation between diabetes rates and the latitude of each state’s capital to be r = -0.54.
  1. See if you can think of a scenario which would result in the following scatterplot. What is to be learned from this?
test <- tibble(
  x = c(rnorm(100, 6, 2), rnorm(100, 10, 2), rnorm(100, 14, 2)),
  y = c(rnorm(100, 6, 2), rnorm(100, 10, 2), rnorm(100, 14, 2)),
  group = as.factor(c(rep(1,100), rep(2,100), rep(3,100)))
)

# Calculate overall correlation
overall <- test %>%
  summarize(r = cor(x, y))

# Calculate correlation for each subgroup
subs <- test %>%
  group_by(group) %>%
  summarize(r = cor(x, y))

# Plot and annotate with correlations
ggplot(data = test, aes(x, y, color = group)) +
  geom_point(size = 2, alpha=0.7) + 
  geom_smooth(method="lm", se=FALSE) + 
  theme(legend.position="none") +
  annotate("text", x = 5, y = 18, label=paste("overall r =", round(overall, 2)), color="black") +
  annotate("text", x = 2, y = 4,  label = paste("r =", round(subs[1,2], 2)), color="red") +
  annotate("text", x = 16, y = 9, label = paste("r =", round(subs[2,2], 2)), color="forestgreen") +
  annotate("text", x = 18.8, y = 16, label = paste("r = ", round(subs[3,2], 2)), color = "blue")

If you’d like to explore other examples in which correlation is mistaken for causation, check out:

Spurious correlations

Correlation or Causation



Nonzero correlations?

  1. What correlation would you expect to find between two sets of random values? Expand the code to see –>
set.seed(3141)
# Generate random values for X and Y
random_xy <- tibble(
  x = rnorm(100, 0, 1),
  y = rnorm(100, 0, 1)
) 

cor(random_xy$x, random_xy$y)

ggplot(data = random_xy, aes(x, y)) +
  geom_point(size=3, color="steelblue") +
    annotate("text", x = 0, y = 2.8, label=paste("overall r =", round(cor(random_xy$x, random_xy$y), 3)), color="black")
  

The correlation of any two variables in any sample of data will be non-zero, so how can we have any confidence that a correlation is “real?”


Randomization-based test

  1. A subset of our data is displayed below. What would the (null) randomization hypothesis be in this scenario? Under that hypothesis, how likely were the Raiders (with a 5.1 malevolence rating) to have 1.19, 0.48, or 0.27 standardized penalty yards?

    Explain how will we use randomization-based methods to test our randomization hypothesis.

Let’s take a look at some randomizations of our data:

# Store our sample correlation as test_stat
test_stat <- cor(nfl$malevolence, nfl$penalty)

# Randomize our data 
p1 <- nfl %>%
  ggplot(., aes(x = malevolence, y = penalty)) +
  geom_point(color="steelblue", size=2) +
  geom_smooth(method="lm", se=FALSE, color="red", size=.2) +
  annotate("text", x = 4, y = 1, label = "actual data", color="red", size=3)

p2 <- nfl %>%
  mutate(shuffled_penalty = sample(penalty)) %>%
  ggplot(., aes(x = malevolence, y = shuffled_penalty)) +
  geom_point(color="steelblue", size=2) +
  geom_smooth(method="lm", se=FALSE, color="red", size=.2) +
  annotate("text", x = 4, y = 1, label = "randomization #1", color="red", size=3)

p3 <- nfl %>%
  mutate(shuffled_penalty = sample(penalty)) %>%
  ggplot(., aes(x = malevolence, y = shuffled_penalty)) +
  geom_point(color="steelblue", size=2) +
  geom_smooth(method="lm", se=FALSE, color="red", size=.2) +
  annotate("text", x = 4, y = 1, label = "randomization #2", color="red", size=3)

p4 <- nfl %>%
  mutate(shuffled_penalty = sample(penalty)) %>%
  ggplot(., aes(x = malevolence, y = shuffled_penalty)) +
  geom_point(color="steelblue", size=2) +
  geom_smooth(method="lm", se=FALSE, color="red", size=.2) +
  annotate("text", x = 4, y = 1, label = "randomization #3", color="red", size=3)

p5 <- nfl %>%
  mutate(shuffled_penalty = sample(penalty)) %>%
  ggplot(., aes(x = malevolence, y = shuffled_penalty)) +
  geom_point(color="steelblue", size=2) +
  geom_smooth(method="lm", se=FALSE, color="red", size=.2) +
  annotate("text", x = 4, y = 1, label = "randomization #4", color="red", size=3)

p6 <- nfl %>%
  mutate(shuffled_penalty = sample(penalty)) %>%
  ggplot(., aes(x = malevolence, y = shuffled_penalty)) +
  geom_point(color="steelblue", size=2) +
  geom_smooth(method="lm", se=FALSE, color="red", size=.2) +
  annotate("text", x = 4, y = 1, label = "randomization #5", color="red", size=3)

# This uses a "multiplot" function that was loaded silently
multiplot(p1, p4, p2, p5, p3, p6, cols=3)


Each randomization of data yields a non-zero correlation. Let’s run 10,000 randomizations and plot all 10,000 correlations.

# Using cor()
shuffled_r <- Do(10000) * cor(malevolence ~ shuffle(penalty), data=nfl)

# Using dplyr
# shuffled_rs <- Do(10000) * nfl %>%
#   mutate(shuffled = sample(penalty)) %>%
#   summarize(cor = cor(shuffled, nfl$malevolence))

# Calculate likelihood of test statistic
pvalue <- prop(shuffled_r$cor >= test_stat)
pvalue

# Plot simple histogram
# histogram(~cor, data=shuffled_r,
#           width=.04,
#           xlab="Possible correlations assuming null hypothesis is true")
# ladd(panel.abline(v=test_stat))   # Add vertical line at test statistic

# Plot
ggplot(data = shuffled_r, aes(x = cor)) +
  geom_histogram(binwidth = 0.04, fill="lightblue", color="white", alpha=0.8) +
  annotate("segment", x = test_stat, xend = test_stat, y = 0, yend = 600, color = "red") +
  annotate("text", x = test_stat, y = 630, label = "observed r = 0.43", color = "red") +
  annotate("text", x = test_stat+.11, y = 170, label = paste("p =",round(pvalue,4)), color = "red") +
  labs(
      title = "Randomized Correlations",
      x = "r"
      ) +
  scale_x_continuous(limits = c(-.7,.7), breaks=seq(-.6, .6, .3), minor_breaks=NULL) +
    theme(
    axis.text.x = element_text(size = 11, color="grey10"),
    legend.position = "none",
    panel.grid.major.y = element_line(colour = "white", size=.25),
    panel.grid.major.x = element_line(colour = "white", size=.15),
    panel.grid.minor = element_blank(),
    panel.background = element_rect(fill = "grey93")
  )


Bootstrap confidence interval

  1. Interpret the following confidence interval and explain how it was constructed.
boot <- Do(10000) * cor( malevolence ~ penalty, data=resample(nfl) )

bootstrapCI <- confint(boot, level = 0.95, method = "quantile")
lower <- as.numeric(bootstrapCI[2]) # Store lower CI bound
upper <- as.numeric(bootstrapCI[3]) # Store upper CI bound

# Density plot
ggplot(data = boot, aes(x = cor)) +
  geom_density(fill="lightblue", color="white", alpha = 0.8) +
  labs(
      title = "Bootstrap distribution of correlations",
      x = "bootstrap correlations"
      ) +
  scale_x_continuous(breaks=seq(-0.8, 0.8, 0.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")
  ) +
  annotate("text", x = lower, y = .2, label = round(lower,3)) +
  annotate("text", x = upper, y = .2, label = round(upper,3)) +
  annotate("text", x = median(boot$cor), y = .25, label = paste("95%")) +
  annotate("segment", x = lower, xend = upper, y = 0.1, yend = .1, color = "red")

t-test for correlation

If we assume our data are sampled from populations with normal distributions, we can use a t-test.

In the next lesson, we’ll derive the following test statistic for the correlation coefficient:

\(t_{n-2}=\frac{r_{xy}-0}{\textrm{SE}_{r_{xy}}}=\frac{r_{xy}}{\sqrt{\frac{1-r_{xy}^{2}}{n-2}}}=\frac{r_{xy}\sqrt{n-2}}{\sqrt{1-r_{xy}^{2}}}=2.427\ \ (p=.011)\)


We can run this test in R using the cor.test() function:

cor.test(nfl$malevolence, nfl$penalty, alternative = c("greater"), method=c("pearson"))

    Pearson's product-moment correlation

data:  nfl$malevolence and nfl$penalty
t = 2.4272, df = 26, p-value = 0.01122
alternative hypothesis: true correlation is greater than 0
95 percent confidence interval:
 0.1299369 1.0000000
sample estimates:
     cor 
0.429796 


  1. Is there anything (worthwhile) we can conclude from this t-test?

Other types of correlations

Nonparametric correlations (correlations of ranks)

Spearman’s rho

Spearman’s rho uses the same formula as Pearson’s r, except the data are converted to ranks.


  1. Using data from 8 MBA graduates, calculate Spearman’s rho to estimate the strength of the relationship between scores on the GMAT (a test they took prior to entering graduate school) and their grade point average in the MBA program. To do this, first convert the scores into ranks.



# Enter the raw data
mba <- tibble(
  gmat = c(710, 610, 640, 580, 545, 560, 610, 530),
  gpa  = c(4, 4, 3.9, 3.8, 3.7, 3.6, 3.5, 3.5)

)

# Calculate Pearson's r and Spearman's rho
mba %>%
  summarize(r = cor(gmat, gpa),
            rho = cor(gmat, gpa, method="spearman"))


Kendall’s tau

Kendall’s Tau is calculated as:

\(\tau =\frac{\textrm{(number of concordant pairs)}-\textrm{(number of discordant pairs)}}{\textrm{(number of concordant pairs)}+\textrm{(number of discordant pairs)}}\)


  1. A new worker is assigned to a machine that manufactures bolts. Each day, a sample of bolts is examined and the percent defective is recorded. Do the following data indicate a significant improvement over time for that worker? Calculate Kendall’s tau by first calculating the number of concordant and discordant observations below each value.



Correlations for categorical data

We’ve calculated correlation coefficients for continuous variables, but correlation coefficients can also be calculated for categorical data.

We won’t have time to investigate these in class, but they are all very easy to learn:



Correlations for nonlinear relationships

Distance correlation

The distance correlation is a modern measure of the relationship between two variables, even when the relationship is nonlinear.

Earlier in this lesson, you saw a figure with values of Pearson’s r for various scatterplots. Compare that to the following figure showing values of the distance correlation:



Recall Pearson’s r is a standardized covariance. The numerator of the covariance is the sum of the products of two distances (distance from the mean of X and distance from the mean of Y) over all points: \(\frac{\sum (x_{i}-\bar{X})(y_{i}-\bar{Y})}{n-1}\). The covariance is maximized when all data points are arranged along a straight line.

The numerator of the distance covariance is similar to that of the covariance. The difference is that the distances are between varying data points; not between a data point and the mean. The distance covariance is defined by the sum of the products of the two distances over all pairs of points. The distance covariance is maximized when the data are arranged along a straight line locally (when the data overall represent a chain in any shape).

Let’s calculate the distance correlation for an extremely small dataset. Our dataset will be:

X = 1, 2, 3

Y = 3, 1, 5

Let’s first calculate Pearson’s r and Spearman’s rho for this data:

cor(x, y, method="spearman")
[1] 0.5


Now, let’s calculate the distance covariance:

  • First, we calculate all euclidean distances between pairs of observations for X and then for Y.

\(a_{x}=\begin{bmatrix}0 &1 &2 \\ 1&0 &1 \\ 2 &1 &0 \end{bmatrix} \textrm{ and } b_{y}=\begin{bmatrix}0 &2 &2 \\ 2&0 &4 \\ 2 &4 &0 \end{bmatrix}\)

These represent the distances between observations (1 and 2), (2 and 3), and (1 and 3).

  • Next, convert this to a euclidean norm by taking each value in each matrix and (1) subtracting its row mean, (2) subtracting its column mean, and (3) adding the grand mean.

This gives us:

\(A_{x}=\begin{bmatrix} -1.11 &0.22 &0.89 \\ 0.22&-0.44 &0.22 \\ 0.89 &0.22 &-1.11\end{bmatrix} \textrm{ and } B_{y}=\begin{bmatrix} -0.89 &0.44 &0.44 \\ 0.44&-2.22 &1.78 \\ 0.44 &1.78 &-2.22\end{bmatrix}\)

  • Now, multiply each value in the A matrix by its corresponding value in the B matrix

\(AxB=\begin{bmatrix}0.9877 &0.0987 &0.3951 \\ 0.0987&0.9877 &0.3951 \\ 0.3951 &0.3951 &2.4691\end{bmatrix}\)

  • Calculate the average of all the values in this matrix

\(cov_{xy}^{2}=0.69136\)

  • Finally, take the square root of that value

\(cov_{xy}=\sqrt{0.69136}=0.8315\)


To convert that distance covariance to a distance correlation, we standardize by dividing by the product of the distance variances of X and Y.

Thankfully, we can let R do all these calculations for us:

dcor(x, y)
[1] 0.83666



Comparing correlation coefficients

Let’s look at some correlation coefficients for various simulated datasets.



Your turn

  1. The data.frame midwest contains demographic information about 437 counties in the Midwest. The list of variables is displayed below.

    (a) Calculate Pearson’s r to measure the correlation between percollege (the percent of people in the county with a college education) and ’percbelowpoverty` (the proportion of people below the poverty line).

    (b) Calculate Spearman’s rho for the same two variables.

    (c) Calculate Kendall’s tau for those two variables.

    (d) Calculate a distance correlation between those variables.

    (e) Conduct a randomization-based test of Pearson’s r and state your conclusions.

    (f) Construct a bootstrap confidence interval for Pearson’s r.

    (g) Conduct a t-test for pearson’s r.
# Load data
data(midwest)
# Display variable names and types
str(midwest)
# Display scatterplot
ggplot(data=midwest, aes(x = percollege, y = percbelowpoverty)) +
  geom_point(alpha=.5, color="steelblue", fill="lightblue", shape=20, size=5) +
  theme_grey()



  1. Calculate Kendall’s tau for the NFL malevolence data we’ve used in this lesson. Then, conduct a randomization-based test for Kendall’s tau. State your conclusion.




Publishing Your Turn solutions


License

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

LS0tCnRpdGxlOiAiQ29ycmVsYXRpb24iCm91dHB1dDoKICBodG1sX2RvY3VtZW50OgogICAgY29kZV9mb2xkaW5nOiBoaWRlCiAgICBjc3M6IGh0dHA6Ly93d3cuYnJhZHRoaWVzc2VuLmNvbS9iYXRsYWIzLmNzcwogICAgZGZfcHJpbnQ6IHRpYmJsZQogICAgZmlnX2hlaWdodDogMy45CiAgICBmaWdfd2lkdGg6IDYuMwogICAgaGlnaGxpZ2h0OiBweWdtZW50cwogICAgdGhlbWU6IHNwYWNlbGFiCiAgICB0b2M6IHllcwogICAgdG9jX2Zsb2F0OiB5ZXMKICBodG1sX25vdGVib29rOgogICAgY29kZV9mb2xkaW5nOiBoaWRlCiAgICBjc3M6IGh0dHA6Ly93d3cuYnJhZHRoaWVzc2VuLmNvbS9iYXRsYWIzLmNzcwogICAgZmlnX2hlaWdodDogMy45CiAgICBmaWdfd2lkdGg6IDYuMwogICAgaGlnaGxpZ2h0OiBweWdtZW50cwogICAgdGhlbWU6IHNwYWNlbGFiCiAgICB0b2M6IHllcwotLS0KCmBgYHtyICdnbG9iYWwgb3B0aW9ucycsIGVjaG89RkFMU0UsIG1lc3NhZ2U9RkFMU0UsIHJlc3VsdHM9J2hpZGUnfQprbml0cjo6b3B0c19jaHVuayRzZXQoCiAgY29tbWVudCA9ICIjICAgIiwKICBjb2xsYXBzZSA9IFRSVUUsCiAgZmlnLmhlaWdodCA9IDMuOSwKICBmaWcud2lkdGggPSA2LjMKKQpgYGAKCmBgYHtyICdwcmVyZXFzJywgbWVzc2FnZT1GQUxTRX0KIyBJbnN0YWxsIGFuZCBsb2FkIGFsbCBuZWNlc3NhcnkgcGFja2FnZXMKbGlzdC5vZi5wYWNrYWdlcyA8LSBjKCJ0aWR5dmVyc2UiLCAibW9zYWljIiwgImVuZXJneSIsICJnZ0V4dHJhIikKbmV3LnBhY2thZ2VzIDwtIGxpc3Qub2YucGFja2FnZXNbIShsaXN0Lm9mLnBhY2thZ2VzICVpbiUgaW5zdGFsbGVkLnBhY2thZ2VzKClbLCJQYWNrYWdlIl0pXQppZihsZW5ndGgobmV3LnBhY2thZ2VzKSkgaW5zdGFsbC5wYWNrYWdlcyhuZXcucGFja2FnZXMpCmxpYnJhcnkodGlkeXZlcnNlKQpsaWJyYXJ5KG1vc2FpYykKbGlicmFyeShlbmVyZ3kpCmxpYnJhcnkoZ2dFeHRyYSkKCmBgYAoKKioqKioKCiMgU2NlbmFyaW86ICBWaWxsYWlucyB3ZWFyIGJsYWNrCgpEbyBORkwgdGVhbXMgd2l0aCBtb3JlIG1hbGV2b2xlbnQtbG9va2luZyB1bmlmb3JtcyBwbGF5IG1vcmUgYWdncmVzc2l2ZWx5IHRoYW4gb3RoZXIgdGVhbXM/CgohW10odW5pZm9ybXMuanBnKQoKPGJyIC8+CgpbSW4gMTk4OF0oaHR0cDovL2ZhY3VsdHkuc211LmVkdS9jaHJpc2wvY291cnNlcy9wc3ljNTM1MS9hcnRpY2xlcy9ibGFja3VuaWZvcm1zLnBkZikgMjUgdm9sdW50ZWVycyB3ZXJlIHBhaWQgJDIgZWFjaCB0byByYXRlIHRoZSAqKnBlcmNlaXZlZCBtYWxldm9sZW5jZSoqIG9mIGVhY2ggTkZMIHRlYW0ncyB1bmlmb3JtIGFuZCBbbG9nb10oaHR0cDovL3d3dy5zcG9ydHNsb2dvcy5uZXQvdGVhbXMvbGlzdF9ieV9sZWFndWUvNykuICBUaGUgdG90YWwgeWFyZHMgZWFjaCB0ZWFtIHdhcyBwZW5hbGl6ZWQgdGhhdCB5ZWFyIHdlcmUgdGhlbiByZWNvcmRlZCBhcyB6LXNjb3JlcyAodG8gcmVwcmVzZW50ICoqc3RhbmRhcmRpemVkIHBlbmFsdHkgeWFyZHMqKiBmb3IgZWFjaCB0ZWFtKS4KCkhlcmUncyBhIHNhbXBsZSBvZiB0aGUgW2RhdGFdKGh0dHA6Ly93d3cuYnJhZHRoaWVzc2VuLmNvbS9odG1sNS9kYXRhL25mbG0uY3N2KToKCiFbXShkYXRhLnBuZykKCjxiciAvPgoKCgoqKioqKgoKIyMgQ292YXJpYW5jZQoKMS4gRGVzY3JpYmUgdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIHRoZXNlIHR3byB2YXJpYWJsZXMuICBBcmUgbGFyZ2VyIHZhbHVlcyBvZiAqbWFsZXZvbGVuY2UqIGFzc29jaWF0ZWQgd2l0aCBsYXJnZXIgdmFsdWVzIG9mICpwZW5hbHR5IHlhcmRzKj8gIFdoZW4gb25lIHZhcmlhYmxlIGRldmlhdGVzIGZyb20gaXRzIG1lYW4sIGRvZXMgdGhlIG90aGVyIHZhcmlhYmxlIGRldmlhdGUgZnJvbSBpdHMgbWVhbiBpbiBhIHNpbWlsYXIgd2F5PwoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoxMDBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KClJlY2FsbCB0aGUgY29uY2VwdCBvZiAqKnZhcmlhbmNlKio6ICRzX3t4fV57Mn09XGZyYWN7XHN1bSBcbGVmdCAoIHhfe2l9LVxiYXJ7WH0gXHJpZ2h0ICleezJ9fXtuLTF9PVxmcmFje1xzdW0gXGxlZnQgKCB4X3tpfS1cYmFye1h9IFxyaWdodCApXGxlZnQgKCB4X3tpfS1cYmFye1h9IFxyaWdodCApfXtuLTF9PXNfe3h4fSQKCjxiciAvPgoKV2UgY2FuIHZpc3VhbGl6ZSB0aGUgKmNvdmFyaWFuY2UqIG9mIG91ciBkYXRhIGluIGEgY291cGxlIHdheXM6CgojIyMjIyB7LnRhYnNldCAudGFic2V0LWZhZGV9CgojIyMjIyMgZGlhZ3JhbQoKPGJyIC8+CgohW10oY292LnBuZykKCjxiciAvPgoKIyMjIyMjIHNjYXR0ZXJwbG90CgpgYGB7ciAnbmZsLXNhbXBsZS1zY2F0dGVycGxvdCcsIG1lc3NhZ2U9RkFMU0UsIGZpZy5oZWlnaHQgPSAzLCBmaWcud2lkdGggPSA0fQpuZmxfc2FtcGxlIDwtIHRpYmJsZSgKICBtYWxldm9sZW5jZSA9IGMoNS4xLCA0LjY4LCA0LCAzLjksIDMuMzgsIDIuOCksCiAgcGVuYWx0eSA9IGMoMS4xOSwgMC4yOSwgLTAuNzMsIC0wLjgxLCAwLjA0LCAtMS42KSkKCmdncGxvdChuZmxfc2FtcGxlLCBhZXMoeCA9IG1hbGV2b2xlbmNlLCB5PXBlbmFsdHkpKSArCiAgZ2VvbV9wb2ludChjb2xvcj0ic3RlZWxibHVlIiwgc2l6ZT0zKQoKYGBgCgoKIyMjIyAuCgpUbyBjYWxjdWxhdGUgdGhlICoqY292YXJpYW5jZSoqLCB3ZSBjYW4gbW9kaWZ5IHRoZSBmb3JtdWxhIGZvciB2YXJpYW5jZTogIAoKJFx0ZXh0cm17Y292fV97eHl9PXNfe3h5fT1cZnJhY3tcc3VtIFxsZWZ0ICggeF97aX0tXGJhcntYfSBccmlnaHQgKVxsZWZ0ICggeV97aX0tXGJhcntZfSBccmlnaHQgKX17bi0xfT1cZnJhY3soNS4xLTMuOTc3KSgxLjE5LSAtMC4yNykrLi4uKygyLjgtMy45NzcpKC0xLjYwLSAtMC4yNyl9ezYtMX09MC42ODg5JC4KCmBgYHtyICdjYWxjdWxhdGUtY292YXJpYW5jZSd9CiMgVGhpcyBjYWxjdWxhdGVzIHRoZSB2YXJpYW5jZS1jb3ZhcmlhbmNlIG1hdHJpeApjb3YobmZsX3NhbXBsZSkKCmBgYAoKCjxiciAvPgoKMi4gVW5kZXIgd2hhdCBjb25kaXRpb25zIHdvdWxkIHRoZSBjb3ZhcmlhbmNlIGJlIG5lZ2F0aXZlPyAgcG9zaXRpdmU/ICB6ZXJvPwoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoxMDBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCgozLiBJZiB3ZSBtdWx0aXBsaWVkIGV2ZXJ5IHZhbHVlIG9mIFggYW5kIFkgYnkgMTAuICBXaGF0IHdvdWxkIGhhcHBlbiB0byB0aGUgY292YXJpYW5jZT8gIFdoYXQgYXJlIHRoZSBsb3dlc3QgYW5kIGhpZ2hlc3QgcG9zc2libGUgdmFsdWVzIGZvciBhIGNvdmFyaWFuY2U/CgpgYGB7ciAnY2FsY3VsYXRlLWNvdmFyaWFuY2Utd2l0aC1iaWdnZXItZGF0YSd9CiMgTXVsdGlwbHkgZXZlcnl0aGluZyBieSAxMAp0ZW5fdGltZXMgPC0gbmZsX3NhbXBsZSAqIDEwCmNvdih0ZW5fdGltZXMpCgpgYGAKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgpUaGF0J3Mgb25lIGxpbWl0YXRpb24gb2YgdXNpbmcgY292YXJpYW5jZSBhcyBhIG1lYXN1cmUgb2YgdGhlIHN0cmVuZ3RoIG9mIHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiB0d28gdmFyaWFibGVzOiAgKippdCBkZXBlbmRzIG9uIHRoZSBzY2FsZSBvZiBtZWFzdXJlbWVudCB1c2VkKiouICAKCkZvciBleGFtcGxlLCB0aGUgcGVuYWx0eSB5YXJkcyB2YXJpYWJsZSBpbiBvdXIgZGF0YSBhcmUgcmVwb3J0ZWQgYXMgei1zY29yZXMuICBJZiB3ZSB3ZXJlIHRvIHVzZSAodW5zdGFuZGFyZGl6ZWQpIHBlbmFsdHkgeWFyZHMsIHRoZSBjb3ZhcmlhbmNlIGluY3JlYXNlcyB0byA0LjgyMi4gIElmIHdlIHVzZWQgcGVuYWx0eSAqKmZlZXQqKiAoaW5zdGVhZCBvZiB5YXJkcyksIHRoZSBjb3ZhcmlhbmNlIHdvdWxkIGFnYWluIGluY3JlYXNlIHRvIDE0LjQ2Ny4gIFRoZXNlIGV4YW1wbGVzIHNob3cgdGhhdCBsYXJnZXIgdmFsdWVzIG9mIGNvdmFyaWFuY2UgZG8gbm90IG5lY2Vzc2FyaWx5IGluZGljYXRlIHN0cm9uZ2VyIHJlbGF0aW9uc2hpcHMgYmV0d2VlbiB2YXJpYWJsZXMuICBCZWNhdXNlIG9mIHRoaXMsIHdlIHNob3VsZG4ndCBkaXJlY3RseSBjb21wYXJlIGNvdmFyaWFuY2VzLgoKPGJyIC8+CgoqKioqKgoKIyMjIENvcnJlbGF0aW9uOiBzdGFuZGFyZGl6ZWQgY292YXJpYW5jZQoKVG8gYWRkcmVzcyB0aGlzIHNjYWxlIGRlcGVuZGVuY2UsIHdlIG5lZWQgdG8gdXNlIGEgdW5pdCBvZiBtZWFzdXJlbWVudCAqKGUuZy4sIG1ldGVycywgcG91bmRzLCBkZWNpYmVscykqIGludG8gd2hpY2ggYW55IHNjYWxlIG9mIG1lYXN1cmVtZW50ICooZS5nLiwgcGVuYWx0eSBkaXN0YW5jZSwgdW5pZm9ybSBtYWxldm9sZW5jZSkqY291bGQgYmUgY29udmVydGVkLgoKSWYgd2UgdXNlIHRoZSBzdGFuZGFyZCBkZXZpYXRpb24gYXMgdGhhdCBzdGFuZGFyZCB1bml0IG9mIG1lYXN1cmVtZW50LCB3ZSBjYW4gc3RhbmRhcmRpemUgdGhlIGNvdmFyaWFuY2UgYnk6CgoKLSBTdGFuZGFyZGl6aW5nIG91ciB2YXJpYWJsZXMgYW5kIGNhbGN1bGF0aW5nIHRoZSBjb3ZhcmlhbmNlOiAkXGZyYWN7XHN1bSBcbGVmdCAoIHpfe3hpfS1cYmFye1pfe3h9fSBccmlnaHQgKVxsZWZ0ICggel97eWl9LVxiYXJ7Wl97eX19IFxyaWdodCApfXtuLTF9PVxmcmFje1xzdW0gel97eH16X3t5fX17bi0xfT1yX3t4eX0kCgpvcgoKLSBDYWxjdWxhdGluZyB0aGUgY292YXJpYW5jZSBhbmQgc3RhbmRhcmRpemluZyBpdDogICRcZnJhY3tcc3VtIFxsZWZ0ICggeF97aX0tXGJhcntYfSBccmlnaHQgKVxsZWZ0ICggeV97aX0tXGJhcntZfSBccmlnaHQgKX17KG4tMSlzX3t4fXNfe3l9fT1yX3t4eX0kCgo8YnIgLz4KClRoaXMgaXMgY2FsbGVkIGBQZWFyc29uJ3MgcHJvZHVjdC1tb21lbnQgY29ycmVsYXRpb24gY29lZmZpY2llbnRgIChvciAqciopLgoKPGJyIC8+Cgo0LiBXaGF0IGFyZSB0aGUgc21hbGxlc3QgYW5kIGxhcmdlc3QgdmFsdWVzIG9mIHI/Cgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjEwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKKioqKioKCiMjIyBDYWxjdWxhdGluZyB0aGUgY29ycmVsYXRpb24gY29lZmZpY2llbnQKCgohW10oY29yLnBuZykKCjUuIEV4cGFuZCB0aGUgY29kZSB0byBzZWUgaG93IHRvIGNhbGN1bGF0ZSB0aGUgY29ycmVsYXRpb24gY29lZmZpY2llbnQgaW4gUi4gIEludGVycHJldCB0aGlzIGNvcnJlbGF0aW9uLgoKYGBge3IgJ2NhbGN1bGF0ZS1yJ30KIyBVc2luZyBjb3IoKQpjb3IobmZsX3NhbXBsZSRtYWxldm9sZW5jZSwgbmZsX3NhbXBsZSRwZW5hbHR5KQoKIyBVc2luZyBjb3IoKSB3aXRoaW4gZHBseXIKbmZsX3NhbXBsZSAlPiUKICBzdW1tYXJpemUoY29ycmVsYXRpb24gPSBjb3IobWFsZXZvbGVuY2UsIHBlbmFsdHkpKQoKYGBgCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0Ojc1cHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgpMZXQncyBub3cgY2FsY3VsYXRlIHRoZSBjb3JyZWxhdGlvbiBmb3Igb3VyIGZ1bGwgZGF0YXNldDoKCmBgYHtyICdmdWxsLW5mbC1kYXRhJ30KIyBMb2FkIGRhdGEKbmZsIDwtIHJlYWQuY3N2KCJodHRwOi8vd3d3LmJyYWR0aGllc3Nlbi5jb20vaHRtbDUvZGF0YS9ORkx1bmlmb3Jtcy5jc3YiKQoKIyBSZW5hbWUgdmFyaWFibGVzCm5hbWVzKG5mbCkgPC0gYygidGVhbSIsICJtYWxldm9sZW5jZSIsICJwZW5hbHR5IikKCiMgUGxvdCBhbmQgYW5ub3RhdGUgd2l0aCBjb3JyZWxhdGlvbgpnZ3Bsb3QobmZsLCBhZXMoeCA9IG1hbGV2b2xlbmNlLCB5PXBlbmFsdHkpKSArCiAgZ2VvbV9wb2ludChjb2xvcj0ic3RlZWxibHVlIiwgc2l6ZT0zKSArCiAgYW5ub3RhdGUoInRleHQiLCB4PTQuNSwgeT0tMS4yLCBsYWJlbD1wYXN0ZSgiciA9ICIsIHJvdW5kKGNvcihuZmwkbWFsZXZvbGVuY2UsIG5mbCRwZW5hbHR5KSw0KSkpCgpgYGAKCioqKioqCgojIyMgQ2hhcmFjdGVyaXN0aWNzIG9mIHRoZSBQZXJzb24gcHJvZHVjdC1tb21lbnQgY29ycmVsYXRpb24KCiMjIyMgU2NhbGUgaW52YXJpYW5jZQoKVGhlIG1hZ25pdHVkZSBvZiB0aGUgY29ycmVsYXRpb24gZG9lcyBub3QgY2hhbmdlIHVuZGVyIGFueSBsaW5lYXIgdHJhbnNmb3JtYXRpb24gb2YgdGhlIHZhcmlhYmxlcy4gIEV4cGFuZCB0aGUgY29kZSB0byBzZWUgYW4gZXhhbXBsZS4KCmBgYHtyICdzY2FsZS1pbnZhcmlhbmNlJ30KIyBMZXQncyBjb252ZXJ0IG1hbGV2b2xlbmNlIHRvIGNlbnRpLW1hbGV2b2xlbmNlCm5mbCRjZW50aV9tYWxldm9sZW5jZSA8LSBuZmwkbWFsZXZvbGVuY2UgLyAxMDAKCiMgVGhlIGNvcnJlbGF0aW9uIGRvZXMgbm90IGNoYW5nZQpjb3IobmZsJGNlbnRpX21hbGV2b2xlbmNlLCBuZmwkcGVuYWx0eSkKCmBgYAoKPGJyIC8+CgojIyMjIFN0cmVuZ3RoIG9mICpsaW5lYXIqIHJlbGF0aW9uc2hpcAoKUGVhcnNvbidzIHIgb25seSBtZWFzdXJlcyB0aGUgc3RyZW5ndGggb2YgYSAqbGluZWFyKiByZWxhdGlvbnNoaXAuICBJZiB3ZSB3YW50IHRvIG1lYXN1cmUgdGhlIHN0cmVuZ3RoIG9mIG5vbmxpbmVhciByZWxhdGlvbnNoaXBzLCB3ZSdsbCBuZWVkIGFub3RoZXIgc3RhdGlzdGljLiAgSGVyZSBhcmUgc29tZSB2YWx1ZXMgb2YgciBmb3IgdmFyaW91cyBzY2F0dGVycGxvdHM6CgohW10oc2NhdHRlcmNvci5wbmcpCgo8YnIgLz4KCiMjIyMgSW1wYWN0IG9mIG91dGxpZXJzCgo2LiBCZWxvdywgSSd2ZSBoaWdobGlnaHRlZCB0aGUgdHdvIG1vc3QgZXh0cmVtZSBvYnNlcnZhdGlvbnMgaW4gb3VyIGRhdGEuICBXZSBrbm93IHRoZSBjb3JyZWxhdGlvbiBiZXR3ZWVuIG1hbGV2b2xlbmNlIGFuZCBwZW5hbHR5IHlhcmRzIGZvciAqKmFsbCoqIHRoZSBkYXRhIGlzICRyX3t4eX09MC40MyQuICBXaGF0IHdvdWxkIGhhcHBlbiB0byB0aGUgdmFsdWUgb2YgdGhpcyBjb3JyZWxhdGlvbiBpZiB0aG9zZSB0d28gaGlnaGxpZ2h0ZWQgdmFsdWVzIHdlcmUgcmVtb3ZlZD8KCmBgYHtyICdjYWxjdWxhdGUtci13aXRoLW91dGxpZXJzLXJlbW92ZWQnfQojIEFkZCBhIHZhcmlhYmxlIHRvIGhpZ2hsaWdodCB0aGUgdHdvIG91dGxpZXJzCm5mbCRvdXRsaWVyIDwtIGMoMSwgcmVwKDAsMjYpLCAxKQoKIyBDcmVhdGUgc2NhdHRlcnBsb3Qgd2l0aCB0aGUgb3V0bGllcnMgaGlnaGxpZ2h0ZWQKZ2dwbG90KGRhdGE9bmZsLCBhZXMoeCA9IG1hbGV2b2xlbmNlLCB5ID0gcGVuYWx0eSwgY29sb3I9b3V0bGllcikpICsKICBnZW9tX3BvaW50KHNpemU9MykgKyB0aGVtZShsZWdlbmQucG9zaXRpb249Im5vbmUiKQoKIyBDYWxjdWxhdGUgY29ycmVsYXRpb24gd2l0aG91dCBoaWdobGlnaHRlZCBvYnNlcnZhdGlvbnMKbmZsICU+JQogIGZpbHRlcihvdXRsaWVyPT0wKSAlPiUKICBzdW1tYXJpemUoY29ycmVsYXRpb24gPSBjb3IobWFsZXZvbGVuY2UsIHBlbmFsdHkpKQpgYGAKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoKIyMjIyBJbXBhY3Qgb2YgcmFuZ2UgcmVzdHJpY3Rpb24KCkxldCdzIHNpbXVsYXRlIGEgbGFyZ2UgZGF0YXNldCB3aXRoIGEgc3Ryb25nLCBwb3NpdGl2ZSBjb3JyZWxhdGlvbjoKCmBgYHtyICdzaW11bGF0ZS1kYXRhJ30KIyBTaW11bGF0ZSBkYXRhCnNpbV9kYXRhIDwtIHRpYmJsZSgKICB4ID0gYygxOjEwMDApLAogIHkgPSAyKnggKyBybm9ybSgxMDAwLCAzLCAzMDApKQoKIyBQbG90CmdncGxvdChkYXRhID0gc2ltX2RhdGEsIGFlcyh4LCB5KSkgKwogIGdlb21fcG9pbnQoY29sb3I9InN0ZWVsYmx1ZSIsIHNpemU9MiwgYWxwaGE9MC41KSArCiAgYW5ub3RhdGUoInRleHQiLCB4ID0gMTI1LCB5ID0gMjIyMiwgbGFiZWw9cGFzdGUoInIgPSAiLHJvdW5kKGNvcihzaW1fZGF0YSR4LCBzaW1fZGF0YSR5KSwzKSkpICsKICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQ9OTAwLCBjb2xvcj0icmVkIikKCmBgYAoKPGJyIC8+Cgo3LiBMb29rIGF0IHRoYXQgc2FtcGxlIGRhdGEuICBTdXBwb3NlIHdlIG9ubHkgcmVzdHJpY3QgdGhlIHJhbmdlIHRvIG9ubHkgaW5jbHVkZSBkYXRhIGluIHdoaWNoICR4PjkwMCQgKHRoZSBkYXRhIHRvIHRoZSByaWdodCBvZiB0aGUgcmVkIGxpbmUpLiAgV2hhdCB3aWxsIGhhcHBlbiB0byB0aGUgbWFnbml0dWRlIG9mIHRoZSBjb3JyZWxhdGlvbj8gIFdoeT8KCmBgYHtyICdjYWxjdWxhdGUtci13aXRoLXJhbmdlLXJlc3RyaWN0aW9uJ30KIyBDYWxjdWxhdGUgY29ycmVsYXRpb24gd2hlbiBvbmx5IGNvbnNpZGVyaW5nIHg+OTAwCnNpbV9kYXRhICU+JQogIGZpbHRlcih4PjkwMCkgJT4lCiAgc3VtbWFyaXplKGNvcnJlbGF0aW9uID0gY29yKHgsIHkpKQoKYGBgCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjEwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKCiMjIyMgQ29ycmVsYXRpb24gZG9lcyBub3QgaW1wbHkgY2F1c2F0aW9uCgpFdmVyeW9uZSB0aGlua3MgdGhleSBrbm93ICpjb3JyZWxhdGlvbiBpcyBub3QgY2F1c2F0aW9uKi4gIEEgY29ycmVsYXRpb24gZG9lcyBub3QgaW1wbHkgYSBjYXVzYWwgcmVsYXRpb25zaGlwLCBhbmQgYSAqbGFjayBvZiBjb3JyZWxhdGlvbiogZG9lcyBub3QgbWVhbiB0aGVyZSBpcyAqbm8qIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIHR3byB2YXJpYWJsZXMuCgpUaGVyZSBhcmUgbWFueSBmYWN0b3JzIHRoYXQgY29udHJpYnV0ZSB0byB0aGlzLiAgV2UndmUgYWxyZWFkeSBzZWVuIHRoYXQgb3V0bGllcnMgYW5kIHJhbmdlIHJlc3RyaWN0aW9uIGNhbiBpbmZsdWVuY2UgdGhlIG1hZ25pdHVkZSAoYW5kIGV2ZW4gc2lnbikgb2YgUGVhcnNvbidzIHIuICBMaWtld2lzZSwgaWYgdHdvIHZhcmlhYmxlcyBoYXZlIGEgbm9ubGluZWFyIHJlbGF0aW9uc2hpcCwgUGVhcnNvbidzIHIgd2lsbCBtaXNyZXByZXNlbnQgdGhhdCByZWxhdGlvbnNoaXAuCgpZZXQgYW5vdGhlciByZWFzb24gaXMgdGhlICoqdGhpcmQtdmFyaWFibGUgcHJvYmxlbSoqLgoKOC4gVGhlIG51bWJlciBvZiBoZWFydCBhdHRhY2tzIGluIGEgZ2l2ZW4gbW9udGggaXMgcG9zaXRpdmVseSBjb3JyZWxhdGVkIHdpdGggaWNlIGNyZWFtIHNhbGVzLiAgRXhwbGFpbiB3aHkgdGhpcyBkb2VzIG5vdCBwcm92aWRlIGV2aWRlbmNlIHRoYXQgaWNlIGNyZWFtIGNhdXNlcyBoZWFydCBhdHRhY2tzLgoKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoxMDBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCjxkaXYgaWQ9ImxpY2Vuc2UiIHN0eWxlPSJmb250LXNpemU6MTRweDsiPgoKQSBbMjAwNCBhcnRpY2xlXShodHRwOi8vaWplLm94Zm9yZGpvdXJuYWxzLm9yZy9jb250ZW50LzMzLzMvNDY0LmxvbmcpIHF1ZXN0aW9uZWQgcmVzZWFyY2ggaW50byB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gaG9ybW9uZSByZXBsYWNlbWVudCB0aGVyYXB5IChIUlQpIGFuZCBjb3JvbmFyeSBoZWFydCBkaXNlYXNlIChDSEQpLiAgTnVtZXJvdXMgc3R1ZGllcyBoYWQgc2hvd24gbmVnYXRpdmUgY29ycmVsYXRpb25zIGJldHdlZW4gSFJUIGFuZCBDSEQsIGxlYWRpbmcgZG9jdG9ycyB0byBwcm9wb3NlIHRoYXQgSFJUIHdhcyBwcm90ZWN0aXZlIGFnYWluc3QgQ0hELgoKUmFuZG9taXplZCBjb250cm9sbGVkIHRyaWFscyBzaG93ZWQsIGluIGZhY3QsIHRoYXQgSFJUIGNhdXNlZCBhIHNtYWxsIGJ1dCBzaWduaWZpY2FudCAqKmluY3JlYXNlKiogaW4gdGhlIHJpc2sgb2YgQ0hELgoKUmUtYW5hbHlzaXMgb2YgdGhlIGRhdGEgZnJvbSB0aGUgb3JpZ2luYWwgc3R1ZGllcyBzaG93ZWQgdGhhdCB3b21lbiB1bmRlcnRha2luZyBIUlQgd2VyZSBtb3JlIGxpa2VseSB0byBiZSBmcm9tICpoaWdoZXIgc29jaW9lY29ub21pYyBncm91cHMqLiAgVGhlc2Ugd29tZW4sIHRoZXJlZm9yZSwgd2VyZSBtb3JlIGxpa2VseSB0byBoYXZlIGJldHRlci10aGFuLWF2ZXJhZ2UgZGlldHMgYW5kIGV4ZXJjaXNlIHJlZ2ltZW5zLgoKVGhlICoqdGhpcmQgdmFyaWFibGUqKiBsaW5raW5nIGhvcm1vbmUgcmVwbGFjZW1lbnQgdGhlcmFweSB0byBjb3JvbmFyeSBoZWFydCBkaXNlYXNlIHdhcyAqKnNvY2lvZWNvbm9taWMgc3RhdHVzKiouCgo8L2Rpdj4KCjxiciAvPgoKCjkuIEluIDIwMDIsIGEgbGV0dGVyIGluIERpYWJldGVzIENhcmUgZGV0YWlsZWQgdGhlIGNhbGN1bGF0aW9uIGJlaGluZCB0aGUgY29ycmVsYXRpb24gb2YgKipyID0gMC41NCoqIGJldHdlZW4gKipkaWFiZXRlcyByYXRlcyoqIGFuZCAqKnBvbGx1dGlvbiBsZXZlbHMqKiBhY3Jvc3MgYWxsIDUwIHN0YXRlcy48YnIgLz48YnIgLz5Ib3cgd291bGQgeW91IGludGVycHJldCB0aGlzIGNvcnJlbGF0aW9uPzxiciAvPGJyIC8+VGhlIGF1dGhvciB0cmllZCB0byBiZSBjYXJlZnVsIGluIGNvbmNsdWRpbmcsICoiLi4uIHRoZSBjb3JyZWxhdGlvbiBiZXR3ZWVuIGFpciBlbWlzc2lvbnMgYW5kIHRoZSBwcmV2YWxlbmNlIG9mIGRpYWJldGVzIGRvZXMgbm90IHByb3ZlIGEgY2F1c2UtYW5kLWVmZmVjdCByZWxhdGlvbnNoaXA7IHRoZSBzaWduaWZpY2FuY2Ugb2YgdGhlIHJlbGF0aW9uc2hpcCBkZW1hbmRzIGF0dGVudGlvbi4iKi4gIDxiciAvPjxiciAvPkluIHJlc3BvbnNlLCBNYXJrIE5pY29saWNoIHF1ZXN0aW9uZWQgaWYgdGhlIHJlbGF0aW9uc2hpcCBldmVuIGRlbWFuZGVkIGF0dGVudGlvbi4gIFVzaW5nIHRoZSBzYW1lIGRpYWJldGVzIGRhdGEsIE5pY29saWNoIGNhbGN1bGF0ZWQgdGhlIGNvcnJlbGF0aW9uIGJldHdlZW4gKipkaWFiZXRlcyByYXRlcyoqIGFuZCB0aGUgKiphbHBoYWJldGl6ZWQgcmFuayBvZiBlYWNoIHN0YXRlKiogdG8gYmUgKipyID0gMC40OSoqLiAgSGUgYWxzbyBmb3VuZCB0aGUgY29ycmVsYXRpb24gYmV0d2VlbiAqKmRpYWJldGVzIHJhdGVzKiogYW5kIHRoZSAqKmxhdGl0dWRlIG9mIGVhY2ggc3RhdGUncyBjYXBpdGFsKiogdG8gYmUgKipyID0gLTAuNTQqKi4KCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjEwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKMTAuIFNlZSBpZiB5b3UgY2FuIHRoaW5rIG9mIGEgc2NlbmFyaW8gd2hpY2ggd291bGQgcmVzdWx0IGluIHRoZSBmb2xsb3dpbmcgc2NhdHRlcnBsb3QuICBXaGF0IGlzIHRvIGJlIGxlYXJuZWQgZnJvbSB0aGlzPwoKCmBgYHtyICdzaW11bGF0ZWQtdGVzdC1zY29yZXMnfQp0ZXN0IDwtIHRpYmJsZSgKICB4ID0gYyhybm9ybSgxMDAsIDYsIDIpLCBybm9ybSgxMDAsIDEwLCAyKSwgcm5vcm0oMTAwLCAxNCwgMikpLAogIHkgPSBjKHJub3JtKDEwMCwgNiwgMiksIHJub3JtKDEwMCwgMTAsIDIpLCBybm9ybSgxMDAsIDE0LCAyKSksCiAgZ3JvdXAgPSBhcy5mYWN0b3IoYyhyZXAoMSwxMDApLCByZXAoMiwxMDApLCByZXAoMywxMDApKSkKKQoKIyBDYWxjdWxhdGUgb3ZlcmFsbCBjb3JyZWxhdGlvbgpvdmVyYWxsIDwtIHRlc3QgJT4lCiAgc3VtbWFyaXplKHIgPSBjb3IoeCwgeSkpCgojIENhbGN1bGF0ZSBjb3JyZWxhdGlvbiBmb3IgZWFjaCBzdWJncm91cApzdWJzIDwtIHRlc3QgJT4lCiAgZ3JvdXBfYnkoZ3JvdXApICU+JQogIHN1bW1hcml6ZShyID0gY29yKHgsIHkpKQoKIyBQbG90IGFuZCBhbm5vdGF0ZSB3aXRoIGNvcnJlbGF0aW9ucwpnZ3Bsb3QoZGF0YSA9IHRlc3QsIGFlcyh4LCB5LCBjb2xvciA9IGdyb3VwKSkgKwogIGdlb21fcG9pbnQoc2l6ZSA9IDIsIGFscGhhPTAuNykgKyAKICBnZW9tX3Ntb290aChtZXRob2Q9ImxtIiwgc2U9RkFMU0UpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uPSJub25lIikgKwogIGFubm90YXRlKCJ0ZXh0IiwgeCA9IDUsIHkgPSAxOCwgbGFiZWw9cGFzdGUoIm92ZXJhbGwgciA9Iiwgcm91bmQob3ZlcmFsbCwgMikpLCBjb2xvcj0iYmxhY2siKSArCiAgYW5ub3RhdGUoInRleHQiLCB4ID0gMiwgeSA9IDQsICBsYWJlbCA9IHBhc3RlKCJyID0iLCByb3VuZChzdWJzWzEsMl0sIDIpKSwgY29sb3I9InJlZCIpICsKICBhbm5vdGF0ZSgidGV4dCIsIHggPSAxNiwgeSA9IDksIGxhYmVsID0gcGFzdGUoInIgPSIsIHJvdW5kKHN1YnNbMiwyXSwgMikpLCBjb2xvcj0iZm9yZXN0Z3JlZW4iKSArCiAgYW5ub3RhdGUoInRleHQiLCB4ID0gMTguOCwgeSA9IDE2LCBsYWJlbCA9IHBhc3RlKCJyID0gIiwgcm91bmQoc3Vic1szLDJdLCAyKSksIGNvbG9yID0gImJsdWUiKQpgYGAKCjxkaXYgc3R5bGU9IndpZHRoOjM1MHB4OyBoZWlnaHQ6MTAwcHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+Cgo8ZGl2IGlkPSJsaWNlbnNlIiBzdHlsZT0iZm9udC1zaXplOjE0cHg7Ij4KSWYgeW91J2QgbGlrZSB0byBleHBsb3JlIG90aGVyIGV4YW1wbGVzIGluIHdoaWNoIGNvcnJlbGF0aW9uIGlzIG1pc3Rha2VuIGZvciBjYXVzYXRpb24sIGNoZWNrIG91dDoKCltTcHVyaW91cyBjb3JyZWxhdGlvbnNdKGh0dHA6Ly93d3cudHlsZXJ2aWdlbi5jb20vc3B1cmlvdXMtY29ycmVsYXRpb25zKQoKW0NvcnJlbGF0aW9uIG9yIENhdXNhdGlvbl0oaHR0cDovL2pmbXVlbGxlci5mYWN1bHR5Lm5vY3RybC5lZHUvMTAwL2NvcnJlbGF0aW9uX29yX2NhdXNhdGlvbi5odG0pCgo8L2Rpdj4KCjxiciAvPgoKKioqKioKCiMjIE5vbnplcm8gY29ycmVsYXRpb25zPwoKMTEuIFdoYXQgY29ycmVsYXRpb24gd291bGQgeW91IGV4cGVjdCB0byBmaW5kIGJldHdlZW4gdHdvIHNldHMgb2YgcmFuZG9tIHZhbHVlcz8gIEV4cGFuZCB0aGUgY29kZSB0byBzZWUgLS0+CgpgYGB7ciAncmFuZG9tLWNvcnJlbGF0aW9ucyd9CnNldC5zZWVkKDMxNDEpCiMgR2VuZXJhdGUgcmFuZG9tIHZhbHVlcyBmb3IgWCBhbmQgWQpyYW5kb21feHkgPC0gdGliYmxlKAogIHggPSBybm9ybSgxMDAsIDAsIDEpLAogIHkgPSBybm9ybSgxMDAsIDAsIDEpCikgCgpjb3IocmFuZG9tX3h5JHgsIHJhbmRvbV94eSR5KQoKZ2dwbG90KGRhdGEgPSByYW5kb21feHksIGFlcyh4LCB5KSkgKwogIGdlb21fcG9pbnQoc2l6ZT0zLCBjb2xvcj0ic3RlZWxibHVlIikgKwogICAgYW5ub3RhdGUoInRleHQiLCB4ID0gMCwgeSA9IDIuOCwgbGFiZWw9cGFzdGUoIm92ZXJhbGwgciA9Iiwgcm91bmQoY29yKHJhbmRvbV94eSR4LCByYW5kb21feHkkeSksIDMpKSwgY29sb3I9ImJsYWNrIikKICAKYGBgCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjEwMHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKVGhlIGNvcnJlbGF0aW9uIG9mIGFueSB0d28gdmFyaWFibGVzIGluIGFueSBzYW1wbGUgb2YgZGF0YSB3aWxsIGJlIG5vbi16ZXJvLCBzbyBob3cgY2FuIHdlIGhhdmUgYW55IGNvbmZpZGVuY2UgdGhhdCBhIGNvcnJlbGF0aW9uIGlzICJyZWFsPyIKCjxiciAvPgoKIyMjIFJhbmRvbWl6YXRpb24tYmFzZWQgdGVzdAoKMTIuIEEgc3Vic2V0IG9mIG91ciBkYXRhIGlzIGRpc3BsYXllZCBiZWxvdy4gIFdoYXQgd291bGQgdGhlIChudWxsKSByYW5kb21pemF0aW9uIGh5cG90aGVzaXMgYmUgaW4gdGhpcyBzY2VuYXJpbz8gIFVuZGVyIHRoYXQgaHlwb3RoZXNpcywgaG93IGxpa2VseSB3ZXJlIHRoZSBSYWlkZXJzICh3aXRoIGEgNS4xIG1hbGV2b2xlbmNlIHJhdGluZykgdG8gaGF2ZSAxLjE5LCAwLjQ4LCBvciAwLjI3IHN0YW5kYXJkaXplZCBwZW5hbHR5IHlhcmRzPzxiciAvPjxiciAvPkV4cGxhaW4gaG93IHdpbGwgd2UgdXNlIHJhbmRvbWl6YXRpb24tYmFzZWQgbWV0aG9kcyB0byB0ZXN0IG91ciByYW5kb21pemF0aW9uIGh5cG90aGVzaXMuCgohW10oc2FtcGxlLnBuZykKPGRpdiBzdHlsZT0id2lkdGg6MzUwcHg7IGhlaWdodDoxMDBweCI+IVtdKFRyYW5zcGFyZW50LmdpZik8L2Rpdj4KCkxldCdzIHRha2UgYSBsb29rIGF0IHNvbWUgcmFuZG9taXphdGlvbnMgb2Ygb3VyIGRhdGE6CgpgYGB7ciAnZnVuY3Rpb24tdG8tcGxvdC1tdWx0aXBsZS1wbG90cycsIGVjaG89RkFMU0V9CiMgTXVsdGlwbGUgcGxvdCBmdW5jdGlvbgptdWx0aXBsb3QgPC0gZnVuY3Rpb24oLi4uLCBwbG90bGlzdD1OVUxMLCBmaWxlLCBjb2xzPTEsIGxheW91dD1OVUxMKSB7CiAgbGlicmFyeShncmlkKQoKICAjIE1ha2UgYSBsaXN0IGZyb20gdGhlIC4uLiBhcmd1bWVudHMgYW5kIHBsb3RsaXN0CiAgcGxvdHMgPC0gYyhsaXN0KC4uLiksIHBsb3RsaXN0KQoKICBudW1QbG90cyA9IGxlbmd0aChwbG90cykKCiAgIyBJZiBsYXlvdXQgaXMgTlVMTCwgdGhlbiB1c2UgJ2NvbHMnIHRvIGRldGVybWluZSBsYXlvdXQKICBpZiAoaXMubnVsbChsYXlvdXQpKSB7CiAgICAjIE1ha2UgdGhlIHBhbmVsCiAgICAjIG5jb2w6IE51bWJlciBvZiBjb2x1bW5zIG9mIHBsb3RzCiAgICAjIG5yb3c6IE51bWJlciBvZiByb3dzIG5lZWRlZCwgY2FsY3VsYXRlZCBmcm9tICMgb2YgY29scwogICAgbGF5b3V0IDwtIG1hdHJpeChzZXEoMSwgY29scyAqIGNlaWxpbmcobnVtUGxvdHMvY29scykpLAogICAgICAgICAgICAgICAgICAgIG5jb2wgPSBjb2xzLCBucm93ID0gY2VpbGluZyhudW1QbG90cy9jb2xzKSkKICB9CgogaWYgKG51bVBsb3RzPT0xKSB7CiAgICBwcmludChwbG90c1tbMV1dKQoKICB9IGVsc2UgewogICAgIyBTZXQgdXAgdGhlIHBhZ2UKICAgIGdyaWQubmV3cGFnZSgpCiAgICBwdXNoVmlld3BvcnQodmlld3BvcnQobGF5b3V0ID0gZ3JpZC5sYXlvdXQobnJvdyhsYXlvdXQpLCBuY29sKGxheW91dCkpKSkKCiAgICAjIE1ha2UgZWFjaCBwbG90LCBpbiB0aGUgY29ycmVjdCBsb2NhdGlvbgogICAgZm9yIChpIGluIDE6bnVtUGxvdHMpIHsKICAgICAgIyBHZXQgdGhlIGksaiBtYXRyaXggcG9zaXRpb25zIG9mIHRoZSByZWdpb25zIHRoYXQgY29udGFpbiB0aGlzIHN1YnBsb3QKICAgICAgbWF0Y2hpZHggPC0gYXMuZGF0YS5mcmFtZSh3aGljaChsYXlvdXQgPT0gaSwgYXJyLmluZCA9IFRSVUUpKQoKICAgICAgcHJpbnQocGxvdHNbW2ldXSwgdnAgPSB2aWV3cG9ydChsYXlvdXQucG9zLnJvdyA9IG1hdGNoaWR4JHJvdywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsYXlvdXQucG9zLmNvbCA9IG1hdGNoaWR4JGNvbCkpCiAgICB9CiAgfQp9CmBgYAoKCmBgYHtyICdyYW5kb21pemVkLWNvcnJlbGF0aW9ucyd9CiMgU3RvcmUgb3VyIHNhbXBsZSBjb3JyZWxhdGlvbiBhcyB0ZXN0X3N0YXQKdGVzdF9zdGF0IDwtIGNvcihuZmwkbWFsZXZvbGVuY2UsIG5mbCRwZW5hbHR5KQoKIyBSYW5kb21pemUgb3VyIGRhdGEgCnAxIDwtIG5mbCAlPiUKICBnZ3Bsb3QoLiwgYWVzKHggPSBtYWxldm9sZW5jZSwgeSA9IHBlbmFsdHkpKSArCiAgZ2VvbV9wb2ludChjb2xvcj0ic3RlZWxibHVlIiwgc2l6ZT0yKSArCiAgZ2VvbV9zbW9vdGgobWV0aG9kPSJsbSIsIHNlPUZBTFNFLCBjb2xvcj0icmVkIiwgc2l6ZT0uMikgKwogIGFubm90YXRlKCJ0ZXh0IiwgeCA9IDQsIHkgPSAxLCBsYWJlbCA9ICJhY3R1YWwgZGF0YSIsIGNvbG9yPSJyZWQiLCBzaXplPTMpCgpwMiA8LSBuZmwgJT4lCiAgbXV0YXRlKHNodWZmbGVkX3BlbmFsdHkgPSBzYW1wbGUocGVuYWx0eSkpICU+JQogIGdncGxvdCguLCBhZXMoeCA9IG1hbGV2b2xlbmNlLCB5ID0gc2h1ZmZsZWRfcGVuYWx0eSkpICsKICBnZW9tX3BvaW50KGNvbG9yPSJzdGVlbGJsdWUiLCBzaXplPTIpICsKICBnZW9tX3Ntb290aChtZXRob2Q9ImxtIiwgc2U9RkFMU0UsIGNvbG9yPSJyZWQiLCBzaXplPS4yKSArCiAgYW5ub3RhdGUoInRleHQiLCB4ID0gNCwgeSA9IDEsIGxhYmVsID0gInJhbmRvbWl6YXRpb24gIzEiLCBjb2xvcj0icmVkIiwgc2l6ZT0zKQoKcDMgPC0gbmZsICU+JQogIG11dGF0ZShzaHVmZmxlZF9wZW5hbHR5ID0gc2FtcGxlKHBlbmFsdHkpKSAlPiUKICBnZ3Bsb3QoLiwgYWVzKHggPSBtYWxldm9sZW5jZSwgeSA9IHNodWZmbGVkX3BlbmFsdHkpKSArCiAgZ2VvbV9wb2ludChjb2xvcj0ic3RlZWxibHVlIiwgc2l6ZT0yKSArCiAgZ2VvbV9zbW9vdGgobWV0aG9kPSJsbSIsIHNlPUZBTFNFLCBjb2xvcj0icmVkIiwgc2l6ZT0uMikgKwogIGFubm90YXRlKCJ0ZXh0IiwgeCA9IDQsIHkgPSAxLCBsYWJlbCA9ICJyYW5kb21pemF0aW9uICMyIiwgY29sb3I9InJlZCIsIHNpemU9MykKCnA0IDwtIG5mbCAlPiUKICBtdXRhdGUoc2h1ZmZsZWRfcGVuYWx0eSA9IHNhbXBsZShwZW5hbHR5KSkgJT4lCiAgZ2dwbG90KC4sIGFlcyh4ID0gbWFsZXZvbGVuY2UsIHkgPSBzaHVmZmxlZF9wZW5hbHR5KSkgKwogIGdlb21fcG9pbnQoY29sb3I9InN0ZWVsYmx1ZSIsIHNpemU9MikgKwogIGdlb21fc21vb3RoKG1ldGhvZD0ibG0iLCBzZT1GQUxTRSwgY29sb3I9InJlZCIsIHNpemU9LjIpICsKICBhbm5vdGF0ZSgidGV4dCIsIHggPSA0LCB5ID0gMSwgbGFiZWwgPSAicmFuZG9taXphdGlvbiAjMyIsIGNvbG9yPSJyZWQiLCBzaXplPTMpCgpwNSA8LSBuZmwgJT4lCiAgbXV0YXRlKHNodWZmbGVkX3BlbmFsdHkgPSBzYW1wbGUocGVuYWx0eSkpICU+JQogIGdncGxvdCguLCBhZXMoeCA9IG1hbGV2b2xlbmNlLCB5ID0gc2h1ZmZsZWRfcGVuYWx0eSkpICsKICBnZW9tX3BvaW50KGNvbG9yPSJzdGVlbGJsdWUiLCBzaXplPTIpICsKICBnZW9tX3Ntb290aChtZXRob2Q9ImxtIiwgc2U9RkFMU0UsIGNvbG9yPSJyZWQiLCBzaXplPS4yKSArCiAgYW5ub3RhdGUoInRleHQiLCB4ID0gNCwgeSA9IDEsIGxhYmVsID0gInJhbmRvbWl6YXRpb24gIzQiLCBjb2xvcj0icmVkIiwgc2l6ZT0zKQoKcDYgPC0gbmZsICU+JQogIG11dGF0ZShzaHVmZmxlZF9wZW5hbHR5ID0gc2FtcGxlKHBlbmFsdHkpKSAlPiUKICBnZ3Bsb3QoLiwgYWVzKHggPSBtYWxldm9sZW5jZSwgeSA9IHNodWZmbGVkX3BlbmFsdHkpKSArCiAgZ2VvbV9wb2ludChjb2xvcj0ic3RlZWxibHVlIiwgc2l6ZT0yKSArCiAgZ2VvbV9zbW9vdGgobWV0aG9kPSJsbSIsIHNlPUZBTFNFLCBjb2xvcj0icmVkIiwgc2l6ZT0uMikgKwogIGFubm90YXRlKCJ0ZXh0IiwgeCA9IDQsIHkgPSAxLCBsYWJlbCA9ICJyYW5kb21pemF0aW9uICM1IiwgY29sb3I9InJlZCIsIHNpemU9MykKCiMgVGhpcyB1c2VzIGEgIm11bHRpcGxvdCIgZnVuY3Rpb24gdGhhdCB3YXMgbG9hZGVkIHNpbGVudGx5Cm11bHRpcGxvdChwMSwgcDQsIHAyLCBwNSwgcDMsIHA2LCBjb2xzPTMpCgpgYGAKCjxiciAvPgoKRWFjaCByYW5kb21pemF0aW9uIG9mIGRhdGEgeWllbGRzIGEgbm9uLXplcm8gY29ycmVsYXRpb24uICBMZXQncyBydW4gMTAsMDAwIHJhbmRvbWl6YXRpb25zIGFuZCBwbG90IGFsbCAxMCwwMDAgY29ycmVsYXRpb25zLgoKYGBge3IgJ3JhbmRvbWl6YXRpb24tY29ycmVsYXRpb24nfQojIFVzaW5nIGNvcigpCnNodWZmbGVkX3IgPC0gRG8oMTAwMDApICogY29yKG1hbGV2b2xlbmNlIH4gc2h1ZmZsZShwZW5hbHR5KSwgZGF0YT1uZmwpCgojIFVzaW5nIGRwbHlyCiMgc2h1ZmZsZWRfcnMgPC0gRG8oMTAwMDApICogbmZsICU+JQojICAgbXV0YXRlKHNodWZmbGVkID0gc2FtcGxlKHBlbmFsdHkpKSAlPiUKIyAgIHN1bW1hcml6ZShjb3IgPSBjb3Ioc2h1ZmZsZWQsIG5mbCRtYWxldm9sZW5jZSkpCgojIENhbGN1bGF0ZSBsaWtlbGlob29kIG9mIHRlc3Qgc3RhdGlzdGljCnB2YWx1ZSA8LSBwcm9wKHNodWZmbGVkX3IkY29yID49IHRlc3Rfc3RhdCkKcHZhbHVlCgojIFBsb3Qgc2ltcGxlIGhpc3RvZ3JhbQojIGhpc3RvZ3JhbSh+Y29yLCBkYXRhPXNodWZmbGVkX3IsCiMgICAgICAgICAgIHdpZHRoPS4wNCwKIyAgICAgICAgICAgeGxhYj0iUG9zc2libGUgY29ycmVsYXRpb25zIGFzc3VtaW5nIG51bGwgaHlwb3RoZXNpcyBpcyB0cnVlIikKIyBsYWRkKHBhbmVsLmFibGluZSh2PXRlc3Rfc3RhdCkpICAgIyBBZGQgdmVydGljYWwgbGluZSBhdCB0ZXN0IHN0YXRpc3RpYwoKIyBQbG90CmdncGxvdChkYXRhID0gc2h1ZmZsZWRfciwgYWVzKHggPSBjb3IpKSArCiAgZ2VvbV9oaXN0b2dyYW0oYmlud2lkdGggPSAwLjA0LCBmaWxsPSJsaWdodGJsdWUiLCBjb2xvcj0id2hpdGUiLCBhbHBoYT0wLjgpICsKICBhbm5vdGF0ZSgic2VnbWVudCIsIHggPSB0ZXN0X3N0YXQsIHhlbmQgPSB0ZXN0X3N0YXQsIHkgPSAwLCB5ZW5kID0gNjAwLCBjb2xvciA9ICJyZWQiKSArCiAgYW5ub3RhdGUoInRleHQiLCB4ID0gdGVzdF9zdGF0LCB5ID0gNjMwLCBsYWJlbCA9ICJvYnNlcnZlZCByID0gMC40MyIsIGNvbG9yID0gInJlZCIpICsKICBhbm5vdGF0ZSgidGV4dCIsIHggPSB0ZXN0X3N0YXQrLjExLCB5ID0gMTcwLCBsYWJlbCA9IHBhc3RlKCJwID0iLHJvdW5kKHB2YWx1ZSw0KSksIGNvbG9yID0gInJlZCIpICsKICBsYWJzKAogICAgICB0aXRsZSA9ICJSYW5kb21pemVkIENvcnJlbGF0aW9ucyIsCiAgICAgIHggPSAiciIKICAgICAgKSArCiAgc2NhbGVfeF9jb250aW51b3VzKGxpbWl0cyA9IGMoLS43LC43KSwgYnJlYWtzPXNlcSgtLjYsIC42LCAuMyksIG1pbm9yX2JyZWFrcz1OVUxMKSArCiAgICB0aGVtZSgKICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMSwgY29sb3I9ImdyZXkxMCIpLAogICAgbGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiLAogICAgcGFuZWwuZ3JpZC5tYWpvci55ID0gZWxlbWVudF9saW5lKGNvbG91ciA9ICJ3aGl0ZSIsIHNpemU9LjI1KSwKICAgIHBhbmVsLmdyaWQubWFqb3IueCA9IGVsZW1lbnRfbGluZShjb2xvdXIgPSAid2hpdGUiLCBzaXplPS4xNSksCiAgICBwYW5lbC5ncmlkLm1pbm9yID0gZWxlbWVudF9ibGFuaygpLAogICAgcGFuZWwuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsID0gImdyZXk5MyIpCiAgKQoKYGBgCgo8YnIgLz4KCiMjIyBCb290c3RyYXAgY29uZmlkZW5jZSBpbnRlcnZhbAoKMTMuIEludGVycHJldCB0aGUgZm9sbG93aW5nIGNvbmZpZGVuY2UgaW50ZXJ2YWwgYW5kIGV4cGxhaW4gaG93IGl0IHdhcyBjb25zdHJ1Y3RlZC4KCmBgYHtyICdib290c3RyYXAtY2knfQpib290IDwtIERvKDEwMDAwKSAqIGNvciggbWFsZXZvbGVuY2UgfiBwZW5hbHR5LCBkYXRhPXJlc2FtcGxlKG5mbCkgKQoKYm9vdHN0cmFwQ0kgPC0gY29uZmludChib290LCBsZXZlbCA9IDAuOTUsIG1ldGhvZCA9ICJxdWFudGlsZSIpCmxvd2VyIDwtIGFzLm51bWVyaWMoYm9vdHN0cmFwQ0lbMl0pICMgU3RvcmUgbG93ZXIgQ0kgYm91bmQKdXBwZXIgPC0gYXMubnVtZXJpYyhib290c3RyYXBDSVszXSkgIyBTdG9yZSB1cHBlciBDSSBib3VuZAoKIyBEZW5zaXR5IHBsb3QKZ2dwbG90KGRhdGEgPSBib290LCBhZXMoeCA9IGNvcikpICsKICBnZW9tX2RlbnNpdHkoZmlsbD0ibGlnaHRibHVlIiwgY29sb3I9IndoaXRlIiwgYWxwaGEgPSAwLjgpICsKICBsYWJzKAogICAgICB0aXRsZSA9ICJCb290c3RyYXAgZGlzdHJpYnV0aW9uIG9mIGNvcnJlbGF0aW9ucyIsCiAgICAgIHggPSAiYm9vdHN0cmFwIGNvcnJlbGF0aW9ucyIKICAgICAgKSArCiAgc2NhbGVfeF9jb250aW51b3VzKGJyZWFrcz1zZXEoLTAuOCwgMC44LCAwLjIpLCBtaW5vcl9icmVha3M9TlVMTCkgKwogICAgdGhlbWUoCiAgICBheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTEsIGNvbG9yPSJncmV5MTAiKSwKICAgIGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIiwKICAgIHBhbmVsLmdyaWQubWFqb3IueSA9IGVsZW1lbnRfbGluZShjb2xvdXIgPSAid2hpdGUiKSwKICAgIHBhbmVsLmdyaWQubWFqb3IueCA9IGVsZW1lbnRfbGluZShjb2xvdXIgPSAid2hpdGUiLCBzaXplPS4xNSksCiAgICBwYW5lbC5ncmlkLm1pbm9yID0gZWxlbWVudF9ibGFuaygpLAogICAgcGFuZWwuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsID0gImdyZXk5MyIpCiAgKSArCiAgYW5ub3RhdGUoInRleHQiLCB4ID0gbG93ZXIsIHkgPSAuMiwgbGFiZWwgPSByb3VuZChsb3dlciwzKSkgKwogIGFubm90YXRlKCJ0ZXh0IiwgeCA9IHVwcGVyLCB5ID0gLjIsIGxhYmVsID0gcm91bmQodXBwZXIsMykpICsKICBhbm5vdGF0ZSgidGV4dCIsIHggPSBtZWRpYW4oYm9vdCRjb3IpLCB5ID0gLjI1LCBsYWJlbCA9IHBhc3RlKCI5NSUiKSkgKwogIGFubm90YXRlKCJzZWdtZW50IiwgeCA9IGxvd2VyLCB4ZW5kID0gdXBwZXIsIHkgPSAwLjEsIHllbmQgPSAuMSwgY29sb3IgPSAicmVkIikKCmBgYAoKCgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0OjE1MHB4Ij4hW10oVHJhbnNwYXJlbnQuZ2lmKTwvZGl2PgoKIyMjIHQtdGVzdCBmb3IgY29ycmVsYXRpb24KCklmIHdlIGFzc3VtZSBvdXIgZGF0YSBhcmUgc2FtcGxlZCBmcm9tIHBvcHVsYXRpb25zIHdpdGggbm9ybWFsIGRpc3RyaWJ1dGlvbnMsIHdlIGNhbiB1c2UgYSB0LXRlc3QuCgpJbiB0aGUgbmV4dCBsZXNzb24sIHdlJ2xsIGRlcml2ZSB0aGUgZm9sbG93aW5nIFt0ZXN0IHN0YXRpc3RpYyBmb3IgdGhlIGNvcnJlbGF0aW9uIGNvZWZmaWNpZW50XShodHRwczovL29ubGluZWNvdXJzZXMuc2NpZW5jZS5wc3UuZWR1L3N0YXQ0MTQvbm9kZS8yNTQpOgoKJHRfe24tMn09XGZyYWN7cl97eHl9LTB9e1x0ZXh0cm17U0V9X3tyX3t4eX19fT1cZnJhY3tyX3t4eX19e1xzcXJ0e1xmcmFjezEtcl97eHl9XnsyfX17bi0yfX19PVxmcmFje3Jfe3h5fVxzcXJ0e24tMn19e1xzcXJ0ezEtcl97eHl9XnsyfX19PTIuNDI3XCBcIChwPS4wMTEpJAoKPGJyIC8+CgpXZSBjYW4gcnVuIHRoaXMgdGVzdCBpbiBSIHVzaW5nIHRoZSBgY29yLnRlc3QoKWAgZnVuY3Rpb246CgpgYGB7cn0KIyB0LXRlc3QgZm9yIGNvcnJlbGF0aW9uIGNvZWZmaWNpZW50CmNvci50ZXN0KG5mbCRtYWxldm9sZW5jZSwgbmZsJHBlbmFsdHksIGFsdGVybmF0aXZlID0gYygiZ3JlYXRlciIpLCBtZXRob2Q9YygicGVhcnNvbiIpKQoKYGBgCgo8YnIgLz4KCjE0LiBJcyB0aGVyZSBhbnl0aGluZyAod29ydGh3aGlsZSkgd2UgY2FuIGNvbmNsdWRlIGZyb20gdGhpcyB0LXRlc3Q/Cgo8ZGl2IHN0eWxlPSJ3aWR0aDozNTBweDsgaGVpZ2h0Ojc1cHgiPiFbXShUcmFuc3BhcmVudC5naWYpPC9kaXY+CgoqKioqKgoKIyMgT3RoZXIgdHlwZXMgb2YgY29ycmVsYXRpb25zCgojIyMgTm9ucGFyYW1ldHJpYyBjb3JyZWxhdGlvbnMgKGNvcnJlbGF0aW9ucyBvZiByYW5rcykKCiMjIyMgU3BlYXJtYW4ncyByaG8KCioqU3BlYXJtYW4ncyByaG8qKiB1c2VzIHRoZSBzYW1lIGZvcm11bGEgYXMgUGVhcnNvbidzIHIsIGV4Y2VwdCB0aGUgZGF0YSBhcmUgKipjb252ZXJ0ZWQgdG8gcmFua3MqKi4KCjxiciAvPgoKMTUuIFVzaW5nIGRhdGEgZnJvbSA4IE1CQSBncmFkdWF0ZXMsIGNhbGN1bGF0ZSBTcGVhcm1hbidzIHJobyB0byBlc3RpbWF0ZSB0aGUgc3RyZW5ndGggb2YgdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIHNjb3JlcyBvbiB0aGUgR01BVCAoYSB0ZXN0IHRoZXkgdG9vayBwcmlvciB0byBlbnRlcmluZyBncmFkdWF0ZSBzY2hvb2wpIGFuZCB0aGVpciBncmFkZSBwb2ludCBhdmVyYWdlIGluIHRoZSBNQkEgcHJvZ3JhbS4gIFRvIGRvIHRoaXMsIGZpcnN0IGNvbnZlcnQgdGhlIHNjb3JlcyBpbnRvIHJhbmtzLgoKPGJyIC8+CgohW10oc3BlYXJtYW4ucG5nKQoKPGJyIC8+CgpgYGB7cn0KIyBFbnRlciB0aGUgcmF3IGRhdGEKbWJhIDwtIHRpYmJsZSgKICBnbWF0ID0gYyg3MTAsIDYxMCwgNjQwLCA1ODAsIDU0NSwgNTYwLCA2MTAsIDUzMCksCiAgZ3BhICA9IGMoNCwgNCwgMy45LCAzLjgsIDMuNywgMy42LCAzLjUsIDMuNSkKCikKCiMgQ2FsY3VsYXRlIFBlYXJzb24ncyByIGFuZCBTcGVhcm1hbidzIHJobwptYmEgJT4lCiAgc3VtbWFyaXplKHIgPSBjb3IoZ21hdCwgZ3BhKSwKICAgICAgICAgICAgcmhvID0gY29yKGdtYXQsIGdwYSwgbWV0aG9kPSJzcGVhcm1hbiIpKQpgYGAKCjxiciAvPgoKIyMjIyBLZW5kYWxsJ3MgdGF1CgoqKktlbmRhbGwncyBUYXUqKiBpcyBjYWxjdWxhdGVkIGFzOgoKJFx0YXUgPVxmcmFje1x0ZXh0cm17KG51bWJlciBvZiBjb25jb3JkYW50IHBhaXJzKX0tXHRleHRybXsobnVtYmVyIG9mIGRpc2NvcmRhbnQgcGFpcnMpfX17XHRleHRybXsobnVtYmVyIG9mIGNvbmNvcmRhbnQgcGFpcnMpfStcdGV4dHJteyhudW1iZXIgb2YgZGlzY29yZGFudCBwYWlycyl9fSQKCjxiciAvPgoKMTQuIEEgbmV3IHdvcmtlciBpcyBhc3NpZ25lZCB0byBhIG1hY2hpbmUgdGhhdCBtYW51ZmFjdHVyZXMgYm9sdHMuICBFYWNoIGRheSwgYSBzYW1wbGUgb2YgYm9sdHMgaXMgZXhhbWluZWQgYW5kIHRoZSBwZXJjZW50IGRlZmVjdGl2ZSBpcyByZWNvcmRlZC4gIERvIHRoZSBmb2xsb3dpbmcgZGF0YSBpbmRpY2F0ZSBhIHNpZ25pZmljYW50IGltcHJvdmVtZW50IG92ZXIgdGltZSBmb3IgdGhhdCB3b3JrZXI/ICBDYWxjdWxhdGUgS2VuZGFsbCdzIHRhdSBieSBmaXJzdCBjYWxjdWxhdGluZyB0aGUgbnVtYmVyIG9mIGNvbmNvcmRhbnQgYW5kIGRpc2NvcmRhbnQgb2JzZXJ2YXRpb25zIGJlbG93IGVhY2ggdmFsdWUuCgo8YnIgLz4KCiFbXShrZW5kYWxsLnBuZykKCjxiciAvPgoKIyMjIENvcnJlbGF0aW9ucyBmb3IgY2F0ZWdvcmljYWwgZGF0YQoKV2UndmUgY2FsY3VsYXRlZCBjb3JyZWxhdGlvbiBjb2VmZmljaWVudHMgZm9yIGNvbnRpbnVvdXMgdmFyaWFibGVzLCBidXQgY29ycmVsYXRpb24gY29lZmZpY2llbnRzIGNhbiBhbHNvIGJlIGNhbGN1bGF0ZWQgZm9yIGNhdGVnb3JpY2FsIGRhdGEuICAKCldlIHdvbid0IGhhdmUgdGltZSB0byBpbnZlc3RpZ2F0ZSB0aGVzZSBpbiBjbGFzcywgYnV0IHRoZXkgYXJlIGFsbCB2ZXJ5IGVhc3kgdG8gbGVhcm46CgotIFtQaGkgY29lZmZpY2llbnRdKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL1BoaV9jb2VmZmljaWVudCkgZm9yIDJ4MiBjb250aW5nZW5jeSB0YWJsZXMKLSBbQ3JhbWVyJ3MgVl0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvQ3JhbcODwqlyJTI3c19WKSBmb3IgdHdvIG5vbWluYWwgdmFyaWFibGVzCi0gW0Jpc2VyaWFsIGNvcnJlbGF0aW9uXShodHRwOi8vd3d3LmFuZHJld3MuZWR1L35jYWxraW5zL21hdGgvZWRybTYxMS9lZHJtMTMuaHRtI0JJUykgZm9yIG9uZSBjb250aW51b3VzIGFuZCBvbmUgYXJ0aWZpY2lhbGx5IGRpY2hvdG9tb3VzIChvcmRpbmFsKSB2YXJpYWJsZQotIFtQb2ludC1iaXNlcmlhbCBjb3JyZWxhdGlvbl0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvUG9pbnQtYmlzZXJpYWxfY29ycmVsYXRpb25fY29lZmZpY2llbnQpIGZvciBvbmUgY29udGludW91cyBhbmQgb25lIChhcnRpZmljaWFsIG9yIG5hdHVyYWwpIGRpY2hvdG9tb3VzIHZhcmlhYmxlCi0gW1BvbHljaG9yaWMgY29ycmVsYXRpb25dKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL1BvbHljaG9yaWNfY29ycmVsYXRpb24pIGZvciB0d28gb3JkaW5hbCB2YXJpYWJsZXMgd2l0aCB1bmRlcmx5aW5nIGNvbnRpbnVvdXMgZGlzdHJpYnV0aW9ucwoKPGJyIC8+CjxiciAvPgoKIyMjIENvcnJlbGF0aW9ucyBmb3Igbm9ubGluZWFyIHJlbGF0aW9uc2hpcHMKCiMjIyMgRGlzdGFuY2UgY29ycmVsYXRpb24KClRoZSBbZGlzdGFuY2UgY29ycmVsYXRpb25dKGh0dHA6Ly9wZXJzb25hbC5iZ3N1LmVkdS9+bXJpenpvL2VuZXJneS9BT0FTMzEyLnBkZikgaXMgYSBtb2Rlcm4gbWVhc3VyZSBvZiB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gdHdvIHZhcmlhYmxlcywgZXZlbiB3aGVuIHRoZSByZWxhdGlvbnNoaXAgaXMgKm5vbmxpbmVhciouICAKCkVhcmxpZXIgaW4gdGhpcyBsZXNzb24sIHlvdSBzYXcgYSBmaWd1cmUgd2l0aCB2YWx1ZXMgb2YgUGVhcnNvbidzIHIgZm9yIHZhcmlvdXMgc2NhdHRlcnBsb3RzLiAgQ29tcGFyZSB0aGF0IHRvIHRoZSBmb2xsb3dpbmcgZmlndXJlIHNob3dpbmcgdmFsdWVzIG9mIHRoZSBkaXN0YW5jZSBjb3JyZWxhdGlvbjoKCjxiciAvPgoKIVtdKGRpc3RhbmNlLnBuZykKCjxiciAvPgoKUmVjYWxsIFBlYXJzb24ncyByIGlzIGEgc3RhbmRhcmRpemVkICpjb3ZhcmlhbmNlKi4gIFRoZSBudW1lcmF0b3Igb2YgdGhlIGNvdmFyaWFuY2UgaXMgdGhlIHN1bSBvZiB0aGUgcHJvZHVjdHMgb2YgdHdvIGRpc3RhbmNlcyAoZGlzdGFuY2UgZnJvbSB0aGUgbWVhbiBvZiBYIGFuZCBkaXN0YW5jZSBmcm9tIHRoZSBtZWFuIG9mIFkpIG92ZXIgYWxsIHBvaW50czogICRcZnJhY3tcc3VtICh4X3tpfS1cYmFye1h9KSh5X3tpfS1cYmFye1l9KX17bi0xfSQuICBUaGUgY292YXJpYW5jZSBpcyBtYXhpbWl6ZWQgd2hlbiBhbGwgZGF0YSBwb2ludHMgYXJlIGFycmFuZ2VkIGFsb25nIGEgc3RyYWlnaHQgbGluZS4KClRoZSBudW1lcmF0b3Igb2YgdGhlICoqZGlzdGFuY2UgY292YXJpYW5jZSoqIGlzIHNpbWlsYXIgdG8gdGhhdCBvZiB0aGUgY292YXJpYW5jZS4gIFRoZSBkaWZmZXJlbmNlIGlzIHRoYXQgdGhlIGRpc3RhbmNlcyBhcmUgYmV0d2VlbiB2YXJ5aW5nICpkYXRhIHBvaW50cyo7IG5vdCBiZXR3ZWVuICphIGRhdGEgcG9pbnQgYW5kIHRoZSBtZWFuKi4gIFRoZSBkaXN0YW5jZSBjb3ZhcmlhbmNlIGlzIGRlZmluZWQgYnkgdGhlIHN1bSBvZiB0aGUgcHJvZHVjdHMgb2YgdGhlIHR3byBkaXN0YW5jZXMgb3ZlciBhbGwgcGFpcnMgb2YgcG9pbnRzLiAgVGhlIGRpc3RhbmNlIGNvdmFyaWFuY2UgaXMgbWF4aW1pemVkIHdoZW4gdGhlIGRhdGEgYXJlIGFycmFuZ2VkIGFsb25nIGEgc3RyYWlnaHQgbGluZSAqbG9jYWxseSogKHdoZW4gdGhlIGRhdGEgb3ZlcmFsbCByZXByZXNlbnQgYSBjaGFpbiBpbiBhbnkgc2hhcGUpLgoKTGV0J3MgY2FsY3VsYXRlIHRoZSBkaXN0YW5jZSBjb3JyZWxhdGlvbiBmb3IgYW4gZXh0cmVtZWx5IHNtYWxsIGRhdGFzZXQuICBPdXIgZGF0YXNldCB3aWxsIGJlOgoKYFggPSAxLCAyLCAzYAoKYFkgPSAzLCAxLCA1YAoKTGV0J3MgZmlyc3QgY2FsY3VsYXRlIFBlYXJzb24ncyByIGFuZCBTcGVhcm1hbidzIHJobyBmb3IgdGhpcyBkYXRhOgoKYGBge3J9CiMgSW5wdXQgZGF0YQp4IDwtIGMoMSwgMiwgMykKeSA8LSBjKDMsIDEsIDUpCgojIFBlYXJzb24ncyByCmNvcih4LCB5LCBtZXRob2Q9InBlYXJzb24iKQoKIyBTcGVhcm1hbidzIHJobwpjb3IoeCwgeSwgbWV0aG9kPSJzcGVhcm1hbiIpCmBgYAoKCjxiciAvPgoKTm93LCBsZXQncyBjYWxjdWxhdGUgdGhlIGRpc3RhbmNlIGNvdmFyaWFuY2U6CgotIEZpcnN0LCB3ZSBjYWxjdWxhdGUgYWxsIGV1Y2xpZGVhbiBkaXN0YW5jZXMgYmV0d2VlbiBwYWlycyBvZiBvYnNlcnZhdGlvbnMgZm9yIFggYW5kIHRoZW4gZm9yIFkuCgokYV97eH09XGJlZ2lue2JtYXRyaXh9MCAmMSAgJjIgXFwgIDEmMCAgJjEgXFwgMiAmMSAgJjAgXGVuZHtibWF0cml4fSBcdGV4dHJteyBhbmQgIH0gYl97eX09XGJlZ2lue2JtYXRyaXh9MCAmMiAgJjIgXFwgMiYwICAmNCBcXCAyICY0ICAmMCBcZW5ke2JtYXRyaXh9JAoKVGhlc2UgcmVwcmVzZW50IHRoZSBkaXN0YW5jZXMgYmV0d2VlbiBvYnNlcnZhdGlvbnMgKDEgYW5kIDIpLCAoMiBhbmQgMyksIGFuZCAoMSBhbmQgMykuCgotIE5leHQsIGNvbnZlcnQgdGhpcyB0byBhICpldWNsaWRlYW4gbm9ybSogYnkgdGFraW5nIGVhY2ggdmFsdWUgaW4gZWFjaCBtYXRyaXggYW5kICgxKSBzdWJ0cmFjdGluZyBpdHMgcm93IG1lYW4sICgyKSBzdWJ0cmFjdGluZyBpdHMgY29sdW1uIG1lYW4sIGFuZCAoMykgYWRkaW5nIHRoZSBncmFuZCBtZWFuLgoKVGhpcyBnaXZlcyB1czoKCiRBX3t4fT1cYmVnaW57Ym1hdHJpeH0gLTEuMTEgJjAuMjIgJjAuODkgXFwgIDAuMjImLTAuNDQgICYwLjIyIFxcIDAuODkgJjAuMjIgICYtMS4xMVxlbmR7Ym1hdHJpeH0gXHRleHRybXsgYW5kICB9IEJfe3l9PVxiZWdpbntibWF0cml4fSAtMC44OSAmMC40NCAgJjAuNDQgXFwgMC40NCYtMi4yMiAgJjEuNzggXFwgMC40NCAmMS43OCAgJi0yLjIyXGVuZHtibWF0cml4fSQKCi0gTm93LCBtdWx0aXBseSBlYWNoIHZhbHVlIGluIHRoZSBBIG1hdHJpeCBieSBpdHMgY29ycmVzcG9uZGluZyB2YWx1ZSBpbiB0aGUgQiBtYXRyaXgKCiRBeEI9XGJlZ2lue2JtYXRyaXh9MC45ODc3ICYwLjA5ODcgJjAuMzk1MSBcXCAgMC4wOTg3JjAuOTg3NyAmMC4zOTUxIFxcIDAuMzk1MSAmMC4zOTUxICAmMi40NjkxXGVuZHtibWF0cml4fSQKCi0gQ2FsY3VsYXRlIHRoZSBhdmVyYWdlIG9mIGFsbCB0aGUgdmFsdWVzIGluIHRoaXMgbWF0cml4CgokY292X3t4eX1eezJ9PTAuNjkxMzYkCgotIEZpbmFsbHksIHRha2UgdGhlIHNxdWFyZSByb290IG9mIHRoYXQgdmFsdWUKCiRjb3Zfe3h5fT1cc3FydHswLjY5MTM2fT0wLjgzMTUkCgo8YnIgLz4KClRvIGNvbnZlcnQgdGhhdCBkaXN0YW5jZSAqY292YXJpYW5jZSogdG8gYSBkaXN0YW5jZSAqY29ycmVsYXRpb24qLCB3ZSBzdGFuZGFyZGl6ZSBieSBkaXZpZGluZyBieSB0aGUgcHJvZHVjdCBvZiB0aGUgKmRpc3RhbmNlIHZhcmlhbmNlcyogb2YgWCBhbmQgWS4KClRoYW5rZnVsbHksIHdlIGNhbiBsZXQgUiBkbyBhbGwgdGhlc2UgY2FsY3VsYXRpb25zIGZvciB1czoKCmBgYHtyfQojIFdlIGxvYWRlZCB0aGUgZW5lcmd5IHBhY2thZ2UgZWFybGllcgojIGxpYnJhcnkoZW5lcmd5KQoKIyBDYWxjdWxhdGUgYWxsIHRoZSBjb21wb25lbnRzIG9mIGEgZGlzdGFuY2UgY29ycmVsYXRpb24KIyBWZXJpZnkgdGhlIGNvdmFyaWFuY2UgaXMgMC44MzE1CkRDT1IoeCwgeSkKCiMgSGVyZSdzIGEgZGlyZWN0IHdheSB0byBjYWxjdWxhdGUgdGhlIGRpc3RhbmNlIGNvcnJlbGF0aW9uCmRjb3IoeCwgeSkKYGBgCgo8YnIgLz4KCioqKioqCgojIyBDb21wYXJpbmcgY29ycmVsYXRpb24gY29lZmZpY2llbnRzCgpMZXQncyBsb29rIGF0IHNvbWUgY29ycmVsYXRpb24gY29lZmZpY2llbnRzIGZvciB2YXJpb3VzIHNpbXVsYXRlZCBkYXRhc2V0cy4KCmBgYHtyLCBlY2hvPUZBTFNFLCBtZXNzYWdlPUZBTFNFLCB3YXJuaW5nPUZBTFNFfQpsaW5lYXJkYXRhIDwtIHRpYmJsZSgKICB4ID0gYygxOjEwMDApLAogIHkgPSAyKnggKyAzICsgcm5vcm0oMTAwMCwwLDMwMCkKKQoKcDEgPC0gZ2dwbG90KGRhdGE9bGluZWFyZGF0YSwgYWVzKHggPSB4LCB5ID0geSkpICsKICBnZW9tX3BvaW50KGFscGhhPS4zLCBjb2xvcj0ic3RlZWxibHVlIiwgZmlsbD0ibGlnaHRibHVlIiwgc2hhcGU9MjAsIHNpemU9NSkgKwogIHRoZW1lX2dyZXkoKSArIGdndGl0bGUocGFzdGUoInIgPSIsIHJvdW5kKGNvcihsaW5lYXJkYXRhJHgsIGxpbmVhcmRhdGEkeSksMiksICI7IHJobyA9Iiwgcm91bmQoY29yKGxpbmVhcmRhdGEkeCwgbGluZWFyZGF0YSR5LCBtZXRob2Q9InNwZWFybWFuIiksMiksICI7IEQgPSIsIHJvdW5kKGRjb3IobGluZWFyZGF0YSR4LCBsaW5lYXJkYXRhJHksIGluZGV4PTEuMCksMikpKQoKbGluZWFyZGF0YSA8LSB0aWJibGUoCiAgeCA9IGMoMToxMDAwKSwKICB5ID0gLTIqeCArIDMgKyBybm9ybSgxMDAwLDAsMzAwMCkKKQoKcDIgPC0gZ2dwbG90KGRhdGE9bGluZWFyZGF0YSwgYWVzKHggPSB4LCB5ID0geSkpICsKICBnZW9tX3BvaW50KGFscGhhPS4zLCBjb2xvcj0ic3RlZWxibHVlIiwgZmlsbD0ibGlnaHRibHVlIiwgc2hhcGU9MjAsIHNpemU9NSkgKwogIHRoZW1lX2dyZXkoKSArIGdndGl0bGUocGFzdGUoInIgPSIsIHJvdW5kKGNvcihsaW5lYXJkYXRhJHgsIGxpbmVhcmRhdGEkeSksMiksICI7IHJobyA9Iiwgcm91bmQoY29yKGxpbmVhcmRhdGEkeCwgbGluZWFyZGF0YSR5LCBtZXRob2Q9InNwZWFybWFuIiksMiksICI7IEQgPSIsIHJvdW5kKGRjb3IobGluZWFyZGF0YSR4LCBsaW5lYXJkYXRhJHksIGluZGV4PTEuMCksMikpKQoKbGluZWFyZGF0YSA8LSB0aWJibGUoCiAgeCA9IGMoLTUwMDo1MDApLAogIHkgPSAyKnheMiArIDMgKyBybm9ybSgxMDAxLDAsNTAwMDApCikKCnAzIDwtIGdncGxvdChkYXRhPWxpbmVhcmRhdGEsIGFlcyh4ID0geCwgeSA9IHkpKSArCiAgZ2VvbV9wb2ludChhbHBoYT0uMywgY29sb3I9InN0ZWVsYmx1ZSIsIGZpbGw9ImxpZ2h0Ymx1ZSIsIHNoYXBlPTIwLCBzaXplPTUpICsKICB0aGVtZV9ncmV5KCkgKyBnZ3RpdGxlKHBhc3RlKCJyID0iLCByb3VuZChjb3IobGluZWFyZGF0YSR4LCBsaW5lYXJkYXRhJHkpLDIpLCAiOyByaG8gPSIsIHJvdW5kKGNvcihsaW5lYXJkYXRhJHgsIGxpbmVhcmRhdGEkeSwgbWV0aG9kPSJzcGVhcm1hbiIpLDIpLCAiOyBEID0iLCByb3VuZChkY29yKGxpbmVhcmRhdGEkeCwgbGluZWFyZGF0YSR5LCBpbmRleD0xLjApLDIpKSkKCmxpbmVhcmRhdGEgPC0gdGliYmxlKAogIHggPSBjKCgwOjUwMCkvMjApLAogIHkgPSBzaW4oeCkrcm5vcm0oNTAwLDAsLjEpCikKCnA0IDwtIGdncGxvdChkYXRhPWxpbmVhcmRhdGEsIGFlcyh4ID0geCwgeSA9IHkpKSArCiAgZ2VvbV9wb2ludChhbHBoYT0uMywgY29sb3I9InN0ZWVsYmx1ZSIsIGZpbGw9ImxpZ2h0Ymx1ZSIsIHNoYXBlPTIwLCBzaXplPTUpICsKICB0aGVtZV9ncmV5KCkgKyBnZ3RpdGxlKHBhc3RlKCJyID0iLCByb3VuZChjb3IobGluZWFyZGF0YSR4LCBsaW5lYXJkYXRhJHkpLDIpLCAiOyByaG8gPSIsIHJvdW5kKGNvcihsaW5lYXJkYXRhJHgsIGxpbmVhcmRhdGEkeSwgbWV0aG9kPSJzcGVhcm1hbiIpLDIpLCAiOyBEID0iLCByb3VuZChkY29yKGxpbmVhcmRhdGEkeCwgbGluZWFyZGF0YSR5LCBpbmRleD0xLjApLDIpKSkKCnggPC0gYXMubnVtZXJpYyhjKDE6MTAwMCkpCnogPC0gc2luKHgtNTAwKQp5IDwtIGNvcyh4LTUwMCkKbGluZWFyZGF0YSA8LSBkYXRhLmZyYW1lKHg9eix5PXkpCnA1IDwtIGdncGxvdChkYXRhPWxpbmVhcmRhdGEsIGFlcyh4ID0geCwgeSA9IHkpKSArCiAgZ2VvbV9wb2ludChhbHBoYT0uMywgY29sb3I9InN0ZWVsYmx1ZSIsIGZpbGw9ImxpZ2h0Ymx1ZSIsIHNoYXBlPTIwLCBzaXplPTUpICsKICB0aGVtZV9ncmV5KCkgKyBnZ3RpdGxlKHBhc3RlKCJyID0iLCByb3VuZChjb3IobGluZWFyZGF0YSR4LCBsaW5lYXJkYXRhJHkpLDIpLCAiOyByaG8gPSIsIHJvdW5kKGNvcihsaW5lYXJkYXRhJHgsIGxpbmVhcmRhdGEkeSwgbWV0aG9kPSJzcGVhcm1hbiIpLDIpLCAiOyBEID0iLCByb3VuZChkY29yKGxpbmVhcmRhdGEkeCwgbGluZWFyZGF0YSR5LCBpbmRleD0xLjApLDIpKSkKCnAxCnAyCnAzCnA0CnA1CgpgYGAKCjxiciAvPgoKKioqKioKCiMgWW91ciB0dXJuCgoxOS4gVGhlIGRhdGEuZnJhbWUgYG1pZHdlc3RgIGNvbnRhaW5zIGRlbW9ncmFwaGljIGluZm9ybWF0aW9uIGFib3V0IDQzNyBjb3VudGllcyBpbiB0aGUgTWlkd2VzdC4gIFRoZSBsaXN0IG9mIHZhcmlhYmxlcyBpcyBkaXNwbGF5ZWQgYmVsb3cuPGJyIC8+PGJyIC8+KiooYSkqKiBDYWxjdWxhdGUgUGVhcnNvbidzIHIgdG8gbWVhc3VyZSB0aGUgY29ycmVsYXRpb24gYmV0d2VlbiBgcGVyY29sbGVnZWAgKHRoZSBwZXJjZW50IG9mIHBlb3BsZSBpbiB0aGUgY291bnR5IHdpdGggYSBjb2xsZWdlIGVkdWNhdGlvbikgYW5kICdwZXJjYmVsb3dwb3ZlcnR5YCAodGhlIHByb3BvcnRpb24gb2YgcGVvcGxlIGJlbG93IHRoZSBwb3ZlcnR5IGxpbmUpLjxiciAvPjxiciAvPioqKGIpKiogQ2FsY3VsYXRlIFNwZWFybWFuJ3MgcmhvIGZvciB0aGUgc2FtZSB0d28gdmFyaWFibGVzLjxiciAvPjxiciAvPioqKGMpKiogQ2FsY3VsYXRlIEtlbmRhbGwncyB0YXUgZm9yIHRob3NlIHR3byB2YXJpYWJsZXMuPGJyIC8+PGJyIC8+KiooZCkqKiBDYWxjdWxhdGUgYSBkaXN0YW5jZSBjb3JyZWxhdGlvbiBiZXR3ZWVuIHRob3NlIHZhcmlhYmxlcy48YnIgLz48YnIgLz4qKihlKSoqIENvbmR1Y3QgYSByYW5kb21pemF0aW9uLWJhc2VkIHRlc3Qgb2YgUGVhcnNvbidzIHIgYW5kIHN0YXRlIHlvdXIgY29uY2x1c2lvbnMuPGJyIC8+PGJyIC8+KiooZikqKiBDb25zdHJ1Y3QgYSBib290c3RyYXAgY29uZmlkZW5jZSBpbnRlcnZhbCBmb3IgUGVhcnNvbidzIHIuPGJyIC8+PGJyIC8+KiooZykqKiBDb25kdWN0IGEgdC10ZXN0IGZvciBwZWFyc29uJ3Mgci4KCmBgYHtyfQojIExvYWQgZGF0YQpkYXRhKG1pZHdlc3QpCiMgRGlzcGxheSB2YXJpYWJsZSBuYW1lcyBhbmQgdHlwZXMKc3RyKG1pZHdlc3QpCiMgRGlzcGxheSBzY2F0dGVycGxvdApnZ3Bsb3QoZGF0YT1taWR3ZXN0LCBhZXMoeCA9IHBlcmNvbGxlZ2UsIHkgPSBwZXJjYmVsb3dwb3ZlcnR5KSkgKwogIGdlb21fcG9pbnQoYWxwaGE9LjUsIGNvbG9yPSJzdGVlbGJsdWUiLCBmaWxsPSJsaWdodGJsdWUiLCBzaGFwZT0yMCwgc2l6ZT01KSArCiAgdGhlbWVfZ3JleSgpCmBgYAoKCjxiciAvPgo8YnIgLz4KCjIwLiBDYWxjdWxhdGUgS2VuZGFsbCdzIHRhdSBmb3IgdGhlIE5GTCBtYWxldm9sZW5jZSBkYXRhIHdlJ3ZlIHVzZWQgaW4gdGhpcyBsZXNzb24uICBUaGVuLCBjb25kdWN0IGEgcmFuZG9taXphdGlvbi1iYXNlZCB0ZXN0IGZvciBLZW5kYWxsJ3MgdGF1LiAgU3RhdGUgeW91ciBjb25jbHVzaW9uLgoKPGJyIC8+CjxiciAvPgoKKioqKioKCiMgUHVibGlzaGluZyBZb3VyIFR1cm4gc29sdXRpb25zCgotIERvd25sb2FkIHRoZSBbWW91ciBUdXJuIFNvbHV0aW9ucyB0ZW1wbGF0ZV0oaHR0cDovL3d3dy5icmFkdGhpZXNzZW4uY29tL2h0bWw1L3N0YXRzL20zMDEveW91cnR1cm45LlJtZCkKICAgICAtIFJTdHVkaW8gd2lsbCBhdXRvbWF0aWNhbGx5IG9wZW4gaXQgaWYgaXQgZG93bmxvYWRzIGFzIGEgKi5SbWQqIGZpbGUuCiAgICAgLSBJZiBpdCBkb3dubG9hZHMgYXMgYSAqLnR4dCogZmlsZSwgdGhlbiB5b3UgY2FuOgogICAgICAgIC0gb3BlbiB0aGUgZmlsZSBpbiBhIHRleHQgZWRpdG9yIGFuZCBjb3B5IGl0cyBjb250ZW50cwogICAgICAgIC0gb3BlbiBSU3R1ZGlvIGFuZCBjcmVhdGUgYSBuZXcgKipSIE1hcmtkb3duKiogZmlsZSAoaHRtbCBmaWxlKQogICAgICAgIC0gZGVsZXRlIGV2ZXJ5dGhpbmcgYW5kIHBhc3RlIHRoZSBjb250ZW50cyBvZiB0aGUgdGVtcGxhdGUgCgotIFR5cGUgeW91ciBzb2x1dGlvbnMgaW50byB0aGUgW1lvdXIgVHVybiBTb2x1dGlvbnMgdGVtcGxhdGVdKGh0dHA6Ly93d3cuYnJhZHRoaWVzc2VuLmNvbS9odG1sNS9zdGF0cy9tMzAxL3lvdXJ0dXJuOS5SbWQpCiAgICAgLSBJJ3ZlIHRyaWVkIHRvIHNob3cgeW91ICp3aGVyZSogdG8gdHlwZSBlYWNoIG9mIHlvdXIgc29sdXRpb25zL2Fuc3dlcnMKICAgICAtIFlvdSBjYW4gcnVuIHRoZSBjb2RlIGFzIHlvdSB0eXBlIGl0LgoKLSBXaGVuIHlvdSd2ZSBhbnN3ZXJlZCBldmVyeSBxdWVzdGlvbiwgY2xpY2sgdGhlICoqS25pdCBIVE1MKiogYnV0dG9uIGxvY2F0ZWQgYXQgdGhlIHRvcCBvZiBSU3R1ZGlvOiA8aW1nIHNyYz0iaHR0cDovL2JyYWR0aGllc3Nlbi5jb20vaHRtbDUvc3RhdHMvbTMwMC9rbml0ci5wbmciIGFsdD0iRHJhd2luZyIvPgogICAgIC0gUlN0dWRpbyB3aWxsIGJlZ2luIHdvcmtpbmcgdG8gY3JlYXRlIGEgLmh0bWwgZmlsZSBvZiB5b3VyIHNvbHV0aW9ucy4KICAgICAtIEl0IG1heSB0YWtlIGEgZmV3IG1pbnV0ZXMgdG8gY29tcGlsZSBldmVyeXRoaW5nIChpZiB5b3UndmUgY29uZHVjdGVkIGFueSBzaW11bGF0aW9ucykKICAgICAtIFdoaWxlIHRoZSBmaWxlIGlzIGNvbXBpbGluZywgeW91IG1heSBzZWUgc29tZSByZWQgdGV4dCBpbiB0aGUgY29uc29sZS4gIAogICAgIC0gV2hlbiB0aGUgZmlsZSBpcyBmaW5pc2hlZCwgaXQgd2lsbCBvcGVuIGF1dG9tYXRpY2FsbHkgKG9yIGl0IHdpbGwgZ2l2ZSB5b3UgYW4gZXJyb3IgbWVzc2FnZSkuCiAgICAgICAgLSBJZiB0aGUgZmlsZSBkb2VzIG5vdCBvcGVuIGF1dG9tYXRpY2FsbHksIHlvdSdsbCBzZWUgYW4gZXJyb3IgbWVzc2FnZSBpbiB0aGUgY29uc29sZS4KCi0gT25jZSB5b3UncmUgc2F0aXNmaWVkIHdpdGggeW91ciBzb2x1dGlvbnMsIGVtYWlsIHRoYXQgaHRtbCBmaWxlIHRvIFt0aGllc3NlbmJyYWRsZXlhQHNhdS5lZHVdKG1haWx0bzp0aGllc3NlbmJyYWRsZXlhQHNhdS5lZHUpIG9yIGdpdmUgbWUgYSBwcmludGVkIGNvcHkuCgotIElmIHlvdSBydW4gaW50byBhbnkgcHJvYmxlbXMsIFtlbWFpbCBtZV0obWFpbHRvOnRoaWVzc2VuYnJhZGxleWFAc2F1LmVkdSkgb3Igd29yayB3aXRoIG90aGVyIHN0dWRlbnRzIGluIHRoZSBjbGFzcy4KCjxiciAvPgoKPGRpdiBpZD0ibGljZW5zZSI+CioqTGljZW5zZSoqCgpUaGlzIGRvY3VtZW50IGlzIHJlbGVhc2VkIHVuZGVyIGEgW0NyZWF0aXZlIENvbW1vbnMgQXR0cmlidXRpb24tU2hhcmVBbGlrZSAzLjAgVW5wb3J0ZWRdKGh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL2xpY2Vuc2VzL2J5LXNhLzMuMCkgbGljZW5zZS4KPC9kaXY+Cg==