Link naar webpage/script die wordt gebruik in de kennisclips:

1 Inleiding

  • Tot nu toe: één uitkomst \(Y\) en één predictor \(X\).
  • Vaak handig om meerdere predictors te gebruiken om de respons te modelleren. bijv
  1. Associatie tussen X en Y verstoord door confounder: blootstelling aan asbest (X) op de longfunctie (Y ), is leeftijd (C).
  2. Welke van een groep variabelen beïnvloedt een gegeven uitkomst. Habitat en menselijke activiteit op biodiversiteit in het regenwoud. (grootte, ouderdom, hoogteligging van het woud \(\rightarrow\) bestudeer het simultane effect van die verschillende variabelen
  3. Voorspellen van uitkomst voor individuen: zoveel mogelijk predictieve informatie simultaan gebruiken. Verwante predicties (maar dan voor het risico op sterfte) worden dagdagelijks gebruikt in eenheden intensieve zorgen om de ernst van de gezondheidstoestand van een patiënt uit te drukken.

\(\rightarrow\) Uitbreiden van enkelvoudige lineaire regressie naar meerdere predictoren.


1.1 Prostaatkanker voorbeeld

  • Prostaat specific antigen (PSA) en een aantal klinische metingen bij 97 mannen waarvan de prostaat werd verwijderd.

  • Associatie van PSA i.f.v.

    • tumor volume (lcavol)
    • het gewicht van de prostaat (lweight)
    • leeftijd (age)
    • de goedaardige prostaathypertrofie hoeveelheid (lbph)
    • een indicator voor de aantasting van de zaadblaasjes (svi)
    • capsulaire penetratie (lcp)
    • Gleason score (gleason)
    • precentage gleason score 4/5 (pgg45)

prostate <- read_csv("https://raw.githubusercontent.com/statomics/sbc20/master/data/prostate.csv")
prostate
prostate$svi <- as.factor(prostate$svi)


2 Additief meervoudig lineaire regressie model

Afzonderlijke lineaire regressiemodellen, zoals

\[E(Y|X_v)=\beta_0 + \beta_v X_v\]

  • Associatie tussen lpsa en 1 variabele vb (lcavol).
  • Meer accurate predicties door meerdere predictoren simultaan in rekening te brengen
  • Schatting voor parameter \(\beta_v\) mogelijks geen zuiver effect van tumor volume.
  • \(\beta_v\) gemiddeld verschil in log-psa voor patiënten die 1 eenheid in het log tumor volume (lcavol) verschillen.
  • Zelfs als lcavol niet is geassocieerd met het lpsa, dan nog kunnen patiënten met een groter tumor volume een hoger lpsa hebben omdat ze bijvoorbeeld een aantasting van de zaadblaasjes hebben (svi status 1). \(\rightarrow\) Confounding.
  • Vergelijken van patiënten met zelfde svi status
  • Kan eenvoudig via meervoudige lineaire regressiemodellen

2.1 Statistisch model

  • \(p-1\) predictors \(X_1,...,X_{p-1}\) en uitkomst \(Y\) voor \(n\) subjecten
  • bijvoorbeeld p-1=3: log kanker volume (\(X_v\)), log gewicht van de prostaat (\(X_w\)) en status van de zaadblaasjes (\(X_s\))

\[\begin{equation} Y_i =\beta_0 + \beta_1 X_{i1} + ... +\beta_{p-1} X_{ip-1} + \epsilon_i \end{equation}\] \[\begin{equation} Y_i =\beta_0 + \beta_v X_{iv} +\beta_{w} X_{iw} \beta_{s}X_{is}+ \epsilon_i \end{equation}\]

  • \(\beta_0,\beta_1,...,\beta_{p-1}\) ongekende parameters
  • \(\epsilon_i\) residuen die niet verklaard kunnen worden door de predictors
  • Schatting met kleinste kwadraten techniek

Model staat toe om:

  1. de verwachte uitkomst te voorspellen voor subjecten gegeven hun waarden \(x_1,...,x_{p-1}\) voor de predictoren.

\[ E[Y\vert X_1=x_1, \ldots X_{p-1}=x_{p-1}]=\hat{\beta}_0+\hat{\beta}_1x_1+...+\hat{\beta}_{p-1}x_{p-1}\]

  1. Verschilt gemiddelde uitkomst tussen 2 groepen subjecten die \(\delta\) eenheden verschillen in een verklarende variabele \(X_j\) maar dezelfde waarden hebben voor alle andere variabelen \(\{X_k,k=1,...,p,k\ne j\}\).

\[ \begin{array}{l} E(Y|X_v=x_v+\delta,X_w=x_w,X_{s}=x_{s}) \\ \quad\quad - E(Y|X_v=x_v,X_w=x_w,X_{s}=x_{s}) \\\\ \quad =\beta_0 + \beta_v (x_v +\delta) + \beta_w x_w+ \beta_{s} x_{s}\\ \quad\quad- \beta_0 - \beta_v x_v - \beta_wx_w-\beta_{s} x_{s} \\\\ \quad= \beta_v\delta \end{array} \]

Interpretatie \(\beta_v\):

  • verschil in gemiddelde uitkomst tussen subjecten die in één eenheid van log tumor volume (\(X_v\)) verschillen, maar dezelfde waarde hebben voor de overige verklarende variabelen (\(X_w\) en \(X_s\)) in het model.

of

  • Effect van predictor log tumor volume waarbij gecorrigeerd wordt voor de overige predictoren, hier dus associatie van tumor volume na correctie voor prostaatgewicht en svi-status.

2.1.1 Prostate voorbeeld

lmV <- lm(lpsa ~ lcavol, prostate)
summary(lmV)

Call:
lm(formula = lpsa ~ lcavol, data = prostate)

Residuals:
     Min       1Q   Median       3Q      Max 
-1.67624 -0.41648  0.09859  0.50709  1.89672 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)    
(Intercept)  1.50730    0.12194   12.36   <2e-16 ***
lcavol       0.71932    0.06819   10.55   <2e-16 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 0.7875 on 95 degrees of freedom
Multiple R-squared:  0.5394,    Adjusted R-squared:  0.5346 
F-statistic: 111.3 on 1 and 95 DF,  p-value: < 2.2e-16

lmVWS <- lm(lpsa~lcavol + lweight + svi, prostate)
summary(lmVWS)

Call:
lm(formula = lpsa ~ lcavol + lweight + svi, data = prostate)

Residuals:
     Min       1Q   Median       3Q      Max 
-1.72966 -0.45767  0.02814  0.46404  1.57012 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)    
(Intercept) -0.26807    0.54350  -0.493  0.62301    
lcavol       0.55164    0.07467   7.388  6.3e-11 ***
lweight      0.50854    0.15017   3.386  0.00104 ** 
sviinvasion  0.66616    0.20978   3.176  0.00203 ** 
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 0.7168 on 93 degrees of freedom
Multiple R-squared:  0.6264,    Adjusted R-squared:  0.6144 
F-statistic: 51.99 on 3 and 93 DF,  p-value: < 2.2e-16

Na terug transformatie

lmVWS$coef %>% exp
(Intercept)      lcavol     lweight sviinvasion 
  0.7648524   1.7360954   1.6628548   1.9467442 


3 Besluitvorming in algemeen lineair regressiemodellen

Als gegevens representatief zijn dan zijn kleinste kwadraten schatters voor het intercept en de hellingen onvertekend. \[E[\hat \beta_j]=\beta_j,\quad j=0,\ldots,p-1.\]

  • Om resultaten uit de steekproef te kunnen veralgemenen naar de populatie is inzicht nodig in de verdeling van de parameterschatters.

  • Om dat op basis van slechts één steekproef te kunnen doen zijn bijkomende veronderstellingen nodig.

  1. Lineariteit

  2. Onafhankelijkheid

  3. Homoscedasticiteit of gelijke variantie

  4. Normaliteit: residuen \(\epsilon_i\) zijn normaal verdeeld

Onder deze aannames geldt: \[\epsilon_i \sim N(0,\sigma^2).\] en \[Y_i\sim N(\beta_0+\beta_1 X_{i1}+\ldots+\beta_{p-1} X_{ip-1},\sigma^2)\]


  • Hellingen zullen opnieuw nauwkeuriger worden geschat als de observaties meer gespreid zijn.

  • De conditionele variantie (\(\sigma^2\)) opnieuw schatten op basis van de mean squared error (MSE):

\[\hat\sigma^2=MSE=\frac{\sum\limits_{i=1}^n \left(y_i-\hat\beta_0-\hat\beta_1 X_{i1}-\ldots-\hat\beta_{p-1} X_{ip-1}\right)^2}{n-p}=\frac{\sum\limits_{i=1}^n e^2_i}{n-p}.\]

Opnieuw toetsen en betrouwbaarheidsintervallen via
\[T_k=\frac{\hat{\beta}_k-\beta_k}{SE(\hat{\beta}_k)} \text{ met } k=0, \ldots, p-1.\]

Als aan alle aannames is voldaan dan volgen deze statistieken \(T_k\) een t-verdeling met \(n-p\) vrijheidsgraden.


Wanneer niet is voldaan aan de veronderstelling van normaliteit maar wel aan lineariteit, onafhankelijkheid en homoscedasticiteit dan kunnen we voor inferentie opnieuw beroep doen op de centrale limietstelling die zegt dat de statistiek \(T_k\) bij benadering een standaard Normale verdeling zal volgen wanneer het aantal observaties voldoende groot is.


Voor het prostaatkanker voorbeeld kunnen we de effecten in de steekproef opnieuw veralgemenen naar de populatie toe door betrouwbaarheidsintervallen te bouwen voor de hellingen:

\[[\hat\beta_j - t_{n-p,\alpha/2} \text{SE}_{\hat\beta_j},\hat\beta_j + t_{n-p,\alpha/2} \text{SE}_{\hat\beta_j}]\].

confint(lmVWS)
                 2.5 %    97.5 %
(Intercept) -1.3473509 0.8112061
lcavol       0.4033628 0.6999144
lweight      0.2103288 0.8067430
sviinvasion  0.2495824 1.0827342

Formele hypothese testen: \[H_0: \beta_j=0\] \[H_1: \beta_j\neq0\]

met test statistiek \[T=\frac{\hat{\beta}_j-0}{SE(\hat{\beta}_j)}\] die een t-verdeling volgt met \(n-p\) vrijheidsgraden onder \(H_0\)


summary(lmVWS)

Call:
lm(formula = lpsa ~ lcavol + lweight + svi, data = prostate)

Residuals:
     Min       1Q   Median       3Q      Max 
-1.72966 -0.45767  0.02814  0.46404  1.57012 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)    
(Intercept) -0.26807    0.54350  -0.493  0.62301    
lcavol       0.55164    0.07467   7.388  6.3e-11 ***
lweight      0.50854    0.15017   3.386  0.00104 ** 
sviinvasion  0.66616    0.20978   3.176  0.00203 ** 
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 0.7168 on 93 degrees of freedom
Multiple R-squared:  0.6264,    Adjusted R-squared:  0.6144 
F-statistic: 51.99 on 3 and 93 DF,  p-value: < 2.2e-16

3.1 Nagaan van modelveronderstellingen

plot(lmVWS)


3.2 Het niet-additieve meervoudig lineair regressiemodel

3.2.1 Interactie tussen een continue variabele en een factor variabele

  • Het vorige model wordt het additief model genoemd omdat de bijdrage van het kanker volume in lpsa niet afhangt van de hoogte van het prostaat gewicht en de status van de zaadblaasjes.
  • De helling voor lcavol hangt m.a.w. niet af van de hoogte van het log prostaat gewicht en de status van de zaadblaasjes.

\[ \begin{array}{l} E[Y\vert X_v=x_v + \delta_v, X_w = x_w, X_S = x_s] - E[Y\vert X_v=x_v, X_w=x_w, X_s=x_s] = \\ \quad\quad \left[\beta_0 + \beta_v (x_{v}+\delta_v) + \beta_w x_{w} +\beta_s x_{s}\right] - \left[\beta_0 + \beta_v x_{v} + \beta_w x_{w} +\beta_s x_s\right] \\ \quad\quad = \beta_v \delta_v \end{array} \]

De svi status en de hoogte van het log-prostaatgewicht (\(x_w\)) heeft geen invloed op de bijdrage van het log-tumorvolume (\(x_v\)) in de gemiddelde log-prostaat antigeen concentratie en vice versa.


  • Het zou nu echter kunnen zijn dat de associatie tussen lpsa en lcavol wel afhangt van de status van de zaadblaasjes.

  • De gemiddelde toename in lpsa tussen patiënten die één eenheid van log-tumorvolume verschillen zou bijvoorbeeld lager kunnen zijn voor patiënten met aangetaste zaadblaasjes dan voor patienten met niet-aangetaste zaadblaasjes.

  • Het effect van het tumorvolume op de prostaat antigeen concentratie hangt in dit geval af van de status van de zaadblaasjes.

Om deze interactie of effectmodificatie tussen variabelen \(X_v\) en \(X_s\), en \(X_w\) en \(X_s\) statistisch te modelleren, kan men de producten van beide variabelen in kwestie aan het model toevoegen

\[ Y_i = \beta_0 + \beta_v x_{iv} + \beta_w x_{iw} +\beta_s x_{is} + \beta_{vs} x_{iv}x_{is} + \beta_{ws} x_{iw}x_{is} +\epsilon_i \]

Deze termen kwantificeren de interactie-effecten van respectievelijk de predictoren \(x_v\) en \(x_s\), en, \(x_v\) en \(x_s\) op de gemiddelde uitkomst.

In dit model worden de termen \(\beta_vx_{iv}\), \(\beta_wx_{iw}\) en $sx{is} dikwijls de hoofdeffecten van de predictoren \(x_v\), \(x_w\) en \(x_s\) genoemd.

lmVWS_IntVS_WS <- lm(
  lpsa ~
    lcavol +
    lweight +
    svi +
    svi:lcavol +
    svi:lweight,
  data = prostate)

summary(lmVWS_IntVS_WS)

Call:
lm(formula = lpsa ~ lcavol + lweight + svi + svi:lcavol + svi:lweight, 
    data = prostate)

Residuals:
     Min       1Q   Median       3Q      Max 
-1.50902 -0.44807  0.06455  0.45657  1.54354 

Coefficients:
                    Estimate Std. Error t value Pr(>|t|)    
(Intercept)         -0.52642    0.56793  -0.927 0.356422    
lcavol               0.54060    0.07821   6.912 6.38e-10 ***
lweight              0.58292    0.15699   3.713 0.000353 ***
sviinvasion          3.43653    1.93954   1.772 0.079771 .  
lcavol:sviinvasion   0.13467    0.25550   0.527 0.599410    
lweight:sviinvasion -0.82740    0.52224  -1.584 0.116592    
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 0.7147 on 91 degrees of freedom
Multiple R-squared:  0.6367,    Adjusted R-squared:  0.6167 
F-statistic: 31.89 on 5 and 91 DF,  p-value: < 2.2e-16

Omdat \(X_S\) een dummy variabele is, verkrijgen we verschillende regressievlakken:

  1. Model voor \(X_s=0\): \[Y=\beta_0+\beta_vX_v+\beta_wX_w + \epsilon\] waar de hoofdeffecten de hellingen voor lcavol en lweight zijn

  2. en het model voor \(X_s=1\): \[\begin{array}{lcl} Y&=&\beta_0+\beta_vX_v+\beta_s+\beta_wX_w+\beta_{vs}X_v + \beta_{ws}X_w +\epsilon\\ &=& (\beta_0+\beta_s)+(\beta_v+\beta_{vs})X_v+(\beta_w+\beta_{ws})X_w+\epsilon \end{array}\] met intercept \(\beta_0+\beta_s\) en hellingen \(\beta_v+\beta_{vs}\) en \(\beta_w+\beta_{ws}\)


3.2.2 Interactie tussen twee continue variabelen

  • Het zou nu echter kunnen zijn dat de associatie tussen lpsa en lcavol afhangt van het prostaatgewicht.
  • De gemiddelde toename in lpsa tussen patiënten die één eenheid van log-tumorvolume verschillen zou bijvoorbeeld lager kunnen zijn voor patiënten met een hoog prostaatgewicht dan bij patiënten met een laag prostaatgewicht.
  • Het effect van het tumorvolume op de prostaat antigeen concentratie hangt in dit geval af van het prostaatgewicht.

Om deze interactie of effectmodificatie tussen 2 variabelen \(X_v\) en \(X_w\) statistisch te modelleren, kan men het product van beide variabelen in kwestie aan het model toevoegen

\[ Y_i = \beta_0 + \beta_v x_{iv} + \beta_w x_{iw} +\beta_s x_{is} + \beta_{vw} x_{iv}x_{iw} +\epsilon_i \]

Deze term kwantificeert het interactie-effect van de predictoren \(x_v\) en \(x_w\) op de gemiddelde uitkomst.


Het effect van een verschil in 1 eenheid in \(X_v\) op de gemiddelde uitkomst bedraagt nu:

\[ \begin{array}{l} E(Y | X_v=x_v +1, X_w=x_w, X_s=x_s) - E(Y | X_v=x_v, X_w=x_w, X_s=x_s) \\ \quad = \left[\beta_0 + \beta_v (x_{v}+1) + \beta_w x_w +\beta_s x_{s} + \beta_{vw} (x_{v}+1) x_w \right] - \left[\beta_0 + \beta_v x_{v} + \beta_w x_w + \beta_s x_{s} + \beta_{vw} (x_{v}) x_w \right]\\ \quad = \beta_v + \beta_{vw} x_w \end{array} \]


lmVWS_IntVW <- lm(
  lpsa ~ lcavol +
    lweight +
    svi +
    lcavol:lweight,
  prostate)

summary(lmVWS_IntVW)

Call:
lm(formula = lpsa ~ lcavol + lweight + svi + lcavol:lweight, 
    data = prostate)

Residuals:
     Min       1Q   Median       3Q      Max 
-1.65886 -0.44673  0.02082  0.50244  1.57457 

Coefficients:
               Estimate Std. Error t value Pr(>|t|)   
(Intercept)     -0.6430     0.7030  -0.915  0.36278   
lcavol           1.0046     0.5427   1.851  0.06734 . 
lweight          0.6146     0.1961   3.134  0.00232 **
sviinvasion      0.6859     0.2114   3.244  0.00164 **
lcavol:lweight  -0.1246     0.1478  -0.843  0.40156   
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 0.7179 on 92 degrees of freedom
Multiple R-squared:  0.6293,    Adjusted R-squared:  0.6132 
F-statistic: 39.05 on 4 and 92 DF,  p-value: < 2.2e-16


  • Merk op, dat het interactie effect dat geobserveerd wordt in de steekproef echter statistisch niet significant is (p=0.4).

  • Gezien de hoofdeffecten die betrokken zijn in een interactie term niet los van elkaar kunnen worden geïnterpreteerd is de conventie om een interactieterm uit het model te verwijderen wanneer die niet significant is.

  • Na verwijdering van de niet-significante interactieterm kunnen de hoofdeffecten worden geïnterpreteerd.


3.2.3 Interactie tussen twee factor variabelen

Zie aparte presentatie over factoriële designs


4 ANOVA Tabel

De totale kwadratensom SSTot is opnieuw

\[ \text{SSTot} = \sum_{i=1}^n (Y_i - \bar{Y})^2. \]

Ook de residuele kwadratensom is zoals voorheen. \[ \text{SSE} = \sum_{i=1}^n (Y_i-\hat{Y}_i)^2. \]

Dan geldt de volgende decompositie van de totale kwadratensom, \[ \text{SSTot} = \text{SSR} + \text{SSE} , \] met \[ \text{SSR} = \sum_{i=1}^n (\hat{Y}_i-\bar{Y})^2. \]


Voor de vrijheidsgraden en de gemiddelde kwadratensommen geldt:

  • SSTot heeft \(n-1\) vrijheidsgraden en \(\text{SSTot}/(n-1)\) is een schatter voor de variantie van \(Y\) (van de marginale distributie van \(Y\)).
  • SSE heeft \(n-p\) vrijheidsgraden en \(\text{MSE}=\text{SSE}/(n-p)\) is een schatter voor de residuele variantie van \(Y\) gegeven de regressoren (i.e. een schatter voor de residuele variantie \(\sigma^2\) van de foutterm \(\epsilon\)).
  • SSR heeft \(p-1\) vrijheidsgraden en \(\text{MSR}=\text{SSR}/(p-1)\) is de gemiddelde kwadratensom van de regressie.

De determinatiecoëfficiënt blijft zoals voorheen, i.e. \[ R^2 = 1-\frac{\text{SSE}}{\text{SSTot}} = \frac{\text{SSR}}{\text{SSTot}} \] is de fractie van de totale variabiliteit in de uitkomsten die verklaard wordt door het regressiemodel.

De teststatistiek \(F=\text{MSR}/\text{MSE}\) is onder \(H_0:\beta_1=\ldots=\beta_{p-1}=0\) verdeeld als \(F_{p-1;n-p}\).



Call:
lm(formula = lpsa ~ lcavol + lweight + svi, data = prostate)

Residuals:
     Min       1Q   Median       3Q      Max 
-1.72966 -0.45767  0.02814  0.46404  1.57012 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)    
(Intercept) -0.26807    0.54350  -0.493  0.62301    
lcavol       0.55164    0.07467   7.388  6.3e-11 ***
lweight      0.50854    0.15017   3.386  0.00104 ** 
sviinvasion  0.66616    0.20978   3.176  0.00203 ** 
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 0.7168 on 93 degrees of freedom
Multiple R-squared:  0.6264,    Adjusted R-squared:  0.6144 
F-statistic: 51.99 on 3 and 93 DF,  p-value: < 2.2e-16

4.1 Extra Kwadratensommen

Beschouw de volgende twee regressiemodellen voor regressoren \(x_1\) en \(x_2\): \[ Y_i = \beta_0+\beta_1 x_{i1} + \epsilon_i, \] met \(\epsilon_i\text{ iid } N(0,\sigma_1^{2})\), en \[ Y_i = \beta_0+\beta_1 x_{i1}+\beta_2 x_{i2} + \epsilon_i, \] met \(\epsilon_i\text{ iid } N(0,\sigma_2^{2})\).

Voor het eerste (gereduceerde) model geldt de decompositie \[ \text{SSTot} = \text{SSR}_1 + \text{SSE}_1 \] en voor het tweede (niet-gereduceerde) model \[ \text{SSTot} = \text{SSR}_2 + \text{SSE}_2 \] (SSTot is uiteraard dezelfde in beide modellen omdat dit niet afhangt van het regressiemodel).


Definitie extra kwadratensom De extra kwadratensom (Engels: extra sum of squares) van predictor \(x_2\) t.o.v. het model met enkel \(x_1\) als predictor wordt gegeven door \[ \text{SSR}_{2\mid 1} = \text{SSE}_1-\text{SSE}_2=\text{SSR}_2-\text{SSR}_1. \]

Einde definitie

Merk eerst op dat \(\text{SSE}_1-\text{SSE}_2=\text{SSR}_2-\text{SSR}_1\) triviaal is gezien de decomposities van de totale kwadratensommen.

De extra kwadratensom \(\text{SSR}_{2\mid 1}\) kan eenvoudig geïnterpreteerd worden als de extra variantie van de uitkomst die verklaard kan worden door regressor \(x_2\) toe te voegen aan een model waarin regressor \(x_1\) reeds aanwezig is.

Met dit nieuw soort kwadratensom kunnen we voor het model met twee predictoren schrijven \[ \text{SSTot} = \text{SSR}_1+ \text{SSR}_{2\mid 1} + \text{SSE}. \] Dit volgt rechtstreeks uit de definitie van de extra kwadratensom \(\text{SSR}_{2\mid 1}\).


Uitbreiding: Zonder in te boeten in algemeenheid starten we met de regressiemodellen (\(s<p-1\)) \[ Y_i = \beta_0 + \beta_1 x_{i1} + \cdots + \beta_{s} x_{is} + \epsilon_i \] met \(\epsilon_i\text{ iid }N(0,\sigma_1^{2})\), en (\(s< q\leq p-1\)) \[ Y_i = \beta_0 + \beta_1 x_{i1} + \cdots + \beta_{s} x_{is} + \beta_{s+1} x_{is+1} + \cdots \beta_{q}x_{iq}+ \epsilon_i \] met \(\epsilon_i\text{ iid } N(0,\sigma_2^{2})\).

De extra kwadratensom van predictoren \(x_{s+1}, \ldots, x_q\) t.o.v. het model met enkel de predictoren \(x_1,\ldots, x_{s}\) wordt gegeven door \[ \text{SSR}_{s+1, \ldots, q\mid 1,\ldots, s} = \text{SSE}_1-\text{SSE}_2=\text{SSR}_2-\text{SSR}_1. \]


4.1.1 Type I Kwadratensommen

Stel dat \(p-1\) regressoren beschouwd worden, en beschouw een sequentie van modellen (\(s=2,\ldots, p-1\)) \[ Y_i = \beta_0 + \sum_{j=1}^{s} \beta_j x_{ij} + \epsilon_i \] met \(\epsilon_i\text{ iid } N(0,\sigma^{2})\).

  • De overeenkomstige kwadratensommen worden genoteerd als \(\text{SSR}_{s}\) en \(\text{SSE}_{s}\).
  • De modelsequentie geeft ook aanleiding tot extra kwadratensommen \(\text{SSR}_{s\mid 1,\ldots, s-1}\).
  • Deze laatste kwadratensom wordt een type I kwadratensom genoemd. Merk op dat deze afhangt van de volgorde (nummering) van regressoren.

Er kan aangetoond worden dat voor Model met \(s=p-1\) geldt \[ \text{SSTot} = \text{SSR}_1 + \text{SSR}_{2\mid 1} + \text{SSR}_{3\mid 1,2} + \cdots + \text{SSR}_{p-1\mid 1,\ldots, p-2} + \text{SSE}, \] met \(\text{SSE}\) de residuele kwadratensom van het model met alle \(p-1\) regressoren en \[ \text{SSR}_1 + \text{SSR}_{2\mid 1} + \text{SSR}_{3\mid 1,2} + \cdots + \text{SSR}_{p-1\mid 1,\ldots, p-2} = \text{SSR} \] met \(\text{SSR}\) de kwadratensom van de regressie van het model met alle \(p-1\) regressoren.

  • Interpretatie van iedere term afhangt van de volgorde van de regressoren in de sequentie van regressiemodellen.

  • Iedere type I SSR heeft betrekking op het effect van 1 regressor en heeft dus 1 vrijheidsgraad.
  • Voor iedere type I SSR term kan een gemiddelde kwadratensom gedefinieerd worden als \(\text{MSR}_{j\mid 1,\ldots, j-1}=\text{SSR}_{j\mid 1,\ldots, j-1}/1\).
  • De teststatistiek \(F=\text{MSR}_{j\mid 1,\ldots, j-1}/\text{MSE}\) is onder \(H_0:\beta_j=0\) met \(s=j\) verdeeld als \(F_{1;n-(j+1)}\).
  • Deze kwadratensommen worden standaard weergegeven door de anova functie in R.

4.1.2 Type III Kwadratensommen

De type III kwadratensom van regressor \(x_j\) wordt gegeven door de extra kwadratensom \[ \text{SSR}_{j \mid 1,\ldots, j-1,j+1,\ldots, p-1} = \text{SSE}_1-\text{SSE}_2 \]

  • \(\text{SSE}_2\) de residuele kwadratensom van regressiemodel met alle \(p-1\) regressoren.
  • \(\text{SSE}_1\) de residuele kwadratensom van regressiemodel met alle \(p-1\) regressoren, uitgezonderd regressor \(x_j\).

De type III kwadratensom \(\text{SSR}_{j \mid 1,\ldots, j-1,j+1,\ldots, p-1}\) kwantificeert dus het aandeel van de totale variantie van de uitkomst dat door regressor \(x_j\) verklaard wordt en dat niet door de andere \(p-2\) regressoren verklaard wordt.


De type III kwadratensom heeft ook 1 vrijheidsgraad omdat het om 1 \(\beta\)-parameter gaat.

Voor iedere type III SSR term kan een gemiddelde kwadratensom gedefinieerd worden als \(\text{MSR}_{j \mid 1,\ldots, j-1,j+1,\ldots, p-1}=\text{SSR}_{j \mid 1,\ldots, j-1,j+1,\ldots, p-1}/1\).

De teststatistiek \(F=\text{MSR}_{j \mid 1,\ldots, j-1,j+1,\ldots, p-1}/\text{MSE}\) is onder \(H_0:\beta_j=0\) verdeeld as \(F_{1;n-p}\).


library(car)
Anova(lmVWS,type=3)

p-waarden identiek aan die van tweezijdige t-testen

Merk echter op dat alle dummy’s voor factoren met meerdere niveaus in één keer uit het model worden gehaald. De type III sum of squares heeft dus evenveel vrijheidsgraden als het aantal dummy’s en er wordt een omnibustest uitgevoerd voor het effect van de factor.


5 Diagnostiek

5.1 Multicollineariteit


Call:
lm(formula = lpsa ~ lcavol + lweight + svi, data = prostate)

Residuals:
     Min       1Q   Median       3Q      Max 
-1.72966 -0.45767  0.02814  0.46404  1.57012 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)    
(Intercept) -0.26807    0.54350  -0.493  0.62301    
lcavol       0.55164    0.07467   7.388  6.3e-11 ***
lweight      0.50854    0.15017   3.386  0.00104 ** 
sviinvasion  0.66616    0.20978   3.176  0.00203 ** 
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 0.7168 on 93 degrees of freedom
Multiple R-squared:  0.6264,    Adjusted R-squared:  0.6144 
F-statistic: 51.99 on 3 and 93 DF,  p-value: < 2.2e-16


Call:
lm(formula = lpsa ~ lcavol + lweight + svi + lcavol:lweight, 
    data = prostate)

Residuals:
     Min       1Q   Median       3Q      Max 
-1.65886 -0.44673  0.02082  0.50244  1.57457 

Coefficients:
               Estimate Std. Error t value Pr(>|t|)   
(Intercept)     -0.6430     0.7030  -0.915  0.36278   
lcavol           1.0046     0.5427   1.851  0.06734 . 
lweight          0.6146     0.1961   3.134  0.00232 **
sviinvasion      0.6859     0.2114   3.244  0.00164 **
lcavol:lweight  -0.1246     0.1478  -0.843  0.40156   
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 0.7179 on 92 degrees of freedom
Multiple R-squared:  0.6293,    Adjusted R-squared:  0.6132 
F-statistic: 39.05 on 4 and 92 DF,  p-value: < 2.2e-16

  • Schattingen verschillend van additief model en standaardfouten zijn veel groter!
  • De oorzaak is probleem van multicollineariteit.
  • Als 2 predictoren sterk gecorreleerd zijn, dan delen ze voor een groot stuk dezelfde informatie
  • Moeilijk om de afzonderlijke effecten van beiden op de uitkomst te schatten.
  • Kleinste kwadratenschatters onstabiel wordt
  • Standaard errors kunnen worden opgeblazen
  • Zolang men enkel predicties tracht te bekomen op basis van het regressiemodel zonder daarbij te extrapoleren buiten het bereik van de predictoren is multicollineariteit geen probleem.
  • Wel probleem voor inferentie

cor(cbind(prostate$lcavol,prostate$lweight,prostate$lcavol*prostate$lweight))
          [,1]      [,2]      [,3]
[1,] 1.0000000 0.1941283 0.9893127
[2,] 0.1941283 1.0000000 0.2835608
[3,] 0.9893127 0.2835608 1.0000000
  • hoge correlatie tussen log-tumorvolume en interactieterm.
  • Is een gekend probleem voor hogere orde termen (interacties en kwadratische termen)

  • Multicollineariteit opsporen a.d.h.v. correlatie matrix of scatterplot matrix is niet ideaal.
  • Geen idee in welke mate de geobserveerde multicollineariteit de resultaten onstabiel maakt.
  • In modellen met 3 of meerdere predictoren, zeg X1, X2, X3 kan er zware multicollineariteit optreden ondanks dat alle paarsgewijze correlaties tussen de predictoren laag zijn.
  • Ook multicollineariteit als er een hoge correlatie is tussen X1 en een lineaire combinatie van X2 en X3.

5.1.1 Variance inflation factor (VIF)

Voor de \(j\)-de parameter in het regressiemodel gedefinieerd wordt als \[\textrm{VIF}_j=\left(1-R_j^2\right)^{-1}\]

  • In deze uitdrukking stelt \(R_j^2\) de meervoudige determinatiecoëfficiënt voor van een lineaire regressie van de \(j\)-de predictor op alle andere predictoren in het model.
  • VIF is 1 als \(j\)-de predictor niet lineair geassocieerd is met de andere predictoren in het model.
  • VIF is groter dan 1 in alle andere gevallen.
  • VIF is factor waarmee geobserveerde variantie groter is dan wanneer alle predictoren onafhankelijk zouden zijn.
  • VIF > 10 \(\rightarrow\) ernstige multicollineariteit.

5.1.2 Body fat voorbeeld



Call:
lm(formula = Body_fat ~ Triceps + Thigh + Midarm, data = bodyfat)

Residuals:
    Min      1Q  Median      3Q     Max 
-3.7263 -1.6111  0.3923  1.4656  4.1277 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)
(Intercept)  117.085     99.782   1.173    0.258
Triceps        4.334      3.016   1.437    0.170
Thigh         -2.857      2.582  -1.106    0.285
Midarm        -2.186      1.595  -1.370    0.190

Residual standard error: 2.48 on 16 degrees of freedom
Multiple R-squared:  0.8014,    Adjusted R-squared:  0.7641 
F-statistic: 21.52 on 3 and 16 DF,  p-value: 7.343e-06

vif(lmFat)
 Triceps    Thigh   Midarm 
708.8429 564.3434 104.6060 


Call:
lm(formula = Midarm ~ Triceps + Thigh, data = bodyfat)

Residuals:
     Min       1Q   Median       3Q      Max 
-0.58200 -0.30625  0.02592  0.29526  0.56102 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)    
(Intercept) 62.33083    1.23934   50.29   <2e-16 ***
Triceps      1.88089    0.04498   41.82   <2e-16 ***
Thigh       -1.60850    0.04316  -37.26   <2e-16 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 0.377 on 17 degrees of freedom
Multiple R-squared:  0.9904,    Adjusted R-squared:  0.9893 
F-statistic: 880.7 on 2 and 17 DF,  p-value: < 2.2e-16

We evalueren nu de VIF in het prostaatkanker voorbeeld voor het additieve model en het model met interactie.

vif(lmVWS)
  lcavol  lweight      svi 
1.447048 1.039188 1.409189 
vif(lmVWS_IntVW)
        lcavol        lweight            svi lcavol:lweight 
     76.193815       1.767121       1.426646      80.611657 
  • Inflatie voor interactietermen wordt vaak veroorzaakt door het feit dat het hoofdeffect een andere interpretatie krijgt.

5.2 Invloedrijke Observaties


  • Het is niet wenselijk dat een enkele waarneming het resultaat van een lineaire regressieanalyse grotendeels bepaald
  • Diagnostieken die ons toelaten om extreme observaties op te sporen
  • Studentized residu’s om outliers op te sporen
  • leverage (invloed, hefboom) om observaties met extreem covariaatpatroon op te sporen

5.2.1 Cook’s distance

  • Een meer rechtstreekse maat om de invloed van elke observatie op de regressie-analyse uit te drukken
  • Cook’s distance voor \(i\)-de observatie is een diagnostische maat voor de invloed van die observatie op alle predicties of voor haar invloed op alle geschatte parameters. \[D_i=\frac{\sum_{j=1}^n(\hat{Y}_j-\hat{Y}_{j(i)})^2}{p\textrm{MSE}}\]
  • Als Cook’s distance \(D_i\) groot is, dan heeft de \(i\)-de observatie een grote invloed op de predicties en geschatte parameters.
  • Extreme Cook’s distance als het het 50% percentiel van de \(F_{p,n-p}\)-verdeling overschrijdt.


  • Eenmaal men vastgesteld heeft dat een observatie invloedrijk is, kan men zogenaamde DFBETAS gebruiken om te bepalen op welke parameter(s) ze een grote invloed uitoefent.
  • DFBETAS van de \(i\)-de observatie vormen een diagnostische maat voor de invloed van die observatie op elke regressieparameter afzonderlijk \[\textrm{DFBETAS}_{j(i)}=\frac{\hat{\beta}_{j}-\hat{\beta}_{j(i)}}{\textrm{SD}(\hat{\beta}_{j})}\]
  • DFBETAS extreem is wanneer ze 1 overschrijdt in kleine tot middelgrote datasets en \(2/\sqrt{n}\) in grote datasets




6 Constrasten

  • Bij meer complexe algemene lineaire modellen wenst men dikwijls meerdere hypothesen te toetsen.
  • Bovendien vertalen de onderzoekshypotheses zich hierbij niet steeds in één parameter, maar in een lineaire combinatie van modelparameters.
  • Een lineaire combinatie van modelparameters wordt ook een contrast genoemd.

6.1 NHANES voorbeeld

  • Stel dat de onderzoekers de associatie tussen de leeftijd en de bloeddruk wensen te bestuderen. Mogelijks is die associatie anders is bij mannen dan vrouwen.

  • De onderzoekers wensen de volgende onderzoeksvragen te beantwoorden:

    • Is er een associatie tussen leeftijd en de bloeddruk bij vrouwen?
    • Is er een associatie tussen leeftijd en de bloeddruk bij mannen?
    • Is de associatie tussen leeftijd en de bloeddruk verschillend bij mannen dan bij vrouwen?

6.2 Model

We fitten een model op basis van de gemiddelde systolische bloeddruk (BPSysAve) in functie van de leeftijd, geslacht en een interactie tussen leeftijd en geslacht voor volwassen blanke subjecten uit de NHANES studie.

library(NHANES)
bpData <- NHANES %>%
filter(
  Race1 =="White" &
    Age >= 18 &
    !is.na(BPSysAve)
    )

mBp1 <- lm(BPSysAve ~ Age*Gender, bpData)
par(mfrow = c(2,2))
plot(mBp1)

  • Aannames van het model zijn niet voldaan!

    • lineariteit lijkt ok
    • heteroscedasticiteit
    • Geen normaliteit: scheve verdeling met staart naar rechts.
    • Grote dataset we kunnen steunen op de CLT

6.2.1 Transformatie

We fitten een model op basis van de log2 getransformeerde gemiddelde systolische bloeddruk (BPSysAve) in functie van de leeftijd, geslacht en een interactie tussen leeftijd en geslacht.

mBp2 <- lm(BPSysAve %>% log2 ~ Age*Gender, bpData)
par(mfrow = c(2,2))
plot(mBp2)

  • De residuen tonen nog steeds heteroscedasticiteit.

6.2.2 Remediëren voor heteroscedasticiteit

  • Als de plot van de residuen i.f.v. de geschatte waarden een toetervorm vertoont kan men toch correcte inferentie bekomen voor grote steekproeven als men de variantie van de response kan schatten.
  • De inverse variantie voor elke observatie kan dan als gewicht worden gebruikt in de lm functie.
  1. We zullen daarom de standard deviatie modelleren in functie van het gemiddelde.
  2. Dat kan door de absolute waarde van de residuen te modelleren in functie van de gefitte waarden.
  3. We kunnen de variantie van Y schatten voor elke observatie d.m.v de kwadraten van de predicties voor alle data punten a.d.h.v model voor de standard deviatie.
  4. De inferentie blijft asymptotisch geldig.
mSd <- lm(mBp1$res %>% abs ~ mBp2$fitted)

We schatten het model nu opnieuw:

mBp3 <- lm(BPSysAve ~ Age*Gender, bpData, w = 1/mSd$fitted^2)

De residuen vertonen nog steeds heteroscedasticiteit.

data.frame(residuals = mBp3$residuals, fit = mBp3$fitted) %>%
  ggplot(aes(fit,residuals)) +
  geom_point()

Na het herschalen van de residuen a.d.h.v. de standard deviatie (vermenigvuldigen met vierkantswortel van het gewicht) zijn de geschaalde residuen homoscedastisch.

De parameters worden geschat door de gewogen kleinste kwadraten techniek.

\[ SSE = \sum\limits_{i=1}^n w_i e_i^2\]

met \(w_i = 1/\hat \sigma^2_i\).

De gewogen regressie zal dus correct rekening houden met heteroscedasticiteit.

data.frame(scaled_residuals = mBp3$residuals/mSd$fitted, fit = mBp3$fitted) %>%
  ggplot(aes(fit,scaled_residuals)) +
  geom_point()

6.2.3 Besluitvorming

summary(mBp3)

Call:
lm(formula = BPSysAve ~ Age * Gender, data = bpData, weights = 1/mSd$fitted^2)

Weighted Residuals:
    Min      1Q  Median      3Q     Max 
-4.3642 -0.8494 -0.0940  0.7605  6.5701 

Coefficients:
               Estimate Std. Error t value Pr(>|t|)    
(Intercept)    97.59709    0.63501 153.693  < 2e-16 ***
Age             0.44082    0.01505  29.294  < 2e-16 ***
Gendermale     13.36724    1.09017  12.262  < 2e-16 ***
Age:Gendermale -0.19115    0.02420  -7.899 3.45e-15 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 1.319 on 4828 degrees of freedom
Multiple R-squared:  0.2182,    Adjusted R-squared:  0.2178 
F-statistic: 449.3 on 3 and 4828 DF,  p-value: < 2.2e-16

De onderzoeksvragen vertalen zich in de volgende nullhypotheses:

  1. Associatie tussen bloeddruk en leeftijd bij de vrouwen? \[H_0: \beta_\text{Age} = 0 \text{ vs } H_1: \beta_\text{Age} \neq 0 \]

  2. Associatie tussen bloeddruk en leeftijd bij de mannen? \[H_0: \beta_\text{Age} + \beta_\text{Age:Gendermale} = 0 \text{ vs } H_1: \beta_\text{Age} + \beta_\text{Age:Gendermale} \neq 0 \]

  3. Is de Associatie tussen bloeddruk en leeftijd verschillend bij mannen en vrouwen? \[H_0: \beta_\text{Age:Gendermale} = 0 \text{ vs } H_1: \beta_\text{Age:Gendermale} \neq 0 \]

  • We kunnen onderzoeksvraag 1 en 3 onmiddelijk toetsen o.b.v. de model output.
  • Onderzoeksvraag 2 is echter een lineaire combinatie van twee parameters.
  • Bovendien is er ook het probleem dat we meerdere toetsen nodig hebben om de associatie te bestuderen.

We kunnen opnieuw gebruik maken van een Anova approach.

  1. We toetsen eerste de omnibus hypothese dat er geen associatie is tussen leeftijd en de bloeddruk.

\[ H_0: \beta_\text{Age} = \beta_\text{Age} + \beta_\text{Age:Gendermale} = \beta_\text{Age:Gendermale} = 0 \]

  • Dat vereenvoudigt zich tot het toetsen dat

\[ H_0: \beta_\text{Age} = \beta_\text{Age:Gendermale} = 0 \]

  • Wat we kunnen evalueren door twee modellen te vergelijken. Een model met enkel het gender effect en volledige model met Gender, Age en Gender x Age interactie.
  1. Als we deze hypothese kunnen verwerpen voeren we posthoc analyses uit voor elk van de 3 contrasten.

6.2.3.1 Omnibus test

mBp0 <- lm(BPSysAve ~ Gender, bpData, w = 1/mSd$fitted^2)
anova(mBp0, mBp3)

6.2.3.2 Posthoc testen

De posthoc testen kunnen we opnieuw uitvoeren a.d.h.v. het multcomp pakket.

library(multcomp)
bpPosthoc <- glht(mBp3, linfct=c(
  "Age = 0",
  "Age + Age:Gendermale = 0",
  "Age:Gendermale = 0")
  )
bpPosthoc %>% summary

     Simultaneous Tests for General Linear Hypotheses

Fit: lm(formula = BPSysAve ~ Age * Gender, data = bpData, weights = 1/mSd$fitted^2)

Linear Hypotheses:
                          Estimate Std. Error t value Pr(>|t|)    
Age == 0                   0.44082    0.01505  29.294   <1e-10 ***
Age + Age:Gendermale == 0  0.24967    0.01895  13.175   <1e-10 ***
Age:Gendermale == 0       -0.19115    0.02420  -7.899   <1e-10 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
(Adjusted p values reported -- single-step method)
bpPosthocBI <- bpPosthoc %>% confint
bpPosthocBI

     Simultaneous Confidence Intervals

Fit: lm(formula = BPSysAve ~ Age * Gender, data = bpData, weights = 1/mSd$fitted^2)

Quantile = 2.3154
95% family-wise confidence level
 

Linear Hypotheses:
                          Estimate lwr     upr    
Age == 0                   0.4408   0.4060  0.4757
Age + Age:Gendermale == 0  0.2497   0.2058  0.2936
Age:Gendermale == 0       -0.1911  -0.2472 -0.1351

Merk op dat de glht functie ons toelaat om de contrasten te definiëren door de nulhypotheses expliciet te formuleren in een karaktervector waarbij gebruik wordt gemaakt van de naam van de pararameters in het model.

6.2.3.3 Conclusie

We kunnen besluiten dat er een extreem significante associatie is tussen leeftijd en de bloeddruk (p << 0.001). De bloeddruk bij twee vrouwen die in leeftijd verschillen is gemiddeld 0.44 mm Hg hoger per jaar leeftijdsverschil bij de oudste vrouw en dat verschil is extreem significant (p << 0.001, 95% BI [0.41, 0.48]. De bloeddruk bij mannen die in leeftijd verschillen is gemiddeld 0.25 mm Hg hoger per jaar leeftijdsverschil bij de oudere man. (p << 0.001, 95% BI [0.21, 0.29]. Het gemiddelde bloeddrukverschil tussen personen in leeftijd verschillen is gemiddeld -0.19 mm Hg/jaar hoger bij vrouwen dan mannen (p << 0.001, 95% BI [-0.25, -0.14]).

LS0tCnRpdGxlOiAiMTAuIEFsZ2VtZWVuIGxpbmVhaXIgbW9kZWwiCmF1dGhvcjogIkxpZXZlbiBDbGVtZW50IgpkYXRlOiAic3RhdE9taWNzLCBHaGVudCBVbml2ZXJzaXR5IChodHRwczovL3N0YXRvbWljcy5naXRodWIuaW8pIgpvdXRwdXQ6CiAgICBodG1sX2RvY3VtZW50OgogICAgICBjb2RlX2Rvd25sb2FkOiB0cnVlICAgIAogICAgICB0aGVtZTogY29zbW8KICAgICAgdG9jOiB0cnVlCiAgICAgIHRvY19mbG9hdDogdHJ1ZQogICAgICBoaWdobGlnaHQ6IHRhbmdvCiAgICAgIG51bWJlcl9zZWN0aW9uczogdHJ1ZQogICAgcGRmX2RvY3VtZW50OgogICAgICB0b2M6IHRydWUKICAgICAgbnVtYmVyX3NlY3Rpb25zOiB0cnVlCiAgICAgIGxhdGV4X2VuZ2luZTogeGVsYXRleAotLS0KCmBgYHtyIHNldHVwLCBpbmNsdWRlPUZBTFNFfQprbml0cjo6b3B0c19jaHVuayRzZXQoaW5jbHVkZSA9IFRSVUUsIGNvbW1lbnQgPSBOQSwgZWNobyA9IFRSVUUsCiAgICAgICAgICAgICAgICAgICAgICBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSkKbGlicmFyeSh0aWR5dmVyc2UpCmBgYAoKTGluayBuYWFyIHdlYnBhZ2Uvc2NyaXB0IGRpZSB3b3JkdCBnZWJydWlrIGluIGRlIGtlbm5pc2NsaXBzOgoKLSBbc2NyaXB0IEhvb2Zkc3R1ayAxMF0oaHR0cHM6Ly9zdGF0b21pY3MuZ2l0aHViLmlvL3NiYzIxL3JtZC8xMC1NdWx0aXBsZVJlZ3Jlc3Npb24uaHRtbCkKCgojIElubGVpZGluZwoKCjxpZnJhbWUgd2lkdGg9IjU2MCIgaGVpZ2h0PSIzMTUiCnNyYz0iaHR0cHM6Ly93d3cueW91dHViZS5jb20vZW1iZWQvM2x0Q1RsNHN0RUkiCmZyYW1lYm9yZGVyPSIwIgpzdHlsZT0iZGlzcGxheTogYmxvY2s7IG1hcmdpbjogYXV0bzsiCmFsbG93PSJhdXRvcGxheTsgZW5jcnlwdGVkLW1lZGlhIiBhbGxvd2Z1bGxzY3JlZW4+PC9pZnJhbWU+CgotIFRvdCBudSB0b2U6IMOpw6luIHVpdGtvbXN0ICRZJCBlbiDDqcOpbiBwcmVkaWN0b3IgJFgkLgotIFZhYWsgaGFuZGlnIG9tIG1lZXJkZXJlIHByZWRpY3RvcnMgdGUgZ2VicnVpa2VuIG9tIGRlIHJlc3BvbnMgdGUgbW9kZWxsZXJlbi4gYmlqdgoKMS4gQXNzb2NpYXRpZSB0dXNzZW4gWCBlbiBZIHZlcnN0b29yZCBkb29yIGNvbmZvdW5kZXI6IGJsb290c3RlbGxpbmcgYWFuIGFzYmVzdCAoWCkgb3AgZGUgbG9uZ2Z1bmN0aWUgKFkgKSwgaXMgbGVlZnRpamQgKEMpLgoyLiBXZWxrZSB2YW4gZWVuIGdyb2VwIHZhcmlhYmVsZW4gYmXDr252bG9lZHQgZWVuIGdlZ2V2ZW4gdWl0a29tc3QuIEhhYml0YXQgZW4gbWVuc2VsaWprZSBhY3Rpdml0ZWl0IG9wIGJpb2RpdmVyc2l0ZWl0IGluIGhldCByZWdlbndvdWQuIChncm9vdHRlLCBvdWRlcmRvbSwgaG9vZ3RlbGlnZ2luZyB2YW4gaGV0IHdvdWQgJFxyaWdodGFycm93JCBiZXN0dWRlZXIgaGV0IHNpbXVsdGFuZSBlZmZlY3QgdmFuIGRpZSB2ZXJzY2hpbGxlbmRlIHZhcmlhYmVsZW4KMy4gVm9vcnNwZWxsZW4gdmFuIHVpdGtvbXN0IHZvb3IgaW5kaXZpZHVlbjogem92ZWVsIG1vZ2VsaWprIHByZWRpY3RpZXZlIGluZm9ybWF0aWUgc2ltdWx0YWFuIGdlYnJ1aWtlbi4gVmVyd2FudGUgcHJlZGljdGllcyAobWFhciBkYW4gdm9vciBoZXQgcmlzaWNvIG9wIHN0ZXJmdGUpIHdvcmRlbiBkYWdkYWdlbGlqa3MgZ2VicnVpa3QgaW4gZWVuaGVkZW4gaW50ZW5zaWV2ZSB6b3JnZW4gb20gZGUgZXJuc3QgdmFuIGRlIGdlem9uZGhlaWRzdG9lc3RhbmQgdmFuIGVlbiBwYXRpw6tudCB1aXQgdGUgZHJ1a2tlbi4KCiRccmlnaHRhcnJvdyQgVWl0YnJlaWRlbiB2YW4gZW5rZWx2b3VkaWdlIGxpbmVhaXJlIHJlZ3Jlc3NpZSBuYWFyIG1lZXJkZXJlIHByZWRpY3RvcmVuLgoKLS0tCgojIyBQcm9zdGFhdGthbmtlciB2b29yYmVlbGQKCgo8aWZyYW1lIHdpZHRoPSI1NjAiIGhlaWdodD0iMzE1IgpzcmM9Imh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL2VtYmVkL1pkWjVYTGFEWmkwIgpmcmFtZWJvcmRlcj0iMCIKc3R5bGU9ImRpc3BsYXk6IGJsb2NrOyBtYXJnaW46IGF1dG87IgphbGxvdz0iYXV0b3BsYXk7IGVuY3J5cHRlZC1tZWRpYSIgYWxsb3dmdWxsc2NyZWVuPjwvaWZyYW1lPgoKLSBQcm9zdGFhdCBzcGVjaWZpYyBhbnRpZ2VuIChQU0EpIGVuIGVlbiBhYW50YWwga2xpbmlzY2hlIG1ldGluZ2VuIGJpaiA5NyBtYW5uZW4gd2FhcnZhbiBkZSBwcm9zdGFhdCB3ZXJkIHZlcndpamRlcmQuCi0gQXNzb2NpYXRpZSB2YW4gUFNBIGkuZi52LgoKICAgIC0gdHVtb3Igdm9sdW1lIChsY2F2b2wpCiAgICAtIGhldCBnZXdpY2h0IHZhbiBkZSBwcm9zdGFhdCAobHdlaWdodCkKICAgIC0gbGVlZnRpamQgKGFnZSkKICAgIC0gZGUgZ29lZGFhcmRpZ2UgcHJvc3RhYXRoeXBlcnRyb2ZpZSBob2V2ZWVsaGVpZCAobGJwaCkKICAgIC0gZWVuIGluZGljYXRvciB2b29yIGRlIGFhbnRhc3RpbmcgdmFuIGRlIHphYWRibGFhc2plcyAoc3ZpKQogICAgLSBjYXBzdWxhaXJlIHBlbmV0cmF0aWUgKGxjcCkKICAgIC0gR2xlYXNvbiBzY29yZSAoZ2xlYXNvbikKICAgIC0gcHJlY2VudGFnZSBnbGVhc29uIHNjb3JlIDQvNSAocGdnNDUpCgotLS0KCmBgYHtyfQpwcm9zdGF0ZSA8LSByZWFkX2NzdigiaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3N0YXRvbWljcy9zYmMyMC9tYXN0ZXIvZGF0YS9wcm9zdGF0ZS5jc3YiKQpwcm9zdGF0ZQpwcm9zdGF0ZSRzdmkgPC0gYXMuZmFjdG9yKHByb3N0YXRlJHN2aSkKYGBgCgotLS0KCmBgYHtyICwgb3V0LndpZHRoPScxMDAlJywgZmlnLmFzcD0uOCwgZmlnLmFsaWduPSdjZW50ZXInLGVjaG89RkFMU0V9CmxpYnJhcnkoR0dhbGx5KQpwcm9zdGF0ZSAlPiUKICBkcGx5cjo6c2VsZWN0KC1wZ2c0NSkgICU+JQogIGdncGFpcnMoKQpgYGAKCi0tLQoKIyBBZGRpdGllZiBtZWVydm91ZGlnIGxpbmVhaXJlIHJlZ3Jlc3NpZSBtb2RlbAoKCjxpZnJhbWUgd2lkdGg9IjU2MCIgaGVpZ2h0PSIzMTUiCnNyYz0iaHR0cHM6Ly93d3cueW91dHViZS5jb20vZW1iZWQvdkFUQUN0RDZrcFUiCmZyYW1lYm9yZGVyPSIwIgpzdHlsZT0iZGlzcGxheTogYmxvY2s7IG1hcmdpbjogYXV0bzsiCmFsbG93PSJhdXRvcGxheTsgZW5jcnlwdGVkLW1lZGlhIiBhbGxvd2Z1bGxzY3JlZW4+PC9pZnJhbWU+CgpBZnpvbmRlcmxpamtlIGxpbmVhaXJlIHJlZ3Jlc3NpZW1vZGVsbGVuLCB6b2FscwoKJCRFKFl8WF92KT1cYmV0YV8wICsgXGJldGFfdiBYX3YkJAoKLSBBc3NvY2lhdGllIHR1c3NlbiBscHNhIGVuIDEgdmFyaWFiZWxlIHZiIChsY2F2b2wpLgotIE1lZXIgYWNjdXJhdGUgcHJlZGljdGllcyBkb29yIG1lZXJkZXJlIHByZWRpY3RvcmVuIHNpbXVsdGFhbiBpbiByZWtlbmluZyB0ZSBicmVuZ2VuCi0gU2NoYXR0aW5nIHZvb3IgcGFyYW1ldGVyICRcYmV0YV92JCBtb2dlbGlqa3MgZ2VlbiB6dWl2ZXIgZWZmZWN0IHZhbiB0dW1vciB2b2x1bWUuCi0gJFxiZXRhX3YkIGdlbWlkZGVsZCB2ZXJzY2hpbCBpbiBsb2ctcHNhIHZvb3IgcGF0acOrbnRlbiBkaWUgMSBlZW5oZWlkIGluIGhldCBsb2cgdHVtb3Igdm9sdW1lIChsY2F2b2wpIHZlcnNjaGlsbGVuLgotIFplbGZzIGFscyBsY2F2b2wgbmlldCBpcyBnZWFzc29jaWVlcmQgbWV0IGhldCBscHNhLCAgZGFuIG5vZyBrdW5uZW4gcGF0acOrbnRlbiBtZXQgZWVuIGdyb3RlciB0dW1vciB2b2x1bWUgZWVuIGhvZ2VyIGxwc2EgaGViYmVuIG9tZGF0IHplIGJpanZvb3JiZWVsZCBlZW4gYWFudGFzdGluZyB2YW4gZGUgemFhZGJsYWFzamVzIGhlYmJlbiAoc3ZpIHN0YXR1cyAxKS4KJFxyaWdodGFycm93JCBDb25mb3VuZGluZy4KLSBWZXJnZWxpamtlbiB2YW4gcGF0acOrbnRlbiBtZXQgemVsZmRlIHN2aSBzdGF0dXMKLSBLYW4gZWVudm91ZGlnIHZpYSBtZWVydm91ZGlnZSBsaW5lYWlyZSByZWdyZXNzaWVtb2RlbGxlbgoKCi0tLQoKIyMgU3RhdGlzdGlzY2ggbW9kZWwKCi0gJHAtMSQgcHJlZGljdG9ycyAkWF8xLC4uLixYX3twLTF9JCBlbiB1aXRrb21zdCAkWSQgdm9vciAkbiQgc3ViamVjdGVuCi0gYmlqdm9vcmJlZWxkIHAtMT0zOiBsb2cga2Fua2VyIHZvbHVtZSAoJFhfdiQpLCBsb2cgZ2V3aWNodCB2YW4gZGUgcHJvc3RhYXQgKCRYX3ckKSBlbiBzdGF0dXMgdmFuIGRlIHphYWRibGFhc2plcyAoJFhfcyQpCgpcYmVnaW57ZXF1YXRpb259ICAKWV9pID1cYmV0YV8wICsgXGJldGFfMSBYX3tpMX0gKyAuLi4gK1xiZXRhX3twLTF9IFhfe2lwLTF9ICsgXGVwc2lsb25faQpcZW5ke2VxdWF0aW9ufQpcYmVnaW57ZXF1YXRpb259ICAKWV9pID1cYmV0YV8wICsgXGJldGFfdiBYX3tpdn0gK1xiZXRhX3t3fSBYX3tpd30gXGJldGFfe3N9WF97aXN9KyBcZXBzaWxvbl9pClxlbmR7ZXF1YXRpb259CgotICRcYmV0YV8wLFxiZXRhXzEsLi4uLFxiZXRhX3twLTF9JCBvbmdla2VuZGUgcGFyYW1ldGVycwotICRcZXBzaWxvbl9pJCByZXNpZHVlbiBkaWUgbmlldCB2ZXJrbGFhcmQga3VubmVuIHdvcmRlbiBkb29yIGRlIHByZWRpY3RvcnMKLSBTY2hhdHRpbmcgbWV0ICprbGVpbnN0ZSBrd2FkcmF0ZW4gdGVjaG5pZWsqCgotLS0KCk1vZGVsIHN0YWF0IHRvZSBvbToKCjEuIGRlIHZlcndhY2h0ZSB1aXRrb21zdCB0ZSB2b29yc3BlbGxlbiB2b29yIHN1YmplY3RlbiBnZWdldmVuIGh1biB3YWFyZGVuICR4XzEsLi4uLHhfe3AtMX0kIHZvb3IgZGUgcHJlZGljdG9yZW4uCgpcWyBFW1lcdmVydCBYXzE9eF8xLCBcbGRvdHMgWF97cC0xfT14X3twLTF9XT1caGF0e1xiZXRhfV8wK1xoYXR7XGJldGF9XzF4XzErLi4uK1xoYXR7XGJldGF9X3twLTF9eF97cC0xfVxdCgoyLiBWZXJzY2hpbHQgZ2VtaWRkZWxkZSB1aXRrb21zdCB0dXNzZW4gMiBncm9lcGVuIHN1YmplY3RlbiBkaWUgJFxkZWx0YSQgZWVuaGVkZW4gdmVyc2NoaWxsZW4gaW4gZWVuIHZlcmtsYXJlbmRlIHZhcmlhYmVsZSAkWF9qJCBtYWFyIGRlemVsZmRlIHdhYXJkZW4gaGViYmVuIHZvb3IgYWxsZSBhbmRlcmUgdmFyaWFiZWxlbiAkXHtYX2ssaz0xLC4uLixwLGtcbmUgalx9JC4KCiQkClxiZWdpbnthcnJheX17bH0KRShZfFhfdj14X3YrXGRlbHRhLFhfdz14X3csWF97c309eF97c30pIFxcClxxdWFkXHF1YWQgLSBFKFl8WF92PXhfdixYX3c9eF93LFhfe3N9PXhfe3N9KSBcXFxcClxxdWFkID1cYmV0YV8wICsgXGJldGFfdiAoeF92ICtcZGVsdGEpICsgIFxiZXRhX3cgeF93KyBcYmV0YV97c30geF97c31cXApccXVhZFxxdWFkLSBcYmV0YV8wIC0gXGJldGFfdiB4X3YgIC0gXGJldGFfd3hfdy1cYmV0YV97c30geF97c30gXFxcXApccXVhZD0gXGJldGFfdlxkZWx0YQpcZW5ke2FycmF5fQokJAoKSW50ZXJwcmV0YXRpZSAkXGJldGFfdiQ6CgotIHZlcnNjaGlsIGluIGdlbWlkZGVsZGUgdWl0a29tc3QgdHVzc2VuIHN1YmplY3RlbiBkaWUgaW4gw6nDqW4gZWVuaGVpZCB2YW4gbG9nIHR1bW9yIHZvbHVtZSAoJFhfdiQpIHZlcnNjaGlsbGVuLCBtYWFyIGRlemVsZmRlIHdhYXJkZSBoZWJiZW4gdm9vciBkZSBvdmVyaWdlIHZlcmtsYXJlbmRlIHZhcmlhYmVsZW4gKCRYX3ckIGVuICRYX3MkKSBpbiBoZXQgbW9kZWwuCgpvZgoKLSBFZmZlY3QgdmFuIHByZWRpY3RvciBsb2cgdHVtb3Igdm9sdW1lIHdhYXJiaWogZ2Vjb3JyaWdlZXJkIHdvcmR0IHZvb3IgZGUgb3ZlcmlnZSBwcmVkaWN0b3JlbiwgIGhpZXIgZHVzIGFzc29jaWF0aWUgdmFuIHR1bW9yIHZvbHVtZSBuYSBjb3JyZWN0aWUgdm9vciBwcm9zdGFhdGdld2ljaHQgZW4gc3ZpLXN0YXR1cy4KCi0tLQoKIyMjIFByb3N0YXRlIHZvb3JiZWVsZAoKCjxpZnJhbWUgd2lkdGg9IjU2MCIgaGVpZ2h0PSIzMTUiCnNyYz0iaHR0cHM6Ly93d3cueW91dHViZS5jb20vZW1iZWQvNUltLWxJQjRqR3MiCmZyYW1lYm9yZGVyPSIwIgpzdHlsZT0iZGlzcGxheTogYmxvY2s7IG1hcmdpbjogYXV0bzsiCmFsbG93PSJhdXRvcGxheTsgZW5jcnlwdGVkLW1lZGlhIiBhbGxvd2Z1bGxzY3JlZW4+PC9pZnJhbWU+CgoKYGBge3J9CmxtViA8LSBsbShscHNhIH4gbGNhdm9sLCBwcm9zdGF0ZSkKc3VtbWFyeShsbVYpCmBgYAoKLS0tCgpgYGB7cn0KbG1WV1MgPC0gbG0obHBzYX5sY2F2b2wgKyBsd2VpZ2h0ICsgc3ZpLCBwcm9zdGF0ZSkKc3VtbWFyeShsbVZXUykKYGBgCgpOYSB0ZXJ1ZyB0cmFuc2Zvcm1hdGllCmBgYHtyfQpsbVZXUyRjb2VmICU+JSBleHAKYGBgCgotLS0KCmBgYHtyIG91dC53aWR0aD0nODAlJywgZmlnLmFzcD0uOCwgZmlnLmFsaWduPSdjZW50ZXInLCBtZXNzYWdlPUZBTFNFLGVjaG89RkFMU0V9CmxpYnJhcnkocGxvdDNEKQpncmlkLmxpbmVzID0gMTAKeCA8LSBwcm9zdGF0ZSRsY2F2b2wKeSA8LSBwcm9zdGF0ZSRsd2VpZ2h0CnogPC0gcHJvc3RhdGUkbHBzYQpmaXQgPC0gbG0oen54K3krc3ZpLGRhdGE9cHJvc3RhdGUpCngucHJlZCA8LSBzZXEobWluKHgpLCBtYXgoeCksIGxlbmd0aC5vdXQgPSBncmlkLmxpbmVzKQp5LnByZWQgPC0gc2VxKG1pbih5KSwgbWF4KHkpLCBsZW5ndGgub3V0ID0gZ3JpZC5saW5lcykKCiMgZml0dGVkIHBvaW50cyBmb3IgZHJvcGxpbmVzIHRvIHN1cmZhY2UKdGg9MjAKcGg9NQpzY2F0dGVyM0QoeCwgeSwgeiwgcGNoID0gMTYsY29sPWMoImRhcmtibHVlIiwicmVkIilbYXMuZG91YmxlKHByb3N0YXRlJHN2aSldLCBjZXggPSAuNzUsCiAgICB0aGV0YSA9IHRoLCBwaGkgPSBwaCwgdGlja3R5cGUgPSAiZGV0YWlsZWQiLAogICAgeGxhYiA9ICJsY2F2b2wiLCB5bGFiID0gImx3ZWlnaHQiLCB6bGFiID0gImxwc2EiLCAgCiAgIGNvbHZhcj1GQUxTRSxidHkgPSAiZyIpCgpmb3IgKGkgaW4gMTpucm93KHByb3N0YXRlKSkKbGluZXMzRCh4PXJlcChwcm9zdGF0ZSRsY2F2b2xbaV0sMikseT1yZXAocHJvc3RhdGUkbHdlaWdodFtpXSwyKSx6PWMocHJvc3RhdGUkbHBzYVtpXSxsbVZXUyRmaXRbaV0pLGNvbD1jKCJkYXJrYmx1ZSIsInJlZCIpW2FzLmRvdWJsZShwcm9zdGF0ZSRzdmkpW2ldXSxhZGQ9VFJVRSxsdHk9MikKCnoucHJlZDNEIDwtIG91dGVyKHgucHJlZCwgeS5wcmVkLCBmdW5jdGlvbih4LHkpIHtsbVZXUyRjb2VmWzFdK2xtVldTJGNvZWZbMl0qeCtsbVZXUyRjb2VmWzNdKnl9KQp4LnByZWQzRCA8LSBvdXRlcih4LnByZWQseS5wcmVkLGZ1bmN0aW9uKHgseSkgeCkKeS5wcmVkM0QgPC0gb3V0ZXIoeC5wcmVkLHkucHJlZCxmdW5jdGlvbih4LHkpIHkpCnN1cmYzRCh4LnByZWQzRCx5LnByZWQzRCx6LnByZWQzRCxjb2w9ImJsdWUiLGZhY2V0cz1OQSxhZGQ9VFJVRSkKejIucHJlZDNEIDwtIG91dGVyKHgucHJlZCwgeS5wcmVkLCBmdW5jdGlvbih4LHkpIHtsbVZXUyRjb2VmWzFdK2xtVldTJGNvZWZbNF0rbG1WV1MkY29lZlsyXSp4K2xtVldTJGNvZWZbM10qeX0pCnN1cmYzRCh4LnByZWQzRCx5LnByZWQzRCx6Mi5wcmVkM0QsY29sPSJyZWQiLGZhY2V0cz1OQSxhZGQ9VFJVRSkKYGBgCgotLS0KCiMgQmVzbHVpdHZvcm1pbmcgaW4gYWxnZW1lZW4gbGluZWFpciByZWdyZXNzaWVtb2RlbGxlbgoKCjxpZnJhbWUgd2lkdGg9IjU2MCIgaGVpZ2h0PSIzMTUiCnNyYz0iaHR0cHM6Ly93d3cueW91dHViZS5jb20vZW1iZWQvNkFick12ZkdYVnciCmZyYW1lYm9yZGVyPSIwIgpzdHlsZT0iZGlzcGxheTogYmxvY2s7IG1hcmdpbjogYXV0bzsiCmFsbG93PSJhdXRvcGxheTsgZW5jcnlwdGVkLW1lZGlhIiBhbGxvd2Z1bGxzY3JlZW4+PC9pZnJhbWU+CgpBbHMgZ2VnZXZlbnMgcmVwcmVzZW50YXRpZWYgemlqbiBkYW4gemlqbiBrbGVpbnN0ZSBrd2FkcmF0ZW4gc2NoYXR0ZXJzIHZvb3IgaGV0IGludGVyY2VwdCBlbiBkZSBoZWxsaW5nZW4gb252ZXJ0ZWtlbmQuCiQkRVtcaGF0IFxiZXRhX2pdPVxiZXRhX2osXHF1YWQgaj0wLFxsZG90cyxwLTEuJCQKCi0gT20gcmVzdWx0YXRlbiB1aXQgZGUgc3RlZWtwcm9lZiB0ZSBrdW5uZW4gdmVyYWxnZW1lbmVuIG5hYXIgZGUgcG9wdWxhdGllIGlzIGluemljaHQgbm9kaWcgaW4gZGUgdmVyZGVsaW5nIHZhbiBkZSBwYXJhbWV0ZXJzY2hhdHRlcnMuCgotIE9tIGRhdCBvcCBiYXNpcyB2YW4gc2xlY2h0cyDDqcOpbiBzdGVla3Byb2VmIHRlIGt1bm5lbiBkb2VuIHppam4gYmlqa29tZW5kZSB2ZXJvbmRlcnN0ZWxsaW5nZW4gbm9kaWcuCgoKMS4gKkxpbmVhcml0ZWl0KgoKMi4gKk9uYWZoYW5rZWxpamtoZWlkKgoKMy4gKkhvbW9zY2VkYXN0aWNpdGVpdCogb2YgKmdlbGlqa2UgdmFyaWFudGllKgo0LiAqTm9ybWFsaXRlaXQqOiByZXNpZHVlbiAkXGVwc2lsb25faSQgemlqbiBub3JtYWFsIHZlcmRlZWxkCgpPbmRlciBkZXplIGFhbm5hbWVzIGdlbGR0OgokJFxlcHNpbG9uX2kgXHNpbSBOKDAsXHNpZ21hXjIpLiQkCmVuCiQkWV9pXHNpbSBOKFxiZXRhXzArXGJldGFfMSBYX3tpMX0rXGxkb3RzK1xiZXRhX3twLTF9IFhfe2lwLTF9LFxzaWdtYV4yKSQkCgotLS0KCgotIEhlbGxpbmdlbiB6dWxsZW4gb3BuaWV1dyBuYXV3a2V1cmlnZXIgd29yZGVuIGdlc2NoYXQgYWxzIGRlIG9ic2VydmF0aWVzIG1lZXIgZ2VzcHJlaWQgemlqbi4KCi0gRGUgY29uZGl0aW9uZWxlIHZhcmlhbnRpZSAoJFxzaWdtYV4yJCkgb3BuaWV1dyBzY2hhdHRlbiBvcCBiYXNpcyB2YW4gIGRlICptZWFuIHNxdWFyZWQgZXJyb3IqIChNU0UpOgoKJCRcaGF0XHNpZ21hXjI9TVNFPVxmcmFje1xzdW1cbGltaXRzX3tpPTF9Xm4gXGxlZnQoeV9pLVxoYXRcYmV0YV8wLVxoYXRcYmV0YV8xIFhfe2kxfS1cbGRvdHMtXGhhdFxiZXRhX3twLTF9IFhfe2lwLTF9XHJpZ2h0KV4yfXtuLXB9PVxmcmFje1xzdW1cbGltaXRzX3tpPTF9Xm4gZV4yX2l9e24tcH0uJCQKCk9wbmlldXcgdG9ldHNlbiBlbiBiZXRyb3V3YmFhcmhlaWRzaW50ZXJ2YWxsZW4gdmlhICAKJCRUX2s9XGZyYWN7XGhhdHtcYmV0YX1fay1cYmV0YV9rfXtTRShcaGF0e1xiZXRhfV9rKX0gXHRleHR7IG1ldCB9IGs9MCwgXGxkb3RzLCBwLTEuJCQKCkFscyBhYW4gYWxsZSBhYW5uYW1lcyBpcyB2b2xkYWFuIGRhbiB2b2xnZW4gZGV6ZSBzdGF0aXN0aWVrZW4gJFRfayQgZWVuIHQtdmVyZGVsaW5nIG1ldCAkbi1wJCB2cmlqaGVpZHNncmFkZW4uCgotLS0KCldhbm5lZXIgbmlldCBpcyB2b2xkYWFuIGFhbiBkZSB2ZXJvbmRlcnN0ZWxsaW5nIHZhbiBub3JtYWxpdGVpdCBtYWFyIHdlbCBhYW4gbGluZWFyaXRlaXQsIG9uYWZoYW5rZWxpamtoZWlkIGVuIGhvbW9zY2VkYXN0aWNpdGVpdCBkYW4ga3VubmVuIHdlIHZvb3IgaW5mZXJlbnRpZSBvcG5pZXV3IGJlcm9lcCBkb2VuIG9wIGRlIGNlbnRyYWxlIGxpbWlldHN0ZWxsaW5nIGRpZSB6ZWd0IGRhdCBkZSBzdGF0aXN0aWVrICRUX2skIGJpaiBiZW5hZGVyaW5nIGVlbiBzdGFuZGFhcmQgTm9ybWFsZSB2ZXJkZWxpbmcgemFsIHZvbGdlbiB3YW5uZWVyIGhldCBhYW50YWwgb2JzZXJ2YXRpZXMgdm9sZG9lbmRlIGdyb290IGlzLgoKLS0tCgpWb29yIGhldCBwcm9zdGFhdGthbmtlciB2b29yYmVlbGQga3VubmVuIHdlIGRlIGVmZmVjdGVuIGluIGRlIHN0ZWVrcHJvZWYgb3BuaWV1dyB2ZXJhbGdlbWVuZW4gbmFhciBkZSBwb3B1bGF0aWUgdG9lIGRvb3IgYmV0cm91d2JhYXJoZWlkc2ludGVydmFsbGVuIHRlIGJvdXdlbiB2b29yIGRlIGhlbGxpbmdlbjoKCiQkW1xoYXRcYmV0YV9qIC0gdF97bi1wLFxhbHBoYS8yfSBcdGV4dHtTRX1fe1xoYXRcYmV0YV9qfSxcaGF0XGJldGFfaiArIHRfe24tcCxcYWxwaGEvMn0gXHRleHR7U0V9X3tcaGF0XGJldGFfan1dJCQuCgpgYGB7cn0KY29uZmludChsbVZXUykKYGBgCgotLS0KCkZvcm1lbGUgaHlwb3RoZXNlIHRlc3RlbjoKJCRIXzA6IFxiZXRhX2o9MCQkCiQkSF8xOiBcYmV0YV9qXG5lcTAkJAoKbWV0IHRlc3Qgc3RhdGlzdGllawokJFQ9XGZyYWN7XGhhdHtcYmV0YX1fai0wfXtTRShcaGF0e1xiZXRhfV9qKX0kJApkaWUgZWVuIHQtdmVyZGVsaW5nIHZvbGd0IG1ldCAkbi1wJCB2cmlqaGVpZHNncmFkZW4gb25kZXIgJEhfMCQKCi0tLQoKYGBge3J9CnN1bW1hcnkobG1WV1MpCmBgYAoKLS0tCgojIyBOYWdhYW4gdmFuIG1vZGVsdmVyb25kZXJzdGVsbGluZ2VuCgpgYGB7cn0KcGxvdChsbVZXUykKYGBgCgotLS0KCgojIyBIZXQgbmlldC1hZGRpdGlldmUgbWVlcnZvdWRpZyBsaW5lYWlyIHJlZ3Jlc3NpZW1vZGVsCgo8aWZyYW1lIHdpZHRoPSI1NjAiIGhlaWdodD0iMzE1IgpzcmM9Imh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL2VtYmVkL2dkbTBfSGg3QWJjIgpmcmFtZWJvcmRlcj0iMCIKc3R5bGU9ImRpc3BsYXk6IGJsb2NrOyBtYXJnaW46IGF1dG87IgphbGxvdz0iYXV0b3BsYXk7IGVuY3J5cHRlZC1tZWRpYSIgYWxsb3dmdWxsc2NyZWVuPjwvaWZyYW1lPgoKIyMjIEludGVyYWN0aWUgdHVzc2VuIGVlbiBjb250aW51ZSB2YXJpYWJlbGUgZW4gZWVuIGZhY3RvciB2YXJpYWJlbGUKCi0gSGV0IHZvcmlnZSBtb2RlbCB3b3JkdCBoZXQgYWRkaXRpZWYgbW9kZWwgZ2Vub2VtZCBvbWRhdCBkZSBiaWpkcmFnZSB2YW4gaGV0IGthbmtlciB2b2x1bWUgaW4gbHBzYSBuaWV0IGFmaGFuZ3QgdmFuIGRlIGhvb2d0ZSB2YW4gaGV0IHByb3N0YWF0IGdld2ljaHQgZW4gZGUgc3RhdHVzIHZhbiBkZSB6YWFkYmxhYXNqZXMuCi0gRGUgaGVsbGluZyB2b29yIGxjYXZvbCBoYW5ndCBtLmEudy4gbmlldCBhZiB2YW4gZGUgaG9vZ3RlIHZhbiBoZXQgbG9nIHByb3N0YWF0IGdld2ljaHQgZW4gZGUgc3RhdHVzIHZhbiBkZSB6YWFkYmxhYXNqZXMuCgokJApcYmVnaW57YXJyYXl9e2x9CkVbWVx2ZXJ0IFhfdj14X3YgKyBcZGVsdGFfdiwgWF93ID0geF93LCBYX1MgPSB4X3NdIC0gRVtZXHZlcnQgWF92PXhfdiwgWF93PXhfdywgWF9zPXhfc10gPSBcXApccXVhZFxxdWFkIFxsZWZ0W1xiZXRhXzAgKyBcYmV0YV92ICh4X3t2fStcZGVsdGFfdikgKyBcYmV0YV93IHhfe3d9ICtcYmV0YV9zIHhfe3N9XHJpZ2h0XSAtIFxsZWZ0W1xiZXRhXzAgKyBcYmV0YV92IHhfe3Z9ICsgXGJldGFfdyB4X3t3fSArXGJldGFfcyB4X3NccmlnaHRdIFxcClxxdWFkXHF1YWQgPSBcYmV0YV92IFxkZWx0YV92ClxlbmR7YXJyYXl9CiQkCgpEZSBzdmkgc3RhdHVzIGVuIGRlIGhvb2d0ZSB2YW4gaGV0IGxvZy1wcm9zdGFhdGdld2ljaHQgKCR4X3ckKSBoZWVmdCBnZWVuIGludmxvZWQgb3AgZGUgYmlqZHJhZ2UgdmFuIGhldCBsb2ctdHVtb3J2b2x1bWUgKCR4X3YkKSBpbiBkZSBnZW1pZGRlbGRlIGxvZy1wcm9zdGFhdCBhbnRpZ2VlbiBjb25jZW50cmF0aWUgZW4gdmljZSB2ZXJzYS4KCgotLS0KCgotIEhldCB6b3UgbnUgZWNodGVyIGt1bm5lbiB6aWpuIGRhdCBkZSBhc3NvY2lhdGllIHR1c3NlbiBscHNhIGVuIGxjYXZvbCB3ZWwgYWZoYW5ndCB2YW4gZGUgc3RhdHVzIHZhbiBkZSB6YWFkYmxhYXNqZXMuCi0gRGUgZ2VtaWRkZWxkZSB0b2VuYW1lIGluIGxwc2EgdHVzc2VuIHBhdGnDq250ZW4gZGllIMOpw6luIGVlbmhlaWQgdmFuIGxvZy10dW1vcnZvbHVtZSB2ZXJzY2hpbGxlbiB6b3UgYmlqdm9vcmJlZWxkIGxhZ2VyIGt1bm5lbiB6aWpuIHZvb3IgcGF0acOrbnRlbiBtZXQgIGFhbmdldGFzdGUgemFhZGJsYWFzamVzIGRhbiB2b29yIHBhdGllbnRlbiBtZXQgbmlldC1hYW5nZXRhc3RlIHphYWRibGFhc2plcy4KCi0gSGV0IGVmZmVjdCB2YW4gaGV0IHR1bW9ydm9sdW1lIG9wIGRlIHByb3N0YWF0IGFudGlnZWVuIGNvbmNlbnRyYXRpZSBoYW5ndCBpbiBkaXQgZ2V2YWwgYWYgdmFuIGRlIHN0YXR1cyB2YW4gZGUgemFhZGJsYWFzamVzLgoKT20gZGV6ZSAqKmludGVyYWN0aWUqKiBvZiAqKmVmZmVjdG1vZGlmaWNhdGllKiogdHVzc2VuIHZhcmlhYmVsZW4gJFhfdiQgZW4gJFhfcyQsIGVuICRYX3ckIGVuICRYX3MkIHN0YXRpc3Rpc2NoIHRlIG1vZGVsbGVyZW4sIGthbiBtZW4gZGUgcHJvZHVjdGVuIHZhbiBiZWlkZSB2YXJpYWJlbGVuIGluIGt3ZXN0aWUgYWFuIGhldCBtb2RlbCB0b2V2b2VnZW4KCiQkCllfaSA9IFxiZXRhXzAgKyBcYmV0YV92IHhfe2l2fSArIFxiZXRhX3cgeF97aXd9ICtcYmV0YV9zIHhfe2lzfSArIFxiZXRhX3t2c30geF97aXZ9eF97aXN9ICsgXGJldGFfe3dzfSB4X3tpd314X3tpc30gICtcZXBzaWxvbl9pCiQkCgpEZXplIHRlcm1lbiBrd2FudGlmaWNlcmVuIGRlICppbnRlcmFjdGllLWVmZmVjdGVuKiB2YW4gcmVzcGVjdGlldmVsaWprIGRlIHByZWRpY3RvcmVuICR4X3YkIGVuICR4X3MkLCBlbiwgJHhfdiQgZW4gJHhfcyQgb3AgZGUgZ2VtaWRkZWxkZSB1aXRrb21zdC4KCkluIGRpdCBtb2RlbCB3b3JkZW4gZGUgdGVybWVuICRcYmV0YV92eF97aXZ9JCwgJFxiZXRhX3d4X3tpd30kIGVuICRcYmV0YV9zeF97aXN9IGRpa3dpamxzIGRlICpob29mZGVmZmVjdGVuKiB2YW4gZGUgcHJlZGljdG9yZW4gJHhfdiQsICR4X3ckIGVuICR4X3MkIGdlbm9lbWQuCgpgYGB7cn0KbG1WV1NfSW50VlNfV1MgPC0gbG0oCiAgbHBzYSB+CiAgICBsY2F2b2wgKwogICAgbHdlaWdodCArCiAgICBzdmkgKwogICAgc3ZpOmxjYXZvbCArCiAgICBzdmk6bHdlaWdodCwKICBkYXRhID0gcHJvc3RhdGUpCgpzdW1tYXJ5KGxtVldTX0ludFZTX1dTKQpgYGAKCi0tLQoKT21kYXQgJFhfUyQgZWVuIGR1bW15IHZhcmlhYmVsZSBpcywgdmVya3JpamdlbiB3ZSB2ZXJzY2hpbGxlbmRlIHJlZ3Jlc3NpZXZsYWtrZW46CgoxLiBNb2RlbCB2b29yICRYX3M9MCQ6ICQkWT1cYmV0YV8wK1xiZXRhX3ZYX3YrXGJldGFfd1hfdyArIFxlcHNpbG9uJCQgd2FhciBkZSBob29mZGVmZmVjdGVuIGRlIGhlbGxpbmdlbiB2b29yIGxjYXZvbCBlbiBsd2VpZ2h0IHppam4KCjIuIGVuIGhldCBtb2RlbCB2b29yICRYX3M9MSQ6CiAgICQkXGJlZ2lue2FycmF5fXtsY2x9CiAgIFkmPSZcYmV0YV8wK1xiZXRhX3ZYX3YrXGJldGFfcytcYmV0YV93WF93K1xiZXRhX3t2c31YX3YgKyBcYmV0YV97d3N9WF93ICtcZXBzaWxvblxcCiAgJj0mIChcYmV0YV8wK1xiZXRhX3MpKyhcYmV0YV92K1xiZXRhX3t2c30pWF92KyhcYmV0YV93K1xiZXRhX3t3c30pWF93K1xlcHNpbG9uCiAgXGVuZHthcnJheX0kJAptZXQgaW50ZXJjZXB0ICRcYmV0YV8wK1xiZXRhX3MkIGVuIGhlbGxpbmdlbiAkXGJldGFfditcYmV0YV97dnN9JCBlbiAkXGJldGFfdytcYmV0YV97d3N9JAoKLS0tCgpgYGB7ciBvdXQud2lkdGg9JzEwMCUnLCBmaWcuYXNwPS44LCBmaWcuYWxpZ249J2NlbnRlcicsIG1lc3NhZ2U9RkFMU0UsZWNobz1GQUxTRX0KcGFyKG1mcm93PWMoMSwyKSkKbGlicmFyeShwbG90M0QpCmdyaWQubGluZXMgPSAxMAp4PC1wcm9zdGF0ZSRsY2F2b2wKeTwtcHJvc3RhdGUkbHdlaWdodAp6PC1wcm9zdGF0ZSRscHNhCmZpdDwtbG0oen54K3krc3ZpLGRhdGE9cHJvc3RhdGUpCngucHJlZCA8LSBzZXEobWluKHgpLCBtYXgoeCksIGxlbmd0aC5vdXQgPSBncmlkLmxpbmVzKQp5LnByZWQgPC0gc2VxKG1pbih5KSwgbWF4KHkpLCBsZW5ndGgub3V0ID0gZ3JpZC5saW5lcykKCiMgZml0dGVkIHBvaW50cyBmb3IgZHJvcGxpbmVzIHRvIHN1cmZhY2UKdGg9LTI1CnBoPTUKc2NhdHRlcjNEKHgsIHksIHosIHBjaCA9IDE2LGNvbD1jKCJkYXJrYmx1ZSIsInJlZCIpW2FzLmRvdWJsZShwcm9zdGF0ZSRzdmkpXSwgY2V4ID0gLjc1LAogICAgdGhldGEgPSB0aCwgcGhpID0gcGgsIHRpY2t0eXBlID0gImRldGFpbGVkIiwKICAgIHhsYWIgPSAibGNhdm9sIiwgeWxhYiA9ICJsd2VpZ2h0IiwgemxhYiA9ICJscHNhIiwgIAogICBjb2x2YXI9RkFMU0UsYnR5ID0gImciLG1haW49IkFkZGl0aXZlIG1vZGVsIikKCmZvciAoaSBpbiB3aGljaChwcm9zdGF0ZSRzdmk9PSJoZWFsdGh5IikpCmxpbmVzM0QoeD1yZXAocHJvc3RhdGUkbGNhdm9sW2ldLDIpLHk9cmVwKHByb3N0YXRlJGx3ZWlnaHRbaV0sMiksej1jKHByb3N0YXRlJGxwc2FbaV0sbG1WV1MkZml0W2ldKSxjb2w9YygiZGFya2JsdWUiLCJyZWQiKVthcy5kb3VibGUocHJvc3RhdGUkc3ZpKVtpXV0sYWRkPVRSVUUsbHR5PTIpCgp6LnByZWQzRCA8LSBvdXRlcih4LnByZWQsIHkucHJlZCwgZnVuY3Rpb24oeCx5KSB7bG1WV1MkY29lZlsxXStsbVZXUyRjb2VmWzJdKngrbG1WV1MkY29lZlszXSp5fSkKeC5wcmVkM0QgPC0gb3V0ZXIoeC5wcmVkLHkucHJlZCxmdW5jdGlvbih4LHkpIHgpCnkucHJlZDNEIDwtIG91dGVyKHgucHJlZCx5LnByZWQsZnVuY3Rpb24oeCx5KSB5KQpzdXJmM0QoeC5wcmVkM0QseS5wcmVkM0Qsei5wcmVkM0QsY29sPSJibHVlIixmYWNldHM9TkEsYWRkPVRSVUUpCnoyLnByZWQzRCA8LSBvdXRlcih4LnByZWQsIHkucHJlZCwgZnVuY3Rpb24oeCx5KSB7bG1WV1MkY29lZlsxXStsbVZXUyRjb2VmWzRdK2xtVldTJGNvZWZbMl0qeCtsbVZXUyRjb2VmWzNdKnl9KQpzdXJmM0QoeC5wcmVkM0QseS5wcmVkM0QsejIucHJlZDNELGNvbD0ib3JhbmdlIixmYWNldHM9TkEsYWRkPVRSVUUpCgoKc2NhdHRlcjNEKHgsIHksIHosIHBjaCA9IDE2LGNvbD1jKCJkYXJrYmx1ZSIsInJlZCIpW2FzLmRvdWJsZShwcm9zdGF0ZSRzdmkpXSwgY2V4ID0gLjc1LAogICAgdGhldGEgPSB0aCwgcGhpID0gcGgsIHRpY2t0eXBlID0gImRldGFpbGVkIiwKICAgIHhsYWIgPSAibGNhdm9sIiwgeWxhYiA9ICJsd2VpZ2h0IiwgemxhYiA9ICJscHNhIiwgIAogICBjb2x2YXI9RkFMU0UsYnR5ID0gImciLG1haW49Ik1vZGVsIG1ldCBsY2F2b2w6bHdlaWdodCBpbnRlcmFjdGllIikKCmZvciAoaSBpbiB3aGljaChwcm9zdGF0ZSRzdmk9PSJoZWFsdGh5IikpCmxpbmVzM0QoeD1yZXAocHJvc3RhdGUkbGNhdm9sW2ldLDIpLHk9cmVwKHByb3N0YXRlJGx3ZWlnaHRbaV0sMiksej1jKHByb3N0YXRlJGxwc2FbaV0sbG1WV1NfSW50VlNfV1MkZml0W2ldKSxjb2w9YygiZGFya2JsdWUiLCJyZWQiKVthcy5kb3VibGUocHJvc3RhdGUkc3ZpKVtpXV0sYWRkPVRSVUUsbHR5PTIpCgp6LnByZWQzRCA8LSBvdXRlcih4LnByZWQsIHkucHJlZCwgZnVuY3Rpb24oeCx5KSB7bG1WV1NfSW50VlNfV1MkY29lZlsxXStsbVZXU19JbnRWU19XUyRjb2VmWzJdKngrbG1WV1NfSW50VlNfV1MkY29lZlszXSp5fSkKeC5wcmVkM0QgPC0gb3V0ZXIoeC5wcmVkLHkucHJlZCxmdW5jdGlvbih4LHkpIHgpCnkucHJlZDNEIDwtIG91dGVyKHgucHJlZCx5LnByZWQsZnVuY3Rpb24oeCx5KSB5KQpzdXJmM0QoeC5wcmVkM0QseS5wcmVkM0Qsei5wcmVkM0QsY29sPSJibHVlIixmYWNldHM9TkEsYWRkPVRSVUUpCnoyLnByZWQzRCA8LSBvdXRlcih4LnByZWQsIHkucHJlZCwgZnVuY3Rpb24oeCx5KSB7bG1WV1NfSW50VlNfV1MkY29lZlsxXStsbVZXU19JbnRWU19XUyRjb2VmWzRdK2xtVldTX0ludFZTX1dTJGNvZWZbMl0qeCtsbVZXU19JbnRWU19XUyRjb2VmWzNdKnkrbG1WV1NfSW50VlNfV1MkY29lZls1XSp4K2xtVldTX0ludFZTX1dTJGNvZWZbNl0qeX0pCnN1cmYzRCh4LnByZWQzRCx5LnByZWQzRCx6Mi5wcmVkM0QsY29sPSJvcmFuZ2UiLGZhY2V0cz1OQSxhZGQ9VFJVRSkKYGBgCgojIyMgSW50ZXJhY3RpZSB0dXNzZW4gdHdlZSBjb250aW51ZSB2YXJpYWJlbGVuCgoKPGlmcmFtZSB3aWR0aD0iNTYwIiBoZWlnaHQ9IjMxNSIKc3JjPSJodHRwczovL3d3dy55b3V0dWJlLmNvbS9lbWJlZC9MV0JYUzRtWWJNSSIKZnJhbWVib3JkZXI9IjAiCnN0eWxlPSJkaXNwbGF5OiBibG9jazsgbWFyZ2luOiBhdXRvOyIKYWxsb3c9ImF1dG9wbGF5OyBlbmNyeXB0ZWQtbWVkaWEiIGFsbG93ZnVsbHNjcmVlbj48L2lmcmFtZT4KCgotIEhldCB6b3UgbnUgZWNodGVyIGt1bm5lbiB6aWpuIGRhdCBkZSBhc3NvY2lhdGllIHR1c3NlbiBscHNhIGVuIGxjYXZvbCBhZmhhbmd0IHZhbiBoZXQgcHJvc3RhYXRnZXdpY2h0LgotIERlIGdlbWlkZGVsZGUgdG9lbmFtZSBpbiBscHNhIHR1c3NlbiBwYXRpw6tudGVuIGRpZSDDqcOpbiBlZW5oZWlkIHZhbiBsb2ctdHVtb3J2b2x1bWUgdmVyc2NoaWxsZW4gem91IGJpanZvb3JiZWVsZCBsYWdlciBrdW5uZW4gemlqbiB2b29yIHBhdGnDq250ZW4gbWV0IGVlbiBob29nIHByb3N0YWF0Z2V3aWNodCBkYW4gYmlqIHBhdGnDq250ZW4gbWV0IGVlbiBsYWFnIHByb3N0YWF0Z2V3aWNodC4KLSBIZXQgZWZmZWN0IHZhbiBoZXQgdHVtb3J2b2x1bWUgb3AgZGUgcHJvc3RhYXQgYW50aWdlZW4gY29uY2VudHJhdGllIGhhbmd0IGluIGRpdCBnZXZhbCBhZiB2YW4gaGV0IHByb3N0YWF0Z2V3aWNodC4KCk9tIGRlemUgKippbnRlcmFjdGllKiogb2YgKiplZmZlY3Rtb2RpZmljYXRpZSoqIHR1c3NlbiAyIHZhcmlhYmVsZW4gJFhfdiQgZW4gJFhfdyQgc3RhdGlzdGlzY2ggdGUgbW9kZWxsZXJlbiwga2FuIG1lbiBoZXQgcHJvZHVjdCB2YW4gYmVpZGUgdmFyaWFiZWxlbiBpbiBrd2VzdGllIGFhbiBoZXQgbW9kZWwgdG9ldm9lZ2VuCgokJApZX2kgPSBcYmV0YV8wICsgXGJldGFfdiB4X3tpdn0gKyBcYmV0YV93IHhfe2l3fSArXGJldGFfcyB4X3tpc30gKyBcYmV0YV97dnd9IHhfe2l2fXhfe2l3fSArXGVwc2lsb25faQokJAoKRGV6ZSB0ZXJtIGt3YW50aWZpY2VlcnQgaGV0ICppbnRlcmFjdGllLWVmZmVjdCogdmFuIGRlIHByZWRpY3RvcmVuICR4X3YkIGVuICR4X3ckIG9wIGRlIGdlbWlkZGVsZGUgdWl0a29tc3QuCgotLS0KCkhldCBlZmZlY3QgdmFuIGVlbiB2ZXJzY2hpbCBpbiAxIGVlbmhlaWQgaW4gJFhfdiQgb3AgZGUgZ2VtaWRkZWxkZSB1aXRrb21zdCBiZWRyYWFndCBudToKCiQkClxiZWdpbnthcnJheX17bH0KRShZIHwgWF92PXhfdiArMSwgWF93PXhfdywgWF9zPXhfcykgLSBFKFkgfCBYX3Y9eF92LCBYX3c9eF93LCBYX3M9eF9zKSBcXApccXVhZCA9IFxsZWZ0W1xiZXRhXzAgKyBcYmV0YV92ICh4X3t2fSsxKSArIFxiZXRhX3cgeF93ICtcYmV0YV9zIHhfe3N9ICsgXGJldGFfe3Z3fSAoeF97dn0rMSkgeF93IFxyaWdodF0gLSBcbGVmdFtcYmV0YV8wICsgXGJldGFfdiB4X3t2fSArIFxiZXRhX3cgeF93ICArIFxiZXRhX3MgeF97c30gKyBcYmV0YV97dnd9ICh4X3t2fSkgeF93IFxyaWdodF1cXApccXVhZCA9IFxiZXRhX3YgKyAgXGJldGFfe3Z3fSB4X3cKIFxlbmR7YXJyYXl9CiAkJAoKCi0tLQoKYGBge3J9CmxtVldTX0ludFZXIDwtIGxtKAogIGxwc2EgfiBsY2F2b2wgKwogICAgbHdlaWdodCArCiAgICBzdmkgKwogICAgbGNhdm9sOmx3ZWlnaHQsCiAgcHJvc3RhdGUpCgpzdW1tYXJ5KGxtVldTX0ludFZXKQpgYGAKCi0tLQoKCmBgYHtyIG91dC53aWR0aD0nMTAwJScsIGZpZy5hc3A9LjgsIGZpZy5hbGlnbj0nY2VudGVyJywgbWVzc2FnZT1GQUxTRSxlY2hvPUZBTFNFfQpwYXIobWZyb3c9YygxLDIpKQpsaWJyYXJ5KHBsb3QzRCkKZ3JpZC5saW5lcyA9IDEwCng8LXByb3N0YXRlJGxjYXZvbAp5PC1wcm9zdGF0ZSRsd2VpZ2h0Cno8LXByb3N0YXRlJGxwc2EKZml0PC1sbSh6fngreStzdmksZGF0YT1wcm9zdGF0ZSkKeC5wcmVkIDwtIHNlcShtaW4oeCksIG1heCh4KSwgbGVuZ3RoLm91dCA9IGdyaWQubGluZXMpCnkucHJlZCA8LSBzZXEobWluKHkpLCBtYXgoeSksIGxlbmd0aC5vdXQgPSBncmlkLmxpbmVzKQoKIyBmaXR0ZWQgcG9pbnRzIGZvciBkcm9wbGluZXMgdG8gc3VyZmFjZQp0aD0tMjUKcGg9NQpzY2F0dGVyM0QoeCwgeSwgeiwgcGNoID0gMTYsY29sPWMoImRhcmtibHVlIiwicmVkIilbYXMuZG91YmxlKHByb3N0YXRlJHN2aSldLCBjZXggPSAuNzUsCiAgICB0aGV0YSA9IHRoLCBwaGkgPSBwaCwgdGlja3R5cGUgPSAiZGV0YWlsZWQiLAogICAgeGxhYiA9ICJsY2F2b2wiLCB5bGFiID0gImx3ZWlnaHQiLCB6bGFiID0gImxwc2EiLCAgCiAgIGNvbHZhcj1GQUxTRSxidHkgPSAiZyIsbWFpbj0iQWRkaXRpdmUgbW9kZWwiKQoKZm9yIChpIGluIHdoaWNoKHByb3N0YXRlJHN2aT09ImhlYWx0aHkiKSkKbGluZXMzRCh4PXJlcChwcm9zdGF0ZSRsY2F2b2xbaV0sMikseT1yZXAocHJvc3RhdGUkbHdlaWdodFtpXSwyKSx6PWMocHJvc3RhdGUkbHBzYVtpXSxsbVZXUyRmaXRbaV0pLGNvbD1jKCJkYXJrYmx1ZSIsInJlZCIpW2FzLmRvdWJsZShwcm9zdGF0ZSRzdmkpW2ldXSxhZGQ9VFJVRSxsdHk9MikKCnoucHJlZDNEIDwtIG91dGVyKHgucHJlZCwgeS5wcmVkLCBmdW5jdGlvbih4LHkpIHtsbVZXUyRjb2VmWzFdK2xtVldTJGNvZWZbMl0qeCtsbVZXUyRjb2VmWzNdKnl9KQp4LnByZWQzRCA8LSBvdXRlcih4LnByZWQseS5wcmVkLGZ1bmN0aW9uKHgseSkgeCkKeS5wcmVkM0QgPC0gb3V0ZXIoeC5wcmVkLHkucHJlZCxmdW5jdGlvbih4LHkpIHkpCnN1cmYzRCh4LnByZWQzRCx5LnByZWQzRCx6LnByZWQzRCxjb2w9ImJsdWUiLGZhY2V0cz1OQSxhZGQ9VFJVRSkKCgpzY2F0dGVyM0QoeCwgeSwgeiwgcGNoID0gMTYsY29sPWMoImRhcmtibHVlIiwicmVkIilbYXMuZG91YmxlKHByb3N0YXRlJHN2aSldLCBjZXggPSAuNzUsCiAgICB0aGV0YSA9IHRoLCBwaGkgPSBwaCwgdGlja3R5cGUgPSAiZGV0YWlsZWQiLAogICAgeGxhYiA9ICJsY2F2b2wiLCB5bGFiID0gImx3ZWlnaHQiLCB6bGFiID0gImxwc2EiLCAgCiAgIGNvbHZhcj1GQUxTRSxidHkgPSAiZyIsbWFpbj0iTW9kZWwgbWV0IGxjYXZvbDpsd2VpZ2h0IGludGVyYWN0aWUiKQoKZm9yIChpIGluIHdoaWNoKHByb3N0YXRlJHN2aT09ImhlYWx0aHkiKSkKbGluZXMzRCh4PXJlcChwcm9zdGF0ZSRsY2F2b2xbaV0sMikseT1yZXAocHJvc3RhdGUkbHdlaWdodFtpXSwyKSx6PWMocHJvc3RhdGUkbHBzYVtpXSxsbVZXU19JbnRWVyRmaXRbaV0pLGNvbD1jKCJkYXJrYmx1ZSIsInJlZCIpW2FzLmRvdWJsZShwcm9zdGF0ZSRzdmkpW2ldXSxhZGQ9VFJVRSxsdHk9MikKCnoucHJlZDNEIDwtIG91dGVyKHgucHJlZCwgeS5wcmVkLCBmdW5jdGlvbih4LHkpIHtsbVZXU19JbnRWVyRjb2VmWzFdK2xtVldTX0ludFZXJGNvZWZbMl0qeCtsbVZXU19JbnRWVyRjb2VmWzNdKnkrbG1WV1NfSW50VlckY29lZls1XSp4Knl9KQp4LnByZWQzRCA8LSBvdXRlcih4LnByZWQseS5wcmVkLGZ1bmN0aW9uKHgseSkgeCkKeS5wcmVkM0QgPC0gb3V0ZXIoeC5wcmVkLHkucHJlZCxmdW5jdGlvbih4LHkpIHkpCnN1cmYzRCh4LnByZWQzRCx5LnByZWQzRCx6LnByZWQzRCxjb2w9ImJsdWUiLGZhY2V0cz1OQSxhZGQ9VFJVRSkKYGBgCgotLS0KCi0gTWVyayBvcCwgZGF0IGhldCBpbnRlcmFjdGllIGVmZmVjdCBkYXQgZ2VvYnNlcnZlZXJkIHdvcmR0IGluIGRlIHN0ZWVrcHJvZWYgZWNodGVyIHN0YXRpc3Rpc2NoIG5pZXQgc2lnbmlmaWNhbnQgaXMgKHA9YHIgcm91bmQoc3VtbWFyeShsbVZXU19JbnRWVykkY29lZls1LDRdLDIpYCkuCgotIEdlemllbiBkZSBob29mZGVmZmVjdGVuIGRpZSBiZXRyb2trZW4gemlqbiBpbiBlZW4gaW50ZXJhY3RpZSB0ZXJtIG5pZXQgbG9zIHZhbiBlbGthYXIga3VubmVuIHdvcmRlbiBnZcOvbnRlcnByZXRlZXJkIGlzIGRlIGNvbnZlbnRpZSBvbSBlZW4gaW50ZXJhY3RpZXRlcm0gdWl0IGhldCBtb2RlbCB0ZSB2ZXJ3aWpkZXJlbiB3YW5uZWVyIGRpZSBuaWV0IHNpZ25pZmljYW50IGlzLgoKLSBOYSB2ZXJ3aWpkZXJpbmcgdmFuIGRlIG5pZXQtc2lnbmlmaWNhbnRlIGludGVyYWN0aWV0ZXJtIGt1bm5lbiBkZSBob29mZGVmZmVjdGVuIHdvcmRlbiBnZcOvbnRlcnByZXRlZXJkLgoKCi0tLQoKIyMjIEludGVyYWN0aWUgdHVzc2VuIHR3ZWUgZmFjdG9yIHZhcmlhYmVsZW4KClppZSBhcGFydGUgcHJlc2VudGF0aWUgb3ZlciBmYWN0b3Jpw6tsZSBkZXNpZ25zCgotLS0KCiMgQU5PVkEgVGFiZWwKCgo8aWZyYW1lIHdpZHRoPSI1NjAiIGhlaWdodD0iMzE1IgpzcmM9Imh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL2VtYmVkL1k0a2xUM2VYSndJIgpmcmFtZWJvcmRlcj0iMCIKc3R5bGU9ImRpc3BsYXk6IGJsb2NrOyBtYXJnaW46IGF1dG87IgphbGxvdz0iYXV0b3BsYXk7IGVuY3J5cHRlZC1tZWRpYSIgYWxsb3dmdWxsc2NyZWVuPjwvaWZyYW1lPgoKCkRlIHRvdGFsZSBrd2FkcmF0ZW5zb20gU1NUb3QgaXMgb3BuaWV1dwoKJCQKICBcdGV4dHtTU1RvdH0gPSBcc3VtX3tpPTF9Xm4gKFlfaSAtIFxiYXJ7WX0pXjIuCiQkCgpPb2sgZGUgcmVzaWR1ZWxlIGt3YWRyYXRlbnNvbSBpcyB6b2FscyB2b29yaGVlbi4KJCQKICBcdGV4dHtTU0V9ID0gXHN1bV97aT0xfV5uIChZX2ktXGhhdHtZfV9pKV4yLgokJAoKRGFuIGdlbGR0IGRlIHZvbGdlbmRlIGRlY29tcG9zaXRpZSB2YW4gZGUgdG90YWxlIGt3YWRyYXRlbnNvbSwKJCQKICBcdGV4dHtTU1RvdH0gPSBcdGV4dHtTU1J9ICsgXHRleHR7U1NFfSAsCiQkCm1ldAokJAogIFx0ZXh0e1NTUn0gPSBcc3VtX3tpPTF9Xm4gKFxoYXR7WX1faS1cYmFye1l9KV4yLgokJAoKLS0tCgoKVm9vciBkZSB2cmlqaGVpZHNncmFkZW4gZW4gZGUgZ2VtaWRkZWxkZSBrd2FkcmF0ZW5zb21tZW4gZ2VsZHQ6CgotIFNTVG90IGhlZWZ0ICRuLTEkIHZyaWpoZWlkc2dyYWRlbiBlbiAkXHRleHR7U1NUb3R9LyhuLTEpJCBpcyBlZW4gc2NoYXR0ZXIgdm9vciBkZSB2YXJpYW50aWUgdmFuICRZJCAodmFuIGRlIG1hcmdpbmFsZSBkaXN0cmlidXRpZSB2YW4gJFkkKS4KLSBTU0UgaGVlZnQgJG4tcCQgdnJpamhlaWRzZ3JhZGVuIGVuICRcdGV4dHtNU0V9PVx0ZXh0e1NTRX0vKG4tcCkkIGlzIGVlbiBzY2hhdHRlciB2b29yIGRlIHJlc2lkdWVsZSB2YXJpYW50aWUgdmFuICRZJCBnZWdldmVuIGRlIHJlZ3Jlc3NvcmVuIChpLmUuIGVlbiBzY2hhdHRlciB2b29yIGRlIHJlc2lkdWVsZSB2YXJpYW50aWUgJFxzaWdtYV4yJCB2YW4gZGUgZm91dHRlcm0gJFxlcHNpbG9uJCkuCi0gU1NSIGhlZWZ0ICRwLTEkIHZyaWpoZWlkc2dyYWRlbiBlbiAkXHRleHR7TVNSfT1cdGV4dHtTU1J9LyhwLTEpJCBpcyBkZSBnZW1pZGRlbGRlIGt3YWRyYXRlbnNvbSB2YW4gZGUgcmVncmVzc2llLgoKRGUgZGV0ZXJtaW5hdGllY2/Dq2ZmaWNpw6tudCBibGlqZnQgem9hbHMgdm9vcmhlZW4sIGkuZS4KJCQKICBSXjIgPSAxLVxmcmFje1x0ZXh0e1NTRX19e1x0ZXh0e1NTVG90fX0gPSBcZnJhY3tcdGV4dHtTU1J9fXtcdGV4dHtTU1RvdH19CiQkCmlzIGRlIGZyYWN0aWUgdmFuIGRlIHRvdGFsZSB2YXJpYWJpbGl0ZWl0IGluIGRlIHVpdGtvbXN0ZW4gZGllIHZlcmtsYWFyZCB3b3JkdCBkb29yIGhldCByZWdyZXNzaWVtb2RlbC4KCkRlIHRlc3RzdGF0aXN0aWVrICRGPVx0ZXh0e01TUn0vXHRleHR7TVNFfSQgaXMgb25kZXIgJEhfMDpcYmV0YV8xPVxsZG90cz1cYmV0YV97cC0xfT0wJCB2ZXJkZWVsZCBhbHMgJEZfe3AtMTtuLXB9JC4KCi0tLQoKYGBge3IsIGVjaG89RkFMU0V9CnN1bW1hcnkobG1WV1MpCmBgYAoKLS0tCgojIyBFeHRyYSBLd2FkcmF0ZW5zb21tZW4KCgo8aWZyYW1lIHdpZHRoPSI1NjAiIGhlaWdodD0iMzE1IgpzcmM9Imh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL2VtYmVkL0xYZ2gzMlU5RHE0IgpmcmFtZWJvcmRlcj0iMCIKc3R5bGU9ImRpc3BsYXk6IGJsb2NrOyBtYXJnaW46IGF1dG87IgphbGxvdz0iYXV0b3BsYXk7IGVuY3J5cHRlZC1tZWRpYSIgYWxsb3dmdWxsc2NyZWVuPjwvaWZyYW1lPgoKQmVzY2hvdXcgZGUgdm9sZ2VuZGUgdHdlZSByZWdyZXNzaWVtb2RlbGxlbiB2b29yIHJlZ3Jlc3NvcmVuICR4XzEkIGVuICR4XzIkOgokJAogIFlfaSA9IFxiZXRhXzArXGJldGFfMSB4X3tpMX0gKyBcZXBzaWxvbl9pLAokJAptZXQgJFxlcHNpbG9uX2lcdGV4dHsgaWlkIH0gTigwLFxzaWdtYV8xXnsyfSkkLCBlbgokJApZX2kgPSBcYmV0YV8wK1xiZXRhXzEgeF97aTF9K1xiZXRhXzIgeF97aTJ9ICsgXGVwc2lsb25faSwKJCQKbWV0ICRcZXBzaWxvbl9pXHRleHR7IGlpZCB9IE4oMCxcc2lnbWFfMl57Mn0pJC4KClZvb3IgaGV0IGVlcnN0ZSAoZ2VyZWR1Y2VlcmRlKSBtb2RlbCBnZWxkdCBkZSBkZWNvbXBvc2l0aWUKXFsKICBcdGV4dHtTU1RvdH0gPSBcdGV4dHtTU1J9XzEgKyBcdGV4dHtTU0V9XzEKXF0KZW4gdm9vciBoZXQgdHdlZWRlIChuaWV0LWdlcmVkdWNlZXJkZSkgbW9kZWwKXFsKICBcdGV4dHtTU1RvdH0gPSBcdGV4dHtTU1J9XzIgKyBcdGV4dHtTU0V9XzIKXF0KKFNTVG90IGlzIHVpdGVyYWFyZCBkZXplbGZkZSBpbiBiZWlkZSBtb2RlbGxlbiBvbWRhdCBkaXQgbmlldCBhZmhhbmd0IHZhbiBoZXQgcmVncmVzc2llbW9kZWwpLgoKLS0tCgoqKkRlZmluaXRpZSBleHRyYSBrd2FkcmF0ZW5zb20qKgpEZSAqZXh0cmEga3dhZHJhdGVuc29tKiAoRW5nZWxzOiAqZXh0cmEgc3VtIG9mIHNxdWFyZXMqKSB2YW4gcHJlZGljdG9yICR4XzIkIHQuby52LiBoZXQgbW9kZWwgbWV0IGVua2VsICR4XzEkIGFscyBwcmVkaWN0b3Igd29yZHQgZ2VnZXZlbiBkb29yCiQkCiAgXHRleHR7U1NSfV97MlxtaWQgMX0gPSBcdGV4dHtTU0V9XzEtXHRleHR7U1NFfV8yPVx0ZXh0e1NTUn1fMi1cdGV4dHtTU1J9XzEuCiQkICAKCioqRWluZGUgZGVmaW5pdGllKioKCk1lcmsgZWVyc3Qgb3AgZGF0ICRcdGV4dHtTU0V9XzEtXHRleHR7U1NFfV8yPVx0ZXh0e1NTUn1fMi1cdGV4dHtTU1J9XzEkIHRyaXZpYWFsIGlzIGdlemllbiBkZSBkZWNvbXBvc2l0aWVzIHZhbiBkZSB0b3RhbGUga3dhZHJhdGVuc29tbWVuLgoKRGUgZXh0cmEga3dhZHJhdGVuc29tICRcdGV4dHtTU1J9X3syXG1pZCAxfSQga2FuIGVlbnZvdWRpZyBnZcOvbnRlcnByZXRlZXJkIHdvcmRlbiBhbHMgZGUgZXh0cmEgdmFyaWFudGllIHZhbiBkZSB1aXRrb21zdCBkaWUgdmVya2xhYXJkIGthbiB3b3JkZW4gZG9vciByZWdyZXNzb3IgJHhfMiQgdG9lIHRlIHZvZWdlbiBhYW4gZWVuIG1vZGVsIHdhYXJpbiByZWdyZXNzb3IgJHhfMSQgcmVlZHMgYWFud2V6aWcgaXMuCgpNZXQgZGl0IG5pZXV3IHNvb3J0IGt3YWRyYXRlbnNvbSBrdW5uZW4gd2Ugdm9vciBoZXQgbW9kZWwgbWV0IHR3ZWUgcHJlZGljdG9yZW4gc2NocmlqdmVuCiQkCiAgXHRleHR7U1NUb3R9ID0gXHRleHR7U1NSfV8xKyBcdGV4dHtTU1J9X3syXG1pZCAxfSArIFx0ZXh0e1NTRX0uCiQkCkRpdCB2b2xndCByZWNodHN0cmVla3MgdWl0IGRlIGRlZmluaXRpZSB2YW4gZGUgZXh0cmEga3dhZHJhdGVuc29tICRcdGV4dHtTU1J9X3syXG1pZCAxfSQuCgoKLS0tCgoKVWl0YnJlaWRpbmc6IFpvbmRlciBpbiB0ZSBib2V0ZW4gaW4gYWxnZW1lZW5oZWlkIHN0YXJ0ZW4gd2UgbWV0IGRlIHJlZ3Jlc3NpZW1vZGVsbGVuICgkczxwLTEkKQokJApZX2kgPSBcYmV0YV8wICsgXGJldGFfMSB4X3tpMX0gKyBcY2RvdHMgKyBcYmV0YV97c30geF97aXN9ICsgXGVwc2lsb25faQokJAptZXQgJFxlcHNpbG9uX2lcdGV4dHsgaWlkIH1OKDAsXHNpZ21hXzFeezJ9KSQsIGVuICgkczwgcVxsZXEgcC0xJCkKJCQKWV9pID0gXGJldGFfMCArIFxiZXRhXzEgeF97aTF9ICsgXGNkb3RzICsgXGJldGFfe3N9IHhfe2lzfSArIFxiZXRhX3tzKzF9IHhfe2lzKzF9ICsgXGNkb3RzIFxiZXRhX3txfXhfe2lxfSsgXGVwc2lsb25faQokJAptZXQgJFxlcHNpbG9uX2lcdGV4dHsgaWlkIH0gTigwLFxzaWdtYV8yXnsyfSkkLgoKCkRlICoqZXh0cmEga3dhZHJhdGVuc29tKiogdmFuIHByZWRpY3RvcmVuICR4X3tzKzF9LCBcbGRvdHMsIHhfcSQgdC5vLnYuIGhldCBtb2RlbCBtZXQgZW5rZWwgZGUgcHJlZGljdG9yZW4gICR4XzEsXGxkb3RzLCB4X3tzfSQgd29yZHQgZ2VnZXZlbiBkb29yCiQkCiAgXHRleHR7U1NSfV97cysxLCBcbGRvdHMsIHFcbWlkIDEsXGxkb3RzLCBzfSA9IFx0ZXh0e1NTRX1fMS1cdGV4dHtTU0V9XzI9XHRleHR7U1NSfV8yLVx0ZXh0e1NTUn1fMS4KJCQgIAoKLS0tCgojIyMgVHlwZSBJIEt3YWRyYXRlbnNvbW1lbgoKU3RlbCBkYXQgJHAtMSQgcmVncmVzc29yZW4gYmVzY2hvdXdkIHdvcmRlbiwgZW4gYmVzY2hvdXcgZWVuIHNlcXVlbnRpZSB2YW4gbW9kZWxsZW4gKCRzPTIsXGxkb3RzLCBwLTEkKQokJApZX2kgPSBcYmV0YV8wICsgXHN1bV97aj0xfV57c30gXGJldGFfaiB4X3tpan0gKyBcZXBzaWxvbl9pCiQkCm1ldCAkXGVwc2lsb25faVx0ZXh0eyBpaWQgfSBOKDAsXHNpZ21hXnsyfSkkLgoKLSBEZSBvdmVyZWVua29tc3RpZ2Uga3dhZHJhdGVuc29tbWVuIHdvcmRlbiBnZW5vdGVlcmQgYWxzICRcdGV4dHtTU1J9X3tzfSQgZW4gJFx0ZXh0e1NTRX1fe3N9JC4KLSBEZSBtb2RlbHNlcXVlbnRpZSBnZWVmdCBvb2sgYWFubGVpZGluZyB0b3QgZXh0cmEga3dhZHJhdGVuc29tbWVuICRcdGV4dHtTU1J9X3tzXG1pZCAxLFxsZG90cywgcy0xfSQuCi0gRGV6ZSBsYWF0c3RlIGt3YWRyYXRlbnNvbSB3b3JkdCBlZW4gdHlwZSBJIGt3YWRyYXRlbnNvbSBnZW5vZW1kLiBNZXJrIG9wIGRhdCBkZXplIGFmaGFuZ3QgdmFuIGRlIHZvbGdvcmRlIChudW1tZXJpbmcpIHZhbiByZWdyZXNzb3Jlbi4KCi0tLQoKRXIga2FuIGFhbmdldG9vbmQgd29yZGVuIGRhdCB2b29yIE1vZGVsIG1ldCAkcz1wLTEkIGdlbGR0CiQkCiBcdGV4dHtTU1RvdH0gPSBcdGV4dHtTU1J9XzEgKyBcdGV4dHtTU1J9X3syXG1pZCAxfSArIFx0ZXh0e1NTUn1fezNcbWlkIDEsMn0gKyBcY2RvdHMgKyBcdGV4dHtTU1J9X3twLTFcbWlkIDEsXGxkb3RzLCBwLTJ9ICsgXHRleHR7U1NFfSwKJCQKbWV0ICRcdGV4dHtTU0V9JCBkZSByZXNpZHVlbGUga3dhZHJhdGVuc29tIHZhbiBoZXQgbW9kZWwgbWV0IGFsbGUgJHAtMSQgcmVncmVzc29yZW4gZW4KJCQKICBcdGV4dHtTU1J9XzEgKyBcdGV4dHtTU1J9X3syXG1pZCAxfSArIFx0ZXh0e1NTUn1fezNcbWlkIDEsMn0gKyBcY2RvdHMgKyBcdGV4dHtTU1J9X3twLTFcbWlkIDEsXGxkb3RzLCBwLTJ9ID0gXHRleHR7U1NSfQokJAptZXQgJFx0ZXh0e1NTUn0kIGRlIGt3YWRyYXRlbnNvbSB2YW4gZGUgcmVncmVzc2llIHZhbiBoZXQgbW9kZWwgbWV0IGFsbGUgJHAtMSQgcmVncmVzc29yZW4uCgotIEludGVycHJldGF0aWUgdmFuIGllZGVyZSB0ZXJtIGFmaGFuZ3QgdmFuIGRlIHZvbGdvcmRlIHZhbiBkZSByZWdyZXNzb3JlbiBpbiBkZSBzZXF1ZW50aWUgdmFuIHJlZ3Jlc3NpZW1vZGVsbGVuLgoKCi0tLQoKCi0gSWVkZXJlIHR5cGUgSSBTU1IgaGVlZnQgYmV0cmVra2luZyBvcCBoZXQgZWZmZWN0IHZhbiAxIHJlZ3Jlc3NvciBlbiBoZWVmdCBkdXMgMSB2cmlqaGVpZHNncmFhZC4KLSBWb29yIGllZGVyZSB0eXBlIEkgU1NSIHRlcm0ga2FuIGVlbiBnZW1pZGRlbGRlIGt3YWRyYXRlbnNvbSBnZWRlZmluaWVlcmQgd29yZGVuIGFscyAkXHRleHR7TVNSfV97alxtaWQgMSxcbGRvdHMsIGotMX09XHRleHR7U1NSfV97alxtaWQgMSxcbGRvdHMsIGotMX0vMSQuCi0gRGUgdGVzdHN0YXRpc3RpZWsgJEY9XHRleHR7TVNSfV97alxtaWQgMSxcbGRvdHMsIGotMX0vXHRleHR7TVNFfSQgaXMgb25kZXIgJEhfMDpcYmV0YV9qPTAkIG1ldCAkcz1qJCB2ZXJkZWVsZCBhbHMgJEZfezE7bi0oaisxKX0kLgotIERlemUga3dhZHJhdGVuc29tbWVuIHdvcmRlbiBzdGFuZGFhcmQgd2VlcmdlZ2V2ZW4gZG9vciBkZSBhbm92YSBmdW5jdGllIGluIFIuCgotLS0KCiMjIyBUeXBlIElJSSBLd2FkcmF0ZW5zb21tZW4KCkRlIHR5cGUgSUlJIGt3YWRyYXRlbnNvbSB2YW4gcmVncmVzc29yICR4X2okIHdvcmR0IGdlZ2V2ZW4gZG9vciBkZSBleHRyYSBrd2FkcmF0ZW5zb20KJCQKICBcdGV4dHtTU1J9X3tqIFxtaWQgMSxcbGRvdHMsIGotMSxqKzEsXGxkb3RzLCBwLTF9ID0gXHRleHR7U1NFfV8xLVx0ZXh0e1NTRX1fMgokJAoKLSAkXHRleHR7U1NFfV8yJCBkZSByZXNpZHVlbGUga3dhZHJhdGVuc29tIHZhbiByZWdyZXNzaWVtb2RlbCBtZXQgYWxsZSAkcC0xJCByZWdyZXNzb3Jlbi4KLSAkXHRleHR7U1NFfV8xJCBkZSByZXNpZHVlbGUga3dhZHJhdGVuc29tIHZhbiByZWdyZXNzaWVtb2RlbCBtZXQgYWxsZSAkcC0xJCByZWdyZXNzb3JlbiwgdWl0Z2V6b25kZXJkIHJlZ3Jlc3NvciAkeF9qJC4KCkRlIHR5cGUgSUlJIGt3YWRyYXRlbnNvbSAkXHRleHR7U1NSfV97aiBcbWlkIDEsXGxkb3RzLCBqLTEsaisxLFxsZG90cywgcC0xfSQga3dhbnRpZmljZWVydCBkdXMgaGV0IGFhbmRlZWwgdmFuIGRlIHRvdGFsZSB2YXJpYW50aWUgdmFuIGRlIHVpdGtvbXN0IGRhdCBkb29yIHJlZ3Jlc3NvciAkeF9qJCB2ZXJrbGFhcmQgd29yZHQgZW4gZGF0IG5pZXQgZG9vciBkZSBhbmRlcmUgJHAtMiQgcmVncmVzc29yZW4gdmVya2xhYXJkIHdvcmR0LgoKLS0tCgpEZSB0eXBlIElJSSBrd2FkcmF0ZW5zb20gaGVlZnQgb29rIDEgdnJpamhlaWRzZ3JhYWQgb21kYXQgaGV0IG9tIDEgJFxiZXRhJC1wYXJhbWV0ZXIgZ2FhdC4KClZvb3IgaWVkZXJlIHR5cGUgSUlJIFNTUiB0ZXJtIGthbiBlZW4gZ2VtaWRkZWxkZSBrd2FkcmF0ZW5zb20gZ2VkZWZpbmllZXJkIHdvcmRlbiBhbHMgJFx0ZXh0e01TUn1fe2ogXG1pZCAxLFxsZG90cywgai0xLGorMSxcbGRvdHMsIHAtMX09XHRleHR7U1NSfV97aiBcbWlkIDEsXGxkb3RzLCBqLTEsaisxLFxsZG90cywgcC0xfS8xJC4KCkRlIHRlc3RzdGF0aXN0aWVrICRGPVx0ZXh0e01TUn1fe2ogXG1pZCAxLFxsZG90cywgai0xLGorMSxcbGRvdHMsIHAtMX0vXHRleHR7TVNFfSQgaXMgb25kZXIgJEhfMDpcYmV0YV9qPTAkIHZlcmRlZWxkIGFzICRGX3sxO24tcH0kLgoKLS0tCgpgYGB7cn0KbGlicmFyeShjYXIpCkFub3ZhKGxtVldTLHR5cGU9MykKYGBgCgpwLXdhYXJkZW4gaWRlbnRpZWsgYWFuIGRpZSB2YW4gdHdlZXppamRpZ2UgdC10ZXN0ZW4KCk1lcmsgZWNodGVyIG9wIGRhdCBhbGxlIGR1bW15J3Mgdm9vciBmYWN0b3JlbiBtZXQgbWVlcmRlcmUgbml2ZWF1cyBpbiDDqcOpbiBrZWVyIHVpdCBoZXQgbW9kZWwgd29yZGVuIGdlaGFhbGQuIERlIHR5cGUgSUlJIHN1bSBvZiBzcXVhcmVzIGhlZWZ0IGR1cyBldmVudmVlbCB2cmlqaGVpZHNncmFkZW4gYWxzIGhldCBhYW50YWwgZHVtbXkncyBlbiBlciB3b3JkdCBlZW4gb21uaWJ1c3Rlc3QgdWl0Z2V2b2VyZCB2b29yIGhldCBlZmZlY3QgdmFuIGRlIGZhY3Rvci4KCi0tLQoKIyBEaWFnbm9zdGllawoKCjxpZnJhbWUgd2lkdGg9IjU2MCIgaGVpZ2h0PSIzMTUiCnNyYz0iaHR0cHM6Ly93d3cueW91dHViZS5jb20vZW1iZWQvNnQ4NHRPMnpPY3MiCmZyYW1lYm9yZGVyPSIwIgpzdHlsZT0iZGlzcGxheTogYmxvY2s7IG1hcmdpbjogYXV0bzsiCmFsbG93PSJhdXRvcGxheTsgZW5jcnlwdGVkLW1lZGlhIiBhbGxvd2Z1bGxzY3JlZW4+PC9pZnJhbWU+CgojIyBNdWx0aWNvbGxpbmVhcml0ZWl0CgpgYGB7ciBlY2hvPUZBTFNFfQpzdW1tYXJ5KGxtVldTKQpgYGAKCi0tLQoKYGBge3IgZWNobz1GQUxTRX0Kc3VtbWFyeShsbVZXU19JbnRWVykKYGBgCgotLS0KCgotIFNjaGF0dGluZ2VuIHZlcnNjaGlsbGVuZCB2YW4gYWRkaXRpZWYgbW9kZWwgZW4gc3RhbmRhYXJkZm91dGVuIHppam4gdmVlbCBncm90ZXIhCi0gRGUgb29yemFhayBpcyBwcm9ibGVlbSB2YW4gbXVsdGljb2xsaW5lYXJpdGVpdC4KLSBBbHMgMiBwcmVkaWN0b3JlbiBzdGVyayBnZWNvcnJlbGVlcmQgemlqbiwgZGFuIGRlbGVuIHplIHZvb3IgZWVuIGdyb290IHN0dWsgZGV6ZWxmZGUgaW5mb3JtYXRpZQotIE1vZWlsaWprIG9tIGRlIGFmem9uZGVybGlqa2UgZWZmZWN0ZW4gdmFuIGJlaWRlbiBvcCBkZSB1aXRrb21zdCB0ZSBzY2hhdHRlbi4KLSBLbGVpbnN0ZSBrd2FkcmF0ZW5zY2hhdHRlcnMgb25zdGFiaWVsIHdvcmR0Ci0gU3RhbmRhYXJkIGVycm9ycyBrdW5uZW4gd29yZGVuIG9wZ2VibGF6ZW4KLSBab2xhbmcgbWVuIGVua2VsIHByZWRpY3RpZXMgdHJhY2h0IHRlIGJla29tZW4gb3AgYmFzaXMgdmFuIGhldCByZWdyZXNzaWVtb2RlbCB6b25kZXIgZGFhcmJpaiB0ZSBleHRyYXBvbGVyZW4gYnVpdGVuIGhldCBiZXJlaWsgdmFuIGRlIHByZWRpY3RvcmVuIGlzIG11bHRpY29sbGluZWFyaXRlaXQgZ2VlbiBwcm9ibGVlbS4KLSBXZWwgcHJvYmxlZW0gdm9vciBpbmZlcmVudGllCgoKLS0tCgpgYGB7cn0KY29yKGNiaW5kKHByb3N0YXRlJGxjYXZvbCxwcm9zdGF0ZSRsd2VpZ2h0LHByb3N0YXRlJGxjYXZvbCpwcm9zdGF0ZSRsd2VpZ2h0KSkKYGBgCgotIGhvZ2UgY29ycmVsYXRpZSB0dXNzZW4gbG9nLXR1bW9ydm9sdW1lIGVuIGludGVyYWN0aWV0ZXJtLgotIElzIGVlbiBnZWtlbmQgcHJvYmxlZW0gdm9vciBob2dlcmUgb3JkZSB0ZXJtZW4gKGludGVyYWN0aWVzIGVuIGt3YWRyYXRpc2NoZSB0ZXJtZW4pCgotLS0KCi0gTXVsdGljb2xsaW5lYXJpdGVpdCBvcHNwb3JlbiBhLmQuaC52LiBjb3JyZWxhdGllIG1hdHJpeCBvZiBzY2F0dGVycGxvdCBtYXRyaXggaXMgbmlldCBpZGVhYWwuCi0gR2VlbiBpZGVlIGluIHdlbGtlIG1hdGUgZGUgZ2VvYnNlcnZlZXJkZSBtdWx0aWNvbGxpbmVhcml0ZWl0IGRlIHJlc3VsdGF0ZW4gb25zdGFiaWVsIG1hYWt0LgotIEluIG1vZGVsbGVuIG1ldCAzIG9mIG1lZXJkZXJlIHByZWRpY3RvcmVuLCB6ZWcgWDEsIFgyLCBYMyBrYW4gZXIgendhcmUgbXVsdGljb2xsaW5lYXJpdGVpdCBvcHRyZWRlbiBvbmRhbmtzIGRhdCBhbGxlIHBhYXJzZ2V3aWp6ZSBjb3JyZWxhdGllcyB0dXNzZW4gZGUgcHJlZGljdG9yZW4gbGFhZyB6aWpuLgotIE9vayBtdWx0aWNvbGxpbmVhcml0ZWl0IGFscyBlciBlZW4gaG9nZSBjb3JyZWxhdGllIGlzIHR1c3NlbiBYMSBlbiBlZW4gbGluZWFpcmUgY29tYmluYXRpZSB2YW4gWDIgZW4gWDMuCgoKLS0tCgojIyMgVmFyaWFuY2UgaW5mbGF0aW9uIGZhY3RvciAoVklGKQoKClZvb3IgZGUgJGokLWRlIHBhcmFtZXRlciBpbiBoZXQgcmVncmVzc2llbW9kZWwgZ2VkZWZpbmllZXJkIHdvcmR0IGFscwpcW1x0ZXh0cm17VklGfV9qPVxsZWZ0KDEtUl9qXjJccmlnaHQpXnstMX1cXQoKLSBJbiBkZXplIHVpdGRydWtraW5nIHN0ZWx0ICRSX2peMiQgZGUgbWVlcnZvdWRpZ2UgZGV0ZXJtaW5hdGllY2/Dq2ZmaWNpw6tudCB2b29yIHZhbiBlZW4gbGluZWFpcmUgcmVncmVzc2llIHZhbiBkZSAkaiQtZGUgcHJlZGljdG9yIG9wIGFsbGUgYW5kZXJlIHByZWRpY3RvcmVuIGluIGhldCBtb2RlbC4KLSBWSUYgaXMgMSBhbHMgJGokLWRlIHByZWRpY3RvciBuaWV0IGxpbmVhaXIgZ2Vhc3NvY2llZXJkIGlzIG1ldCBkZSBhbmRlcmUgcHJlZGljdG9yZW4gaW4gaGV0IG1vZGVsLgotIFZJRiBpcyBncm90ZXIgZGFuIDEgaW4gYWxsZSBhbmRlcmUgZ2V2YWxsZW4uCi0gVklGIGlzIGZhY3RvciB3YWFybWVlIGdlb2JzZXJ2ZWVyZGUgdmFyaWFudGllIGdyb3RlciBpcyBkYW4gd2FubmVlciBhbGxlIHByZWRpY3RvcmVuIG9uYWZoYW5rZWxpamsgem91ZGVuIHppam4uCi0gVklGID4gMTAgJFxyaWdodGFycm93JCBlcm5zdGlnZSBtdWx0aWNvbGxpbmVhcml0ZWl0LgoKLS0tCgojIyMgQm9keSBmYXQgdm9vcmJlZWxkCgpgYGB7ciBvdXQud2lkdGg9JzkwJScsIGZpZy5hc3A9LjgsIGZpZy5hbGlnbj0nY2VudGVyJywgbWVzc2FnZT1GQUxTRSxlY2hvPUZBTFNFfQpib2R5ZmF0IDwtIHJlYWRfZGVsaW0oImh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9zdGF0b21pY3Mvc2JjMjAvbWFzdGVyL2RhdGEvYm9keWZhdC50eHQiLGRlbGltPSIgIikKYm9keWZhdCAlPiUgZ2dwYWlycygpCmBgYAoKLS0tCgpgYGB7ciBlY2hvPUZBTFNFfQpsbUZhdCA8LSBsbSgKICBCb2R5X2ZhdCB+CiAgICBUcmljZXBzICsKICAgIFRoaWdoICsKICAgIE1pZGFybSwKICBkYXRhID0gYm9keWZhdCkKCnN1bW1hcnkobG1GYXQpCmBgYAoKLS0tCgpgYGB7cn0KdmlmKGxtRmF0KQpgYGAKCi0tLQoKYGBge3IgZWNobz1GQUxTRX0KbG1NaWRhcm0gPC0gbG0oCiAgTWlkYXJtIH4KICAgIFRyaWNlcHMgKwogICAgVGhpZ2gsCiAgICBkYXRhID0gYm9keWZhdCkKCnN1bW1hcnkobG1NaWRhcm0pCmBgYAoKLS0tCgoKV2UgZXZhbHVlcmVuIG51IGRlIFZJRiBpbiBoZXQgcHJvc3RhYXRrYW5rZXIgdm9vcmJlZWxkIHZvb3IgaGV0IGFkZGl0aWV2ZSBtb2RlbCBlbiBoZXQgbW9kZWwgbWV0IGludGVyYWN0aWUuCgpgYGB7cn0KdmlmKGxtVldTKQp2aWYobG1WV1NfSW50VlcpCmBgYAoKCi0gSW5mbGF0aWUgdm9vciBpbnRlcmFjdGlldGVybWVuIHdvcmR0IHZhYWsgdmVyb29yemFha3QgZG9vciBoZXQgZmVpdCBkYXQgaGV0IGhvb2ZkZWZmZWN0IGVlbiBhbmRlcmUgaW50ZXJwcmV0YXRpZSBrcmlqZ3QuCgotLS0KCiMjIEludmxvZWRyaWprZSBPYnNlcnZhdGllcwoKCjxpZnJhbWUgd2lkdGg9IjU2MCIgaGVpZ2h0PSIzMTUiCnNyYz0iaHR0cHM6Ly93d3cueW91dHViZS5jb20vZW1iZWQvQ2NaRTltUlVTVEUiCmZyYW1lYm9yZGVyPSIwIgpzdHlsZT0iZGlzcGxheTogYmxvY2s7IG1hcmdpbjogYXV0bzsiCmFsbG93PSJhdXRvcGxheTsgZW5jcnlwdGVkLW1lZGlhIiBhbGxvd2Z1bGxzY3JlZW4+PC9pZnJhbWU+CgoKYGBge3IgZWNobz1GQUxTRX0Kc2V0LnNlZWQoMTEyMzU4KQpub2JzPC0yMApzZHk8LTEKeDwtc2VxKDAsMSxsZW5ndGg9bm9icykKeTwtMTArNSp4K3Jub3JtKG5vYnMsc2Q9c2R5KQp4MTwtYyh4LDAuNSkKeTEgPC0gYyh5LDEwKzUqMS41K3Jub3JtKDEsc2Q9c2R5KSkKeDIgPC0gYyh4LDEuNSkKeTIgPC0gYyh5LHkxWzIxXSkKeDMgPC0gYyh4LDEuNSkKeTMgPC0gYyh5LDExKQpwbG90KHgseSx4bGltPXJhbmdlKGMoeDEseDIseDMpKSx5bGltPXJhbmdlKGMoeTEseTIseTMpKSkKcG9pbnRzKGMoeDFbMjFdLHgyWzIxXSx4M1syMV0pLGMoeTFbMjFdLHkyWzIxXSx5M1syMV0pLHBjaD1hcy5jaGFyYWN0ZXIoMTozKSxjb2w9Mjo0KQphYmxpbmUobG0oeX54KSxsd2Q9MikKYWJsaW5lKGxtKHkxfngxKSxjb2w9MixsdHk9Mixsd2Q9MikKYWJsaW5lKGxtKHkyfngyKSxjb2w9MyxsdHk9Myxsd2Q9MikKYWJsaW5lKGxtKHkzfngzKSxjb2w9NCxsdHk9NCxsd2Q9MikKbGVnZW5kKCJ0b3BsZWZ0Iixjb2w9MTo0LGx0eT0xOjQsbGVnZW5kPXBhc3RlKCJsbSIsYygiIixhcy5jaGFyYWN0ZXIoMTozKSkpLHRleHQuY29sPTE6NCkKYGBgCgotLS0KCi0gSGV0IGlzIG5pZXQgd2Vuc2VsaWprIGRhdCBlZW4gZW5rZWxlIHdhYXJuZW1pbmcgaGV0IHJlc3VsdGFhdCB2YW4gZWVuIGxpbmVhaXJlIHJlZ3Jlc3NpZWFuYWx5c2UgZ3JvdGVuZGVlbHMgYmVwYWFsZAotIERpYWdub3N0aWVrZW4gZGllIG9ucyB0b2VsYXRlbiBvbSBleHRyZW1lIG9ic2VydmF0aWVzIG9wIHRlIHNwb3JlbgotICpTdHVkZW50aXplZCByZXNpZHUncyogb20gb3V0bGllcnMgb3AgdGUgc3BvcmVuCi0gKmxldmVyYWdlIChpbnZsb2VkLCBoZWZib29tKSogb20gb2JzZXJ2YXRpZXMgbWV0IGV4dHJlZW0gY292YXJpYWF0cGF0cm9vbiBvcCB0ZSBzcG9yZW4KCi0tLQoKIyMjIENvb2sncyBkaXN0YW5jZQoKCi0gRWVuIG1lZXIgcmVjaHRzdHJlZWtzZSBtYWF0IG9tIGRlIGludmxvZWQgdmFuIGVsa2Ugb2JzZXJ2YXRpZSBvcCBkZSByZWdyZXNzaWUtYW5hbHlzZSB1aXQgdGUgZHJ1a2tlbgotIENvb2sncyBkaXN0YW5jZSB2b29yICRpJC1kZSBvYnNlcnZhdGllIGlzIGVlbiBkaWFnbm9zdGlzY2hlIG1hYXQgdm9vciBkZSBpbnZsb2VkIHZhbiBkaWUgb2JzZXJ2YXRpZSBvcCBhbGxlIHByZWRpY3RpZXMgb2Ygdm9vciBoYWFyIGludmxvZWQgIG9wICphbGxlKiBnZXNjaGF0dGUgcGFyYW1ldGVycy4KXFtEX2k9XGZyYWN7XHN1bV97aj0xfV5uKFxoYXR7WX1fai1caGF0e1l9X3tqKGkpfSleMn17cFx0ZXh0cm17TVNFfX1cXQotIEFscyBDb29rJ3MgZGlzdGFuY2UgJERfaSQgZ3Jvb3QgaXMsIGRhbiBoZWVmdCBkZSAkaSQtZGUgb2JzZXJ2YXRpZSBlZW4gZ3JvdGUgaW52bG9lZCBvcCBkZSBwcmVkaWN0aWVzIGVuIGdlc2NoYXR0ZSBwYXJhbWV0ZXJzLgotIEV4dHJlbWUgQ29vaydzIGRpc3RhbmNlIGFscyBoZXQgaGV0IDUwXCUgcGVyY2VudGllbCB2YW4gZGUgJEZfe3Asbi1wfSQtdmVyZGVsaW5nIG92ZXJzY2hyaWpkdC4KCgotLS0KCgpgYGB7ciAgb3V0LndpZHRoPScxMDAlJywgZmlnLmFzcD0uOCwgZmlnLmFsaWduPSdjZW50ZXInLCBtZXNzYWdlPUZBTFNFLCBlY2hvPUZBTFNFfQpwYXIobWZyb3c9YygyLDIpKQpwbG90KGxtVldTLHdoaWNoPTUpCnBsb3QobG1WV1NfSW50Vlcsd2hpY2g9NSkKcGxvdChjb29rcy5kaXN0YW5jZShsbVZXUyksdHlwZT0iaCIseWxpbT1jKDAsMSksbWFpbj0iQWRkaXRpdmUgbW9kZWwiKQphYmxpbmUoaD1xZigwLjUsbGVuZ3RoKGxtVldTJGNvZWYpLG5yb3cocHJvc3RhdGUpLWxlbmd0aChsbVZXUyRjb2VmKSksbHR5PTIpCnBsb3QoY29va3MuZGlzdGFuY2UobG1WV1NfSW50VlcpLHR5cGU9ImgiLHlsaW09YygwLDEpLCBtYWluPSJNb2RlbCB3aXRoIGxjYXZvbDpsd2VpZ2h0IGludGVyYWN0aW9uIikKYWJsaW5lKGg9cWYoMC41LGxlbmd0aChsbVZXU19JbnRWVyRjb2VmKSxucm93KHByb3N0YXRlKS1sZW5ndGgobG1WV1NfSW50VlckY29lZikpLGx0eT0yKQpgYGAKCi0tLQoKLSBFZW5tYWFsIG1lbiB2YXN0Z2VzdGVsZCBoZWVmdCBkYXQgZWVuIG9ic2VydmF0aWUgaW52bG9lZHJpamsgaXMsIGthbiBtZW4gem9nZW5hYW1kZSAqREZCRVRBUyogZ2VicnVpa2VuIG9tIHRlIGJlcGFsZW4gb3Agd2Vsa2UgcGFyYW1ldGVyKHMpIHplIGVlbiBncm90ZSBpbnZsb2VkIHVpdG9lZmVudC4KLSAgREZCRVRBUyB2YW4gZGUgJGkkLWRlIG9ic2VydmF0aWUgdm9ybWVuIGVlbiBkaWFnbm9zdGlzY2hlIG1hYXQgdm9vciBkZSBpbnZsb2VkIHZhbiBkaWUgb2JzZXJ2YXRpZSAqb3AgZWxrZSByZWdyZXNzaWVwYXJhbWV0ZXIgYWZ6b25kZXJsaWprKgokJFx0ZXh0cm17REZCRVRBU31fe2ooaSl9PVxmcmFje1xoYXR7XGJldGF9X3tqfS1caGF0e1xiZXRhfV97aihpKX19e1x0ZXh0cm17U0R9KFxoYXR7XGJldGF9X3tqfSl9JCQKLSAgREZCRVRBUyBleHRyZWVtIGlzIHdhbm5lZXIgemUgMSBvdmVyc2NocmlqZHQgaW4ga2xlaW5lIHRvdCBtaWRkZWxncm90ZSBkYXRhc2V0cyBlbiAkMi9cc3FydHtufSQgaW4gZ3JvdGUgZGF0YXNldHMKCi0tLQoKCmBgYHtyIG91dC53aWR0aD0nOTAlJywgZmlnLmFzcD0uOCwgZmlnLmFsaWduPSdjZW50ZXInLCBtZXNzYWdlPUZBTFNFLCBlY2hvPUZBTFNFfQpwYXIobWZyb3c9YygyLDIpKQpkZmJldGFzUGxvdHMobG1WV1MpCmBgYAoKLS0tCgpgYGB7ciBvdXQud2lkdGg9JzkwJScsIGZpZy5hbGlnbj0nY2VudGVyJywgbWVzc2FnZT1GQUxTRSwgZWNobz1GQUxTRX0KcGFyKG1mcm93PWMoMiwyKSkKZGZiZXRhc1Bsb3RzKGxtVldTX0ludFZXKQpgYGAKCi0tLQoKYGBge3Igb3V0LndpZHRoPScxMDAlJyxmaWcuYWxpZ249J2NlbnRlcicsIG1lc3NhZ2U9RkFMU0UsIGVjaG89RkFMU0V9CmJveHBsb3QoZXhwKHByb3N0YXRlJGx3ZWlnaHQpLHlsYWI9IlByb3N0YXRlIFdlaWdodCAoZykiKQpgYGAKCgotLS0KCiMgQ29uc3RyYXN0ZW4KCgo8aWZyYW1lIHdpZHRoPSI1NjAiIGhlaWdodD0iMzE1IgpzcmM9Imh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL2VtYmVkL1RYdVh1dmUyMHdNIgpmcmFtZWJvcmRlcj0iMCIKc3R5bGU9ImRpc3BsYXk6IGJsb2NrOyBtYXJnaW46IGF1dG87IgphbGxvdz0iYXV0b3BsYXk7IGVuY3J5cHRlZC1tZWRpYSIgYWxsb3dmdWxsc2NyZWVuPjwvaWZyYW1lPgoKLSBCaWogbWVlciBjb21wbGV4ZSBhbGdlbWVuZSBsaW5lYWlyZSBtb2RlbGxlbiB3ZW5zdCBtZW4gZGlrd2lqbHMgbWVlcmRlcmUgaHlwb3RoZXNlbiB0ZSB0b2V0c2VuLgotIEJvdmVuZGllbiB2ZXJ0YWxlbiBkZSBvbmRlcnpvZWtzaHlwb3RoZXNlcyB6aWNoIGhpZXJiaWogbmlldCBzdGVlZHMgaW4gw6nDqW4gcGFyYW1ldGVyLCBtYWFyIGluIGVlbiBsaW5lYWlyZSBjb21iaW5hdGllIHZhbiBtb2RlbHBhcmFtZXRlcnMuCi0gRWVuIGxpbmVhaXJlIGNvbWJpbmF0aWUgdmFuIG1vZGVscGFyYW1ldGVycyB3b3JkdCBvb2sgZWVuIGNvbnRyYXN0IGdlbm9lbWQuCgojIyBOSEFORVMgdm9vcmJlZWxkCgotIFN0ZWwgZGF0IGRlIG9uZGVyem9la2VycyBkZSBhc3NvY2lhdGllIHR1c3NlbiBkZSBsZWVmdGlqZCBlbiBkZSBibG9lZGRydWsgd2Vuc2VuIHRlIGJlc3R1ZGVyZW4uIE1vZ2VsaWprcyBpcyBkaWUgYXNzb2NpYXRpZSBhbmRlcnMgaXMgYmlqIG1hbm5lbiBkYW4gdnJvdXdlbi4KCi0gRGUgb25kZXJ6b2VrZXJzIHdlbnNlbiBkZSB2b2xnZW5kZSBvbmRlcnpvZWtzdnJhZ2VuIHRlIGJlYW50d29vcmRlbjoKCiAgICAtIElzIGVyIGVlbiBhc3NvY2lhdGllIHR1c3NlbiBsZWVmdGlqZCBlbiBkZSBibG9lZGRydWsgIGJpaiB2cm91d2VuPwogICAgLSBJcyBlciBlZW4gYXNzb2NpYXRpZSB0dXNzZW4gbGVlZnRpamQgZW4gZGUgYmxvZWRkcnVrIGJpaiBtYW5uZW4/CiAgICAtIElzIGRlIGFzc29jaWF0aWUgdHVzc2VuIGxlZWZ0aWpkIGVuIGRlIGJsb2VkZHJ1ayB2ZXJzY2hpbGxlbmQgYmlqIG1hbm5lbiBkYW4gYmlqIHZyb3V3ZW4/CgojIyBNb2RlbAoKV2UgZml0dGVuIGVlbiBtb2RlbCBvcCBiYXNpcyB2YW4gZGUgZ2VtaWRkZWxkZSBzeXN0b2xpc2NoZSBibG9lZGRydWsgKGBCUFN5c0F2ZWApIGluIGZ1bmN0aWUgdmFuIGRlIGxlZWZ0aWpkLCBnZXNsYWNodCBlbiBlZW4gaW50ZXJhY3RpZSB0dXNzZW4gbGVlZnRpamQgZW4gZ2VzbGFjaHQgdm9vciB2b2x3YXNzZW4gYmxhbmtlIHN1YmplY3RlbiB1aXQgZGUgTkhBTkVTIHN0dWRpZS4KCmBgYHtyfQpsaWJyYXJ5KE5IQU5FUykKYnBEYXRhIDwtIE5IQU5FUyAlPiUKZmlsdGVyKAogIFJhY2UxID09IldoaXRlIiAmCiAgICBBZ2UgPj0gMTggJgogICAgIWlzLm5hKEJQU3lzQXZlKQogICAgKQoKbUJwMSA8LSBsbShCUFN5c0F2ZSB+IEFnZSpHZW5kZXIsIGJwRGF0YSkKcGFyKG1mcm93ID0gYygyLDIpKQpwbG90KG1CcDEpCmBgYAoKLSBBYW5uYW1lcyB2YW4gaGV0IG1vZGVsIHppam4gbmlldCB2b2xkYWFuIQoKICAgIC0gbGluZWFyaXRlaXQgbGlqa3Qgb2sKICAgIC0gaGV0ZXJvc2NlZGFzdGljaXRlaXQKICAgIC0gR2VlbiBub3JtYWxpdGVpdDogc2NoZXZlIHZlcmRlbGluZyBtZXQgc3RhYXJ0IG5hYXIgcmVjaHRzLgogICAgLSBHcm90ZSBkYXRhc2V0IHdlIGt1bm5lbiBzdGV1bmVuIG9wIGRlIENMVAoKIyMjIFRyYW5zZm9ybWF0aWUKCldlIGZpdHRlbiBlZW4gbW9kZWwgb3AgYmFzaXMgdmFuIGRlIGxvZzIgZ2V0cmFuc2Zvcm1lZXJkZSBnZW1pZGRlbGRlIHN5c3RvbGlzY2hlIGJsb2VkZHJ1ayAoYEJQU3lzQXZlYCkgaW4gZnVuY3RpZSB2YW4gZGUgbGVlZnRpamQsIGdlc2xhY2h0IGVuIGVlbiBpbnRlcmFjdGllIHR1c3NlbiBsZWVmdGlqZCBlbiBnZXNsYWNodC4KCmBgYHtyfQptQnAyIDwtIGxtKEJQU3lzQXZlICU+JSBsb2cyIH4gQWdlKkdlbmRlciwgYnBEYXRhKQpwYXIobWZyb3cgPSBjKDIsMikpCnBsb3QobUJwMikKYGBgCgotIERlIHJlc2lkdWVuIHRvbmVuIG5vZyBzdGVlZHMgaGV0ZXJvc2NlZGFzdGljaXRlaXQuCgojIyMgUmVtZWRpw6tyZW4gdm9vciBoZXRlcm9zY2VkYXN0aWNpdGVpdAoKLSBBbHMgZGUgcGxvdCB2YW4gZGUgcmVzaWR1ZW4gaS5mLnYuIGRlIGdlc2NoYXR0ZSB3YWFyZGVuIGVlbiB0b2V0ZXJ2b3JtIHZlcnRvb250IGthbiBtZW4gdG9jaCBjb3JyZWN0ZSBpbmZlcmVudGllIGJla29tZW4gdm9vciBncm90ZSBzdGVla3Byb2V2ZW4gYWxzIG1lbiBkZSB2YXJpYW50aWUgdmFuIGRlIHJlc3BvbnNlIGthbiBzY2hhdHRlbi4KLSBEZSBpbnZlcnNlIHZhcmlhbnRpZSB2b29yIGVsa2Ugb2JzZXJ2YXRpZSBrYW4gZGFuIGFscyBnZXdpY2h0IHdvcmRlbiBnZWJydWlrdCBpbiBkZSBsbSBmdW5jdGllLgoKMS4gV2UgenVsbGVuIGRhYXJvbSBkZSBzdGFuZGFyZCBkZXZpYXRpZSBtb2RlbGxlcmVuIGluIGZ1bmN0aWUgdmFuIGhldCBnZW1pZGRlbGRlLgoyLiBEYXQga2FuIGRvb3IgZGUgYWJzb2x1dGUgd2FhcmRlIHZhbiBkZSByZXNpZHVlbiB0ZSBtb2RlbGxlcmVuIGluIGZ1bmN0aWUgdmFuIGRlIGdlZml0dGUgd2FhcmRlbi4KMy4gV2Uga3VubmVuIGRlIHZhcmlhbnRpZSB2YW4gWSBzY2hhdHRlbiB2b29yIGVsa2Ugb2JzZXJ2YXRpZSBkLm0udiBkZSBrd2FkcmF0ZW4gdmFuIGRlIHByZWRpY3RpZXMgdm9vciBhbGxlIGRhdGEgcHVudGVuIGEuZC5oLnYgbW9kZWwgdm9vciBkZSBzdGFuZGFyZCBkZXZpYXRpZS4KNC4gRGUgaW5mZXJlbnRpZSBibGlqZnQgYXN5bXB0b3Rpc2NoIGdlbGRpZy4KCmBgYHtyfQptU2QgPC0gbG0obUJwMSRyZXMgJT4lIGFicyB+IG1CcDIkZml0dGVkKQpgYGAKCldlIHNjaGF0dGVuIGhldCBtb2RlbCBudSBvcG5pZXV3OgoKYGBge3J9Cm1CcDMgPC0gbG0oQlBTeXNBdmUgfiBBZ2UqR2VuZGVyLCBicERhdGEsIHcgPSAxL21TZCRmaXR0ZWReMikKYGBgCgpEZSByZXNpZHVlbiB2ZXJ0b25lbiBub2cgc3RlZWRzIGhldGVyb3NjZWRhc3RpY2l0ZWl0LgoKYGBge3J9CmRhdGEuZnJhbWUocmVzaWR1YWxzID0gbUJwMyRyZXNpZHVhbHMsIGZpdCA9IG1CcDMkZml0dGVkKSAlPiUKICBnZ3Bsb3QoYWVzKGZpdCxyZXNpZHVhbHMpKSArCiAgZ2VvbV9wb2ludCgpCmBgYAoKTmEgaGV0IGhlcnNjaGFsZW4gdmFuIGRlIHJlc2lkdWVuIGEuZC5oLnYuIGRlIHN0YW5kYXJkIGRldmlhdGllICh2ZXJtZW5pZ3Z1bGRpZ2VuIG1ldCB2aWVya2FudHN3b3J0ZWwgdmFuIGhldCBnZXdpY2h0KSB6aWpuIGRlIGdlc2NoYWFsZGUgcmVzaWR1ZW4gaG9tb3NjZWRhc3Rpc2NoLgoKRGUgcGFyYW1ldGVycyB3b3JkZW4gZ2VzY2hhdCBkb29yIGRlIGdld29nZW4ga2xlaW5zdGUga3dhZHJhdGVuIHRlY2huaWVrLgoKXFsgU1NFID0gXHN1bVxsaW1pdHNfe2k9MX1ebiB3X2kgZV9pXjJcXQoKbWV0ICR3X2kgPSAxL1xoYXQgXHNpZ21hXjJfaSQuCgpEZSBnZXdvZ2VuIHJlZ3Jlc3NpZSB6YWwgZHVzIGNvcnJlY3QgcmVrZW5pbmcgaG91ZGVuIG1ldCBoZXRlcm9zY2VkYXN0aWNpdGVpdC4KCmBgYHtyfQpkYXRhLmZyYW1lKHNjYWxlZF9yZXNpZHVhbHMgPSBtQnAzJHJlc2lkdWFscy9tU2QkZml0dGVkLCBmaXQgPSBtQnAzJGZpdHRlZCkgJT4lCiAgZ2dwbG90KGFlcyhmaXQsc2NhbGVkX3Jlc2lkdWFscykpICsKICBnZW9tX3BvaW50KCkKYGBgCgojIyMgQmVzbHVpdHZvcm1pbmcKCjxpZnJhbWUgd2lkdGg9IjU2MCIgaGVpZ2h0PSIzMTUiCnNyYz0iaHR0cHM6Ly93d3cueW91dHViZS5jb20vZW1iZWQvR0xhbjEwSDRrdWsiCmZyYW1lYm9yZGVyPSIwIgpzdHlsZT0iZGlzcGxheTogYmxvY2s7IG1hcmdpbjogYXV0bzsiCmFsbG93PSJhdXRvcGxheTsgZW5jcnlwdGVkLW1lZGlhIiBhbGxvd2Z1bGxzY3JlZW4+PC9pZnJhbWU+CgpgYGB7cn0Kc3VtbWFyeShtQnAzKQpgYGAKCkRlIG9uZGVyem9la3N2cmFnZW4gdmVydGFsZW4gemljaCBpbiBkZSB2b2xnZW5kZSBudWxsaHlwb3RoZXNlczoKCjEuIEFzc29jaWF0aWUgdHVzc2VuIGJsb2VkZHJ1ayBlbiBsZWVmdGlqZCBiaWogZGUgdnJvdXdlbj8gXFtIXzA6IFxiZXRhX1x0ZXh0e0FnZX0gPSAwIFx0ZXh0eyB2cyB9IEhfMTogXGJldGFfXHRleHR7QWdlfSBcbmVxIDAgXF0KCjIuIEFzc29jaWF0aWUgdHVzc2VuIGJsb2VkZHJ1ayBlbiBsZWVmdGlqZCBiaWogZGUgbWFubmVuPyBcW0hfMDogXGJldGFfXHRleHR7QWdlfSArIFxiZXRhX1x0ZXh0e0FnZTpHZW5kZXJtYWxlfSA9IDAgXHRleHR7IHZzIH0gSF8xOiBcYmV0YV9cdGV4dHtBZ2V9ICsgXGJldGFfXHRleHR7QWdlOkdlbmRlcm1hbGV9IFxuZXEgMCBcXQoKMy4gSXMgZGUgQXNzb2NpYXRpZSB0dXNzZW4gYmxvZWRkcnVrIGVuIGxlZWZ0aWpkIHZlcnNjaGlsbGVuZCBiaWogbWFubmVuIGVuIHZyb3V3ZW4/IFxbSF8wOiBcYmV0YV9cdGV4dHtBZ2U6R2VuZGVybWFsZX0gPSAwIFx0ZXh0eyB2cyB9IEhfMTogXGJldGFfXHRleHR7QWdlOkdlbmRlcm1hbGV9IFxuZXEgMCBcXQoKCi0gV2Uga3VubmVuIG9uZGVyem9la3N2cmFhZyAxIGVuIDMgb25taWRkZWxpamsgdG9ldHNlbiBvLmIudi4gZGUgbW9kZWwgb3V0cHV0LgotIE9uZGVyem9la3N2cmFhZyAyIGlzIGVjaHRlciBlZW4gbGluZWFpcmUgY29tYmluYXRpZSB2YW4gdHdlZSBwYXJhbWV0ZXJzLgotIEJvdmVuZGllbiBpcyBlciBvb2sgaGV0IHByb2JsZWVtIGRhdCB3ZSBtZWVyZGVyZSB0b2V0c2VuIG5vZGlnIGhlYmJlbiBvbSBkZSBhc3NvY2lhdGllIHRlIGJlc3R1ZGVyZW4uCgpXZSBrdW5uZW4gb3BuaWV1dyBnZWJydWlrIG1ha2VuIHZhbiBlZW4gQW5vdmEgYXBwcm9hY2guCgoxLiAgV2UgdG9ldHNlbiBlZXJzdGUgZGUgb21uaWJ1cyBoeXBvdGhlc2UgZGF0IGVyIGdlZW4gYXNzb2NpYXRpZSBpcyB0dXNzZW4gbGVlZnRpamQgZW4gZGUgYmxvZWRkcnVrLgoKXFsgSF8wOiBcYmV0YV9cdGV4dHtBZ2V9ID0gXGJldGFfXHRleHR7QWdlfSArIFxiZXRhX1x0ZXh0e0FnZTpHZW5kZXJtYWxlfSA9IFxiZXRhX1x0ZXh0e0FnZTpHZW5kZXJtYWxlfSA9IDAKXF0KCi0gRGF0IHZlcmVlbnZvdWRpZ3QgemljaCB0b3QgaGV0IHRvZXRzZW4gZGF0CgpcWyBIXzA6IFxiZXRhX1x0ZXh0e0FnZX0gPSBcYmV0YV9cdGV4dHtBZ2U6R2VuZGVybWFsZX0gPSAwClxdCgotIFdhdCB3ZSBrdW5uZW4gZXZhbHVlcmVuIGRvb3IgdHdlZSBtb2RlbGxlbiB0ZSB2ZXJnZWxpamtlbi4gRWVuIG1vZGVsIG1ldCBlbmtlbCBoZXQgZ2VuZGVyIGVmZmVjdCBlbiB2b2xsZWRpZ2UgbW9kZWwgbWV0IEdlbmRlciwgQWdlIGVuIEdlbmRlciB4IEFnZSBpbnRlcmFjdGllLgoKMi4gQWxzIHdlIGRlemUgaHlwb3RoZXNlIGt1bm5lbiB2ZXJ3ZXJwZW4gdm9lcmVuIHdlIHBvc3Rob2MgYW5hbHlzZXMgdWl0IHZvb3IgZWxrIHZhbiBkZSAzIGNvbnRyYXN0ZW4uCgoKIyMjIyBPbW5pYnVzIHRlc3QKCmBgYHtyfQptQnAwIDwtIGxtKEJQU3lzQXZlIH4gR2VuZGVyLCBicERhdGEsIHcgPSAxL21TZCRmaXR0ZWReMikKYW5vdmEobUJwMCwgbUJwMykKYGBgICAKCgojIyMjIFBvc3Rob2MgdGVzdGVuCgpEZSBwb3N0aG9jIHRlc3RlbiBrdW5uZW4gd2Ugb3BuaWV1dyB1aXR2b2VyZW4gYS5kLmgudi4gaGV0IG11bHRjb21wIHBha2tldC4KCmBgYHtyfQpsaWJyYXJ5KG11bHRjb21wKQpicFBvc3Rob2MgPC0gZ2xodChtQnAzLCBsaW5mY3Q9YygKICAiQWdlID0gMCIsCiAgIkFnZSArIEFnZTpHZW5kZXJtYWxlID0gMCIsCiAgIkFnZTpHZW5kZXJtYWxlID0gMCIpCiAgKQpicFBvc3Rob2MgJT4lIHN1bW1hcnkKCmJwUG9zdGhvY0JJIDwtIGJwUG9zdGhvYyAlPiUgY29uZmludApicFBvc3Rob2NCSQpgYGAKCk1lcmsgb3AgZGF0IGRlIGdsaHQgZnVuY3RpZSBvbnMgdG9lbGFhdCBvbSBkZSBjb250cmFzdGVuIHRlIGRlZmluacOrcmVuIGRvb3IgZGUgbnVsaHlwb3RoZXNlcyBleHBsaWNpZXQgdGUgZm9ybXVsZXJlbiBpbiBlZW4ga2FyYWt0ZXJ2ZWN0b3Igd2FhcmJpaiBnZWJydWlrIHdvcmR0IGdlbWFha3QgdmFuIGRlIG5hYW0gdmFuIGRlIHBhcmFyYW1ldGVycyBpbiBoZXQgbW9kZWwuCgojIyMjIENvbmNsdXNpZQoKV2Uga3VubmVuIGJlc2x1aXRlbiBkYXQgZXIgZWVuIGV4dHJlZW0gc2lnbmlmaWNhbnRlIGFzc29jaWF0aWUgaXMgdHVzc2VuIGxlZWZ0aWpkIGVuIGRlIGJsb2VkZHJ1ayAocCA8PCAwLjAwMSkuCkRlIGJsb2VkZHJ1ayBiaWogdHdlZSB2cm91d2VuIGRpZSBpbiBsZWVmdGlqZCB2ZXJzY2hpbGxlbiBpcyBnZW1pZGRlbGQgYHIgcm91bmQoYnBQb3N0aG9jQkkkY29uZmludFsxLDFdLDIpYCBtbSBIZyBob2dlciBwZXIgamFhciBsZWVmdGlqZHN2ZXJzY2hpbCBiaWogZGUgb3Vkc3RlIHZyb3V3IGVuIGRhdCB2ZXJzY2hpbCBpcyBleHRyZWVtIHNpZ25pZmljYW50IChwIDw8IDAuMDAxLCA5NSUgQkkgW2ByIHBhc3RlKHJvdW5kKGJwUG9zdGhvY0JJJGNvbmZpbnRbMSwtMV0sMiksY29sbGFwc2U9IiwgIilgXS4KRGUgYmxvZWRkcnVrIGJpaiBtYW5uZW4gZGllIGluIGxlZWZ0aWpkIHZlcnNjaGlsbGVuIGlzIGdlbWlkZGVsZCBgciByb3VuZChicFBvc3Rob2NCSSRjb25maW50WzIsMV0sMilgIG1tIEhnIGhvZ2VyIHBlciBqYWFyIGxlZWZ0aWpkc3ZlcnNjaGlsIGJpaiBkZSBvdWRlcmUgbWFuLiAocCA8PCAwLjAwMSwgOTUlIEJJIFtgciBwYXN0ZShyb3VuZChicFBvc3Rob2NCSSRjb25maW50WzIsLTFdLDIpLGNvbGxhcHNlPSIsICIpYF0uCkhldCBnZW1pZGRlbGRlIGJsb2VkZHJ1a3ZlcnNjaGlsIHR1c3NlbiBwZXJzb25lbiBpbiBsZWVmdGlqZCB2ZXJzY2hpbGxlbiBpcyBnZW1pZGRlbGQgYHIgcm91bmQoYnBQb3N0aG9jQkkkY29uZmludFszLDFdLDIpYCBtbSBIZy9qYWFyIGhvZ2VyIGJpaiB2cm91d2VuIGRhbiBtYW5uZW4gKHAgPDwgMC4wMDEsIDk1JSBCSSBbYHIgcGFzdGUocm91bmQoYnBQb3N0aG9jQkkkY29uZmludFszLC0xXSwyKSxjb2xsYXBzZT0iLCAiKWBdKS4gIAoKIyBbSG9tZV0oaHR0cHM6Ly9zdGF0b21pY3MuZ2l0aHViLmlvL3NiYzIwLykgey19Cg==