Hands-on RNA-seq Analysis from fastq

Sheffield Bioinformatics Core

web : sbc.shef.ac.uk
twitter: [@SheffBioinfCore](https://twitter.com/SheffBioinfCore)
email: bioinformatics-core@sheffield.ac.uk


Tutorial Overview

This tutorial will cover the basics of RNA-seq using Galaxy and Degust; two open-source web-based platforms for the analysis of biological data. You should gain an appreciation of the tasks involved in a typical RNA-seq analysis and be comfortable with the outputs generated by the Bioinformatician.

Using these notes

Sections with this background indicate exercises to be completed during the workshop.

Sections with this background highlight particular shortcuts or other references that might be useful.

Sections with this background give information about potential error messages or might encounter, or problems that might arise in your own data analysis.

More on Galaxy

The official Galaxy page has many tutorials on using the service, and examples of other types of analysis that can be performed on the platform.

Those eventually wanted to perform their own RNA-seq analysis (for example in R), should look out for other courses

Courses on analysing RNA-seq data in R

Techniques not covered in the workshop

From our experience, the most common application of RNA-seq is still to perform a “differential expression” in order to identify genes, and eventually pathways, that are altered between a set of biological conditions. Therefore we will concentrate on this task in the workshop. We will also be considering bulk RNA-seq only. i.e. where our biological sample may be comprised of a pool of heterogeneous cells. Single-cell approaches are becoming more popular, and although there are some similarities in how these data are processed, a different downstream analysis approach is required.

The Galaxy Training Network provides materials on single-cell analysis, and other applications not covered in this workshop.

RNA-seq workflow

Two workflows are possible with RNA-seq data - with the difference being whether one performs an alignment to the reference genome or not.

Recent tools for RNA-seq analysis (e.g. salmon, kallisto) do not require the time-consuming step of whole-genome alignment to be performed, and can therefore produce gene-level counts in a much faster time frame. They not require the creation of large bam files, which is useful if constrained by file space on Galaxy.

(image from Harvard Bioinformatics Core)


Background

Where do the data in this tutorial come from?

The data for this tutorial comes from a Journal of Experimental Medicine paper “Itraconazole targets cell cycle heterogeneity in colorectal cancer”. This study examines the expression profiles of two cell lines in response to treatment with itraconazole.

For this tutorial, we will assume that the wet-lab stages of the experiment have been performed and in this tutorial we will demonstrate the steps of Quality assessment, alignment, quantification and differential expression testing.

The fastq data for this experiment were made available on the Sequencing Read Archive (SRA) with accession SRP144496. For the purposes of this workshop we have created a downsampled dataset

The experimental design for the dataset is summarised in the table below.

run name cell_line condition
SRR7108388 HT55_CONT_1 HT55 DMSO
SRR7108389 HT55_CONT_2 HT55 DMSO
SRR7108390 HT55_CONT_3 HT55 DMSO
SRR7108391 HT55_CONT_4 HT55 DMSO
SRR7108392 HT55_ITRA_1 HT55 ITRACONAZOLE
SRR7108393 HT55_ITRA_2 HT55 ITRACONAZOLE
SRR7108394 HT55_ITRA_3 HT55 ITRACONAZOLE
SRR7108395 HT55_ITRA_4 HT55 ITRACONAZOLE
SRR7108396 SW948_CONT_1 SW948 DMSO
SRR7108397 SW948_CONT_2 SW948 DMSO
SRR7108398 SW948_CONT_3 SW948 DMSO
SRR7108399 SW948_CONT_4 SW948 DMSO
SRR7108400 SW948_ITRA_1 SW948 ITRACONAZOLE
SRR7108401 SW948_ITRA_2 SW948 ITRACONAZOLE
SRR7108402 SW948_ITRA_3 SW948 ITRACONAZOLE
SRR7108403 SW948_ITRA_4 SW948 ITRACONAZOLE

Digression: How to download raw sequencing files

The Sequencing Read Archive (SRA) is commonly-used to store the raw data from sequencing experiments and can be accessed through the NCBI website. However, the interface is not particularly friendly and the links to download data and not easy to obtain.

An easier alternative exists in the form of SRA Explorer

The SRA accession (usually found in a paper describing the dataset) can be entered into the Search box, and all the samples belonging to that dataset should be found. Samples of interest can be saved, and upon “checkout” the download links (URLs) will be displayed. A command-line tool such as curl or wget can then be used to download the files locally.

Section 1: Preparation

1. Sign-up to the European Galaxy server

Make sure you check your email to activate your account

2. Download the course data

The data for this course have all been shared on a google drive. If you have not done so already, please go to this directory and download the following files

https://drive.google.com/drive/folders/1RSuvl9shAw12Bj77uYSUdWtkZ5ST5EWi?usp=sharing

  • SRR7108388.fastq.gz
  • SRR7108389.fastq.gz
  • SRR7108392.fastq.gz
  • SRR7108393.fastq.gz
  • Homo_sapiens.GRCh38.cdna.all.fa.gz
  • tx2gene.txt

3. Import the RNA-seq data for the workshop.

We can going to import the fastq files for this experiment. This is a standard format for storing raw sequencing reads and their associated quality scores. To make the practical quicker, we have downsampled the original fastq files to a quarter of a million reads.

Get Data -> Upload File

You can import the data by:

  1. In the tool panel located on the left, under Basic Tools select Get Data > Upload File. Click on the Choose local file button on the bottom section of the pop-up window.
  2. Navigate to the fastq directory of the zip file that you downloaded from google drive and select these two files from the HT55-DMSO condition.

SRR7108388.fastq.gz SRR7108389.fastq.gz

and these two files are from the HT55 ITRACONAZOLE condition.

SRR7108392.fastq.gz SRR7108393.fastq.gz

also upload the files Homo_sapiens.GRCh38.cdna.all.fa.gz and tx2gene.txt. These are reference files that we will use later.

  1. You should now have these 4 fastq files in your history:
    • SRR7108388.fastq.gz
    • SRR7108389.fastq.gz
    • SRR7108392.fastq.gz
    • SRR7108393.fastq.gz

The annotation files may take a while longer to upload

The .gz at the end of each file name means that it is compressed (like a zip file).

You can upload the other files for extra practice if you wish

Alternative upload

To save time, the raw data for this dataset have been uploaded to Galaxy. To access the history, use the link below

There should be a button to import the history in the top-right corner. The raw data for the practical should now be available to you.

Quality assessment with FastQC (Optional)

FastQC is a popular tool from Babraham Institute Bioinformatics Group used for quality assessment of sequencing data. Most Bioinformatics pipelines will use FastQC, or similar tools in the first stage of the analysis. The documentation for FastQC will help you to interpret the plots and stats produced by the tool. A traffic light system is used to alert the user’s attention to possible issues. However, it is worth bearing in mind that the tool is blind to the particular type of sequencing you are performing (i.e. whole-genome, ChIP-seq, RNA-seq), so some warnings might be expected due to the nature of your experiment.

FastQ Quality Control -> FastQC Read Quality reports

  • Select one of the FASTQ files as input and Execute the tool.
  • When the tool finishes running, you should have an HTML file in your History. Click on the eye icon to view the various quality metrics.
  • Run Fastqc on the remaing fastq files, but don’t examine the results just yet.

Question: Do the data seem to be of reasonable quality?

You can use the documentation to help interpret the plots

If poor quality reads towards the ends of reads are considered to be a problem, or there is considerable adapter contamination, we can employ various tools to trim our data.

However, a recent paper demonstrated that read trimming is no longer required prior to alignment:- https://www.biorxiv.org/content/10.1101/833962v1

If you also suspect contamination by another organism, or rRNA present in your data, you can use the sortMeRNA tool to remove this artefact.

Combining QC reports

It can be quite tiresome to click through multiple QC reports and compare the results for different samples. It is useful to have all the QC plots on the same page so that we can more easily spot trends in the data.

The multiqc tool has been designed for the tasks of aggregating qc reports and combining into a single report that is easy to digest.

FASTQ Quality Control -> Multiqc

Under Which tool was used generate logs? Choose fastqc and select the RawData output from the fastqc run on each of your bam files.

Question: Repeat the FastQC analysis for the remaining fastq files and combine the reports with multiQC. Do the fastq files seem to have consistently high-quality?

Section 2: Quantification

Traditionally, workflows for RNA-seq would align reads to a reference genome, and then overlap with know gene coordinates. However, many now prefer to align directly to the transcriptome sequences using a method such as salmon or kallisto. We will demonstrate the salmon protocol.

Obtaining the files for salmon

  1. cDNA fasta file

As we are going to align to a set of transcripts rather than the whole genome, we require a file that contains the sequence of each transcript. This file has been provided in the google drive folder and should have been uploaded to your Galaxy history.

However, it is useful to know where this file came from in case you are not working with Human data. The file was obtained from Ensembl by clicking on the cDNA (FASTA) link for the appropriate organism (Human).

On the next screen, Right-click to save the .cdna.all.fa.gz to your computer

The FASTA file is a large text file that lists all the transcripts for the given organism and their genomic sequence. You could open this in a standard text editor if you wished to see the contents. The contents are similar to that of a FASTQ file. In fact, the FASTQ file is a FASTA file with extra quality scores added.

The identifier line for each sequence (starting with >) names the transcript and the gene it is associated with. Since we obtained the file from Ensembl, the Transcripts and Genes begin with the ENST and ENSG respectively. The numbers after each transcript or gene are the version numbers; the sequence and definition of each transcript / gene can evolve over time.

>ENST00000631435.1 cdna chromosome:GRCh38:CHR_HSCHR7_2_CTG6:142847306:142847317:1 gene:ENSG00000282253.1 gene_biotype:TR_D_gene transcript_biotype:TR_D_gene gene_symbol:TRBD1 description:T cell receptor beta diversity 1 [Source:HGNC Symbol;Acc:HGNC:12158]
GGGACAGGGGGC
>ENST00000415118.1 cdna chromosome:GRCh38:14:22438547:22438554:1 gene:ENSG00000223997.1 gene_biotype:TR_D_gene transcript_biotype:TR_D_gene gene_symbol:TRDD1 description:T cell receptor delta diversity 1 [Source:HGNC Symbol;Acc:HGNC:12254]
GAAATAGT
>ENST00000448914.1 cdna chromosome:GRCh38:14:22449113:22449125:1 gene:ENSG00000228985.1 gene_biotype:TR_D_gene transcript_biotype:TR_D_gene gene_symbol:TRDD3 description:T cell receptor delta diversity 3 [Source:HGNC Symbol;Acc:HGNC:12256]
ACTGGGGGATACG
>ENST00000434970.2 cdna chromosome:GRCh38:14:22439007:22439015:1 gene:ENSG00000237235.2 gene_biotype:TR_D_gene transcript_biotype:TR_D_gene gene_symbol:TRDD2 description:T cell receptor delta diversity 2 [Source:HGNC Symbol;Acc:HGNC:12255]
CCTTCCTAC
>ENST00000632684.1 cdna chromosome:GRCh38:7:142786213:142786224:1 gene:ENSG00000282431.1 gene_biotype:TR_D_gene transcript_biotype:TR_D_gene gene_symbol:TRBD1 description:T cell receptor beta diversity 1 [Source:HGNC Symbol;Acc:HGNC:12158]
GGGACAGGGGGC
  1. Transcript mapping file

By default, salmon will produce counts for each transcript. This might be what we want, but for most standard analyses it is preferable to work at the gene-level. We therefore have to tell salmon how the transcripts in the cDNA file relate to known genes. Such a file can be obtained from biomart.

  • Select the Ensembl genes (104) database
  • Select the dataset Human genes (GRCh38.13)
  • In Attributes, click the “+” button next to GENE and select Transcript stable ID version and Gene stable ID. Make sure the order on the left-hand panel is Transcript stable ID version, followed by Gene stable ID. This will affect the column order in the file. You can tick / untick the IDs to make sure the order is correct.

  • Click the Results button in the top left corner to see a preview of the results. Clicking the Go button will export the results to a file

It is important that the file downloaded from Biomart is edited so that the column headings do not contain any spaces. You can do this in a text editor. The edited file should look like this.

It is important to make sure the version number of your transcript file and the biomaRt dataset are the same, otherwise some of the steps downstream might not work as expected.

If you have problems, this mapping file is also provided in the google drive as tx2gene.txt. The contents of the first column have to be in the same format as the transcript names in the fasta file. i.e. in this case the version number must be present.

  1. Annotation file

The Ensembl gene IDs are not particularly memorable, so it would be highly beneficial to have other annotations at hand to help us interpret the data downstream. We can use the biomart website again to produce a table to aid downstream interpretation.

This time, select only the Gene Stable ID tickbox in the GENE box. Expand the EXTERNAL panel by clicking the “+” next to EXTERNAL, and select HGNC symbol and NCBI gene (formerly Entrezgene) ID

salmon configuration and running

RNA Analysis -> Salmon quant

  • Select the Homo_sapiens_GRCh38.cdna.all.fa.gz file as the Transcripts fasta file
  • Select all your uploaded fastq files as your Data Input FASTQ/FASTA file
  • Scroll down to File containing a mapping of transcripts to genes and select the tx2gene.txt file from your history

Two jobs will now be queued for each sample fastq file. The Quantification output will contain transcript-level data, and the Gene Quantification output will be at the gene-level. We should expect the number of lines in the Gene Quantification file to be substantially less. If not, you will need to check that your transcript mapping file was correct.

The Gene Quantification output from each sample comprises the following columns (taken from the salmon documentation)

  • Name — This is the name of the target transcript provided in the input transcript database (FASTA file).
  • Length — This is the length of the target transcript in nucleotides.
  • EffectiveLength — This is the computed effective length of the target transcript. It takes into account all factors being modeled that will effect the probability of sampling fragments from this transcript, including the fragment length distribution and sequence-specific and gc-fragment bias (if they are being modeled).
  • TPM — This is salmon’s estimate of the relative abundance of this transcript in units of Transcripts Per Million (TPM). TPM is the recommended relative abundance measure to use for downstream analysis.
  • NumReads — This is salmon’s estimate of the number of reads mapping to each transcript that was quantified. It is an “estimate” insofar as it is the expected number of reads that have originated from each transcript given the structure of the uniquely mapping and multi-mapping reads and the relative abundance estimates for each transcript.

Note that we are using a downsampled dataset, so the majority of NumReads will be zero.

Create a count matrix

Methods for detecting differential expression are likely to want data in the form of a table; where every row is a different gene and each column is a unique biological sample. Before we can proceed we will therefore need to merge our salmon results into a single output. This can be down using the Salmon quantmerge tool

RNA Analysis -> Salmon quantmerge

Use the +Insert Quant file and names button repeatedly to select each of your Gene Quantification outputs. The One-word sample names text box can be used to create a shorter column name for each output.

Once all the Gene Quantification files have been selected the drop-down menu under Columns should be changed from Length to NumReads.

The first step in merging our salmon output is to produce a table for each sample that contains just the gene name and the number of reads for that gene. This can be done with the advanced cut tool

Text Manipulation -> Advanced Cut columns from a table

  • In File to cut, select all the salmon Gene Quantification outputs
  • In the List of Fields box under Cut by, select Column: 1 and Column: 5.

These outputs can be merged using the Column join tool

Collection Operations -> Column join on multiple datasets

  • Select the Advanced Cut outputs and keep the Identifier column as 1. This will “join” the files together based on the gene names; which hopefully should be the same for each sample.

After the tool has finished you should have an output table with one row for each gene and a column containing the counts for that gene.

Exercise: Upload the annotation file from biomart containing Ensembl Gene ID, hgnc and Entrez. Use the column join tool to create a table that allows you to identify the counts for given genes more easily.

(Optional) Alternative workflow involving genome alignment

If time allows, we will also follow this section

The workflow that people used for many years is summarised in this image from Ting-you Wang’s RNA-seq data analysis page, and may still be preferable if your analysis doesn’t just call for gene-level counts.

Mapping -> HISAT2

1. Map/align the reads with HISAT2 to the hg38 reference genome

In the left tool panel menu, under NGS Analysis, select Mapping > HISAT2 and set the parameters as follows:

  • Is this single-end or paired-end data? Single-end (as individual datasets)
  • FASTQ file
    (Click on the multiple datasets icon and select all four of the FASTQ files)
    • SRR7108388.fastq.gz
    • SRR7108389.fastq.gz
    • SRR7108392.fastq.gz
    • SRR7108393.fastq.gz
  • Source for the reference genome Use built-in genome
  • Select a reference genome: Human Dec 2013. (GRCh38/hg38) (hg38)
  • Use defaults for the other fields
  • Execute

Quantification (Counting reads in features)

HTSeq-count creates a count matrix using the number of the reads from each bam file that map to the genomic features. For each feature (a gene for example) a count matrix shows how many reads were mapped to this feature.

Various rules can be used to assign counts to features

To obtain the coordinates of each gene, we can use the UCSC genome browser which is integrated into Galaxy. Unfortunately, the Ensembl FTP site cannot be used as the chromosome naming conventions used in Ensembl are different to the chromosome naming scheme used in the reference genomes supplied by Galaxy (1,2,.. vs chr1, chr2). The alternative would be to download a matching gtf and genome reference sequence from Ensembl and upload both to Galaxy. This would take more time and space on the Galaxy server.

Obtaining gene coordinates

Get Data -> UCSC Main table browser

Selecting the UCSC Main tool from Galaxy will take you to the UCSC table browser. From here we can extract gene coordinates for our genome of interest (hg38) in gtf format for processing with galaxy.

  • Set clade to Mammal
  • Set genome to Human
  • assembly Dec.2013 (GRCm38/hg38)
  • group Genes and Gene Prediction
  • track NCBI RefSeq
  • table UCSC RefSeq (refGene)
  • region genome
  • output format GTF - gene transfer format (limited) and send output to Galaxy

Click get output and send query to Galaxy to be returned to Galaxy. A new job will be submitted to retrieve the coordinates from UCSC

When you are returned to Galaxy from UCSC it might look like you have lost all th files in your analysis and are no longer logged in.

To solve this, log back in and choose the View all histories option under the History panel.

There should be two “histories”; one containing all the outputs you generated before accessing UCSC, and one containing the UCSC output. All this point you can switch back to your previous history, and drag the box containing the UCSC ouput to this history

Counting reads in genes

RNA Analysis > htseq-count

  1. Use HTSeq-count to count the number of reads for each feature.
    In the left tool panel menu, under NGS Analysis, select NGS Analysis > htseq-count and set the parameters as follows:
    • Aligned SAM/BAM file
      (Select all four bam files using the multiple datasets option)
    • GFF file UCSC Main on Human:refGene (genome)
    • Use defaults for the other fields
    • Execute
  2. Repeat for the remaining bam files if running on each bam separately.
  3. To make things easier to track, rename the ht-seq output for each sample to contain the corresponding sample name (e.g. SRR1552444.htseq). Do not rename the outputs that have “(no feature)” in their name

Create a count matrix

The htseq tool is designed to produce a separate table of counts for each sample. This is not particularly useful for other tools such as Degust (see next section) which require the counts to be presented in a data matrix where each row is a gene and each column is a particular sample in the dataset.

Collection Operations -> Column Join on Collections

  • In the Tabular Files section, select the ht-seq count files from your history SRR1552444.htseq, SRR1552450, etc… Holding the CTRL key allows multiple files to be selected
  • Keep Identifier column as 1

Differential Expression using Degust

Differential expression is possible using Galaxy using the DESeq2 tool (for example). However, our particular recommendation is to use Degust for a more interactive experience. For this section, we will be using counts generated on the full dataset, rather than the downsampled data analysed in the previous section. These counts are available in the file GSE114013_salmon_counts.csv from the google drive, or can be downloaded using the link below

Differential expression

The term differential expression was first used to refer to the process of finding statistically significant genes from a microarray gene expression study.

Such methods were developed on the premise that microarray expression values are approximately normally-distributed when appropriately transformed (e.g. by using a log\(_2\) transformation) so that a modified version of the standard t-test can be used. The same test is applied to each gene under investigation yielding a test statistic, fold-change and p-value. Similar methods have been adapted to RNA-seq data to account for the fact that the data are count-based and do not follow a normal distribution.

http://degust.erc.monash.edu/

Degust is a web tool that can analyse the counts files produced in the step above, to test for differential gene expression. It offers and interactive view of the differential expression results

The input file is a count matrix where each row is a measured gene, and each column is a different biological sample. Within the tool we can configure which samples belong to the different biological groups of interest. It is important that no normalisation has been applied to the counts that are submitted to DEGUST.

Read counts have to be normalised first prior to differential expression testing. This page from Harvard Bioinformatics summarises the main biases in “raw” RNA-seq counts; Sequencing depth, Gene length and RNA composition

Methods such as edgeR (implemented in Degust) and DESeq2 have their own method of normalising counts. You will probably encounter other methods of normalising RNA-seq reads such as RPKM, CPM, TPM etc. This blog provides a nice explanation of the current thinking. As part of the Degust output, you have the option of downloading normalised counts in various formats. Some other online visualisation tools require normalised counts as input, so it is good to have these to-hand.

Uploading the count matrix to Degust

  • From the main degust page, click Upload your counts file
  • Click on Browse
  • Select the location of the file GSE114013_salmon_counts.csv, and click Open.
  • Click Upload
  • A Configuration page will appear.

  • For Name type “GSE114013” (or whatever you want to call the analysis)
  • For Info columns select SYMBOL
  • Click Add condition
    • Referring to the experiment design (below), select the DMSO samples and call the condition DMSO
    • Repeat for the ITRACONAZOLE samples. We will refer to this as ITRA for short.
  • Save the settings and then View the results
run name cell_line condition
SRR7108388 HT55_CONT_1 HT55 DMSO
SRR7108389 HT55_CONT_2 HT55 DMSO
SRR7108390 HT55_CONT_3 HT55 DMSO
SRR7108391 HT55_CONT_4 HT55 DMSO
SRR7108392 HT55_ITRA_1 HT55 ITRACONAZOLE
SRR7108393 HT55_ITRA_2 HT55 ITRACONAZOLE
SRR7108394 HT55_ITRA_3 HT55 ITRACONAZOLE
SRR7108395 HT55_ITRA_4 HT55 ITRACONAZOLE
SRR7108396 SW948_CONT_1 SW948 DMSO
SRR7108397 SW948_CONT_2 SW948 DMSO
SRR7108398 SW948_CONT_3 SW948 DMSO
SRR7108399 SW948_CONT_4 SW948 DMSO
SRR7108400 SW948_ITRA_1 SW948 ITRACONAZOLE
SRR7108401 SW948_ITRA_2 SW948 ITRACONAZOLE
SRR7108402 SW948_ITRA_3 SW948 ITRACONAZOLE
SRR7108403 SW948_ITRA_4 SW948 ITRACONAZOLE

Overview of Degust sections

  • Top black panel with Configure settings at right. Used to change the experimental conditions
  • Left: Control over which conditions are compared and the cut-offs for deciding DE genes.
  • Top centre: Interactive plots (see below)
  • A table of genes (or features); expression in “treatment” relative to “control”; and significance (FDR column).

(Not that the screenshots are for illustration purposes and taken from a different dataset to that being analysed in the tutorial.)

MA-plot

Each dot shows the change in expression in one gene.

  • The average expression (over both condition and treatment samples) is represented on the x-axis.
    • Plot points should be symmetrical around the x-axis.
    • We can see that many genes are expressed at a low level, and some are highly expressed.
  • The fold change is represented on the y axis.
    • If expression is significantly different between batch and chem, the dots are red. If not, they are blue. (In Degust, significant means FDR <0.05).
    • At low levels of gene expression (low values of the x axis), fold changes are less likely to be significant.

Click on the dot to see the gene name. The panel on the right of the MA-plot will also update to show the expression of this gene in each sample in the form of a dot plot.

MDS plot

This is a multidimensional scaling plot which represents the variation between samples. It is a similar concept to a Principal Components Analysis (PCA) plot. The x-axis is the dimension with the highest magnitude. In a standard control/treatment setup, samples should be split along this axis. A desirable plot is shown below:-

Volcano plot

Similar to the MA-plot, this shows the significance of each gene (y-axis) and magnitude of change (x-axis).

Table of genes

Initially this will contain the details of all genes present in the dataset. Once the FDR and logFC cut-offs are altered, any genes that do not meet the cut-offs will be removed.

  • INFO; The columns that you selected in the configuration to contain gene information (e.g. gene name, gene symbol, Entrez ID)
  • FDR: False Discovery Rate. This is an adjusted p value to show the significance of the difference in gene expression between two conditions. Click on column headings to sort. By default, this table is sorted by FDR.
  • Treatment and Control: log2(Fold Change) of gene expression. The default display is of fold change in the treatment relative to the control. Therefore, values in the batch column are zero. This can be changed in the Options panel at the top right.
    • In some cases, a large fold change will be meaningful but in others, even a small fold change can be important biologically.

The table can be sorted according to any of the columns (e.g. fold-change or p-value)

Download and R code

Above the genes table is the option to download the results of the current analysis to a csv file. You can also download the R code required to reproduce the analysis by clicking the Show R code box underneath the Options box.

Exercise

Question: Make a note of how many genes are detected with a FDR < 0.05 and abs logFC > 1 It seems that the differential expression analysis is detecting lots of genes. However, does this tell the whole story about the dataset? What do you think is the main factor separating samples on the x-axis, and thus explaining the most variation in the data?

Modifying the analysis

We will now repeat the analysis, but only for samples from the HT55 cell-line. The correct configuration for this analysis is shown below:-

Exercise: How many genes are differentially-expressed with an FDR < 0.05 and abs logFC > 1. Download this file and rename it to HT55.ITRACONAZOLE_vs_DMSO.csv.

Exercise: Rest the FDR cut-off and abs LogFC cutoffs back to default (1 and 0 respectively) and download the file. Rename the file to background.csv. We will use this later.

Exercise: Repeat the analysis for SW948 samples and download the table of differentially-expressed results (same FDR and log fold-change) to SW948.ITRACONAZOLE_vs_DMSO.csv

File Downloads

If you didn’t manage to complete these analyses, you can download the files from here by right-clicking on each link and selecting “Save Link as” (or equivalent). They are also available in the course google drive.

Overlapping Gene Lists

We might sometimes want to compare the lists of genes that we identify using different methods, or genes identified from more than one contrast. In our example dataset we can compare the genes in the contrast of ITRACONAZOLE vs DMSO in HT55 and SW948 cells

The website venny provides a really nice interface for doing this.

  • Open both your HT55: ITRACONAZOLE vs DMSO and SW948: ITRACONAZOLE vs DMSO results files in Excel
  • Go to the venny website
  • Copy the names of genes in the HT55 file into the List 1 box on the venny website. List 1 can be renamed to HT55
    • You can select all entries in a column with the shortcut CTRL + SPACE
  • Copy the names of genes in the SW948 analysis into the List 2 box on the venny website. List 2 can be renamed to SW948
  • venny should now report the number of genes found in each list, the size of the intersection, and genes unique to each method

Refined analysis

The final analysis we will perform is to include all the samples, but correct for the differences in cell-line. This is achieved by telling Degust about the hidden factors in our dataset. The hidden factor in this dataset is whether the sample is from the HT55 or SW948 samples. In other words, this is a technical factor that influences our results but not a factor that we wish to compare. We only need to specify which samples are from HT55 and DEGUST will infer that the other samples belong to a different cell line. Other hidden factors you might need to include could be (depending on the MDS plot) :-

  • sample batch
  • gender

See below for the correct configuration to include the hidden factors.

Exercise: How many genes are identified with an FDR < 0.05 and abs logFC > 1 for this “hidden factor” analysis. How does it compare to the initial comparison of DMSO vs ITRA using all samples?

Enrichment and pathways analysis

In this section we move towards discovering if our results are biologically significant. Are the genes that we have picked statistical flukes, or are there some commonalities.

There are two different approaches one might use, and we will cover the theory behind both. The distinction is whether you are happy to use a hard (and arbitrary) threshold to identify DE genes.

Over-representation analysis

“Threshold-based” methods require the application of a statistical threshold to define list of genes to test (e.g. FDR < 0.01). Then a hypergeometric test or Fisher’s Exact test generally used. These are typically used in situations where plenty of DE genes have been identified, and people often use quite relaxed criteria for identifying DE genes (e.g. raw rather than adjusted p-values or FDR value)

The question we are asking here is;

“Are the number of DE genes associated with Theme X significantly greater than what we might expect by chance alone?”

or

“If I picked a set of genes at random that is the same as the number of DE genes, how many genes from Theme X would I expect to find”?

We can answer this question by knowing

  • the total number of DE genes
  • the number of genes in the gene set (pathway or process)
  • the number of genes in the gene set that are found to be DE
  • the total number of tested genes (background)

The formula for Fishers exact test is;

\[ p = \frac{\binom{a + b}{a}\binom{c +d}{c}}{\binom{n}{a +c}} = \frac{(a+b)!(c+d)!(a+c)!(b+d)!}{a!b!c!d!n!} \]

with:-

is DE Not DE Row Total
In Gene Set a b a + b
Not in Gene Set c d c + d
Column Total a + c b + d a + b + c + d = n

This formula is printed here for your information. The software we use will perform all the calculations for us.

In this first test, our genes will be grouped together according to their Gene Ontology (GO) terms:- http://www.geneontology.org/

Using GOrilla

There are several popular online tools for performing enrichment analysis. We will be using the online tool GOrilla to perform the pathways analysis as it is particularly straightforward. It has two modes; the first of which accepts a list of background and target genes.

  1. Go to http://cbl-gorilla.cs.technion.ac.il/
  2. Choose Organism: Homo Sapiens
  3. Choose running mode: Two unranked lists of genes
  4. Paste the gene symbols corresponding to DE genes in SW948: ITRACONAZOLE vs DMSO into the Target set.
  • The shortcut CTRL + SPACE will let you select an entire column
  1. Paste the gene symbols from the Background set into the other box. This should be the names of all genes present in the Background file. i.e. all genes that were tested.
  2. Choose an Ontology: Process
  3. Under advanced options, tick Output results in Microsoft Excel format
  4. Search Enriched GO terms

You should be presented with a graph of enriched GO terms showing the relationship between the terms. Each GO term is coloured according to its statistical significance.

Below the figure is the results table. This links to more information about each GO term, and lists each gene in the category that was found in your list. The enrichment column gives 4 numbers that are used to determine enrichment (similar to the Fisher exact test we saw earlier)

  • N, total number of genes (should be the same in all rows)
  • B, total number of genes annotated with the GO term
  • n, total number of genes that were found in the list you uploaded (same for all rows)
  • b, number of genes in the list you uploaded that intersect with this GO term

Exercise: Repeat the GOrilla to find enriched pathways in the HT55: ITRACONAZOLE vs DMSO analysis. What do you notice?

Threshold-free analysis

This type of analysis is popular for datasets where differential expression analysis does not reveal many genes that are differentially-expressed on their own. Instead, it seeks to identify genes that as a group have a tendency to be near the extremes of the log-fold changes. The results are typically presented in the following way.

The “barcode”-like panel represents where genes from a particular pathway (HALLMARK_E2F_TARGETS in this case) are located in a gene list ranked from most up-regulated to most down-regulated. The peak in the green curve is used to indicate where the majority of genes are located. If this is shifted to the left or the right it indicates that genes belonging to this gene set have a tendency to be up- or down-regulated.

As such, it does not rely on having to impose arbitrary cut-offs on the data. Instead, we need to provide a measure of the importance of each gene such as it’s fold-change. These are then used the rank the genes.

The Broad institute has made this analysis method popular and provides a version of GSEA that can be run via a java application. However, the application can be a bit fiddly to run, so we will use the GeneTrail website instead

  • Open the file background.csv in Excel and delete all columns except the SYMBOL and ITRA column.
  • Go to the GeneTrail website, and select Transcriptomics from the front page
  • Select the Paste the content of a text file in a tabular format option and the contents of your modified excel file into the box. Do not paste the column headings
  • Click Upload

Hopefully it should recognise your input without any errors, and on the next screen the Set-level statistic should be automatically set to GSEA

If your data does not get uploaded, double-check that the column heading ITRA has not been pasted into the text box

To make the analysis run faster, you can de-select the GO pathways (biological processes, molecular function and cellular compartment)

For your own analyses, you should consider analysing these categories. We are only de-selecting them here to make things run quicker

After a short wait, you will be able to view and download the results. The tested pathways are grouped into different sources (Kegg, Reactome or Wikipathways)

Each of the significant pathways can be explored in detail; such as showing which genes in that pathways are up- or downregulated.

The Rank of the gene shown is the position of the gene in the ranked list; with 1 being most up-regulated gene. The score is the score used to rank the genes (fold-change in our example).

Exercise: Explore the GeneTrail results. Does the method identify significant pathways for the HT55: ITRACONAZOLE vs DMSO analysis?

Recap

LS0tCnRpdGxlOiAiUk5BLXNlcSB0dXRvcmlhbCIKYXV0aG9yOiAiTWFyayBEdW5uaW5nIgpvdXRwdXQ6CiAgaHRtbF9ub3RlYm9vazoKICAgIHRvYzogeWVzCiAgICB0b2NfZmxvYXQ6IHllcwogICAgY3NzOiBzdHlsZXNoZWV0cy9zdHlsZXMuY3NzCiAgaHRtbF9kb2N1bWVudDoKICAgIGRmX3ByaW50OiBwYWdlZAogICAgdG9jOiB5ZXMKZWRpdG9yX29wdGlvbnM6CiAgY2h1bmtfb3V0cHV0X3R5cGU6IGlubGluZQotLS0KCgoKIyBIYW5kcy1vbiBSTkEtc2VxIEFuYWx5c2lzIGZyb20gZmFzdHEKCgojIyMgU2hlZmZpZWxkIEJpb2luZm9ybWF0aWNzIENvcmUKPGltZyBzcmM9Im1lZGlhL2xvZ28tc20ucG5nIiBhbGlnbj1yaWdodD4KCndlYiA6IFtzYmMuc2hlZi5hYy51a10oaHR0cHM6Ly9zYmMuc2hlZi5hYy51aykgIAp0d2l0dGVyOiBbQFNoZWZmQmlvaW5mQ29yZV0oaHR0cHM6Ly90d2l0dGVyLmNvbS9TaGVmZkJpb2luZkNvcmUpICAKZW1haWw6IFtiaW9pbmZvcm1hdGljcy1jb3JlQHNoZWZmaWVsZC5hYy51a10oYmlvaW5mb3JtYXRpY3MtY29yZUBzaGVmZmllbGQuYWMudWspCgotLS0tLQoKIyMgVHV0b3JpYWwgT3ZlcnZpZXcKClRoaXMgdHV0b3JpYWwgd2lsbCBjb3ZlciB0aGUgYmFzaWNzIG9mIFJOQS1zZXEgdXNpbmcgR2FsYXh5IGFuZCBEZWd1c3Q7IHR3byBvcGVuLXNvdXJjZSB3ZWItYmFzZWQgcGxhdGZvcm1zIGZvciB0aGUgYW5hbHlzaXMgb2YgYmlvbG9naWNhbCBkYXRhLiBZb3Ugc2hvdWxkIGdhaW4gYW4gYXBwcmVjaWF0aW9uIG9mIHRoZSB0YXNrcyBpbnZvbHZlZCBpbiBhIHR5cGljYWwgUk5BLXNlcSBhbmFseXNpcyBhbmQgYmUgY29tZm9ydGFibGUgd2l0aCB0aGUgb3V0cHV0cyBnZW5lcmF0ZWQgYnkgdGhlIEJpb2luZm9ybWF0aWNpYW4uCgoKIyMgVXNpbmcgdGhlc2Ugbm90ZXMKCjxkaXYgY2xhc3M9ImV4ZXJjaXNlIj4KU2VjdGlvbnMgd2l0aCB0aGlzIGJhY2tncm91bmQgaW5kaWNhdGUgZXhlcmNpc2VzIHRvIGJlIGNvbXBsZXRlZCBkdXJpbmcgdGhlIHdvcmtzaG9wLgo8L2Rpdj4KCjxkaXYgY2xhc3M9ImluZm9ybWF0aW9uIj4KU2VjdGlvbnMgd2l0aCB0aGlzIGJhY2tncm91bmQgaGlnaGxpZ2h0IHBhcnRpY3VsYXIgc2hvcnRjdXRzIG9yIG90aGVyIHJlZmVyZW5jZXMgdGhhdCBtaWdodCBiZSB1c2VmdWwuCjwvZGl2PgoKPGRpdiBjbGFzcz0id2FybmluZyI+ClNlY3Rpb25zIHdpdGggdGhpcyBiYWNrZ3JvdW5kIGdpdmUgaW5mb3JtYXRpb24gYWJvdXQgcG90ZW50aWFsIGVycm9yIG1lc3NhZ2VzIG9yIG1pZ2h0IGVuY291bnRlciwgb3IgcHJvYmxlbXMgdGhhdCBtaWdodCBhcmlzZSBpbiB5b3VyIG93biBkYXRhIGFuYWx5c2lzLgo8L2Rpdj4KCiMjIyBNb3JlIG9uIEdhbGF4eQoKVGhlIG9mZmljaWFsIEdhbGF4eSBwYWdlIGhhcyBtYW55IFt0dXRvcmlhbHNdKGh0dHBzOi8vZ2FsYXh5cHJvamVjdC5vcmcvbGVhcm4vKSBvbiB1c2luZyB0aGUgc2VydmljZSwgYW5kIGV4YW1wbGVzIG9mIG90aGVyIHR5cGVzIG9mIGFuYWx5c2lzIHRoYXQgY2FuIGJlIHBlcmZvcm1lZCBvbiB0aGUgcGxhdGZvcm0uCgpUaG9zZSBldmVudHVhbGx5IHdhbnRlZCB0byBwZXJmb3JtIHRoZWlyIG93biBSTkEtc2VxIGFuYWx5c2lzIChmb3IgZXhhbXBsZSBpbiBSKSwgc2hvdWxkIGxvb2sgb3V0IGZvciBvdGhlciBjb3Vyc2VzCgojIyMgQ291cnNlcyBvbiBhbmFseXNpbmcgUk5BLXNlcSBkYXRhIGluIFIKCi0gW1NoZWZmaWVsZCBCaW9pbmZvcm1hdGljcyBDb3JlXShodHRwczovL3NiYy5zaGVmLmFjLnVrL3RyYWluaW5nL3JuYS1zZXEtaW4tci0yMDIwLTAyLTEzLykKLSBbTW9uYXNoIEJpb2luZm9ybWF0aWNzIFBsYXRmb3JtXShodHRwOi8vbW9uYXNoYmlvaW5mb3JtYXRpY3NwbGF0Zm9ybS5naXRodWIuaW8vUk5Bc2VxLURFLWFuYWx5c2lzLXdpdGgtUi8pCgojIyMgVGVjaG5pcXVlcyBub3QgY292ZXJlZCBpbiB0aGUgd29ya3Nob3AKCkZyb20gb3VyIGV4cGVyaWVuY2UsIHRoZSBtb3N0IGNvbW1vbiBhcHBsaWNhdGlvbiBvZiBSTkEtc2VxIGlzIHN0aWxsIHRvIHBlcmZvcm0gYSAiZGlmZmVyZW50aWFsIGV4cHJlc3Npb24iIGluIG9yZGVyIHRvIGlkZW50aWZ5IGdlbmVzLCBhbmQgZXZlbnR1YWxseSBwYXRod2F5cywgdGhhdCBhcmUgYWx0ZXJlZCBiZXR3ZWVuIGEgc2V0IG9mIGJpb2xvZ2ljYWwgY29uZGl0aW9ucy4gVGhlcmVmb3JlIHdlIHdpbGwgY29uY2VudHJhdGUgb24gdGhpcyB0YXNrIGluIHRoZSB3b3Jrc2hvcC4gV2Ugd2lsbCBhbHNvIGJlIGNvbnNpZGVyaW5nICpidWxrKiBSTkEtc2VxIG9ubHkuIGkuZS4gd2hlcmUgb3VyIGJpb2xvZ2ljYWwgc2FtcGxlIG1heSBiZSBjb21wcmlzZWQgb2YgYSBwb29sIG9mIGhldGVyb2dlbmVvdXMgY2VsbHMuIFNpbmdsZS1jZWxsIGFwcHJvYWNoZXMgYXJlIGJlY29taW5nIG1vcmUgcG9wdWxhciwgYW5kIGFsdGhvdWdoIHRoZXJlIGFyZSBzb21lIHNpbWlsYXJpdGllcyBpbiBob3cgdGhlc2UgZGF0YSBhcmUgcHJvY2Vzc2VkLCBhIGRpZmZlcmVudCBkb3duc3RyZWFtIGFuYWx5c2lzIGFwcHJvYWNoIGlzIHJlcXVpcmVkLgoKVGhlIEdhbGF4eSBUcmFpbmluZyBOZXR3b3JrIHByb3ZpZGVzIG1hdGVyaWFscyBvbiBzaW5nbGUtY2VsbCBhbmFseXNpcywgYW5kIG90aGVyIGFwcGxpY2F0aW9ucyBub3QgY292ZXJlZCBpbiB0aGlzIHdvcmtzaG9wLgoKLSBbR2FsYXh5IFRyYWluaW5nIE5ldHdvcmsgLSBUcmFuc2NyaXB0b21pY3NdKGh0dHBzOi8vdHJhaW5pbmcuZ2FsYXh5cHJvamVjdC5vcmcvdHJhaW5pbmctbWF0ZXJpYWwvdG9waWNzL3RyYW5zY3JpcHRvbWljcy8pCgoKIyMgUk5BLXNlcSB3b3JrZmxvdwoKVHdvIHdvcmtmbG93cyBhcmUgcG9zc2libGUgd2l0aCBSTkEtc2VxIGRhdGEgLSB3aXRoIHRoZSBkaWZmZXJlbmNlIGJlaW5nIHdoZXRoZXIgb25lIHBlcmZvcm1zIGFuIGFsaWdubWVudCB0byB0aGUgcmVmZXJlbmNlIGdlbm9tZSBvciBub3QuCgpSZWNlbnQgdG9vbHMgZm9yIFJOQS1zZXEgYW5hbHlzaXMgKGUuZy4gYHNhbG1vbmAsIGBrYWxsaXN0b2ApIGRvIG5vdCByZXF1aXJlIHRoZSB0aW1lLWNvbnN1bWluZyBzdGVwIG9mIHdob2xlLWdlbm9tZSBhbGlnbm1lbnQgdG8gYmUgcGVyZm9ybWVkLCBhbmQgY2FuIHRoZXJlZm9yZSBwcm9kdWNlIGdlbmUtbGV2ZWwgY291bnRzIGluIGEgbXVjaCBmYXN0ZXIgdGltZSBmcmFtZS4gVGhleSBub3QgcmVxdWlyZSB0aGUgY3JlYXRpb24gb2YgbGFyZ2UgYmFtIGZpbGVzLCB3aGljaCBpcyB1c2VmdWwgaWYgY29uc3RyYWluZWQgYnkgZmlsZSBzcGFjZSBvbiBHYWxheHkuCgohW10oaHR0cHM6Ly9oYmN0cmFpbmluZy5naXRodWIuaW8vSW50cm8tdG8tcm5hc2VxLWhwYy1ndC9pbWcvYWxpZ25tZW50ZnJlZV93b3JrZmxvd19hdWcyMDE3LnBuZykKCihpbWFnZSBmcm9tIEhhcnZhcmQgQmlvaW5mb3JtYXRpY3MgQ29yZSkKCgotLS0tLQoKIyMgQmFja2dyb3VuZAoKIyMjIyBXaGVyZSBkbyB0aGUgZGF0YSBpbiB0aGlzIHR1dG9yaWFsIGNvbWUgZnJvbT8KVGhlIGRhdGEgZm9yIHRoaXMgdHV0b3JpYWwgY29tZXMgZnJvbSBhIEpvdXJuYWwgb2YgRXhwZXJpbWVudGFsIE1lZGljaW5lIHBhcGVyIFsiSXRyYWNvbmF6b2xlIHRhcmdldHMgY2VsbCBjeWNsZSBoZXRlcm9nZW5laXR5IGluIGNvbG9yZWN0YWwgY2FuY2VyIl0oaHR0cHM6Ly9wdWJtZWQubmNiaS5ubG0ubmloLmdvdi8yOTg1MzYwNy8pLiBUaGlzIHN0dWR5IGV4YW1pbmVzIHRoZSBleHByZXNzaW9uIHByb2ZpbGVzIG9mIHR3byBjZWxsIGxpbmVzIGluIHJlc3BvbnNlIHRvIHRyZWF0bWVudCB3aXRoIGl0cmFjb25hem9sZS4KCgoKRm9yIHRoaXMgdHV0b3JpYWwsIHdlIHdpbGwgYXNzdW1lIHRoYXQgdGhlICp3ZXQtbGFiKiBzdGFnZXMgb2YgdGhlIGV4cGVyaW1lbnQgaGF2ZSBiZWVuIHBlcmZvcm1lZCBhbmQgaW4gdGhpcyB0dXRvcmlhbCB3ZSB3aWxsIGRlbW9uc3RyYXRlIHRoZSBzdGVwcyBvZiAqKlF1YWxpdHkgYXNzZXNzbWVudCoqLCAqKmFsaWdubWVudCoqLCAqKnF1YW50aWZpY2F0aW9uKiogYW5kICoqZGlmZmVyZW50aWFsIGV4cHJlc3Npb24gdGVzdGluZyoqLgoKVGhlIGZhc3RxIGRhdGEgZm9yIHRoaXMgZXhwZXJpbWVudCB3ZXJlIG1hZGUgYXZhaWxhYmxlIG9uIHRoZSBTZXF1ZW5jaW5nIFJlYWQgQXJjaGl2ZSAoU1JBKSB3aXRoIGFjY2Vzc2lvbiBTUlAxNDQ0OTYuICoqRm9yIHRoZSBwdXJwb3NlcyBvZiB0aGlzIHdvcmtzaG9wIHdlIGhhdmUgY3JlYXRlZCBhIGRvd25zYW1wbGVkIGRhdGFzZXQqKgoKVGhlIGV4cGVyaW1lbnRhbCBkZXNpZ24gZm9yIHRoZSBkYXRhc2V0IGlzIHN1bW1hcmlzZWQgaW4gdGhlIHRhYmxlIGJlbG93LgoKcnVuCXwgbmFtZQl8IGNlbGxfbGluZQl8IGNvbmRpdGlvbgotLS0tfC0tLS0tLS18LS0tLS0tLS0tLS18LS0tLS0tLS0tIApTUlI3MTA4Mzg4CXwgSFQ1NV9DT05UXzEJfCBIVDU1CXwgRE1TTwpTUlI3MTA4Mzg5CXwgSFQ1NV9DT05UXzIJfCBIVDU1CXwgRE1TTwpTUlI3MTA4MzkwCXwgSFQ1NV9DT05UXzMJfCBIVDU1CXwgRE1TTwpTUlI3MTA4MzkxCXwgSFQ1NV9DT05UXzQJfCBIVDU1CXwgRE1TTwpTUlI3MTA4MzkyCXwgSFQ1NV9JVFJBXzEJfCBIVDU1CXwgSVRSQUNPTkFaT0xFClNSUjcxMDgzOTMJfCBIVDU1X0lUUkFfMgl8IEhUNTUJfCBJVFJBQ09OQVpPTEUKU1JSNzEwODM5NAl8IEhUNTVfSVRSQV8zCXwgSFQ1NQl8IElUUkFDT05BWk9MRQpTUlI3MTA4Mzk1CXwgSFQ1NV9JVFJBXzQJfCBIVDU1CXwgSVRSQUNPTkFaT0xFClNSUjcxMDgzOTYJfCBTVzk0OF9DT05UXzEgfCBTVzk0OAl8IERNU08KU1JSNzEwODM5Nwl8IFNXOTQ4X0NPTlRfMiB8IFNXOTQ4CXwgRE1TTwpTUlI3MTA4Mzk4CXwgU1c5NDhfQ09OVF8zIHwgU1c5NDgJfCBETVNPClNSUjcxMDgzOTkJfCBTVzk0OF9DT05UXzQgfCBTVzk0OAl8IERNU08KU1JSNzEwODQwMAl8IFNXOTQ4X0lUUkFfMSB8IFNXOTQ4CXwgSVRSQUNPTkFaT0xFClNSUjcxMDg0MDEJfCBTVzk0OF9JVFJBXzIgfCBTVzk0OAl8IElUUkFDT05BWk9MRQpTUlI3MTA4NDAyCXwgU1c5NDhfSVRSQV8zIHwgU1c5NDgJfCBJVFJBQ09OQVpPTEUKU1JSNzEwODQwMwl8IFNXOTQ4X0lUUkFfNCB8IFNXOTQ4CXwgSVRSQUNPTkFaT0xFCgojIyMjIERpZ3Jlc3Npb246IEhvdyB0byBkb3dubG9hZCByYXcgc2VxdWVuY2luZyBmaWxlcwoKVGhlIFNlcXVlbmNpbmcgUmVhZCBBcmNoaXZlIChTUkEpIGlzIGNvbW1vbmx5LXVzZWQgdG8gc3RvcmUgdGhlIHJhdyBkYXRhIGZyb20gc2VxdWVuY2luZyBleHBlcmltZW50cyBhbmQgY2FuIGJlIGFjY2Vzc2VkIHRocm91Z2ggdGhlIE5DQkkgd2Vic2l0ZS4gSG93ZXZlciwgdGhlIGludGVyZmFjZSBpcyBub3QgcGFydGljdWxhcmx5IGZyaWVuZGx5IGFuZCB0aGUgbGlua3MgdG8gZG93bmxvYWQgZGF0YSBhbmQgbm90IGVhc3kgdG8gb2J0YWluLgoKQW4gZWFzaWVyIGFsdGVybmF0aXZlIGV4aXN0cyBpbiB0aGUgZm9ybSBvZiBTUkEgRXhwbG9yZXIKCi0gW2h0dHBzOi8vc3JhLWV4cGxvcmVyLmluZm8vXShodHRwczovL3NyYS1leHBsb3Jlci5pbmZvLykKCjxpbWcgc3JjPSJtZWRpYS9zcmFfZXhwbG9yZXIucG5nIi8+CgpUaGUgU1JBIGFjY2Vzc2lvbiAodXN1YWxseSBmb3VuZCBpbiBhIHBhcGVyIGRlc2NyaWJpbmcgdGhlIGRhdGFzZXQpIGNhbiBiZSBlbnRlcmVkIGludG8gdGhlIFNlYXJjaCBib3gsIGFuZCBhbGwgdGhlIHNhbXBsZXMgYmVsb25naW5nIHRvIHRoYXQgZGF0YXNldCBzaG91bGQgYmUgZm91bmQuIFNhbXBsZXMgb2YgaW50ZXJlc3QgY2FuIGJlIHNhdmVkLCBhbmQgdXBvbiAiY2hlY2tvdXQiIHRoZSBkb3dubG9hZCBsaW5rcyAoVVJMcykgd2lsbCBiZSBkaXNwbGF5ZWQuIEEgY29tbWFuZC1saW5lIHRvb2wgc3VjaCBhcyBgY3VybGAgb3IgYHdnZXRgIGNhbiB0aGVuIGJlIHVzZWQgdG8gZG93bmxvYWQgdGhlIGZpbGVzIGxvY2FsbHkuCgoKIyMgU2VjdGlvbiAxOiBQcmVwYXJhdGlvbgojIyMjIDEuIFNpZ24tdXAgdG8gdGhlIEV1cm9wZWFuIEdhbGF4eSBzZXJ2ZXIKCgotIGh0dHBzOi8vdXNlZ2FsYXh5LmV1Ci0gVGhlIEF1c3RyYWxpYW4gc2VydmVyIGlzIGFuIGFsdGVybmF0aXZlIGlmIHRoZSBFdXJvcGVhbiBvbmUgaXMgZG93bjotIGh0dHBzOi8vdXNlZ2FsYXh5Lm9yZy5hdS8KCgoqKk1ha2Ugc3VyZSB5b3UgY2hlY2sgeW91ciBlbWFpbCB0byBhY3RpdmF0ZSB5b3VyIGFjY291bnQqKgoKCgojIyMjIDIuIERvd25sb2FkIHRoZSBjb3Vyc2UgZGF0YQoKVGhlIGRhdGEgZm9yIHRoaXMgY291cnNlIGhhdmUgYWxsIGJlZW4gc2hhcmVkIG9uIGEgZ29vZ2xlIGRyaXZlLiBJZiB5b3UgaGF2ZSBub3QgZG9uZSBzbyBhbHJlYWR5LCBwbGVhc2UgZ28gdG8gdGhpcyBkaXJlY3RvcnkgYW5kIGRvd25sb2FkIHRoZSBmb2xsb3dpbmcgZmlsZXMKCmh0dHBzOi8vZHJpdmUuZ29vZ2xlLmNvbS9kcml2ZS9mb2xkZXJzLzFSU3V2bDlzaEF3MTJCajc3dVlTVWRXdGtaNVNUNUVXaT91c3A9c2hhcmluZwoKLSBgU1JSNzEwODM4OC5mYXN0cS5nemAKLSBgU1JSNzEwODM4OS5mYXN0cS5nemAKLSBgU1JSNzEwODM5Mi5mYXN0cS5nemAKLSBgU1JSNzEwODM5My5mYXN0cS5nemAKLSBgSG9tb19zYXBpZW5zLkdSQ2gzOC5jZG5hLmFsbC5mYS5nemAKLSBgdHgyZ2VuZS50eHRgCgoKIyMjIyAzLiAgSW1wb3J0IHRoZSBSTkEtc2VxIGRhdGEgZm9yIHRoZSB3b3Jrc2hvcC4KCldlIGNhbiBnb2luZyB0byBpbXBvcnQgdGhlIFsqZmFzdHEqIGZpbGVzXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9GQVNUUV9mb3JtYXQpIGZvciB0aGlzIGV4cGVyaW1lbnQuIFRoaXMgaXMgYSBzdGFuZGFyZCBmb3JtYXQgZm9yIHN0b3JpbmcgcmF3IHNlcXVlbmNpbmcgcmVhZHMgYW5kIHRoZWlyIGFzc29jaWF0ZWQgcXVhbGl0eSBzY29yZXMuIFRvIG1ha2UgdGhlIHByYWN0aWNhbCBxdWlja2VyLCB3ZSBoYXZlICpkb3duc2FtcGxlZCogdGhlIG9yaWdpbmFsIGZhc3RxIGZpbGVzIHRvIGEgcXVhcnRlciBvZiBhIG1pbGxpb24gcmVhZHMuCgo8ZGl2IGNsYXNzPSJpbmZvcm1hdGlvbiI+CgojIyMjICoqR2V0IERhdGEgLT4gVXBsb2FkIEZpbGUgKiogCgo8L2Rpdj4KCllvdSBjYW4gaW1wb3J0IHRoZSBkYXRhIGJ5OgoKMS4gIEluIHRoZSB0b29sIHBhbmVsIGxvY2F0ZWQgb24gdGhlIGxlZnQsIHVuZGVyIEJhc2ljIFRvb2xzIHNlbGVjdCAqKkdldCBEYXRhID4gVXBsb2FkIEZpbGUqKi4gQ2xpY2sgb24gdGhlICoqQ2hvb3NlIGxvY2FsIGZpbGUqKiBidXR0b24gb24gdGhlCiAgICBib3R0b20gc2VjdGlvbiBvZiB0aGUgcG9wLXVwIHdpbmRvdy4KMi4gIE5hdmlnYXRlIHRvIHRoZSBgZmFzdHFgIGRpcmVjdG9yeSBvZiB0aGUgemlwIGZpbGUgdGhhdCB5b3UgZG93bmxvYWRlZCBmcm9tIGdvb2dsZSBkcml2ZSBhbmQgc2VsZWN0IHRoZXNlIHR3byBmaWxlcyBmcm9tIHRoZSBIVDU1LURNU08gY29uZGl0aW9uLiAKCmBTUlI3MTA4Mzg4LmZhc3RxLmd6YApgU1JSNzEwODM4OS5mYXN0cS5nemAKIAogYW5kIHRoZXNlIHR3byBmaWxlcyBhcmUgZnJvbSB0aGUgSFQ1NSBJVFJBQ09OQVpPTEUgY29uZGl0aW9uLgoKYFNSUjcxMDgzOTIuZmFzdHEuZ3pgCmBTUlI3MTA4MzkzLmZhc3RxLmd6YAoKYWxzbyB1cGxvYWQgdGhlIGZpbGVzIGBIb21vX3NhcGllbnMuR1JDaDM4LmNkbmEuYWxsLmZhLmd6YCBhbmQgYHR4MmdlbmUudHh0YC4gVGhlc2UgYXJlIHJlZmVyZW5jZSBmaWxlcyB0aGF0IHdlIHdpbGwgdXNlIGxhdGVyLgoKMy4gIFlvdSBzaG91bGQgbm93IGhhdmUgdGhlc2UgNCBmYXN0cSBmaWxlcyBpbiB5b3VyIGhpc3Rvcnk6CiAgICAtIGBTUlI3MTA4Mzg4LmZhc3RxLmd6YAogICAgLSBgU1JSNzEwODM4OS5mYXN0cS5nemAKICAgIC0gYFNSUjcxMDgzOTIuZmFzdHEuZ3pgCiAgICAtIGBTUlI3MTA4MzkzLmZhc3RxLmd6YAoKVGhlIGFubm90YXRpb24gZmlsZXMgbWF5IHRha2UgYSB3aGlsZSBsb25nZXIgdG8gdXBsb2FkCgo8ZGl2IGNsYXNzPSJpbmZvcm1hdGlvbiI+ClRoZSBgLmd6YCBhdCB0aGUgZW5kIG9mIGVhY2ggZmlsZSBuYW1lIG1lYW5zIHRoYXQgaXQgaXMgKmNvbXByZXNzZWQqIChsaWtlIGEgemlwIGZpbGUpLiAKPC9kaXY+Cgo8ZGl2IGNsYXNzPSJpbmZvcm1hdGlvbiI+CllvdSBjYW4gdXBsb2FkIHRoZSBvdGhlciBmaWxlcyBmb3IgZXh0cmEgcHJhY3RpY2UgaWYgeW91IHdpc2gKPC9kaXY+CgojIyBBbHRlcm5hdGl2ZSB1cGxvYWQKClRvIHNhdmUgdGltZSwgdGhlIHJhdyBkYXRhIGZvciB0aGlzIGRhdGFzZXQgaGF2ZSBiZWVuIHVwbG9hZGVkIHRvIEdhbGF4eS4gVG8gYWNjZXNzIHRoZSBoaXN0b3J5LCB1c2UgdGhlIGxpbmsgYmVsb3cKCi0gW2h0dHBzOi8vdXNlZ2FsYXh5LmV1L3UvbWFya2R1bm5pbmcvaC9iZWdpbm5lcnMtcm5hLXNlceKAlHNhbG1vbl0oaHR0cHM6Ly91c2VnYWxheHkuZXUvdS9tYXJrZHVubmluZy9oL2JlZ2lubmVycy1ybmEtc2Vx4oCUc2FsbW9uKQoKLSBbaHR0cHM6Ly91c2VnYWxheHkub3JnLmF1L3UvbWFya2R1bm5pbmcvaC9iZWdpbm5lcnNybmFzZXFzYWxtb25dKGh0dHBzOi8vdXNlZ2FsYXh5Lm9yZy5hdS91L21hcmtkdW5uaW5nL2gvYmVnaW5uZXJzcm5hc2Vxc2FsbW9uKSAtICoqZ2FsYXh5Lm9yZy5hdSoqCgpUaGVyZSBzaG91bGQgYmUgYSBidXR0b24gdG8gaW1wb3J0IHRoZSBoaXN0b3J5IGluIHRoZSB0b3AtcmlnaHQgY29ybmVyLiBUaGUgcmF3IGRhdGEgZm9yIHRoZSBwcmFjdGljYWwgc2hvdWxkIG5vdyBiZSBhdmFpbGFibGUgdG8geW91LgoKIyMjIFF1YWxpdHkgYXNzZXNzbWVudCB3aXRoIEZhc3RRQyAoT3B0aW9uYWwpCgpbRmFzdFFDXShodHRwczovL3d3dy5iaW9pbmZvcm1hdGljcy5iYWJyYWhhbS5hYy51ay9wcm9qZWN0cy9mYXN0cWMvKSBpcyBhIHBvcHVsYXIgdG9vbCBmcm9tIFtCYWJyYWhhbSBJbnN0aXR1dGUgQmlvaW5mb3JtYXRpY3MgR3JvdXBdKGh0dHBzOi8vd3d3LmJpb2luZm9ybWF0aWNzLmJhYnJhaGFtLmFjLnVrL2luZGV4Lmh0bWwpIHVzZWQgZm9yICpxdWFsaXR5IGFzc2Vzc21lbnQqIG9mIHNlcXVlbmNpbmcgZGF0YS4gTW9zdCBCaW9pbmZvcm1hdGljcyBwaXBlbGluZXMgd2lsbCB1c2UgRmFzdFFDLCBvciBzaW1pbGFyIHRvb2xzIGluIHRoZSBmaXJzdCBzdGFnZSBvZiB0aGUgYW5hbHlzaXMuIFRoZSBbZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly93d3cuYmlvaW5mb3JtYXRpY3MuYmFicmFoYW0uYWMudWsvcHJvamVjdHMvZmFzdHFjL0hlbHAvKSBmb3IgRmFzdFFDIHdpbGwgaGVscCB5b3UgdG8gaW50ZXJwcmV0IHRoZSBwbG90cyBhbmQgc3RhdHMgcHJvZHVjZWQgYnkgdGhlIHRvb2wuIEEgdHJhZmZpYyBsaWdodCBzeXN0ZW0gaXMgdXNlZCB0byBhbGVydCB0aGUgdXNlcidzIGF0dGVudGlvbiB0byBwb3NzaWJsZSBpc3N1ZXMuIEhvd2V2ZXIsIGl0IGlzIHdvcnRoIGJlYXJpbmcgaW4gbWluZCB0aGF0IHRoZSB0b29sIGlzIGJsaW5kIHRvIHRoZSBwYXJ0aWN1bGFyIHR5cGUgb2Ygc2VxdWVuY2luZyB5b3UgYXJlIHBlcmZvcm1pbmcgKGkuZS4gd2hvbGUtZ2Vub21lLCBDaElQLXNlcSwgUk5BLXNlcSksIHNvIHNvbWUgd2FybmluZ3MgbWlnaHQgYmUgZXhwZWN0ZWQgZHVlIHRvIHRoZSBuYXR1cmUgb2YgeW91ciBleHBlcmltZW50LgoKPGRpdiBjbGFzcz0iaW5mb3JtYXRpb24iPgoKIyMjIyAqRmFzdFEgUXVhbGl0eSBDb250cm9sKiAtPiAqRmFzdFFDIFJlYWQgUXVhbGl0eSByZXBvcnRzKgoKPC9kaXY+CgotIFNlbGVjdCBvbmUgb2YgdGhlIEZBU1RRIGZpbGVzIGFzIGlucHV0IGFuZCAqRXhlY3V0ZSogdGhlIHRvb2wuCi0gV2hlbiB0aGUgdG9vbCBmaW5pc2hlcyBydW5uaW5nLCB5b3Ugc2hvdWxkIGhhdmUgYW4gSFRNTCBmaWxlIGluIHlvdXIgSGlzdG9yeS4gQ2xpY2sgb24gdGhlIGV5ZSBpY29uIHRvIHZpZXcgdGhlIHZhcmlvdXMgcXVhbGl0eSBtZXRyaWNzLgotIFJ1biBGYXN0cWMgb24gdGhlIHJlbWFpbmcgZmFzdHEgZmlsZXMsIGJ1dCBkb24ndCBleGFtaW5lIHRoZSByZXN1bHRzIGp1c3QgeWV0LgoKCjxkaXYgY2xhc3M9ImV4ZXJjaXNlIj4KCioqUXVlc3Rpb246IERvIHRoZSBkYXRhIHNlZW0gdG8gYmUgb2YgcmVhc29uYWJsZSBxdWFsaXR5PyAqKgoKWW91IGNhbiB1c2UgdGhlIFtkb2N1bWVudGF0aW9uXShodHRwczovL3d3dy5iaW9pbmZvcm1hdGljcy5iYWJyYWhhbS5hYy51ay9wcm9qZWN0cy9mYXN0cWMvSGVscC8zJTIwQW5hbHlzaXMlMjBNb2R1bGVzLykgdG8gaGVscCBpbnRlcnByZXQgdGhlIHBsb3RzCgo8L2Rpdj4KCklmIHBvb3IgcXVhbGl0eSByZWFkcyB0b3dhcmRzIHRoZSBlbmRzIG9mIHJlYWRzIGFyZSBjb25zaWRlcmVkIHRvIGJlIGEgcHJvYmxlbSwgb3IgdGhlcmUgaXMgY29uc2lkZXJhYmxlIGFkYXB0ZXIgY29udGFtaW5hdGlvbiwgd2UgY2FuIGVtcGxveSB2YXJpb3VzIHRvb2xzIHRvICp0cmltKiBvdXIgZGF0YS4KCkhvd2V2ZXIsIGEgcmVjZW50IHBhcGVyIGRlbW9uc3RyYXRlZCB0aGF0IHJlYWQgdHJpbW1pbmcgaXMgbm8gbG9uZ2VyIHJlcXVpcmVkIHByaW9yIHRvIGFsaWdubWVudDotIGh0dHBzOi8vd3d3LmJpb3J4aXYub3JnL2NvbnRlbnQvMTAuMTEwMS84MzM5NjJ2MQoKCjxkaXYgY2xhc3M9Indhcm5pbmciPgoKSWYgeW91IGFsc28gc3VzcGVjdCBjb250YW1pbmF0aW9uIGJ5IGFub3RoZXIgb3JnYW5pc20sIG9yIHJSTkEgcHJlc2VudCBpbiB5b3VyIGRhdGEsIHlvdSBjYW4gdXNlIHRoZSBzb3J0TWVSTkEgdG9vbCB0byByZW1vdmUgdGhpcyBhcnRlZmFjdC4KPC9kaXY+CgoKIyMjIENvbWJpbmluZyBRQyByZXBvcnRzCgpJdCBjYW4gYmUgcXVpdGUgdGlyZXNvbWUgdG8gY2xpY2sgdGhyb3VnaCBtdWx0aXBsZSBRQyByZXBvcnRzIGFuZCBjb21wYXJlIHRoZSByZXN1bHRzIGZvciBkaWZmZXJlbnQgc2FtcGxlcy4gSXQgaXMgdXNlZnVsIHRvIGhhdmUgYWxsIHRoZSBRQyBwbG90cyBvbiB0aGUgc2FtZSBwYWdlIHNvIHRoYXQgd2UgY2FuIG1vcmUgZWFzaWx5IHNwb3QgdHJlbmRzIGluIHRoZSBkYXRhLgoKVGhlIFttdWx0aXFjXShodHRwczovL211bHRpcWMuaW5mby8pIHRvb2wgaGFzIGJlZW4gZGVzaWduZWQgZm9yIHRoZSB0YXNrcyBvZiBhZ2dyZWdhdGluZyBxYyByZXBvcnRzIGFuZCBjb21iaW5pbmcgaW50byBhIHNpbmdsZSByZXBvcnQgdGhhdCBpcyBlYXN5IHRvIGRpZ2VzdC4KCgo8ZGl2IGNsYXNzPSJpbmZvcm1hdGlvbiI+CgoqRkFTVFEgUXVhbGl0eSBDb250cm9sKiAtPiAqTXVsdGlxYyoKCjwvZGl2PgoKVW5kZXIgKldoaWNoIHRvb2wgd2FzIHVzZWQgZ2VuZXJhdGUgbG9ncz8qIENob29zZSAqZmFzdHFjKiBhbmQgc2VsZWN0IHRoZSBSYXdEYXRhIG91dHB1dCBmcm9tIHRoZSBmYXN0cWMgcnVuIG9uIGVhY2ggb2YgeW91ciBiYW0gZmlsZXMuCgoKPGRpdiBjbGFzcz0iZXhlcmNpc2UiPgoKUXVlc3Rpb246IFJlcGVhdCB0aGUgRmFzdFFDIGFuYWx5c2lzIGZvciB0aGUgcmVtYWluaW5nIGZhc3RxIGZpbGVzIGFuZCBjb21iaW5lIHRoZSByZXBvcnRzIHdpdGggYG11bHRpUUNgLiBEbyB0aGUgZmFzdHEgZmlsZXMgc2VlbSB0byBoYXZlIGNvbnNpc3RlbnRseSBoaWdoLXF1YWxpdHk/CjwvZGl2PgoKCiMjIFNlY3Rpb24gMjogUXVhbnRpZmljYXRpb24KClRyYWRpdGlvbmFsbHksIHdvcmtmbG93cyBmb3IgUk5BLXNlcSB3b3VsZCBhbGlnbiByZWFkcyB0byBhIHJlZmVyZW5jZSAqZ2Vub21lKiwgYW5kIHRoZW4gb3ZlcmxhcCB3aXRoIGtub3cgZ2VuZSBjb29yZGluYXRlcy4gSG93ZXZlciwgbWFueSBub3cgcHJlZmVyIHRvIGFsaWduIGRpcmVjdGx5IHRvIHRoZSAqdHJhbnNjcmlwdG9tZSogc2VxdWVuY2VzIHVzaW5nIGEgbWV0aG9kIHN1Y2ggYXMgW2BzYWxtb25gXShodHRwczovL3NhbG1vbi5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvKSBvciBbYGthbGxpc3RvYF0oaHR0cHM6Ly9wYWNodGVybGFiLmdpdGh1Yi5pby9rYWxsaXN0by9hYm91dCkuIFdlIHdpbGwgZGVtb25zdHJhdGUgdGhlIHNhbG1vbiBwcm90b2NvbC4KCiMjIyBPYnRhaW5pbmcgdGhlIGZpbGVzIGZvciBzYWxtb24KCjEpIGNETkEgZmFzdGEgZmlsZQoKQXMgd2UgYXJlIGdvaW5nIHRvIGFsaWduIHRvIGEgc2V0IG9mICp0cmFuc2NyaXB0cyogcmF0aGVyIHRoYW4gdGhlIHdob2xlIGdlbm9tZSwgd2UgcmVxdWlyZSBhIGZpbGUgdGhhdCBjb250YWlucyB0aGUgc2VxdWVuY2Ugb2YgZWFjaCB0cmFuc2NyaXB0LiBUaGlzIGZpbGUgaGFzIGJlZW4gcHJvdmlkZWQgaW4gdGhlIGdvb2dsZSBkcml2ZSBmb2xkZXIgYW5kIHNob3VsZCBoYXZlIGJlZW4gdXBsb2FkZWQgdG8geW91ciBHYWxheHkgaGlzdG9yeS4KCi0gW2h0dHBzOi8vZHJpdmUuZ29vZ2xlLmNvbS9kcml2ZS9mb2xkZXJzLzFSU3V2bDlzaEF3MTJCajc3dVlTVWRXdGtaNVNUNUVXaT91c3A9c2hhcmluZ10oaHR0cHM6Ly9kcml2ZS5nb29nbGUuY29tL2RyaXZlL2ZvbGRlcnMvMVJTdXZsOXNoQXcxMkJqNzd1WVNVZFd0a1o1U1Q1RVdpP3VzcD1zaGFyaW5nKQoKSG93ZXZlciwgaXQgaXMgdXNlZnVsIHRvIGtub3cgd2hlcmUgdGhpcyBmaWxlIGNhbWUgZnJvbSBpbiBjYXNlIHlvdSBhcmUgbm90IHdvcmtpbmcgd2l0aCBIdW1hbiBkYXRhLiBUaGUgZmlsZSB3YXMgb2J0YWluZWQgZnJvbSBbRW5zZW1ibF0oaHR0cDovL20uZW5zZW1ibC5vcmcvaW5mby9kYXRhL2Z0cC9pbmRleC5odG1sKSBieSBjbGlja2luZyBvbiB0aGUgKipjRE5BIChGQVNUQSkqKiBsaW5rIGZvciB0aGUgYXBwcm9wcmlhdGUgb3JnYW5pc20gKEh1bWFuKS4gCgo8aW1nIHNyYz0ibWVkaWEvZW5zZW1ibF9kb3dubG9hZC5wbmciLz4KCk9uIHRoZSBuZXh0IHNjcmVlbiwgIFJpZ2h0LWNsaWNrIHRvIHNhdmUgdGhlIGAuY2RuYS5hbGwuZmEuZ3pgIHRvIHlvdXIgY29tcHV0ZXIKCjxpbWcgc3JjPSJtZWRpYS9jZG5hX2Rvd25sb2FkLnBuZyIvPgoKVGhlIEZBU1RBIGZpbGUgaXMgYSBsYXJnZSB0ZXh0IGZpbGUgdGhhdCBsaXN0cyBhbGwgdGhlIHRyYW5zY3JpcHRzIGZvciB0aGUgZ2l2ZW4gb3JnYW5pc20gYW5kIHRoZWlyIGdlbm9taWMgc2VxdWVuY2UuIFlvdSAqY291bGQqIG9wZW4gdGhpcyBpbiBhIHN0YW5kYXJkIHRleHQgZWRpdG9yIGlmIHlvdSB3aXNoZWQgdG8gc2VlIHRoZSBjb250ZW50cy4gVGhlIGNvbnRlbnRzIGFyZSBzaW1pbGFyIHRvIHRoYXQgb2YgYSBGQVNUUSBmaWxlLiBJbiBmYWN0LCB0aGUgRkFTVFEgZmlsZSBpcyBhIEZBU1RBIGZpbGUgd2l0aCBleHRyYSBxdWFsaXR5IHNjb3JlcyBhZGRlZC4KClRoZSBpZGVudGlmaWVyIGxpbmUgZm9yIGVhY2ggc2VxdWVuY2UgKHN0YXJ0aW5nIHdpdGggYD5gKSBuYW1lcyB0aGUgdHJhbnNjcmlwdCBhbmQgdGhlIGdlbmUgaXQgaXMgYXNzb2NpYXRlZCB3aXRoLiBTaW5jZSB3ZSBvYnRhaW5lZCB0aGUgZmlsZSBmcm9tIEVuc2VtYmwsIHRoZSBUcmFuc2NyaXB0cyBhbmQgR2VuZXMgYmVnaW4gd2l0aCB0aGUgYEVOU1RgIGFuZCBgRU5TR2AgcmVzcGVjdGl2ZWx5LiBUaGUgbnVtYmVycyBhZnRlciBlYWNoIHRyYW5zY3JpcHQgb3IgZ2VuZSBhcmUgdGhlICp2ZXJzaW9uIG51bWJlcnMqOyB0aGUgc2VxdWVuY2UgYW5kIGRlZmluaXRpb24gb2YgZWFjaCB0cmFuc2NyaXB0IC8gZ2VuZSBjYW4gZXZvbHZlIG92ZXIgdGltZS4KCmBgYAo+RU5TVDAwMDAwNjMxNDM1LjEgY2RuYSBjaHJvbW9zb21lOkdSQ2gzODpDSFJfSFNDSFI3XzJfQ1RHNjoxNDI4NDczMDY6MTQyODQ3MzE3OjEgZ2VuZTpFTlNHMDAwMDAyODIyNTMuMSBnZW5lX2Jpb3R5cGU6VFJfRF9nZW5lIHRyYW5zY3JpcHRfYmlvdHlwZTpUUl9EX2dlbmUgZ2VuZV9zeW1ib2w6VFJCRDEgZGVzY3JpcHRpb246VCBjZWxsIHJlY2VwdG9yIGJldGEgZGl2ZXJzaXR5IDEgW1NvdXJjZTpIR05DIFN5bWJvbDtBY2M6SEdOQzoxMjE1OF0KR0dHQUNBR0dHR0dDCj5FTlNUMDAwMDA0MTUxMTguMSBjZG5hIGNocm9tb3NvbWU6R1JDaDM4OjE0OjIyNDM4NTQ3OjIyNDM4NTU0OjEgZ2VuZTpFTlNHMDAwMDAyMjM5OTcuMSBnZW5lX2Jpb3R5cGU6VFJfRF9nZW5lIHRyYW5zY3JpcHRfYmlvdHlwZTpUUl9EX2dlbmUgZ2VuZV9zeW1ib2w6VFJERDEgZGVzY3JpcHRpb246VCBjZWxsIHJlY2VwdG9yIGRlbHRhIGRpdmVyc2l0eSAxIFtTb3VyY2U6SEdOQyBTeW1ib2w7QWNjOkhHTkM6MTIyNTRdCkdBQUFUQUdUCj5FTlNUMDAwMDA0NDg5MTQuMSBjZG5hIGNocm9tb3NvbWU6R1JDaDM4OjE0OjIyNDQ5MTEzOjIyNDQ5MTI1OjEgZ2VuZTpFTlNHMDAwMDAyMjg5ODUuMSBnZW5lX2Jpb3R5cGU6VFJfRF9nZW5lIHRyYW5zY3JpcHRfYmlvdHlwZTpUUl9EX2dlbmUgZ2VuZV9zeW1ib2w6VFJERDMgZGVzY3JpcHRpb246VCBjZWxsIHJlY2VwdG9yIGRlbHRhIGRpdmVyc2l0eSAzIFtTb3VyY2U6SEdOQyBTeW1ib2w7QWNjOkhHTkM6MTIyNTZdCkFDVEdHR0dHQVRBQ0cKPkVOU1QwMDAwMDQzNDk3MC4yIGNkbmEgY2hyb21vc29tZTpHUkNoMzg6MTQ6MjI0MzkwMDc6MjI0MzkwMTU6MSBnZW5lOkVOU0cwMDAwMDIzNzIzNS4yIGdlbmVfYmlvdHlwZTpUUl9EX2dlbmUgdHJhbnNjcmlwdF9iaW90eXBlOlRSX0RfZ2VuZSBnZW5lX3N5bWJvbDpUUkREMiBkZXNjcmlwdGlvbjpUIGNlbGwgcmVjZXB0b3IgZGVsdGEgZGl2ZXJzaXR5IDIgW1NvdXJjZTpIR05DIFN5bWJvbDtBY2M6SEdOQzoxMjI1NV0KQ0NUVENDVEFDCj5FTlNUMDAwMDA2MzI2ODQuMSBjZG5hIGNocm9tb3NvbWU6R1JDaDM4Ojc6MTQyNzg2MjEzOjE0Mjc4NjIyNDoxIGdlbmU6RU5TRzAwMDAwMjgyNDMxLjEgZ2VuZV9iaW90eXBlOlRSX0RfZ2VuZSB0cmFuc2NyaXB0X2Jpb3R5cGU6VFJfRF9nZW5lIGdlbmVfc3ltYm9sOlRSQkQxIGRlc2NyaXB0aW9uOlQgY2VsbCByZWNlcHRvciBiZXRhIGRpdmVyc2l0eSAxIFtTb3VyY2U6SEdOQyBTeW1ib2w7QWNjOkhHTkM6MTIxNThdCkdHR0FDQUdHR0dHQwpgYGAKCjIpIFRyYW5zY3JpcHQgbWFwcGluZyBmaWxlCgpCeSBkZWZhdWx0LCBzYWxtb24gd2lsbCBwcm9kdWNlIGNvdW50cyBmb3IgZWFjaCAqdHJhbnNjcmlwdCouIFRoaXMgbWlnaHQgYmUgd2hhdCB3ZSB3YW50LCBidXQgZm9yIG1vc3Qgc3RhbmRhcmQgYW5hbHlzZXMgaXQgaXMgcHJlZmVyYWJsZSB0byB3b3JrIGF0IHRoZSBnZW5lLWxldmVsLiBXZSB0aGVyZWZvcmUgaGF2ZSB0byB0ZWxsIHNhbG1vbiBob3cgdGhlIHRyYW5zY3JpcHRzIGluIHRoZSBjRE5BIGZpbGUgcmVsYXRlIHRvIGtub3duIGdlbmVzLiBTdWNoIGEgZmlsZSBjYW4gYmUgb2J0YWluZWQgZnJvbSBbYmlvbWFydF0oaHR0cHM6Ly9lbnNlbWJsLm9yZy9iaW9tYXJ0L21hcnR2aWV3LykuIAoKLSBTZWxlY3QgdGhlIEVuc2VtYmwgZ2VuZXMgKDEwNCkgZGF0YWJhc2UKLSBTZWxlY3QgdGhlIGRhdGFzZXQgSHVtYW4gZ2VuZXMgKEdSQ2gzOC4xMykKLSBJbiBBdHRyaWJ1dGVzLCBjbGljayB0aGUgIisiIGJ1dHRvbiBuZXh0IHRvIEdFTkUgYW5kIHNlbGVjdCAqVHJhbnNjcmlwdCBzdGFibGUgSUQgdmVyc2lvbiogYW5kICpHZW5lIHN0YWJsZSBJRCouIE1ha2Ugc3VyZSB0aGUgb3JkZXIgb24gdGhlIGxlZnQtaGFuZCBwYW5lbCBpcyAqVHJhbnNjcmlwdCBzdGFibGUgSUQgdmVyc2lvbiosIGZvbGxvd2VkIGJ5ICpHZW5lIHN0YWJsZSBJRCouIFRoaXMgd2lsbCBhZmZlY3QgdGhlIGNvbHVtbiBvcmRlciBpbiB0aGUgZmlsZS4gWW91IGNhbiB0aWNrIC8gdW50aWNrIHRoZSBJRHMgdG8gbWFrZSBzdXJlIHRoZSBvcmRlciBpcyBjb3JyZWN0LgoKPGltZyBzcmM9Im1lZGlhL2Jpb21hcnRfdHJhbnNjcmlwdF9leHBvcnQuUE5HIi8+CgotIENsaWNrIHRoZSBSZXN1bHRzIGJ1dHRvbiBpbiB0aGUgdG9wIGxlZnQgY29ybmVyIHRvIHNlZSBhIHByZXZpZXcgb2YgdGhlIHJlc3VsdHMuIENsaWNraW5nIHRoZSBHbyBidXR0b24gd2lsbCBleHBvcnQgdGhlIHJlc3VsdHMgdG8gYSBmaWxlCgo8aW1nIHNyYz0ibWVkaWEvYmlvbWFydF90cmFuc2NyaXB0X3ByZXZpZXcuUE5HIi8+CgoKKipJdCBpcyBpbXBvcnRhbnQgdGhhdCB0aGUgZmlsZSBkb3dubG9hZGVkIGZyb20gQmlvbWFydCBpcyBlZGl0ZWQgc28gdGhhdCB0aGUgY29sdW1uIGhlYWRpbmdzIGRvIG5vdCBjb250YWluIGFueSBzcGFjZXMqKi4gWW91IGNhbiBkbyB0aGlzIGluIGEgdGV4dCBlZGl0b3IuIFRoZSBlZGl0ZWQgZmlsZSBzaG91bGQgbG9vayBsaWtlIHRoaXMuCgo8aW1nIHNyYz0ibWVkaWEvYmlvbWFydF90cmFuc2NyaXB0X2VkaXQuUE5HIi8+Cgo8ZGl2IGNsYXNzPSJ3YXJuaW5nIj4KCkl0IGlzIGltcG9ydGFudCB0byBtYWtlIHN1cmUgdGhlIHZlcnNpb24gbnVtYmVyIG9mIHlvdXIgdHJhbnNjcmlwdCBmaWxlIGFuZCB0aGUgYmlvbWFSdCBkYXRhc2V0IGFyZSAqKnRoZSBzYW1lKiosIG90aGVyd2lzZSBzb21lIG9mIHRoZSBzdGVwcyBkb3duc3RyZWFtIG1pZ2h0IG5vdCB3b3JrIGFzIGV4cGVjdGVkLgo8L2Rpdj4KCklmIHlvdSBoYXZlIHByb2JsZW1zLCB0aGlzIG1hcHBpbmcgZmlsZSBpcyBhbHNvIHByb3ZpZGVkIGluIHRoZSBnb29nbGUgZHJpdmUgYXMgYHR4MmdlbmUudHh0YC4gVGhlIGNvbnRlbnRzIG9mIHRoZSBmaXJzdCBjb2x1bW4gaGF2ZSB0byBiZSBpbiB0aGUgc2FtZSBmb3JtYXQgYXMgdGhlIHRyYW5zY3JpcHQgbmFtZXMgaW4gdGhlIGZhc3RhIGZpbGUuIGkuZS4gaW4gdGhpcyBjYXNlIHRoZSB2ZXJzaW9uIG51bWJlciBtdXN0IGJlIHByZXNlbnQuIAoKMykgQW5ub3RhdGlvbiBmaWxlIAoKVGhlIEVuc2VtYmwgZ2VuZSBJRHMgYXJlIG5vdCBwYXJ0aWN1bGFybHkgbWVtb3JhYmxlLCBzbyBpdCB3b3VsZCBiZSBoaWdobHkgYmVuZWZpY2lhbCB0byBoYXZlIG90aGVyIGFubm90YXRpb25zIGF0IGhhbmQgdG8gaGVscCB1cyBpbnRlcnByZXQgdGhlIGRhdGEgZG93bnN0cmVhbS4gV2UgY2FuIHVzZSB0aGUgYmlvbWFydCB3ZWJzaXRlIGFnYWluIHRvIHByb2R1Y2UgYSB0YWJsZSB0byBhaWQgZG93bnN0cmVhbSBpbnRlcnByZXRhdGlvbi4gCgpUaGlzIHRpbWUsIHNlbGVjdCBvbmx5IHRoZSAqR2VuZSBTdGFibGUgSUQqIHRpY2tib3ggaW4gdGhlIEdFTkUgYm94LiBFeHBhbmQgdGhlIEVYVEVSTkFMIHBhbmVsIGJ5IGNsaWNraW5nIHRoZSAiKyIgbmV4dCB0byBFWFRFUk5BTCwgYW5kIHNlbGVjdCAqSEdOQyBzeW1ib2wqIGFuZCAqTkNCSSBnZW5lIChmb3JtZXJseSBFbnRyZXpnZW5lKSBJRCoKCjxpbWcgc3JjPSJtZWRpYS9iaW9tYXJ0X2Fubm90YXRpb24uUE5HIi8+CgojIyMgc2FsbW9uIGNvbmZpZ3VyYXRpb24gYW5kIHJ1bm5pbmcKCjxkaXYgY2xhc3M9ImluZm9ybWF0aW9uIj4KKipSTkEgQW5hbHlzaXMqKiAtPiAqKlNhbG1vbiBxdWFudCoqCjwvZGl2PgoKCi0gU2VsZWN0IHRoZSAqSG9tb19zYXBpZW5zX0dSQ2gzOC5jZG5hLmFsbC5mYS5neiogZmlsZSBhcyB0aGUgVHJhbnNjcmlwdHMgZmFzdGEgZmlsZQotIFNlbGVjdCBhbGwgeW91ciB1cGxvYWRlZCBmYXN0cSBmaWxlcyBhcyB5b3VyIERhdGEgSW5wdXQgRkFTVFEvRkFTVEEgZmlsZQotIFNjcm9sbCBkb3duIHRvICpGaWxlIGNvbnRhaW5pbmcgYSBtYXBwaW5nIG9mIHRyYW5zY3JpcHRzIHRvIGdlbmVzKiBhbmQgc2VsZWN0IHRoZSBgdHgyZ2VuZS50eHRgIGZpbGUgZnJvbSB5b3VyIGhpc3RvcnkKClR3byBqb2JzIHdpbGwgbm93IGJlIHF1ZXVlZCBmb3IgZWFjaCBzYW1wbGUgZmFzdHEgZmlsZS4gVGhlIFF1YW50aWZpY2F0aW9uIG91dHB1dCB3aWxsIGNvbnRhaW4gdHJhbnNjcmlwdC1sZXZlbCBkYXRhLCBhbmQgdGhlIEdlbmUgUXVhbnRpZmljYXRpb24gb3V0cHV0IHdpbGwgYmUgYXQgdGhlICpnZW5lLWxldmVsKi4gV2Ugc2hvdWxkIGV4cGVjdCB0aGUgbnVtYmVyIG9mIGxpbmVzIGluIHRoZSBHZW5lIFF1YW50aWZpY2F0aW9uIGZpbGUgdG8gYmUgc3Vic3RhbnRpYWxseSBsZXNzLiBJZiBub3QsIHlvdSB3aWxsIG5lZWQgdG8gY2hlY2sgdGhhdCB5b3VyIHRyYW5zY3JpcHQgbWFwcGluZyBmaWxlIHdhcyBjb3JyZWN0LgoKVGhlIEdlbmUgUXVhbnRpZmljYXRpb24gb3V0cHV0IGZyb20gZWFjaCBzYW1wbGUgY29tcHJpc2VzIHRoZSBmb2xsb3dpbmcgY29sdW1ucyAodGFrZW4gZnJvbSB0aGUgW3NhbG1vbiBkb2N1bWVudGF0aW9uXShodHRwczovL3NhbG1vbi5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvZmlsZV9mb3JtYXRzLmh0bWwpKQoKLSBOYW1lIOKAlCBUaGlzIGlzIHRoZSBuYW1lIG9mIHRoZSB0YXJnZXQgdHJhbnNjcmlwdCBwcm92aWRlZCBpbiB0aGUgaW5wdXQgdHJhbnNjcmlwdCBkYXRhYmFzZSAoRkFTVEEgZmlsZSkuCi0gTGVuZ3RoIOKAlCBUaGlzIGlzIHRoZSBsZW5ndGggb2YgdGhlIHRhcmdldCB0cmFuc2NyaXB0IGluIG51Y2xlb3RpZGVzLgotIEVmZmVjdGl2ZUxlbmd0aCDigJQgVGhpcyBpcyB0aGUgY29tcHV0ZWQgZWZmZWN0aXZlIGxlbmd0aCBvZiB0aGUgdGFyZ2V0IHRyYW5zY3JpcHQuIEl0IHRha2VzIGludG8gYWNjb3VudCBhbGwgZmFjdG9ycyBiZWluZyBtb2RlbGVkIHRoYXQgd2lsbCBlZmZlY3QgdGhlIHByb2JhYmlsaXR5IG9mIHNhbXBsaW5nIGZyYWdtZW50cyBmcm9tIHRoaXMgdHJhbnNjcmlwdCwgaW5jbHVkaW5nIHRoZSBmcmFnbWVudCBsZW5ndGggZGlzdHJpYnV0aW9uIGFuZCBzZXF1ZW5jZS1zcGVjaWZpYyBhbmQgZ2MtZnJhZ21lbnQgYmlhcyAoaWYgdGhleSBhcmUgYmVpbmcgbW9kZWxlZCkuCi0gVFBNIOKAlCBUaGlzIGlzIHNhbG1vbuKAmXMgZXN0aW1hdGUgb2YgdGhlIHJlbGF0aXZlIGFidW5kYW5jZSBvZiB0aGlzIHRyYW5zY3JpcHQgaW4gdW5pdHMgb2YgVHJhbnNjcmlwdHMgUGVyIE1pbGxpb24gKFRQTSkuIFRQTSBpcyB0aGUgcmVjb21tZW5kZWQgcmVsYXRpdmUgYWJ1bmRhbmNlIG1lYXN1cmUgdG8gdXNlIGZvciBkb3duc3RyZWFtIGFuYWx5c2lzLgotIE51bVJlYWRzIOKAlCBUaGlzIGlzIHNhbG1vbuKAmXMgZXN0aW1hdGUgb2YgdGhlIG51bWJlciBvZiByZWFkcyBtYXBwaW5nIHRvIGVhY2ggdHJhbnNjcmlwdCB0aGF0IHdhcyBxdWFudGlmaWVkLiBJdCBpcyBhbiDigJxlc3RpbWF0ZeKAnSBpbnNvZmFyIGFzIGl0IGlzIHRoZSBleHBlY3RlZCBudW1iZXIgb2YgcmVhZHMgdGhhdCBoYXZlIG9yaWdpbmF0ZWQgZnJvbSBlYWNoIHRyYW5zY3JpcHQgZ2l2ZW4gdGhlIHN0cnVjdHVyZSBvZiB0aGUgdW5pcXVlbHkgbWFwcGluZyBhbmQgbXVsdGktbWFwcGluZyByZWFkcyBhbmQgdGhlIHJlbGF0aXZlIGFidW5kYW5jZSBlc3RpbWF0ZXMgZm9yIGVhY2ggdHJhbnNjcmlwdC4KCk5vdGUgdGhhdCB3ZSBhcmUgdXNpbmcgYSBkb3duc2FtcGxlZCBkYXRhc2V0LCBzbyB0aGUgbWFqb3JpdHkgb2YgTnVtUmVhZHMgd2lsbCBiZSB6ZXJvLgoKIyMjIENyZWF0ZSBhIGNvdW50IG1hdHJpeAoKTWV0aG9kcyBmb3IgZGV0ZWN0aW5nIGRpZmZlcmVudGlhbCBleHByZXNzaW9uIGFyZSBsaWtlbHkgdG8gd2FudCBkYXRhIGluIHRoZSBmb3JtIG9mIGEgdGFibGU7IHdoZXJlIGV2ZXJ5IHJvdyBpcyBhIGRpZmZlcmVudCBnZW5lIGFuZCBlYWNoIGNvbHVtbiBpcyBhIHVuaXF1ZSBiaW9sb2dpY2FsIHNhbXBsZS4gQmVmb3JlIHdlIGNhbiBwcm9jZWVkIHdlIHdpbGwgdGhlcmVmb3JlIG5lZWQgdG8gKm1lcmdlKiBvdXIgc2FsbW9uIHJlc3VsdHMgaW50byBhIHNpbmdsZSBvdXRwdXQuIH5+VGhpcyBjYW4gYmUgZG93biB1c2luZyB0aGUgKlNhbG1vbiBxdWFudG1lcmdlKiB0b29sfn4KCjxkaXYgY2xhc3M9ImluZm9ybWF0aW9uIj4Kfn4qKlJOQSBBbmFseXNpcyoqIC0+ICoqU2FsbW9uIHF1YW50bWVyZ2UqKn5+CjwvZGl2PgoKfn5Vc2UgdGhlICtJbnNlcnQgUXVhbnQgZmlsZSBhbmQgbmFtZXMgYnV0dG9uIHJlcGVhdGVkbHkgdG8gc2VsZWN0IGVhY2ggb2YgeW91ciAqKkdlbmUgUXVhbnRpZmljYXRpb24qKiBvdXRwdXRzLiBUaGUgT25lLXdvcmQgc2FtcGxlIG5hbWVzIHRleHQgYm94IGNhbiBiZSB1c2VkIHRvIGNyZWF0ZSBhIHNob3J0ZXIgY29sdW1uIG5hbWUgZm9yIGVhY2ggb3V0cHV0Ln5+Cgp+fk9uY2UgYWxsIHRoZSBHZW5lIFF1YW50aWZpY2F0aW9uIGZpbGVzIGhhdmUgYmVlbiBzZWxlY3RlZCB0aGUgZHJvcC1kb3duIG1lbnUgdW5kZXIgKipDb2x1bW5zKiogc2hvdWxkIGJlIGNoYW5nZWQgZnJvbSBMZW5ndGggdG8gKipOdW1SZWFkcyoqLn5+CgpUaGUgZmlyc3Qgc3RlcCBpbiBtZXJnaW5nIG91ciBzYWxtb24gb3V0cHV0IGlzIHRvIHByb2R1Y2UgYSB0YWJsZSBmb3IgZWFjaCBzYW1wbGUgdGhhdCBjb250YWlucyBqdXN0IHRoZSBnZW5lIG5hbWUgYW5kIHRoZSBudW1iZXIgb2YgcmVhZHMgZm9yIHRoYXQgZ2VuZS4gVGhpcyBjYW4gYmUgZG9uZSB3aXRoIHRoZSBhZHZhbmNlZCBjdXQgdG9vbAoKPGRpdiBjbGFzcz0iaW5mb3JtYXRpb24iPgoqKlRleHQgTWFuaXB1bGF0aW9uKiogLT4gKipBZHZhbmNlZCBDdXQqKiBjb2x1bW5zIGZyb20gYSB0YWJsZQo8L2Rpdj4KCi0gSW4gKipGaWxlIHRvIGN1dCoqLCBzZWxlY3QgYWxsIHRoZSBzYWxtb24gKipHZW5lIFF1YW50aWZpY2F0aW9uKiogb3V0cHV0cwotIEluIHRoZSAqKkxpc3Qgb2YgRmllbGRzKiogYm94IHVuZGVyICoqQ3V0IGJ5KiosIHNlbGVjdCBDb2x1bW46IDEgYW5kIENvbHVtbjogNS4KClRoZXNlIG91dHB1dHMgY2FuIGJlIG1lcmdlZCB1c2luZyB0aGUgQ29sdW1uIGpvaW4gdG9vbAoKPGRpdiBjbGFzcz0iaW5mb3JtYXRpb24iPgoqKkNvbGxlY3Rpb24gT3BlcmF0aW9ucyoqIC0+ICoqQ29sdW1uIGpvaW4qKiBvbiBtdWx0aXBsZSBkYXRhc2V0cwo8L2Rpdj4KCi0gU2VsZWN0IHRoZSAqKkFkdmFuY2VkIEN1dCoqIG91dHB1dHMgYW5kIGtlZXAgdGhlICoqSWRlbnRpZmllciBjb2x1bW4qKiBhcyAxLiBUaGlzIHdpbGwgImpvaW4iIHRoZSBmaWxlcyB0b2dldGhlciBiYXNlZCBvbiB0aGUgZ2VuZSBuYW1lczsgd2hpY2ggaG9wZWZ1bGx5IHNob3VsZCBiZSB0aGUgc2FtZSBmb3IgZWFjaCBzYW1wbGUuCgpBZnRlciB0aGUgdG9vbCBoYXMgZmluaXNoZWQgeW91IHNob3VsZCBoYXZlIGFuIG91dHB1dCB0YWJsZSB3aXRoIG9uZSByb3cgZm9yIGVhY2ggZ2VuZSBhbmQgYSBjb2x1bW4gY29udGFpbmluZyB0aGUgY291bnRzIGZvciB0aGF0IGdlbmUuCgo8ZGl2IGNsYXNzPSJleGVyY2lzZSI+CioqRXhlcmNpc2UqKjogVXBsb2FkIHRoZSBhbm5vdGF0aW9uIGZpbGUgZnJvbSBiaW9tYXJ0IGNvbnRhaW5pbmcgRW5zZW1ibCBHZW5lIElELCBoZ25jIGFuZCBFbnRyZXouIFVzZSB0aGUgY29sdW1uIGpvaW4gdG9vbCB0byBjcmVhdGUgYSB0YWJsZSB0aGF0IGFsbG93cyB5b3UgdG8gaWRlbnRpZnkgdGhlIGNvdW50cyBmb3IgZ2l2ZW4gZ2VuZXMgbW9yZSBlYXNpbHkuCgo8L2V4ZXJjaXNlPgoKIyAqKihPcHRpb25hbCkqKiBBbHRlcm5hdGl2ZSB3b3JrZmxvdyBpbnZvbHZpbmcgZ2Vub21lIGFsaWdubWVudAoKSWYgdGltZSBhbGxvd3MsIHdlIHdpbGwgYWxzbyBmb2xsb3cgdGhpcyBzZWN0aW9uCgohW10oaHR0cHM6Ly9kYXRhYmVhdXR5LmNvbS9maWd1cmVzLzIwMTYtMDktMTMtUk5BLXNlcS1hbmFseXNpcy9ybmFfc2VxX3dvcmtmbG93LnBuZykKClRoZSB3b3JrZmxvdyB0aGF0IHBlb3BsZSB1c2VkIGZvciBtYW55IHllYXJzIGlzIHN1bW1hcmlzZWQgaW4gdGhpcyBpbWFnZSBmcm9tIFRpbmcteW91IFdhbmcncyBbUk5BLXNlcSBkYXRhIGFuYWx5c2lzIHBhZ2VdKGh0dHBzOi8vZGF0YWJlYXV0eS5jb20vYmxvZy90dXRvcmlhbC8yMDE2LzA5LzEzL1JOQS1zZXEtYW5hbHlzaXMuaHRtbCksIGFuZCBtYXkgc3RpbGwgYmUgcHJlZmVyYWJsZSBpZiB5b3VyIGFuYWx5c2lzIGRvZXNuJ3QganVzdCBjYWxsIGZvciBnZW5lLWxldmVsIGNvdW50cy4gCjxkaXYgY2xhc3M9ImluZm9ybWF0aW9uIj4KCk1hcHBpbmcgLT4gSElTQVQyCgo8L2Rpdj4KCiMjIyMgMS4gIE1hcC9hbGlnbiB0aGUgcmVhZHMgd2l0aCBISVNBVDIgdG8gdGhlIGhnMzggcmVmZXJlbmNlIGdlbm9tZQpJbiB0aGUgbGVmdCB0b29sIHBhbmVsIG1lbnUsIHVuZGVyIE5HUyBBbmFseXNpcywgc2VsZWN0CioqTWFwcGluZyA+IEhJU0FUMioqIGFuZCBzZXQgdGhlIHBhcmFtZXRlcnMgYXMgZm9sbG93czogIAoKLSAqKklzIHRoaXMgc2luZ2xlLWVuZCBvciBwYWlyZWQtZW5kIGRhdGE/KiogU2luZ2xlLWVuZCAoYXMgaW5kaXZpZHVhbCBkYXRhc2V0cykgIAotICoqRkFTVFEgZmlsZSoqICAKKENsaWNrIG9uIHRoZSBtdWx0aXBsZSBkYXRhc2V0cyBpY29uIGFuZCBzZWxlY3QgYWxsIGZvdXIgb2YgdGhlCkZBU1RRIGZpbGVzKQogICAgLSBgU1JSNzEwODM4OC5mYXN0cS5nemAKICAgIC0gYFNSUjcxMDgzODkuZmFzdHEuZ3pgCiAgICAtIGBTUlI3MTA4MzkyLmZhc3RxLmd6YAogICAgLSBgU1JSNzEwODM5My5mYXN0cS5nemAKCi0gKipTb3VyY2UgZm9yIHRoZSByZWZlcmVuY2UgZ2Vub21lKiogVXNlCmJ1aWx0LWluIGdlbm9tZQotICoqU2VsZWN0IGEgcmVmZXJlbmNlIGdlbm9tZToqKiBIdW1hbiBEZWMgMjAxMy4gKEdSQ2gzOC9oZzM4KSAoaGczOCkKLSBVc2UgZGVmYXVsdHMgZm9yIHRoZSBvdGhlciBmaWVsZHMKLSBFeGVjdXRlCgoKCiMjIyBRdWFudGlmaWNhdGlvbiAoQ291bnRpbmcgcmVhZHMgaW4gZmVhdHVyZXMpCgpgSFRTZXEtY291bnRgIGNyZWF0ZXMgYSBjb3VudCBtYXRyaXggdXNpbmcgdGhlIG51bWJlciBvZiB0aGUgcmVhZHMgZnJvbSBlYWNoIGJhbSBmaWxlIHRoYXQgbWFwIHRvIHRoZSBnZW5vbWljIGZlYXR1cmVzLiBGb3IgZWFjaCBmZWF0dXJlIChhIGdlbmUgZm9yIGV4YW1wbGUpIGEgY291bnQgbWF0cml4IHNob3dzIGhvdyBtYW55IHJlYWRzIHdlcmUgbWFwcGVkIHRvIHRoaXMKZmVhdHVyZS4KClZhcmlvdXMgcnVsZXMgY2FuIGJlIHVzZWQgdG8gYXNzaWduIGNvdW50cyB0byBmZWF0dXJlcwoKIVtdKG1lZGlhL2h0c2VxLnBuZykKClRvIG9idGFpbiB0aGUgY29vcmRpbmF0ZXMgb2YgZWFjaCBnZW5lLCB3ZSBjYW4gdXNlIHRoZSBVQ1NDIGdlbm9tZSBicm93c2VyIHdoaWNoIGlzIGludGVncmF0ZWQgaW50byBHYWxheHkuIFVuZm9ydHVuYXRlbHksIHRoZSBFbnNlbWJsIEZUUCBzaXRlICoqY2Fubm90KiogYmUgdXNlZCBhcyB0aGUgY2hyb21vc29tZSBuYW1pbmcgY29udmVudGlvbnMgdXNlZCBpbiBFbnNlbWJsIGFyZSBkaWZmZXJlbnQgdG8gdGhlIGNocm9tb3NvbWUgbmFtaW5nIHNjaGVtZSB1c2VkIGluIHRoZSByZWZlcmVuY2UgZ2Vub21lcyBzdXBwbGllZCBieSBHYWxheHkgKDEsMiwuLiB2cyBjaHIxLCBjaHIyKS4gClRoZSBhbHRlcm5hdGl2ZSB3b3VsZCBiZSB0byBkb3dubG9hZCBhIG1hdGNoaW5nIGd0ZiBhbmQgZ2Vub21lIHJlZmVyZW5jZSBzZXF1ZW5jZSBmcm9tIEVuc2VtYmwgYW5kIHVwbG9hZCBib3RoIHRvIEdhbGF4eS4gVGhpcyB3b3VsZCB0YWtlIG1vcmUgdGltZSBhbmQgc3BhY2Ugb24gdGhlIEdhbGF4eSBzZXJ2ZXIuCgojIyMgT2J0YWluaW5nIGdlbmUgY29vcmRpbmF0ZXMKCjxkaXYgY2xhc3M9ImluZm9ybWF0aW9uIj4KCioqR2V0IERhdGEqKiAtPiAqKlVDU0MgTWFpbioqIHRhYmxlIGJyb3dzZXIKCjwvZGl2PgoKCiFbXShtZWRpYS91Y3NjX2Jyb3dzZXIucG5nKQoKU2VsZWN0aW5nIHRoZSAqKlVDU0MgTWFpbioqIHRvb2wgZnJvbSBHYWxheHkgd2lsbCB0YWtlIHlvdSB0byB0aGUgVUNTQyB0YWJsZSBicm93c2VyLiBGcm9tIGhlcmUgd2UgY2FuIGV4dHJhY3QgZ2VuZSBjb29yZGluYXRlcyBmb3Igb3VyIGdlbm9tZSBvZiBpbnRlcmVzdCAoYGhnMzhgKSBpbiBgZ3RmYCBmb3JtYXQgZm9yIHByb2Nlc3Npbmcgd2l0aCBnYWxheHkuCgotIFNldCAqY2xhZGUqIHRvICoqTWFtbWFsKioKLSBTZXQgKmdlbm9tZSogdG8gKipIdW1hbioqCi0gKmFzc2VtYmx5KiAqKkRlYy4yMDEzIChHUkNtMzgvaGczOCkqKgotICpncm91cCogKipHZW5lcyBhbmQgR2VuZSBQcmVkaWN0aW9uKioKLSAqdHJhY2sqICoqTkNCSSBSZWZTZXEqKgotICp0YWJsZSogKipVQ1NDIFJlZlNlcSAocmVmR2VuZSkqKgotICpyZWdpb24qICoqZ2Vub21lKioKLSAqb3V0cHV0IGZvcm1hdCogKipHVEYgLSBnZW5lIHRyYW5zZmVyIGZvcm1hdCAobGltaXRlZCkqKiBhbmQgKnNlbmQgb3V0cHV0IHRvKiAqKkdhbGF4eSoqCgpDbGljayAqZ2V0IG91dHB1dCogYW5kICpzZW5kIHF1ZXJ5IHRvIEdhbGF4eSogdG8gYmUgcmV0dXJuZWQgdG8gR2FsYXh5LiBBIG5ldyBqb2Igd2lsbCBiZSBzdWJtaXR0ZWQgdG8gcmV0cmlldmUgdGhlIGNvb3JkaW5hdGVzIGZyb20gVUNTQwoKPGRpdiBjbGFzcz0id2FybmluZyI+CgpXaGVuIHlvdSBhcmUgcmV0dXJuZWQgdG8gR2FsYXh5IGZyb20gVUNTQyBpdCBtaWdodCBsb29rIGxpa2UgeW91IGhhdmUgbG9zdCBhbGwgdGggZmlsZXMgaW4geW91ciBhbmFseXNpcyBhbmQgYXJlIG5vIGxvbmdlciBsb2dnZWQgaW4uIAoKVG8gc29sdmUgdGhpcywgbG9nIGJhY2sgaW4gYW5kIGNob29zZSB0aGUgKipWaWV3IGFsbCBoaXN0b3JpZXMqKiBvcHRpb24gdW5kZXIgdGhlIEhpc3RvcnkgcGFuZWwuCgohW10obWVkaWEvZ2FsYXh5X2hpc3RvcnkuUE5HKQoKVGhlcmUgc2hvdWxkIGJlIHR3byAiaGlzdG9yaWVzIjsgb25lIGNvbnRhaW5pbmcgYWxsIHRoZSBvdXRwdXRzIHlvdSBnZW5lcmF0ZWQgYmVmb3JlIGFjY2Vzc2luZyBVQ1NDLCBhbmQgb25lIGNvbnRhaW5pbmcgdGhlIFVDU0Mgb3V0cHV0LiBBbGwgdGhpcyBwb2ludCB5b3UgY2FuIHN3aXRjaCBiYWNrIHRvIHlvdXIgcHJldmlvdXMgaGlzdG9yeSwgYW5kIGRyYWcgdGhlIGJveCBjb250YWluaW5nIHRoZSBVQ1NDIG91cHV0IHRvIHRoaXMgaGlzdG9yeQoKIVtdKG1lZGlhL3N3aXRjaF9oaXN0b3JpZXMuUE5HKQoKPC9kaXY+CgoKIyMjIENvdW50aW5nIHJlYWRzIGluIGdlbmVzCgo8ZGl2IGNsYXNzPSJpbmZvcm1hdGlvbiI+CioqUk5BIEFuYWx5c2lzID4gaHRzZXEtY291bnQqKgo8L2Rpdj4KCjEuICBVc2UgSFRTZXEtY291bnQgdG8gY291bnQgdGhlIG51bWJlciBvZiByZWFkcyBmb3IgZWFjaCBmZWF0dXJlLiAgCiAgICBJbiB0aGUgbGVmdCB0b29sIHBhbmVsIG1lbnUsIHVuZGVyIE5HUyBBbmFseXNpcywgc2VsZWN0CiAgICAqKk5HUyBBbmFseXNpcyA+IGh0c2VxLWNvdW50KiogYW5kIHNldCB0aGUgcGFyYW1ldGVycyBhcyBmb2xsb3dzOiAgCiAgICAtICoqQWxpZ25lZCBTQU0vQkFNIGZpbGUqKiAgCiAgICAgIChTZWxlY3QgYWxsIGZvdXIgYmFtIGZpbGVzIHVzaW5nIHRoZSBtdWx0aXBsZSBkYXRhc2V0cyBvcHRpb24pCiAgICAtICoqR0ZGIGZpbGUqKiBVQ1NDIE1haW4gb24gSHVtYW46cmVmR2VuZSAoZ2Vub21lKQogICAgLSBVc2UgZGVmYXVsdHMgZm9yIHRoZSBvdGhlciBmaWVsZHMKICAgIC0gRXhlY3V0ZQoyLiAgUmVwZWF0IGZvciB0aGUgcmVtYWluaW5nIGJhbSBmaWxlcyBpZiBydW5uaW5nIG9uIGVhY2ggYmFtIHNlcGFyYXRlbHkuCjMuICBUbyBtYWtlIHRoaW5ncyBlYXNpZXIgdG8gdHJhY2ssIHJlbmFtZSB0aGUgaHQtc2VxIG91dHB1dCBmb3IgZWFjaCBzYW1wbGUgdG8gY29udGFpbiB0aGUgY29ycmVzcG9uZGluZyBzYW1wbGUgbmFtZSAoZS5nLiBTUlIxNTUyNDQ0Lmh0c2VxKS4gKipEbyBub3QgcmVuYW1lIHRoZSBvdXRwdXRzIHRoYXQgaGF2ZSAiKG5vIGZlYXR1cmUpIiBpbiB0aGVpciBuYW1lKioKCgoKIyMjIENyZWF0ZSBhIGNvdW50IG1hdHJpeAoKVGhlIGh0c2VxIHRvb2wgaXMgZGVzaWduZWQgdG8gcHJvZHVjZSBhIHNlcGFyYXRlIHRhYmxlIG9mIGNvdW50cyBmb3IgZWFjaCBzYW1wbGUuIFRoaXMgaXMgbm90IHBhcnRpY3VsYXJseSB1c2VmdWwgZm9yIG90aGVyIHRvb2xzIHN1Y2ggYXMgRGVndXN0IChzZWUgbmV4dCBzZWN0aW9uKSB3aGljaCByZXF1aXJlIHRoZSBjb3VudHMgdG8gYmUgcHJlc2VudGVkIGluIGEgZGF0YSBtYXRyaXggd2hlcmUgZWFjaCByb3cgaXMgYSBnZW5lIGFuZCBlYWNoIGNvbHVtbiBpcyBhIHBhcnRpY3VsYXIgc2FtcGxlIGluIHRoZSBkYXRhc2V0LgoKPGRpdiBjbGFzcz0iaW5mb3JtYXRpb24iPgoqQ29sbGVjdGlvbiBPcGVyYXRpb25zIC0+IENvbHVtbiBKb2luKiBvbiBDb2xsZWN0aW9ucwo8L2Rpdj4KCi0gSW4gdGhlICpUYWJ1bGFyIEZpbGVzKiBzZWN0aW9uLCBzZWxlY3QgdGhlIGBodC1zZXFgIGNvdW50IGZpbGVzIGZyb20geW91ciBoaXN0b3J5ICpTUlIxNTUyNDQ0Lmh0c2VxKiwgKlNSUjE1NTI0NTAqLCBldGMuLi4gSG9sZGluZyB0aGUgQ1RSTCBrZXkgYWxsb3dzIG11bHRpcGxlIGZpbGVzIHRvIGJlIHNlbGVjdGVkCi0gS2VlcCAqSWRlbnRpZmllciBjb2x1bW4qIGFzIGAxYAoKCgoKIyBEaWZmZXJlbnRpYWwgRXhwcmVzc2lvbiB1c2luZyBEZWd1c3QKCkRpZmZlcmVudGlhbCBleHByZXNzaW9uIGlzIHBvc3NpYmxlIHVzaW5nIEdhbGF4eSB1c2luZyB0aGUgREVTZXEyIHRvb2wgKGZvciBleGFtcGxlKS4gSG93ZXZlciwgb3VyIHBhcnRpY3VsYXIgcmVjb21tZW5kYXRpb24gaXMgdG8gdXNlIERlZ3VzdCBmb3IgYSBtb3JlIGludGVyYWN0aXZlIGV4cGVyaWVuY2UuIEZvciB0aGlzIHNlY3Rpb24sIHdlIHdpbGwgYmUgdXNpbmcgY291bnRzIGdlbmVyYXRlZCBvbiB0aGUgKmZ1bGwgZGF0YXNldCosIHJhdGhlciB0aGFuIHRoZSAqZG93bnNhbXBsZWQqIGRhdGEgYW5hbHlzZWQgaW4gdGhlIHByZXZpb3VzIHNlY3Rpb24uIFRoZXNlIGNvdW50cyBhcmUgYXZhaWxhYmxlIGluIHRoZSBmaWxlIGBHU0UxMTQwMTNfc2FsbW9uX2NvdW50cy5jc3ZgIGZyb20gdGhlIGdvb2dsZSBkcml2ZSwgb3IgY2FuIGJlIGRvd25sb2FkZWQgdXNpbmcgdGhlIGxpbmsgYmVsb3cKCi0gW0dTRTExNDAxM19zYWxtb25fY291bnRzLmNzdl0oR1NFMTE0MDEzX3NhbG1vbl9jb3VudHMuY3N2KQoKCiMjIERpZmZlcmVudGlhbCBleHByZXNzaW9uCgpUaGUgdGVybSAqZGlmZmVyZW50aWFsIGV4cHJlc3Npb24qIHdhcyBmaXJzdCB1c2VkIHRvIHJlZmVyIHRvIHRoZSBwcm9jZXNzIG9mIGZpbmRpbmcgc3RhdGlzdGljYWxseSBzaWduaWZpY2FudCBnZW5lcyBmcm9tIGEgKm1pY3JvYXJyYXkqIGdlbmUgZXhwcmVzc2lvbiBzdHVkeS4KCiFbXShtZWRpYS9kZV9leHBsYWluZWQucG5nKQoKClN1Y2ggbWV0aG9kcyB3ZXJlIGRldmVsb3BlZCBvbiB0aGUgcHJlbWlzZSB0aGF0IG1pY3JvYXJyYXkgZXhwcmVzc2lvbiB2YWx1ZXMgYXJlIGFwcHJveGltYXRlbHkgKm5vcm1hbGx5LWRpc3RyaWJ1dGVkKiB3aGVuIGFwcHJvcHJpYXRlbHkgdHJhbnNmb3JtZWQgKGUuZy4gYnkgdXNpbmcgYSBsb2ckXzIkIHRyYW5zZm9ybWF0aW9uKSBzbyB0aGF0IGEgbW9kaWZpZWQgdmVyc2lvbiBvZiB0aGUgc3RhbmRhcmQgKnQtdGVzdCogY2FuIGJlIHVzZWQuIFRoZSBzYW1lIHRlc3QgaXMgYXBwbGllZCB0byBlYWNoIGdlbmUgdW5kZXIgaW52ZXN0aWdhdGlvbiB5aWVsZGluZyBhICp0ZXN0IHN0YXRpc3RpYyosICpmb2xkLWNoYW5nZSogYW5kICpwLXZhbHVlKi4gU2ltaWxhciBtZXRob2RzIGhhdmUgYmVlbiBhZGFwdGVkIHRvIFJOQS1zZXEgZGF0YSB0byBhY2NvdW50IGZvciB0aGUgZmFjdCB0aGF0IHRoZSBkYXRhIGFyZSAqY291bnQtYmFzZWQqIGFuZCBkbyBub3QgZm9sbG93IGEgbm9ybWFsIGRpc3RyaWJ1dGlvbi4KCgohW10obWVkaWEvcm5hX2FkdmFuY2VkX2RlZ3VzdF8xLnBuZykKCjxmb250IHNpemU9IjgiPltodHRwOi8vZGVndXN0LmVyYy5tb25hc2guZWR1L10oaHR0cDovL2RlZ3VzdC5lcmMubW9uYXNoLmVkdS8pPC9mb250PgoKYERlZ3VzdGAgaXMgYSB3ZWIgdG9vbCB0aGF0IGNhbiBhbmFseXNlIHRoZSBjb3VudHMgZmlsZXMgcHJvZHVjZWQgaW4gdGhlIHN0ZXAgYWJvdmUsIHRvIHRlc3QgZm9yIGRpZmZlcmVudGlhbCBnZW5lIGV4cHJlc3Npb24uIEl0IG9mZmVycyBhbmQgaW50ZXJhY3RpdmUgdmlldyBvZiB0aGUgZGlmZmVyZW50aWFsIGV4cHJlc3Npb24gcmVzdWx0cwoKVGhlIGlucHV0IGZpbGUgaXMgYSBjb3VudCBtYXRyaXggd2hlcmUgZWFjaCByb3cgaXMgYSBtZWFzdXJlZCBnZW5lLCBhbmQgZWFjaCBjb2x1bW4gaXMgYSBkaWZmZXJlbnQgYmlvbG9naWNhbCBzYW1wbGUuIFdpdGhpbiB0aGUgdG9vbCB3ZSBjYW4gY29uZmlndXJlIHdoaWNoIHNhbXBsZXMgYmVsb25nIHRvIHRoZSBkaWZmZXJlbnQgYmlvbG9naWNhbCBncm91cHMgb2YgaW50ZXJlc3QuICoqSXQgaXMgaW1wb3J0YW50IHRoYXQgbm8gbm9ybWFsaXNhdGlvbiBoYXMgYmVlbiBhcHBsaWVkIHRvIHRoZSBjb3VudHMgdGhhdCBhcmUgc3VibWl0dGVkIHRvIERFR1VTVCoqLgoKUmVhZCBjb3VudHMgaGF2ZSB0byBiZSBub3JtYWxpc2VkIGZpcnN0IHByaW9yIHRvIGRpZmZlcmVudGlhbCBleHByZXNzaW9uIHRlc3RpbmcuIFRoaXMgcGFnZSBmcm9tIEhhcnZhcmQgQmlvaW5mb3JtYXRpY3Mgc3VtbWFyaXNlcyB0aGUgbWFpbiBiaWFzZXMgaW4gInJhdyIgUk5BLXNlcSBjb3VudHM7IFNlcXVlbmNpbmcgZGVwdGgsIEdlbmUgbGVuZ3RoIGFuZCBSTkEgY29tcG9zaXRpb24KCi0gW2h0dHBzOi8vaGJjdHJhaW5pbmcuZ2l0aHViLmlvL0RHRV93b3Jrc2hvcC9sZXNzb25zLzAyX0RHRV9jb3VudF9ub3JtYWxpemF0aW9uLmh0bWxdKGh0dHBzOi8vaGJjdHJhaW5pbmcuZ2l0aHViLmlvL0RHRV93b3Jrc2hvcC9sZXNzb25zLzAyX0RHRV9jb3VudF9ub3JtYWxpemF0aW9uLmh0bWwpCgoKTWV0aG9kcyBzdWNoIGFzIGBlZGdlUmAgKGltcGxlbWVudGVkIGluIERlZ3VzdCkgYW5kIGBERVNlcTJgIGhhdmUgdGhlaXIgb3duIG1ldGhvZCBvZiBub3JtYWxpc2luZyBjb3VudHMuIFlvdSB3aWxsIHByb2JhYmx5IGVuY291bnRlciBvdGhlciBtZXRob2RzIG9mIG5vcm1hbGlzaW5nIFJOQS1zZXEgcmVhZHMgc3VjaCBhcyAqUlBLTSosICpDUE0qLCAqVFBNKiBldGMuIFtUaGlzIGJsb2ddKGh0dHBzOi8vd3d3LnJuYS1zZXFibG9nLmNvbS9ycGttLWZwa20tYW5kLXRwbS1jbGVhcmx5LWV4cGxhaW5lZC8pIHByb3ZpZGVzIGEgbmljZSBleHBsYW5hdGlvbiBvZiB0aGUgY3VycmVudCB0aGlua2luZy4gQXMgcGFydCBvZiB0aGUgYERlZ3VzdGAgb3V0cHV0LCB5b3UgaGF2ZSB0aGUgb3B0aW9uIG9mIGRvd25sb2FkaW5nIG5vcm1hbGlzZWQgY291bnRzIGluIHZhcmlvdXMgZm9ybWF0cy4gU29tZSBvdGhlciBvbmxpbmUgdmlzdWFsaXNhdGlvbiB0b29scyByZXF1aXJlIG5vcm1hbGlzZWQgY291bnRzIGFzIGlucHV0LCBzbyBpdCBpcyBnb29kIHRvIGhhdmUgdGhlc2UgdG8taGFuZC4KCgojIyMgVXBsb2FkaW5nIHRoZSBjb3VudCBtYXRyaXggdG8gRGVndXN0CgoKLSBGcm9tIHRoZSBtYWluIGRlZ3VzdCBwYWdlLCBjbGljayAqVXBsb2FkIHlvdXIgY291bnRzIGZpbGUqCi0gQ2xpY2sgb24gQnJvd3NlCi0gU2VsZWN0IHRoZSBsb2NhdGlvbiBvZiB0aGUgZmlsZSBgR1NFMTE0MDEzX3NhbG1vbl9jb3VudHMuY3N2YCwgYW5kIGNsaWNrICpPcGVuKi4KLSBDbGljayAqVXBsb2FkKgotIEEgQ29uZmlndXJhdGlvbiBwYWdlIHdpbGwgYXBwZWFyLgoKIVtdKG1lZGlhL2RlZ3VzdF9jb25maWcucG5nKQoKLSBGb3IgTmFtZSB0eXBlICIqR1NFMTE0MDEzKiIgKG9yIHdoYXRldmVyIHlvdSB3YW50IHRvIGNhbGwgdGhlIGFuYWx5c2lzKQotIEZvciBJbmZvIGNvbHVtbnMgc2VsZWN0ICpTWU1CT0wqCi0gQ2xpY2sgQWRkIGNvbmRpdGlvbgogICAgKyBSZWZlcnJpbmcgdG8gdGhlIGV4cGVyaW1lbnQgZGVzaWduIChiZWxvdyksIHNlbGVjdCB0aGUgRE1TTyBzYW1wbGVzIGFuZCBjYWxsIHRoZSBjb25kaXRpb24gRE1TTwogICAgKyBSZXBlYXQgZm9yIHRoZSBJVFJBQ09OQVpPTEUgc2FtcGxlcy4gV2Ugd2lsbCByZWZlciB0byB0aGlzIGFzICoqSVRSQSoqIGZvciBzaG9ydC4KLSBTYXZlIHRoZSBzZXR0aW5ncyBhbmQgdGhlbiBWaWV3IHRoZSByZXN1bHRzCgpydW4JfCBuYW1lCXwgY2VsbF9saW5lCXwgY29uZGl0aW9uCi0tLS18LS0tLS0tLXwtLS0tLS0tLS0tLXwtLS0tLS0tLS0gClNSUjcxMDgzODgJfCBIVDU1X0NPTlRfMQl8IEhUNTUJfCBETVNPClNSUjcxMDgzODkJfCBIVDU1X0NPTlRfMgl8IEhUNTUJfCBETVNPClNSUjcxMDgzOTAJfCBIVDU1X0NPTlRfMwl8IEhUNTUJfCBETVNPClNSUjcxMDgzOTEJfCBIVDU1X0NPTlRfNAl8IEhUNTUJfCBETVNPClNSUjcxMDgzOTIJfCBIVDU1X0lUUkFfMQl8IEhUNTUJfCBJVFJBQ09OQVpPTEUKU1JSNzEwODM5Mwl8IEhUNTVfSVRSQV8yCXwgSFQ1NQl8IElUUkFDT05BWk9MRQpTUlI3MTA4Mzk0CXwgSFQ1NV9JVFJBXzMJfCBIVDU1CXwgSVRSQUNPTkFaT0xFClNSUjcxMDgzOTUJfCBIVDU1X0lUUkFfNAl8IEhUNTUJfCBJVFJBQ09OQVpPTEUKU1JSNzEwODM5Ngl8IFNXOTQ4X0NPTlRfMSB8IFNXOTQ4CXwgRE1TTwpTUlI3MTA4Mzk3CXwgU1c5NDhfQ09OVF8yIHwgU1c5NDgJfCBETVNPClNSUjcxMDgzOTgJfCBTVzk0OF9DT05UXzMgfCBTVzk0OAl8IERNU08KU1JSNzEwODM5OQl8IFNXOTQ4X0NPTlRfNCB8IFNXOTQ4CXwgRE1TTwpTUlI3MTA4NDAwCXwgU1c5NDhfSVRSQV8xIHwgU1c5NDgJfCBJVFJBQ09OQVpPTEUKU1JSNzEwODQwMQl8IFNXOTQ4X0lUUkFfMiB8IFNXOTQ4CXwgSVRSQUNPTkFaT0xFClNSUjcxMDg0MDIJfCBTVzk0OF9JVFJBXzMgfCBTVzk0OAl8IElUUkFDT05BWk9MRQpTUlI3MTA4NDAzCXwgU1c5NDhfSVRSQV80IHwgU1c5NDgJfCBJVFJBQ09OQVpPTEUKCgoKIyMjIE92ZXJ2aWV3IG9mIERlZ3VzdCBzZWN0aW9ucwoKLSBUb3AgYmxhY2sgcGFuZWwgd2l0aCBDb25maWd1cmUgc2V0dGluZ3MgYXQgcmlnaHQuIFVzZWQgdG8gY2hhbmdlIHRoZSBleHBlcmltZW50YWwgY29uZGl0aW9ucwotIExlZnQ6IENvbnRyb2wgb3ZlciB3aGljaCBjb25kaXRpb25zIGFyZSBjb21wYXJlZCBhbmQgdGhlIGN1dC1vZmZzIGZvciBkZWNpZGluZyBERSBnZW5lcy4KLSBUb3AgY2VudHJlOiBJbnRlcmFjdGl2ZSBwbG90cyAoc2VlIGJlbG93KQotIEEgdGFibGUgb2YgZ2VuZXMgKG9yIGZlYXR1cmVzKTsgZXhwcmVzc2lvbiBpbiAidHJlYXRtZW50IiByZWxhdGl2ZSB0byAiY29udHJvbCI7IGFuZCBzaWduaWZpY2FuY2UgKEZEUiBjb2x1bW4pLgoKKCoqTm90IHRoYXQgdGhlIHNjcmVlbnNob3RzIGFyZSBmb3IgaWxsdXN0cmF0aW9uIHB1cnBvc2VzIGFuZCB0YWtlbiBmcm9tIGEgZGlmZmVyZW50IGRhdGFzZXQgdG8gdGhhdCBiZWluZyBhbmFseXNlZCBpbiB0aGUgdHV0b3JpYWwuKiopCgohW10oaHR0cDovL3NlcHNpcy1vbWljcy5naXRodWIuaW8vdHV0b3JpYWxzL21vZHVsZXMvZGdlL2ltYWdlcy9pbWFnZTEyLnBuZykKCgojIyMgTUEtcGxvdAoKIVtdKG1lZGlhL2RlZ3VzdF9tYS5wbmcpCgpFYWNoIGRvdCBzaG93cyB0aGUgY2hhbmdlIGluIGV4cHJlc3Npb24gaW4gb25lIGdlbmUuCgotIFRoZSBhdmVyYWdlIGV4cHJlc3Npb24gKG92ZXIgYm90aCBjb25kaXRpb24gYW5kIHRyZWF0bWVudCBzYW1wbGVzKSBpcyByZXByZXNlbnRlZCBvbiB0aGUgeC1heGlzLgogICAgKyBQbG90IHBvaW50cyBzaG91bGQgYmUgc3ltbWV0cmljYWwgYXJvdW5kIHRoZSB4LWF4aXMuCiAgICArIFdlIGNhbiBzZWUgdGhhdCBtYW55IGdlbmVzIGFyZSBleHByZXNzZWQgYXQgYSBsb3cgbGV2ZWwsIGFuZCBzb21lIGFyZSBoaWdobHkgZXhwcmVzc2VkLgotIFRoZSBmb2xkIGNoYW5nZSBpcyByZXByZXNlbnRlZCBvbiB0aGUgeSBheGlzLgogICAgKyBJZiBleHByZXNzaW9uIGlzIHNpZ25pZmljYW50bHkgZGlmZmVyZW50IGJldHdlZW4gYmF0Y2ggYW5kIGNoZW0sIHRoZSBkb3RzIGFyZSByZWQuIElmIG5vdCwgdGhleSBhcmUgYmx1ZS4gKEluIERlZ3VzdCwgc2lnbmlmaWNhbnQgbWVhbnMgRkRSIDwwLjA1KS4KICAgICsgQXQgbG93IGxldmVscyBvZiBnZW5lIGV4cHJlc3Npb24gKGxvdyB2YWx1ZXMgb2YgdGhlIHggYXhpcyksIGZvbGQgY2hhbmdlcyBhcmUgbGVzcyBsaWtlbHkgdG8gYmUgc2lnbmlmaWNhbnQuCgpDbGljayBvbiB0aGUgZG90IHRvIHNlZSB0aGUgZ2VuZSBuYW1lLiBUaGUgcGFuZWwgb24gdGhlIHJpZ2h0IG9mIHRoZSBNQS1wbG90IHdpbGwgYWxzbyB1cGRhdGUgdG8gc2hvdyB0aGUgZXhwcmVzc2lvbiBvZiB0aGlzIGdlbmUgaW4gZWFjaCBzYW1wbGUgaW4gdGhlIGZvcm0gb2YgYSBkb3QgcGxvdC4KCgojIyMgTURTIHBsb3QKClRoaXMgaXMgYSBtdWx0aWRpbWVuc2lvbmFsIHNjYWxpbmcgcGxvdCB3aGljaCByZXByZXNlbnRzIHRoZSB2YXJpYXRpb24gYmV0d2VlbiBzYW1wbGVzLiBJdCBpcyBhIHNpbWlsYXIgY29uY2VwdCB0byBhIFByaW5jaXBhbCBDb21wb25lbnRzIEFuYWx5c2lzIChQQ0EpIHBsb3QuIFRoZSB4LWF4aXMgaXMgdGhlIGRpbWVuc2lvbiB3aXRoIHRoZSBoaWdoZXN0IG1hZ25pdHVkZS4gSW4gYSBzdGFuZGFyZCBjb250cm9sL3RyZWF0bWVudCBzZXR1cCwgc2FtcGxlcyBzaG91bGQgYmUgc3BsaXQgYWxvbmcgdGhpcyBheGlzLiBBIGRlc2lyYWJsZSBwbG90IGlzIHNob3duIGJlbG93Oi0KCiFbXShtZWRpYS9kZWd1c3RfbWRzLnBuZykKCiMjIyBWb2xjYW5vIHBsb3QKClNpbWlsYXIgdG8gdGhlIE1BLXBsb3QsIHRoaXMgc2hvd3MgdGhlIHNpZ25pZmljYW5jZSBvZiBlYWNoIGdlbmUgKHktYXhpcykgYW5kIG1hZ25pdHVkZSBvZiBjaGFuZ2UgKHgtYXhpcykuIAoKIyMjIFRhYmxlIG9mIGdlbmVzCgpJbml0aWFsbHkgdGhpcyB3aWxsIGNvbnRhaW4gdGhlIGRldGFpbHMgb2YgYWxsIGdlbmVzIHByZXNlbnQgaW4gdGhlIGRhdGFzZXQuIE9uY2UgdGhlIEZEUiBhbmQgbG9nRkMgY3V0LW9mZnMgYXJlIGFsdGVyZWQsIGFueSBnZW5lcyB0aGF0IGRvIG5vdCBtZWV0IHRoZSBjdXQtb2ZmcyB3aWxsIGJlIHJlbW92ZWQuCgohW10obWVkaWEvZGVndXN0X2dlbmVfdGFibGUucG5nKQoKCi0gSU5GTzsgVGhlIGNvbHVtbnMgdGhhdCB5b3Ugc2VsZWN0ZWQgaW4gdGhlIGNvbmZpZ3VyYXRpb24gdG8gY29udGFpbiBnZW5lIGluZm9ybWF0aW9uIChlLmcuIGdlbmUgbmFtZSwgZ2VuZSBzeW1ib2wsIEVudHJleiBJRCkKLSBGRFI6IEZhbHNlIERpc2NvdmVyeSBSYXRlLiBUaGlzIGlzIGFuIGFkanVzdGVkIHAgdmFsdWUgdG8gc2hvdyB0aGUgc2lnbmlmaWNhbmNlIG9mIHRoZSBkaWZmZXJlbmNlIGluIGdlbmUgZXhwcmVzc2lvbiBiZXR3ZWVuIHR3byBjb25kaXRpb25zLiBDbGljayBvbiBjb2x1bW4gaGVhZGluZ3MgdG8gc29ydC4gQnkgZGVmYXVsdCwgdGhpcyB0YWJsZSBpcyBzb3J0ZWQgYnkgRkRSLgotIFRyZWF0bWVudCBhbmQgQ29udHJvbDogbG9nMihGb2xkIENoYW5nZSkgb2YgZ2VuZSBleHByZXNzaW9uLiBUaGUgZGVmYXVsdCBkaXNwbGF5IGlzIG9mIGZvbGQgY2hhbmdlIGluIHRoZSB0cmVhdG1lbnQgcmVsYXRpdmUgdG8gdGhlIGNvbnRyb2wuIFRoZXJlZm9yZSwgdmFsdWVzIGluIHRoZSBiYXRjaCBjb2x1bW4gYXJlIHplcm8uIFRoaXMgY2FuIGJlIGNoYW5nZWQgaW4gdGhlIE9wdGlvbnMgcGFuZWwgYXQgdGhlIHRvcCByaWdodC4KICAgICsgSW4gc29tZSBjYXNlcywgYSBsYXJnZSBmb2xkIGNoYW5nZSB3aWxsIGJlIG1lYW5pbmdmdWwgYnV0IGluIG90aGVycywgZXZlbiBhIHNtYWxsIGZvbGQgY2hhbmdlIGNhbiBiZSBpbXBvcnRhbnQgYmlvbG9naWNhbGx5LgoKVGhlIHRhYmxlIGNhbiBiZSBzb3J0ZWQgYWNjb3JkaW5nIHRvIGFueSBvZiB0aGUgY29sdW1ucyAoZS5nLiBmb2xkLWNoYW5nZSBvciBwLXZhbHVlKQoKCiMjIyBEb3dubG9hZCBhbmQgUiBjb2RlCgpBYm92ZSB0aGUgZ2VuZXMgdGFibGUgaXMgdGhlIG9wdGlvbiB0byBkb3dubG9hZCB0aGUgcmVzdWx0cyBvZiB0aGUgY3VycmVudCBhbmFseXNpcyB0byBhIGNzdiBmaWxlLiBZb3UgY2FuIGFsc28gZG93bmxvYWQgdGhlICpSKiBjb2RlIHJlcXVpcmVkIHRvIHJlcHJvZHVjZSB0aGUgYW5hbHlzaXMgYnkgY2xpY2tpbmcgdGhlICpTaG93IFIgY29kZSogYm94IHVuZGVybmVhdGggdGhlIE9wdGlvbnMgYm94LgoKIyMjIEV4ZXJjaXNlCgo8ZGl2IGNsYXNzPSJleGVyY2lzZSI+CioqUXVlc3Rpb246KiogTWFrZSBhIG5vdGUgb2YgaG93IG1hbnkgZ2VuZXMgYXJlIGRldGVjdGVkIHdpdGggYSAqKkZEUioqIDwgMC4wNSBhbmQgKiphYnMgbG9nRkMqKiA+IDEgSXQgc2VlbXMgdGhhdCB0aGUgZGlmZmVyZW50aWFsIGV4cHJlc3Npb24gYW5hbHlzaXMgaXMgZGV0ZWN0aW5nIGxvdHMgb2YgZ2VuZXMuIEhvd2V2ZXIsIGRvZXMgdGhpcyB0ZWxsIHRoZSB3aG9sZSBzdG9yeSBhYm91dCB0aGUgZGF0YXNldD8gV2hhdCBkbyB5b3UgdGhpbmsgaXMgdGhlIG1haW4gZmFjdG9yIHNlcGFyYXRpbmcgc2FtcGxlcyBvbiB0aGUgeC1heGlzLCBhbmQgdGh1cyBleHBsYWluaW5nIHRoZSBtb3N0IHZhcmlhdGlvbiBpbiB0aGUgZGF0YT8KCjwvZGl2PgoKCgojIyBNb2RpZnlpbmcgdGhlIGFuYWx5c2lzCgpXZSB3aWxsIG5vdyByZXBlYXQgdGhlIGFuYWx5c2lzLCBidXQgb25seSBmb3Igc2FtcGxlcyBmcm9tIHRoZSAqSFQ1NSogY2VsbC1saW5lLiBUaGUgY29ycmVjdCBjb25maWd1cmF0aW9uIGZvciB0aGlzIGFuYWx5c2lzIGlzIHNob3duIGJlbG93Oi0KCiFbXShtZWRpYS9kZWd1c3RfaHQ1NS5QTkcpCgo8ZGl2IGNsYXNzPSJleGVyY2lzZSI+CioqRXhlcmNpc2UqKjogSG93IG1hbnkgZ2VuZXMgYXJlIGRpZmZlcmVudGlhbGx5LWV4cHJlc3NlZCB3aXRoIGFuIEZEUiA8IDAuMDUgYW5kIGFicyBsb2dGQyA+IDEuIERvd25sb2FkIHRoaXMgZmlsZSBhbmQgcmVuYW1lIGl0IHRvIGBIVDU1LklUUkFDT05BWk9MRV92c19ETVNPLmNzdmAuCjwvZGl2PgoKPGRpdiBjbGFzcz0iZXhlcmNpc2UiPgoqKkV4ZXJjaXNlOioqIFJlc3QgdGhlIEZEUiBjdXQtb2ZmIGFuZCBhYnMgTG9nRkMgY3V0b2ZmcyBiYWNrIHRvIGRlZmF1bHQgKDEgYW5kIDAgcmVzcGVjdGl2ZWx5KSBhbmQgKmRvd25sb2FkKiB0aGUgZmlsZS4gUmVuYW1lIHRoZSBmaWxlIHRvIGBiYWNrZ3JvdW5kLmNzdmAuIFdlIHdpbGwgdXNlIHRoaXMgbGF0ZXIuCjwvZGl2PgoKPGRpdiBjbGFzcz0iZXhlcmNpc2UiPgoqKkV4ZXJjaXNlKio6IFJlcGVhdCB0aGUgYW5hbHlzaXMgZm9yIFNXOTQ4IHNhbXBsZXMgYW5kIGRvd25sb2FkIHRoZSB0YWJsZSBvZiBkaWZmZXJlbnRpYWxseS1leHByZXNzZWQgcmVzdWx0cyAoc2FtZSBGRFIgYW5kIGxvZyBmb2xkLWNoYW5nZSkgdG8gYFNXOTQ4LklUUkFDT05BWk9MRV92c19ETVNPLmNzdmAKPC9kaXY+CgoKCiMjIyBGaWxlIERvd25sb2FkcwoKPGRpdiBjbGFzcz0iaW5mb3JtYXRpb24iPgpJZiB5b3UgZGlkbid0IG1hbmFnZSB0byBjb21wbGV0ZSB0aGVzZSBhbmFseXNlcywgeW91IGNhbiBkb3dubG9hZCB0aGUgZmlsZXMgZnJvbSBoZXJlIGJ5IHJpZ2h0LWNsaWNraW5nIG9uIGVhY2ggbGluayBhbmQgc2VsZWN0aW5nICJTYXZlIExpbmsgYXMiIChvciBlcXVpdmFsZW50KS4gVGhleSBhcmUgYWxzbyBhdmFpbGFibGUgaW4gdGhlIGNvdXJzZSBnb29nbGUgZHJpdmUuCgotIFtIVDU1LklUUkFDT05BWk9MRV92c19ETVNPLmNzdl0oSFQ1NS5JVFJBQ09OQVpPTEVfdnNfRE1TTy5jc3YpCi0gW1NXOTQ4LklUUkFDT05BWk9MRV92c19ETVNPLmNzdl0oU1c5NDguSVRSQUNPTkFaT0xFX3ZzX0RNU08uY3N2KQotIFtiYWNrZ3JvdW5kLmNzdl0oYmFja2dyb3VuZC5jc3YpCgo8L2Rpdj4KCiMjIyBPdmVybGFwcGluZyBHZW5lIExpc3RzCgpXZSBtaWdodCBzb21ldGltZXMgd2FudCB0byBjb21wYXJlIHRoZSBsaXN0cyBvZiBnZW5lcyB0aGF0IHdlIGlkZW50aWZ5IHVzaW5nIGRpZmZlcmVudCBtZXRob2RzLCBvciBnZW5lcyBpZGVudGlmaWVkIGZyb20gbW9yZSB0aGFuIG9uZSBjb250cmFzdC4gSW4gb3VyIGV4YW1wbGUgZGF0YXNldCB3ZSBjYW4gY29tcGFyZSB0aGUgZ2VuZXMgaW4gdGhlIGNvbnRyYXN0IG9mIElUUkFDT05BWk9MRSB2cyBETVNPIGluIEhUNTUgYW5kIFNXOTQ4IGNlbGxzCgpUaGUgd2Vic2l0ZSAqdmVubnkqIHByb3ZpZGVzIGEgcmVhbGx5IG5pY2UgaW50ZXJmYWNlIGZvciBkb2luZyB0aGlzLgoKIVtdKG1lZGlhL3Zlbm55X2NvbmZpZy5wbmcpCgotIE9wZW4gYm90aCB5b3VyICpIVDU1OiBJVFJBQ09OQVpPTEUgdnMgRE1TTyogYW5kICpTVzk0ODogSVRSQUNPTkFaT0xFIHZzIERNU08qIHJlc3VsdHMgZmlsZXMgaW4gRXhjZWwKLSBHbyB0byB0aGUgdmVubnkgd2Vic2l0ZQogICAgKyBodHRwOi8vYmlvaW5mb2dwLmNuYi5jc2ljLmVzL3Rvb2xzL3Zlbm55LwotIENvcHkgdGhlIG5hbWVzIG9mIGdlbmVzIGluIHRoZSBIVDU1IGZpbGUgaW50byB0aGUgKipMaXN0IDEqKiBib3ggb24gdGhlIHZlbm55IHdlYnNpdGUuICoqTGlzdCAxKiogY2FuIGJlIHJlbmFtZWQgdG8gKkhUNTUqCiAgICArICpZb3UgY2FuIHNlbGVjdCBhbGwgZW50cmllcyBpbiBhIGNvbHVtbiB3aXRoIHRoZSBzaG9ydGN1dCBDVFJMICsgU1BBQ0UqCi0gQ29weSB0aGUgbmFtZXMgb2YgZ2VuZXMgaW4gdGhlIFNXOTQ4IGFuYWx5c2lzIGludG8gdGhlICoqTGlzdCAyKiogYm94IG9uIHRoZSB2ZW5ueSB3ZWJzaXRlLiAqKkxpc3QgMioqIGNhbiBiZSByZW5hbWVkIHRvICoqU1c5NDgqKgotIHZlbm55IHNob3VsZCBub3cgcmVwb3J0IHRoZSBudW1iZXIgb2YgZ2VuZXMgZm91bmQgaW4gZWFjaCBsaXN0LCB0aGUgc2l6ZSBvZiB0aGUgaW50ZXJzZWN0aW9uLCBhbmQgZ2VuZXMgdW5pcXVlIHRvIGVhY2ggbWV0aG9kCgojIyMgUmVmaW5lZCBhbmFseXNpcwoKVGhlIGZpbmFsIGFuYWx5c2lzIHdlIHdpbGwgcGVyZm9ybSBpcyB0byBpbmNsdWRlIGFsbCB0aGUgc2FtcGxlcywgYnV0IGNvcnJlY3QgZm9yIHRoZSBkaWZmZXJlbmNlcyBpbiBjZWxsLWxpbmUuIFRoaXMgaXMgYWNoaWV2ZWQgYnkgdGVsbGluZyBEZWd1c3QgYWJvdXQgdGhlICpoaWRkZW4gZmFjdG9ycyogaW4gb3VyIGRhdGFzZXQuIFRoZSBoaWRkZW4gZmFjdG9yIGluIHRoaXMgZGF0YXNldCBpcyB3aGV0aGVyIHRoZSBzYW1wbGUgaXMgZnJvbSB0aGUgKipIVDU1Kiogb3IgKipTVzk0OCoqIHNhbXBsZXMuIEluIG90aGVyIHdvcmRzLCB0aGlzIGlzIGEgdGVjaG5pY2FsIGZhY3RvciB0aGF0IGluZmx1ZW5jZXMgb3VyIHJlc3VsdHMgYnV0IG5vdCBhIGZhY3RvciB0aGF0IHdlIHdpc2ggdG8gY29tcGFyZS4gV2Ugb25seSBuZWVkIHRvIHNwZWNpZnkgd2hpY2ggc2FtcGxlcyBhcmUgZnJvbSBIVDU1IGFuZCBERUdVU1Qgd2lsbCBpbmZlciB0aGF0IHRoZSBvdGhlciBzYW1wbGVzIGJlbG9uZyB0byBhIGRpZmZlcmVudCBjZWxsIGxpbmUuIE90aGVyIGhpZGRlbiBmYWN0b3JzIHlvdSBtaWdodCBuZWVkIHRvIGluY2x1ZGUgY291bGQgYmUgKGRlcGVuZGluZyBvbiB0aGUgTURTIHBsb3QpIDotCgotIHNhbXBsZSBiYXRjaAotIGdlbmRlcgoKU2VlIGJlbG93IGZvciB0aGUgY29ycmVjdCBjb25maWd1cmF0aW9uIHRvIGluY2x1ZGUgdGhlIGhpZGRlbiBmYWN0b3JzLgoKPGltZyBzcmM9Im1lZGlhL2hpZGRlbl9mYWN0b3IucG5nIi8+Cgo8ZGl2IGNsYXNzPSJleGVyY2lzZSI+CioqRXhlcmNpc2UqKjogSG93IG1hbnkgZ2VuZXMgYXJlIGlkZW50aWZpZWQgd2l0aCBhbiBGRFIgPCAwLjA1IGFuZCBhYnMgbG9nRkMgPiAxIGZvciB0aGlzICJoaWRkZW4gZmFjdG9yIiBhbmFseXNpcy4gSG93IGRvZXMgaXQgY29tcGFyZSB0byB0aGUgaW5pdGlhbCBjb21wYXJpc29uIG9mIERNU08gdnMgSVRSQSB1c2luZyBhbGwgc2FtcGxlcz8KPC9kaXY+CgojIEVucmljaG1lbnQgYW5kIHBhdGh3YXlzIGFuYWx5c2lzCgpJbiB0aGlzIHNlY3Rpb24gd2UgbW92ZSB0b3dhcmRzIGRpc2NvdmVyaW5nIGlmIG91ciByZXN1bHRzIGFyZSAqKipiaW9sb2dpY2FsbHkgc2lnbmlmaWNhbnQqKiouIEFyZSB0aGUgZ2VuZXMgdGhhdCB3ZSBoYXZlIHBpY2tlZCBzdGF0aXN0aWNhbCBmbHVrZXMsIG9yIGFyZSB0aGVyZSBzb21lIGNvbW1vbmFsaXRpZXMuIAoKVGhlcmUgYXJlIHR3byBkaWZmZXJlbnQgYXBwcm9hY2hlcyBvbmUgbWlnaHQgdXNlLCBhbmQgd2Ugd2lsbCBjb3ZlciB0aGUgdGhlb3J5IGJlaGluZCBib3RoLiBUaGUgZGlzdGluY3Rpb24gaXMgd2hldGhlciB5b3UgYXJlIGhhcHB5IHRvIHVzZSBhIGhhcmQgKGFuZCBhcmJpdHJhcnkpIHRocmVzaG9sZCB0byBpZGVudGlmeSBERSBnZW5lcy4KCgojIyBPdmVyLXJlcHJlc2VudGF0aW9uIGFuYWx5c2lzCgoiVGhyZXNob2xkLWJhc2VkIiBtZXRob2RzIHJlcXVpcmUgdGhlIGFwcGxpY2F0aW9uIG9mIGEgc3RhdGlzdGljYWwgdGhyZXNob2xkIHRvIGRlZmluZSBsaXN0IG9mIGdlbmVzIHRvIHRlc3QgKGUuZy4gRkRSIDwgMC4wMSkuIFRoZW4gYSAqaHlwZXJnZW9tZXRyaWMqIHRlc3Qgb3IgKkZpc2hlcidzIEV4YWN0KiB0ZXN0IGdlbmVyYWxseSB1c2VkLiBUaGVzZSBhcmUgdHlwaWNhbGx5IHVzZWQgaW4gc2l0dWF0aW9ucyB3aGVyZSBwbGVudHkgb2YgREUgZ2VuZXMgaGF2ZSBiZWVuIGlkZW50aWZpZWQsIGFuZCBwZW9wbGUgb2Z0ZW4gdXNlIHF1aXRlIHJlbGF4ZWQgY3JpdGVyaWEgZm9yIGlkZW50aWZ5aW5nIERFIGdlbmVzIChlLmcuIHJhdyByYXRoZXIgdGhhbiBhZGp1c3RlZCBwLXZhbHVlcyBvciBGRFIgdmFsdWUpCgpUaGUgcXVlc3Rpb24gd2UgYXJlIGFza2luZyBoZXJlIGlzOwoKPiAiQXJlIHRoZSBudW1iZXIgb2YgREUgZ2VuZXMgYXNzb2NpYXRlZCB3aXRoIFRoZW1lIFggc2lnbmlmaWNhbnRseSBncmVhdGVyIHRoYW4gd2hhdCB3ZSBtaWdodCBleHBlY3QgYnkgY2hhbmNlIGFsb25lPyIKCm9yCgo+ICJJZiBJIHBpY2tlZCBhIHNldCBvZiBnZW5lcyAqYXQgcmFuZG9tKiB0aGF0IGlzIHRoZSBzYW1lIGFzIHRoZSBudW1iZXIgb2YgREUgZ2VuZXMsIGhvdyBtYW55IGdlbmVzIGZyb20gVGhlbWUgWCB3b3VsZCBJIGV4cGVjdCB0byBmaW5kIj8KCldlIGNhbiBhbnN3ZXIgdGhpcyBxdWVzdGlvbiBieSBrbm93aW5nCgotIHRoZSB0b3RhbCBudW1iZXIgb2YgREUgZ2VuZXMKLSB0aGUgbnVtYmVyIG9mIGdlbmVzIGluIHRoZSBnZW5lIHNldCAocGF0aHdheSBvciBwcm9jZXNzKQotIHRoZSBudW1iZXIgb2YgZ2VuZXMgaW4gdGhlIGdlbmUgc2V0IHRoYXQgYXJlIGZvdW5kIHRvIGJlIERFCi0gdGhlIHRvdGFsIG51bWJlciBvZiB0ZXN0ZWQgZ2VuZXMgKGJhY2tncm91bmQpCgpUaGUgZm9ybXVsYSBmb3IgRmlzaGVycyBleGFjdCB0ZXN0IGlzOwoKJCQgcCA9IFxmcmFje1xiaW5vbXthICsgYn17YX1cYmlub217YyArZH17Y319e1xiaW5vbXtufXthICtjfX0gPSBcZnJhY3soYStiKSEoYytkKSEoYStjKSEoYitkKSF9e2EhYiFjIWQhbiF9ICQkCgp3aXRoOi0KCnwgICAgICAgICAgICAgIHwgaXMgREUgfCBOb3QgREUgfCBSb3cgVG90YWwgfAp8LS0tLS0tLS0tLS0tLSB8IC0tLS0tLXwtLS0tLS0tLXwtLS0tLS0tLS0tLXwKfCBJbiBHZW5lIFNldCAgIHwgYSAgICAgfCBiICAgICAgfCBhICsgYiAgICAgfAp8IE5vdCBpbiBHZW5lIFNldCAgfCBjICAgICB8IGQgICAgICB8IGMgKyBkICAgICB8CnwgQ29sdW1uIFRvdGFsICAgIHwgIGEgKyBjIHwgYiArIGQgIHwgYSArIGIgKyBjICsgZCA9IG4gfAoKPGRpdiBjbGFzcz0iaW5mb3JtYXRpb24iPgpUaGlzIGZvcm11bGEgaXMgcHJpbnRlZCBoZXJlIGZvciB5b3VyIGluZm9ybWF0aW9uLiBUaGUgc29mdHdhcmUgd2UgdXNlIHdpbGwgcGVyZm9ybSBhbGwgdGhlIGNhbGN1bGF0aW9ucyBmb3IgdXMuCjwvZGl2PgoKSW4gdGhpcyBmaXJzdCB0ZXN0LCBvdXIgZ2VuZXMgd2lsbCBiZSBncm91cGVkIHRvZ2V0aGVyIGFjY29yZGluZyB0byB0aGVpciBHZW5lIE9udG9sb2d5IChHTykgdGVybXM6LSBodHRwOi8vd3d3LmdlbmVvbnRvbG9neS5vcmcvCgoKIyMgVXNpbmcgR09yaWxsYQoKVGhlcmUgYXJlIHNldmVyYWwgcG9wdWxhciBvbmxpbmUgdG9vbHMgZm9yIHBlcmZvcm1pbmcgZW5yaWNobWVudCBhbmFseXNpcy4gV2Ugd2lsbCBiZSB1c2luZyB0aGUgb25saW5lIHRvb2wgW0dPcmlsbGFdKGh0dHA6Ly9jYmwtZ29yaWxsYS5jcy50ZWNobmlvbi5hYy5pbC8pIHRvIHBlcmZvcm0gdGhlIHBhdGh3YXlzIGFuYWx5c2lzIGFzIGl0IGlzIHBhcnRpY3VsYXJseSBzdHJhaWdodGZvcndhcmQuIEl0IGhhcyB0d28gbW9kZXM7IHRoZSBmaXJzdCBvZiB3aGljaCBhY2NlcHRzIGEgbGlzdCBvZiAqYmFja2dyb3VuZCogYW5kICp0YXJnZXQqIGdlbmVzLiAKCjEuIEdvIHRvIGh0dHA6Ly9jYmwtZ29yaWxsYS5jcy50ZWNobmlvbi5hYy5pbC8KMi4gQ2hvb3NlIE9yZ2FuaXNtOiBgSG9tbyBTYXBpZW5zYAozLiBDaG9vc2UgcnVubmluZyBtb2RlOiBgVHdvIHVucmFua2VkIGxpc3RzIG9mIGdlbmVzYAo0LiBQYXN0ZSB0aGUgZ2VuZSBzeW1ib2xzIGNvcnJlc3BvbmRpbmcgdG8gREUgZ2VuZXMgaW4gKlNXOTQ4OiBJVFJBQ09OQVpPTEUgdnMgRE1TTyogaW50byB0aGUgVGFyZ2V0IHNldC4KICArICoqVGhlIHNob3J0Y3V0IENUUkwgKyBTUEFDRSB3aWxsIGxldCB5b3Ugc2VsZWN0IGFuIGVudGlyZSBjb2x1bW4qKgo1LiBQYXN0ZSB0aGUgZ2VuZSBzeW1ib2xzIGZyb20gdGhlIEJhY2tncm91bmQgc2V0IGludG8gdGhlIG90aGVyIGJveC4gVGhpcyBzaG91bGQgYmUgdGhlIG5hbWVzIG9mIGFsbCBnZW5lcyBwcmVzZW50IGluIHRoZSBCYWNrZ3JvdW5kIGZpbGUuIGkuZS4gYWxsIGdlbmVzIHRoYXQgd2VyZSB0ZXN0ZWQuCjYuIENob29zZSBhbiBPbnRvbG9neTogYFByb2Nlc3NgCjcuIFVuZGVyIGFkdmFuY2VkIG9wdGlvbnMsIHRpY2sgKipPdXRwdXQgcmVzdWx0cyBpbiBNaWNyb3NvZnQgRXhjZWwgZm9ybWF0KioKOC4gYFNlYXJjaCBFbnJpY2hlZCBHTyB0ZXJtc2AKCllvdSBzaG91bGQgYmUgcHJlc2VudGVkIHdpdGggYSBncmFwaCBvZiBlbnJpY2hlZCBHTyB0ZXJtcyBzaG93aW5nIHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiB0aGUgdGVybXMuIEVhY2ggR08gdGVybSBpcyBjb2xvdXJlZCBhY2NvcmRpbmcgdG8gaXRzIHN0YXRpc3RpY2FsIHNpZ25pZmljYW5jZS4KCkJlbG93IHRoZSBmaWd1cmUgaXMgdGhlIHJlc3VsdHMgdGFibGUuIFRoaXMgbGlua3MgdG8gbW9yZSBpbmZvcm1hdGlvbiBhYm91dCBlYWNoIEdPIHRlcm0sIGFuZCBsaXN0cyBlYWNoIGdlbmUgaW4gdGhlIGNhdGVnb3J5IHRoYXQgd2FzIGZvdW5kIGluIHlvdXIgbGlzdC4gVGhlIGVucmljaG1lbnQgY29sdW1uIGdpdmVzIDQgbnVtYmVycyB0aGF0IGFyZSB1c2VkIHRvIGRldGVybWluZSBlbnJpY2htZW50IChzaW1pbGFyIHRvIHRoZSBGaXNoZXIgZXhhY3QgdGVzdCB3ZSBzYXcgZWFybGllcikKCi0gTiwgdG90YWwgbnVtYmVyIG9mIGdlbmVzIChzaG91bGQgYmUgdGhlIHNhbWUgaW4gYWxsIHJvd3MpCi0gQiwgdG90YWwgbnVtYmVyIG9mIGdlbmVzIGFubm90YXRlZCB3aXRoIHRoZSBHTyB0ZXJtCi0gbiwgdG90YWwgbnVtYmVyIG9mIGdlbmVzIHRoYXQgd2VyZSBmb3VuZCBpbiB0aGUgbGlzdCB5b3UgdXBsb2FkZWQgKHNhbWUgZm9yIGFsbCByb3dzKQotIGIsIG51bWJlciBvZiBnZW5lcyBpbiB0aGUgbGlzdCB5b3UgdXBsb2FkZWQgdGhhdCBpbnRlcnNlY3Qgd2l0aCB0aGlzIEdPIHRlcm0KCjxkaXYgY2xhc3M9ImV4ZXJjaXNlIj4KKipFeGVyY2lzZToqKiBSZXBlYXQgdGhlIEdPcmlsbGEgdG8gZmluZCBlbnJpY2hlZCBwYXRod2F5cyBpbiB0aGUgSFQ1NTogSVRSQUNPTkFaT0xFIHZzIERNU08gYW5hbHlzaXMuIFdoYXQgZG8geW91IG5vdGljZT8KPC9kaXY+CgoKIyMgVGhyZXNob2xkLWZyZWUgYW5hbHlzaXMKClRoaXMgdHlwZSBvZiBhbmFseXNpcyBpcyBwb3B1bGFyIGZvciBkYXRhc2V0cyB3aGVyZSBkaWZmZXJlbnRpYWwgZXhwcmVzc2lvbiBhbmFseXNpcyBkb2VzIG5vdCByZXZlYWwgbWFueSBnZW5lcyB0aGF0IGFyZSBkaWZmZXJlbnRpYWxseS1leHByZXNzZWQgb24gdGhlaXIgb3duLiBJbnN0ZWFkLCBpdCBzZWVrcyB0byBpZGVudGlmeSBnZW5lcyB0aGF0IGFzIGEgZ3JvdXAgaGF2ZSBhIHRlbmRlbmN5IHRvIGJlIG5lYXIgdGhlIGV4dHJlbWVzIG9mIHRoZSBsb2ctZm9sZCBjaGFuZ2VzLiBUaGUgcmVzdWx0cyBhcmUgdHlwaWNhbGx5IHByZXNlbnRlZCBpbiB0aGUgZm9sbG93aW5nIHdheS4KCiFbXShtZWRpYS9vdmVyZXhwcmVzc2VkLWdzZWEucG5nKQoKVGhlICJiYXJjb2RlIi1saWtlIHBhbmVsIHJlcHJlc2VudHMgd2hlcmUgZ2VuZXMgZnJvbSBhIHBhcnRpY3VsYXIgcGF0aHdheSAoKipIQUxMTUFSS19FMkZfVEFSR0VUUyoqIGluIHRoaXMgY2FzZSkgYXJlIGxvY2F0ZWQgaW4gYSBnZW5lIGxpc3QgKnJhbmtlZCogZnJvbSBtb3N0IHVwLXJlZ3VsYXRlZCB0byBtb3N0IGRvd24tcmVndWxhdGVkLiBUaGUgcGVhayBpbiB0aGUgZ3JlZW4gY3VydmUgaXMgdXNlZCB0byBpbmRpY2F0ZSB3aGVyZSB0aGUgbWFqb3JpdHkgb2YgZ2VuZXMgYXJlIGxvY2F0ZWQuIElmIHRoaXMgaXMgc2hpZnRlZCB0byB0aGUgbGVmdCBvciB0aGUgcmlnaHQgaXQgaW5kaWNhdGVzIHRoYXQgZ2VuZXMgYmVsb25naW5nIHRvIHRoaXMgZ2VuZSBzZXQgaGF2ZSBhIHRlbmRlbmN5IHRvIGJlIHVwLSBvciBkb3duLXJlZ3VsYXRlZC4KCkFzIHN1Y2gsIGl0IGRvZXMgbm90IHJlbHkgb24gaGF2aW5nIHRvIGltcG9zZSBhcmJpdHJhcnkgY3V0LW9mZnMgb24gdGhlIGRhdGEuIEluc3RlYWQsIHdlIG5lZWQgdG8gcHJvdmlkZSBhIG1lYXN1cmUgb2YgdGhlIGltcG9ydGFuY2Ugb2YgZWFjaCBnZW5lIHN1Y2ggYXMgaXQncyBmb2xkLWNoYW5nZS4gVGhlc2UgYXJlIHRoZW4gdXNlZCB0aGUgcmFuayB0aGUgZ2VuZXMuCgpUaGUgQnJvYWQgaW5zdGl0dXRlIGhhcyBtYWRlIHRoaXMgYW5hbHlzaXMgbWV0aG9kIHBvcHVsYXIgYW5kIHByb3ZpZGVzIFthIHZlcnNpb24gb2YgR1NFQV0oaHR0cDovL3NvZnR3YXJlLmJyb2FkaW5zdGl0dXRlLm9yZy9nc2VhL2luZGV4LmpzcCkgdGhhdCBjYW4gYmUgcnVuIHZpYSBhIGphdmEgYXBwbGljYXRpb24uIEhvd2V2ZXIsIHRoZSBhcHBsaWNhdGlvbiBjYW4gYmUgYSBiaXQgZmlkZGx5IHRvIHJ1biwgc28gd2Ugd2lsbCB1c2UgdGhlIEdlbmVUcmFpbCB3ZWJzaXRlIGluc3RlYWQKCi0gT3BlbiB0aGUgZmlsZSBgYmFja2dyb3VuZC5jc3ZgIGluIEV4Y2VsIGFuZCBkZWxldGUgYWxsIGNvbHVtbnMgZXhjZXB0IHRoZSBgU1lNQk9MYCBhbmQgYElUUkFgIGNvbHVtbi4KPGltZyBzcmM9Im1lZGlhL0dlbmVUcmFpbF9wcmVwLnBuZyIvPgotIEdvIHRvIHRoZSBHZW5lVHJhaWwgd2Vic2l0ZSwgYW5kIHNlbGVjdCBUcmFuc2NyaXB0b21pY3MgZnJvbSB0aGUgZnJvbnQgcGFnZQotIFNlbGVjdCB0aGUgKipQYXN0ZSB0aGUgY29udGVudCBvZiBhIHRleHQgZmlsZSBpbiBhIHRhYnVsYXIgZm9ybWF0IG9wdGlvbioqIGFuZCB0aGUgY29udGVudHMgb2YgeW91ciBtb2RpZmllZCBleGNlbCBmaWxlIGludG8gdGhlIGJveC4gKipEbyBub3QgcGFzdGUgdGhlIGNvbHVtbiBoZWFkaW5ncyoqCi0gQ2xpY2sgVXBsb2FkCgpIb3BlZnVsbHkgaXQgc2hvdWxkIHJlY29nbmlzZSB5b3VyIGlucHV0IHdpdGhvdXQgYW55IGVycm9ycywgYW5kIG9uIHRoZSBuZXh0IHNjcmVlbiB0aGUgKipTZXQtbGV2ZWwgc3RhdGlzdGljKiogc2hvdWxkIGJlIGF1dG9tYXRpY2FsbHkgc2V0IHRvICoqR1NFQSoqCgo8ZGl2IGNsYXNzPSJ3YXJuaW5nIj4KSWYgeW91ciBkYXRhIGRvZXMgbm90IGdldCB1cGxvYWRlZCwgZG91YmxlLWNoZWNrIHRoYXQgdGhlIGNvbHVtbiBoZWFkaW5nICoqSVRSQSoqIGhhcyBub3QgYmVlbiBwYXN0ZWQgaW50byB0aGUgdGV4dCBib3gKPC9kaXY+CgpUbyBtYWtlIHRoZSBhbmFseXNpcyBydW4gZmFzdGVyLCB5b3UgY2FuIGRlLXNlbGVjdCB0aGUgR08gcGF0aHdheXMgKGJpb2xvZ2ljYWwgcHJvY2Vzc2VzLCBtb2xlY3VsYXIgZnVuY3Rpb24gYW5kIGNlbGx1bGFyIGNvbXBhcnRtZW50KQoKKipGb3IgeW91ciBvd24gYW5hbHlzZXMsIHlvdSBzaG91bGQgY29uc2lkZXIgYW5hbHlzaW5nIHRoZXNlIGNhdGVnb3JpZXMuIFdlIGFyZSBvbmx5IGRlLXNlbGVjdGluZyB0aGVtIGhlcmUgdG8gbWFrZSB0aGluZ3MgcnVuIHF1aWNrZXIqKgoKPGltZyBzcmM9Im1lZGlhL2dlbmV0cmFpbF9zZXR1cC5QTkciLz4KCkFmdGVyIGEgc2hvcnQgd2FpdCwgeW91IHdpbGwgYmUgYWJsZSB0byB2aWV3IGFuZCBkb3dubG9hZCB0aGUgcmVzdWx0cy4gVGhlIHRlc3RlZCBwYXRod2F5cyBhcmUgZ3JvdXBlZCBpbnRvIGRpZmZlcmVudCBzb3VyY2VzIChLZWdnLCBSZWFjdG9tZSBvciBXaWtpcGF0aHdheXMpIAoKPGltZyBzcmM9Im1lZGlhL2dlbmV0cmFpbF9LRUdHLlBORyIvPgoKRWFjaCBvZiB0aGUgc2lnbmlmaWNhbnQgcGF0aHdheXMgY2FuIGJlIGV4cGxvcmVkIGluIGRldGFpbDsgc3VjaCBhcyBzaG93aW5nIHdoaWNoIGdlbmVzIGluIHRoYXQgcGF0aHdheXMgYXJlIHVwLSBvciBkb3ducmVndWxhdGVkLgoKPGltZyBzcmM9Im1lZGlhL2dlbmV0cmFpbF9LRUdHX3Jlc3VsdC5QTkciLz4KClRoZSBSYW5rIG9mIHRoZSBnZW5lIHNob3duIGlzIHRoZSBwb3NpdGlvbiBvZiB0aGUgZ2VuZSBpbiB0aGUgcmFua2VkIGxpc3Q7IHdpdGggMSBiZWluZyBtb3N0IHVwLXJlZ3VsYXRlZCBnZW5lLiBUaGUgc2NvcmUgaXMgdGhlIHNjb3JlIHVzZWQgdG8gcmFuayB0aGUgZ2VuZXMgKGZvbGQtY2hhbmdlIGluIG91ciBleGFtcGxlKS4KCjxkaXYgY2xhc3M9ImV4ZXJjaXNlIj4KKipFeGVyY2lzZSoqOiBFeHBsb3JlIHRoZSBHZW5lVHJhaWwgcmVzdWx0cy4gRG9lcyB0aGUgbWV0aG9kIGlkZW50aWZ5IHNpZ25pZmljYW50IHBhdGh3YXlzIGZvciB0aGUgSFQ1NTogSVRSQUNPTkFaT0xFIHZzIERNU08gYW5hbHlzaXM/CjwvZGl2PgoKIyBSZWNhcAoK