Introduction

This is our first lab when we are considering 2 dimensions and instead of calculating univariate statistics by groups (or factors) of other variable - we will measure their common relationships based on co-variance and correlation coefficients.

*Please be very careful when choosing the measure of correlation! In case of different measurument scales we have to recode one of the variables into weaker scale.

It would be nice to add some additional plots in the background. Feel free to add your own sections and use external packages.

Data

This time we are going to use a typical credit scoring data with predefined “default” variables and personal demografic and income data. Please take a look closer at headers and descriptions of each variable.

Scatterplots

First let’s visualize our quantitative relationships using scatterplots.

## [1] 0.574346
##    estimate      p.value statistic   n gp  Method
## 1 0.2775388 4.816032e-28  10.97912 700  1 kendall

You can also normalize the skewed distribution of incomes using log:

We can add an estimated linear regression line:

Scatterplots by groups

We can finally see if there any differences between risk status:

## `geom_smooth()` using formula = 'y ~ x'

We can also see more closely if there any differences between those two distributions adding their estimated density plots:

Scatterplots with density curves

We can also see more closely if there any differences between those two distributions adding their estimated density plots:

Correlation coefficients - Pearson’s linear correlation

Ok, let’s move to some calculations. In R, we can use the cor() function. It takes three arguments and the method: cor(x, y, method) For 2 quantitative data, with all assumptions met, we can calculate simple Pearson’s coefficient of linear correlation:

## [1] 0.574346

Ok, what about the percentage of the explained variability?

## [1] "Percentage of explained variability: 32.9873365718318 %"

So as we can see almost ??? of total log of incomes’ variability is explained by differences in age. The rest (???) is probably explained by other factors.

Partial and semipartial correlation

The partial and semi-partial (also known as part) correlations are used to express the specific portion of variance explained by eliminating the effect of other variables when assessing the correlation between two variables.

Partial correlation holds constant one variable when computing the relations to others. Suppose we want to know the correlation between X and Y holding Z constant for both X and Y. That would be the partial correlation between X and Y controlling for Z.

Semipartial correlation holds Z constant for either X or Y, but not both, so if we wanted to control X for Z, we could compute the semipartial correlation between X and Y holding Z constant for X.

Suppose we want to know the correlation between the log of income and age controlling for years of employment. How highly correlated are these after controlling for tenure?

**There can be more than one control variable.

##    estimate      p.value statistic   n gp  Method
## 1 0.3194263 4.805085e-18  8.899323 700  1 pearson
##    estimate      p.value statistic   n gp  Method
## 1 0.2203711 3.899134e-09  5.964597 700  1 pearson

How can we interpret the obtained partial correlation coefficient? What is the difference between that one and the semi-partial coefficient:

##    estimate      p.value statistic   n gp  Method
## 1 0.3194263 4.805085e-18  8.899323 700  1 pearson
##    estimate      p.value statistic   n gp  Method
## 1 0.2203711 3.899134e-09  5.964597 700  1 pearson

Rank correlation

For 2 different scales - like for example this pair of variables: income vs. education levels - we cannot use Pearson’s coefficient. The only possibility is to rank also incomes… and lose some more detailed information about them.

First, let’s see boxplots of income by education levels.

######################## Now, let’s see Kendal’s coefficient of rank correlation (robust for ties).

## [1] 0.1577567

Point-biserial correlation

Let’s try to verify if there is a significant relationship between incomes and risk status. First, let’s take a look at the boxplot:

If you would like to compare 1 quantitative variable (income) and 1 dychotomous variable (default status - binary), then you can use point-biserial coefficient:

## [1] -0.07096966

Nonlinear correlation - eta coefficient

If you would like to check if there are any nonlinearities between 2 variables, the only possibility (beside transformations and linear analysis) is to calculate “eta” coefficient and compare it with the Pearson’s linear coefficient.

## [1] 1.725159e-05

Correlation matrix

We can also prepare the correlation matrix for all quantitative variables stored in our data frame.

We can use ggcorr() function:

## Warning in ggcorr(bank, label = TRUE): data in column(s) 'def', 'educ' are not
## numeric and were ignored

As you can see - the default correlation matrix is not the best idea for all measurement scales (including binary variable “default”).

That’s why now we can perform our bivariate analysis with ggpair with grouping.

Correlation matrix with scatterplots

Here is what we are about to calculate: - The correlation matrix between age, log_income, employ, address, debtinc, creddebt, and othdebt variable grouped by whether the person has a default status or not. - Plot the distribution of each variable by group - Display the scatter plot with the trend by group

## `stat_bin()` using `bins = 30`. Pick better value with `binwidth`.
## `stat_bin()` using `bins = 30`. Pick better value with `binwidth`.
## `stat_bin()` using `bins = 30`. Pick better value with `binwidth`.
## `stat_bin()` using `bins = 30`. Pick better value with `binwidth`.
## `stat_bin()` using `bins = 30`. Pick better value with `binwidth`.
## `stat_bin()` using `bins = 30`. Pick better value with `binwidth`.
## `stat_bin()` using `bins = 30`. Pick better value with `binwidth`.

Qualitative data

In case of two variables measured on nominal or ordinal&nominal scale - we are forced to organize so called “contingency” table with frequencies and calculate some kind of the correlation coefficient based on them. This is so called “contingency analysis”.

Let’s consider one example based on our data: verify, if there is any significant correlation between education level and credit risk.

#contingency table
contingency_table <- table(bank$ed, bank$debtinc)
print(contingency_table)
##    
##     0.4 0.6 0.7 0.8 0.9 1 1.1 1.2 1.3 1.4 1.6 1.7 1.8 1.9 2.1 2.2 2.3 2.4 2.5
##   1   1   1   0   0   2 0   2   4   1   0   2   2   1   1   3   0   1   3   3
##   2   0   1   1   1   1 1   1   0   1   2   2   1   0   3   1   1   0   2   1
##   3   0   0   0   0   0 0   0   0   0   0   0   1   0   0   0   1   2   1   1
##   4   0   0   0   0   0 0   0   0   0   0   1   1   0   1   0   0   0   0   1
##   5   0   0   0   0   0 0   0   0   0   0   0   0   0   0   0   0   0   0   0
##    
##     2.6 2.7 2.8 2.9 3 3.1 3.2 3.3 3.4 3.5 3.6 3.7 3.8 3.9 4 4.1 4.2 4.3 4.4 4.5
##   1   4   1   1   4 2   1   3   2   1   1   4   4   0   0 2   3   3   0   5   5
##   2   1   0   2   1 2   2   0   0   2   1   0   3   1   1 1   1   1   3   2   3
##   3   1   0   2   0 1   1   0   1   0   0   0   1   0   0 1   1   1   0   2   2
##   4   0   0   0   1 0   0   0   0   0   0   1   0   0   0 0   0   0   1   0   0
##   5   0   0   0   0 0   1   0   0   0   0   0   0   0   0 0   0   0   0   0   0
##    
##     4.6 4.7 4.8 4.9 5 5.1 5.2 5.4 5.5 5.6 5.7 5.8 5.9 6 6.1 6.2 6.3 6.4 6.5 6.6
##   1   1   4   6   4 5   1   3   5   3   3   3   1   2 3   3   3   2   4   3   3
##   2   2   0   1   1 1   1   1   2   0   3   1   4   0 0   1   1   3   2   1   2
##   3   0   1   0   1 0   0   0   2   1   0   1   1   0 2   1   0   1   1   0   2
##   4   0   0   0   0 1   0   0   0   1   0   1   0   0 0   0   0   0   1   1   1
##   5   0   0   0   0 0   0   0   0   0   0   0   0   1 0   0   0   0   0   0   0
##    
##     6.7 6.8 6.9 7 7.1 7.2 7.3 7.4 7.5 7.6 7.7 7.8 7.9 8 8.1 8.2 8.3 8.4 8.5 8.6
##   1   5   1   3 3   2   4   2   2   3   3   1   1   2 6   1   2   2   4   2   1
##   2   2   4   1 2   1   4   2   0   0   2   1   2   2 0   0   1   1   0   2   3
##   3   1   0   0 0   0   0   0   0   0   0   3   0   1 0   0   0   0   1   0   0
##   4   1   0   0 0   0   0   0   0   0   0   1   0   1 1   0   0   0   0   1   0
##   5   0   0   0 0   0   0   0   0   0   0   0   1   0 0   1   0   0   0   0   0
##    
##     8.7 8.8 8.9 9 9.1 9.2 9.3 9.4 9.5 9.6 9.7 9.8 9.9 10 10.1 10.2 10.3 10.4
##   1   0   1   4 4   5   1   2   0   2   3   4   3   1  2    2    2    2    2
##   2   1   0   0 1   1   0   1   0   4   2   2   0   0  0    1    1    1    1
##   3   1   0   0 0   0   1   2   1   0   0   0   0   0  0    0    0    0    2
##   4   0   0   0 0   0   0   0   0   0   0   0   1   0  1    0    0    0    1
##   5   0   0   0 0   0   0   0   1   0   0   0   0   0  0    0    0    0    0
##    
##     10.5 10.6 10.7 10.8 10.9 11 11.1 11.2 11.3 11.4 11.5 11.6 11.7 11.8 11.9 12
##   1    5    3    1    1    2  1    1    3    4    1    1    1    2    0    0  3
##   2    0    0    0    0    0  0    2    0    0    2    4    0    0    1    1  0
##   3    1    0    0    1    0  1    0    0    0    0    0    0    0    0    0  1
##   4    0    1    0    0    0  1    1    0    0    0    1    0    0    0    0  0
##   5    0    0    0    0    0  0    0    0    0    0    0    0    0    0    0  0
##    
##     12.1 12.2 12.3 12.4 12.5 12.6 12.7 12.8 12.9 13 13.1 13.2 13.3 13.4 13.5
##   1    1    1    1    1    1    2    1    2    3  2    1    2    0    0    1
##   2    1    1    0    2    2    0    0    1    1  2    0    3    2    1    1
##   3    1    1    0    0    0    0    0    0    1  1    0    2    1    1    0
##   4    1    0    0    1    0    0    1    0    0  0    0    0    0    0    0
##   5    0    0    0    0    0    0    0    0    0  0    0    0    0    0    0
##    
##     13.6 13.7 13.8 13.9 14 14.1 14.2 14.3 14.4 14.5 14.6 14.7 14.8 15 15.1 15.2
##   1    1    2    2    1  2    1    1    2    3    0    1    2    3  3    1    1
##   2    1    0    0    0  0    0    0    1    1    0    0    1    1  4    0    0
##   3    1    0    1    0  0    1    1    0    1    1    1    1    0  0    1    0
##   4    0    0    0    0  0    0    0    0    0    0    0    0    0  0    0    0
##   5    0    0    0    0  0    0    0    0    0    0    0    0    0  0    0    0
##    
##     15.3 15.4 15.6 15.7 16 16.1 16.3 16.4 16.5 16.6 16.7 16.9 17 17.1 17.2 17.3
##   1    0    1    2    0  1    2    3    1    1    1    1    1  2    1    0    3
##   2    0    2    0    0  1    2    0    2    0    0    0    1  0    1    1    1
##   3    1    0    0    1  2    0    0    0    1    0    0    0  0    0    0    0
##   4    0    0    0    1  0    1    0    0    1    0    0    0  0    0    0    2
##   5    0    0    0    0  0    0    0    0    0    0    0    0  0    0    0    0
##    
##     17.4 17.5 17.6 17.8 18 18.1 18.2 18.3 18.4 18.5 18.6 18.7 18.8 18.9 19.1
##   1    1    1    0    2  0    1    0    0    0    1    2    0    1    2    0
##   2    1    0    0    0  1    2    1    1    2    0    0    3    0    1    1
##   3    0    0    1    0  0    0    0    0    0    0    0    0    0    0    0
##   4    0    0    1    0  0    0    0    0    1    0    0    0    0    0    0
##   5    0    0    0    0  0    0    0    0    0    0    0    0    0    0    0
##    
##     19.2 19.4 19.5 19.6 19.7 19.8 19.9 20 20.1 20.2 20.4 20.5 20.9 21 21.2 21.3
##   1    1    1    0    1    1    2    1  1    0    1    0    1    0  1    0    2
##   2    1    0    1    0    0    0    0  0    1    0    1    0    1  0    1    0
##   3    1    0    0    0    0    1    3  0    1    0    0    0    0  0    0    0
##   4    0    0    0    0    0    0    0  0    0    0    0    0    0  0    0    0
##   5    0    0    0    0    0    0    0  0    0    0    0    0    0  0    0    0
##    
##     21.4 21.7 22.3 22.7 22.9 23 23.1 23.3 23.6 23.7 23.8 24.2 24.4 24.5 24.6
##   1    1    0    1    0    0  0    3    0    0    1    2    0    1    1    0
##   2    0    1    0    1    1  0    2    1    1    0    2    0    0    0    0
##   3    0    0    0    0    0  1    0    0    0    0    0    0    0    0    1
##   4    0    0    0    0    0  0    0    0    0    0    0    1    0    0    0
##   5    0    0    0    0    0  0    0    0    0    0    0    0    0    0    0
##    
##     24.7 25.1 25.2 25.3 25.4 25.6 25.7 26 26.3 26.5 27.1 27.7 28.5 28.9 29.7
##   1    1    1    0    1    1    1    1  0    0    1    1    1    1    2    0
##   2    0    1    0    0    0    0    0  1    0    0    0    1    0    0    0
##   3    0    0    1    0    0    0    0  0    1    1    0    0    0    0    1
##   4    0    0    0    0    0    0    0  0    0    0    0    0    0    0    0
##   5    0    0    0    0    0    0    0  0    0    0    0    0    0    0    0
##    
##     30.1 30.6 30.7 30.8 32.5 33.3 33.4 35.3 36.6 41.3
##   1    1    1    1    1    1    1    0    0    0    1
##   2    0    0    0    0    0    0    0    0    0    0
##   3    0    0    0    0    0    0    0    0    1    0
##   4    0    0    0    0    0    0    1    1    0    0
##   5    0    0    0    0    0    0    0    0    0    0
#chi-squared test of independence
chi_squared_test <- chisq.test(contingency_table)
## Warning in chisq.test(contingency_table): Chi-squared approximation may be
## incorrect
print(chi_squared_test)
## 
##  Pearson's Chi-squared test
## 
## data:  contingency_table
## X-squared = 879.06, df = 920, p-value = 0.8298

Exercise 1. Contingency analysis.

Do you believe in the Afterlife? https://nationalpost.com/news/canada/millennials-do-you-believe-in-life-after-life A survey was conducted and a random sample of 1091 questionnaires is given in the form of the following contingency table:

##         Believe
## Gender   Yes  No
##   Female 435 375
##   Male   147 134

Our task is to check if there is a significant relationship between the belief in the afterlife and gender. We can perform this procedure with the simple chi-square statistics and chosen qualitative correlation coefficient (two-way 2x2 table).

## 
##  Pearson's Chi-squared test with Yates' continuity correction
## 
## data:  dane
## X-squared = 0.11103, df = 1, p-value = 0.739
##         Believe
## Gender         Yes        No
##   Female 0.3987168 0.3437214
##   Male   0.1347388 0.1228231

As you can see we can calculate our chi-square statistic really quickly for two-way tables or larger. Now we can standardize this contingency measure to see if the relationship is significant.

## [1] 0.01218871

Exercise 2. Contingency analysis for the ‘Titanic’ data.

Let’s consider the titanic dataset which contains a complete list of passengers and crew members on the RMS Titanic. It includes a variable indicating whether a person did survive the sinking of the RMS Titanic on April 15, 1912. A data frame contains 2456 observations on 14 variables.

The website http://www.encyclopedia-titanica.org/ offers detailed information about passengers and crew members on the RMS Titanic. According to the website 1317 passengers and 890 crew member were aboard.

8 musicians and 9 employees of the shipyard company are listed as passengers, but travelled with a free ticket, which is why they have NA values in fare. In addition to that, fare is truely missing for a few regular passengers.

# Clean the data by replacing empty strings with NA and dropping all NA values
titanic[titanic == ""] <- NA
titanic <- na.omit(titanic)

# Recoding variables as factors for categorical analysis
titanic$Status <- as.factor(titanic$Status)
titanic$Gender <- as.factor(titanic$Gender)
titanic$Embarked <- as.factor(titanic$Embarked)

# Plot the distribution of Age by Survival Status
library(ggplot2)
ggplot(titanic, aes(x = Age, fill = Status)) + 
    geom_histogram(binwidth = 5, position = "fill", color = "black") +
    labs(y = "Proportion", x = "Age", fill = "Survival Status", title = "Survival by Age Group")

# Contingency tables and Chi-squared tests
# 1. Class and Survival
table_Class_Survived <- table(titanic$Class, titanic$Status)
chi_Class_Survived <- chisq.test(table_Class_Survived)
print(table_Class_Survived)
##            
##             Survivor Victim
##   1st Class       50     53
##   2nd Class       11     83
##   3rd Class       33    341
print(chi_Class_Survived)
## 
##  Pearson's Chi-squared test
## 
## data:  table_Class_Survived
## X-squared = 94.503, df = 2, p-value < 2.2e-16
# 2. Gender and Survival
table_Gender_Survived <- table(titanic$Gender, titanic$Status)
chi_Gender_Survived <- chisq.test(table_Gender_Survived)
print(table_Gender_Survived)
##         
##          Survivor Victim
##   Female       35     36
##   Male         59    441
print(chi_Gender_Survived)
## 
##  Pearson's Chi-squared test with Yates' continuity correction
## 
## data:  table_Gender_Survived
## X-squared = 60.863, df = 1, p-value = 6.12e-15
# 3. Embarkation point and Survival
table_Embarked_Survived <- table(titanic$Embarked, titanic$Status)
chi_Embarked_Survived <- chisq.test(table_Embarked_Survived)
## Warning in chisq.test(table_Embarked_Survived): Chi-squared approximation may
## be incorrect
print(table_Embarked_Survived)
##              
##               Survivor Victim
##   Belfast            0      1
##   Cherbourg         30     55
##   Queenstown         9     48
##   Southampton       55    373
print(chi_Embarked_Survived)
## 
##  Pearson's Chi-squared test
## 
## data:  table_Embarked_Survived
## X-squared = 26.195, df = 3, p-value = 8.68e-06
LS0tDQp0aXRsZTogJ0Rlc2NyaXB0aXZlIFN0YXRpc3RpY3MnDQpzdWJ0aXRsZTogJ0JpdmFyaWF0ZSBBbmFseXNpcycNCmRhdGU6ICJgciBTeXMuRGF0ZSgpYCINCmF1dGhvcjogIlNlbWloIEVMTUFTLCBFbXJlIEFZRElOIg0Kb3V0cHV0Og0KICBodG1sX2RvY3VtZW50OiANCiAgICB0aGVtZTogY2VydWxlYW4NCiAgICBoaWdobGlnaHQ6IHRleHRtYXRlDQogICAgZm9udHNpemU6IDEwcHQNCiAgICB0b2M6IHllcw0KICAgIGNvZGVfZG93bmxvYWQ6IHllcw0KICAgIHRvY19mbG9hdDoNCiAgICAgIGNvbGxhcHNlZDogbm8NCiAgICBkZl9wcmludDogZGVmYXVsdA0KICAgIHRvY19kZXB0aDogNQ0KZWRpdG9yX29wdGlvbnM6IA0KICBtYXJrZG93bjogDQogICAgd3JhcDogNzINCi0tLQ0KDQpgYGB7ciBzZXR1cCwJbWVzc2FnZSA9IEZBTFNFLAl3YXJuaW5nID0gRkFMU0UsCWluY2x1ZGUgPSBGQUxTRX0NCmxpYnJhcnkoZHBseXIpDQpsaWJyYXJ5KHRpZHl2ZXJzZSkNCmxpYnJhcnkoSFNBVVIzKQ0KbGlicmFyeShoYXZlbikNCmxpYnJhcnkoZ2dwbG90MikNCmxpYnJhcnkoZ3JpZEV4dHJhKQ0KbGlicmFyeShwcGNvcikgIyB0aGlzIHBhY2thZ2UgY29tcHV0ZXMgcGFydGlhbCBhbmQgc2VtaXBhcnRpYWwgY29ycmVsYXRpb25zLg0KbGlicmFyeShsdG0pICMgdGhpcyBwYWNrYWdlIGNvbXB1dGVzIHBvaW50LWJpc2VyaWFsIGNvcnJlbGF0aW9ucy4NCmxpYnJhcnkoZGV2dG9vbHMpIA0KaW5zdGFsbF9naXRodWIoIm1hcmtoZWNrbWFubi9yeW91cmVhZHkiKSAjIHBsZWFzZSBpbnN0YWxsIHBhY2thZ2UgInJ5b3VyZWFkeSIgZnJvbSBnaXRodWIhICh0aGVuICMgaXQpDQpsaWJyYXJ5KHJ5b3VyZWFkeSkgIyB0aGlzIHBhY2thZ2UgY29tcHV0ZXMgbm9ubGluZWFyICJldGEiIGNvcnJlbGF0aW9ucy4NCmxpYnJhcnkoR0dhbGx5KSAjIHRoaXMgcGFja2FnZSBjb21wdXRlcyBjb3JyZWxhdGlvbiBtYXRyaXguDQpsaWJyYXJ5KHBzeWNoKSAjIHRoaXMgcGFja2FnZSBjb21wdXRlcyBxdWFsaXRhdGl2ZSBjb3JyZWxhdGlvbnMuDQpsaWJyYXJ5KERlc2NUb29scykgIyB0aGlzIHBhY2thZ2UgY29tcHV0ZXMgcXVhbGl0YXRpdmUgY29ycmVsYXRpb25zLg0KYGBgDQoNCg0KIyMgSW50cm9kdWN0aW9uDQoNClRoaXMgaXMgb3VyIGZpcnN0IGxhYiB3aGVuIHdlIGFyZSBjb25zaWRlcmluZyAyIGRpbWVuc2lvbnMgYW5kIGluc3RlYWQgb2YgY2FsY3VsYXRpbmcgdW5pdmFyaWF0ZSBzdGF0aXN0aWNzIGJ5IGdyb3VwcyAob3IgZmFjdG9ycykgb2Ygb3RoZXIgdmFyaWFibGUgLSB3ZSB3aWxsIG1lYXN1cmUgdGhlaXIgY29tbW9uIHJlbGF0aW9uc2hpcHMgYmFzZWQgb24gY28tdmFyaWFuY2UgYW5kIGNvcnJlbGF0aW9uIGNvZWZmaWNpZW50cy4gDQoNCipQbGVhc2UgYmUgdmVyeSBjYXJlZnVsIHdoZW4gY2hvb3NpbmcgdGhlIG1lYXN1cmUgb2YgY29ycmVsYXRpb24hIEluIGNhc2Ugb2YgZGlmZmVyZW50IG1lYXN1cnVtZW50IHNjYWxlcyB3ZSBoYXZlIHRvIHJlY29kZSBvbmUgb2YgdGhlIHZhcmlhYmxlcyBpbnRvIHdlYWtlciBzY2FsZS4NCg0KSXQgd291bGQgYmUgbmljZSB0byBhZGQgc29tZSBhZGRpdGlvbmFsIHBsb3RzIGluIHRoZSBiYWNrZ3JvdW5kLiBGZWVsIGZyZWUgdG8gYWRkIHlvdXIgb3duIHNlY3Rpb25zIGFuZCB1c2UgZXh0ZXJuYWwgcGFja2FnZXMuDQoNCiMjIERhdGENCg0KVGhpcyB0aW1lIHdlIGFyZSBnb2luZyB0byB1c2UgYSB0eXBpY2FsIGNyZWRpdCBzY29yaW5nIGRhdGEgd2l0aCBwcmVkZWZpbmVkICJkZWZhdWx0IiB2YXJpYWJsZXMgYW5kIHBlcnNvbmFsIGRlbW9ncmFmaWMgYW5kIGluY29tZSBkYXRhLiBQbGVhc2UgdGFrZSBhIGxvb2sgY2xvc2VyIGF0IGhlYWRlcnMgYW5kIGRlc2NyaXB0aW9ucyBvZiBlYWNoIHZhcmlhYmxlLg0KDQpgYGB7ciBsb2FkLWRhdGEsIHdhcm5pbmc9VFJVRSwgaW5jbHVkZT1GQUxTRX0NCmRvd25sb2FkLmZpbGUoImh0dHBzOi8vZ2l0aHViLmNvbS9rZmxpc2lrb3dza2kvZHMvYmxvYi9tYXN0ZXIvYmFua19kZWZhdWx0cy5zYXY/cmF3PXRydWUiLCBkZXN0ZmlsZSA9ImJhbmtfZGVmYXVsdHMuc2F2Iixtb2RlPSJ3YiIpDQpiYW5rX2RlZmF1bHRzIDwtIHJlYWRfc2F2KCJiYW5rX2RlZmF1bHRzLnNhdiIpDQpiYW5rPC1uYS5vbWl0KGJhbmtfZGVmYXVsdHMpDQpiYW5rJGRlZjwtYXMuZmFjdG9yKGJhbmskZGVmYXVsdCkNCmJhbmskZWR1YzwtYXMuZmFjdG9yKGJhbmskZWQpDQpgYGANCg0KIyMgU2NhdHRlcnBsb3RzDQoNCkZpcnN0IGxldCdzIHZpc3VhbGl6ZSBvdXIgcXVhbnRpdGF0aXZlIHJlbGF0aW9uc2hpcHMgdXNpbmcgc2NhdHRlcnBsb3RzLiANCg0KYGBge3IgZWNobz1GQUxTRSwgd2FybmluZz1UUlVFfQ0KYmFuayRsb2dpbmNvbWUgPC0gbG9nKGJhbmskaW5jb21lKQ0KDQpjb2xzX3RvX2NoZWNrIDwtIGMoImxvZ2luY29tZSIsICJhZ2UiLCAiaW5jb21lIiwgImVtcGxveSIpDQoNCmJhbmsgPC0gYmFua1tjb21wbGV0ZS5jYXNlcyhiYW5rWywgY29sc190b19jaGVja10pLCBdDQoNCnBsb3QoYmFuayRsb2dpbmNvbWUsIGJhbmskYWdlLCBtYWluPSJMb2coSW5jb21lKSB2cy4gQWdlIiwNCiAgICAgeGxhYj0iTG9nIG9mIEluY29tZSIsIHlsYWI9IkFnZSIsIHBjaD0xOSwgY29sPXJnYigwLjEsIDAuMSwgMC44LCAwLjcpKQ0KY29ycmVsYXRpb25fcmVzdWx0IDwtIGNvcihiYW5rJGFnZSwgYmFuayRsb2dpbmNvbWUpDQpwcmludChjb3JyZWxhdGlvbl9yZXN1bHQpDQoNCnJlc3VsdCA8LSBwY29yLnRlc3QoYmFuayRhZ2UsIGJhbmskbG9naW5jb21lLCBiYW5rJGVtcGxveSwgbWV0aG9kID0gImtlbmRhbGwiKQ0KcHJpbnQocmVzdWx0KQ0KYGBgDQoNCllvdSBjYW4gYWxzbyBub3JtYWxpemUgdGhlIHNrZXdlZCBkaXN0cmlidXRpb24gb2YgaW5jb21lcyB1c2luZyBsb2c6DQoNCmBgYHtyIGVjaG89RkFMU0UsIHdhcm5pbmc9VFJVRX0NCiMgQmFzaWMgc2NhdHRlciBwbG90IHdpdGggdGhlIGxvZyBvZiBpbmNvbWUNCmJhbmskbG9naW5jb21lIDwtIGxvZyhiYW5rJGluY29tZSkNCg0KcGxvdChiYW5rJGxvZ2luY29tZSwgYmFuayRhZ2UsIG1haW49IlNjYXR0ZXIgUGxvdCBvZiBMb2coSW5jb21lKSB2cy4gQWdlIiwNCiAgICAgeGxhYj0iTG9nIG9mIEluY29tZSIsIHlsYWI9IkFnZSIsIHBjaD0xOSwgY29sPXJnYigwLjEsIDAuMSwgMC44LCAwLjcpKQ0KYGBgDQoNCldlIGNhbiBhZGQgYW4gZXN0aW1hdGVkIGxpbmVhciByZWdyZXNzaW9uIGxpbmU6DQoNCmBgYHtyIGVjaG89RkFMU0UsIHdhcm5pbmc9VFJVRX0NCg0KcGxvdChiYW5rJGxvZ2luY29tZSwgYmFuayRhZ2UsIG1haW49IkxvZyhJbmNvbWUpIHZzLiBBZ2UiLA0KICAgICB4bGFiPSJMb2cgb2YgSW5jb21lIiwgeWxhYj0iQWdlIiwgcGNoPTE5LCBjb2w9cmdiKDAuMSwgMC4xLCAwLjgsIDAuNykpDQoNCg0KbW9kZWwgPC0gbG0oYmFuayRhZ2UgfiBiYW5rJGxvZ2luY29tZSkgIA0KYWJsaW5lKG1vZGVsLCBjb2w9InJlZCIpDQoNCmBgYA0KDQojIyBTY2F0dGVycGxvdHMgYnkgZ3JvdXBzIA0KDQpXZSBjYW4gZmluYWxseSBzZWUgaWYgdGhlcmUgYW55IGRpZmZlcmVuY2VzIGJldHdlZW4gcmlzayBzdGF0dXM6DQoNCmBgYHtyIGVjaG89RkFMU0UsIHdhcm5pbmc9VFJVRX0NCmdncGxvdChiYW5rLCBhZXMoeCA9IGxvZ2luY29tZSwgeSA9IGNyZWRkZWJ0KSkgKw0KICBnZW9tX3BvaW50KCkgKw0KICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEZBTFNFKSArDQogIGxhYnModGl0bGUgPSAiTG9nKEluY29tZSkgdnMuIENyZWRpdCBDYXJkIERlYnQiLA0KICAgICAgIHggPSAiTG9nIG9mIEluY29tZSIsDQogICAgICAgeSA9ICJDcmVkaXQgQ2FyZCBEZWJ0IikgDQoNCmBgYA0KDQpXZSBjYW4gYWxzbyBzZWUgbW9yZSBjbG9zZWx5IGlmIHRoZXJlIGFueSBkaWZmZXJlbmNlcyBiZXR3ZWVuIHRob3NlIHR3byBkaXN0cmlidXRpb25zIGFkZGluZyB0aGVpciBlc3RpbWF0ZWQgZGVuc2l0eSBwbG90czoNCg0KYGBge3IgZWNobz1GQUxTRSwgd2FybmluZz1UUlVFfQ0KDQojIG1haW4gc2NhdHRlciBwbG90DQpzY2F0dGVyX3Bsb3QgPC0gZ2dwbG90KGJhbmssIGFlcyh4ID0gYWdlLCB5ID0gbG9naW5jb21lKSkgKw0KICBnZW9tX3BvaW50KCkgKw0KICB0aGVtZV9taW5pbWFsKCkNCg0KIyBtYXJnaW5hbCBkZW5zaXR5IHBsb3QgZm9yICdhZ2UnDQpkZW5zaXR5X2FnZSA8LSBnZ3Bsb3QoYmFuaywgYWVzKHggPSBhZ2UpKSArDQogIGdlb21fZGVuc2l0eShhbHBoYSA9IDAuNSkgKw0KICB0aGVtZV9taW5pbWFsKCkNCg0KIyBtYXJnaW5hbCBkZW5zaXR5IHBsb3QgZm9yICdsb2dpbmNvbWUnDQpkZW5zaXR5X3kgPC0gZ2dwbG90KGJhbmssIGFlcyh4ID0gbG9naW5jb21lKSkgKw0KICBnZW9tX2RlbnNpdHkoYWxwaGEgPSAwLjUpICsNCiAgdGhlbWVfbWluaW1hbCgpDQoNCiMgY29tYmluZSB0aGUgcGxvdHMNCmdyaWQuYXJyYW5nZSgNCiAgZGVuc2l0eV9hZ2UsIA0KICBOVUxMLCANCiAgc2NhdHRlcl9wbG90LCANCiAgZGVuc2l0eV95LA0KICBuY29sID0gMiwgbnJvdyA9IDIsDQogIHdpZHRocyA9IGMoNCwgMSksDQogIGhlaWdodHMgPSBjKDEsIDQpDQopDQoNCmBgYA0KDQojIyBTY2F0dGVycGxvdHMgd2l0aCBkZW5zaXR5IGN1cnZlcyANCg0KV2UgY2FuIGFsc28gc2VlIG1vcmUgY2xvc2VseSBpZiB0aGVyZSBhbnkgZGlmZmVyZW5jZXMgYmV0d2VlbiB0aG9zZSB0d28gZGlzdHJpYnV0aW9ucyBhZGRpbmcgdGhlaXIgZXN0aW1hdGVkIGRlbnNpdHkgcGxvdHM6DQoNCmBgYHtyIGVjaG89RkFMU0UsIHdhcm5pbmc9VFJVRX0NCg0KcGxvdCA8LSBnZ3Bsb3QoYmFuaywgYWVzKHggPSBhZ2UsIHkgPSBsb2dpbmNvbWUsIGNvbG9yID0pKSArDQogIGdlb21fcG9pbnQoYWxwaGEgPSAwLjUpICsNCiAgZ2VvbV9kZW5zaXR5XzJkKGFlcyhmaWxsID0pLCBhbHBoYSA9IDAuNSkgKw0KICBsYWJzKHRpdGxlID0gIlNjYXR0ZXIgUGxvdCB3aXRoIERlbnNpdHkgQ3VydmVzIiwNCiAgICAgICB4ID0gIkFnZSIsDQogICAgICAgeSA9ICJsb2coSW5jb21lKSIpICsNCiAgdGhlbWVfbWluaW1hbCgpDQoNCiMgRGlzcGxheSB0aGUgcGxvdA0KcHJpbnQocGxvdCkNCg0KYGBgDQoNCiMjIENvcnJlbGF0aW9uIGNvZWZmaWNpZW50cyAtIFBlYXJzb24ncyBsaW5lYXIgY29ycmVsYXRpb24NCg0KT2ssIGxldCdzIG1vdmUgdG8gc29tZSBjYWxjdWxhdGlvbnMuDQpJbiBSLCB3ZSBjYW4gdXNlIHRoZSBjb3IoKSBmdW5jdGlvbi4gSXQgdGFrZXMgdGhyZWUgYXJndW1lbnRzIGFuZCB0aGUgbWV0aG9kOiBjb3IoeCwgeSwgbWV0aG9kKQ0KRm9yIDIgcXVhbnRpdGF0aXZlIGRhdGEsIHdpdGggYWxsIGFzc3VtcHRpb25zIG1ldCwgd2UgY2FuIGNhbGN1bGF0ZSBzaW1wbGUgUGVhcnNvbidzIGNvZWZmaWNpZW50IG9mIGxpbmVhciBjb3JyZWxhdGlvbjoNCg0KYGBge3IgZWNobz1GQUxTRSwgd2FybmluZz1UUlVFfQ0KDQpjb3JyZWxhdGlvbl9jb2VmZmljaWVudCA8LSBjb3IoYmFuayRhZ2UsIGJhbmskbG9naW5jb21lLCBtZXRob2QgPSAicGVhcnNvbiIpDQoNCnByaW50KGNvcnJlbGF0aW9uX2NvZWZmaWNpZW50KQ0KDQoNCmBgYA0KDQpPaywgd2hhdCBhYm91dCB0aGUgcGVyY2VudGFnZSBvZiB0aGUgZXhwbGFpbmVkIHZhcmlhYmlsaXR5Pw0KDQpgYGB7ciBlY2hvPUZBTFNFLCB3YXJuaW5nPVRSVUV9DQoNCnIgPC0gY29yKGJhbmskYWdlLCBiYW5rJGxvZ2luY29tZSwgbWV0aG9kID0gInBlYXJzb24iKQ0KDQpyX3NxdWFyZWQgPC0gcl4yDQoNCnBlcmNlbnRhZ2VfZXhwbGFpbmVkIDwtIHJfc3F1YXJlZCAqIDEwMA0KcHJpbnQocGFzdGUoIlBlcmNlbnRhZ2Ugb2YgZXhwbGFpbmVkIHZhcmlhYmlsaXR5OiIsIHBlcmNlbnRhZ2VfZXhwbGFpbmVkLCAiJSIpKQ0KDQoNCmBgYA0KU28gYXMgd2UgY2FuIHNlZSBhbG1vc3QgPz8/IG9mIHRvdGFsIGxvZyBvZiBpbmNvbWVzJyB2YXJpYWJpbGl0eSBpcyBleHBsYWluZWQgYnkgZGlmZmVyZW5jZXMgaW4gYWdlLiBUaGUgcmVzdCAoPz8/KSBpcyBwcm9iYWJseSBleHBsYWluZWQgYnkgb3RoZXIgZmFjdG9ycy4NCg0KIyMgUGFydGlhbCBhbmQgc2VtaXBhcnRpYWwgY29ycmVsYXRpb24gDQoNClRoZSBwYXJ0aWFsIGFuZCBzZW1pLXBhcnRpYWwgKGFsc28ga25vd24gYXMgcGFydCkgY29ycmVsYXRpb25zIGFyZSB1c2VkIHRvIGV4cHJlc3MgdGhlIHNwZWNpZmljIHBvcnRpb24gb2YgdmFyaWFuY2UgZXhwbGFpbmVkIGJ5IGVsaW1pbmF0aW5nIHRoZSBlZmZlY3Qgb2Ygb3RoZXIgdmFyaWFibGVzIHdoZW4gYXNzZXNzaW5nIHRoZSBjb3JyZWxhdGlvbiBiZXR3ZWVuIHR3byB2YXJpYWJsZXMuDQoNClBhcnRpYWwgY29ycmVsYXRpb24gaG9sZHMgY29uc3RhbnQgb25lIHZhcmlhYmxlIHdoZW4gY29tcHV0aW5nIHRoZSByZWxhdGlvbnMgdG8gb3RoZXJzLiBTdXBwb3NlIHdlIHdhbnQgdG8ga25vdyB0aGUgY29ycmVsYXRpb24gYmV0d2VlbiBYIGFuZCBZIGhvbGRpbmcgWiBjb25zdGFudCBmb3IgYm90aCBYIGFuZCBZLiBUaGF0IHdvdWxkIGJlIHRoZSBwYXJ0aWFsIGNvcnJlbGF0aW9uIGJldHdlZW4gWCBhbmQgWSBjb250cm9sbGluZyBmb3IgWi4gDQoNClNlbWlwYXJ0aWFsIGNvcnJlbGF0aW9uIGhvbGRzIFogY29uc3RhbnQgZm9yIGVpdGhlciBYIG9yIFksIGJ1dCBub3QgYm90aCwgc28gaWYgd2Ugd2FudGVkIHRvIGNvbnRyb2wgWCBmb3IgWiwgd2UgY291bGQgY29tcHV0ZSB0aGUgc2VtaXBhcnRpYWwgY29ycmVsYXRpb24gYmV0d2VlbiBYIGFuZCBZIGhvbGRpbmcgWiBjb25zdGFudCBmb3IgWC4NCg0KU3VwcG9zZSB3ZSB3YW50IHRvIGtub3cgdGhlIGNvcnJlbGF0aW9uIGJldHdlZW4gdGhlIGxvZyBvZiBpbmNvbWUgYW5kIGFnZSBjb250cm9sbGluZyBmb3IgeWVhcnMgb2YgZW1wbG95bWVudC4gSG93IGhpZ2hseSBjb3JyZWxhdGVkIGFyZSB0aGVzZSBhZnRlciBjb250cm9sbGluZyBmb3IgdGVudXJlPyANCg0KKipUaGVyZSBjYW4gYmUgbW9yZSB0aGFuIG9uZSBjb250cm9sIHZhcmlhYmxlLg0KDQpgYGB7ciBlY2hvPUZBTFNFLCB3YXJuaW5nPUZBTFNFfQ0KcGFydGlhbF9jb3JyIDwtIHBjb3IudGVzdChiYW5rJGxvZ2luY29tZSwgYmFuayRhZ2UsIGJhbmskZW1wbG95LCBtZXRob2Q9InBlYXJzb24iKQ0KcHJpbnQocGFydGlhbF9jb3JyKQ0KDQpzZW1pcGFydGlhbF9jb3JyIDwtIHNwY29yLnRlc3QoYmFuayRsb2dpbmNvbWUsIGJhbmskYWdlLCBiYW5rJGVtcGxveSwgbWV0aG9kPSJwZWFyc29uIikNCnByaW50KHNlbWlwYXJ0aWFsX2NvcnIpDQoNCmBgYA0KDQpIb3cgY2FuIHdlIGludGVycHJldCB0aGUgb2J0YWluZWQgcGFydGlhbCBjb3JyZWxhdGlvbiBjb2VmZmljaWVudD8gV2hhdCBpcyB0aGUgZGlmZmVyZW5jZSBiZXR3ZWVuIHRoYXQgb25lIGFuZCB0aGUgc2VtaS1wYXJ0aWFsIGNvZWZmaWNpZW50Og0KDQpgYGB7ciBlY2hvPUZBTFNFLCB3YXJuaW5nPUZBTFNFfQ0KDQojIHBhcnRpYWwgY29ycmVsYXRpb24gY29udHJvbGxpbmcgZm9yIFoNCnJlc3VsdF9wYXJ0aWFsIDwtIHBjb3IudGVzdCh4PWJhbmskbG9naW5jb21lLCB5PWJhbmskYWdlLCB6PWJhbmskZW1wbG95LCBtZXRob2Q9InBlYXJzb24iKQ0KcHJpbnQocmVzdWx0X3BhcnRpYWwpDQoNCiMgc2VtaXBhcnRpYWwgY29ycmVsYXRpb24gd2hlcmUgWiBpcyBjb250cm9sbGVkIG9ubHkgaW4gcmVsYXRpb24gdG8gWA0KcmVzdWx0X3NlbWkgPC0gc3Bjb3IudGVzdCh4PWJhbmskbG9naW5jb21lLCB5PWJhbmskYWdlLCB6PWJhbmskZW1wbG95LCBtZXRob2Q9InBlYXJzb24iKQ0KcHJpbnQocmVzdWx0X3NlbWkpDQoNCmBgYA0KDQojIyBSYW5rIGNvcnJlbGF0aW9uIA0KDQpGb3IgMiBkaWZmZXJlbnQgc2NhbGVzIC0gbGlrZSBmb3IgZXhhbXBsZSB0aGlzIHBhaXIgb2YgdmFyaWFibGVzOiBpbmNvbWUgdnMuIGVkdWNhdGlvbiBsZXZlbHMgLSB3ZSBjYW5ub3QgdXNlIFBlYXJzb24ncyBjb2VmZmljaWVudC4gVGhlIG9ubHkgcG9zc2liaWxpdHkgaXMgdG8gcmFuayBhbHNvIGluY29tZXMuLi4gYW5kIGxvc2Ugc29tZSBtb3JlIGRldGFpbGVkIGluZm9ybWF0aW9uIGFib3V0IHRoZW0uIA0KDQpGaXJzdCwgbGV0J3Mgc2VlIGJveHBsb3RzIG9mIGluY29tZSBieSBlZHVjYXRpb24gbGV2ZWxzLg0KDQpgYGB7ciBlY2hvPUZBTFNFLCB3YXJuaW5nPVRSVUV9DQoNCiMgYm94cGxvdCBvZiBpbmNvbWUgYnkgZWR1Y2F0aW9uIGxldmVscw0KZ2dwbG90KGJhbmssIGFlcyh4PWZhY3RvcihlZCksIHk9bG9naW5jb21lKSkgKyANCiAgZ2VvbV9ib3hwbG90KCkgKw0KICBsYWJzKHRpdGxlPSJCb3hwbG90IG9mIEluY29tZSBieSBFZHVjYXRpb24gTGV2ZWxzIiwNCiAgICAgICB4PSJFZHVjYXRpb24gTGV2ZWxzIiwNCiAgICAgICB5PSJJbmNvbWUiKSArDQogIHRoZW1lX21pbmltYWwoKQ0KDQpgYGANCiMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIw0KTm93LCBsZXQncyBzZWUgS2VuZGFsJ3MgY29lZmZpY2llbnQgb2YgcmFuayBjb3JyZWxhdGlvbiAocm9idXN0IGZvciB0aWVzKS4NCg0KYGBge3IgZWNobz1GQUxTRSwgd2FybmluZz1UUlVFfQ0Ka2VuZGFsIDwtIGNvcihiYW5rJGluY29tZSwgYmFuayRlZCwgbWV0aG9kID0gImtlbmRhbGwiKQ0KcHJpbnQoa2VuZGFsKQ0KYGBgDQoNCiMjIFBvaW50LWJpc2VyaWFsIGNvcnJlbGF0aW9uDQoNCkxldCdzIHRyeSB0byB2ZXJpZnkgaWYgdGhlcmUgaXMgYSBzaWduaWZpY2FudCByZWxhdGlvbnNoaXAgYmV0d2VlbiBpbmNvbWVzIGFuZCByaXNrIHN0YXR1cy4gRmlyc3QsIGxldCdzIHRha2UgYSBsb29rIGF0IHRoZSBib3hwbG90Og0KDQpgYGB7ciBlY2hvPUZBTFNFLCB3YXJuaW5nPVRSVUV9DQoNCiMgQ3JlYXRlIGJveHBsb3Qgb2YgaW5jb21lIGJ5IHJpc2sgc3RhdHVzDQpnZ3Bsb3QoYmFuaywgYWVzKHg9ZmFjdG9yKGRlYnRpbmMpLCB5PWluY29tZSkpICsgDQogIGdlb21fYm94cGxvdCgpICsNCiAgbGFicyh0aXRsZT0iQm94cGxvdCBvZiBJbmNvbWUgYnkgRGVidCB0byBJbmNvbWUgUmF0aW8iLA0KICAgICAgIHg9IkRlYnQgdG8gSW5jb21lIFJhdGlvIiwNCiAgICAgICB5PSJJbmNvbWUiKQ0KYGBgDQoNCklmIHlvdSB3b3VsZCBsaWtlIHRvIGNvbXBhcmUgMSBxdWFudGl0YXRpdmUgdmFyaWFibGUgKGluY29tZSkgYW5kIDEgZHljaG90b21vdXMgdmFyaWFibGUgKGRlZmF1bHQgc3RhdHVzIC0gYmluYXJ5KSwgdGhlbiB5b3UgY2FuIHVzZSBwb2ludC1iaXNlcmlhbCBjb2VmZmljaWVudDoNCg0KYGBge3IgZWNobz1GQUxTRSwgd2FybmluZz1GQUxTRX0NCnBvaW50X2Jpc2VyaWFsIDwtIGNvcihiYW5rJGluY29tZSwgYXMubnVtZXJpYyhiYW5rJGRlZmF1bHQpLCBtZXRob2QgPSAicGVhcnNvbiIpDQpwcmludChwb2ludF9iaXNlcmlhbCkNCmBgYA0KDQoNCiMjIE5vbmxpbmVhciBjb3JyZWxhdGlvbiAtIGV0YSBjb2VmZmljaWVudA0KDQpJZiB5b3Ugd291bGQgbGlrZSB0byBjaGVjayBpZiB0aGVyZSBhcmUgYW55IG5vbmxpbmVhcml0aWVzIGJldHdlZW4gMiB2YXJpYWJsZXMsIHRoZSBvbmx5IHBvc3NpYmlsaXR5IChiZXNpZGUgdHJhbnNmb3JtYXRpb25zIGFuZCBsaW5lYXIgYW5hbHlzaXMpIGlzIHRvIGNhbGN1bGF0ZSAiZXRhIiBjb2VmZmljaWVudCBhbmQgY29tcGFyZSBpdCB3aXRoIHRoZSBQZWFyc29uJ3MgbGluZWFyIGNvZWZmaWNpZW50LiANCg0KYGBge3IgZWNobz1GQUxTRSwgd2FybmluZz1GQUxTRX0NCg0KYW5vdmFfcmVzdWx0IDwtIGFvdihpbmNvbWUgfiBlZCwgZGF0YT1iYW5rKQ0Kc3VtbWFyeV9hbm92YSA8LSBzdW1tYXJ5KGFub3ZhX3Jlc3VsdCkNCmV0YV9zcXVhcmVkIDwtIHN1bW1hcnlfYW5vdmFbWzFdXVtbIlByKD5GKSJdXVsxXQ0KZXRhIDwtIHNxcnQoZXRhX3NxdWFyZWQpDQpwcmludChldGEpDQoNCmBgYA0KDQojIyBDb3JyZWxhdGlvbiBtYXRyaXgNCg0KV2UgY2FuIGFsc28gcHJlcGFyZSB0aGUgY29ycmVsYXRpb24gbWF0cml4IGZvciBhbGwgcXVhbnRpdGF0aXZlIHZhcmlhYmxlcyBzdG9yZWQgaW4gb3VyIGRhdGEgZnJhbWUuIA0KDQpXZSBjYW4gdXNlIGdnY29ycigpIGZ1bmN0aW9uOg0KDQpgYGB7ciBlY2hvPUZBTFNFLCB3YXJuaW5nPVRSVUV9DQoNCmlmICghcmVxdWlyZShHR2FsbHkpKSBpbnN0YWxsLnBhY2thZ2VzKCJHR2FsbHkiKQ0KbGlicmFyeShHR2FsbHkpDQoNCmNvcnJlbGF0aW9uX21hdHJpeCA8LSBnZ2NvcnIoYmFuaywgbGFiZWwgPSBUUlVFKQ0KcHJpbnQoY29ycmVsYXRpb25fbWF0cml4KQ0KDQoNCmBgYA0KICANCkFzIHlvdSBjYW4gc2VlIC0gdGhlIGRlZmF1bHQgY29ycmVsYXRpb24gbWF0cml4IGlzIG5vdCB0aGUgYmVzdCBpZGVhIGZvciBhbGwgbWVhc3VyZW1lbnQgc2NhbGVzIChpbmNsdWRpbmcgYmluYXJ5IHZhcmlhYmxlICJkZWZhdWx0IikuIA0KDQpUaGF0J3Mgd2h5IG5vdyB3ZSBjYW4gcGVyZm9ybSBvdXIgYml2YXJpYXRlIGFuYWx5c2lzIHdpdGggZ2dwYWlyIHdpdGggZ3JvdXBpbmcuDQoNCiMjIENvcnJlbGF0aW9uIG1hdHJpeCB3aXRoIHNjYXR0ZXJwbG90cyANCg0KSGVyZSBpcyB3aGF0IHdlIGFyZSBhYm91dCB0byBjYWxjdWxhdGU6DQotIFRoZSBjb3JyZWxhdGlvbiBtYXRyaXggYmV0d2VlbiBhZ2UsIGxvZ19pbmNvbWUsIGVtcGxveSwgYWRkcmVzcywgZGVidGluYywgY3JlZGRlYnQsIGFuZCBvdGhkZWJ0IHZhcmlhYmxlIGdyb3VwZWQgYnkgd2hldGhlciB0aGUgcGVyc29uIGhhcyBhIGRlZmF1bHQgc3RhdHVzIG9yIG5vdC4NCi0gUGxvdCB0aGUgZGlzdHJpYnV0aW9uIG9mIGVhY2ggdmFyaWFibGUgYnkgZ3JvdXANCi0gRGlzcGxheSB0aGUgc2NhdHRlciBwbG90IHdpdGggdGhlIHRyZW5kIGJ5IGdyb3VwDQoNCmBgYHtyIGVjaG89RkFMU0UsIHdhcm5pbmc9VFJVRX0NCg0KbGlicmFyeShHR2FsbHkpDQoNCnNlbGVjdGVkX2RhdGEgPC0gYmFua1ssIGMoImFnZSIsICJpbmNvbWUiLCAiZW1wbG95IiwgImFkZHJlc3MiLCAiZGVidGluYyIsICJjcmVkZGVidCIsICJvdGhkZWJ0IiwgImRlZmF1bHQiKV0NCnNlbGVjdGVkX2RhdGEkZGVmYXVsdCA8LSBhcy5mYWN0b3Ioc2VsZWN0ZWRfZGF0YSRkZWZhdWx0KQ0KDQojIEdlbmVyYXRlIGdncGFpcnMgcGxvdA0KZ2dwYWlycyhzZWxlY3RlZF9kYXRhLCBtYXBwaW5nID0gYWVzKGNvbG9yID0gZGVmYXVsdCksIA0KICAgICAgICB1cHBlciA9IGxpc3QoY29udGludW91cyA9IHdyYXAoImNvciIsIHNpemUgPSA0KSksDQogICAgICAgIGxvd2VyID0gbGlzdChjb250aW51b3VzID0gd3JhcCgic21vb3RoIiwgbWV0aG9kID0gImxtIiwgc2UgPSBUUlVFKSksDQogICAgICAgIGRpYWcgPSBsaXN0KGNvbnRpbnVvdXMgPSAiZGVuc2l0eURpYWciKQ0KKQ0KYGBgDQoNCg0KIyMgUXVhbGl0YXRpdmUgZGF0YQ0KDQpJbiBjYXNlIG9mIHR3byB2YXJpYWJsZXMgbWVhc3VyZWQgb24gbm9taW5hbCBvciBvcmRpbmFsJm5vbWluYWwgc2NhbGUgLSB3ZSBhcmUgZm9yY2VkIHRvIG9yZ2FuaXplIHNvIGNhbGxlZCAiY29udGluZ2VuY3kiIHRhYmxlIHdpdGggZnJlcXVlbmNpZXMgYW5kIGNhbGN1bGF0ZSBzb21lIGtpbmQgb2YgdGhlIGNvcnJlbGF0aW9uIGNvZWZmaWNpZW50IGJhc2VkIG9uIHRoZW0uIFRoaXMgaXMgc28gY2FsbGVkICJjb250aW5nZW5jeSBhbmFseXNpcyIuIA0KDQpMZXQncyBjb25zaWRlciBvbmUgZXhhbXBsZSBiYXNlZCBvbiBvdXIgZGF0YTogdmVyaWZ5LCBpZiB0aGVyZSBpcyBhbnkgc2lnbmlmaWNhbnQgY29ycmVsYXRpb24gYmV0d2VlbiBlZHVjYXRpb24gbGV2ZWwgYW5kIGNyZWRpdCByaXNrLg0KDQpgYGB7cn0NCg0KI2NvbnRpbmdlbmN5IHRhYmxlDQpjb250aW5nZW5jeV90YWJsZSA8LSB0YWJsZShiYW5rJGVkLCBiYW5rJGRlYnRpbmMpDQpwcmludChjb250aW5nZW5jeV90YWJsZSkNCg0KI2NoaS1zcXVhcmVkIHRlc3Qgb2YgaW5kZXBlbmRlbmNlDQpjaGlfc3F1YXJlZF90ZXN0IDwtIGNoaXNxLnRlc3QoY29udGluZ2VuY3lfdGFibGUpDQpwcmludChjaGlfc3F1YXJlZF90ZXN0KQ0KDQpgYGANCg0KDQojIyBFeGVyY2lzZSAxLiBDb250aW5nZW5jeSBhbmFseXNpcy4NCg0KRG8geW91IGJlbGlldmUgaW4gdGhlIEFmdGVybGlmZT8NCmh0dHBzOi8vbmF0aW9uYWxwb3N0LmNvbS9uZXdzL2NhbmFkYS9taWxsZW5uaWFscy1kby15b3UtYmVsaWV2ZS1pbi1saWZlLWFmdGVyLWxpZmUNCkEgc3VydmV5IHdhcyBjb25kdWN0ZWQgYW5kIGEgcmFuZG9tIHNhbXBsZSBvZiAxMDkxIHF1ZXN0aW9ubmFpcmVzIGlzIGdpdmVuIGluIHRoZSBmb3JtIG9mIHRoZSBmb2xsb3dpbmcgY29udGluZ2VuY3kgdGFibGU6DQoNCmBgYHtyIGVjaG89RkFMU0UsIHdhcm5pbmc9RkFMU0V9DQp4PWMoNDM1LDE0NywzNzUsMTM0KQ0KZGltKHgpPWMoMiwyKQ0KZGFuZTwtYXMudGFibGUoeCkNCmRpbW5hbWVzKGRhbmUpPWxpc3QoR2VuZGVyPWMoJ0ZlbWFsZScsJ01hbGUnKSxCZWxpZXZlPWMoJ1llcycsJ05vJykpDQpkYW5lDQpmb3VyZm9sZHBsb3QoZGFuZSkNCmBgYA0KDQpPdXIgdGFzayBpcyB0byBjaGVjayBpZiB0aGVyZSBpcyBhIHNpZ25pZmljYW50IHJlbGF0aW9uc2hpcCBiZXR3ZWVuIHRoZSBiZWxpZWYgaW4gdGhlIGFmdGVybGlmZSBhbmQgZ2VuZGVyLiBXZSBjYW4gcGVyZm9ybSB0aGlzIHByb2NlZHVyZSB3aXRoIHRoZSBzaW1wbGUgY2hpLXNxdWFyZSBzdGF0aXN0aWNzIGFuZCBjaG9zZW4gcXVhbGl0YXRpdmUgY29ycmVsYXRpb24gY29lZmZpY2llbnQgKHR3by13YXkgMngyIHRhYmxlKS4NCg0KYGBge3IgZWNobz1GQUxTRSwgd2FybmluZz1GQUxTRX0NCnllczwtYyg0MzUsMTQ3KQ0Kbm88LWMoMzc1LDEzNCkNCiNjb2hlbi5rYXBwYShjYmluZCh5ZXMsbm8pKQ0KY2hpc3EudGVzdChkYW5lKQ0KcHJvcC50YWJsZShkYW5lKQ0KYGBgDQoNCkFzIHlvdSBjYW4gc2VlIHdlIGNhbiBjYWxjdWxhdGUgb3VyIGNoaS1zcXVhcmUgc3RhdGlzdGljIHJlYWxseSBxdWlja2x5IGZvciB0d28td2F5IHRhYmxlcyBvciBsYXJnZXIuIA0KTm93IHdlIGNhbiBzdGFuZGFyZGl6ZSB0aGlzIGNvbnRpbmdlbmN5IG1lYXN1cmUgdG8gc2VlIGlmIHRoZSByZWxhdGlvbnNoaXAgaXMgc2lnbmlmaWNhbnQuDQoNCmBgYHtyIGVjaG89RkFMU0UsIHdhcm5pbmc9RkFMU0V9DQpQaGkoZGFuZSkNCiM/Q29udENvZWYNCiNDb250Q29lZihkYW5lKQ0KI0NyYW1lclYoZGFuZSkNCiNUc2NodXByb3dUKGRhbmUpDQptb3NhaWNwbG90KGRhbmUpDQpiYXJwbG90KGRhbmUpDQpgYGANCg0KDQojIyBFeGVyY2lzZSAyLiBDb250aW5nZW5jeSBhbmFseXNpcyBmb3IgdGhlICdUaXRhbmljJyBkYXRhLg0KDQpMZXQncyBjb25zaWRlciB0aGUgdGl0YW5pYyBkYXRhc2V0IHdoaWNoIGNvbnRhaW5zIGEgY29tcGxldGUgbGlzdCBvZiBwYXNzZW5nZXJzIGFuZCBjcmV3IG1lbWJlcnMgb24gdGhlIFJNUyBUaXRhbmljLiBJdCBpbmNsdWRlcyBhIHZhcmlhYmxlIGluZGljYXRpbmcgd2hldGhlciBhIHBlcnNvbiBkaWQgc3Vydml2ZSB0aGUgc2lua2luZyBvZiB0aGUgUk1TIFRpdGFuaWMgb24gQXByaWwgMTUsIDE5MTIuDQpBIGRhdGEgZnJhbWUgY29udGFpbnMgMjQ1NiBvYnNlcnZhdGlvbnMgb24gMTQgdmFyaWFibGVzLg0KDQpgYGB7ciBsb2FkLWRhdGEyLCB3YXJuaW5nPVRSVUUsIGluY2x1ZGU9RkFMU0V9DQpkb3dubG9hZC5maWxlKCJodHRwczovL2dpdGh1Yi5jb20va2ZsaXNpa293c2tpL2RzL2Jsb2IvbWFzdGVyL3RpdGFuaWMuY3N2P3Jhdz10cnVlIiwgZGVzdGZpbGUgPSJ0aXRhbmljLmNzdiIsbW9kZT0id2IiKQ0KdGl0YW5pYyA8LSByZWFkLmNzdigidGl0YW5pYy5jc3YiLHJvdy5uYW1lcz0xLHNlcD0iOyIpDQpgYGANCg0KVGhlIHdlYnNpdGUgaHR0cDovL3d3dy5lbmN5Y2xvcGVkaWEtdGl0YW5pY2Eub3JnLyBvZmZlcnMgZGV0YWlsZWQgaW5mb3JtYXRpb24gYWJvdXQgcGFzc2VuZ2VycyBhbmQgY3JldyBtZW1iZXJzIG9uIHRoZSBSTVMgVGl0YW5pYy4gQWNjb3JkaW5nIHRvIHRoZSB3ZWJzaXRlIDEzMTcgcGFzc2VuZ2VycyBhbmQgODkwIGNyZXcgbWVtYmVyIHdlcmUgYWJvYXJkLg0KDQo4IG11c2ljaWFucyBhbmQgOSBlbXBsb3llZXMgb2YgdGhlIHNoaXB5YXJkIGNvbXBhbnkgYXJlIGxpc3RlZCBhcyBwYXNzZW5nZXJzLCBidXQgdHJhdmVsbGVkIHdpdGggYSBmcmVlIHRpY2tldCwgd2hpY2ggaXMgd2h5IHRoZXkgaGF2ZSBOQSB2YWx1ZXMgaW4gZmFyZS4gSW4gYWRkaXRpb24gdG8gdGhhdCwgZmFyZSBpcyB0cnVlbHkgbWlzc2luZyBmb3IgYSBmZXcgcmVndWxhciBwYXNzZW5nZXJzLiANCg0KYGBge3J9DQoNCiMgQ2xlYW4gdGhlIGRhdGEgYnkgcmVwbGFjaW5nIGVtcHR5IHN0cmluZ3Mgd2l0aCBOQSBhbmQgZHJvcHBpbmcgYWxsIE5BIHZhbHVlcw0KdGl0YW5pY1t0aXRhbmljID09ICIiXSA8LSBOQQ0KdGl0YW5pYyA8LSBuYS5vbWl0KHRpdGFuaWMpDQoNCiMgUmVjb2RpbmcgdmFyaWFibGVzIGFzIGZhY3RvcnMgZm9yIGNhdGVnb3JpY2FsIGFuYWx5c2lzDQp0aXRhbmljJFN0YXR1cyA8LSBhcy5mYWN0b3IodGl0YW5pYyRTdGF0dXMpDQp0aXRhbmljJEdlbmRlciA8LSBhcy5mYWN0b3IodGl0YW5pYyRHZW5kZXIpDQp0aXRhbmljJEVtYmFya2VkIDwtIGFzLmZhY3Rvcih0aXRhbmljJEVtYmFya2VkKQ0KDQojIFBsb3QgdGhlIGRpc3RyaWJ1dGlvbiBvZiBBZ2UgYnkgU3Vydml2YWwgU3RhdHVzDQpsaWJyYXJ5KGdncGxvdDIpDQpnZ3Bsb3QodGl0YW5pYywgYWVzKHggPSBBZ2UsIGZpbGwgPSBTdGF0dXMpKSArIA0KICAgIGdlb21faGlzdG9ncmFtKGJpbndpZHRoID0gNSwgcG9zaXRpb24gPSAiZmlsbCIsIGNvbG9yID0gImJsYWNrIikgKw0KICAgIGxhYnMoeSA9ICJQcm9wb3J0aW9uIiwgeCA9ICJBZ2UiLCBmaWxsID0gIlN1cnZpdmFsIFN0YXR1cyIsIHRpdGxlID0gIlN1cnZpdmFsIGJ5IEFnZSBHcm91cCIpDQoNCiMgQ29udGluZ2VuY3kgdGFibGVzIGFuZCBDaGktc3F1YXJlZCB0ZXN0cw0KIyAxLiBDbGFzcyBhbmQgU3Vydml2YWwNCnRhYmxlX0NsYXNzX1N1cnZpdmVkIDwtIHRhYmxlKHRpdGFuaWMkQ2xhc3MsIHRpdGFuaWMkU3RhdHVzKQ0KY2hpX0NsYXNzX1N1cnZpdmVkIDwtIGNoaXNxLnRlc3QodGFibGVfQ2xhc3NfU3Vydml2ZWQpDQpwcmludCh0YWJsZV9DbGFzc19TdXJ2aXZlZCkNCnByaW50KGNoaV9DbGFzc19TdXJ2aXZlZCkNCg0KIyAyLiBHZW5kZXIgYW5kIFN1cnZpdmFsDQp0YWJsZV9HZW5kZXJfU3Vydml2ZWQgPC0gdGFibGUodGl0YW5pYyRHZW5kZXIsIHRpdGFuaWMkU3RhdHVzKQ0KY2hpX0dlbmRlcl9TdXJ2aXZlZCA8LSBjaGlzcS50ZXN0KHRhYmxlX0dlbmRlcl9TdXJ2aXZlZCkNCnByaW50KHRhYmxlX0dlbmRlcl9TdXJ2aXZlZCkNCnByaW50KGNoaV9HZW5kZXJfU3Vydml2ZWQpDQoNCiMgMy4gRW1iYXJrYXRpb24gcG9pbnQgYW5kIFN1cnZpdmFsDQp0YWJsZV9FbWJhcmtlZF9TdXJ2aXZlZCA8LSB0YWJsZSh0aXRhbmljJEVtYmFya2VkLCB0aXRhbmljJFN0YXR1cykNCmNoaV9FbWJhcmtlZF9TdXJ2aXZlZCA8LSBjaGlzcS50ZXN0KHRhYmxlX0VtYmFya2VkX1N1cnZpdmVkKQ0KcHJpbnQodGFibGVfRW1iYXJrZWRfU3Vydml2ZWQpDQpwcmludChjaGlfRW1iYXJrZWRfU3Vydml2ZWQpDQoNCmBgYA0KDQoNCg==