Objectives:

  • Discuss count normalizations
  • Execute model fitting for differential expression comparisons

1 Differential Expression Workflow

Here we will proceed with count normalizations and fit our DESeq2 model.


2 Count normalizations

Since counts of mapped reads for each gene is proportional to the expression of RNA in addition to many “uninteresting” other factors, normalization is the process of scaling raw count values to account for the “uninteresting” factors and ensure expression levels are more comparable.

2.1 Normalization goals

Two common factors that need to be accounted for during normalization are sequencing depth and gene length. Common normalization approaches (such as FPKM, RPKM, CPM, TPM, etc.) account for one or both of these factors.

  • Sequencing depth normalization is necessary to account for the proportion of reads per gene expected for more deeply sequenced samples (like in pink below) versus a less deeply sequenced sample (like in green below).

Note that each pink or green rectangle represents an aligned read, with reads spanning an intron connected by a dashed line. Figure from HBC training materials

  • Gene length normalization is necessary since genes of different lengths have different probabilities of generating fragments that end up in the library. In the example below, both genes have similar levels of expression. However, the longer gene (Gene X) will have more mapped reads than the short gene (Gene Y).

Each pink rectangle represents an aligned read, with reads spanning an intron connected by a dashed line. Figure from HBC training materials

Note: The above figures are originally from a HBC tutorial that also includes a detailed comparison of different normalization (CPM, TPM, FPKM) approaches and their best uses.

Check-in: Questions about normalizations?

2.2 DESeq2 normalizations

DESeq2 has an internal normalization process that accounts for RNA composition. A few highly differentially expressed genes, differences in the number of genes expressed between samples, or contamination are not accounted for by depth or gene length normalization methods. Accounting for RNA composition is particularly important for differential expression analyses, regardless of the tool used.

For data exploration and visualizations, it is helpful to generate an object of independently normalized counts. We will use the rlog transformation that accounts for sequencing depth for each sample and RNA composition differences in the overall experiment for downstream quality control visualizations.

The rlog transformation produces log2 scale data that has also been normalized to overall library size as well as variance across genes at different mean expression levels. For larger numbers of samples, there is an alternative transformation method, vst that can be used instead for count normalizations.

The command to generate the normalized count object has a few parts, including dds as an input and providing a value to the option blind.

rld <- rlog(dds, blind = TRUE)

*Note: We see a message here that our data did not fit the default ‘parametric’ dispersion model so a local regression was substituted. If we had more time, we might look at a dispersion plot with the plotDispEsts(dds) function, but as this bioconductor thread discusses, other visualizations of our data might be more helpful and/or easier to interpret.

Next, we’ll look at the results of the transformation.

head(assay(rld), 2)
##                    Sample_116498 Sample_116499 Sample_116500 Sample_116501 Sample_116502 Sample_116503
## ENSMUSG00000000001     12.635215     12.609457      12.76686     12.616921     12.705707     12.661095
## ENSMUSG00000000028      8.096766      8.350835       8.17923      8.391328      8.616073      8.653318
##                    Sample_116504 Sample_116505 Sample_116506 Sample_116507 Sample_116508 Sample_116509
## ENSMUSG00000000001      12.54194      12.52138      12.49388      12.71019      12.52632      12.48844
## ENSMUSG00000000028      10.71402      10.77955      10.88032      11.01995      10.93564      10.96100

Looking at the rld values, we can see that they are now in log scale. Since we set blind=TRUE, the transformation is blind to the sample information we specified in the design formula. The normalized counts are helpful for visualization methods during expression-level quality assessment but aren’t used in the model fitting.

3 DESeq2 Model Fitting

Next, we’ll fit our standard model using the DESeq function and take a look at the objects we generate.

dds <- DESeq(dds)

This command applies the model to our data, using the sample information supplied when generating the dds object so it takes some time to run.

resultsNames(dds)
## [1] "Intercept"                    "Gtype.Tx_ko.Tx_vs_wt.Tx"      "Gtype.Tx_ko.control_vs_wt.Tx"
## [4] "Gtype.Tx_wt.control_vs_wt.Tx"

The results include three pairwise comparisons to the specified control as default but other information is now stored in the dds object so we can generate additional pairwise comparisons.

head(dds)

Checkpoint: If you see the same results when you execute resultsNames(dds), please indicate with the green ‘yes’ button. Otherwise, please use the red ‘x’ button to get help before the break

3.0.0.1 [Optional Exercise]

If you executed the commands in the option exercise from Module 07, you can fit a seperate DESeq2 model for the patient sample example, which included a covariate in our model.

Click for fitting a model that includes a covariate

r dds_patient <- DESeq(dds_patient)

r resultsNames(dds_patient) If you run through the optional exercises, you can explore the impact of adding a covariate by substituting dds_patient for dds and re-running those commands since both DESeq2 objects have their data organized in the same way.

4 Summary

In this section, we:

  • Learned about count normalizations and uses
  • Generated a normalized count table
  • Fit two DESeq2 models for our data
  • Saw the impact of including a covariate in our model

5 Sources

5.1 Training resources used to develop materials

These materials have been adapted and extended from materials listed above. 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.

LS0tCnRpdGxlOiAiRGF5IDIgLSBNb2R1bGUgMDg6IENvdW50IG5vcm1hbGl6YXRpb25zIGFuZCBERVNlcTIgbW9kZWwgZml0dGluZyIKYXV0aG9yOiAiVU0gQmlvaW5mb3JtYXRpY3MgQ29yZSIKZGF0ZTogImByIFN5cy5EYXRlKClgIgpvdXRwdXQ6CiAgICAgICAgaHRtbF9kb2N1bWVudDoKICAgICAgICAgICAgaW5jbHVkZXM6CiAgICAgICAgICAgICAgICBpbl9oZWFkZXI6IGhlYWRlci5odG1sCiAgICAgICAgICAgIHRoZW1lOiBwYXBlcgogICAgICAgICAgICB0b2M6IHRydWUKICAgICAgICAgICAgdG9jX2RlcHRoOiA0CiAgICAgICAgICAgIHRvY19mbG9hdDogdHJ1ZQogICAgICAgICAgICBudW1iZXJfc2VjdGlvbnM6IHRydWUKICAgICAgICAgICAgZmlnX2NhcHRpb246IHRydWUKICAgICAgICAgICAgbWFya2Rvd246IEdGTQogICAgICAgICAgICBjb2RlX2Rvd25sb2FkOiB0cnVlCi0tLQo8c3R5bGUgdHlwZT0idGV4dC9jc3MiPgpib2R5eyAvKiBOb3JtYWwgICovCiAgICAgIGZvbnQtc2l6ZTogMTRwdDsKICB9CnByZSB7CiAgZm9udC1zaXplOiAxMnB0Cn0KY29kZS5yewogIGZvbnQtc2l6ZTogMTJwdDsKfQo8L3N0eWxlPgoKPCEtLS0gQWxsb3cgdGhlIHBhZ2UgdG8gYmUgd2lkZXIgLS0tPgo8c3R5bGU+CiAgICBib2R5IC5tYWluLWNvbnRhaW5lciB7CiAgICAgICAgbWF4LXdpZHRoOiAxMjAwcHg7CiAgICB9Cjwvc3R5bGU+Cgo+ICMgT2JqZWN0aXZlczogICAgCj4gKiBEaXNjdXNzIGNvdW50IG5vcm1hbGl6YXRpb25zCj4gKiBFeGVjdXRlIG1vZGVsIGZpdHRpbmcgZm9yIGRpZmZlcmVudGlhbCBleHByZXNzaW9uIGNvbXBhcmlzb25zCgpgYGB7ciBMb2FkUnVubmluZ0RhdGEsIGV2YWw9VFJVRSwgZWNobz1GQUxTRSwgbWVzc2FnZT1GQUxTRSwgd2FybmluZz1GQUxTRX0KbGlicmFyeShERVNlcTIpCmxvYWQoInJkYXRhL1J1bm5pbmdEYXRhLlJEYXRhIikKYGBgCgoKIyBEaWZmZXJlbnRpYWwgRXhwcmVzc2lvbiBXb3JrZmxvdwoKSGVyZSB3ZSB3aWxsIHByb2NlZWQgd2l0aCBjb3VudCBub3JtYWxpemF0aW9ucyBhbmQgZml0IG91ciBERVNlcTIgbW9kZWwuCgohW10oLi9pbWFnZXMvd2F5ZmluZGVyL3dheWZpbmRlci0wNC5wbmcpe3dpZHRoPTc1JX0KCi0tLQoKIyBDb3VudCBub3JtYWxpemF0aW9ucwoKU2luY2UgY291bnRzIG9mIG1hcHBlZCByZWFkcyBmb3IgZWFjaCBnZW5lIGlzIHByb3BvcnRpb25hbCB0byB0aGUgZXhwcmVzc2lvbiBvZiBSTkEgaW4gYWRkaXRpb24gdG8gbWFueSDigJx1bmludGVyZXN0aW5n4oCdIG90aGVyIGZhY3RvcnMsIG5vcm1hbGl6YXRpb24gaXMgdGhlIHByb2Nlc3Mgb2Ygc2NhbGluZyByYXcgY291bnQgdmFsdWVzIHRvIGFjY291bnQgZm9yIHRoZSDigJx1bmludGVyZXN0aW5n4oCdIGZhY3RvcnMgYW5kIGVuc3VyZSBleHByZXNzaW9uIGxldmVscyBhcmUgbW9yZSBjb21wYXJhYmxlLgoKIyMgTm9ybWFsaXphdGlvbiBnb2FscwoKVHdvIGNvbW1vbiBmYWN0b3JzIHRoYXQgbmVlZCB0byBiZSBhY2NvdW50ZWQgZm9yIGR1cmluZyBub3JtYWxpemF0aW9uIGFyZSAqKnNlcXVlbmNpbmcgZGVwdGgqKiBhbmQgKipnZW5lIGxlbmd0aCoqLiBDb21tb24gbm9ybWFsaXphdGlvbiBhcHByb2FjaGVzIChzdWNoIGFzIEZQS00sIFJQS00sIENQTSwgVFBNLCBldGMuKSBhY2NvdW50IGZvciBvbmUgb3IgYm90aCBvZiB0aGVzZSBmYWN0b3JzLiAgICAKCiogKipTZXF1ZW5jaW5nIGRlcHRoKiogbm9ybWFsaXphdGlvbiBpcyBuZWNlc3NhcnkgdG8gYWNjb3VudCBmb3IgdGhlIHByb3BvcnRpb24gb2YgcmVhZHMgcGVyIGdlbmUgZXhwZWN0ZWQgZm9yIG1vcmUgZGVlcGx5IHNlcXVlbmNlZCBzYW1wbGVzIChsaWtlIGluIHBpbmsgYmVsb3cpIHZlcnN1cyBhIGxlc3MgZGVlcGx5IHNlcXVlbmNlZCBzYW1wbGUgKGxpa2UgaW4gZ3JlZW4gYmVsb3cpLiAgICAKCiFbKk5vdGUgdGhhdCBlYWNoIHBpbmsgb3IgZ3JlZW4gcmVjdGFuZ2xlIHJlcHJlc2VudHMgYW4gYWxpZ25lZCByZWFkLCB3aXRoIHJlYWRzIHNwYW5uaW5nIGFuIGludHJvbiBjb25uZWN0ZWQgYnkgYSBkYXNoZWQgbGluZS4gRmlndXJlIGZyb20gW0hCQyAgdHJhaW5pbmcgbWF0ZXJpYWxzXShodHRwczovL2hiY3RyYWluaW5nLmdpdGh1Yi5pby9ER0Vfd29ya3Nob3AvbGVzc29ucy8wMl9ER0VfY291bnRfbm9ybWFsaXphdGlvbi5odG1sKSpdKC4vaW1hZ2VzL25vcm1hbGl6YXRpb25fbWV0aG9kc19kZXB0aC5wbmcpICAgIAoKCiogKipHZW5lIGxlbmd0aCoqIG5vcm1hbGl6YXRpb24gaXMgbmVjZXNzYXJ5IHNpbmNlIGdlbmVzIG9mIGRpZmZlcmVudCBsZW5ndGhzIGhhdmUgZGlmZmVyZW50IHByb2JhYmlsaXRpZXMgb2YgZ2VuZXJhdGluZyBmcmFnbWVudHMgdGhhdCBlbmQgdXAgaW4gdGhlIGxpYnJhcnkuIEluIHRoZSBleGFtcGxlIGJlbG93LCBib3RoIGdlbmVzIGhhdmUgc2ltaWxhciBsZXZlbHMgb2YgZXhwcmVzc2lvbi4gSG93ZXZlciwgdGhlIGxvbmdlciBnZW5lIChHZW5lIFgpIHdpbGwgaGF2ZSBtb3JlIG1hcHBlZCByZWFkcyB0aGFuIHRoZSBzaG9ydCBnZW5lIChHZW5lIFkpLiAgICAKCiFbKkVhY2ggcGluayByZWN0YW5nbGUgcmVwcmVzZW50cyBhbiBhbGlnbmVkIHJlYWQsIHdpdGggcmVhZHMgc3Bhbm5pbmcgYW4gaW50cm9uIGNvbm5lY3RlZCBieSBhIGRhc2hlZCBsaW5lLiBGaWd1cmUgZnJvbSBbSEJDICB0cmFpbmluZyBtYXRlcmlhbHNdKGh0dHBzOi8vaGJjdHJhaW5pbmcuZ2l0aHViLmlvL0RHRV93b3Jrc2hvcC9sZXNzb25zLzAyX0RHRV9jb3VudF9ub3JtYWxpemF0aW9uLmh0bWwpKl0oLi9pbWFnZXMvbm9ybWFsaXphdGlvbl9tZXRob2RzX2xlbmd0aC5wbmcpe3dpZHRoPTc1JX0gICAgICAKCgo+ICoqTm90ZSoqOiBUaGUgYWJvdmUgZmlndXJlcyBhcmUgb3JpZ2luYWxseSBmcm9tIGEgW0hCQyAgdHV0b3JpYWxdKGh0dHBzOi8vaGJjdHJhaW5pbmcuZ2l0aHViLmlvL0RHRV93b3Jrc2hvcC9sZXNzb25zLzAyX0RHRV9jb3VudF9ub3JtYWxpemF0aW9uLmh0bWwpIHRoYXQgYWxzbyBpbmNsdWRlcyBhIGRldGFpbGVkIGNvbXBhcmlzb24gb2YgZGlmZmVyZW50IG5vcm1hbGl6YXRpb24gKENQTSwgVFBNLCBGUEtNKSBhcHByb2FjaGVzIGFuZCB0aGVpciBiZXN0IHVzZXMuCgoqKkNoZWNrLWluOiBRdWVzdGlvbnMgYWJvdXQgbm9ybWFsaXphdGlvbnM/KioKCiMjIERFU2VxMiBub3JtYWxpemF0aW9ucwoKREVTZXEyIGhhcyBhbiBbaW50ZXJuYWwgbm9ybWFsaXphdGlvbiBwcm9jZXNzXShodHRwczovL2dlbm9tZWJpb2xvZ3kuYmlvbWVkY2VudHJhbC5jb20vYXJ0aWNsZXMvMTAuMTE4Ni9nYi0yMDEwLTExLTEwLXIxMDYpIHRoYXQgYWNjb3VudHMgZm9yICoqUk5BIGNvbXBvc2l0aW9uKiouIEEgZmV3IGhpZ2hseSBkaWZmZXJlbnRpYWxseSBleHByZXNzZWQgZ2VuZXMsIGRpZmZlcmVuY2VzIGluIHRoZSBudW1iZXIgb2YgZ2VuZXMgZXhwcmVzc2VkIGJldHdlZW4gc2FtcGxlcywgb3IgY29udGFtaW5hdGlvbiBhcmUgbm90IGFjY291bnRlZCBmb3IgYnkgZGVwdGggb3IgZ2VuZSBsZW5ndGggbm9ybWFsaXphdGlvbiBtZXRob2RzLiBBY2NvdW50aW5nIGZvciBSTkEgY29tcG9zaXRpb24gaXMgcGFydGljdWxhcmx5IGltcG9ydGFudCBmb3IgZGlmZmVyZW50aWFsIGV4cHJlc3Npb24gYW5hbHlzZXMsIHJlZ2FyZGxlc3Mgb2YgdGhlIHRvb2wgdXNlZC4KCkZvciBkYXRhIGV4cGxvcmF0aW9uIGFuZCB2aXN1YWxpemF0aW9ucywgaXQgaXMgaGVscGZ1bCB0byBnZW5lcmF0ZSBhbiBvYmplY3Qgb2YgaW5kZXBlbmRlbnRseSBub3JtYWxpemVkIGNvdW50cy4gV2Ugd2lsbCB1c2UgdGhlIFtybG9nIHRyYW5zZm9ybWF0aW9uXShodHRwOi8vYmlvY29uZHVjdG9yLm9yZy9wYWNrYWdlcy9kZXZlbC9iaW9jL3ZpZ25ldHRlcy9ERVNlcTIvaW5zdC9kb2MvREVTZXEyLmh0bWwjY291bnQtZGF0YS10cmFuc2Zvcm1hdGlvbnMpIHRoYXQgYWNjb3VudHMgZm9yIHNlcXVlbmNpbmcgZGVwdGggZm9yIGVhY2ggc2FtcGxlIGFuZCBSTkEgY29tcG9zaXRpb24gZGlmZmVyZW5jZXMgaW4gdGhlIG92ZXJhbGwgZXhwZXJpbWVudCBmb3IgZG93bnN0cmVhbSBxdWFsaXR5IGNvbnRyb2wgdmlzdWFsaXphdGlvbnMuCgpUaGUgcmxvZyB0cmFuc2Zvcm1hdGlvbiBwcm9kdWNlcyBsb2cyIHNjYWxlIGRhdGEgdGhhdCBoYXMgYWxzbyBiZWVuIG5vcm1hbGl6ZWQgdG8gb3ZlcmFsbCBsaWJyYXJ5IHNpemUgYXMgd2VsbCBhcyB2YXJpYW5jZSBhY3Jvc3MgZ2VuZXMgYXQgZGlmZmVyZW50IG1lYW4gZXhwcmVzc2lvbiBsZXZlbHMuIEZvciBsYXJnZXIgbnVtYmVycyBvZiBzYW1wbGVzLCB0aGVyZSBpcyBhbiBhbHRlcm5hdGl2ZSB0cmFuc2Zvcm1hdGlvbiBtZXRob2QsIFt2c3RdKGh0dHA6Ly93d3cuYmlvY29uZHVjdG9yLm9yZy9wYWNrYWdlcy9yZWxlYXNlL2Jpb2MvdmlnbmV0dGVzL0RFU2VxMi9pbnN0L2RvYy9ERVNlcTIuaHRtbCNjb3VudC1kYXRhLXRyYW5zZm9ybWF0aW9ucykgdGhhdCBjYW4gYmUgdXNlZCBpbnN0ZWFkIGZvciBjb3VudCBub3JtYWxpemF0aW9ucy4KCgpUaGUgY29tbWFuZCB0byBnZW5lcmF0ZSB0aGUgbm9ybWFsaXplZCBjb3VudCBvYmplY3QgaGFzIGEgZmV3IHBhcnRzLCBpbmNsdWRpbmcgYGRkc2AgYXMgYW4gaW5wdXQgYW5kIHByb3ZpZGluZyBhIHZhbHVlIHRvIHRoZSBvcHRpb24gYGJsaW5kYC4KYGBge3IgQ291bnROb3JtfQpybGQgPC0gcmxvZyhkZHMsIGJsaW5kID0gVFJVRSkKYGBgCgoqTm90ZTogV2Ugc2VlIGEgbWVzc2FnZSBoZXJlIHRoYXQgb3VyIGRhdGEgZGlkIG5vdCBmaXQgdGhlIGRlZmF1bHQgJ3BhcmFtZXRyaWMnIGRpc3BlcnNpb24gbW9kZWwgc28gYSBsb2NhbCByZWdyZXNzaW9uIHdhcyBzdWJzdGl0dXRlZC4gSWYgd2UgaGFkIG1vcmUgdGltZSwgd2UgbWlnaHQgbG9vayBhdCBhIGRpc3BlcnNpb24gcGxvdCB3aXRoIHRoZSBgcGxvdERpc3BFc3RzKGRkcylgIGZ1bmN0aW9uLCBidXQgYXMgW3RoaXMgYmlvY29uZHVjdG9yIHRocmVhZF0oaHR0cHM6Ly9zdXBwb3J0LmJpb2NvbmR1Y3Rvci5vcmcvcC8xMDc5MzcvKSBkaXNjdXNzZXMsIG90aGVyIHZpc3VhbGl6YXRpb25zIG9mIG91ciBkYXRhIG1pZ2h0IGJlIG1vcmUgaGVscGZ1bCBhbmQvb3IgZWFzaWVyIHRvIGludGVycHJldC4KCk5leHQsIHdlJ2xsIGxvb2sgYXQgdGhlIHJlc3VsdHMgb2YgdGhlIHRyYW5zZm9ybWF0aW9uLgpgYGB7ciBDb3VudE5vcm1DaGVja30KaGVhZChhc3NheShybGQpLCAyKQpgYGAKCkxvb2tpbmcgYXQgdGhlIHJsZCB2YWx1ZXMsIHdlIGNhbiBzZWUgdGhhdCB0aGV5IGFyZSBub3cgaW4gbG9nIHNjYWxlLiBTaW5jZSB3ZSBzZXQgYGJsaW5kPVRSVUVgLCB0aGUgdHJhbnNmb3JtYXRpb24gaXMgYmxpbmQgdG8gdGhlIHNhbXBsZSBpbmZvcm1hdGlvbiB3ZSBzcGVjaWZpZWQgaW4gdGhlIGRlc2lnbiBmb3JtdWxhLiBUaGUgbm9ybWFsaXplZCBjb3VudHMgYXJlIGhlbHBmdWwgZm9yIHZpc3VhbGl6YXRpb24gbWV0aG9kcyBkdXJpbmcgZXhwcmVzc2lvbi1sZXZlbCBxdWFsaXR5IGFzc2Vzc21lbnQgYnV0ICoqYXJlbid0IHVzZWQgaW4gdGhlIG1vZGVsIGZpdHRpbmcqKi4KCiMgREVTZXEyIE1vZGVsIEZpdHRpbmcKCk5leHQsIHdlJ2xsIGZpdCBvdXIgc3RhbmRhcmQgbW9kZWwgdXNpbmcgdGhlIGBERVNlcWAgZnVuY3Rpb24gYW5kIHRha2UgYSBsb29rIGF0IHRoZSBvYmplY3RzIHdlIGdlbmVyYXRlLgpgYGB7ciBGaXRNb2RlbFN0YW5kYXJkLCBtZXNzYWdlPUZBTFNFfQpkZHMgPC0gREVTZXEoZGRzKQpgYGAKClRoaXMgY29tbWFuZCBhcHBsaWVzIHRoZSBtb2RlbCB0byBvdXIgZGF0YSwgdXNpbmcgdGhlIHNhbXBsZSBpbmZvcm1hdGlvbiBzdXBwbGllZCB3aGVuIGdlbmVyYXRpbmcgdGhlIGBkZHNgIG9iamVjdCBzbyBpdCB0YWtlcyBzb21lIHRpbWUgdG8gcnVuLgoKYGBge3IgRml0TW9kZWxTdGFuZGFyZENoZWNrMSwgZXZhbD1UUlVFfQpyZXN1bHRzTmFtZXMoZGRzKQpgYGAKClRoZSByZXN1bHRzIGluY2x1ZGUgdGhyZWUgcGFpcndpc2UgY29tcGFyaXNvbnMgdG8gdGhlIHNwZWNpZmllZCBjb250cm9sIGFzIGRlZmF1bHQgYnV0IG90aGVyIGluZm9ybWF0aW9uIGlzIG5vdyBzdG9yZWQgaW4gdGhlIGBkZHNgIG9iamVjdCBzbyB3ZSBjYW4gZ2VuZXJhdGUgYWRkaXRpb25hbCBwYWlyd2lzZSBjb21wYXJpc29ucy4KCmBgYHtyIEZpdE1vZGVsU3RhbmRhcmRDaGVjazIsIGV2YWw9RkFMU0V9CmhlYWQoZGRzKQpgYGAKCioqQ2hlY2twb2ludCoqOiAqSWYgeW91IHNlZSB0aGUgc2FtZSByZXN1bHRzIHdoZW4geW91IGV4ZWN1dGUgYHJlc3VsdHNOYW1lcyhkZHMpYCwgcGxlYXNlIGluZGljYXRlIHdpdGggdGhlIGdyZWVuICd5ZXMnIGJ1dHRvbi4gT3RoZXJ3aXNlLCBwbGVhc2UgdXNlIHRoZSByZWQgJ3gnIGJ1dHRvbiB0byBnZXQgaGVscCBiZWZvcmUgdGhlIGJyZWFrKgoKCgojIyMjIFtPcHRpb25hbCBFeGVyY2lzZV0KCklmIHlvdSBleGVjdXRlZCB0aGUgY29tbWFuZHMgaW4gdGhlIG9wdGlvbiBleGVyY2lzZSBmcm9tIE1vZHVsZSAwNywgeW91IGNhbiBmaXQgYSBzZXBlcmF0ZSBERVNlcTIgbW9kZWwgZm9yIHRoZSBwYXRpZW50IHNhbXBsZSBleGFtcGxlLCB3aGljaCBpbmNsdWRlZCBhIGNvdmFyaWF0ZSBpbiBvdXIgbW9kZWwuCgo8ZGV0YWlscz4KICAgIDxzdW1tYXJ5PipDbGljayBmb3IgZml0dGluZyBhIG1vZGVsIHRoYXQgaW5jbHVkZXMgYSBjb3ZhcmlhdGUqPC9zdW1tYXJ5PgogICAgYGBge3IgRml0TW9kZWxDb3ZhcmlhdGUsIGV2YWw9IEZBTFNFLCBtZXNzYWdlPUZBTFNFfQogICAgZGRzX3BhdGllbnQgPC0gREVTZXEoZGRzX3BhdGllbnQpCiAgICBgYGAKICAgIGBgYHtyIEZpdE1vZGVsQ292YXJpYXRlQ2hlY2sxLGV2YWw9RkFMU0V9CiAgICByZXN1bHRzTmFtZXMoZGRzX3BhdGllbnQpCiAgICBgYGAKICAgIElmIHlvdSBydW4gdGhyb3VnaCB0aGUgb3B0aW9uYWwgZXhlcmNpc2VzLCB5b3UgY2FuIGV4cGxvcmUgdGhlIGltcGFjdCBvZiBhZGRpbmcgYSBjb3ZhcmlhdGUgYnkgc3Vic3RpdHV0aW5nIGBkZHNfcGF0aWVudGAgZm9yIGBkZHNgIGFuZCByZS1ydW5uaW5nIHRob3NlIGNvbW1hbmRzIHNpbmNlIGJvdGggREVTZXEyIG9iamVjdHMgaGF2ZSB0aGVpciBkYXRhIG9yZ2FuaXplZCBpbiB0aGUgc2FtZSB3YXkuCjwvZGV0YWlscz4KCgoKCiMgU3VtbWFyeQoKSW4gdGhpcyBzZWN0aW9uLCB3ZTogICAKCiogTGVhcm5lZCBhYm91dCBjb3VudCBub3JtYWxpemF0aW9ucyBhbmQgdXNlcyAgICAKKiBHZW5lcmF0ZWQgYSBub3JtYWxpemVkIGNvdW50IHRhYmxlICAgIAoqIEZpdCB0d28gREVTZXEyIG1vZGVscyBmb3Igb3VyIGRhdGEgICAgCiogU2F3IHRoZSBpbXBhY3Qgb2YgaW5jbHVkaW5nIGEgY292YXJpYXRlIGluIG91ciBtb2RlbAoKCgotLS0KCiMgU291cmNlcwojIyBUcmFpbmluZyByZXNvdXJjZXMgdXNlZCB0byBkZXZlbG9wIG1hdGVyaWFscwoqIEhCQyBER0Ugc2V0dXA6IGh0dHBzOi8vaGJjdHJhaW5pbmcuZ2l0aHViLmlvL0RHRV93b3Jrc2hvcC9sZXNzb25zLzAxX0RHRV9zZXR1cF9hbmRfb3ZlcnZpZXcuaHRtbAoqIEhCQyBDb3VudCBOb3JtYWxpemF0aW9uOiBodHRwczovL2hiY3RyYWluaW5nLmdpdGh1Yi5pby9ER0Vfd29ya3Nob3AvbGVzc29ucy8wMl9ER0VfY291bnRfbm9ybWFsaXphdGlvbi5odG1sCiogREVTZXEyIHN0YW5kYXJkIHZpZ25ldHRlOiBodHRwOi8vYmlvY29uZHVjdG9yLm9yZy9wYWNrYWdlcy9kZXZlbC9iaW9jL3ZpZ25ldHRlcy9ERVNlcTIvaW5zdC9kb2MvREVTZXEyLmh0bWwKKiBERVNlcTIgYmVnaW5uZXJzIHZpZ25ldHRlOiBodHRwczovL2Jpb2MuaXNtLmFjLmpwL3BhY2thZ2VzLzIuMTQvYmlvYy92aWduZXR0ZXMvREVTZXEyL2luc3QvZG9jL2JlZ2lubmVyLnBkZgoqIEJpb2NvbmR1Y3RvciBSTkEtc2VxIFdvcmtmbG93czogaHR0cHM6Ly93d3cuYmlvY29uZHVjdG9yLm9yZy9oZWxwL2NvdXJzZS1tYXRlcmlhbHMvMjAxNS9MZWFybkJpb2NvbmR1Y3RvckZlYjIwMTUvQjAyLjFfUk5BU2VxLmh0bWwKCgoKYGBge3IgV3JpdGVPdXQuUkRhdGEsIGV2YWw9VFJVRSwgZWNobz1GQUxTRSwgbWVzc2FnZT1GQUxTRSwgd2FybmluZz1GQUxTRX0KI0hpZGRlbiBjb2RlIGJsb2NrIHRvIHdyaXRlIG91dCBkYXRhIGZvciBrbml0dGluZwpzYXZlLmltYWdlKGZpbGUgPSAicmRhdGEvUnVubmluZ0RhdGEuUkRhdGEiKQoKYGBgCgoKVGhlc2UgbWF0ZXJpYWxzIGhhdmUgYmVlbiBhZGFwdGVkIGFuZCBleHRlbmRlZCBmcm9tIG1hdGVyaWFscyBsaXN0ZWQgYWJvdmUuIFRoZXNlIGFyZSBvcGVuIGFjY2VzcyBtYXRlcmlhbHMgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBbQ3JlYXRpdmUgQ29tbW9ucyBBdHRyaWJ1dGlvbiBsaWNlbnNlIChDQyBCWSA0LjApXShodHRwOi8vY3JlYXRpdmVjb21tb25zLm9yZy9saWNlbnNlcy9ieS80LjAvKSwgd2hpY2ggcGVybWl0cyB1bnJlc3RyaWN0ZWQgdXNlLCBkaXN0cmlidXRpb24sIGFuZCByZXByb2R1Y3Rpb24gaW4gYW55IG1lZGl1bSwgcHJvdmlkZWQgdGhlIG9yaWdpbmFsIGF1dGhvciBhbmQgc291cmNlIGFyZSBjcmVkaXRlZC4K