Reference Genomes
In this module, we will learn:
- what a reference genome is and what it contains
- details about the FASTA and GTF formats
- to appreciate the differences in gene identifiers
- how to download a reference genome
Differential Expression Workflow
Here we will set the stage for the next steps by discussing reference genomes, which are integral to genome alignments and gene/isoform quantification. Along the way we will touch on some things to be aware of.
Reference Genomes
A reference genome consists of the reference sequence and, optionally, any number of genomic annotations that describe attributes about that sequence. Examples of annotations include:
- Gene models consisting of the location and other information about genes.
- Variants consisting of the location of common or rare genetic variants, their alleles, and frequencies.
- Small RNAs consisting of the location and other information about various types of small RNAs.
Of particular relevance to us for this workshop are the reference sequence and gene models.
Reference Sequence
Reference sequence is stored in FASTA files. They are similar to FASTQ files in their storage of sequence information, but their format is a little different in a couple ways:
- Records are separated by lines beginning with
>
instead of @
.
- Only the sequence is stored in a FASTA file, there is no notion of quality attached to the nucleotides.
>chrM
GATCACAGGTCTATCACCCTATTAACCACTCACGGGAGCTCTCCATGCAT
TTGGTATTTTCGTCTGGGGGGTGTGCACGCGATAGCATTGCGAGACGCTG
GAGCCGGAGCACCCTATGTCGCAGTATCTGTCTTTGATTCCTGCCTCATT
CTATTATTTATCGCACCTACGTTCAATATTACAGGCGAACATACCTACTA
AAGTGTGTTAATTAATTAATGCTTGTAGGACATAATAATAACAATTGAAT
GTCTGCACAGCCGCTTTCCACACAGACATCATAACAAAAAATTTCCACCA
AACCCCCCCCTCCCCCCGCTTCTGGCCACAGCACTTAAACACATCTCTGC
CAAACCCCAAAAACAAAGAACCCTAACACCAGCCTAACCAGATTTCAAAT
TTTATCTTTAGGCGGTATGCACTTTTAACAGTCACCCCCCAACTAACACA
Gene Models
Well-characterized organisms (e.g. human, mouse, zebrafish) have fairly mature gene models. These are stored in GTF format, which gives location and other information about each gene feature. Below are two examples:
chr1 unknown exon 11874 12227 . + . gene_id "DDX11L1"; gene_name "DDX11L1"; transcript_id "NR_046018"; tss_id "TSS16932";
chr1 unknown exon 12613 12721 . + . gene_id "DDX11L1"; gene_name "DDX11L1"; transcript_id "NR_046018"; tss_id "TSS16932";
chr1 unknown exon 13221 14409 . + . gene_id "DDX11L1"; gene_name "DDX11L1"; transcript_id "NR_046018"; tss_id "TSS16932";
chr1 unknown exon 14362 14829 . - . gene_id "WASH7P"; gene_name "WASH7P"; transcript_id "NR_024540"; tss_id "TSS8568";
1 havana gene 11869 14409 . + . gene_id "ENSG00000223972"; gene_version "5"; gene_name "DDX11L1"; gene_source "havana"; gene_biotype "transcribed_unprocessed_pseudogene";
1 havana transcript 11869 14409 . + . gene_id "ENSG00000223972"; gene_version "5"; transcript_id "ENST00000456328"; transcript_version "2"; gene_name "DDX11L1"; gene_source "havana"; gene_biotype "transcribed_unprocessed_pseudogene"; transcript_name "DDX11L1-202"; transcript_source "havana"; transcript_biotype "lncRNA"; tag "basic"; transcript_support_level "1";
1 havana exon 11869 12227 . + . gene_id "ENSG00000223972"; gene_version "5"; transcript_id "ENST00000456328"; transcript_version "2"; exon_number "1"; gene_name "DDX11L1"; gene_source "havana"; gene_biotype "transcribed_unprocessed_pseudogene"; transcript_name "DDX11L1-202"; transcript_source "havana"; transcript_biotype "lncRNA"; exon_id "ENSE00002234944"; exon_version "1"; tag "basic"; transcript_support_level "1";
The GTF format stores specific information in each column:
1 |
Chromosome |
2 |
Source, e.g. ensembl, havana |
3 |
Gene feature, e.g. exon, intron, mRNA, transcript |
4 |
Start location, 1-based |
5 |
End location, 1-based |
6 |
Score |
7 |
Strand |
8 |
Frame, relating to codons |
9 |
Attribute, a semicolon separated list of key/value pairs giving additional information about the feature. |
Minutiae, Very Briefly
Bioinformatics is a relatively new, fast-changing, field and its data standards and formats are no different. Consequently there are some oddities and tedious items of note which we would like to only briefly touch on here.
Genome Builds
On occassion new reference genomes are released, and the genome build number changes. You may be familiar with the UCSC manner of naming human genome builds: hg18, hg19, hg38. ENSEMBL, naturally, has their own way of referring to genome builds: GRCh36, GRCh37, and GRCh38. Notice with the most recent human reference, the numbering now aligns between UCSC and ENSEMBL.
Different organisms have their own versioning.
Gene IDs
The two GTF examples above highlight different ways of referring to the same gene. In the first GTF we see:
And in the second GTF we see:
- ENSG00000223972, the ENSEMBL gene ID
- DDX11L1, the gene symbol, thankfully the same
- ENST00000456328, the ENSEMBL transcript ID
Translating between different gene IDs is possible, as we will see in Day Two with biomaRt
. But in terms of best practice it is generally a good idea to avoid using the gene symbol as the primary gene identifier because not everyone refers to the same gene by the same symbol.
Getting a Reference Genome
The Illumina iGenomes resource is one of the easiest, and most comprehensive, ways to download a reference genome. iGenomes includes both the reference sequence and gene models.
Reference genomes can be very large, depending on the organism, and so we will not download one to the Amazon instance we are using for this workshop. We’ve included instructions for downloading these, in case you want to download these to the server where you intend to later do a similar RNA-seq analysis (e.g. on High-Performance Compute, GreatLakes).
How would I download references with iGenomes?
As noted, it’s not recommended to download the iGenomes references to the AWS instance. However, if you wanted to know in general how you would do that, the process is described here.
First go to the iGenomes page, find the build you want from the source you want, right click the genome build you want to download, and select “Copy link location”:
Then on the remote server you would go to the directory you’d like to download the genome to and type (that URL is what we copied):
$ wget http://igenomes.illumina.com.s3-website-us-east-1.amazonaws.com/Homo_sapiens/NCBI/GRCh38/Homo_sapiens_NCBI_GRCh38.tar.gz
After the download finishes (it may take a while as it is tens of GB large), you can unpack it with:
$ tar -xf Homo_sapiens_NCBI_GRCh38.tar.gz
Which Reference is Right for Me?
The key is to be consistent in your research. Switching from ENSEMBL to UCSC will create many headaches because of the change in gene identifiers, and differences in the gene models themselves. Often people choose the one they’re most comfortable with, which is often a function of historical accident. The key is not to overthink it.
Another important note is not to mix the sources. If you download reference sequence from UCSC, don’t use an ENSEMBL GTF (and vice versa). One of the quirky differences between the two databases is that ENSEMBL refers to chromosome only by their number, i.e. 1
, whereas UCSC refers to chromsomes as chr1
. This makes reference FASTAs from one source incompatible with gene builds from another.
These materials have been adapted and extended from materials created by the Harvard Chan Bioinformatics Core (HBC). These are open access materials distributed under the terms of the Creative Commons Attribution license (CC BY 4.0), which permits unrestricted use, distribution, and reproduction in any medium, provided the original author and source are credited.
LS0tCnRpdGxlOiAiRGF5IDEgLSBNb2R1bGUgMDM6IFJlZmVyZW5jZSBHZW5vbWVzIgphdXRob3I6ICJVTSBCaW9pbmZvcm1hdGljcyBDb3JlIgpvdXRwdXQ6CiAgICAgICAgaHRtbF9kb2N1bWVudDoKICAgICAgICAgICAgaW5jbHVkZXM6CiAgICAgICAgICAgICAgICBpbl9oZWFkZXI6IGhlYWRlci5odG1sCiAgICAgICAgICAgIHRoZW1lOiBwYXBlcgogICAgICAgICAgICB0b2M6IHRydWUKICAgICAgICAgICAgdG9jX2RlcHRoOiA0CiAgICAgICAgICAgIHRvY19mbG9hdDogdHJ1ZQogICAgICAgICAgICBudW1iZXJfc2VjdGlvbnM6IHRydWUKICAgICAgICAgICAgZmlnX2NhcHRpb246IHRydWUKICAgICAgICAgICAgbWFya2Rvd246IEdGTQogICAgICAgICAgICBjb2RlX2Rvd25sb2FkOiB0cnVlCi0tLQo8c3R5bGUgdHlwZT0idGV4dC9jc3MiPgpib2R5eyAvKiBOb3JtYWwgICovCiAgICAgIGZvbnQtc2l6ZTogMTRwdDsKICB9CnByZSB7CiAgZm9udC1zaXplOiAxMnB0Cn0KPC9zdHlsZT4KCiMgUmVmZXJlbmNlIEdlbm9tZXMKCkluIHRoaXMgbW9kdWxlLCB3ZSB3aWxsIGxlYXJuOgoKKiB3aGF0IGEgcmVmZXJlbmNlIGdlbm9tZSBpcyBhbmQgd2hhdCBpdCBjb250YWlucwoqIGRldGFpbHMgYWJvdXQgdGhlIEZBU1RBIGFuZCBHVEYgZm9ybWF0cwoqIHRvIGFwcHJlY2lhdGUgdGhlIGRpZmZlcmVuY2VzIGluIGdlbmUgaWRlbnRpZmllcnMKKiBob3cgdG8gZG93bmxvYWQgYSByZWZlcmVuY2UgZ2Vub21lCgojIERpZmZlcmVudGlhbCBFeHByZXNzaW9uIFdvcmtmbG93CgpIZXJlIHdlIHdpbGwgc2V0IHRoZSBzdGFnZSBmb3IgdGhlIG5leHQgc3RlcHMgYnkgZGlzY3Vzc2luZyByZWZlcmVuY2UgZ2Vub21lcywgd2hpY2ggYXJlIGludGVncmFsIHRvIGdlbm9tZSBhbGlnbm1lbnRzIGFuZCBnZW5lL2lzb2Zvcm0gcXVhbnRpZmljYXRpb24uIEFsb25nIHRoZSB3YXkgd2Ugd2lsbCB0b3VjaCBvbiBzb21lIHRoaW5ncyB0byBiZSBhd2FyZSBvZi4KCiFbXShpbWFnZXMvd2F5ZmluZGVyL3dheWZpbmRlci0wMi5wbmcpCjxicj4KPGJyPgo8YnI+Cjxicj4KCiMgUmVmZXJlbmNlIEdlbm9tZXMKCkEgcmVmZXJlbmNlIGdlbm9tZSBjb25zaXN0cyBvZiB0aGUgKipyZWZlcmVuY2Ugc2VxdWVuY2UqKiBhbmQsIG9wdGlvbmFsbHksIGFueSBudW1iZXIgb2YgKipnZW5vbWljIGFubm90YXRpb25zKiogdGhhdCBkZXNjcmliZSBhdHRyaWJ1dGVzIGFib3V0IHRoYXQgc2VxdWVuY2UuIEV4YW1wbGVzIG9mIGFubm90YXRpb25zIGluY2x1ZGU6CgoqIEdlbmUgbW9kZWxzIGNvbnNpc3Rpbmcgb2YgdGhlIGxvY2F0aW9uIGFuZCBvdGhlciBpbmZvcm1hdGlvbiBhYm91dCBnZW5lcy4KKiBWYXJpYW50cyBjb25zaXN0aW5nIG9mIHRoZSBsb2NhdGlvbiBvZiBjb21tb24gb3IgcmFyZSBnZW5ldGljIHZhcmlhbnRzLCB0aGVpciBhbGxlbGVzLCBhbmQgZnJlcXVlbmNpZXMuCiogU21hbGwgUk5BcyBjb25zaXN0aW5nIG9mIHRoZSBsb2NhdGlvbiBhbmQgb3RoZXIgaW5mb3JtYXRpb24gYWJvdXQgdmFyaW91cyB0eXBlcyBvZiBzbWFsbCBSTkFzLgoKT2YgcGFydGljdWxhciByZWxldmFuY2UgdG8gdXMgZm9yIHRoaXMgd29ya3Nob3AgYXJlIHRoZSByZWZlcmVuY2Ugc2VxdWVuY2UgYW5kIGdlbmUgbW9kZWxzLgoKIyMgUmVmZXJlbmNlIFNlcXVlbmNlCgpSZWZlcmVuY2Ugc2VxdWVuY2UgaXMgc3RvcmVkIGluIFtGQVNUQV0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvRkFTVEFfZm9ybWF0KSBmaWxlcy4gVGhleSBhcmUgc2ltaWxhciB0byBGQVNUUSBmaWxlcyBpbiB0aGVpciBzdG9yYWdlIG9mIHNlcXVlbmNlIGluZm9ybWF0aW9uLCBidXQgdGhlaXIgZm9ybWF0IGlzIGEgbGl0dGxlIGRpZmZlcmVudCBpbiBhIGNvdXBsZSB3YXlzOgoKMS4gUmVjb3JkcyBhcmUgc2VwYXJhdGVkIGJ5IGxpbmVzIGJlZ2lubmluZyB3aXRoIGA+YCBpbnN0ZWFkIG9mIGBAYC4KMi4gT25seSB0aGUgc2VxdWVuY2UgaXMgc3RvcmVkIGluIGEgRkFTVEEgZmlsZSwgdGhlcmUgaXMgbm8gbm90aW9uIG9mIHF1YWxpdHkgYXR0YWNoZWQgdG8gdGhlIG51Y2xlb3RpZGVzLgoKYGBgCj5jaHJNCkdBVENBQ0FHR1RDVEFUQ0FDQ0NUQVRUQUFDQ0FDVENBQ0dHR0FHQ1RDVENDQVRHQ0FUClRUR0dUQVRUVFRDR1RDVEdHR0dHR1RHVEdDQUNHQ0dBVEFHQ0FUVEdDR0FHQUNHQ1RHCkdBR0NDR0dBR0NBQ0NDVEFUR1RDR0NBR1RBVENUR1RDVFRUR0FUVENDVEdDQ1RDQVRUCkNUQVRUQVRUVEFUQ0dDQUNDVEFDR1RUQ0FBVEFUVEFDQUdHQ0dBQUNBVEFDQ1RBQ1RBCkFBR1RHVEdUVEFBVFRBQVRUQUFUR0NUVEdUQUdHQUNBVEFBVEFBVEFBQ0FBVFRHQUFUCkdUQ1RHQ0FDQUdDQ0dDVFRUQ0NBQ0FDQUdBQ0FUQ0FUQUFDQUFBQUFBVFRUQ0NBQ0NBCkFBQ0NDQ0NDQ0NUQ0NDQ0NDR0NUVENUR0dDQ0FDQUdDQUNUVEFBQUNBQ0FUQ1RDVEdDCkNBQUFDQ0NDQUFBQUFDQUFBR0FBQ0NDVEFBQ0FDQ0FHQ0NUQUFDQ0FHQVRUVENBQUFUClRUVEFUQ1RUVEFHR0NHR1RBVEdDQUNUVFRUQUFDQUdUQ0FDQ0NDQ0NBQUNUQUFDQUNBCmBgYAoKIyMgR2VuZSBNb2RlbHMKCldlbGwtY2hhcmFjdGVyaXplZCBvcmdhbmlzbXMgKGUuZy4gaHVtYW4sIG1vdXNlLCB6ZWJyYWZpc2gpIGhhdmUgZmFpcmx5IG1hdHVyZSBnZW5lIG1vZGVscy4gVGhlc2UgYXJlIHN0b3JlZCBpbiBbR1RGXShodHRwczovL3Vzd2VzdC5lbnNlbWJsLm9yZy9pbmZvL3dlYnNpdGUvdXBsb2FkL2dmZi5odG1sKSBmb3JtYXQsIHdoaWNoIGdpdmVzIGxvY2F0aW9uIGFuZCBvdGhlciBpbmZvcm1hdGlvbiBhYm91dCBlYWNoIGdlbmUgZmVhdHVyZS4gQmVsb3cgYXJlIHR3byBleGFtcGxlczoKCgogICAgY2hyMQl1bmtub3duCWV4b24JMTE4NzQJMTIyMjcJLgkrCS4JZ2VuZV9pZCAiRERYMTFMMSI7IGdlbmVfbmFtZSAiRERYMTFMMSI7IHRyYW5zY3JpcHRfaWQgIk5SXzA0NjAxOCI7IHRzc19pZCAiVFNTMTY5MzIiOwogICAgY2hyMQl1bmtub3duCWV4b24JMTI2MTMJMTI3MjEJLgkrCS4JZ2VuZV9pZCAiRERYMTFMMSI7IGdlbmVfbmFtZSAiRERYMTFMMSI7IHRyYW5zY3JpcHRfaWQgIk5SXzA0NjAxOCI7IHRzc19pZCAiVFNTMTY5MzIiOwogICAgY2hyMQl1bmtub3duCWV4b24JMTMyMjEJMTQ0MDkJLgkrCS4JZ2VuZV9pZCAiRERYMTFMMSI7IGdlbmVfbmFtZSAiRERYMTFMMSI7IHRyYW5zY3JpcHRfaWQgIk5SXzA0NjAxOCI7IHRzc19pZCAiVFNTMTY5MzIiOwogICAgY2hyMQl1bmtub3duCWV4b24JMTQzNjIJMTQ4MjkJLgktCS4JZ2VuZV9pZCAiV0FTSDdQIjsgZ2VuZV9uYW1lICJXQVNIN1AiOyB0cmFuc2NyaXB0X2lkICJOUl8wMjQ1NDAiOyB0c3NfaWQgIlRTUzg1NjgiOwoKCiAgICAxCWhhdmFuYQlnZW5lCTExODY5CTE0NDA5CS4JKwkuCWdlbmVfaWQgIkVOU0cwMDAwMDIyMzk3MiI7IGdlbmVfdmVyc2lvbiAiNSI7IGdlbmVfbmFtZSAiRERYMTFMMSI7IGdlbmVfc291cmNlICJoYXZhbmEiOyBnZW5lX2Jpb3R5cGUgInRyYW5zY3JpYmVkX3VucHJvY2Vzc2VkX3BzZXVkb2dlbmUiOwogICAgMQloYXZhbmEJdHJhbnNjcmlwdAkxMTg2OQkxNDQwOQkuCSsJLglnZW5lX2lkICJFTlNHMDAwMDAyMjM5NzIiOyBnZW5lX3ZlcnNpb24gIjUiOyB0cmFuc2NyaXB0X2lkICJFTlNUMDAwMDA0NTYzMjgiOyB0cmFuc2NyaXB0X3ZlcnNpb24gIjIiOyBnZW5lX25hbWUgIkREWDExTDEiOyBnZW5lX3NvdXJjZSAiaGF2YW5hIjsgZ2VuZV9iaW90eXBlICJ0cmFuc2NyaWJlZF91bnByb2Nlc3NlZF9wc2V1ZG9nZW5lIjsgdHJhbnNjcmlwdF9uYW1lICJERFgxMUwxLTIwMiI7IHRyYW5zY3JpcHRfc291cmNlICJoYXZhbmEiOyB0cmFuc2NyaXB0X2Jpb3R5cGUgImxuY1JOQSI7IHRhZyAiYmFzaWMiOyB0cmFuc2NyaXB0X3N1cHBvcnRfbGV2ZWwgIjEiOwogICAgMQloYXZhbmEJZXhvbgkxMTg2OQkxMjIyNwkuCSsJLglnZW5lX2lkICJFTlNHMDAwMDAyMjM5NzIiOyBnZW5lX3ZlcnNpb24gIjUiOyB0cmFuc2NyaXB0X2lkICJFTlNUMDAwMDA0NTYzMjgiOyB0cmFuc2NyaXB0X3ZlcnNpb24gIjIiOyBleG9uX251bWJlciAiMSI7IGdlbmVfbmFtZSAiRERYMTFMMSI7IGdlbmVfc291cmNlICJoYXZhbmEiOyBnZW5lX2Jpb3R5cGUgInRyYW5zY3JpYmVkX3VucHJvY2Vzc2VkX3BzZXVkb2dlbmUiOyB0cmFuc2NyaXB0X25hbWUgIkREWDExTDEtMjAyIjsgdHJhbnNjcmlwdF9zb3VyY2UgImhhdmFuYSI7IHRyYW5zY3JpcHRfYmlvdHlwZSAibG5jUk5BIjsgZXhvbl9pZCAiRU5TRTAwMDAyMjM0OTQ0IjsgZXhvbl92ZXJzaW9uICIxIjsgdGFnICJiYXNpYyI7IHRyYW5zY3JpcHRfc3VwcG9ydF9sZXZlbCAiMSI7CgoKVGhlIEdURiBmb3JtYXQgc3RvcmVzIHNwZWNpZmljIGluZm9ybWF0aW9uIGluIGVhY2ggY29sdW1uOgoKfCBDb2x1bW4gfCBEZXNjcmlwdGlvbiB8CnwgOi0tLS06IHwgLS0tLS0tLS0tLS0gfAp8IDEgfCBDaHJvbW9zb21lIHwKfCAyIHwgU291cmNlLCBlLmcuIGVuc2VtYmwsIGhhdmFuYSB8CnwgMyB8IEdlbmUgZmVhdHVyZSwgZS5nLiBleG9uLCBpbnRyb24sIG1STkEsIHRyYW5zY3JpcHQgfAp8IDQgfCBTdGFydCBsb2NhdGlvbiwgMS1iYXNlZCB8CnwgNSB8IEVuZCBsb2NhdGlvbiwgMS1iYXNlZCB8CnwgNiB8IFNjb3JlIHwKfCA3IHwgU3RyYW5kIHwKfCA4IHwgRnJhbWUsIHJlbGF0aW5nIHRvIGNvZG9ucyB8CnwgOSB8IEF0dHJpYnV0ZSwgYSBzZW1pY29sb24gc2VwYXJhdGVkIGxpc3Qgb2Yga2V5L3ZhbHVlIHBhaXJzIGdpdmluZyBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIGFib3V0IHRoZSBmZWF0dXJlLiB8CgojIyBNaW51dGlhZSwgVmVyeSBCcmllZmx5CgpCaW9pbmZvcm1hdGljcyBpcyBhIHJlbGF0aXZlbHkgbmV3LCBmYXN0LWNoYW5naW5nLCBmaWVsZCBhbmQgaXRzIGRhdGEgc3RhbmRhcmRzIGFuZCBmb3JtYXRzIGFyZSBubyBkaWZmZXJlbnQuIENvbnNlcXVlbnRseSB0aGVyZSBhcmUgc29tZSBvZGRpdGllcyBhbmQgdGVkaW91cyBpdGVtcyBvZiBub3RlIHdoaWNoIHdlIHdvdWxkIGxpa2UgdG8gb25seSBicmllZmx5IHRvdWNoIG9uIGhlcmUuCgojIyMgR2Vub21lIEJ1aWxkcwoKT24gb2NjYXNzaW9uIG5ldyByZWZlcmVuY2UgZ2Vub21lcyBhcmUgcmVsZWFzZWQsIGFuZCB0aGUgZ2Vub21lIGJ1aWxkIG51bWJlciBjaGFuZ2VzLiBZb3UgbWF5IGJlIGZhbWlsaWFyIHdpdGggdGhlIFVDU0MgbWFubmVyIG9mIG5hbWluZyBodW1hbiBnZW5vbWUgYnVpbGRzOiBoZzE4LCBoZzE5LCBoZzM4LiBFTlNFTUJMLCBuYXR1cmFsbHksIGhhcyB0aGVpciBvd24gd2F5IG9mIHJlZmVycmluZyB0byBnZW5vbWUgYnVpbGRzOiBHUkNoMzYsIEdSQ2gzNywgYW5kIEdSQ2gzOC4gTm90aWNlIHdpdGggdGhlIG1vc3QgcmVjZW50IGh1bWFuIHJlZmVyZW5jZSwgdGhlIG51bWJlcmluZyBub3cgYWxpZ25zIGJldHdlZW4gVUNTQyBhbmQgRU5TRU1CTC4KCkRpZmZlcmVudCBvcmdhbmlzbXMgaGF2ZSB0aGVpciBvd24gdmVyc2lvbmluZy4KCiMjIyBBbm5vdGF0aW9uIFNvdXJjZXMKCltOQ0JJIFJlZlNlcV0oaHR0cHM6Ly93d3cubmNiaS5ubG0ubmloLmdvdi9yZWZzZXEvcnNnLyksIFtFTlNFTUJMXShodHRwczovL3d3dy5lbnNlbWJsLm9yZy9pbmZvL2dlbm9tZS9nZW5lYnVpbGQvaW5kZXguaHRtbCksIGFuZCBbVUNTQyBLbm93biBHZW5lc10oaHR0cHM6Ly9hY2FkZW1pYy5vdXAuY29tL2Jpb2luZm9ybWF0aWNzL2FydGljbGUvMjIvOS8xMDM2LzIwMDA5MykgYXJlIHRoZSB0aHJlZSBwcmltYXJ5IGdlbmUgYW5ub3RhdGlvbiBkYXRhYmFzZXMgKGRpZmZlcmVudCBvcmdhbmlzbXMgaGF2ZSB0aGVpciBvd24gZGF0YWJhc2VzKS4gV2Ugd2lsbCBub3QgZ28gaW50byBleGFjdGx5IGhvdyB0aGUgZ2VuZSBhbm5vdGF0aW9ucyBhcmUgZGlmZmVyZW50LCBidXQgd2Ugbm90ZSB0aGF0IHRoZSBhcmUsIGFuZCBbb3RoZXJzIGhhdmUgZXhhbWluZWQgdGhlIGNvbnNlcXVlbmNlcyBvZiB0aGlzXShodHRwczovL2JtY2dlbm9taWNzLmJpb21lZGNlbnRyYWwuY29tL2FydGljbGVzLzEwLjExODYvczEyODY0LTAxNS0xMzA4LTgpLgoKIyMjIEdlbmUgSURzCgpUaGUgdHdvIEdURiBleGFtcGxlcyBhYm92ZSBoaWdobGlnaHQgZGlmZmVyZW50IHdheXMgb2YgcmVmZXJyaW5nIHRvIHRoZSBzYW1lIGdlbmUuIEluIHRoZSBmaXJzdCBHVEYgd2Ugc2VlOgoKKiBERFgxMUwxLCB0aGUgZ2VuZSBzeW1ib2wsIGNvbnRyb2xsZWQgYnkgdGhlIFtIdW1hbiBHZW5lIE5vbWVuY2xhdHVyZSBDb21taXR0ZWUgKEhVR08pXShodHRwczovL3d3dy5nZW5lbmFtZXMub3JnLykuCiogTlJfMDQ2MDE4LCB0aGUgUmVmU2VxIHRyYW5zY3JpcHQgSUQKCkFuZCBpbiB0aGUgc2Vjb25kIEdURiB3ZSBzZWU6CgoqIEVOU0cwMDAwMDIyMzk3MiwgdGhlIEVOU0VNQkwgZ2VuZSBJRAoqIEREWDExTDEsIHRoZSBnZW5lIHN5bWJvbCwgdGhhbmtmdWxseSB0aGUgc2FtZQoqIEVOU1QwMDAwMDQ1NjMyOCwgdGhlIEVOU0VNQkwgdHJhbnNjcmlwdCBJRAoKVHJhbnNsYXRpbmcgYmV0d2VlbiBkaWZmZXJlbnQgZ2VuZSBJRHMgaXMgcG9zc2libGUsIGFzIHdlIHdpbGwgc2VlIGluIERheSBUd28gd2l0aCBgYmlvbWFSdGAuIEJ1dCBpbiB0ZXJtcyBvZiAqKmJlc3QgcHJhY3RpY2UqKiBpdCBpcyBnZW5lcmFsbHkgYSBnb29kIGlkZWEgdG8gYXZvaWQgdXNpbmcgdGhlIGdlbmUgc3ltYm9sIGFzIHRoZSBwcmltYXJ5IGdlbmUgaWRlbnRpZmllciBiZWNhdXNlIG5vdCBldmVyeW9uZSByZWZlcnMgdG8gdGhlIHNhbWUgZ2VuZSBieSB0aGUgc2FtZSBzeW1ib2wuCgojIEdldHRpbmcgYSBSZWZlcmVuY2UgR2Vub21lCgpUaGUgW0lsbHVtaW5hIGlHZW5vbWVzXShodHRwczovL3N1cHBvcnQuaWxsdW1pbmEuY29tL3NlcXVlbmNpbmcvc2VxdWVuY2luZ19zb2Z0d2FyZS9pZ2Vub21lLmh0bWwpIHJlc291cmNlIGlzIG9uZSBvZiB0aGUgZWFzaWVzdCwgYW5kIG1vc3QgY29tcHJlaGVuc2l2ZSwgd2F5cyB0byBkb3dubG9hZCBhIHJlZmVyZW5jZSBnZW5vbWUuIGlHZW5vbWVzIGluY2x1ZGVzIGJvdGggdGhlIHJlZmVyZW5jZSBzZXF1ZW5jZSBhbmQgZ2VuZSBtb2RlbHMuCgpSZWZlcmVuY2UgZ2Vub21lcyBjYW4gYmUgKip2ZXJ5IGxhcmdlKiosIGRlcGVuZGluZyBvbiB0aGUgb3JnYW5pc20sIGFuZCBzbyB3ZSB3aWxsIG5vdCBkb3dubG9hZCBvbmUgdG8gdGhlIEFtYXpvbiBpbnN0YW5jZSB3ZSBhcmUgdXNpbmcgZm9yIHRoaXMgd29ya3Nob3AuIFdlJ3ZlIGluY2x1ZGVkIGluc3RydWN0aW9ucyBmb3IgZG93bmxvYWRpbmcgdGhlc2UsIGluIGNhc2UgeW91IHdhbnQgdG8gZG93bmxvYWQgdGhlc2UgdG8gdGhlIHNlcnZlciB3aGVyZSB5b3UgaW50ZW5kIHRvIGxhdGVyIGRvIGEgc2ltaWxhciBSTkEtc2VxIGFuYWx5c2lzIChlLmcuIG9uIEhpZ2gtUGVyZm9ybWFuY2UgQ29tcHV0ZSwgR3JlYXRMYWtlcykuCgo8ZGV0YWlscz4KPHN1bW1hcnk+SG93IHdvdWxkIEkgZG93bmxvYWQgcmVmZXJlbmNlcyB3aXRoIGlHZW5vbWVzPzwvc3VtbWFyeT4KCkFzIG5vdGVkLCBpdCdzIG5vdCByZWNvbW1lbmRlZCB0byBkb3dubG9hZCB0aGUgaUdlbm9tZXMgcmVmZXJlbmNlcyB0byB0aGUgQVdTIGluc3RhbmNlLiBIb3dldmVyLCBpZiB5b3Ugd2FudGVkIHRvIGtub3cgaW4gZ2VuZXJhbCBob3cgeW91IHdvdWxkIGRvIHRoYXQsIHRoZSBwcm9jZXNzIGlzIGRlc2NyaWJlZCBoZXJlLgoKRmlyc3QgZ28gdG8gdGhlIFtpR2Vub21lc10oaHR0cHM6Ly9zdXBwb3J0LmlsbHVtaW5hLmNvbS9zZXF1ZW5jaW5nL3NlcXVlbmNpbmdfc29mdHdhcmUvaWdlbm9tZS5odG1sKSBwYWdlLCBmaW5kIHRoZSBidWlsZCB5b3Ugd2FudCBmcm9tIHRoZSBzb3VyY2UgeW91IHdhbnQsIHJpZ2h0IGNsaWNrIHRoZSBnZW5vbWUgYnVpbGQgeW91IHdhbnQgdG8gZG93bmxvYWQsIGFuZCBzZWxlY3QgIkNvcHkgbGluayBsb2NhdGlvbiI6CgohW2lHZW5vbWVzIGltYWdlIGZvciBjb3B5aW5nIGxpbmsgbG9jYXRpb25dKGltYWdlcy9nZW5vbWVfY29weV9saW5rLnBuZykKClRoZW4gb24gdGhlIHJlbW90ZSBzZXJ2ZXIgeW91IHdvdWxkIGdvIHRvIHRoZSBkaXJlY3RvcnkgeW91J2QgbGlrZSB0byBkb3dubG9hZCB0aGUgZ2Vub21lIHRvIGFuZCB0eXBlICh0aGF0IFVSTCBpcyB3aGF0IHdlIGNvcGllZCk6CgpgYGAKJCB3Z2V0IGh0dHA6Ly9pZ2Vub21lcy5pbGx1bWluYS5jb20uczMtd2Vic2l0ZS11cy1lYXN0LTEuYW1hem9uYXdzLmNvbS9Ib21vX3NhcGllbnMvTkNCSS9HUkNoMzgvSG9tb19zYXBpZW5zX05DQklfR1JDaDM4LnRhci5negpgYGAKCkFmdGVyIHRoZSBkb3dubG9hZCBmaW5pc2hlcyAoaXQgbWF5IHRha2UgYSB3aGlsZSBhcyBpdCBpcyB0ZW5zIG9mIEdCIGxhcmdlKSwgeW91IGNhbiB1bnBhY2sgaXQgd2l0aDoKCmBgYAokIHRhciAteGYgSG9tb19zYXBpZW5zX05DQklfR1JDaDM4LnRhci5negpgYGAKCjwvZGV0YWlscz4KCgojIyBXaGljaCBSZWZlcmVuY2UgaXMgUmlnaHQgZm9yIE1lPwoKVGhlIGtleSBpcyB0byBiZSBjb25zaXN0ZW50IGluIHlvdXIgcmVzZWFyY2guIFN3aXRjaGluZyBmcm9tIEVOU0VNQkwgdG8gVUNTQyB3aWxsIGNyZWF0ZSBtYW55IGhlYWRhY2hlcyBiZWNhdXNlIG9mIHRoZSBjaGFuZ2UgaW4gZ2VuZSBpZGVudGlmaWVycywgYW5kIGRpZmZlcmVuY2VzIGluIHRoZSBnZW5lIG1vZGVscyB0aGVtc2VsdmVzLiBPZnRlbiBwZW9wbGUgY2hvb3NlIHRoZSBvbmUgdGhleSdyZSBtb3N0IGNvbWZvcnRhYmxlIHdpdGgsIHdoaWNoIGlzIG9mdGVuIGEgZnVuY3Rpb24gb2YgaGlzdG9yaWNhbCBhY2NpZGVudC4gVGhlIGtleSBpcyBub3QgdG8gb3ZlcnRoaW5rIGl0LgoKQW5vdGhlciBpbXBvcnRhbnQgbm90ZSBpcyBub3QgdG8gbWl4IHRoZSBzb3VyY2VzLiBJZiB5b3UgZG93bmxvYWQgcmVmZXJlbmNlIHNlcXVlbmNlIGZyb20gVUNTQywgZG9uJ3QgdXNlIGFuIEVOU0VNQkwgR1RGIChhbmQgdmljZSB2ZXJzYSkuIE9uZSBvZiB0aGUgcXVpcmt5IGRpZmZlcmVuY2VzIGJldHdlZW4gdGhlIHR3byBkYXRhYmFzZXMgaXMgdGhhdCBFTlNFTUJMIHJlZmVycyB0byBjaHJvbW9zb21lIG9ubHkgYnkgdGhlaXIgbnVtYmVyLCBpLmUuIGAxYCwgd2hlcmVhcyBVQ1NDIHJlZmVycyB0byBjaHJvbXNvbWVzIGFzIGBjaHIxYC4gVGhpcyBtYWtlcyByZWZlcmVuY2UgRkFTVEFzIGZyb20gb25lIHNvdXJjZSBpbmNvbXBhdGlibGUgd2l0aCBnZW5lIGJ1aWxkcyBmcm9tIGFub3RoZXIuCgotLS0KClRoZXNlIG1hdGVyaWFscyBoYXZlIGJlZW4gYWRhcHRlZCBhbmQgZXh0ZW5kZWQgZnJvbSBtYXRlcmlhbHMgY3JlYXRlZCBieSB0aGUgW0hhcnZhcmQgQ2hhbiBCaW9pbmZvcm1hdGljcyBDb3JlIChIQkMpXShodHRwOi8vYmlvaW5mb3JtYXRpY3Muc3BoLmhhcnZhcmQuZWR1LykuIFRoZXNlIGFyZSBvcGVuIGFjY2VzcyBtYXRlcmlhbHMgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBbQ3JlYXRpdmUgQ29tbW9ucyBBdHRyaWJ1dGlvbiBsaWNlbnNlIChDQyBCWSA0LjApXShodHRwOi8vY3JlYXRpdmVjb21tb25zLm9yZy9saWNlbnNlcy9ieS80LjAvKSwgd2hpY2ggcGVybWl0cyB1bnJlc3RyaWN0ZWQgdXNlLCBkaXN0cmlidXRpb24sIGFuZCByZXByb2R1Y3Rpb24gaW4gYW55IG1lZGl1bSwgcHJvdmlkZWQgdGhlIG9yaWdpbmFsIGF1dGhvciBhbmQgc291cmNlIGFyZSBjcmVkaXRlZC4K