Note to self: I changed the clusterProfiler output list to store the enrichResults in [[xx_data]][[enrich]], or in the case of go: [[go_data]][[BP_enrich]] instead of enrich_results.

1 Set some parameters which will be used later

## LogFC cutoff when working on the inclusion sets.
lfc_cutoff <- 0.1
## Adjusted p-value cutoff when working on the inclusion sets.
adjp_cutoff <- 0.1
## Increase the maximum allowed group size when working with clusterProfiler
## This should bring out some of the more general groups like 'cellbody'
max_groupsize <- 2000
## Allow groups higher up in the tree for clusterProfiler results.
go_level <- 2
## Allow 10 GO categories to be displayed when plotting.
go_categories <- 14
## MA plot point outlines
outline <- FALSE
## Speed up clusterProfiler by choosing the correct keytypes
orgdb_from <- "ENSEMBL"
default_fstring <- "~ 0 + condition"

2 TODOs

  • Check this for correctness.
  • Reorganize it
  • GSVA (Hänzelmann, Castelo, and Guinney (2013))
  • Consider embedding some/many/all of the Excel outputs into the html output via xfun::embed_dir(‘excel/’)

3 Meeting with Theresa

Previous papers did not do an explicit subtraction, instead just compared to WT and kept the genes which are > in delta/het vs. wt. There are multiple ways to deal with this and that query has not yet been defined. Later, Theresa came to the conclusion that the subtraction method is not appropriate.

4 Introduction

In this document I hope to explore the freshly processed samples and perform some comparisons to see that we have the expected similarities and differences from the prior analysis performed by Theresa.

There is one way in which I expect any/all of these analyses to be explicitly different: this should include the changes produced by April’s renaming of some samples.

My intention is to produce a sample sheet which includes one column with non-umi-deduplicated results and one with deduplicated results. With the exception of the previous point, I hope that the first will be identical (or at least very close to identical) to Theresa’s result while the second I expect will be subtly different – but I am hoping subtly enough that it will not significantly change the interpretation but be a little more precise.

Lets see! I need therefore to make a change to my metadata gathering function to include the umi deduplicated result. I am thinking therefore to create a separate specification for umi-barcoded samples because looking through the logs for umi stuff when they are not used will be too much of a pain…

4.1 Small random reminder

I have a couple pictures of RPL22 to help me remember the experimental design:

  • The human ribosome with RPL22 in red: human_rpl22_red
  • The mouse ribosome with RPL22 in green at the center: mouse_rpl22

That second picture came from: (Li et al. (2022))

5 A note about implementation

I would like to improve this document by comparing/contrasting the methodologies performed by other groups and those performed by me in it. I never fully appreciated the suite of computational methods applied by previous groups when examining TRAP data; I instead simply followed Theresa’s notebook without considering other possibilities.

I therefore spent a little time stepping through her thesis and pulling out the relevant papers in the hopes of learning these various methods. I should therefore be able soon to compare/contrast the various methods employed by other labs in addition to copying Theresa’s logic.

5.1 The following block cannot work in the container

The following block assumes the full tree of preprocessed data with the logs from the trimmer, mapping, umi deduplication, counting, etc. As a result it cannot work in the container which has only the various count tables.

As a result, I am including a copy of this sheet after running the following block in my working tree. I suppose for the moment you will have to trust that it worked. (for right now, when testing out this container, I am just sending the R working directory to my tree for this block, then moving it back.

I will need to manually edit one column though, the symlink column from Theresa has a series of paths which do not work in the container.

umi_spec <- make_rnaseq_spec(umi = TRUE)
iprgc_2022_meta <- gather_preprocessing_metadata("sample_sheets/20240606_only_umd_sequenced.xlsx",
                                                 spec = umi_spec, species = "mm39_112", verbose = FALSE,
                                                 basedir = "preprocessing/umd_sequenced")
colnames(iprgc_2022_meta[["new_meta"]])
head(iprgc_2022_meta[["new_meta"]])
sample_sheet <- "sample_sheets/20240606_only_umd_sequenced_modified.xlsx"
msigdb <- "reference/msigdb_v2024.1.Mm.db"
msig_data <- NULL
make_transparent <- function() {
  ggplot2::theme(
    panel.background = element_rect(fill = 'transparent'),
    plot.background = element_rect(fill = 'transparent', color = NA),
    panel.grid.major = element_blank(),
    panel.grid.minor = element_blank(),
    legend.background = element_rect(fill = 'transparent'),
    legend.box.background = element_rect(fill = 'transparent'))
}

I will figure out if I can leave mSigDB M2 in this image; if not, then any analyses depending on those gene sets will fail a priori.

m2_gsc <- try(load_gmt_signatures(signatures = msigdb,
                                  signature_category = "M2"), silent = TRUE)
## I do not think I have permission to load the msigdb in the container
## So, if this fails, just load it from GSVAData, oh wait no, GSVAdata is human.
if ("try-error" %in% class(m2_gsc)) {
  warning("Unable to load the M2 MsigDB data.")
}
## Warning: Unable to load the M2 MsigDB data.

From this point on, I am hoping/intending to pull liberally from Theresa’s notebook with a diversion to compare the three datasets:

  • Pre-April renaming: E.g. Theresa’s current dataset
  • Post renaming: Unless I am mistaken, this should be very similar to the above.
  • Post deduplication: Given what I saw from the extracted logs in the sample sheet, I expect this to be similar but not identical to the previous two.

Lets find out! But first, annotations!

6 Annotation data

I am pulling this from Theresa’s anxontrapR_pipeline.Rmd, primarily because it looks similar to the other documents, but was modified more recently. I will change it slightly, primarily because I grabbed a new mmusculus assembly and therefore I will pull the mmusculus annotations from a specific biomart (Smedley et al. (2009)) archive that should match it.

A note from the future: multiple ensembl archive servers have been taken offline since last I ran this. Let us see if Feb. 2023 still works.

6.1 An important note!

In the recent past, ensembl queries have become inconsistent, failing much more often than ever in the past. I do not think this is the fault of ensembl; but I think I need a fallback mechanism for collecting annotation information.

In the case of ensembl, it should be trivial (but less fun) to use a combination of the locally installed orgdb and txdb databases.

This does open a risk that the set of genes with annotations will be different depending on when the container is run due to differences between the orgdb/txdb instance and the Feb 2023 biomart. I am not sure there is much I can do about that except to bundle the set of annotations I downloaded in the container – since load_biomart_annotations() does save a rda copy of its download.

ok, I did both. If you, dear reader, wish to download your own annotations, and ensembl is having troubles, the following should work without a problem; in addition the rda annotations are in /data of the container and should get loaded.

tx_gene_map <- data.frame()
mm_annot <- try(load_biomart_annotations(species = "mmusculus", year = "2023", month = "02"))
## The biomart annotations file already exists, loading from it.
if ("try-error" %in% class(mm_annot)) {
  fields <- c("ACCNUM", "ENSEMBL", "ENSEMBLTRANS", "ENTEZID", "GENENAME", "SYMBOL")
  orgdb_annot <- load_orgdb_annotations("org.Mm.eg.db", fields = fields)
  gene_info <- orgdb_annot[["genes"]]
  ## Note, there are a bunch of variants of the txdb package one might use.
  ## I do not think it matters a lot for our purposes, but I suspect that if we used
  ## a mismatched BSgenome and tried to pull CDS sequences, that might end badly.
  pkg <- "TxDb.Mmusculus.UCSC.mm10.knownGene"
  tx_annot <- load_txdb_annotations(pkg)
  transcripts <- tx_annot[["TX"]]
  transcripts[["tx"]] <- gsub(x = transcripts[["TXNAME"]],
                              pattern = "\\.\\d+$", replacement = "")
  mm_annot <- merge(gene_info, transcripts, by.x = "ensembltrans", by.y = "tx")
  rownames(mm_annot) <- make.names(mm_annot[["ensembl"]], unique = TRUE)
} else {
  mm_annot <- mm_annot[["annotation"]]
  mm_annot[["txid"]] <- paste0(mm_annot[["ensembl_transcript_id"]], ".", mm_annot[["version"]])
  rownames(mm_annot) <- make.names(mm_annot[["ensembl_gene_id"]], unique=TRUE)
  tx_gene_map <- mm_annot[, c("txid", "ensembl_gene_id")]
}

7 Hisat2 summarizedExperiments

The primary difference between my block and Theresa’s are:

  1. I am pulling the metadata directly from the gather_preprocessing_metadata() above.
  2. I am using the column ‘symlink’ which is just a copy of the existing ‘file’ column with a small change so I can load it from my directory without having to copy everything.
  3. I am using the ensembl genome release 39, version 112 and so pulled a somewhat newer copy of the annotation data.
  4. The original is named ‘v1’, followed by ‘v2’ and ‘v3’ for the other two treatments I performed.

7.1 Color choices and reused parameters

Given that we are excluding a bunch of the older samples, the set of colors I expect to find is different; so I will make explicit here the various colors used to denote location/genotype/time/etc.

April turned me onto this website ‘paletton.com’ for this kind of stuff and I will try and pick out palettes which basically match what I am getting with the original colors.

color_choices <- list(
  "all" = list(
    "p08_het_dlgn" = "#E7298A",
    "p15_het_dlgn" = "#E7298A",
    "p08_het_retina" = "#238B45",
    "p15_het_retina" = "#238B45",
    "p08_het_scn" = "#4292C6",
    "p15_het_scn" = "#4292C6",
    "p08_ko_dlgn" = "#C994C7",
    "p15_ko_dlgn" = "#C994C7",
    "p08_ko_retina" = "#74c476",
    "p15_ko_retina" = "#74c476",
    "p08_ko_scn" = "#9BCAE1",
    "p15_ko_scn" = "#9BCAE1",
    "p08_wt_dlgn" = "#980043",
    "p15_wt_dlgn" = "#980043",
    "p08_wt_retina" = "#004008",
    "p15_wt_retina" = "#004008",
    "p08_wt_scn" = "#08519C",
    "p15_wt_scn" = "#08519C",
    "p60_wt_dlgn" = "#333333",
    "p60_wt_retina" = "#222222",
    "p60_wt_scn" = "#111111"),
  "geno_loc" = list(
    "het_dlgn" = "#E7298A",
    "het_retina" = "#238B45",
    "het_scn" = "#4292C6",
    "ko_dlgn" = "#C994C7",
    "ko_retina" = "#74c476",
    "ko_scn" = "#9BCAE1",
    "wt_dlgn" = "#980043",
    "wt_retina" = "#004008",
    "wt_scn" = "#08519C"),
  "location" = list(
    "retina" = "#004008",
    "dlgn" = "#980043",
    "scn" = "#08519C"),
  "genotype" = list(
    "wt" = "#74c476",
    "het" = "#238B45",
    "ko" = "#006D2C"),
  "time" = list(
    "p08" = "#5E104B",
    "p15" = "#4E9231"))
label_column <- "mgi_symbol" ## Set the column used to extract gene symbols rather than ENSG.....
colors <- color_choices[["geno_loc"]]
time_colors <- list(
  "p08_het_dlgn" = "#E7298A",
  "p15_het_dlgn" = "#8a1852",
  "p08_het_retina" = "#238B45",
  "p15_het_retina" = "#155329",
  "p08_het_scn" = "#4292C6",
  "p15_het_scn" = "#275776",
  "p08_ko_dlgn" = "#C994C7",
  "p15_ko_dlgn" = "#785877",
  "p08_ko_retina" = "#74C476",
  "p15_ko_retina" = "#457546",
  "p08_ko_scn" = "#9BCAE1",
  "p15_ko_scn" = "#5d7987")

There is one noteworthy sample: iprgc_103, it was effectively replaced when April renamed the samples and so exists in the v1 data, but not v2/v3; they instead have the newly named samples which I called iprgc_123 to iprgc_130. As a result, I copied the annotations for iprgc_123 to my column so that there is no discrepency in terms of genotype/location/time.

7.2 The original count tables

At the moment I have not included the original counts in this container because we made some changes to the mapping strategy and also found that a couple samples were mixed up in sequencing; as a result I documented all of the changes in the sample sheets and preprocessing documents and excluded the original files.

This is also why some columns in the sample sheet have suffixes like ‘adh’ and ‘atb’, those denote from whom the relevant metadata columns came from.

mm38_hisat_v1 <- create_se(sample_sheet,
                           gene_info = mm_annot,
                           file_column = "symlink") %>%
  set_conditions(fact = "geno_loc_atb") %>%
  set_batches(fact = "time_atb") %>%
  set_colors(color_choices[["geno_loc"]])
mm38_hisat_v1

7.3 Recounted tables and the deduplicated result

In the following I make two more versions of the data, one remapped with the changes to the sample identities, and one with deduplication applied.

mm38_hisat_v2 <- create_se(sample_sheet, gene_info = mm_annot,
                           file_column = "hisat_count_table") %>%
  set_conditions(fact = "geno_loc_atb") %>%
  set_batches(fact = "time_atb") %>%
  set_colors(color_choices[["geno_loc"]])
## Reading the sample metadata.
## Checking the state of the condition column.
## Checking the state of the batch column.
## Checking the condition factor.
## The sample definitions comprises: 69 rows(samples) and 76 columns(metadata fields).
## Warning in create_se(sample_sheet, gene_info = mm_annot, file_column =
## "hisat_count_table"): Some samples were removed when cross referencing the
## samples against the count data.
## Matched 25404 annotations and counts.
## Some annotations were lost in merging, setting them to 'undefined'.
## The final summarized experiment has 25425 rows and 76 columns.
## The numbers of samples by condition are:
## 
##   het_dlgn het_retina    het_scn    ko_dlgn  ko_retina     ko_scn    wt_dlgn 
##          7          7          7          6          6          6         11 
##  wt_retina     wt_scn 
##         11          7
## The number of samples by batch are:
## 
## p08 p15 p60 
##  31  34   3
mm38_hisat_v2
## class: SummarizedExperiment 
## dim: 25425 68 
## metadata(7): notes title ... study researcher
## assays(1): ''
## rownames(25425): ENSMUSG00000000001 ENSMUSG00000000003 ...
##   ENSMUSG00001074846 ENSMUSG00002076083
## rowData names(15): ensembl_gene_id ensembl_transcript_id ...
##   uniprot_gn_symbol txid
## colnames(68): iprgc_62 iprgc_63 ... iprgc_129 iprgc_130
## colData names(76): rownames sampleid ... umi_dedup_mean_umi_per_pos
##   umi_dedup_max_umi_per_pos
mm38_hisat_v3 <- create_se(sample_sheet, gene_info = mm_annot,
                           file_column = "umi_dedup_output_count") %>%
  set_conditions(fact = "geno_loc_atb") %>%
  set_batches(fact = "time_atb") %>%
  set_colors(color_choices[["geno_loc"]])
## Reading the sample metadata.
## Checking the state of the condition column.
## Checking the state of the batch column.
## Checking the condition factor.
## The sample definitions comprises: 69 rows(samples) and 76 columns(metadata fields).
## Warning in create_se(sample_sheet, gene_info = mm_annot, file_column =
## "umi_dedup_output_count"): Some samples were removed when cross referencing the
## samples against the count data.
## Matched 25404 annotations and counts.
## Some annotations were lost in merging, setting them to 'undefined'.
## The final summarized experiment has 25425 rows and 76 columns.
## The numbers of samples by condition are:
## 
##   het_dlgn het_retina    het_scn    ko_dlgn  ko_retina     ko_scn    wt_dlgn 
##          7          7          7          6          6          6         11 
##  wt_retina     wt_scn 
##         11          7
## The number of samples by batch are:
## 
## p08 p15 p60 
##  31  34   3
mm38_hisat_v3
## class: SummarizedExperiment 
## dim: 25425 68 
## metadata(7): notes title ... study researcher
## assays(1): ''
## rownames(25425): ENSMUSG00000000001 ENSMUSG00000000003 ...
##   ENSMUSG00001074846 ENSMUSG00002076083
## rowData names(15): ensembl_gene_id ensembl_transcript_id ...
##   uniprot_gn_symbol txid
## colnames(68): iprgc_62 iprgc_63 ... iprgc_129 iprgc_130
## colData names(76): rownames sampleid ... umi_dedup_mean_umi_per_pos
##   umi_dedup_max_umi_per_pos
all_fact <- paste0(colData(mm38_hisat_v3)[["time_atb"]], "_",
                   colData(mm38_hisat_v3)[["geno_loc_atb"]])
colData(mm38_hisat_v3)[["time_geno_loc"]] <- all_fact

Note the end of the previous block, I created a factor out of the combination of time, genotype, and location. In a future invocation of this notebook, I will change the pairwise comparisons to add each of these three factors to the statistical model instead of this. The code to do that is not quite ready yet.

8 Non-zero Counts per Sample

Let’s look at the number of non-zero genes for all samples versus the coverage.

As above, this does not get run because I did not copy the count tables.

v1_nonzero <- plot_nonzero(mm38_hisat_v1)
v1_nonzero

But these do!

plot_legend(mm38_hisat_v2)
## The colors used in the expressionset are: #004008, #08519C, #238B45, #4292C6, #74c476, #980043, #9BCAE1, #C994C7, #E7298A.

v2_nonzero  <- plot_nonzero(mm38_hisat_v2, y_intercept = 0.65)
## The following samples have less than 16526.25 genes.
##  [1] "iprgc_62"  "iprgc_63"  "iprgc_64"  "iprgc_66"  "iprgc_67"  "iprgc_68" 
##  [7] "iprgc_70"  "iprgc_71"  "iprgc_72"  "iprgc_73"  "iprgc_74"  "iprgc_75" 
## [13] "iprgc_77"  "iprgc_78"  "iprgc_80"  "iprgc_81"  "iprgc_82"  "iprgc_83" 
## [19] "iprgc_84"  "iprgc_85"  "iprgc_86"  "iprgc_87"  "iprgc_88"  "iprgc_89" 
## [25] "iprgc_90"  "iprgc_91"  "iprgc_92"  "iprgc_93"  "iprgc_94"  "iprgc_95" 
## [31] "iprgc_96"  "iprgc_97"  "iprgc_98"  "iprgc_100" "iprgc_102" "iprgc_104"
## [37] "iprgc_105" "iprgc_106" "iprgc_107" "iprgc_108" "iprgc_110" "iprgc_111"
## [43] "iprgc_112" "iprgc_113" "iprgc_114" "iprgc_115" "iprgc_117" "iprgc_118"
## [49] "iprgc_121" "iprgc_123" "iprgc_124" "iprgc_125" "iprgc_126" "iprgc_127"
## [55] "iprgc_128" "iprgc_129" "iprgc_130"
## Scale for colour is already present.
## Adding another scale for colour, which will replace the existing scale.
## Scale for fill is already present.
## Adding another scale for fill, which will replace the existing scale.
## Warning: Using `size` aesthetic for lines was deprecated in ggplot2 3.4.0.
## i Please use `linewidth` instead.
## i The deprecated feature was likely used in the hpgltools package.
##   Please report the issue to the authors.
## This warning is displayed once per session.
## Call `lifecycle::last_lifecycle_warnings()` to see where this warning was
## generated.
v2_nonzero
## A non-zero genes plot of 68 samples.
## These samples have an average 13.7 CPM coverage and 15744 genes observed, ranging from 13692 to
## 17083.

pp(file = "01diagnostic_images/nonzero_v2_unfiltered.pdf")
## Warning in pp(file = "01diagnostic_images/nonzero_v2_unfiltered.pdf"): The
## directory: 01diagnostic_images does not exist, will attempt to create it.
v2_nonzero[["plot"]]
plotted <- dev.off()

v3_nonzero  <- plot_nonzero(mm38_hisat_v3, y_intercept = 0.65)
## The following samples have less than 16526.25 genes.
##  [1] "iprgc_62"  "iprgc_63"  "iprgc_64"  "iprgc_66"  "iprgc_67"  "iprgc_68" 
##  [7] "iprgc_70"  "iprgc_71"  "iprgc_72"  "iprgc_73"  "iprgc_74"  "iprgc_75" 
## [13] "iprgc_77"  "iprgc_78"  "iprgc_80"  "iprgc_81"  "iprgc_82"  "iprgc_83" 
## [19] "iprgc_84"  "iprgc_85"  "iprgc_86"  "iprgc_87"  "iprgc_88"  "iprgc_89" 
## [25] "iprgc_90"  "iprgc_91"  "iprgc_92"  "iprgc_93"  "iprgc_94"  "iprgc_95" 
## [31] "iprgc_96"  "iprgc_97"  "iprgc_98"  "iprgc_100" "iprgc_102" "iprgc_104"
## [37] "iprgc_105" "iprgc_106" "iprgc_107" "iprgc_108" "iprgc_110" "iprgc_111"
## [43] "iprgc_112" "iprgc_113" "iprgc_114" "iprgc_115" "iprgc_117" "iprgc_118"
## [49] "iprgc_119" "iprgc_121" "iprgc_123" "iprgc_124" "iprgc_125" "iprgc_126"
## [55] "iprgc_127" "iprgc_128" "iprgc_129" "iprgc_130"
## Scale for colour is already present.
## Adding another scale for colour, which will replace the existing scale.
## Scale for fill is already present.
## Adding another scale for fill, which will replace the existing scale.
v3_nonzero
## A non-zero genes plot of 68 samples.
## These samples have an average 4.803 CPM coverage and 15787 genes observed, ranging from 13868 to
## 17101.

pp(file = "01diagnostic_images/nonzero_v3_unfiltered.pdf")
v3_nonzero[["plot"]]
plotted <- dev.off()

Oh wow, I did not expect such a profound effect on the cpm values on the more saturated libraries. I guess in retrospect I should have?

Also note to self, we are not messing with p60.

8.1 Exclude p60

mm38_hisat_v2 <- subset_se(mm38_hisat_v2, subset = "time_atb!='p60'")
mm38_hisat_v3 <- subset_se(mm38_hisat_v3, subset = "time_atb!='p60'")

8.2 Replot the nonzero gene plots

v2_nonzero_filt <- plot_nonzero(mm38_hisat_v2, plot_labels = FALSE)
## The following samples have less than 16526.25 genes.
##  [1] "iprgc_62"  "iprgc_63"  "iprgc_64"  "iprgc_66"  "iprgc_67"  "iprgc_68" 
##  [7] "iprgc_70"  "iprgc_71"  "iprgc_72"  "iprgc_73"  "iprgc_74"  "iprgc_75" 
## [13] "iprgc_77"  "iprgc_81"  "iprgc_82"  "iprgc_83"  "iprgc_84"  "iprgc_85" 
## [19] "iprgc_86"  "iprgc_87"  "iprgc_88"  "iprgc_89"  "iprgc_90"  "iprgc_91" 
## [25] "iprgc_92"  "iprgc_93"  "iprgc_94"  "iprgc_95"  "iprgc_96"  "iprgc_97" 
## [31] "iprgc_98"  "iprgc_100" "iprgc_102" "iprgc_104" "iprgc_105" "iprgc_106"
## [37] "iprgc_107" "iprgc_108" "iprgc_110" "iprgc_111" "iprgc_112" "iprgc_113"
## [43] "iprgc_114" "iprgc_115" "iprgc_117" "iprgc_118" "iprgc_121" "iprgc_123"
## [49] "iprgc_124" "iprgc_125" "iprgc_126" "iprgc_127" "iprgc_128" "iprgc_129"
## [55] "iprgc_130"
## Scale for colour is already present.
## Adding another scale for colour, which will replace the existing scale.
## Scale for fill is already present.
## Adding another scale for fill, which will replace the existing scale.
## Not putting labels on the plot.
pp(file = "01diagnostic_images/nonzero_v2_filt.pdf")
v2_nonzero_filt[["plot"]]
plotted <- dev.off()

v3_nonzero_filt <- plot_nonzero(mm38_hisat_v3, plot_labels = FALSE)
## The following samples have less than 16526.25 genes.
##  [1] "iprgc_62"  "iprgc_63"  "iprgc_64"  "iprgc_66"  "iprgc_67"  "iprgc_68" 
##  [7] "iprgc_70"  "iprgc_71"  "iprgc_72"  "iprgc_73"  "iprgc_74"  "iprgc_75" 
## [13] "iprgc_77"  "iprgc_81"  "iprgc_82"  "iprgc_83"  "iprgc_84"  "iprgc_85" 
## [19] "iprgc_86"  "iprgc_87"  "iprgc_88"  "iprgc_89"  "iprgc_90"  "iprgc_91" 
## [25] "iprgc_92"  "iprgc_93"  "iprgc_94"  "iprgc_95"  "iprgc_96"  "iprgc_97" 
## [31] "iprgc_98"  "iprgc_100" "iprgc_102" "iprgc_104" "iprgc_105" "iprgc_106"
## [37] "iprgc_107" "iprgc_108" "iprgc_110" "iprgc_111" "iprgc_112" "iprgc_113"
## [43] "iprgc_114" "iprgc_115" "iprgc_117" "iprgc_118" "iprgc_119" "iprgc_121"
## [49] "iprgc_123" "iprgc_124" "iprgc_125" "iprgc_126" "iprgc_127" "iprgc_128"
## [55] "iprgc_129" "iprgc_130"
## Scale for colour is already present.
## Adding another scale for colour, which will replace the existing scale.
## Scale for fill is already present.
## Adding another scale for fill, which will replace the existing scale.
## Not putting labels on the plot.
pp(file = "01diagnostic_images/nonzero_v3_filt.pdf")
v3_nonzero_filt[["plot"]]
plotted <- dev.off()

Once again, I do not want to lose the previous code, so here is the v1 invocation

mm38_hisat_v1 <- subset_se(mm38_hisat_v1, subset = "time_atb!='p60'")

9 Quick PCA, then return to Theresa’s document

v2_norm <- normalize(mm38_hisat_v2, transform = "log2", convert = "cpm",
                     norm = "quant", filter = TRUE)
## Removing 10298 low-count genes (15127 remaining).
## transform_counts: Found 8465 values equal to 0, adding 1 to the matrix.
v2_norm_pca <- plot_pca(v2_norm)
v2_norm_pca
## The result of performing a fast_svd dimension reduction.
## The x-axis is PC1 and the y-axis is PC2
## Colors are defined by het_dlgn, het_retina, het_scn, ko_dlgn, ko_retina, ko_scn, wt_dlgn, wt_retina, wt_scn
## Shapes are defined by p08, p15.

pp(file = "01diagnostic_images/v2_norm_pca.pdf")
v2_norm_pca[["plot"]]
plotted <- dev.off()

v3_norm <- normalize(mm38_hisat_v3, transform = "log2", convert = "cpm",
                     norm = "quant", filter = TRUE)
## Removing 10156 low-count genes (15269 remaining).
## transform_counts: Found 9347 values equal to 0, adding 1 to the matrix.
v3_norm_pca <- plot_pca(v3_norm)
v3_norm_pca
## The result of performing a fast_svd dimension reduction.
## The x-axis is PC1 and the y-axis is PC2
## Colors are defined by het_dlgn, het_retina, het_scn, ko_dlgn, ko_retina, ko_scn, wt_dlgn, wt_retina, wt_scn
## Shapes are defined by p08, p15.

pp(file = "01diagnostic_images/v3_norm_pca.pdf")
v3_norm_pca[["plot"]]
plotted <- dev.off()

Ibid.

v1_norm <- normalize(mm38_hisat_v1, transform = "log2", convert = "cpm",
                     norm = "quant", filter = TRUE)
plot_pca(v1_norm)

To my eyes it looks like we just have 1 weirdo p15 sample? Deduplication had a minor but significant effect on the PCA.

With that in mind, let us look at Theresa’s WORKING document and see what we can recapitulate.

Theresa’s document: The TRAP protocol has some variability which is introduced at different stpdf including homogenization, antibody labeling, pulldown efficiency/specificity, sample handling during cleanup, and library prep/sequencing. We know from Rashmi’s QC that there is variability at the level of pulldown efficiency (amount of RNA isolated). She is doing a good job of keeping track of this for all her samples and we have validated her P8 results (attached supplementary figure 3D). We consistently see clear differences between control and cre samples for the retina, which makes sense because the cell bodies are in the retina. The target tissue differences are smaller, which also makes sense for axon-TRAP. We think that some of her P15 samples are not good based on low amounts of isolated RNA from cre(+) retina samples. We plan to drop these samples and not perform additional isolations at this time point. Based on this (and the general lack of large developmental effects), we were planning to focus on presenting the P8 data only in the paper. Interested to hear your thoughts in this…

My notes: Theresa’s first operations in this notebook were to:

  1. Set location as condition, genotype as batch.
  2. Perform PCA before/after sva.
v3_loc_geno <- set_conditions(mm38_hisat_v3, fact = "location_atb",
                              colors = color_choices[["location"]]) %>%
  set_batches(fact = "genotype_atb")
## The numbers of samples by condition are:
## 
##   dlgn retina    scn 
##     23     23     19
## The number of samples by batch are:
## 
## het  ko  wt 
##  21  18  26

9.1 The associated PCA

At different times, it appears to me that Theresa has preferred slightly different normalization methods, primarily a mix of TMM and quantile.

Thus I will use different suffix letters to denote various normalizations employed, and if they turn out the same I will pick one arbitrarily.

loc_geno_nq <- normalize(v3_loc_geno, transform = "log2", convert = "cpm",
                         filter = TRUE, norm = "quant")
## Removing 10156 low-count genes (15269 remaining).
## transform_counts: Found 9347 values equal to 0, adding 1 to the matrix.
location_genotype_pca <- plot_pca(loc_geno_nq)
pp(file = "01diagnostic_images/location_genotype_norm_pca.pdf")
location_genotype_pca[["plot"]]
plotted <- dev.off()
location_genotype_pca
## The result of performing a fast_svd dimension reduction.
## The x-axis is PC1 and the y-axis is PC2
## Colors are defined by dlgn, retina, scn
## Shapes are defined by het, ko, wt.

## ok, I have two weirdo samples which look very much like they are actually dlgn.
## These are sample IDs iprgc_66 and iprgc_130

loc_geno_nt <- normalize(v3_loc_geno, transform = "log2", convert = "cpm",
                         filter = TRUE, norm = "tmm")
## Removing 10156 low-count genes (15269 remaining).
## transform_counts: Found 42869 values equal to 0, adding 1 to the matrix.
location_genotype_tmm_pca <- plot_pca(loc_geno_nt)
pp(file = "01diagnostic_images/location_genotype_tmm_pca.pdf")
location_genotype_tmm_pca[["plot"]]
## Warning in MASS::cov.trob(data[, vars], wt = weight * nrow(data)): Probable
## convergence failure
## Warning in MASS::cov.trob(data[, vars], wt = weight * nrow(data)): Probable
## convergence failure
plotted <- dev.off()
location_genotype_tmm_pca
## The result of performing a fast_svd dimension reduction.
## The x-axis is PC1 and the y-axis is PC2
## Colors are defined by dlgn, retina, scn
## Shapes are defined by het, ko, wt.
## Warning in MASS::cov.trob(data[, vars], wt = weight * nrow(data)): Probable
## convergence failure
## Warning in MASS::cov.trob(data[, vars], wt = weight * nrow(data)): Probable
## convergence failure

A random thought about these PCA plots, it might be worth while to add a panel below the legend with the sample numbers per condition/batch.

Of course, the same information is provided in a more fun fashion via my silly sankey function:

sample_sankey <- plot_meta_sankey(v3_loc_geno, color_choices = color_choices,
                                  factors = c("genotype_atb", "location_atb", "time_atb"))
## Warning: attributes are not identical across measure variables; they will be
## dropped
## Warning: The `size` argument of `element_rect()` is deprecated as of ggplot2 3.4.0.
## i Please use the `linewidth` argument instead.
## i The deprecated feature was likely used in the ggsankey package.
##   Please report the issue at <https://github.com/davidsjoberg/ggsankey/issues>.
## This warning is displayed once per session.
## Call `lifecycle::last_lifecycle_warnings()` to see where this warning was
## generated.
pp(file = "01diagnostic_images/design_sankey.pdf")
sample_sankey[["ggplot"]]
plotted <- dev.off()
sample_sankey
## A sankey plot describing the metadata of 65 samples,
## including 30 out of 0 nodes and traversing metadata factors:
## genotype_atb, location_atb, time_atb.

10 A Short conversation with Rashmi

Rashmi came by and we discussed the samples a little. She suggested that is likely that we will need to exclude the 202205 samples, these may be identified by a few ways, most easily I think via the ‘project_ah’ column, they are the 021_1 samples.

My sense was that she concurred with my interpretation of the umi deduplication, so I will continue using the deduplicated results exclusively, at least for now.

11 Melanopsin Sanity Check

One of Theresa’s first checks was wisely for melanopsin. Let us repeat a version of this:

An important note: Indrajeet Patil removed the groupedstats and its associated plotting library from CRAN/github/etc. I am not certain what happened, but that necessitates a change in how I plot this.

opn4_exprs <- data.frame(combined = colData(loc_geno_nt)[["geno_loc_atb"]],
                         location = colData(loc_geno_nt)[["location_atb"]],
                         genotype = colData(loc_geno_nt)[["genotype_atb"]],
                         opn = assay(loc_geno_nt)["ENSMUSG00000021799", ])

## groupedstats::grouped_summary(opn4_exprs, location, opn)
## opn4_location <- ggbetweenstats(data = opn4_exprs, x = location, y = opn)
## pp(file = "images/ggbetween_location.pdf")
## opn4_location
## plotted <- dev.off()
## opn4_location

## opn4_genotype <- ggbetweenstats(data = opn4_exprs, x = genotype, y = opn)
## pp(file = "images/ggbetween_location.pdf")
## opn4_genotype
## plotted <- dev.off()
## opn4_genotype

## opn4_combined <- ggbetweenstats(data = opn4_exprs, x = combined, y = opn)
## pp(file = "images/ggbetween_combined.pdf")
## opn4_combined
## plotted <- dev.off()
## opn4_combined

ok, so I plotted the question a bit differently, but got the same answer.

Here is the text of Theresa’s notebook following this analysis:

“Ugh oh, looks like there is at least one retina KO sample that has some melanopsin expression in it. Turns out ipRGC_07 is a bad egg which is supposed to be a KO but has melanopsin expression. It’s friends which were pooled from the same mice are iprgc_06 and iprgc_08, so we need to exclude all these samples.”

I am also seeing some knockout expression with some caveats: I do not have the affected samples in my dataset (iprgc_07) and the levels I am seeing are quite low – I will look in IGV to double check, but I strongly suspect that these are some piddly reads near the UTRs.

Onward!

12 PCA plots

12.1 PCA of all genes by location

Theresa’s first pca was of log2 cpm values. I might add quantile/tmm to this?

v3_location <- set_conditions(mm38_hisat_v3, fact = "location_atb") %>%
  set_batches(fact = "genotype_atb") %>%
  set_colors(color_choices[["location"]])
## The numbers of samples by condition are:
## 
##   dlgn retina    scn 
##     23     23     19
## The number of samples by batch are:
## 
## het  ko  wt 
##  21  18  26
v3_location_norm <- normalize(v3_location, filter = TRUE, norm = "quant",
                                   transform = "log2", convert = "cpm")
## Removing 10156 low-count genes (15269 remaining).
## transform_counts: Found 9347 values equal to 0, adding 1 to the matrix.
v3_location_pca <- plot_pca(v3_location_norm)
pp(file = "01diagnostic_images/v3_location_norm_pca.pdf")
v3_location_pca
## The result of performing a fast_svd dimension reduction.
## The x-axis is PC1 and the y-axis is PC2
## Colors are defined by dlgn, retina, scn
## Shapes are defined by het, ko, wt.
dev.off()
## png 
##   2
v3_location_pca
## The result of performing a fast_svd dimension reduction.
## The x-axis is PC1 and the y-axis is PC2
## Colors are defined by dlgn, retina, scn
## Shapes are defined by het, ko, wt.

Once again we see that samples iprgc_66 and iprgc_130 are likely actually DLGN and not SCN. I am therefore going to add a column to the sample sheet noting this, and remove them from the expressionset.

I will thus replot the data after removing those two. If we want to see what it looks like with the re-attributed locations, we can do so.

Theresa has a nice change to the PCA plotter in which she sets the alpha channel as an additional visual queue for a metadata factor…

mm38_hisat_v3 <- subset_se(mm38_hisat_v3, subset="sampleid!='iprgc_130'") %>%
  subset_se(subset="sampleid!='iprgc_66'")
v3_location <- set_conditions(mm38_hisat_v3, fact = "location_atb") %>%
  set_batches(fact = "genotype_atb") %>%
  set_colors(color_choices[["location"]])
## The numbers of samples by condition are:
## 
##   dlgn retina    scn 
##     23     23     17
## The number of samples by batch are:
## 
## het  ko  wt 
##  20  18  25
v3_location_norm <- normalize(v3_location, filter = TRUE, norm = "quant",
                                   transform = "log2", convert = "cpm")
## Removing 10162 low-count genes (15263 remaining).
## transform_counts: Found 8867 values equal to 0, adding 1 to the matrix.
filtered_location_pca <- plot_pca(v3_location_norm)
pp(file = "02filtered_images/filtered_location_pca.pdf")
## Warning in pp(file = "02filtered_images/filtered_location_pca.pdf"): The
## directory: 02filtered_images does not exist, will attempt to create it.
filtered_location_pca[["plot"]]
plotted <- dev.off()
filtered_location_pca
## The result of performing a fast_svd dimension reduction.
## The x-axis is PC1 and the y-axis is PC2
## Colors are defined by dlgn, retina, scn
## Shapes are defined by het, ko, wt.

removed_sankey <- plot_meta_sankey(v3_location, color_choices = color_choices,
                                   factors = c("genotype_atb", "location_atb", "time_atb"))
## Warning: attributes are not identical across measure variables; they will be
## dropped
pp(file = "02filtered_images/filtered_sankey.pdf")
removed_sankey[["ggplot"]]
plotted <- dev.off()
removed_sankey
## A sankey plot describing the metadata of 63 samples,
## including 30 out of 0 nodes and traversing metadata factors:
## genotype_atb, location_atb, time_atb.

Here is Theresa’s text, recall once again that I do not have some of these older samples (iprgc_62):

PC1 vs PC2 identifies retina vs axon is still the main component of variation. We do see though that in the PC2 direction, we see with the new samples added, we don’t see separation based on axonal targets (dLGN vs SCN). In the PC1 vs PC3 plot, we see that it’s PC3 where we start to see variation correlated with axonal compartment. Let’s look at PC1 vs PC2 colored by batch (when they were processed/sequenced) to see if that is what is contributing so much variation in PC2.

Side note: ipRGC 62 seems like an odd ball. This seems to me like it should have been a dLGN P08 sample. Is there any possibility this got mislabeled early on? I went back and double checked to see if all my processing is correct and it indeed was labeled an SCN P15 from the time I got the samples, and it is indeed.

13 DE

I now switched to Theresa’s document ‘WORKING_axonTRAP…’ and will start pulling sections from it. I am reasonably certain I have reasonably similar sample distributions, so I presume I can invoke similar/identical calls for DESeq and friends.

13.1 p8 retinas

In the block immediately before the DE analyses, Theresa created a subset expressionset of only p08 retinas. Thus this initial DE I assume will be used to subtract for the SCN/DLGN analyses that follow. (I guess I could read ahead and find out, but no! I want to be a blank slate)

Theresa’s primary workflow makes heavy use of DESeq2 (Love, Huber, and Anders (2014)) and sva (Leek et al. (2012)). In some(most?) of Theresa’s invocations of the all_pairwise() function, she excludes the other methods that it performs. In this workbook, I left those methods on, thus we can evaluate the relative performance DESeq2 vs. some (all? I may have disabled EBSeq/dream because they were taking too long) of the following:

  • limma: (Ritchie et al. (2015)) (among other references) originally written for microarrays.
  • EdgeR: (Robinson, McCarthy, and Smyth (2010)), which shares many assumptions with DESeq2.
  • EBSeq: (Leng et al. (2013)), because I have a soft spot for any Bayesian method.
  • Noiseq: (Tarazona et al. (2011)), which seeks to directly model variance in an RNASeq dataset and use that to improve the sensitivity of the result, much like:
  • Dream: (Gabriel E. Hoffman and Roussos (2020)), written by the same authors (and uses very similar logic) as one of my favorite tools, variancePartition(Gabriel E. Hoffman and Schadt (2016)).
mm38_p8_retina <- subset_se(mm38_hisat_v3, subset = "time_atb=='p08' & location_atb=='retina'")
mm_normal_p8_ret_de <- all_pairwise(mm38_p8_retina, model_svs = "svaseq",
                                    model_fstring = "~ 0 + condition", filter = TRUE)
## het_retina  ko_retina  wt_retina 
##          3          3          5
## Removing 12001 low-count genes (13424 remaining).
## Basic step 0/3: Normalizing data.
## Basic step 0/3: Converting data.
## I think this is failing? SummarizedExperiment
## Basic step 0/3: Transforming data.
## Setting 2593 entries to zero.
## This received a matrix of SVs.
## converting counts to integer mode
## gene-wise dispersion estimates
## mean-dispersion relationship
## final dispersion estimates
## conditions
## het_retina  ko_retina  wt_retina 
##          3          3          5
## conditions
## het_retina  ko_retina  wt_retina 
##          3          3          5
## conditions
## het_retina  ko_retina  wt_retina 
##          3          3          5

mm_normal_p8_ret_de
## A pairwise differential expression with results from: basic, deseq, ebseq, edger, limma, noiseq.
## This used a surrogate/batch estimate from: svaseq.
## The primary analysis performed 3 comparisons.

There seems to be a discrepency with previous iterations of this. Let us simplify to just doing deseq and find what is causing it. In my previous iteration, I got 3632 genes in the unique(c()) or het+ko.

deseq_only <- deseq_pairwise(mm38_p8_retina, model_svs = "svaseq",
                             model_fstring = default_fstring, filter = TRUE)
## converting counts to integer mode
## gene-wise dispersion estimates
## mean-dispersion relationship
## final dispersion estimates
deseq_hetkeeper_genes <- deseq_only$all_tables$wt_retina_vs_het_retina %>%
  filter(logFC <= -0.25 & adj.P.Val <= 0.05)
deseq_kokeeper_genes <- deseq_only$all_tables$wt_retina_vs_ko_retina %>%
  filter(logFC <= -0.25 & adj.P.Val <= 0.05)
deseq_keepergenes <- unique(c(rownames(deseq_hetkeeper_genes),
                        rownames(deseq_kokeeper_genes)))
length(deseq_keepergenes)
## [1] 3632
deseq_pair_hetkeeper_genes <- mm_normal_p8_ret_de$deseq$all_tables$wt_retina_vs_het_retina %>%
  filter(logFC <= -0.25 & adj.P.Val <= 0.05)
deseq_pair_kokeeper_genes <- mm_normal_p8_ret_de$deseq$all_tables$wt_retina_vs_ko_retina %>%
  filter(logFC <= -0.25 & adj.P.Val <= 0.05)
deseq_pair_keepergenes <- unique(c(rownames(deseq_pair_hetkeeper_genes),
                        rownames(deseq_pair_kokeeper_genes)))
length(deseq_pair_keepergenes)
## [1] 3632

The following invocation performed by Theresa filters the wt/het comparison for only those genes which increased by at least 0.25 logFC with a significant adjusted p-value. I assume that this is to use the wt samples as a translational control for the ket/ko comparisons; I am therefore thinking that for my purposes, I will therefore separate the contrasts from all_pairwise do this in a stepwise fashion…

The block of code immediately following Theresa’s all_pairwise() invocation is a little confusing for me and warrants some explanation by me to me in the hopes that I do not misunderstand what is happening and the goals therein.

I think I can safely assume that the goal here is to pull out the IDs which increased in het with respect to wild type; even if by a small margin, as long as it is statistically significant vis a vis the adjusted p-value.

I am going to perform what I think is the same thing in a slightly different fashion so that I can share a copy of the results with whomever is interested. I will also repeat Theresa’s invocation and prove to myself that I understood and got the same answer.

wt_het_keeper <- list("het_vs_wt" = c("het_retina", "wt_retina"))
het_wt_table <- combine_de_tables(
  mm_normal_p8_ret_de, keepers = wt_het_keeper, label_column = label_column,
  excel = "03theresa_comparison_excel/het_retina_control.xlsx")
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
wanted_sig <- extract_significant_genes(
  het_wt_table, lfc = 0.25, according_to = "deseq",
  excel = "03theresa_comparison_excel/het_retina_control-sig.xlsx")
wanted_het_increased <- wanted_sig[["deseq"]][["ups"]][["het_vs_wt"]]
increased_het_genes <- rownames(wanted_het_increased)

Here are Theresa’s next lines:

mm_de_normal_p8_ret <- mm_normal_p8_ret_de
hetkeeper_genes <- mm_de_normal_p8_ret$deseq$all_tables$wt_retina_vs_het_retina %>%
  filter(logFC <= -0.25 & adj.P.Val <= 0.05)
kokeeper_genes <- mm_de_normal_p8_ret$deseq$all_tables$wt_retina_vs_ko_retina %>%
  filter(logFC <= -0.25 & adj.P.Val <= 0.05)
keepergenes <- unique(c(rownames(hetkeeper_genes),
                        rownames(kokeeper_genes)))
## We know a priori that Opn4 is ENSMUSG00000021799
## I do not expect to see it in this set, it should be higher in wt
## retina vs ko retina by a significant margin.
"ENSMUSG00000021799" %in% keepergenes
## [1] TRUE
## Oooohhh but it _is_ higher in het vs. wt, as we saw in
## the violin plot earlier.

I think Rashmi made a compelling point which illustrates why we likely should expect the expression of Opn4 to significantly higher in the heterozygotes vs wild-type:

  1. Recall that the assay is using the immunopurification to extract the RNAs.
  2. The wt samples do not have the cre recombinase and therefore no HA and therefore everything we observe is due to non-specific binding.
  3. The set of genes observed due to non-specific binding is different than het/ko (presumably a larger number of relatively small values), therefore the divisor performed in the cpm is likely relativly large resulting in normalized values getting shifted down to some degree.
  4. On the other hand, the set of genes observed in het/ko are more likely to be only the specific binders and therefore smaller (I can test this) resulting in a smaller divisor and slight shifting up in the cpm values.

This makes me wonder if any normalization methods exist which do something like multiply the values by some value related to the proportion of observed genes; and/or if this is a good/bad/indifferent idea.

Also, just a note for me to remember: RPL22, not RPS22, for some reason I keep thinking the small subunit.

13.2 Prove I understood

hetkeeper_genes <- mm_normal_p8_ret_de$deseq$all_tables$wt_retina_vs_het_retina %>%
  filter(logFC <= -0.25 & adj.P.Val <= 0.05)
testthat::expect_true(nrow(hetkeeper_genes) == length(increased_het_genes))
taa_keepers <- sort(rownames(hetkeeper_genes))
atb_keepers <- sort(increased_het_genes)
testthat::expect_equal(taa_keepers, atb_keepers)

Yay! I can read! Now let us repeat for the KO vs wt

wt_ko_keeper <- list("ko_vs_wt" = c("ko_retina", "wt_retina"))
ko_wt_table <- combine_de_tables(
  mm_normal_p8_ret_de, keepers = wt_ko_keeper, label_column = label_column,
  excel = "03theresa_comparison_excel/ko_retina_control.xlsx")
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
wanted_sig <- extract_significant_genes(
  ko_wt_table, lfc = 0.25, according_to = "deseq",
  excel = "03theresa_comparison_excel/ko_retina_control-sig.xlsx")

wanted_ko_increased <- wanted_sig[["deseq"]][["ups"]][["ko_vs_wt"]]
increased_ko_genes <- rownames(wanted_ko_increased)

The next thing performed in Theresa’s document is a unique(concatenation of these two gene groups), thus sucking up every gene which was significantly higher in either the knockout or heterzyous samples with respect to wild-type.

This was followed by a couple of merge operations of a little bit of the annotation data; I am not sure I understand the goal yet…

Here is her code. I copied the annotation ‘mgi_symbol’ column to ‘external_gene_name’ so that I need not change any of her code. I am assuming this is the appropriate column of interest, I do not know this for certain, but it seems quite likely.

While I am at it, here is the set_sig_limma() function from Theresa’s helpers.R

set_sig_limma <- function(limma_tbl, factors = NULL) {
  if (is.null(factors)) {
    #set significance for plotting colors
    limma_tbl$Significance <- NA
    limma_tbl[abs(limma_tbl$logFC) < 1 | limma_tbl$adj.P.Val > .05, "Significance"] <- "Not \nEnriched"
    limma_tbl[limma_tbl$logFC >= 1  & limma_tbl$adj.P.Val <= .05, ][["Significance"]] <- "Disease \nUpregulated"
    limma_tbl[limma_tbl$logFC <= -1  & limma_tbl$adj.P.Val <= .05, ][["Significance"]] <- "Disease \nDownregulated"
    limma_tbl$Significance <- factor(limma_tbl$Significance, levels = c("Upregulated", "Downregulated",  "Not \nEnriched"))
  } else {
    limma_tbl$Significance <- NA
    limma_tbl[abs(limma_tbl$logFC) < 1 | limma_tbl$adj.P.Val > .05, "Significance"] <- "Not \nEnriched"
    if(nrow(limma_tbl[limma_tbl$logFC >= 1  & limma_tbl$adj.P.Val <= .05, ]) != 0) {
      limma_tbl[limma_tbl$logFC >= 1  & limma_tbl$adj.P.Val <= .05, ][["Significance"]] <- factors[1]
    }
    if (nrow(limma_tbl[limma_tbl$logFC <= -1  & limma_tbl$adj.P.Val <= .05, ]) != 0) {
      limma_tbl[limma_tbl$logFC <= -1  & limma_tbl$adj.P.Val <= .05, ][["Significance"]] <- factors[2]
    }
    limma_tbl$Significance <- factor(limma_tbl$Significance, levels = c(factors,  "Not \nEnriched"))
  }
  return(limma_tbl)
}

13.2.1 Combining het/wt and ko/wt

mm_annot[["external_gene_name"]] <- mm_annot[["mgi_symbol"]]
keepergenes <- unique(c(rownames(hetkeeper_genes), rownames(kokeeper_genes)))
length(keepergenes)
## [1] 3632
annots_to_merge <- mm_annot %>%
  select(ensembl_gene_id, external_gene_name) %>%
  filter(ensembl_gene_id %in%
           rownames(mm_de_normal_p8_ret$deseq$all_tables$ko_retina_vs_het_retina)) %>%
  distinct()
mm_de_normal_p8_ret$deseq$all_tables$ko_retina_vs_het_retina <- merge(
  mm_de_normal_p8_ret$deseq$all_tables$ko_retina_vs_het_retina, annots_to_merge,
  by.x = 0, by.y = "ensembl_gene_id", all.x = TRUE)
df <- mm_de_normal_p8_ret$deseq$all_tables$ko_retina_vs_het_retina %>%
  dplyr::mutate(logFC = -logFC) %>%
  set_sig_limma(factors = c("Het Enriched", "KO Enriched"))

My version of the above task makes use of the excludes option of combine_de_tabes. Given the set of unique gene IDs increased in the het/ko, I can ask to exlude anything not in that set. I could also have more parsimoniously directly excluded any gene ID increased in the wt samples. But, Theresa already provided the code to do the former, so it will be less typing/opportunity for silly mistakes to just do that.

both_increased_genes <- unique(c(increased_het_genes, increased_ko_genes))
## arbitrairly grab all genes from one of my data structures.
all_genes <- rownames(exprs(mm38_hisat_v3))
exclude_idx <- all_genes %in% both_increased_genes
summary(exclude_idx)
##    Mode   FALSE    TRUE 
## logical   21793    3632

My April 2025 version of this shows 21,793 and 3,632 genes in this set. Is that still true? (As of 20260311, it is!)

exclude_increased_genes <- all_genes[exclude_idx]
retina_keepers <- list(
  "het_vs_wt" = c("het_retina", "wt_retina"),
  "ko_vs_wt" = c("ko_retina", "wt_retina"),
  "ko_vs_het" = c("ko_retina", "het_retina"))
## A reminder to myself: there is also a parameter 'wanted_genes'
## which does effectively the same thing as excludes in this context;
## excludes was originally written to allow flexible, keyword-based
## exclusion.
p8_retina_tables <- combine_de_tables(
  mm_normal_p8_ret_de, keepers = retina_keepers,
  wanted_genes = both_increased_genes, label_column = label_column,
  excel = glue("03theresa_comparison_excel/p8_retina_kept_genes_increased_in_wt_tables-v{ver}.xlsx"))
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
p8_retina_sig <- extract_significant_genes(
  p8_retina_tables, according_to = "deseq",
  excel = glue("03theresa_comparison_excel/p8_retina_kept_genes_increased_in_wt_sig-v{ver}.xlsx"))

opposite_p8_retina_tables <- combine_de_tables(
  mm_normal_p8_ret_de, keepers = retina_keepers,
  excludes = both_increased_genes, label_column = label_column,
  excel = glue("03theresa_comparison_excel/p8_retina_removed_genes_increased_in_wt_tables-v{ver}.xlsx"))
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
opposite_p8_retina_sig <- extract_significant_genes(
  p8_retina_tables, according_to = "deseq",
  excel = glue("03theresa_comparison_excel/p8_retina_removed_genes_increased_in_wt_sig-v{ver}.xlsx"))

14 Filtering out non-specific genes and examining the results

The following is a copy/paste from Theresa containing the remaining tasks she performed and will provide the template for implementation of the final tasks.

This picks up with the lines from her notebook immediately following the invocation of ‘set_sig_limma(factors = c(“Het Enriched” …’.

For all of the remaining blocks I will copy in her code, turn off its evaluation, run the blocks manually, compare them to her notebook output, then enable each block as I ensure I understand it.

I will likely therefore introduce some small formatting changes and add some additional GSEA/enrichment tasks once the non-specific filtering is complete.

df <- df %>%
  filter(Row.names %in% keepergenes)
labels_ups <- df %>%
  filter(adj.P.Val <= 0.05 & abs(logFC) > 1) %>%
  arrange(logFC) %>%
  head(n = 9)
labels_downs <- df %>%
  filter(adj.P.Val <= 0.05 & abs(logFC) > 1) %>%
  arrange(-logFC) %>%
  head(n = 11)
labels <- rbind(labels_ups, labels_downs)
res_tbl <- df
DEplot <- ggplot(res_tbl, aes(x = logFC, y = -log10(adj.P.Val), label = external_gene_name)) +
  geom_point(aes(colour = Significance), size = 4) +
  geom_vline(xintercept = c(-1, 1)) +
  geom_hline(yintercept = -log10(0.05)) +
  theme_classic(base_size = 20) +
  xlab("log2(FC)") +
  ylab("-log10(p-value)") +
  theme(legend.position = "right") +
  scale_color_manual(values = c("#F8766D", "#00BFC4", "Grey")) +
  geom_label_repel(
    data = filter(df,
                  ## c('s5_het_dlgn', 's5_het_ret', 's5_het_scn')),
                  external_gene_name %in% labels$external_gene_name),
    ## nudge_x = -0.5,
    nudge_y = 3, max.overlaps = 15) +
  xlim(c(-3, 6))

pp(file = "03theresa_comparison_images/p08_retina_DE_1312024.pdf")
## Warning in pp(file = "03theresa_comparison_images/p08_retina_DE_1312024.pdf"):
## The directory: 03theresa_comparison_images does not exist, will attempt to
## create it.
DEplot
## Warning: Removed 2 rows containing missing values or values outside the scale range
## (`geom_point()`).
## Warning: Removed 2 rows containing missing values or values outside the scale range
## (`geom_label_repel()`).
plotted <- dev.off()
DEplot
## Warning: Removed 2 rows containing missing values or values outside the scale range
## (`geom_point()`).
## Removed 2 rows containing missing values or values outside the scale range
## (`geom_label_repel()`).

write_xlsx(df, excel = "excel/retinahet_vs_retinako_WTfiltered.xlsx")
## write_xlsx() wrote excel/retinahet_vs_retinako_WTfiltered.xlsx.
## The cursor is on sheet first, row: 3635 column: 13.

14.1 How many ups/downs

ko_enriched <- df %>%
  filter(Significance == "KO Enriched")
nrow(ko_enriched)
## [1] 21
het_enriched <-  df %>%
  filter(Significance == "Het Enriched")
nrow(het_enriched)
## [1] 69

14.2 category enrichment/GSEA

regulated_genes <- res_tbl %>%
  filter(adj.P.Val <= 0.05) %>%
  arrange(logFC) %>%
  select(Row.names, logFC, adj.P.Val, external_gene_name, Significance) %>%
  filter(abs(logFC) >= 1)
## gsea_result_ko <- gost(query = ko_genes$external_gene_name,
##                        organism = "mmusculus",
##                        evcodes = TRUE,
##                        ordered_query = TRUE)
gsea_result_het <- gost(query = het_enriched$external_gene_name,
                        organism = "mmusculus",
                        evcodes = TRUE,
                        ordered_query = TRUE)
##gsea_result_alldysregulated <- gost(query = alldysregulated_genes$external_gene_name,
##                                    organism = "mmusculus",
##                                    evcodes = TRUE,
##                                    ordered_query = TRUE)

I have a function in my package which seeks to make gProfiler queries a bit more complete and easy. Let us see how similar the result is…

rownames(alldysregulated_genes) <- alldysregulated_genes[["Row.names"]]
alldysregulated_genes[["Row.names"]] <- NULL

het_gp <- simple_gprofiler(rownames(alldysregulated_genes),
                           species = "mmusculus",
                           excel = glue("excel/het_gprofiler-v{ver}.xlsx"))
het_gp
enrichplot::dotplot(het_gp[["BP_enrich"]])
gp_pair <- enrichplot::pairwise_termsim(het_gp[["BP_enrich"]])
enrichplot::emapplot(gp_pair)
enrichplot::ssplot(gp_pair)
enrichplot::treeplot(gp_pair)
upsetplot(het_gp[["BP_enrich"]])

enrichplot::dotplot(het_gp[["REAC_enrich"]])
gp_pair <- enrichplot::pairwise_termsim(het_gp[["REAC_enrich"]])
enrichplot::emapplot(gp_pair)
enrichplot::ssplot(gp_pair)
enrichplot::treeplot(gp_pair)
upsetplot(het_gp[["REAC_enrich"]])

I make a somewhat arbitrary distinction between the concepts of over-enrichment analyses and GSEA: the former (as performed by gprofiler) (Raudvere et al. (2019)) seeks to find groups of genes overrepresented in GO/reactome/etc. These groups of genes are taken exclusively from the top-n/bottom-n genes with respect to fold-change between conditions of interest; in this case most different than wt in the p08 retina ko or het samples.

With that in mind, I can invoke a similar function using the full table of DE results to get what I call the GSEA result using clusterProfiler (Yu (n.d.)). In the following block I will use the ‘all_cprofiler’ function on the data structures named ‘p8_retina_tables’ and ‘opposite_p8_retina_tables’ in order to get these GSEA results for each contrast performed (het/wt, ko/wt, het/ko). I will follow that up with ‘all_gprofiler’ which does the same, but uses gProfiler’s enrichment analyses (it will therefore include what we just looked at).

p08_retina_all_cp <- all_cprofiler(
  p8_retina_sig, p8_retina_tables, orgdb = "org.Mm.eg.db", orgdb_from = orgdb_from,
  excel = "03theresa_comparison_excel/cprofiler_p08_retina.xlsx")
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error in `simple_cl[["kegg_universe"]]`:
## ! subscript out of bounds
enrichplot::dotplot(p08_retina_all_cp[["ko_vs_het_up"]][["go_data"]][["MF_enrich"]])
## Error in `h()`:
## ! error in evaluating the argument 'object' in selecting a method for function 'dotplot': object 'p08_retina_all_cp' not found
p08_topn_gsea <- plot_topn_gsea(p08_retina_all_cp)
## Error in `h()`:
## ! error in evaluating the argument 'gse' in selecting a method for function 'plot_topn_gsea': object 'p08_retina_all_cp' not found
pp(file = "03theresa_comparison_images/gsea_p08_retina_ko_vs_het_top_hit.pdf")
p08_topn_gsea[["GO_ko_vs_het_up"]][[1]]
## Error:
## ! object 'p08_topn_gsea' not found
plotted <- dev.off()

p08_topn_gsea[["GO_ko_vs_het_up"]][[1]]
## Error:
## ! object 'p08_topn_gsea' not found
p08_topn_gsea[["GO_ko_vs_het_up"]][[2]]
## Error:
## ! object 'p08_topn_gsea' not found
p08_topn_gsea[["GO_ko_vs_het_up"]][[3]]
## Error:
## ! object 'p08_topn_gsea' not found
p08_topn_gsea[["GO_ko_vs_het_up"]][[4]]
## Error:
## ! object 'p08_topn_gsea' not found
p08_topn_gsea[["GO_ko_vs_het_up"]][[5]]
## Error:
## ! object 'p08_topn_gsea' not found
pp(file = "03theresa_comparison_images/gsea_p08_retina_het_vs_wt_top_hit.pdf")
p08_topn_gsea[["GO_het_vs_wt_up"]][[1]]
## Error:
## ! object 'p08_topn_gsea' not found
plotted <- dev.off()
#gsea_ko <-  gsea_result_ko[["result"]] %>%
#    select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
#    arrange(desc(recall)) %>%
#    head(n = 10)
#  gsea_plots_ko <- ggplot(gsea_ko, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
#  geom_bar(stat = "identity")+
#  scale_fill_continuous(low = "blue", high = "red") +
#  theme_bw()+
#  ylab("") +
#  xlab("GSEA Score")
gsea_het <-  gsea_result_het[["result"]] %>%
  select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
  arrange(desc(recall)) %>%
  head(n = 10)
gsea_plots_het <- ggplot(gsea_het, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
  geom_bar(stat = "identity") +
  scale_fill_continuous(low = "blue", high = "red") +
  theme_bw() +
  ylab("") +
  xlab("Over Representation Score")
pp(file = "03theresa_comparison_images/GSEA_p08_axontrap_retinahet_upregulated_vs_retinako.pdf")
gsea_plots_het
plotted <- dev.off()
gsea_plots_het

gsea_all <-  gsea_result_alldysregulated[["result"]] %>%
  select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
  arrange(desc(recall)) %>%
  head(n = 10)
gsea_plots_all <- ggplot(gsea_all, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
  geom_bar(stat = "identity") +
  scale_fill_continuous(low = "blue", high = "red") +
  theme_bw() +
  ylab("") +
  xlab("Over Representation Score")

pp(file = "images/GSEA_p08_retina_axontrap_alldysregulatedgenes.pdf")
gsea_plots_all
plotted <- dev.off()

15 SCN Het vs KO

It is only now that I realized we are splitting the data by location for each set of comparisons. I think that, left to my own devices, I would prefer to keep the input data structure intact, perform the somewhat larger number of contrasts, and then split up the results. Ideally this will slightly improve the fidelity of the results returned by DESeq2 and friends. But, I will run the state of Theresa’s notebook with as few changes as possible first, then add this.

15.1 PCA

I am going to skip this PCA plot for a couple of reasons: I already did a superset of it, and the subset Theresa performed is not valid given the set of samples included in my sample sheet, and figuring out the actually corresponding subset will take me forever… In addition, I want to use my mm38_hisat_v3 for everything…

mm38_subset <- subset_se(
  mm38_hisat,
  subset = "(batch == '4' | batch == '5' | batch == '6') & time == 'p08' & location == 'scn' | sampleid == 'iprgc_03'")
mm38_norm <- normalize(mm38_subset, filter = TRUE, convert = "cpm",
                            transform = "log2", batch = "svaseq")
mm38_norm <- set_batches(mm38_norm, fact = "location")
mm38_norm <- set_conditions(mm38_norm, fact = "genotype")
pca_norm <- plot_pca(mm38_norm, max_overlaps = 70)
pca_norm$plot

Instead I will simplify the subset and see what happens…

scn_samples <- subset_se(mm38_hisat_v3,
                           subset = "location_atb == 'scn'") %>%
  set_batches(fact = "location_atb") %>%
  set_conditions(fact = "genotype_atb", colors = color_choices[["genotype"]])
## The number of samples by batch are:
## 
## scn 
##  17
## The numbers of samples by condition are:
## 
## het  ko  wt 
##   6   6   5
scn_norm <- normalize(scn_samples, filter = TRUE, convert = "cpm",
                           transform = "log2", batch = "svaseq")
## Removing 11109 low-count genes (14316 remaining).
## transform_counts: Found 919 values less than 0.
## transform_counts: Found 919 values equal to 0, adding 1 to the matrix.
scn_norm_pca <- plot_pca(scn_norm)
scn_norm_pca
## The result of performing a fast_svd dimension reduction.
## The x-axis is PC1 and the y-axis is PC2
## Colors are defined by het, ko, wt
## Shapes are defined by scn.

16 Library sizes post-deduplication

Theresa’s next operation was to perform libsize/nonzero plots. I already did the pre/post deduplication nonzero, here is the analagous libsize.

v2 is pre-deduplication and v3 is post.

plot_libsize(mm38_hisat_v2)
## Library sizes of 65 samples, 
## ranging from 3,717,242 to 24,538,069.

post_filter_nonzero <- plot_libsize(mm38_hisat_v3, text = FALSE)
pp(file = "01diagnostic_images/post_all_filteres_nonzero.pdf")
post_filter_nonzero[["plot"]]
plotted <- dev.off()
post_filter_nonzero
## Library sizes of 63 samples, 
## ranging from 1,264,475 to 10,979,038.

I am a bit concerned about some of these library sizes post-deduplication.

Let us look at the relationship between reads and duplication, which I assume will be relatively linear.

test <- colData(mm38_hisat_v3)[, c("hisat_genome_single_all", "umi_dedup_pct_reads")]
test_plot <- plot_linear_scatter(test, loess = TRUE)
test_plot[["scatter"]]
## `geom_smooth()` using formula = 'y ~ x'
## Warning: The following aesthetics were dropped during statistical transformation: label.
## i This can happen when ggplot fails to infer the correct grouping structure in
##   the data.
## i Did you forget to specify a `group` aesthetic or to convert a numerical
##   variable into a factor?

test_plot[["scatter"]]
## `geom_smooth()` using formula = 'y ~ x'
## Warning: The following aesthetics were dropped during statistical transformation: label.
## i This can happen when ggplot fails to infer the correct grouping structure in
##   the data.
## i Did you forget to specify a `group` aesthetic or to convert a numerical
##   variable into a factor?

pp(file = "01diagnostic_images/deduplication_vs_hisat_scatter.pdf")
test_plot[["scatter"]]
## `geom_smooth()` using formula = 'y ~ x'
## Warning: The following aesthetics were dropped during statistical transformation: label.
## i This can happen when ggplot fails to infer the correct grouping structure in
##   the data.
## i Did you forget to specify a `group` aesthetic or to convert a numerical
##   variable into a factor?
dev.off()
## png 
##   2

Theresa also produced a density/sample plot, that might prove quite useful for these due to their significantly larger variance across samples (due to deduplication).

pp(file = "01diagnostic_images/sample_density.pdf")
mm38_density <- plot_density(loc_geno_nt)
mm38_density[["plot"]] +
  theme(legend.position = "none")
plot_boxplot(loc_geno_nt)
##          iprgc_62 iprgc_63 iprgc_64 iprgc_65 iprgc_66 iprgc_67 iprgc_68
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          1.689    1.527    2.480    2.558    1.894    2.186    1.469
## median      3.713    3.521    4.413    4.445    3.909    4.021    3.539
## mean        3.711    3.556    4.250    4.279    3.916    3.995    3.573
## q3          5.460    5.314    5.914    5.904    5.705    5.638    5.333
## max        15.929   16.020   14.188   14.054   14.741   15.920   16.168
## iqr         3.771    3.788    3.434    3.345    3.811    3.452    3.864
## iqr_high   11.115   10.995   11.064   10.922   11.423   10.817   11.129
## iqr_low    -5.656   -5.681   -5.151   -5.018   -5.717   -5.178   -5.796
## sd          2.426    2.434    2.309    2.285    2.439    2.309    2.439
## var         5.885    5.926    5.330    5.219    5.948    5.333    5.948
## stdvar      1.586    1.667    1.254    1.220    1.519    1.335    1.665
##          iprgc_69 iprgc_70 iprgc_71 iprgc_72 iprgc_73 iprgc_74 iprgc_75
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          2.543    2.182    1.968    1.990    1.996    2.404    2.419
## median      4.466    4.015    3.869    3.922    3.907    4.268    4.318
## mean        4.299    3.977    3.865    3.889    3.901    4.169    4.210
## q3          5.929    5.624    5.556    5.584    5.599    5.848    5.883
## max        12.381   15.767   16.057   15.939   15.386   13.760   13.110
## iqr         3.387    3.442    3.587    3.594    3.602    3.444    3.465
## iqr_high   11.009   10.786   10.937   10.975   11.002   11.014   11.080
## iqr_low    -5.080   -5.162   -5.381   -5.391   -5.403   -5.166   -5.197
## sd          2.304    2.325    2.377    2.377    2.384    2.334    2.328
## var         5.308    5.404    5.651    5.648    5.684    5.448    5.421
## stdvar      1.235    1.359    1.462    1.452    1.457    1.307    1.288
##          iprgc_76 iprgc_77 iprgc_81 iprgc_82 iprgc_83 iprgc_84 iprgc_85
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          2.518    2.264    2.084    2.178    2.503    2.531    2.246
## median      4.379    4.066    3.814    3.984    4.354    4.457    3.959
## mean        4.264    4.025    3.848    3.996    4.221    4.288    3.938
## q3          5.908    5.654    5.476    5.617    5.852    5.961    5.531
## max        13.649   15.864   16.157   14.893   13.932   13.337   16.483
## iqr         3.389    3.390    3.392    3.439    3.349    3.430    3.285
## iqr_high   10.992   10.739   10.565   10.776   10.875   11.106   10.459
## iqr_low    -5.084   -5.085   -5.088   -5.159   -5.023   -5.145   -4.928
## sd          2.299    2.295    2.320    2.320    2.298    2.317    2.234
## var         5.287    5.266    5.382    5.384    5.283    5.368    4.989
## stdvar      1.240    1.308    1.399    1.348    1.252    1.252    1.267
##          iprgc_86 iprgc_87 iprgc_88 iprgc_89 iprgc_90 iprgc_91 iprgc_92
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          2.470    1.526    1.858    2.199    2.303    2.420    2.044
## median      4.345    3.317    3.742    4.035    4.389    4.450    3.823
## mean        4.232    3.389    3.773    4.031    4.202    4.259    3.874
## q3          5.941    4.998    5.463    5.664    5.973    5.967    5.531
## max        14.490   16.910   16.308   15.157   14.315   13.393   15.881
## iqr         3.471    3.472    3.604    3.465    3.669    3.547    3.487
## iqr_high   11.148   10.206   10.869   10.862   11.477   11.288   10.761
## iqr_low    -5.207   -5.208   -5.406   -5.198   -5.504   -5.321   -5.230
## sd          2.339    2.276    2.347    2.320    2.403    2.360    2.325
## var         5.471    5.179    5.510    5.381    5.773    5.571    5.404
## stdvar      1.293    1.528    1.460    1.335    1.374    1.308    1.395
##          iprgc_93 iprgc_94 iprgc_95 iprgc_96 iprgc_97 iprgc_98 iprgc_99
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          2.194    1.455    2.093    2.550    2.101    1.934    2.604
## median      3.971    3.413    3.819    4.423    3.892    3.878    4.541
## mean        3.924    3.486    3.853    4.270    3.934    3.908    4.335
## q3          5.516    5.208    5.455    5.926    5.576    5.660    5.976
## max        16.492   16.585   16.084   13.933   15.433   14.700   12.844
## iqr         3.322    3.753    3.362    3.377    3.475    3.726    3.371
## iqr_high   10.498   10.837   10.498   10.991   10.789   11.249   11.033
## iqr_low    -4.982   -5.629   -5.043   -5.065   -5.213   -5.589   -5.057
## sd          2.253    2.401    2.276    2.305    2.332    2.424    2.303
## var         5.075    5.766    5.181    5.314    5.440    5.876    5.302
## stdvar      1.293    1.654    1.344    1.245    1.383    1.504    1.223
##          iprgc_100 iprgc_101 iprgc_102 iprgc_104 iprgc_105 iprgc_106 iprgc_107
## min          0.000     0.000     0.000     0.000     0.000     0.000     0.000
## q1           1.824     2.562     2.159     2.260     2.261     1.787     1.599
## median       3.846     4.490     4.065     4.050     4.187     3.930     3.435
## mean         3.838     4.307     4.032     4.011     4.119     3.888     3.505
## q3           5.593     5.958     5.751     5.639     5.825     5.706     5.147
## max         15.708    13.878    14.894    15.914    14.621    15.187    17.169
## iqr          3.770     3.396     3.592     3.380     3.564     3.919     3.548
## iqr_high    11.248    11.053    11.138    10.709    11.171    11.585    10.469
## iqr_low     -5.654    -5.095    -5.388    -5.069    -5.346    -5.879    -5.322
## sd           2.419     2.294     2.368     2.270     2.351     2.479     2.303
## var          5.852     5.263     5.610     5.155     5.526     6.147     5.305
## stdvar       1.525     1.222     1.391     1.285     1.342     1.581     1.514
##          iprgc_108 iprgc_109 iprgc_110 iprgc_111 iprgc_112 iprgc_113 iprgc_114
## min          0.000     0.000     0.000     0.000     0.000     0.000     0.000
## q1           1.838     2.491     2.183     2.361     2.193     2.358     2.163
## median       4.026     4.483     3.966     4.274     4.066     4.150     4.078
## mean         3.945     4.302     4.002     4.184     4.080     4.108     4.065
## q3           5.784     6.029     5.657     5.855     5.808     5.711     5.801
## max         14.921    12.859    15.079    14.025    13.023    15.794    13.261
## iqr          3.946     3.538     3.473     3.493     3.614     3.353     3.638
## iqr_high    11.703    11.335    10.867    11.095    11.229    10.741    11.257
## iqr_low     -5.919    -5.306    -5.210    -5.240    -5.422    -5.030    -5.457
## sd           2.486     2.354     2.336     2.335     2.386     2.267     2.400
## var          6.179     5.543     5.457     5.450     5.695     5.139     5.760
## stdvar       1.566     1.289     1.364     1.303     1.396     1.251     1.417
##          iprgc_115 iprgc_116 iprgc_117 iprgc_118 iprgc_119 iprgc_120 iprgc_121
## min          0.000     0.000     0.000     0.000     0.000     0.000     0.000
## q1           2.511     2.620     2.396     2.491     2.525     2.346     2.352
## median       4.495     4.525     4.367     4.396     4.404     4.347     4.365
## mean         4.303     4.332     4.223     4.256     4.248     4.214     4.214
## q3           6.020     5.957     5.980     5.985     5.919     5.941     5.942
## max         12.739    12.644    13.541    13.650    13.796    14.752    14.560
## iqr          3.508     3.337     3.584     3.494     3.394     3.596     3.590
## iqr_high    11.282    10.963    11.356    11.225    11.010    11.335    11.327
## iqr_low     -5.262    -5.006    -5.376    -5.241    -5.091    -5.393    -5.385
## sd           2.354     2.289     2.377     2.347     2.310     2.368     2.366
## var          5.543     5.242     5.649     5.509     5.335     5.609     5.596
## stdvar       1.288     1.210     1.338     1.294     1.256     1.331     1.328
##          iprgc_122 iprgc_123 iprgc_124 iprgc_125 iprgc_126 iprgc_127 iprgc_128
## min          0.000     0.000     0.000     0.000     0.000     0.000     0.000
## q1           2.567     1.556     2.038     2.211     2.215     2.323     2.352
## median       4.453     3.322     3.745     4.085     4.082     4.172     4.198
## mean         4.282     3.409     3.761     4.045     4.037     4.138     4.128
## q3           5.915     5.004     5.325     5.709     5.733     5.838     5.782
## max         13.284    17.222    17.196    13.801    15.040    14.076    15.026
## iqr          3.348     3.447     3.287     3.498     3.518     3.515     3.430
## iqr_high    10.937    10.174    10.255    10.956    11.011    11.110    10.926
## iqr_low     -5.022    -5.171    -4.930    -5.247    -5.278    -5.272    -5.145
## sd           2.300     2.242     2.196     2.354     2.363     2.348     2.316
## var          5.291     5.026     4.821     5.543     5.583     5.511     5.364
## stdvar       1.236     1.474     1.282     1.370     1.383     1.332     1.300
##          iprgc_129 iprgc_130
## min          0.000     0.000
## q1           2.462     2.286
## median       4.266     4.221
## mean         4.175     4.132
## q3           5.817     5.836
## max         15.200    13.939
## iqr          3.354     3.550
## iqr_high    10.848    11.162
## iqr_low     -5.031    -5.325
## sd           2.293     2.373
## var          5.258     5.632
## stdvar       1.259     1.363
## Plot describing the gene distribution from a dataset.
dev.off()
## png 
##   2
mm38_density[["plot"]] +
  theme(legend.position = "none")

box <- plot_boxplot(loc_geno_nt)
##          iprgc_62 iprgc_63 iprgc_64 iprgc_65 iprgc_66 iprgc_67 iprgc_68
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          1.689    1.527    2.480    2.558    1.894    2.186    1.469
## median      3.713    3.521    4.413    4.445    3.909    4.021    3.539
## mean        3.711    3.556    4.250    4.279    3.916    3.995    3.573
## q3          5.460    5.314    5.914    5.904    5.705    5.638    5.333
## max        15.929   16.020   14.188   14.054   14.741   15.920   16.168
## iqr         3.771    3.788    3.434    3.345    3.811    3.452    3.864
## iqr_high   11.115   10.995   11.064   10.922   11.423   10.817   11.129
## iqr_low    -5.656   -5.681   -5.151   -5.018   -5.717   -5.178   -5.796
## sd          2.426    2.434    2.309    2.285    2.439    2.309    2.439
## var         5.885    5.926    5.330    5.219    5.948    5.333    5.948
## stdvar      1.586    1.667    1.254    1.220    1.519    1.335    1.665
##          iprgc_69 iprgc_70 iprgc_71 iprgc_72 iprgc_73 iprgc_74 iprgc_75
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          2.543    2.182    1.968    1.990    1.996    2.404    2.419
## median      4.466    4.015    3.869    3.922    3.907    4.268    4.318
## mean        4.299    3.977    3.865    3.889    3.901    4.169    4.210
## q3          5.929    5.624    5.556    5.584    5.599    5.848    5.883
## max        12.381   15.767   16.057   15.939   15.386   13.760   13.110
## iqr         3.387    3.442    3.587    3.594    3.602    3.444    3.465
## iqr_high   11.009   10.786   10.937   10.975   11.002   11.014   11.080
## iqr_low    -5.080   -5.162   -5.381   -5.391   -5.403   -5.166   -5.197
## sd          2.304    2.325    2.377    2.377    2.384    2.334    2.328
## var         5.308    5.404    5.651    5.648    5.684    5.448    5.421
## stdvar      1.235    1.359    1.462    1.452    1.457    1.307    1.288
##          iprgc_76 iprgc_77 iprgc_81 iprgc_82 iprgc_83 iprgc_84 iprgc_85
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          2.518    2.264    2.084    2.178    2.503    2.531    2.246
## median      4.379    4.066    3.814    3.984    4.354    4.457    3.959
## mean        4.264    4.025    3.848    3.996    4.221    4.288    3.938
## q3          5.908    5.654    5.476    5.617    5.852    5.961    5.531
## max        13.649   15.864   16.157   14.893   13.932   13.337   16.483
## iqr         3.389    3.390    3.392    3.439    3.349    3.430    3.285
## iqr_high   10.992   10.739   10.565   10.776   10.875   11.106   10.459
## iqr_low    -5.084   -5.085   -5.088   -5.159   -5.023   -5.145   -4.928
## sd          2.299    2.295    2.320    2.320    2.298    2.317    2.234
## var         5.287    5.266    5.382    5.384    5.283    5.368    4.989
## stdvar      1.240    1.308    1.399    1.348    1.252    1.252    1.267
##          iprgc_86 iprgc_87 iprgc_88 iprgc_89 iprgc_90 iprgc_91 iprgc_92
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          2.470    1.526    1.858    2.199    2.303    2.420    2.044
## median      4.345    3.317    3.742    4.035    4.389    4.450    3.823
## mean        4.232    3.389    3.773    4.031    4.202    4.259    3.874
## q3          5.941    4.998    5.463    5.664    5.973    5.967    5.531
## max        14.490   16.910   16.308   15.157   14.315   13.393   15.881
## iqr         3.471    3.472    3.604    3.465    3.669    3.547    3.487
## iqr_high   11.148   10.206   10.869   10.862   11.477   11.288   10.761
## iqr_low    -5.207   -5.208   -5.406   -5.198   -5.504   -5.321   -5.230
## sd          2.339    2.276    2.347    2.320    2.403    2.360    2.325
## var         5.471    5.179    5.510    5.381    5.773    5.571    5.404
## stdvar      1.293    1.528    1.460    1.335    1.374    1.308    1.395
##          iprgc_93 iprgc_94 iprgc_95 iprgc_96 iprgc_97 iprgc_98 iprgc_99
## min         0.000    0.000    0.000    0.000    0.000    0.000    0.000
## q1          2.194    1.455    2.093    2.550    2.101    1.934    2.604
## median      3.971    3.413    3.819    4.423    3.892    3.878    4.541
## mean        3.924    3.486    3.853    4.270    3.934    3.908    4.335
## q3          5.516    5.208    5.455    5.926    5.576    5.660    5.976
## max        16.492   16.585   16.084   13.933   15.433   14.700   12.844
## iqr         3.322    3.753    3.362    3.377    3.475    3.726    3.371
## iqr_high   10.498   10.837   10.498   10.991   10.789   11.249   11.033
## iqr_low    -4.982   -5.629   -5.043   -5.065   -5.213   -5.589   -5.057
## sd          2.253    2.401    2.276    2.305    2.332    2.424    2.303
## var         5.075    5.766    5.181    5.314    5.440    5.876    5.302
## stdvar      1.293    1.654    1.344    1.245    1.383    1.504    1.223
##          iprgc_100 iprgc_101 iprgc_102 iprgc_104 iprgc_105 iprgc_106 iprgc_107
## min          0.000     0.000     0.000     0.000     0.000     0.000     0.000
## q1           1.824     2.562     2.159     2.260     2.261     1.787     1.599
## median       3.846     4.490     4.065     4.050     4.187     3.930     3.435
## mean         3.838     4.307     4.032     4.011     4.119     3.888     3.505
## q3           5.593     5.958     5.751     5.639     5.825     5.706     5.147
## max         15.708    13.878    14.894    15.914    14.621    15.187    17.169
## iqr          3.770     3.396     3.592     3.380     3.564     3.919     3.548
## iqr_high    11.248    11.053    11.138    10.709    11.171    11.585    10.469
## iqr_low     -5.654    -5.095    -5.388    -5.069    -5.346    -5.879    -5.322
## sd           2.419     2.294     2.368     2.270     2.351     2.479     2.303
## var          5.852     5.263     5.610     5.155     5.526     6.147     5.305
## stdvar       1.525     1.222     1.391     1.285     1.342     1.581     1.514
##          iprgc_108 iprgc_109 iprgc_110 iprgc_111 iprgc_112 iprgc_113 iprgc_114
## min          0.000     0.000     0.000     0.000     0.000     0.000     0.000
## q1           1.838     2.491     2.183     2.361     2.193     2.358     2.163
## median       4.026     4.483     3.966     4.274     4.066     4.150     4.078
## mean         3.945     4.302     4.002     4.184     4.080     4.108     4.065
## q3           5.784     6.029     5.657     5.855     5.808     5.711     5.801
## max         14.921    12.859    15.079    14.025    13.023    15.794    13.261
## iqr          3.946     3.538     3.473     3.493     3.614     3.353     3.638
## iqr_high    11.703    11.335    10.867    11.095    11.229    10.741    11.257
## iqr_low     -5.919    -5.306    -5.210    -5.240    -5.422    -5.030    -5.457
## sd           2.486     2.354     2.336     2.335     2.386     2.267     2.400
## var          6.179     5.543     5.457     5.450     5.695     5.139     5.760
## stdvar       1.566     1.289     1.364     1.303     1.396     1.251     1.417
##          iprgc_115 iprgc_116 iprgc_117 iprgc_118 iprgc_119 iprgc_120 iprgc_121
## min          0.000     0.000     0.000     0.000     0.000     0.000     0.000
## q1           2.511     2.620     2.396     2.491     2.525     2.346     2.352
## median       4.495     4.525     4.367     4.396     4.404     4.347     4.365
## mean         4.303     4.332     4.223     4.256     4.248     4.214     4.214
## q3           6.020     5.957     5.980     5.985     5.919     5.941     5.942
## max         12.739    12.644    13.541    13.650    13.796    14.752    14.560
## iqr          3.508     3.337     3.584     3.494     3.394     3.596     3.590
## iqr_high    11.282    10.963    11.356    11.225    11.010    11.335    11.327
## iqr_low     -5.262    -5.006    -5.376    -5.241    -5.091    -5.393    -5.385
## sd           2.354     2.289     2.377     2.347     2.310     2.368     2.366
## var          5.543     5.242     5.649     5.509     5.335     5.609     5.596
## stdvar       1.288     1.210     1.338     1.294     1.256     1.331     1.328
##          iprgc_122 iprgc_123 iprgc_124 iprgc_125 iprgc_126 iprgc_127 iprgc_128
## min          0.000     0.000     0.000     0.000     0.000     0.000     0.000
## q1           2.567     1.556     2.038     2.211     2.215     2.323     2.352
## median       4.453     3.322     3.745     4.085     4.082     4.172     4.198
## mean         4.282     3.409     3.761     4.045     4.037     4.138     4.128
## q3           5.915     5.004     5.325     5.709     5.733     5.838     5.782
## max         13.284    17.222    17.196    13.801    15.040    14.076    15.026
## iqr          3.348     3.447     3.287     3.498     3.518     3.515     3.430
## iqr_high    10.937    10.174    10.255    10.956    11.011    11.110    10.926
## iqr_low     -5.022    -5.171    -4.930    -5.247    -5.278    -5.272    -5.145
## sd           2.300     2.242     2.196     2.354     2.363     2.348     2.316
## var          5.291     5.026     4.821     5.543     5.583     5.511     5.364
## stdvar       1.236     1.474     1.282     1.370     1.383     1.332     1.300
##          iprgc_129 iprgc_130
## min          0.000     0.000
## q1           2.462     2.286
## median       4.266     4.221
## mean         4.175     4.132
## q3           5.817     5.836
## max         15.200    13.939
## iqr          3.354     3.550
## iqr_high    10.848    11.162
## iqr_low     -5.031    -5.325
## sd           2.293     2.373
## var          5.258     5.632
## stdvar       1.259     1.363
pp(file = "01diagnostic_images/sample_boxplot.pdf")
box
## Plot describing the gene distribution from a dataset.
dev.off()
## png 
##   2
box
## Plot describing the gene distribution from a dataset.

There is some difference across sample densities, but it is not too crazytown.

17 Diverging a little

At this point in the document I read ahead a bit and came to the conclusion that it repeats the above logic of taking the union of wt comparisons to remove genes from the appropriate het/ko or p15/p08 or location comparisons. This seems quite reasonable to me, but I would prefer to not separate all the data, so I will attempt to duplicate and slightly streamline this logic on the full dataset. Thus I am going to skip down to the end and attempt to implement this.

17.1 DE

Note: The following few blocks are all copy/pasted directly from Theresa’s notebook and are not evaluated because they are performed almost identically later but with slightly different logic/orders.

mm_de_normal_p8_scn <- all_pairwise(mm38_subset, model_batch = "svaseq",
                                    parallel = FALSE, do_ebseq = FALSE, do_basic = FALSE,
                                    do_dream = FALSE, do_noiseq = FALSE, do_edger = FALSE,
                                    filter = TRUE)
annots_to_merge <- mm_annot %>%
  select(ensembl_gene_id, external_gene_name) %>%
  filter(ensembl_gene_id %in% rownames(mm_de_normal_p8_scn$deseq$all_tables$ko_scn_vs_het_scn)) %>%
  distinct()
mm_de_normal_p8_scn$deseq$all_tables$ko_scn_vs_het_scn <- merge(
  mm_de_normal_p8_scn$deseq$all_tables$ko_scn_vs_het_scn,
  annots_to_merge, by.x = 0, by.y = "ensembl_gene_id", all.x = TRUE)
hetkeeper_genes <- mm_de_normal_p8_scn$deseq$all_tables$wt_scn_vs_het_scn %>%
  filter(logFC <= -0.1 & adj.P.Val <= 0.05)
kokeeper_genes <- mm_de_normal_p8_scn$deseq$all_tables$wt_scn_vs_ko_scn %>%
  filter(logFC <= -0.1 & adj.P.Val <= 0.05)
keepergenes <- unique(c(rownames(hetkeeper_genes), rownames(kokeeper_genes)))
df <- mm_de_normal_p8_scn$deseq$all_tables$koscn_vs_hetscn %>%
  dplyr::mutate(logFC = -logFC) %>%
  set_sig_limma(factors = c("Het Enriched",
                            "KO Enriched"))
df <- df %>%
  filter(Row.names %in% keepergenes)
labels_ups <- df %>%
  filter(abs(logFC) > 1) %>%
  arrange(logFC) %>%
  head(n = 1)
labels_downs <- df %>%
  filter(abs(logFC) > 1) %>%
  arrange(-logFC) %>%
  head(n = 1)
labels <- rbind(labels_ups, labels_downs)
res_tbl <- df
DEplot <- ggplot(res_tbl, aes(x = logFC, y = -log10(adj.P.Val), label = external_gene_name)) +
  geom_point(aes(colour = Significance), size = 4) +
  geom_vline(xintercept = c(-1, 1)) +
  geom_hline(yintercept = -log10(0.05)) +
  theme_classic(base_size = 20) +
  xlab("log2(FC)") +
  ylab("-log10(p-value)") +
  ## ggtitle(title, subtitle = subtitle) +
  theme(legend.position="right") +
  scale_color_manual(values=c("Het Enriched" = "#F8766D",
                              "KO Enriched" = "#00BFC4",
                              "Not\n Enriched" = "Grey")) +
  geom_label_repel(data=filter(df,
                               ## c('s5_het_dlgn', 's5_het_ret', 's5_het_scn')),
                               external_gene_name %in% labels$external_gene_name),
                   ## nudge_x = -0.5,
                   nudge_y = 3, max.overlaps = 15) +
  ggtitle("SCN Het vs KO Translatome")

pp(file = "images/p08_scn_DE_1312024.pdf")
DEplot
plotted <- dev.off()

write_xlsx(df, excel = "excel/scnhet_vs_scnko_WTfiltered.xlsx")

17.2 How many ups/downs

ko_enriched <- df %>%
  filter(Significance == "KO Enriched")

het_enriched <- df %>%
  filter(Significance == "Het Enriched")

17.3 GSEA

ko_genes <- res_tbl %>%
  filter(adj.P.Val <= 0.05) %>%
  arrange(-abs(logFC)) %>%
  select(Row.names, logFC, adj.P.Val, external_gene_name, Significance) %>%
  filter(logFC <= -1)

het_genes <- res_tbl %>%
  filter(adj.P.Val <= 0.05) %>%
  arrange(-abs(logFC)) %>%
  select(Row.names, logFC, adj.P.Val, external_gene_name, Significance) %>%
  filter(logFC >= 1)

alldysregulated_genes <- res_tbl %>%
  filter(adj.P.Val <= 0.05) %>%
  arrange(logFC) %>%
  select(Row.names, logFC, adj.P.Val, external_gene_name, Significance) %>%
  filter(abs(logFC) >= 1)

gsea_result_ko <- gost(query = ko_genes$external_gene_name,
                       organism = "mmusculus",
                       evcodes = TRUE,
                       ordered_query = TRUE)

gsea_result_het <- gost(query = het_genes$external_gene_name,
                        organism = "mmusculus",
                        evcodes = TRUE,
                        ordered_query = TRUE)

gsea_result_alldysregulated <- gost(query = alldysregulated_genes$external_gene_name,
                                    organism = "mmusculus",
                                    evcodes = TRUE,
                                    ordered_query = TRUE)
gsea_ko <-  gsea_result_ko[["result"]] %>%
  select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
  arrange(desc(recall)) %>%
  head(n = 10)
gsea_plots_ko <- ggplot(gsea_ko, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
  geom_bar(stat = "identity") +
  scale_fill_continuous(low = "blue", high = "red") +
  theme_bw() +
  ylab("") +
  xlab("Over enrichment Score")

gsea_het <-  gsea_result_het[["result"]] %>%
  select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
  arrange(desc(recall)) %>%
  head(n = 10)
gsea_plots_het <- ggplot(gsea_het, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
  geom_bar(stat = "identity") +
  scale_fill_continuous(low = "blue", high = "red") +
  theme_bw() +
  ylab("") +
  xlab("Over enrichment Score")

gsea_all <-  gsea_result_alldysregulated[["result"]] %>%
  select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
  arrange(desc(recall)) %>%
  head(n = 10)
gsea_plots_all <- ggplot(gsea_all, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
  geom_bar(stat = "identity") +
  scale_fill_continuous(low = "blue", high = "red") +
  theme_bw() +
  ylab("") +
  xlab("Over enrichment Score")

pp(file = "images/GSEA_p08_retina_axontrap_alldysregulatedgenes.pdf")
gsea_plots_all
plotted <- dev.off()

18 Het Retina vs Het SCN

mm38_subset2 <- subset_se(
  mm38_hisat,
  subset = "(batch == '4' | batch == '5' | batch == '6') & time == 'p08' & genotype != 'ko' & location != 'dlgn' | sampleid == 'iprgc_03'")

mm38_subset2 <- subset_se(mm38_subset2, subset = "sampleid != 'iprgc_89'")
mm38_subset2$design %>%
  select(genotype, location) %>%
  table()
mm38_norm2 <- normalize(mm38_subset2, filter=TRUE,
                             convert="cpm",
                             transform="log2", batch = "svaseq")

18.1 PCA

mm38_norm2 <- set_batches(mm38_norm2, fact = "location")
mm38_norm2 <- set_conditions(mm38_norm2, fact = "genotype")
pca_norm2 <- plot_pca(mm38_norm2, max_overlaps = 70)
pca_norm2$plot

18.2 DE

mm_de_subset2 <- all_pairwise(mm38_subset2,
                              model_batch="svaseq",
                              parallel=FALSE, do_ebseq=FALSE,
                              do_basic = FALSE, do_dream = FALSE,
                              do_noiseq = FALSE, do_edger = FALSE,
                              filter = TRUE)
retinakeeper_genes <- mm_de_subset2$deseq$all_tables$wt_retina_vs_het_retina %>%
  filter(logFC <= -0.1 & adj.P.Val <= 0.05)

scnkeeper_genes <- mm_de_subset2$deseq$all_tables$wt_scn_vs_het_scn %>%
  filter(logFC <= -0.1 & adj.P.Val <= 0.05)

keepergenes <- unique(c(rownames(retinakeeper_genes), rownames(scnkeeper_genes)))

annots_to_merge <- mm_annot %>%
  select(ensembl_gene_id, external_gene_name) %>%
  filter(ensembl_gene_id %in% rownames(mm_de_subset2$deseq$all_tables$het_scn_vs_het_retina)) %>%
  distinct()

mm_de_subset2$deseq$all_tables$het_scn_vs_het_retina <- merge(
  mm_de_subset2$deseq$all_tables$het_scn_vs_het_retina,
  annots_to_merge, by.x = 0,
  by.y = "ensembl_gene_id", all.x = TRUE)

df <- mm_de_subset2$deseq$all_tables$het_scn_vs_het_retina %>%
  mutate(Significance = case_when(logFC <= -1.0 ~ "Retina Enriched",
                                  logFC >= 1.0 ~ "SCN Enriched",
                                  logFC > -1.0 & logFC < 1.0 ~ "Not\n Enriched"))

df <- df %>%
  filter(Row.names %in% keepergenes)

scn_enriched <- df %>%
  filter(adj.P.Val <= 0.05 & logFC >= 1.0) %>%
  arrange(-logFC) %>%
  select(Row.names, external_gene_name, logFC, adj.P.Val) %>%
  mutate(Significance = "SCN Enriched") %>%
  filter(Row.names %in% rownames(scnkeeper_genes))

retina_enriched <- df %>%
  filter(adj.P.Val <= 0.05 & logFC <= -1.0) %>%
  arrange(logFC)  %>%
  select(Row.names, external_gene_name, logFC, adj.P.Val) %>%
  mutate(Significance = "Retina Enriched") %>%
  filter(Row.names %in% rownames(retinakeeper_genes))

notenriched <- df %>%
  select(Row.names, external_gene_name, logFC, adj.P.Val, Significance) %>%
  filter(Row.names %in% c(rownames(retinakeeper_genes),
                          rownames(scnkeeper_genes))[duplicated(c(rownames(retinakeeper_genes),
                                                                  rownames(scnkeeper_genes)))]) %>%
  filter(Significance == "Not\n Enriched")

df <- rbind(scn_enriched, retina_enriched, notenriched)
df <- df %>%
  distinct()

## writexl::write_xlsx(df, path = "axonTRAP_DE_results_20240202/retinahet_vs_scn_het_WTfiltered.xlsx")
labels_ups <- df %>%
  filter(adj.P.Val <= 0.05 & abs(logFC) > 1.0) %>%
  arrange(logFC) %>%
  head(n = 10)

labels_downs <- df %>%
  filter(adj.P.Val <= 0.05 & abs(logFC) > 1.0) %>%
  arrange(-logFC) %>%
  head(n = 10)

labels <- rbind(labels_ups, labels_downs)

labels_requested <- c("Cdh10","Cdh12","Cdh13","Cdh18",
                      "Cdh7","Cdh8","Cdh9","Cntn3",
                      "Cntn4","Cntn5","Cntn6","Kirrel3",
                      "Nrxn1","Nrxn3","Sema3c","Sema6d",
                      "Tenm1","Tenm2","Tenm4")
res_tbl <- df
DEplot <- ggplot(res_tbl, aes(x = logFC, y = -log10(adj.P.Val), label = external_gene_name)) +
  geom_point(aes(colour = Significance), size = 4) +
  geom_vline(xintercept = c(-1, 1)) +
  geom_hline(yintercept = -log10(0.05)) +
  theme_classic(base_size = 20) +
  xlab("log2(FC)") +
  ylab("-log10(p-value)") +
  ## ggtitle(title, subtitle = subtitle) +
  theme(legend.position="right") +
  scale_color_manual(values=c("Grey", "#F8766D", "#00BFC4")) +
  geom_label_repel(data=filter(df,
                               external_gene_name %in% labels_requested),
                   ## c(labels$external_gene_name, "Opn4")), #c('s5_het_dlgn', 's5_het_ret', 's5_het_scn')),
                   ## nudge_x = -0.5,
                   nudge_y = 15, max.overlaps = 25)

#pp(file = "axonTRAP_Volcanoplots_20240202/p08_retinavsscnhet_DE_requested_genelabels_02052024.pdf")
DEplot
#dev.off()

18.3 How many ups/downs

scn_enriched <- df %>%
  filter(adj.P.Val <= 0.05 & logFC >= 1.0) %>%
  arrange(-logFC) %>%
  select(Row.names, external_gene_name, logFC, adj.P.Val, Significance)

retina_enriched <- df %>%
  filter(adj.P.Val <= 0.05 & logFC <= -1.0) %>%
  arrange(logFC)  %>%
  select(Row.names, external_gene_name, logFC, adj.P.Val, Significance)

scn_enriched
retina_enriched

df %>%
  filter(Significance == "Not\n Enriched")

18.4 GSEA

gsea_result_scn <- gost(query = scn_enriched$external_gene_name,
                        organism = "mmusculus", evcodes = TRUE,
                        ordered_query = TRUE, source = c("GO"))

gsea_result_ret <- gost(query = retina_enriched$external_gene_name,
                        organism = "mmusculus", evcodes = TRUE,
                        ordered_query = TRUE, source = c("GO"))
gsea_scn <-  gsea_result_scn[["result"]] %>%
  select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
  arrange(desc(recall)) %>%
  head(n = 20)
gsea_plots_scn <- ggplot(gsea_scn, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
  geom_bar(stat = "identity") +
  scale_fill_continuous(low = "blue", high = "red") +
  theme_bw() +
  ylab("") +
  xlab("Over enrichment Score")

pp(file = "images/GSEA_SCNhet_vs_retina_enriched_P08.pdf")
gsea_plots_scn
plotted <- dev.off()

gsea_ret <-  gsea_result_ret[["result"]] %>%
  select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
  arrange(desc(recall)) %>%
  head(n = 20)
gsea_plots_ret <- ggplot(gsea_ret, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
  geom_bar(stat = "identity") +
  scale_fill_continuous(low = "blue", high = "red") +
  theme_bw() +
  ylab("") +
  xlab("Over enrichment Score")

pp(file = "images/GSEA_Retinahet_vs_SCN_enriched_P08.pdf")
gsea_plots_ret
plotted <- dev.off()

19 KO Retina vs KO SCN

mm38_subset3 <- subset_se(
  mm38_hisat,
  subset = "(batch == '4' | batch == '5' | batch == '6') & time == 'p08'  & genotype != 'het' & location != 'dlgn' | sampleid == 'iprgc_03'")

mm38_subset3 <- subset_se(mm38_subset3, subset = "sampleid != 'iprgc_86'")
mm38_subset3$design %>%
  select(genotype, location) %>%
  table()
mm38_norm3 <- normalize(mm38_subset3, filter=TRUE,
                             convert="cpm", transform="log2", batch = "svaseq")

19.1 PCA

mm38_norm3 <- set_batches(mm38_norm3, fact = "location")
mm38_norm3 <- set_conditions(mm38_norm3, fact = "genotype")
pca_norm3 <- plot_pca(mm38_norm3, max_overlaps = 70)
pca_norm3$plot

19.2 DE

mm_de_subset3 <- all_pairwise(mm38_subset3,
                              model_batch="svaseq",
                              parallel=FALSE, do_ebseq=FALSE,
                              do_basic = FALSE, do_dream = FALSE,
                              do_noiseq = FALSE, do_edger = FALSE,
                              filter = TRUE)

retinakeeper_genes <- mm_de_subset3$deseq$all_tables$wtretina_vs_koretina %>%
  filter(logFC <= -1.0 & adj.P.Val <= 0.05)

scnkeeper_genes <- mm_de_subset3$deseq$all_tables$wtscn_vs_koscn %>%
  filter(logFC <= -1.0 & adj.P.Val <= 0.05)

keepergenes <- unique(c(rownames(retinakeeper_genes), rownames(scnkeeper_genes)))

annots_to_merge <- mm_annot %>%
  select(ensembl_gene_id, external_gene_name) %>%
  filter(ensembl_gene_id %in% rownames(mm_de_subset3$deseq$all_tables$ko_scn_vs_ko_retina)) %>%
  distinct()

mm_de_subset3$deseq$all_tables$ko_scn_vs_ko_retina <- merge(
  mm_de_subset3$deseq$all_tables$ko_scn_vs_ko_retina,
  annots_to_merge, by.x = 0,
  by.y = "ensembl_gene_id", all.x = TRUE)

df <- mm_de_subset3$deseq$all_tables$ko_scn_vs_ko_retina %>%
  mutate(Significance = case_when(logFC <= -1 ~ "Retina Enriched",
                                  logFC >= 1 ~ "SCN Enriched",
                                  logFC > -1 & logFC < 1 ~ "Not\n Enriched"))

df <- df %>%
  filter(Row.names %in% keepergenes)

scn_enriched <- df %>%
  filter(adj.P.Val <= 0.05 & logFC >= 1) %>%
  arrange(-logFC) %>%
  select(Row.names, external_gene_name, logFC, adj.P.Val) %>%
  mutate(Significance = "SCN Enriched") %>%
  filter(Row.names %in% rownames(scnkeeper_genes))

df %>%
  filter(adj.P.Val <= 0.05 & logFC <= -1) %>%
  arrange(logFC)  %>%
  select(Row.names, external_gene_name, logFC, adj.P.Val) %>%
  mutate(Significance = "Retina Enriched") %>%
  filter(Row.names %in% rownames(retinakeeper_genes)) -> retina_enriched

notenriched <- df %>%
  select(Row.names, external_gene_name, logFC, adj.P.Val, Significance) %>%
  filter(Row.names %in% c(rownames(retinakeeper_genes),
                          rownames(scnkeeper_genes))[duplicated(c(rownames(retinakeeper_genes),
                                                                  rownames(scnkeeper_genes)))])

df <- rbind(scn_enriched, retina_enriched, notenriched)
labels_ups <- df %>%
  filter(adj.P.Val <= 0.05 & abs(logFC) > 1) %>%
  arrange(logFC) %>%
  head(n = 10)

labels_downs <- df %>%
  filter(adj.P.Val <= 0.05 & abs(logFC) > 1) %>%
  arrange(-logFC) %>%
  head(n = 10)

labels <- rbind(labels_ups, labels_downs)
## wanted_column <- "Significance"

res_tbl <- df
DEplot <- ggplot(res_tbl, aes(x = logFC, y = -log10(adj.P.Val), label = external_gene_name)) +
  geom_point(aes(colour = Significance), size = 4) +
  ## geom_point(aes(colour = !!sym(wanted_column)), size = 4) +
  geom_vline(xintercept = c(-1, 1)) +
  geom_hline(yintercept = -log10(0.05)) +
  theme_classic(base_size = 20) +
  xlab("log2(FC)") +
  ylab("-log10(p-value)") +
  ## ggtitle(title, subtitle = subtitle) +
  theme(legend.position = "right") +
  scale_color_manual(values = c("Grey", "#F8766D", "#00BFC4")) +
  geom_label_repel(data = filter(
    df, external_gene_name %in% c(labels$external_gene_name, "Opn4")),
    ## c('s5_het_dlgn', 's5_het_ret', 's5_het_scn')),
    ## nudge_x = -0.5,
    nudge_y = 10, max.overlaps = 25)

pp(file = "images/p08_retinavsscnko_DE_1312024.pdf")
DEplot
plotted <- dev.off()

19.3 How many ups/downs

scn_enriched
retina_enriched
notenriched %>%
  filter(Significance == "Not\n Enriched")

19.4 GSEA

gsea_result_scn <- gost(query = scn_enriched$external_gene_name,
                        organism = "mmusculus",
                        evcodes = TRUE,
                        ordered_query = TRUE,
                        source = c("GO"))

gsea_result_ret <- gost(query = retina_enriched$external_gene_name,
                        organism = "mmusculus",
                        evcodes = TRUE,
                        ordered_query = TRUE,
                        source = c("GO"))
gsea_scn <-  gsea_result_scn[["result"]] %>%
  select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
  arrange(desc(recall)) %>%
  head(n = 20)
gsea_plots_scn <- ggplot(gsea_scn, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
  geom_bar(stat = "identity") +
  scale_fill_continuous(low = "blue", high = "red") +
  theme_bw() +
  ylab("") +
  xlab("GSEA Score")

pp(file = "images/GSEA_SCNko_enriched_vs_retina_P08.pdf")
gsea_plots_scn
plotted <- dev.off()

gsea_ret <-  gsea_result_ret[["result"]] %>%
  select(term_name, p_value, term_size, intersection_size, recall, source, intersection) %>%
  arrange(desc(recall)) %>%
  head(n = 20)
gsea_plots_ret <- ggplot(gsea_ret, aes(x = recall, y = reorder(term_name, recall), fill = p_value)) +
  geom_bar(stat = "identity") +
  scale_fill_continuous(low = "blue", high = "red") +
  theme_bw() +
  ylab("") +
  xlab("GSEA Score")

pp(file = "images/GSEA_Retinako_enriched_vs_SCN_P08.pdf")
gsea_plots_ret
plotted <- dev.off()

20 My version of the global analysis

I want to have an invocation of all_pairwise() which uses all samples, in the following block I will set that up using a set of ‘keepers’ which will be named by time, location, then 2 letters for the numerator/denominator: w for WT, h for het, d for delta; thus “p08_retina_hw” is comparing the het/wt for the p08 retina samples.

If they are of interest, I will have a separate set which follows the same convention with names like “p08_ko_sr” to compare p08 deltas with SCN as the numerator and retina as the denominator.

20.1 Set up the exclusion dataset

The most peculiar aspect of this analysis resides in the choices around choosing which genes to consider when comparing the genotypes/locations/times. The general idea is pretty clear: find the genes which are non-specifically being pulled down in the WT samples and either exclude or discount them. The various potential methods for performing this are confusing:

  1. Which set of comparisons of wt/ko wt/het do we use to exclude/discount genes?
    1. Should it be a combination of all samples wt vs. x?
    2. Should it be only the ‘relevant’ comparison, e.g. if we are comparing p08_dlgn_het vs. p08_scn_het; do we remove genes observed in (p08_dlgn_het/wt && p08_scn_het/wt)
  2. Do we instead attempt to use this x/wt information to normalize the expression values in the other conditions and keep those genes?

Theresa’s current worksheet implements a version of 1b in which she separated the various input gene sets to define the exclusion genes. I am going to repeat this, but leave the starting data structure intact.

In this first iteration, I will do that by creating a simplified model of the data which combines the time/genotype/location and using sva. In my next iteration I will use a full statistical model containing each of those factors (and probably also using sva).

Note: my color choices are kind of garbage.

In addition, the exclusion dataset is the same as the analysis dataset, it is really only the contrasts which will be different.

v3_pairwise_input <- set_conditions(mm38_hisat_v3, fact = "time_geno_loc",
                                    colors = color_choices[["all"]])
## The numbers of samples by condition are:
## 
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Warning in set_se_colors(new_se, colors = colors): Colors for the following
## categories are not being used: p60_wt_dlgn, p60_wt_retina, p60_wt_scn.

20.2 A heatmap of specific genes across all conditions

all_cond_gene_heatmap_start <- normalize(v3_pairwise_input, filter = "simple",
                                         length_column = "cds_length",
                                         convert = "rpkm", transform = "log2")
## Removing 5517 low-count genes (19908 remaining).
## There appear to be 6387 genes without a length.
## transform_counts: Found 592846 values equal to 0, adding 1 to the matrix.
all_cond_gene_heatmap_input <- median_by_factor(all_cond_gene_heatmap_start)
## The factor p08_het_dlgn has 3 rows.
## The factor p08_het_retina has 3 rows.
## The factor p08_het_scn has 3 rows.
## The factor p08_ko_dlgn has 3 rows.
## The factor p08_ko_retina has 3 rows.
## The factor p08_ko_scn has 3 rows.
## The factor p08_wt_dlgn has 5 rows.
## The factor p08_wt_retina has 5 rows.
## The factor p08_wt_scn has 3 rows.
## The factor p15_het_dlgn has 4 rows.
## The factor p15_het_retina has 4 rows.
## The factor p15_het_scn has 3 rows.
## The factor p15_ko_dlgn has 3 rows.
## The factor p15_ko_retina has 3 rows.
## The factor p15_ko_scn has 3 rows.
## The factor p15_wt_dlgn has 5 rows.
## The factor p15_wt_retina has 5 rows.
## The factor p15_wt_scn has 2 rows.
all_cond_mtrx <- all_cond_gene_heatmap_input[["medians"]]
color_order <- colnames(all_cond_mtrx)
na_idx <- is.na(all_cond_mtrx)
all_cond_mtrx[na_idx] <- 0
variances <- matrixStats::rowVars(as.matrix(all_cond_mtrx))
variant_genes <- variances > 6.2
input_mtrx <- all_cond_mtrx[variant_genes, ]
cond_colors <- get_colors_by_condition(v3_pairwise_input, levels = color_order)
dim(input_mtrx)
## [1] 104  18
pp(file = "04inclusion_comparisons/top_104_variant_rpkm_genes_heatmap.pdf")
## Warning in pp(file =
## "04inclusion_comparisons/top_104_variant_rpkm_genes_heatmap.pdf"): The
## directory: 04inclusion_comparisons does not exist, will attempt to create it.
gplots::heatmap.2(as.matrix(input_mtrx), scale = "none", trace = "none",
                  ColSideColors = cond_colors)
dev.off()
## png 
##   2

21 TODO: Change the above to subtract wt

Rashmi suggested we should do the above plot after subtracting the wt counts. This is a good idea, but it will have to wait until we finish the current set.

21.1 Set up the contrasts

In the following few blocks I will set up the various comparisons of interest. Starting with the set of genes to exclude because they were observed to bind non-specifically in the wt samples.

21.1.1 Inclusion contrasts

In each exclusion I will have the contrast first followed by the pair of contrasts which will be used to define the gene set to exclude.

  • p15_het_dlgn/p08_het_dlgn: p15_wt_dlgn/p15_het_dlgn, p08_wt_dlgn/p08_het_dlgn; remove the genes increased in wt.
  • p15_ko_scn/p08_ko_scn: p15_wt_scn/p15_ko_scn, p08_wt_scn/p15_ko_scn
  • p15_het_retina/p08_het_retina: I think you get it, wt/het for both p15 retinas and p08 retinas…

Put slightly differently, for every term of interest I will create a contrast with the wt as numerator and the desired term as denominator, then pull out the genes increased in wt.

inclusions <- list(
  ## I like alphabetizing things, start with dlgn
  "p15_het_dlgn" = c("p15_het_dlgn", "p15_wt_dlgn"),
  "p08_het_dlgn" = c("p08_het_dlgn", "p08_wt_dlgn"),
  "p15_ko_dlgn" = c("p15_ko_dlgn", "p15_wt_dlgn"),
  "p08_ko_dlgn" = c("p08_ko_dlgn", "p08_wt_dlgn"),
  ## Then retinas
  "p15_het_retina" = c("p15_het_retina", "p15_wt_retina"),
  "p08_het_retina" = c("p08_het_retina", "p08_wt_retina"),
  "p15_ko_retina" = c("p15_ko_retina", "p15_wt_retina"),
  "p08_ko_retina" = c("p08_ko_retina", "p08_wt_retina"),
  ## Then scn
  "p15_het_scn" = c("p15_het_scn", "p15_wt_scn"),
  "p08_het_scn" = c("p08_het_scn", "p08_wt_scn"),
  "p15_ko_scn" = c("p15_ko_scn", "p15_wt_scn"),
  "p08_ko_scn" = c("p08_ko_scn", "p08_wt_scn"))

21.1.2 Time contrasts

For each location/genotype of interest, let us compare p15/p08

time_keepers <- list(
  ## DLGN
  "t_het_dlgn" = c("p15_het_dlgn", "p08_het_dlgn"),
  "t_ko_dlgn" = c("p15_ko_dlgn", "p08_ko_dlgn"),
  ## Retina
  "t_het_retina" = c("p15_het_retina", "p08_het_retina"),
  "t_ko_retina" = c("p15_ko_retina", "p08_ko_retina"),
  ## SCN
  "t_het_scn" = c("p15_het_scn", "p08_het_scn"),
  "t_ko_scn" = c("p15_ko_scn", "p08_ko_scn"))

21.1.3 Location contrasts

Compare locations and keep time/genotype consistent. I will use the location initials to define numerator/denominator.

location_keepers <- list(
  ## dlgn/retina
  "dr_p08_het" = c("p08_het_dlgn", "p08_het_retina"),
  "dr_p15_het" = c("p15_het_dlgn", "p15_het_retina"),
  "dr_p08_ko" = c("p08_ko_dlgn", "p08_ko_retina"),
  "dr_p15_ko" = c("p15_ko_dlgn", "p15_ko_retina"),
  ## scn/retina
  "sr_p08_het" = c("p08_het_scn", "p08_het_retina"),
  "sr_p15_het" = c("p15_het_scn", "p15_het_retina"),
  "sr_p08_ko" = c("p08_ko_scn", "p08_ko_retina"),
  "sr_p15_ko" = c("p15_ko_scn", "p15_ko_retina"),
  ## dlgn/scn
  "ds_p08_het" = c("p08_het_dlgn", "p08_het_scn"),
  "ds_p15_het" = c("p15_het_dlgn", "p15_het_scn"),
  "ds_p08_ko" = c("p08_ko_dlgn", "p08_ko_scn"),
  "ds_p15_ko" = c("p15_ko_dlgn", "p15_ko_scn"))

21.1.4 Genotype contrasts

Compare ko/het while keeping time/location constant. Similarly, use the initials to denote numerator/denominator, which will always be kh.

genotype_keepers <- list(
  ## DLGN
  "kh_p08_dlgn" = c("p08_ko_dlgn", "p08_het_dlgn"),
  "kh_p15_dlgn" = c("p15_ko_dlgn", "p15_het_dlgn"),
  ## Retina
  "kh_p08_retina" = c("p08_ko_retina", "p08_het_retina"),
  "kh_p15_retina" = c("p15_ko_retina", "p15_het_retina"),
  ## SCN
  "kh_p08_scn" = c("p08_ko_scn", "p08_het_scn"),
  "kh_p15_scn" = c("p15_ko_scn", "p15_het_scn"))

21.2 Perform the exclusion comparison

My all_pairwise() function now has a parameter which allows me to choose which contrasts to perform instead of literally doing every possible comparison. That is well suited for these operations:

In a container, the following appears to fail with:

“error code 1 from Lapack routine ‘dgesdd’”

Running it manually outside the container results in it working without error. I assume therefore that the problem lies in the compilation flags of LAPACK in the container.

Note: This problem was fixed by removing some parallelization.

inclusion_de <- all_pairwise(
  v3_pairwise_input, filter = "simple", model_fstring = default_fstring,
  keepers = inclusions, model_svs = "svaseq")
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 5517 low-count genes (19908 remaining).
## Basic step 0/3: Normalizing data.
## Basic step 0/3: Converting data.
## I think this is failing? SummarizedExperiment
## Basic step 0/3: Transforming data.
## Setting 394442 entries to zero.
## This received a matrix of SVs.
## converting counts to integer mode
## gene-wise dispersion estimates
## mean-dispersion relationship
## final dispersion estimates
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2

inclusion_de
## A pairwise differential expression with results from: basic, deseq, ebseq, edger, limma, noiseq.
## This used a surrogate/batch estimate from: svaseq.
## The primary analysis performed 12 comparisons.
inclusion_tables <- combine_de_tables(
  inclusion_de, keepers = inclusions, label_column = label_column,
  excel = glue("04inclusion_comparisons/inclusion_tables-v{ver}.xlsx"))
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
inclusion_tables
## A set of combined differential expression results.
##                              table deseq_sigup deseq_sigdown edger_sigup
## 1      p15_het_dlgn_vs_p15_wt_dlgn         536           971         668
## 2      p08_het_dlgn_vs_p08_wt_dlgn          15            79          57
## 3       p15_ko_dlgn_vs_p15_wt_dlgn         903          1333        1029
## 4       p08_ko_dlgn_vs_p08_wt_dlgn         117           282         133
## 5  p15_het_retina_vs_p15_wt_retina         151            58         193
## 6  p08_het_retina_vs_p08_wt_retina         432           110         464
## 7   p15_ko_retina_vs_p15_wt_retina         107            34         178
## 8   p08_ko_retina_vs_p08_wt_retina         533           155         551
## 9        p15_het_scn_vs_p15_wt_scn          11             8          39
## 10       p08_het_scn_vs_p08_wt_scn          48             8          57
## 11        p15_ko_scn_vs_p15_wt_scn           7             5          28
## 12        p08_ko_scn_vs_p08_wt_scn          31            26          83
##    edger_sigdown limma_sigup limma_sigdown
## 1           1013         582           833
## 2            107          40            46
## 3           1346         837          1045
## 4            297         186           356
## 5            114         119            57
## 6            162         315            88
## 7             62          39            25
## 8            224         404           144
## 9             28          19            34
## 10            45          42            20
## 11            30          39            39
## 12            54          65            55
## Warning: `aes_string()` was deprecated in ggplot2 3.0.0.
## i Please use tidy evaluation idioms with `aes()`.
## i See also `vignette("ggplot2-in-packages")` for more information.
## i The deprecated feature was likely used in the UpSetR package.
##   Please report the issue to the authors.
## This warning is displayed once per session.
## Call `lifecycle::last_lifecycle_warnings()` to see where this warning was
## generated.
## Warning: The `size` argument of `element_line()` is deprecated as of ggplot2 3.4.0.
## i Please use the `linewidth` argument instead.
## i The deprecated feature was likely used in the UpSetR package.
##   Please report the issue to the authors.
## This warning is displayed once per session.
## Call `lifecycle::last_lifecycle_warnings()` to see where this warning was
## generated.
## Plot describing unique/shared genes in a differential expression table.

## 202603: I successfully recapitulated previous non-container result.

inclusion_sig <- extract_significant_genes(
  inclusion_tables, lfc = lfc_cutoff, p = adjp_cutoff, according_to = "deseq",
  excel = glue("04inclusion_comparisons/inclusion_sig-v{ver}.xlsx"))
inclusion_sig
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 0.1 adj P cutoff: 0.1
##                deseq_up deseq_down
## p15_het_dlgn       2067       2381
## p08_het_dlgn        180        349
## p15_ko_dlgn        2544       2633
## p08_ko_dlgn         397        572
## p15_het_retina      402        223
## p08_het_retina      976        380
## p15_ko_retina       347         93
## p08_ko_retina      1551        859
## p15_het_scn          15          9
## p08_het_scn         225         55
## p15_ko_scn            8          6
## p08_ko_scn          112        116

202505: A strange thing happened here in this iteration: the plot of the significant genes is the exact same as the previous iteration; but the table of numbers of genes looks different.

For example, the previous table showed: p15_het_dlgn with 2067 up and 2381 down. The plot shows exactly that; but the new table shows 607 up and 1229 down. Let us check the actual data structure and see what is up?

I think I get it: when we do the extract_significant_genes above, we explicitly set a non-standard p-value and logFC because we are explicitly attempting to use a very loose definition of the set of genes which are in greater abundance than their most similar wild-type. However, when I create the barplot of significant genes; those values are explicitly set to 0,1,2 logFCs and p-value 0.05. Therefore, what I need to do, in order to check consistency, is to repeat this call but with the default FC/p values and see what the numbers look like.

test_inclusion <- extract_significant_genes(
  inclusion_tables, according_to = "deseq", excel = "excel/default_inclusion_sig.xlsx")
test_inclusion
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##                deseq_up deseq_down
## p15_het_dlgn        536        971
## p08_het_dlgn         15         79
## p15_ko_dlgn         903       1333
## p08_ko_dlgn         117        282
## p15_het_retina      151         58
## p08_het_retina      432        110
## p15_ko_retina       107         34
## p08_ko_retina       533        155
## p15_het_scn          11          8
## p08_het_scn          48          8
## p15_ko_scn            7          5
## p08_ko_scn           31         26

Yeah, I think this makes sense; what I need to do: change the significant bar plot so that it uses the lfc cutoff argument as the second of its 3 cutoffs. That should ensure that these numbers are consistent across analyses and parameters provided.

dim(inclusion_sig$deseq$ups$p15_het_dlgn)
## [1] 2067   75
test_all_up <- inclusion_tables$data$p15_het_dlgn[["deseq_logfc"]] > 0.1 &
  inclusion_tables$data$p15_het_dlgn[["deseq_adjp"]] <= 0.1
summary(test_all_up)
##    Mode   FALSE    TRUE 
## logical   17841    2067

Ohh, I get it, when I was testing this out manually, I set the logFC to 1.0 instead of the very minimal 0.1 we have been using for this!

22 MA/Volcano plots of x vs wt

Rashmi asked to see the comparisons against wt; I will name each file xw to show that it is x vs wt. for whatever other parameters are being examined. It is likely that some colors will be wrong because this is my first time creating these plots and we are doing them manually.

22.1 p15_het_dlgn

allc <- color_choices[["all"]]
table_name <- "p15_het_dlgn"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p15_het_dlgn"
denom <- "p15_wt_dlgn"
hw_p15_dlgn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)

pp(file = "05inclusion_volcano_ma/hw_p15_dlgn_volcano.pdf", width = 9, height = 9)
## Warning in pp(file = "05inclusion_volcano_ma/hw_p15_dlgn_volcano.pdf", width =
## 9, : The directory: 05inclusion_volcano_ma does not exist, will attempt to
## create it.
hw_p15_dlgn_volcano[["plot"]]
plotted <- dev.off()
hw_p15_dlgn_volcano[["plot"]]

hw_p15_dlgn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)

pp(file = "05inclusion_volcano_ma/hw_p15_dlgn_ma.pdf", width = 9, height = 9)
hw_p15_dlgn_ma[["plot"]]
plotted <- dev.off()
hw_p15_dlgn_ma[["plot"]]

22.2 p08_het_dlgn

table_name <- "p08_het_dlgn"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p08_het_dlgn"
denom <- "p08_wt_dlgn"
hw_p08_dlgn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/hw_p08_dlgn_volcano.pdf", width = 9, height = 9)
hw_p08_dlgn_volcano[["plot"]]
plotted <- dev.off()
hw_p08_dlgn_volcano[["plot"]]

hw_p08_dlgn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/hw_p08_dlgn_ma.pdf", width = 9, height = 9)
hw_p08_dlgn_ma[["plot"]]
plotted <- dev.off()
hw_p08_dlgn_ma[["plot"]]

22.3 p15_ko_dlgn

table_name <- "p15_ko_dlgn"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p15_ko_dlgn"
denom <- "p15_wt_dlgn"
kw_p15_dlgn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/kw_p15_dlgn_volcano.pdf", width = 9, height = 9)
kw_p15_dlgn_volcano[["plot"]]
plotted <- dev.off()
kw_p15_dlgn_volcano[["plot"]]

kw_p15_dlgn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/kw_p15_dlgn_ma.pdf", width = 9, height = 9)
kw_p15_dlgn_ma[["plot"]]
plotted <- dev.off()
kw_p15_dlgn_ma[["plot"]]

22.4 p08_ko_dlgn

table_name <- "p08_ko_dlgn"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p08_ko_dlgn"
denom <- "p08_wt_dlgn"
kw_p08_dlgn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/kw_p08_dlgn_volcano.pdf", width = 9, height = 9)
kw_p08_dlgn_volcano[["plot"]]
plotted <- dev.off()
kw_p08_dlgn_volcano[["plot"]]

kw_p08_dlgn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/kw_p08_dlgn_ma.pdf", width = 9, height = 9)
kw_p08_dlgn_ma[["plot"]]
plotted <- dev.off()
kw_p08_dlgn_ma[["plot"]]

22.5 p15_het_retina

table_name <- "p15_het_retina"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p15_het_retina"
denom <- "p15_wt_retina"
hw_p15_retina_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/hw_p15_retina_volcano.pdf", width = 9, height = 9)
hw_p15_retina_volcano[["plot"]]
plotted <- dev.off()
hw_p15_retina_volcano[["plot"]]

hw_p15_retina_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/hw_p15_retina_ma.pdf", width = 9, height = 9)
hw_p15_retina_ma[["plot"]]
plotted <- dev.off()
hw_p15_retina_ma[["plot"]]

22.6 p08_het_retina

table_name <- "p08_het_retina"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p08_het_retina"
denom <- "p08_wt_retina"
hw_p08_retina_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/hw_p08_retina_volcano.pdf", width = 9, height = 9)
hw_p08_retina_volcano[["plot"]]
plotted <- dev.off()
hw_p08_retina_volcano[["plot"]]

hw_p08_retina_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/hw_p08_retina_ma.pdf", width = 9, height = 9)
hw_p08_retina_ma[["plot"]]
plotted <- dev.off()
hw_p08_retina_ma[["plot"]]

22.7 p15_ko_retina

table_name <- "p15_ko_retina"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p15_ko_retina"
denom <- "p15_wt_retina"
kw_p15_retina_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/kw_p15_retina_volcano.pdf", width = 9, height = 9)
kw_p15_retina_volcano[["plot"]]
plotted <- dev.off()
kw_p15_retina_volcano[["plot"]]

kw_p15_retina_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/kw_p15_retina_ma.pdf", width = 9, height = 9)
kw_p15_retina_ma[["plot"]]
plotted <- dev.off()
kw_p15_retina_ma[["plot"]]

22.8 p08_ko_retina

table_name <- "p08_ko_retina"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p08_ko_retina"
denom <- "p08_wt_retina"
kw_p08_retina_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/kw_p08_retina_volcano.pdf", width = 9, height = 9)
kw_p08_retina_volcano[["plot"]]
plotted <- dev.off()
kw_p08_retina_volcano[["plot"]]

kw_p08_retina_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma_p08_retina_ma.pdf", width = 9, height = 9)
kw_p08_retina_ma[["plot"]]
plotted <- dev.off()
kw_p08_retina_ma[["plot"]]

22.9 p15_het_scn

table_name <- "p15_het_scn"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p15_het_scn"
denom <- "p15_wt_scn"
hw_p15_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/hw_p15_scn_volcano.pdf", width = 9, height = 9)
hw_p15_scn_volcano[["plot"]]
plotted <- dev.off()
hw_p15_scn_volcano[["plot"]]

hw_p15_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/hw_p15_scn_ma.pdf", width = 9, height = 9)
hw_p15_scn_ma[["plot"]]
plotted <- dev.off()
hw_p15_scn_ma[["plot"]]

22.10 p08_het_scn

table_name <- "p08_het_scn"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p08_het_scn"
denom <- "p08_wt_scn"
hw_p08_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/hw_p08_scn_volcano.pdf", width = 9, height = 9)
hw_p08_scn_volcano[["plot"]]
plotted <- dev.off()
hw_p08_scn_volcano[["plot"]]

hw_p08_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/hw_p08_scn_ma.pdf", width = 9, height = 9)
hw_p08_scn_ma[["plot"]]
plotted <- dev.off()
hw_p08_scn_ma[["plot"]]

22.11 p15_ko_scn

table_name <- "p15_ko_scn"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p15_ko_scn"
denom <- "p15_wt_scn"
kw_p15_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/kw_p15_scn_volcano.pdf", width = 9, height = 9)
kw_p15_scn_volcano[["plot"]]
plotted <- dev.off()
kw_p15_scn_volcano[["plot"]]

kw_p15_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/kw_p15_scn_ma.pdf", width = 9, height = 9)
kw_p15_scn_ma[["plot"]]
plotted <- dev.off()
kw_p15_scn_ma[["plot"]]

22.12 p08_ko_scn

table_name <- "p08_ko_scn"
table <- inclusion_tables[["data"]][[table_name]]
num <- "p08_ko_scn"
denom <- "p08_wt_scn"
kw_p08_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = allc[[denom]], color_high = allc[[num]],
  label_column = "mgi_symbol", label = 10, alpha = 1.0,
  size = 4)
pp(file = "05inclusion_volcano_ma/kw_p08_scn_volcano.pdf", width = 9, height = 9)
kw_p08_scn_volcano[["plot"]]
plotted <- dev.off()
kw_p08_scn_volcano[["plot"]]

kw_p08_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = allc[[denom]], color_high = allc[[num]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = 10)
pp(file = "05inclusion_volcano_ma/kw_p08_scn_ma.pdf", width = 9, height = 9)
kw_p08_scn_ma[["plot"]]
plotted <- dev.off()
kw_p08_scn_ma[["plot"]]

See the shared/unique genes in these sets.

inclusion_upsets <- upsetr_sig(inclusion_sig)
inclusion_intersects <- write_upset_groups(
  inclusion_upsets, excel = "04inclusion_comparison/inclusion_gene_groups.xlsx")

22.12.1 Extract genes included for each set of contrasts

Now, using that function, pull out the gene IDs of genes we do not trust because they were too high in wt for every contrast we are likely to perform.

The following was a modified version of the inclusion function which is somewhat more restrictive.

extract_inclusions_strict <- function(inclusion_sig, inclusion_tables, inclusions, keepers,
                                      all_genes, according_to = "deseq", which = "ups") {
  retlist <- list()
  table_names <- names(inclusion_sig[[according_to]][[which]])
  for (c_num in seq_along(keepers)) {
    contrast <- names(keepers)[c_num]
    numerator_name <- keepers[[c_num]][1]
    denominator_name <- keepers[[c_num]][2]
    ## In my new branch I cleaned up the sanitizer function for contrasts so this is not needed.
    ## The following two lines are no longer needed because of the cleanups I performed.
    ##numerator_name <- gsub(x = numerator_name, pattern = "(het|ko|wt)", replacement = "_\\1_")
    ##denominator_name <- gsub(x = denominator_name, pattern = "(het|ko|wt)", replacement = "_\\1_")
    numerator_table <- inclusion_sig[[according_to]][[which]][[numerator_name]]
    numerator_genes <- rownames(numerator_table)
    denominator_table <- inclusion_sig[[according_to]][[which]][[denominator_name]]
    denominator_genes <- rownames(denominator_table)
    df_columns <- paste0("deseq_", c("logfc", "adjp", "den"))
    included_num <- inclusion_tables[["data"]][[numerator_name]][, df_columns]
    colnames(included_num) <- c("numerator_vs_wt_logfc", "numerator_vs_wt_adjp", "num_wt_mean_exprs")
    included_den <- inclusion_tables[["data"]][[denominator_name]][, df_columns]
    colnames(included_den) <- c("denominator_vs_wt_logfc", "denominator_vs_wt_adjp", "den_wt_mean_exprs")
    ## I think this is where things went wrong,
    ## compare this modified line to the original to prove it.
    included_df <- merge(included_num, included_den, by = "row.names")
    ## Previously, I did not specify the merge action, all = FALSE by default.
    ## This then will result in a difference in the rows observed
    ## included_df <- merge(included_num, included_den, by = "row.names", all = FALSE)
    rownames(included_df) <- included_df[["Row.names"]]
    included_df[["Row.names"]] <- NULL
    concatenated_genes <- c(numerator_genes, denominator_genes)
    both_gene_idx <- duplicated(concatenated_genes)
    genes_in_both <- concatenated_genes[both_gene_idx]
    message("The set of unique genes higher in ", numerator_name,
            " vs. wt is ", length(numerator_genes), ".")
    message("The set of unique genes higher in ", denominator_name,
            " vs. wt is ", length(denominator_genes), ".")
    message("The intersection of them is ", length(genes_in_both), " genes.")
    include_name <- paste0("inc_", contrast)
    include_idx <- all_genes %in% genes_in_both
    include_genes <- all_genes[include_idx]
    df_name <- paste0("df_", contrast)
    retlist[[df_name]] <- included_df
    written_inclusion <- write_xlsx(
      data = included_df,
      excel = glue("07included_strict_genes_excel/{include_name}-v{ver}.xlsx"))
    retlist[[include_name]] <- include_genes
    retlist[[contrast]] <- include_genes
  }
  return(retlist)
}

This is the pre-202505 version of this function.

extract_inclusions <- function(inclusion_sig, inclusion_tables, inclusions, keepers, all_genes,
                               according_to = "deseq", which = "ups") {
  retlist <- list()
  table_names <- names(inclusion_sig[[according_to]][[which]])
  for (c_num in seq_along(keepers)) {
    contrast <- names(keepers)[c_num]
    numerator_name <- keepers[[c_num]][1]
    denominator_name <- keepers[[c_num]][2]
    ## In my new branch I cleaned up the sanitizer function for contrasts so this is not needed.
    ## The following two lines are no longer needed because of the cleanups I performed.
    ##numerator_name <- gsub(x = numerator_name, pattern = "(het|ko|wt)", replacement = "_\\1_")
    ##denominator_name <- gsub(x = denominator_name, pattern = "(het|ko|wt)", replacement = "_\\1_")
    numerator_table <- inclusion_sig[[according_to]][[which]][[numerator_name]]
    numerator_genes <- rownames(numerator_table)
    denominator_table <- inclusion_sig[[according_to]][[which]][[denominator_name]]
    denominator_genes <- rownames(denominator_table)
    df_columns <- paste0("deseq_", c("logfc", "adjp", "den"))
    included_num <- inclusion_tables[["data"]][[numerator_name]][, df_columns]
    colnames(included_num) <- c("numerator_vs_wt_logfc", "numerator_vs_wt_adjp", "num_wt_mean_exprs")
    included_den <- inclusion_tables[["data"]][[denominator_name]][, df_columns]
    colnames(included_den) <- c("denominator_vs_wt_logfc", "denominator_vs_wt_adjp", "den_wt_mean_exprs")
    included_df <- merge(included_num, included_den, by = "row.names")
    rownames(included_df) <- included_df[["Row.names"]]
    included_df[["Row.names"]] <- NULL
    include_genes <- unique(c(numerator_genes, denominator_genes))
    message("The set of unique genes higher in ", numerator_name,
            " vs. wt is ", length(numerator_genes), ".")
    message("The set of unique genes higher in ", denominator_name,
            " vs. wt is ", length(denominator_genes), ".")
    message("The unique union of them is ", length(include_genes), " genes.")
    include_name <- paste0("inc_", contrast)
    include_idx <- all_genes %in% include_genes
    include_genes <- all_genes[include_idx]
    df_name <- paste0("df_", contrast)
    retlist[[df_name]] <- included_df
    written_inclusion <- write_xlsx(data = included_df,
                                    excel = glue("included_genes/{include_name}-v{ver}.xlsx"))
    retlist[[include_name]] <- include_genes
    retlist[[contrast]] <- include_genes
  }
  return(retlist)
}

23 ‘Normal’ Inclusion extraction

Here is the full set of gene IDs

all_genes <- rownames(assay(v3_pairwise_input))

In the following blocks I am including the union of genes observed higher than wt in either of the numerator or denominator for each contrast.

23.1 Time

time_inclusions <- extract_inclusions(inclusion_sig, inclusion_tables, inclusions,
                                      time_keepers, all_genes)
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The unique union of them is 2113 genes.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The unique union of them is 2716 genes.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The unique union of them is 1086 genes.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The unique union of them is 1664 genes.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The unique union of them is 238 genes.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The unique union of them is 120 genes.

23.2 Location

location_inclusions <- extract_inclusions(inclusion_sig, inclusion_tables, inclusions,
                                          location_keepers, all_genes)
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The unique union of them is 1134 genes.
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The unique union of them is 2361 genes.
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The unique union of them is 1883 genes.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The unique union of them is 2843 genes.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The unique union of them is 1188 genes.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The unique union of them is 417 genes.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The unique union of them is 1624 genes.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The unique union of them is 355 genes.
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The unique union of them is 402 genes.
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The unique union of them is 2080 genes.
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The unique union of them is 493 genes.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The unique union of them is 2550 genes.

23.3 Genotype

genotype_inclusions <- extract_inclusions(inclusion_sig, inclusion_tables, inclusions,
                                          genotype_keepers, all_genes)
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The unique union of them is 501 genes.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The unique union of them is 2773 genes.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The unique union of them is 1760 genes.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The unique union of them is 571 genes.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The unique union of them is 312 genes.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The unique union of them is 18 genes.

24 The strict ‘inclusion’ set

24.1 Time

time_inclusions_strict <- extract_inclusions_strict(inclusion_sig, inclusion_tables, inclusions,
                                                    time_keepers, all_genes)
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The intersection of them is 134 genes.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The intersection of them is 225 genes.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The intersection of them is 292 genes.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The intersection of them is 234 genes.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The intersection of them is 2 genes.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The intersection of them is 0 genes.

24.2 Location

location_inclusions_strict <- extract_inclusions_strict(inclusion_sig, inclusion_tables, inclusions,
                                                        location_keepers, all_genes)
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The intersection of them is 22 genes.
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The intersection of them is 108 genes.
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The intersection of them is 65 genes.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The intersection of them is 48 genes.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The intersection of them is 13 genes.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The intersection of them is 0 genes.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The intersection of them is 39 genes.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The intersection of them is 0 genes.
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The intersection of them is 3 genes.
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The intersection of them is 2 genes.
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The intersection of them is 16 genes.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The intersection of them is 2 genes.

24.3 Genotype

genotype_inclusions_strict <- extract_inclusions_strict(inclusion_sig, inclusion_tables, inclusions,
                                                        genotype_keepers, all_genes)
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The intersection of them is 76 genes.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The intersection of them is 1838 genes.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The intersection of them is 767 genes.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The intersection of them is 178 genes.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The intersection of them is 25 genes.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The intersection of them is 5 genes.

24.3.1 Check vs Theresa’s filter

Up above Theresa performed a 0.25 log2FC and 0.05 adjp filter which provided a set of 2,640 genes observed higher in the p08 het retinas vs. wt retinas. I should see that in this inclusion_sig data structure.

There is an important caveat though: in Theresa’s filter above, she did a DE of only the retina samples but I did all samples. I expected that this would result in basically the same result (I actually assumed I would get a few more genes), but instead it appears to have retrieved a significantly smaller number of genes (about 1/2, happily they pretty much all appear in the previous filter). As a result, I am going to try relaxing my constraints slightly to see if I can recapitulate her filter (which would match Theresa’s later filter, though I guess that in turn will lead to a smaller set of genes compared to her later, relaxed 0.1 filter).

comparison <- inclusion_sig[["deseq"]][["ups"]][["p08_het_retina"]]
comp <- list(
  "taa" = taa_keepers,
  "new" = rownames(comparison))
test_comparison <- Vennerable::Venn(comp)
Vennerable::plot(test_comparison)

I want to have a little function which, given a contrast of interest, will extract the gene sets which should be included/excluded given the above.

write_all_cp <- function(all_cp, prefix = "12", suffix = "") {
  all_written <- list()
  for (g in seq_len(length(all_cp))) {
    name <- names(all_cp)[g]
    datum <- all_cp[[name]]
    filename <- glue("{prefix}enrichment_excel/{name}_cprofiler{suffix}-v{ver}.xlsx")
    written <- sm(write_cp_data(datum, excel = filename))
    all_written[[g]] <- written
  }
  return(all_written)
}
write_all_gp <- function(all_gp, prefix = "13", suffix = "") {
  all_written <- list()
  for (g in seq_len(length(all_gp))) {
    name <- names(all_gp)[g]
    datum <- all_gp[[name]]
    filename <- glue("{prefix}enrichment_excel/{name}_gprofiler{suffix}-v{ver}.xlsx")
    written <- sm(write_gprofiler_data(datum, excel = filename))
    all_written[[g]] <- written
  }
  return(all_written)
}
write_all_en <- function(all_en, prefix = "14", suffix = "") {
  all_written <- list()
  for (e in seq_len(length(all_en))) {
    name <- names(all_en)[e]
    datum <- all_en[[name]]
    filename <- glue("{prefix}enrichment_excel/{name}_enricher{suffix}-v{ver}.xlsx")
    written <- sm(write_enricher_data(datum, excel = filename))
    all_written[[e]] <- written
  }
  return(all_written)
}

24.3.2 Extract genes included for each set of contrasts

Now, using that function, pull out the gene IDs of genes we do not trust because they were too high in wt for every contrast we are likely to perform.

all_genes <- rownames(assay(v3_pairwise_input))
time_inclusions <- extract_inclusions(inclusion_sig, inclusion_tables, inclusions,
                                      time_keepers, all_genes)
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The unique union of them is 2113 genes.
## Deleting the file included_genes/inc_t_het_dlgn-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The unique union of them is 2716 genes.
## Deleting the file included_genes/inc_t_ko_dlgn-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The unique union of them is 1086 genes.
## Deleting the file included_genes/inc_t_het_retina-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The unique union of them is 1664 genes.
## Deleting the file included_genes/inc_t_ko_retina-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The unique union of them is 238 genes.
## Deleting the file included_genes/inc_t_het_scn-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The unique union of them is 120 genes.
## Deleting the file included_genes/inc_t_ko_scn-v20260331.xlsx before writing the tables.
location_inclusions <- extract_inclusions(inclusion_sig, inclusion_tables, inclusions,
                                          location_keepers, all_genes)
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The unique union of them is 1134 genes.
## Deleting the file included_genes/inc_dr_p08_het-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The unique union of them is 2361 genes.
## Deleting the file included_genes/inc_dr_p15_het-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The unique union of them is 1883 genes.
## Deleting the file included_genes/inc_dr_p08_ko-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The unique union of them is 2843 genes.
## Deleting the file included_genes/inc_dr_p15_ko-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The unique union of them is 1188 genes.
## Deleting the file included_genes/inc_sr_p08_het-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The unique union of them is 417 genes.
## Deleting the file included_genes/inc_sr_p15_het-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The unique union of them is 1624 genes.
## Deleting the file included_genes/inc_sr_p08_ko-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The unique union of them is 355 genes.
## Deleting the file included_genes/inc_sr_p15_ko-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The unique union of them is 402 genes.
## Deleting the file included_genes/inc_ds_p08_het-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The unique union of them is 2080 genes.
## Deleting the file included_genes/inc_ds_p15_het-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The unique union of them is 493 genes.
## Deleting the file included_genes/inc_ds_p08_ko-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The unique union of them is 2550 genes.
## Deleting the file included_genes/inc_ds_p15_ko-v20260331.xlsx before writing the tables.
genotype_inclusions <- extract_inclusions(inclusion_sig, inclusion_tables, inclusions,
                                          genotype_keepers, all_genes)
## The set of unique genes higher in p08_ko_dlgn vs. wt is 397.
## The set of unique genes higher in p08_het_dlgn vs. wt is 180.
## The unique union of them is 501 genes.
## Deleting the file included_genes/inc_kh_p08_dlgn-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_ko_dlgn vs. wt is 2544.
## The set of unique genes higher in p15_het_dlgn vs. wt is 2067.
## The unique union of them is 2773 genes.
## Deleting the file included_genes/inc_kh_p15_dlgn-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p08_ko_retina vs. wt is 1551.
## The set of unique genes higher in p08_het_retina vs. wt is 976.
## The unique union of them is 1760 genes.
## Deleting the file included_genes/inc_kh_p08_retina-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_ko_retina vs. wt is 347.
## The set of unique genes higher in p15_het_retina vs. wt is 402.
## The unique union of them is 571 genes.
## Deleting the file included_genes/inc_kh_p15_retina-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p08_ko_scn vs. wt is 112.
## The set of unique genes higher in p08_het_scn vs. wt is 225.
## The unique union of them is 312 genes.
## Deleting the file included_genes/inc_kh_p08_scn-v20260331.xlsx before writing the tables.
## The set of unique genes higher in p15_ko_scn vs. wt is 8.
## The set of unique genes higher in p15_het_scn vs. wt is 15.
## The unique union of them is 18 genes.
## Deleting the file included_genes/inc_kh_p15_scn-v20260331.xlsx before writing the tables.

24.4 Perform the DE analyses and exclude the target genes

24.4.1 Genotype

genotype_de <- all_pairwise(v3_pairwise_input, filter = TRUE, model_fstring = default_fstring,
                            keepers = genotype_keepers, model_svs = "svaseq")
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 10162 low-count genes (15263 remaining).
## Basic step 0/3: Normalizing data.
## Basic step 0/3: Converting data.
## I think this is failing? SummarizedExperiment
## Basic step 0/3: Transforming data.
## Setting 109425 entries to zero.
## This received a matrix of SVs.
## converting counts to integer mode
## gene-wise dispersion estimates
## mean-dispersion relationship
## final dispersion estimates
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2

genotype_de
## A pairwise differential expression with results from: basic, deseq, ebseq, edger, limma, noiseq.
## This used a surrogate/batch estimate from: svaseq.
## The primary analysis performed 6 comparisons.

24.4.2 Location

location_de <- all_pairwise(v3_pairwise_input, filter = TRUE, model_fstring = default_fstring,
                            keepers = location_keepers, model_svs = "svaseq")
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 10162 low-count genes (15263 remaining).
## Basic step 0/3: Normalizing data.
## Basic step 0/3: Converting data.
## I think this is failing? SummarizedExperiment
## Basic step 0/3: Transforming data.
## Setting 109425 entries to zero.
## This received a matrix of SVs.
## converting counts to integer mode
## gene-wise dispersion estimates
## mean-dispersion relationship
## final dispersion estimates
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2

location_de
## A pairwise differential expression with results from: basic, deseq, ebseq, edger, limma, noiseq.
## This used a surrogate/batch estimate from: svaseq.
## The primary analysis performed 12 comparisons.

24.4.3 Time

time_de <- all_pairwise(v3_pairwise_input, filter = TRUE, model_fstring = default_fstring,
                        keepers = time_keepers, model_svs = "svaseq")
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 10162 low-count genes (15263 remaining).
## Basic step 0/3: Normalizing data.
## Basic step 0/3: Converting data.
## I think this is failing? SummarizedExperiment
## Basic step 0/3: Transforming data.
## Setting 109425 entries to zero.
## This received a matrix of SVs.
## converting counts to integer mode
## gene-wise dispersion estimates
## mean-dispersion relationship
## final dispersion estimates
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2

time_de
## A pairwise differential expression with results from: basic, deseq, ebseq, edger, limma, noiseq.
## This used a surrogate/batch estimate from: svaseq.
## The primary analysis performed 6 comparisons.

It is near here when the computer sometimes fails with no more tempfiles. In another window I am messing with tempfile() in R to try to understand where it is going off the rails…

24.5 Extract the relevant tables and include genes lower in wt

24.5.1 Genotype contrasts

I will start with the tables and no inclusions so I can check my work.

In this first block I will explain a little more thoroughly what is going on:

  1. Dump the full table of the contrasts I defined above comparing the 3 genotypes across time/location.
  2. Iterate over each of those contrasts and do the following:
    1. Extract the name of the contrast, ‘kh_p08_dlgn’ for example
    2. Yank out that specific entry from the keeper list and its name
    3. Yank out the corresponding set of genes to include from the inclusions data structure.
    4. Create a filename given the name in (a) above and the logFC cutoff chosen for the inclusions (I am assuming we may change this)
    5. Given (b), (c), and (d), extract the corresponding table from the differential expression analysis and include the appropriate genes.
    FIXME: my gprofiler function just assumes human and so if passed mmusculus will incorrectly attempt to connect to non-existant databases. Let us fix that now.
genotype_tables_full <- combine_de_tables(
  genotype_de, keepers = genotype_keepers, label_column = label_column,
  fancy = TRUE,
  excel = glue("08full_contrasts_excel/genotype_full_tables-v{ver}.xlsx"))
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
genotype_tables_full
## A set of combined differential expression results.
##                             table deseq_sigup deseq_sigdown edger_sigup
## 1     p08_ko_dlgn_vs_p08_het_dlgn          24             2          42
## 2     p15_ko_dlgn_vs_p15_het_dlgn          50             2          81
## 3 p08_ko_retina_vs_p08_het_retina           6             2           9
## 4 p15_ko_retina_vs_p15_het_retina           9             5           6
## 5       p08_ko_scn_vs_p08_het_scn          54           135          80
## 6       p15_ko_scn_vs_p15_het_scn           0            16           3
##   edger_sigdown limma_sigup limma_sigdown
## 1             1          41             3
## 2             3           0             0
## 3             2           3             1
## 4             4           0             3
## 5           139          32            28
## 6            29           0             1
## Plot describing unique/shared genes in a differential expression table.

genotype_sig_full <- extract_significant_genes(
  genotype_tables_full, according_to = "deseq",
  excel = glue("08full_contrasts_excel/genotype_full_sig-v{ver}.xlsx"))
genotype_sig_full
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##               deseq_up deseq_down
## kh_p08_dlgn         24          2
## kh_p15_dlgn         50          2
## kh_p08_retina        6          2
## kh_p15_retina        9          5
## kh_p08_scn          54        135
## kh_p15_scn           0         16

24.5.2 Search Enrichment of these sets

In this run, we will search the full set of genes, next we will only do the inclusions.

genotype_full_gp <- all_gprofiler(genotype_sig_full, species = "mmusculus",
                                  excel = "09full_contrasts_enrich/genotype_full_gprofiler.xlsx")
genotype_full_cp <- all_cprofiler(genotype_sig_full, genotype_tables_full,
                                  orgdb = "org.Mm.eg.db", go_level = go_level, organism = "mouse",
                                  orgdb_from = orgdb_from, max_groupsize = max_groupsize,
                                  excel = "09full_contrasts_enrich/genotype_full_cprofiler.xlsx")
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error in `simple_cl[["kegg_universe"]]`:
## ! subscript out of bounds
genotype_full_upset <- upsetr_sig(genotype_sig_full)
genotype_full_intersects <- write_upset_groups(genotype_full_upset,
                                               excel = "09full_contrasts_intersections/genotype_full_gene_groups.xlsx")

Now separate the various genotype tables and perform the inclusions of the genes with relatively low wt values.

24.5.2.1 Extract filtered genes and pass to gprofiler2/clusterprofiler

genotype_tables <- list()
genotype_sig <- list()
genotype_gp <- list()
genotype_cp <- list()
genotype_en <- list()
for (k in seq_along(genotype_keepers)) {
  name <- names(genotype_keepers)[k]
  message("Examining ", name)
  keeper <- genotype_keepers[name]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- genotype_inclusions[[include_df_name]]
  includes <- genotype_inclusions[[include_name]]
  summary(rownames(genotype_sig_full[["deseq"]][["ups"]][[name]]) %in% includes)
  include_filename <- glue("10genotype_contrasts_excel/{name}_including_wt_{lfc_cutoff}_decreased_table-v{ver}.xlsx")
  include_sig_filename <- glue("10genotype_contrasts_excel/{name}_including_wt_{lfc_cutoff}_decreased_sig-v{ver}.xlsx")
  genotype_tables[[name]] <- combine_de_tables(
    genotype_de, extra_annot = include_df,
    keepers = keeper, label_column = label_column,
    excel = include_filename, wanted_genes = includes)
  print(genotype_tables[[name]])
  genotype_sig[[name]] <- extract_significant_genes(
    genotype_tables[[name]], according_to = "deseq",
    excel = include_sig_filename)
  print(genotype_sig[[name]])
  num_rows <- nrow(genotype_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(genotype_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  if (num_rows >= 10) {
    message("Performing gprofiler/clusterProfiler.")
    genotype_gp[[name]] <- all_gprofiler(genotype_sig[[name]], species = "mmusculus")
    genotype_cp[[name]] <- all_cprofiler(
      genotype_sig[[name]], genotype_tables[[name]],
      orgdb = "org.Mm.eg.db", orgdb_from = orgdb_from,
      go_level = go_level, max_groupsize = max_groupsize, organism = "mouse")
    #if (!is.null(get0("m2_gsc"))) {
    #  genotype_en[[name]] <- all_enricher(genotype_sig[[name]], gsc = m2_gsc,
    #                                      orgdb = "org.Mm.eg.db", from = "ENSEMBL", to = "SYMBOL")
    #}
    gp_written <- write_all_gp(genotype_gp[[name]], prefix = "11")
    cp_written <- write_all_cp(genotype_cp[[name]], prefix = "11")
    #en_written <- write_all_en(genotype_en[[name]])
  } else {
    warning("There are less than 10 genes up and down in the ", name, " comparison.")
    message("There are less than 10 genes up and down in the ", name, " comparison.")
  }
}
## Examining kh_p08_dlgn
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## A set of combined differential expression results.
##                         table deseq_sigup deseq_sigdown edger_sigup
## 1 p08_ko_dlgn_vs_p08_het_dlgn          23             1          30
##   edger_sigdown limma_sigup limma_sigdown
## 1             0          26             0
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## Plot describing unique/shared genes in a differential expression table.

## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##             deseq_up deseq_down
## kh_p08_dlgn       23          1

## There are 24 significant up and down genes.
## Performing gprofiler/clusterProfiler.
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error in `simple_cl[["kegg_universe"]]`:
## ! subscript out of bounds

Plot the results separately.

for (k in seq_along(genotype_keepers)) {
  name <- names(genotype_keepers)[k]
  message("Examining ", name)
  keeper <- genotype_keepers[name]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- genotype_inclusions[[include_df_name]]
  includes <- genotype_inclusions[[include_name]]
  summary(rownames(genotype_sig_full[["deseq"]][["ups"]][[name]]) %in% includes)
  num_rows <- nrow(genotype_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(genotype_sig[[name]][["deseq"]][["downs"]][[name]])
  nrow(genotype_sig[[name]][["deseq"]][["ups"]][[name]])
  nrow(genotype_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  ## #1 is up and #2 is down, avoiding typeos here.
  num_objects <- length(genotype_cp[[name]])
  if (num_objects == 0) {
    warning("Something went wrong in all_cprofiler.")
  } else {
    upp <- which(grepl(x = names(genotype_cp[[name]]), pattern = "_up$"))
    downp <- which(grepl(x = names(genotype_cp[[name]]), pattern = "_down$"))
    if (length(upp) > 0) {
      mf_sig <- genotype_cp[[name]][[upp]][["go_data"]][["MF_enrich"]]
      cc_sig <- genotype_cp[[name]][[upp]][["go_data"]][["CC_enrich"]]
      bp_sig <- genotype_cp[[name]][[upp]][["go_data"]][["BP_enrich"]]
      mf_plots_up <- plot_enrichresult(mf_sig)
      mf_tree_up_filename <- glue("12clusterProfiler_plots/{name}_up_mf_sig_tree.pdf")
      pp(file = mf_tree_up_filename)
      try(print(mf_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_up_filename <- glue("12clusterProfiler_plots/{name}_up_mf_sig_bar.pdf")
      pp(file = mf_bar_up_filename)
      try(print(mf_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_up <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_up_filename <- glue("12clusterProfiler_plots/{name}_up_cc_sig_tree.pdf")
      pp(file = cc_tree_up_filename)
      try(print(cc_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_up_filename <- glue("12clusterProfiler_plots/{name}_up_cc_sig_bar.pdf")
      pp(file = cc_bar_up_filename)
      try(print(cc_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_up_filename <- glue("12clusterProfiler_plots/{name}_up_bp_sig_tree.pdf")
      pp(file = bp_tree_up_filename)
      try(print(bp_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_up_filename <- glue("12clusterProfiler_plots/{name}_up_bp_sig_bar.pdf")
      pp(file = bp_bar_up_filename)
      try(print(bp_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
    if (length(downp) > 0) {
      mf_sig <- genotype_cp[[name]][[downp]][["go_data"]][["MF_enrich"]]
      cc_sig <- genotype_cp[[name]][[downp]][["go_data"]][["CC_enrich"]]
      bp_sig <- genotype_cp[[name]][[downp]][["go_data"]][["BP_enrich"]]
      mf_plots_down <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_down_filename <- glue("12clusterProfiler_plots/{name}_down_mf_sig_tree.pdf")
      pp(file = mf_tree_down_filename)
      try(print(mf_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_down_filename <- glue("12clusterProfiler_plots/{name}_down_mf_sig_bar.pdf")
      pp(file = mf_bar_down_filename)
      try(print(mf_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_down <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_down_filename <- glue("12clusterProfiler_plots/{name}_down_cc_sig_tree.pdf")
      pp(file = cc_tree_down_filename)
      try(print(cc_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_down_filename <- glue("12clusterProfiler_plots/{name}_down_cc_sig_bar.pdf")
      pp(file = cc_bar_down_filename)
      try(print(cc_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_down_filename <- glue("12clusterProfiler_plots/{name}_down_bp_sig_tree.pdf")
      pp(file = bp_tree_down_filename)
      try(print(bp_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, cateogories = go_categories)
      bp_bar_down_filename <- glue("12clusterProfiler_plots/{name}_down_bp_sig_bar.pdf")
      pp(file = bp_bar_down_filename)
      try(print(bp_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
  }
}
## Examining kh_p08_dlgn
## There are 24 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p15_dlgn
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p08_retina
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p15_retina
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p08_scn
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p15_scn
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.

A few specific plots of interest: Colenso asked to label a few genes for the knockout/het p08_retinas, p08_scn, and p08_dlgn: either the top-15 or all significant. I am pretty sure if I tell it 15 and there are not that many, it will just do the significant? Let us find out!

24.5.2.2 ko/het for p08 retinas

table_name <- "kh_p08_retina"
table_input <- genotype_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
interesting <- c("Opn4", "Gm9008", "Lrr1", "Cnbd1")
kh_p08_retina_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = colors[["het_retina"]], color_high = colors[["ko_retina"]],
  label_column = "mgi_symbol", label = interesting, alpha = 1.0,
  outline = outline, size = 4)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "13genotype_ma_volcano/kh_p08_retina_volcano.pdf", width = 9, height = 9)
## Warning in pp(file = "13genotype_ma_volcano/kh_p08_retina_volcano.pdf", : The
## directory: 13genotype_ma_volcano does not exist, will attempt to create it.
kh_p08_retina_volcano[["plot"]]
## Error:
## ! object 'kh_p08_retina_volcano' not found
plotted <- dev.off()
kh_p08_retina_volcano[["plot"]]
## Error:
## ! object 'kh_p08_retina_volcano' not found
kh_p08_retina_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_retina"]], color_high = colors[["het_retina"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", outline = outline,
  label = interesting)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "13genotype_ma_volcano/kh_p08_retina_ma.pdf", width = 9, height = 9)
kh_p08_retina_ma[["plot"]]
## Error:
## ! object 'kh_p08_retina_ma' not found
plotted <- dev.off()
kh_p08_retina_ma[["plot"]]
## Error:
## ! object 'kh_p08_retina_ma' not found

24.5.2.3 ko/het p08 SCN

I am going to make an executive decision for this plot, 15 is too many and makes it crazy cluttered.

24.5.3 Repeat this with two sets of genes

table_name <- "kh_p08_scn"
table_input <- genotype_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
interesting_genes <- c("Fign", "Nrn1", "Dpysl2", "Actb", "Fgf9", "Otx2", "Sec23",
                       "Ncam1", "Map4", "Sec22b", "Nlgn3", "Marcks", "Cd47",
                       "Dpysl3", "Lin7c", "Cadm1", "Snx12", "Rhoa", "Inpp5f",
                       "Atg12", "Set", "Gsk3b", "Pdcd4", "Gabra2", "Tmco1", "Anapc16")
kh_p08_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq
_adjp",
  label_column = "mgi_symbol", label = interesting_genes, size = 4, alpha = 1.0,
  outline = outline, color_low = colors[["ko_scn"]], color_high = colors[["het_scn"]])
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "13genotype_ma_volcano/kh_p08_scn_volcano.pdf", width = 9, height = 9)
kh_p08_scn_volcano[["plot"]]
## Error:
## ! object 'kh_p08_scn_volcano' not found
plotted <- dev.off()
kh_p08_scn_volcano[["plot"]]
## Error:
## ! object 'kh_p08_scn_volcano' not found
## why in the crap is it double-labelling!?
## My MA plotter isn't as smart as the volcano plotter, the genes are:
kh_p08_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_scn"]], color_high = colors[["het_scn"]],
  outline = outline, p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "13genotype_ma_volcano/kh_p08_scn_ma.pdf", width = 9, height = 9)
kh_p08_scn_ma[["plot"]]
## Error:
## ! object 'kh_p08_scn_ma' not found
plotted <- dev.off()
kh_p08_scn_ma[["plot"]]
## Error:
## ! object 'kh_p08_scn_ma' not found

24.5.3.1 Same plot but a different set of labeled genes

table_name <- "kh_p08_scn"
table_input <- genotype_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
interesting_genes <- c(
  "Anapc16", "Gabra2", "Tmco1", "Sod2", "Fgf9", "Pdcd4", "Rhoa", "Gsk3b", "Foxp1",
  "Ncam1", "Marcks", "Fign", "Dpysl3", "Inpp5f", "Cadm1", "Map4", "Ugcg", "Elovl4",
  "Elavl1", "Cfl2", "Tnnt1", "Gnb1", "Impact", "Nrn1", "Nlgn3", "Actb", "Cd47",
  "Sec22b", "Slc17a7", "Vglut1", "Actb", "B4galt5", "Foxp1", "Otx2", "Lin7c",
  "Snx12", "Atg12", "Set")
kh_p08_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["ko_scn"]], color_high = colors[["het_scn"]],
  outline = outline, label_column = "mgi_symbol", label = interesting_genes, size = 4, alpha = 1.0)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "13genotype_ma_volcano/kh_p08_scn_volcano_v2.pdf", width = 9, height = 9)
kh_p08_scn_volcano[["plot"]]
## Error:
## ! object 'kh_p08_scn_volcano' not found
plotted <- dev.off()
kh_p08_scn_volcano[["plot"]]
## Error:
## ! object 'kh_p08_scn_volcano' not found
## why in the crap is it double-labelling!?
## My MA plotter isn't as smart as the volcano plotter, the genes are:
kh_p08_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_scn"]], color_high = colors[["het_scn"]],
  outline = outline, p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "13genotype_ma_volcano/kh_p08_scn_ma_v2.pdf", width = 9, height = 9)
kh_p08_scn_ma[["plot"]]
## Error:
## ! object 'kh_p08_scn_ma' not found
plotted <- dev.off()
kh_p08_scn_ma[["plot"]]
## Error:
## ! object 'kh_p08_scn_ma' not found

24.5.3.2 ko/het p08 dLGN

table_name <- "kh_p08_dlgn"
table_input <- genotype_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
kh_p08_dlgn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["ko_dlgn"]], color_high = colors[["het_dlgn"]],
  outline = outline, label_column = "mgi_symbol", label = 10, size = 4, alpha = 1.0)
## Warning in ggrepel::geom_text_repel(data = df_subset, nudge_x = nudge_x, :
## Ignoring unknown parameters: `outline`
pp(file = "13genotype_ma_volcano/kh_p08_dlgn_volcano.pdf", width = 9, height = 9)
kh_p08_dlgn_volcano[["plot"]]
plotted <- dev.off()
kh_p08_dlgn_volcano[["plot"]]

## My MA plotter isn't as smart as the volcano plotter, the genes are:
kh_p08_dlgn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_dlgn"]], color_high = colors[["het_dlgn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = 10, outline = outline)
pp(file = "13genotype_ma_volcano/kh_p08_dlgn_ma.pdf", width = 9, height = 9)
kh_p08_dlgn_ma[["plot"]]
plotted <- dev.off()
kh_p08_dlgn_ma[["plot"]]

24.5.3.3 ko/het for p15 retinas

When last I ran this manually, it did not double-label, hopefully that remains true in the container.

table_name <- "kh_p15_retina"
table_input <- genotype_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
interesting <- c("Opn4", "Gm9008", "Lrr1", "Cnbd1")
kh_p15_retina_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp", fill = "black",
  color_low = colors[["ko_retina"]], color_high = colors[["het_retina"]],
  label_column = "mgi_symbol", label = interesting, alpha = 1.0,
  outline = outline, size = 4)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "13genotype_ma_volcano/kh_p15_retina_volcano.pdf", width = 9, height = 9)
kh_p15_retina_volcano[["plot"]]
## Error:
## ! object 'kh_p15_retina_volcano' not found
plotted <- dev.off()
kh_p15_retina_volcano[["plot"]]
## Error:
## ! object 'kh_p15_retina_volcano' not found
## why in the crap is it double-labelling!?
## My MA plotter isn't as smart as the volcano plotter, the genes are:
kh_p15_retina_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_retina"]], color_high = colors[["het_retina"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "13genotype_ma_volcano/kh_p15_retina_ma.pdf", width = 9, height = 9)
kh_p15_retina_ma[["plot"]]
## Error:
## ! object 'kh_p15_retina_ma' not found
plotted <- dev.off()
kh_p15_retina_ma[["plot"]]
## Error:
## ! object 'kh_p15_retina_ma' not found

24.5.3.4 ko/het p15 SCN

24.5.4 Repeat this with two sets of genes

table_name <- "kh_p15_scn"
table_input <- genotype_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
interesting_genes <- c("Fign", "Nrn1", "Dpysl2", "Actb", "Fgf9", "Otx2", "Sec23",
                       "Ncam1", "Map4", "Sec22b", "Nlgn3", "Marcks", "Cd47",
                       "Dpysl3", "Lin7c", "Cadm1", "Snx12", "Rhoa", "Inpp5f",
                       "Atg12", "Set", "Gsk3b", "Pdcd4", "Gabra2", "Tmco1", "Anapc16")
kh_p15_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  label_column = "mgi_symbol", size = 4, alpha = 1.0,
  outline = outline, color_low = colors[["ko_scn"]], color_high = colors[["het_scn"]])
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "13genotype_ma_volcano/kh_p15_scn_volcano.pdf", width = 9, height = 9)
kh_p15_scn_volcano[["plot"]]
## Error:
## ! object 'kh_p15_scn_volcano' not found
plotted <- dev.off()
kh_p15_scn_volcano[["plot"]]
## Error:
## ! object 'kh_p15_scn_volcano' not found
kh_p15_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_scn"]], color_high = colors[["het_scn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "13genotype_ma_volcano/kh_p15_scn_ma.pdf", width = 9, height = 9)
kh_p15_scn_ma[["plot"]]
## Error:
## ! object 'kh_p15_scn_ma' not found
plotted <- dev.off()
kh_p15_scn_ma[["plot"]]
## Error:
## ! object 'kh_p15_scn_ma' not found

Round 2 with a separate gene set.

table_name <- "kh_p15_scn"
table_input <- genotype_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
interesting_genes <- c(
  "Anapc16", "Gabra2", "Tmco1", "Sod2", "Fgf9", "Pdcd4", "Rhoa", "Gsk3b", "Foxp1",
  "Ncam1", "Marcks", "Fign", "Dpysl3", "Inpp5f", "Cadm1", "Map4", "Ugcg", "Elovl4",
  "Elavl1", "Cfl2", "Tnnt1", "Gnb1", "Impact", "Nrn1", "Nlgn3", "Actb", "Cd47",
  "Sec22b", "Slc17a7", "Vglut1", "Actb", "B4galt5", "Foxp1", "Otx2", "Lin7c",
  "Snx12", "Atg12", "Set")
kh_p15_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["ko_scn"]], color_high = colors[["het_scn"]],
  outline = outline, label_column = "mgi_symbol", label = interesting_genes, size = 4, alpha = 1.0)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "13genotype_ma_volcano/kh_p15_scn_volcano_v2.pdf", width = 9, height = 9)
kh_p15_scn_volcano[["plot"]]
## Error:
## ! object 'kh_p15_scn_volcano' not found
plotted <- dev.off()
kh_p15_scn_volcano[["plot"]]
## Error:
## ! object 'kh_p15_scn_volcano' not found
kh_p15_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_scn"]], color_high = colors[["het_scn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "13genotype_ma_volcano/kh_p15_scn_ma_v2.pdf", width = 9, height = 9)
kh_p15_scn_ma[["plot"]]
## Error:
## ! object 'kh_p15_scn_ma' not found
plotted <- dev.off()
kh_p15_scn_ma[["plot"]]
## Error:
## ! object 'kh_p15_scn_ma' not found

24.5.4.1 ko/het p15 dLGN

table_name <- "kh_p15_dlgn"
table_input <- genotype_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
kh_p15_dlgn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["ko_dlgn"]], color_high = colors[["het_dlgn"]],
  outline = outline, label_column = "mgi_symbol", label = 10, size = 4, alpha = 1.0)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "13genotype_ma_volcano/kh_p15_dlgn_volcano.pdf", width = 9, height = 9)
kh_p15_dlgn_volcano[["plot"]]
## Error:
## ! object 'kh_p15_dlgn_volcano' not found
plotted <- dev.off()
kh_p15_dlgn_volcano[["plot"]]
## Error:
## ! object 'kh_p15_dlgn_volcano' not found
## My MA plotter isn't as smart as the volcano plotter, the genes are:
kh_p15_dlgn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_dlgn"]], color_high = colors[["het_dlgn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = 10, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "13genotype_ma_volcano/kh_p15_dlgn_ma.pdf", width = 9, height = 9)
kh_p15_dlgn_ma[["plot"]]
## Error:
## ! object 'kh_p15_dlgn_ma' not found
plotted <- dev.off()
kh_p15_dlgn_ma[["plot"]]
## Error:
## ! object 'kh_p15_dlgn_ma' not found

A query from Rashmi:

“I was discussing with Dr. Speer about the dLGN data and we found mostly retinal genes in dLGN Het/KO or time point comparison. Please check if those are not retina samples.”

I checked the samples etc and everything looks ok to me; perhaps I can use the results to look at this question in another way:

I will therefore load the p08_het_dlgn/p08_ko_dlgn table and compare it to the p08_het_retina/p08_ko_retina table directly. I think that if these turn out to be identical, then the hypothesis suggested by this query is correct.

Note, in order to do this, I must use the full tables, not the post-inclusion tables because I cannot guarantee that they will have identical gene IDs.

retina_table <- genotype_tables_full[["data"]][["kh_p08_retina"]]
dlgn_table <- genotype_tables_full[["data"]][["kh_p08_dlgn"]]
retina_subset <- retina_table[, c("ensembl_gene_id", "deseq_logfc")]
colnames(retina_subset) <- c("ID", "retina_logfc")
dlgn_subset <- dlgn_table[, c("ensembl_gene_id", "deseq_logfc")]
colnames(dlgn_subset) <- c("ID", "dlgn_logfc")
merged <- merge(retina_subset, dlgn_subset, by = "ID")
rownames(merged) <- make.names(merged[["ID"]], unique = TRUE)
merged[["ID"]] <- NULL
plotted <- plot_linear_scatter(merged)
pp(file = "images/kh_p08_retina_vs_dlgn_deseq_logfc_values.png")
## Warning in pp(file = "images/kh_p08_retina_vs_dlgn_deseq_logfc_values.png"):
## The directory: images does not exist, will attempt to create it.
plotted[["scatter"]]
dev.off()
## png 
##   2
plotted[["scatter"]]

Rashmi asked if we could also do the p15 for this comparison:

retina_table <- genotype_tables_full[["data"]][["kh_p15_retina"]]
dlgn_table <- genotype_tables_full[["data"]][["kh_p15_dlgn"]]
retina_subset <- retina_table[, c("ensembl_gene_id", "deseq_logfc")]
colnames(retina_subset) <- c("ID", "retina_logfc")
dlgn_subset <- dlgn_table[, c("ensembl_gene_id", "deseq_logfc")]
colnames(dlgn_subset) <- c("ID", "dlgn_logfc")
merged <- merge(retina_subset, dlgn_subset, by = "ID")
rownames(merged) <- make.names(merged[["ID"]], unique = TRUE)
merged[["ID"]] <- NULL
plotted <- plot_linear_scatter(merged)
pp(file = "images/kh_p15_retina_vs_dlgn_deseq_logfc_values.png")
plotted[["scatter"]]
dev.off()
## png 
##   2
plotted[["scatter"]]

24.5.4.2 Repeat with the strict filter

genotype_strict_tables <- list()
genotype_strict_sig <- list()
genotype_strict_gp <- list()
genotype_strict_cp <- list()
genotype_strict_en <- list()
for (k in seq_along(genotype_keepers)) {
  name <- names(genotype_keepers)[k]
  message("Examining ", name)
  keeper <- genotype_keepers[name]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_strict_df <- genotype_inclusions_strict[[include_df_name]]
  includes_strict <- genotype_inclusions_strict[[include_name]]
  summary(rownames(genotype_sig_full[["deseq"]][["ups"]][[name]]) %in% includes_strict)
  include_filename <- glue("14genotype_strict_contrasts_excel/{name}_including_wt_{lfc_cutoff}_decreased_table-v{ver}.xlsx")
  include_sig_filename <- glue("14genotype_strict_contrasts_excel/{name}_including_wt_{lfc_cutoff}_decreased_sig-v{ver}.xlsx")
  genotype_strict_tables[[name]] <- combine_de_tables(
    genotype_de, extra_annot = include_strict_df,
    keepers = keeper, label_column = label_column,
    excel = include_filename, wanted_genes = includes_strict)
  print(genotype_strict_tables[[name]])
  genotype_strict_sig[[name]] <- extract_significant_genes(
    genotype_strict_tables[[name]], according_to = "deseq",
    excel = include_sig_filename)
  print(genotype_strict_sig[[name]])
  num_rows <- nrow(genotype_strict_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(genotype_strict_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  if (num_rows >= 10) {
    message("Performing gprofiler/clusterProfiler.")
    genotype_strict_gp[[name]] <- all_gprofiler(genotype_strict_sig[[name]], species = "mmusculus")
    genotype_strict_cp[[name]] <- all_cprofiler(
      genotype_strict_sig[[name]], genotype_strict_tables[[name]],
      orgdb = "org.Mm.eg.db", go_level = go_level,
      orgdb_from = orgdb_from, max_groupsize = max_groupsize, organism = "mouse")
    #if (!is.null(get0("m2_gsc"))) {
    #  genotype_strict_en[[name]] <- all_enricher(genotype_strict_sig[[name]], gsc = m2_gsc,
    #                                      orgdb = "org.Mm.eg.db", from = "ENSEMBL", to = "SYMBOL")
    #}
    gp_written <- write_all_gp(genotype_strict_gp[[name]], prefix = "15", suffix = "strict")
    cp_written <- write_all_cp(genotype_strict_cp[[name]], prefix = "15", suffix = "strict")
    #en_written <- write_all_en(genotype_strict_en[[name]])
  } else {
    warning("There are less than 10 genes up and down in the ", name, " comparison.")
    message("There are less than 10 genes up and down in the ", name, " comparison.")
  }
}
## Examining kh_p08_dlgn
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## A set of combined differential expression results.
##                         table deseq_sigup deseq_sigdown edger_sigup
## 1 p08_ko_dlgn_vs_p08_het_dlgn           1             0           1
##   edger_sigdown limma_sigup limma_sigdown
## 1             0           1             0
## Only kh_p08_dlgn_up has information, cannot create an UpSet.
## Plot describing unique/shared genes in a differential expression table.
## NULL
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##             deseq_up deseq_down
## kh_p08_dlgn        1          0

## There are 1 significant up and down genes.
## Warning: There are less than 10 genes up and down in the kh_p08_dlgn
## comparison.
## There are less than 10 genes up and down in the kh_p08_dlgn comparison.
## Examining kh_p15_dlgn
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## A set of combined differential expression results.
##                         table deseq_sigup deseq_sigdown edger_sigup
## 1 p15_ko_dlgn_vs_p15_het_dlgn           0             0           0
##   edger_sigdown limma_sigup limma_sigdown
## 1             0           0             0
## Only  has information, cannot create an UpSet.
## Plot describing unique/shared genes in a differential expression table.
## NULL
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##             deseq_up deseq_down
## kh_p15_dlgn        0          0
## There are 0 significant up and down genes.
## Warning: There are less than 10 genes up and down in the kh_p15_dlgn
## comparison.
## There are less than 10 genes up and down in the kh_p15_dlgn comparison.
## Examining kh_p08_retina
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## A set of combined differential expression results.
##                             table deseq_sigup deseq_sigdown edger_sigup
## 1 p08_ko_retina_vs_p08_het_retina           0             0           0
##   edger_sigdown limma_sigup limma_sigdown
## 1             0           0             0
## Only  has information, cannot create an UpSet.
## Plot describing unique/shared genes in a differential expression table.
## NULL
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##               deseq_up deseq_down
## kh_p08_retina        0          0
## There are 0 significant up and down genes.
## Warning: There are less than 10 genes up and down in the kh_p08_retina
## comparison.
## There are less than 10 genes up and down in the kh_p08_retina comparison.
## Examining kh_p15_retina
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## A set of combined differential expression results.
##                             table deseq_sigup deseq_sigdown edger_sigup
## 1 p15_ko_retina_vs_p15_het_retina           0             0           0
##   edger_sigdown limma_sigup limma_sigdown
## 1             0           0             2
## Only  has information, cannot create an UpSet.
## Plot describing unique/shared genes in a differential expression table.
## NULL
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##               deseq_up deseq_down
## kh_p15_retina        0          0
## There are 0 significant up and down genes.
## Warning: There are less than 10 genes up and down in the kh_p15_retina
## comparison.
## There are less than 10 genes up and down in the kh_p15_retina comparison.
## Examining kh_p08_scn
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## The result table is too small for meaningful comparisons.
## The first table has only: 25.
## A set of combined differential expression results.
##                       table deseq_sigup deseq_sigdown edger_sigup edger_sigdown
## 1 p08_ko_scn_vs_p08_het_scn           0             0           0             0
##   limma_sigup limma_sigdown
## 1           0             0
## Only  has information, cannot create an UpSet.
## Plot describing unique/shared genes in a differential expression table.
## NULL
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##            deseq_up deseq_down
## kh_p08_scn        0          0
## There are 0 significant up and down genes.
## Warning: There are less than 10 genes up and down in the kh_p08_scn comparison.
## There are less than 10 genes up and down in the kh_p08_scn comparison.
## Examining kh_p15_scn
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## The result table is too small for meaningful comparisons.
## The first table has only: 3.
## A set of combined differential expression results.
##                       table deseq_sigup deseq_sigdown edger_sigup edger_sigdown
## 1 p15_ko_scn_vs_p15_het_scn           0             0           0             0
##   limma_sigup limma_sigdown
## 1           0             0
## Only  has information, cannot create an UpSet.
## Plot describing unique/shared genes in a differential expression table.
## NULL
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##            deseq_up deseq_down
## kh_p15_scn        0          0
## There are 0 significant up and down genes.
## Warning: There are less than 10 genes up and down in the kh_p15_scn comparison.
## There are less than 10 genes up and down in the kh_p15_scn comparison.

25 Genotype Strict plots

for (k in seq_along(genotype_keepers)) {
  name <- names(genotype_keepers)[k]
  message("Examining ", name)
  keeper <- genotype_keepers[name]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_strict_df <- genotype_inclusions_strict[[include_df_name]]
  includes_strict <- genotype_inclusions_strict[[include_name]]
  summary(rownames(genotype_sig_full[["deseq"]][["ups"]][[name]]) %in% includes_strict)
  num_rows <- nrow(genotype_strict_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(genotype_strict_sig[[name]][["deseq"]][["downs"]][[name]])
  nrow(genotype_strict_sig[[name]][["deseq"]][["ups"]][[name]])
  nrow(genotype_strict_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  ## #1 is up and #2 is down, avoiding typeos here.
  num_objects <- length(genotype_strict_cp[[name]])
  if (num_objects == 0) {
    warning("Something went wrong in all_cprofiler.")
  } else {
    upp <- which(grepl(x = names(genotype_strict_cp[[name]]), pattern = "_up$"))
    downp <- which(grepl(x = names(genotype_strict_cp[[name]]), pattern = "_down$"))
    if (length(upp) > 0) {
      mf_sig <- genotype_strict_cp[[name]][[upp]][["go_data"]][["MF_enrich"]]
      cc_sig <- genotype_strict_cp[[name]][[upp]][["go_data"]][["CC_enrich"]]
      bp_sig <- genotype_strict_cp[[name]][[upp]][["go_data"]][["BP_enrich"]]
      mf_plots_up <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_up_filename <- glue("16clusterProfiler_plots/{name}_up_mf_sig_tree.pdf")
      pp(file = mf_tree_up_filename)
      try(print(mf_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_up_filename <- glue("16clusterProfiler_plots/{name}_up_mf_sig_bar.pdf")
      pp(file = mf_bar_up_filename)
      try(print(mf_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_up <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_up_filename <- glue("16clusterProfiler_plots/{name}_up_cc_sig_tree.pdf")
      pp(file = cc_tree_up_filename)
      try(print(cc_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_up_filename <- glue("16clusterProfiler_plots/{name}_up_cc_sig_bar.pdf")
      pp(file = cc_bar_up_filename)
      try(print(cc_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_up_filename <- glue("16clusterProfiler_plots/{name}_up_bp_sig_tree.pdf")
      pp(file = bp_tree_up_filename)
      try(print(bp_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_up_filename <- glue("16clusterProfiler_plots/{name}_up_bp_sig_bar.pdf")
      pp(file = bp_bar_up_filename)
      try(print(bp_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
    if (length(downp) > 0) {
      mf_sig <- genotype_strict_cp[[name]][[downp]][["go_data"]][["MF_enrich"]]
      cc_sig <- genotype_strict_cp[[name]][[downp]][["go_data"]][["CC_enrich"]]
      bp_sig <- genotype_strict_cp[[name]][[downp]][["go_data"]][["BP_enrich"]]
      mf_plots_down <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_down_filename <- glue("16clusterProfiler_plots/{name}_down_mf_sig_tree.pdf")
      pp(file = mf_tree_down_filename)
      try(print(mf_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_down_filename <- glue("16clusterProfiler_plots/{name}_down_mf_sig_bar.pdf")
      pp(file = mf_bar_down_filename)
      try(print(mf_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_down <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_down_filename <- glue("16clusterProfiler_plots/{name}_down_cc_sig_tree.pdf")
      pp(file = cc_tree_down_filename)
      try(print(cc_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_down_filename <- glue("16clusterProfiler_plots/{name}_down_cc_sig_bar.pdf")
      pp(file = cc_bar_down_filename)
      try(print(cc_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_down_filename <- glue("16clusterProfiler_plots/{name}_down_bp_sig_tree.pdf")
      pp(file = bp_tree_down_filename)
      try(print(bp_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, cateogories = go_categories)
      bp_bar_down_filename <- glue("16clusterProfiler_plots/{name}_down_bp_sig_bar.pdf")
      pp(file = bp_bar_down_filename)
      try(print(bp_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
  }
}
## Examining kh_p08_dlgn
## There are 1 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p15_dlgn
## There are 0 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p08_retina
## There are 0 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p15_retina
## There are 0 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p08_scn
## There are 0 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining kh_p15_scn
## There are 0 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.

Given this stricter filter, I think no genes pass in the genome comparisons.

25.0.1 Location contrasts with genes removed/kept

We will now repeat the above tasks seeking location differences instead of genotype; essentially I copy/pasted the above with s/genotype/location/g.

location_tables_full <- combine_de_tables(
  location_de, keepers = location_keepers, label_column = label_column,
  excel = glue("17full_location_contrasts/location_full_tables-v{ver}.xlsx"))
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
location_tables_full
## A set of combined differential expression results.
##                             table deseq_sigup deseq_sigdown edger_sigup
## 1  p08_het_dlgn_vs_p08_het_retina        2165          1562        2212
## 2  p15_het_dlgn_vs_p15_het_retina        2437          3369        2587
## 3    p08_ko_dlgn_vs_p08_ko_retina        2180          1868        2144
## 4    p15_ko_dlgn_vs_p15_ko_retina        2715          3942        2934
## 5   p08_het_scn_vs_p08_het_retina        2634          1707        2586
## 6   p15_het_scn_vs_p15_het_retina        2841          2395        2716
## 7     p08_ko_scn_vs_p08_ko_retina        2728          1705        2644
## 8     p15_ko_scn_vs_p15_ko_retina        2613          3005        2612
## 9     p08_het_dlgn_vs_p08_het_scn         648           788         751
## 10    p15_het_dlgn_vs_p15_het_scn        1708          2796        1984
## 11      p08_ko_dlgn_vs_p08_ko_scn        1002          1342        1115
## 12      p15_ko_dlgn_vs_p15_ko_scn        1829          2529        2158
##    edger_sigdown limma_sigup limma_sigdown
## 1           1632        1886          1660
## 2           3339        2748          2639
## 3           2077        2104          1963
## 4           3847        3236          2962
## 5           1882        2226          1889
## 6           2616        2730          2366
## 7           1922        2386          2160
## 8           3143        2812          2623
## 9            780         647           766
## 10          2623        1979          2034
## 11          1439        1169          1317
## 12          2396        1878          2000
## Plot describing unique/shared genes in a differential expression table.

location_sig_full <- extract_significant_genes(
  location_tables_full, according_to = "deseq",
  excel = glue("17full_location_contrasts/location_full_sig-v{ver}.xlsx"))
location_sig_full
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##            deseq_up deseq_down
## dr_p08_het     2165       1562
## dr_p15_het     2437       3369
## dr_p08_ko      2180       1868
## dr_p15_ko      2715       3942
## sr_p08_het     2634       1707
## sr_p15_het     2841       2395
## sr_p08_ko      2728       1705
## sr_p15_ko      2613       3005
## ds_p08_het      648        788
## ds_p15_het     1708       2796
## ds_p08_ko      1002       1342
## ds_p15_ko      1829       2529

location_full_upset <- upsetr_sig(location_sig_full)
location_full_intersects <- write_upset_groups(
  location_full_upset,
  excel = "excel/17full_location_contrasts/location_full_gene_groups.xlsx")
location_tables <- list()
location_sig <- list()
location_gp <- list()
location_cp <- list()
for (k in seq_along(location_keepers)) {
  name <- names(location_keepers)[k]
  message("Examining ", name)
  keeper <- location_keepers[name]
  includes <- location_inclusions[[name]]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- location_inclusions[[include_df_name]]
  includes <- location_inclusions[[include_name]]
  summary(rownames(location_sig_full[["deseq"]][["ups"]][[name]]) %in% includes)
  include_filename <- glue("18location_contrasts/{name}_including_wt_{lfc_cutoff}_decreased_table-v{ver}.xlsx")
  include_sig_filename <- glue("18location_contrasts/{name}_including_wt_{lfc_cutoff}_decreased_sig-v{ver}.xlsx")
  location_tables[[name]] <- combine_de_tables(
    location_de, extra_annot = include_df,
    keepers = keeper, label_column = label_column,
    excel = include_filename, wanted_genes = includes)
  print(location_tables[[name]])
  location_sig[[name]] <- extract_significant_genes(
    location_tables[[name]], according_to = "deseq",
    excel = include_sig_filename)
  print(location_sig[[name]])
  num_rows <- nrow(location_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(location_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  if (num_rows > 10) {
    location_gp[[name]] <- all_gprofiler(location_sig[[name]], species = "mmusculus")
    location_cp[[name]] <- all_cprofiler(
      location_sig[[name]], location_tables[[name]],
      orgdb = "org.Mm.eg.db", go_level = go_level, orgdb_from = orgdb_from,
      max_groupsize = max_groupsize, organism = "mouse")
    cp_written <- write_all_cp(location_cp[[name]], prefix = "19")
    gp_written <- write_all_gp(location_gp[[name]], prefix = "19")
  }
}
## Examining dr_p08_het
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## A set of combined differential expression results.
##                            table deseq_sigup deseq_sigdown edger_sigup
## 1 p08_het_dlgn_vs_p08_het_retina         259            81         259
##   edger_sigdown limma_sigup limma_sigdown
## 1            85         240            81
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## Plot describing unique/shared genes in a differential expression table.

## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##            deseq_up deseq_down
## dr_p08_het      259         81

## There are 340 significant up and down genes.
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error in `simple_cl[["kegg_universe"]]`:
## ! subscript out of bounds

Print out all the plots in a separate block.

for (k in seq_along(location_keepers)) {
  name <- names(location_keepers)[k]
  message("Examining ", name)
  keeper <- location_keepers[name]
  includes <- location_inclusions[[name]]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- location_inclusions[[include_df_name]]
  includes <- location_inclusions[[include_name]]
  summary(rownames(location_sig_full[["deseq"]][["ups"]][[name]]) %in% includes)
  num_rows <- nrow(location_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(location_sig[[name]][["deseq"]][["downs"]][[name]])
  nrow(location_sig[[name]][["deseq"]][["ups"]][[name]])
  nrow(location_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  num_objects <- length(location_cp[[name]])
  if (num_objects == 0) {
    warning("Something went wrong in all_cprofiler.")
  } else {
    upp <- which(grepl(x = names(location_cp[[name]]), pattern = "_up$"))
    downp <- which(grepl(x = names(location_cp[[name]]), pattern = "_down$"))
    if (length(upp) > 0) {
      mf_sig <- location_cp[[name]][[upp]][["go_data"]][["MF_enrich"]]
      cc_sig <- location_cp[[name]][[upp]][["go_data"]][["CC_enrich"]]
      bp_sig <- location_cp[[name]][[upp]][["go_data"]][["BP_enrich"]]
      mf_plots_up <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_up_filename <- glue("19clusterProfiler_plots/{name}_up_mf_sig_tree.pdf")
      pp(file = mf_tree_up_filename)
      try(print(mf_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_up_filename <- glue("19clusterProfiler_plots/{name}_up_mf_sig_bar.pdf")
      pp(file = mf_bar_up_filename)
      try(print(mf_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_up <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_up_filename <- glue("19clusterProfiler_plots/{name}_up_cc_sig_tree.pdf")
      pp(file = cc_tree_up_filename)
      try(print(cc_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_up_filename <- glue("19clusterProfiler_plots/{name}_up_cc_sig_bar.pdf")
      pp(file = cc_bar_up_filename)
      try(print(cc_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_up_filename <- glue("19clusterProfiler_plots/{name}_up_bp_sig_tree.pdf")
      pp(file = bp_tree_up_filename)
      try(print(bp_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_up_filename <- glue("19clusterProfiler_plots/{name}_up_bp_sig_bar.pdf")
      pp(file = bp_bar_up_filename)
      try(print(bp_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
    if (length(downp) > 0) {
      mf_sig <- location_cp[[name]][[downp]][["go_data"]][["MF_enrich"]]
      cc_sig <- location_cp[[name]][[downp]][["go_data"]][["CC_enrich"]]
      bp_sig <- location_cp[[name]][[downp]][["go_data"]][["BP_enrich"]]
      mf_plots_down <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_down_filename <- glue("19clusterProfiler_plots/{name}_down_mf_sig_tree.pdf")
      pp(file = mf_tree_down_filename)
      try(print(mf_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_down_filename <- glue("19clusterProfiler_plots/{name}_down_mf_sig_bar.pdf")
      pp(file = mf_bar_down_filename)
      try(print(mf_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_down <- plot_enrichresult(cc_sig, showCategory = 12)
      cc_tree_down_filename <- glue("19clusterProfiler_plots/{name}_down_cc_sig_tree.pdf")
      pp(file = cc_tree_down_filename)
      try(print(cc_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_down_filename <- glue("19clusterProfiler_plots/{name}_down_cc_sig_bar.pdf")
      pp(file = cc_bar_down_filename)
      try(print(cc_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_down_filename <- glue("19clusterProfiler_plots/{name}_down_bp_sig_tree.pdf")
      pp(file = bp_tree_down_filename)
      try(print(bp_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_down_filename <- glue("19clusterProfiler_plots/{name}_down_bp_sig_bar.pdf")
      pp(file = bp_bar_down_filename)
      try(print(bp_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
  }
}
## Examining dr_p08_het
## There are 340 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining dr_p15_het
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining dr_p08_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining dr_p15_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining sr_p08_het
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining sr_p15_het
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining sr_p08_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining sr_p15_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining ds_p08_het
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining ds_p15_het
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining ds_p08_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining ds_p15_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.

Colenso sent a specific query of interest, comparing SCN vs. Retinas at p08 in the heterozygotes including a set of genes of particular interest. Perhaps I can use some of these as markers to quality control my work in the future?

Here are the genes:

Opn4, Eomes, Trpc7, Oprm1, Nr4a3, Tbx20, Irx6, AW551984, Pcdh19, Adcyap1, Baiap3, Chl1, Grin3a, Igf1, Gria1, Grin2d, Grin3a, Chrna6, Chrna3, Htr5a, Htr2a, Htr7, Irx4, PlxnC1, Sema6d, Sema4f, Sema4a, Sema6b, Lrrc4b, Lrrc58, Lrrc3b, Wnt4, Wnt9b, Ctxn3, Tenm1, Gna14, Rgs4, Rgs6, Rgs5

table_input <- location_tables[["sr_p08_het"]]
table_name <- "sr_p08_het"
table <- table_input[["data"]][[table_name]]
interesting_genes <- c("Opn4", "Eomes", "Trpc7", "Oprm1", "Nr4a3", "Tbx20",
                       "Irx6", "AW551984", "Pcdh19", "Adcyap1r1", "Baiap3",
                       "Chl1", "Grin3a", "Igf1", "Gria1", "Grin2d", "Grin3a",
                       "Chrna6", "Chrna3", "Htr5a", "Htr2a", "Htr7", "Irx4",
                       "PlxnC1", "Sema6d", "Sema4f", "Sema4a", "Sema6b", "Lrrc4b",
                       "Lrrc58", "Lrrc3b", "Wnt4", "Wnt9b", "Ctxn3", "Tenm1", "Gna14",
                       "Rgs4", "Rgs6", "Rgs5", "Pou4f2", "Chrnb3", "Bcan")
sr_p08_het_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["het_retina"]], color_high = colors[["het_scn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "20location_ma_volcano/sr_p08_het_volcano.pdf", width = 9, height = 9)
## Warning in pp(file = "20location_ma_volcano/sr_p08_het_volcano.pdf", width = 9,
## : The directory: 20location_ma_volcano does not exist, will attempt to create
## it.
sr_p08_het_volcano[["plot"]]
## Error:
## ! object 'sr_p08_het_volcano' not found
plotted <- dev.off()
sr_p08_het_volcano[["plot"]]
## Error:
## ! object 'sr_p08_het_volcano' not found
sr_p08_het_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["het_retina"]], color_high = colors[["het_scn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "20location_ma_volcano/sr_p08_het_ma.pdf", width = 9, height = 9)
sr_p08_het_ma[["plot"]]
## Error:
## ! object 'sr_p08_het_ma' not found
plotted <- dev.off()
sr_p08_het_ma[["plot"]]
## Error:
## ! object 'sr_p08_het_ma' not found

25.0.2 Also the ko

table_input <- location_tables[["sr_p08_ko"]]
table_name <- "sr_p08_ko"
table <- table_input[["data"]][[table_name]]
sr_p08_ko_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["ko_retina"]], color_high = colors[["ko_scn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "20location_ma_volcano/sr_p08_ko_volcano.pdf", width = 9, height = 9)
sr_p08_ko_volcano[["plot"]]
## Error:
## ! object 'sr_p08_ko_volcano' not found
plotted <- dev.off()
sr_p08_ko_volcano[["plot"]]
## Error:
## ! object 'sr_p08_ko_volcano' not found
sr_p08_ko_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_retina"]], color_high = colors[["ko_scn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "20location_ma_volcano/sr_p08_ko_ma.pdf", width = 9, height = 9)
sr_p08_ko_ma[["plot"]]
## Error:
## ! object 'sr_p08_ko_ma' not found
plotted <- dev.off()
sr_p08_ko_ma[["plot"]]
## Error:
## ! object 'sr_p08_ko_ma' not found
table_input <- location_tables[["sr_p15_het"]]
table_name <- "sr_p15_het"
table <- table_input[["data"]][[table_name]]
interesting_genes <- c("Opn4", "Eomes", "Trpc7", "Oprm1", "Nr4a3", "Tbx20",
                       "Irx6", "AW551984", "Pcdh19", "Adcyap1r1", "Baiap3",
                       "Chl1", "Grin3a", "Igf1", "Gria1", "Grin2d", "Grin3a",
                       "Chrna6", "Chrna3", "Htr5a", "Htr2a", "Htr7", "Irx4",
                       "PlxnC1", "Sema6d", "Sema4f", "Sema4a", "Sema6b", "Lrrc4b",
                       "Lrrc58", "Lrrc3b", "Wnt4", "Wnt9b", "Ctxn3", "Tenm1", "Gna14",
                       "Rgs4", "Rgs6", "Rgs5", "Pou4f2", "Chrnb3", "Bcan")
sr_p15_het_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["het_retina"]], color_high = colors[["het_scn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "20location_ma_volcano/sr_p15_het_volcano.pdf", width = 9, height = 9)
sr_p15_het_volcano[["plot"]]
## Error:
## ! object 'sr_p15_het_volcano' not found
plotted <- dev.off()
sr_p15_het_volcano[["plot"]]
## Error:
## ! object 'sr_p15_het_volcano' not found
sr_p15_het_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["het_retina"]], color_high = colors[["het_scn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "20location_ma_volcano/sr_p15_het_ma.pdf", width = 9, height = 9)
sr_p15_het_ma[["plot"]]
## Error:
## ! object 'sr_p15_het_ma' not found
plotted <- dev.off()
sr_p15_het_ma[["plot"]]
## Error:
## ! object 'sr_p15_het_ma' not found

25.0.3 Also the ko

table_input <- location_tables[["sr_p15_ko"]]
table_name <- "sr_p15_ko"
table <- table_input[["data"]][[table_name]]
sr_p15_ko_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["ko_retina"]], color_high = colors[["ko_scn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4, min.segment.length = 0, point.padding = 0.2)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "20location_ma_volcano/sr_p15_ko_volcano.pdf", width = 12, height = 12)
sr_p15_ko_volcano[["plot"]]
## Error:
## ! object 'sr_p15_ko_volcano' not found
plotted <- dev.off()
sr_p15_ko_volcano[["plot"]]
## Error:
## ! object 'sr_p15_ko_volcano' not found
sr_p15_ko_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_retina"]], color_high = colors[["ko_scn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "20location_ma_volcano/sr_p15_ko_ma.pdf", width = 9, height = 9)
sr_p15_ko_ma[["plot"]]
## Error:
## ! object 'sr_p15_ko_ma' not found
plotted <- dev.off()
sr_p15_ko_ma[["plot"]]
## Error:
## ! object 'sr_p15_ko_ma' not found

25.0.3.1 Test a specific location query for duplicated IDs

Let us see if any Ensembl gene IDs and/or MGI IDs are shared in the worksheet location_sr_p08_ko_including_wt_0.1_decreased_sig up/down.

test_table_up <- location_sig[["sr_p08_ko"]][["deseq"]][["ups"]][[1]]
test_table_down <- location_sig[["sr_p08_ko"]][["deseq"]][["downs"]][[1]]

query <- list("up" = rownames(test_table_up),
              "down" = rownames(test_table_down))
query_upset <- UpSetR::fromList(query)
UpSetR::upset(query_upset)
## Error in `start_col:end_col`:
## ! argument of length 0
query <- list("up" = test_table_up[["mgi_symbol"]],
              "down" = test_table_down[["mgi_symbol"]])
query_upset <- UpSetR::fromList(query)
UpSetR::upset(query_upset)
## Error in `start_col:end_col`:
## ! argument of length 0
## ok, good.

25.0.3.2 Repeat with the strict filter

location_strict_tables <- list()
location_strict_sig <- list()
location_strict_gp <- list()
location_strict_cp <- list()
for (k in seq_along(location_keepers)) {
  name <- names(location_keepers)[k]
  message("Examining ", name)
  keeper <- location_keepers[name]
  includes <- location_inclusions_strict[[name]]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- location_inclusions_strict[[include_df_name]]
  includes <- location_inclusions_strict[[include_name]]
  found_includes <- rownames(location_sig_full[["deseq"]][["ups"]][[name]]) %in% includes
  summary(found_includes)
  if (sum(found_includes) == 0) {
    next
  }
  include_filename <- glue("21location_strict_contrasts/{name}_including_wt_{lfc_cutoff}_decreased_table-v{ver}.xlsx")
  include_sig_filename <- glue("21location_strict_contrasts/{name}_including_wt_{lfc_cutoff}_decreased_sig-v{ver}.xlsx")
  location_strict_tables[[name]] <- combine_de_tables(
    location_de, extra_annot = include_df,
    keepers = keeper, label_column = label_column,
    excel = include_filename, wanted_genes = includes)
  print(location_strict_tables[[name]])
  location_strict_sig[[name]] <- extract_significant_genes(
    location_strict_tables[[name]], according_to = "deseq",
    excel = include_sig_filename)
  print(location_strict_sig[[name]])
  num_rows <- nrow(location_strict_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(location_strict_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  if (num_rows > 10) {
    location_strict_gp[[name]] <- all_gprofiler(location_strict_sig[[name]], species = "mmusculus")
    location_strict_cp[[name]] <- all_cprofiler(
      location_strict_sig[[name]], location_strict_tables[[name]],
      orgdb = "org.Mm.eg.db", go_level = go_level, orgdb_from = orgdb_from,
      max_groupsize = max_groupsize, organism = "mouse")
    cp_written <- write_all_cp(location_strict_cp[[name]], prefix = "22", suffix = "strict")
    gp_written <- write_all_gp(location_strict_gp[[name]], prefix = "22", suffix = "strict")
  }
}
## Examining dr_p08_het
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## The result table is too small for meaningful comparisons.
## The first table has only: 22.
## A set of combined differential expression results.
##                            table deseq_sigup deseq_sigdown edger_sigup
## 1 p08_het_dlgn_vs_p08_het_retina           7             0           8
##   edger_sigdown limma_sigup limma_sigdown
## 1             0           5             0
## Only dr_p08_het_up has information, cannot create an UpSet.
## Plot describing unique/shared genes in a differential expression table.
## NULL
## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##            deseq_up deseq_down
## dr_p08_het        7          0

## There are 7 significant up and down genes.
## Examining dr_p15_het
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## A set of combined differential expression results.
##                            table deseq_sigup deseq_sigdown edger_sigup
## 1 p15_het_dlgn_vs_p15_het_retina          65             5          65
##   edger_sigdown limma_sigup limma_sigdown
## 1             5          57             4
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## Plot describing unique/shared genes in a differential expression table.

## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##            deseq_up deseq_down
## dr_p15_het       65          5

## There are 70 significant up and down genes.
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error in `simple_cl[["kegg_universe"]]`:
## ! subscript out of bounds

Print out all the plots in a separate block.

for (k in seq_along(location_keepers)) {
  name <- names(location_keepers)[k]
  message("Examining ", name)
  keeper <- location_keepers[name]
  includes <- location_inclusions_strict[[name]]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- location_inclusions_strict[[include_df_name]]
  includes <- location_inclusions_strict[[include_name]]
  summary(rownames(location_sig_full[["deseq"]][["ups"]][[name]]) %in% includes)
  num_rows <- nrow(location_strict_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(location_strict_sig[[name]][["deseq"]][["downs"]][[name]])
  nrow(location_strict_sig[[name]][["deseq"]][["ups"]][[name]])
  nrow(location_strict_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  num_objects <- length(location_strict_cp[[name]])
  if (num_objects == 0) {
    warning("Something went wrong in all_cprofiler.")
  } else {
    upp <- which(grepl(x = names(location_strict_cp[[name]]), pattern = "_up$"))
    downp <- which(grepl(x = names(location_strict_cp[[name]]), pattern = "_down$"))
    if (length(upp) > 0) {
      mf_sig <- location_strict_cp[[name]][[upp]][["go_data"]][["MF_enrich"]]
      cc_sig <- location_strict_cp[[name]][[upp]][["go_data"]][["CC_enrich"]]
      bp_sig <- location_strict_cp[[name]][[upp]][["go_data"]][["BP_enrich"]]
      mf_plots_up <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_up_filename <- glue("23clusterProfiler_plots/{name}_up_mf_sig_tree.pdf")
      pp(file = mf_tree_up_filename)
      try(print(mf_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_up_filename <- glue("23clusterProfiler_plots/{name}_up_mf_sig_bar.pdf")
      pp(file = mf_bar_up_filename)
      try(print(mf_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_up <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_up_filename <- glue("23clusterProfiler_plots/{name}_up_cc_sig_tree.pdf")
      pp(file = cc_tree_up_filename)
      try(print(cc_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_up_filename <- glue("23clusterProfiler_plots/{name}_up_cc_sig_bar.pdf")
      pp(file = cc_bar_up_filename)
      try(print(cc_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_up_filename <- glue("23clusterProfiler_plots/{name}_up_bp_sig_tree.pdf")
      pp(file = bp_tree_up_filename)
      try(print(bp_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_up_filename <- glue("23clusterProfiler_plots/{name}_up_bp_sig_bar.pdf")
      pp(file = bp_bar_up_filename)
      try(print(bp_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
    if (length(downp) > 0) {
      mf_sig <- location_strict_cp[[name]][[downp]][["go_data"]][["MF_enrich"]]
      cc_sig <- location_strict_cp[[name]][[downp]][["go_data"]][["CC_enrich"]]
      bp_sig <- location_strict_cp[[name]][[downp]][["go_data"]][["BP_enrich"]]
      mf_plots_down <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_down_filename <- glue("23clusterProfiler_plots/{name}_down_mf_sig_tree.pdf")
      pp(file = mf_tree_down_filename)
      try(print(mf_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_down_filename <- glue("23clusterProfiler_plots/{name}_down_mf_sig_bar.pdf")
      pp(file = mf_bar_down_filename)
      try(print(mf_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_down <- plot_enrichresult(cc_sig, showCategory = 12)
      cc_tree_down_filename <- glue("23clusterProfiler_plots/{name}_down_cc_sig_tree.pdf")
      pp(file = cc_tree_down_filename)
      try(print(cc_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_down_filename <- glue("23clusterProfiler_plots/{name}_down_cc_sig_bar.pdf")
      pp(file = cc_bar_down_filename)
      try(print(cc_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_down_filename <- glue("23clusterProfiler_plots/{name}_down_bp_sig_tree.pdf")
      pp(file = bp_tree_down_filename)
      try(print(bp_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_down_filename <- glue("23clusterProfiler_plots/{name}_down_bp_sig_bar.pdf")
      pp(file = bp_bar_down_filename)
      try(print(bp_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
  }
}
## Examining dr_p08_het
## There are 7 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining dr_p15_het
## There are 70 significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining dr_p08_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining dr_p15_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining sr_p08_het
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining sr_p15_het
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining sr_p08_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining sr_p15_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining ds_p08_het
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining ds_p15_het
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining ds_p08_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.
## Examining ds_p15_ko
## There are  significant up and down genes.
## Warning: Something went wrong in all_cprofiler.

Colenso sent a specific query of interest, comparing SCN vs. Retinas at p08 in the heterozygotes including a set of genes of particular interest. Perhaps I can use some of these as markers to quality control my work in the future?

Here are the genes:

Opn4, Eomes, Trpc7, Oprm1, Nr4a3, Tbx20, Irx6, AW551984, Pcdh19, Adcyap1, Baiap3, Chl1, Grin3a, Igf1, Gria1, Grin2d, Grin3a, Chrna6, Chrna3, Htr5a, Htr2a, Htr7, Irx4, PlxnC1, Sema6d, Sema4f, Sema4a, Sema6b, Lrrc4b, Lrrc58, Lrrc3b, Wnt4, Wnt9b, Ctxn3, Tenm1, Gna14, Rgs4, Rgs6, Rgs5

table_input <- location_strict_tables[["sr_p08_het"]]
table_name <- "sr_p08_het"
table <- table_input[["data"]][[table_name]]
interesting_genes <- c("Opn4", "Eomes", "Trpc7", "Oprm1", "Nr4a3", "Tbx20",
                       "Irx6", "AW551984", "Pcdh19", "Adcyap1r1", "Baiap3",
                       "Chl1", "Grin3a", "Igf1", "Gria1", "Grin2d", "Grin3a",
                       "Chrna6", "Chrna3", "Htr5a", "Htr2a", "Htr7", "Irx4",
                       "PlxnC1", "Sema6d", "Sema4f", "Sema4a", "Sema6b", "Lrrc4b",
                       "Lrrc58", "Lrrc3b", "Wnt4", "Wnt9b", "Ctxn3", "Tenm1", "Gna14",
                       "Rgs4", "Rgs6", "Rgs5", "Pou4f2", "Chrnb3", "Bcan")
sr_p08_het_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["het_retina"]], color_high = colors[["het_scn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "24location_ma_volcano_strict/sr_p08_het_volcano.pdf", width = 9, height = 9)
## Warning in pp(file = "24location_ma_volcano_strict/sr_p08_het_volcano.pdf", :
## The directory: 24location_ma_volcano_strict does not exist, will attempt to
## create it.
sr_p08_het_volcano[["plot"]]
## Error:
## ! object 'sr_p08_het_volcano' not found
plotted <- dev.off()
sr_p08_het_volcano[["plot"]]
## Error:
## ! object 'sr_p08_het_volcano' not found
sr_p08_het_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["het_retina"]], color_high = colors[["het_scn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "24location_ma_volcano_strict/sr_p08_het_ma.pdf", width = 9, height = 9)
sr_p08_het_ma[["plot"]]
## Error:
## ! object 'sr_p08_het_ma' not found
plotted <- dev.off()
sr_p08_het_ma[["plot"]]
## Error:
## ! object 'sr_p08_het_ma' not found

25.0.4 Also the ko

table_input <- location_strict_tables[["sr_p08_ko"]]
table_name <- "sr_p08_ko"
table <- table_input[["data"]][[table_name]]
sr_p08_ko_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["ko_retina"]], color_high = colors[["ko_scn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "24location_ma_volcano_strict/sr_p08_ko_volcano.pdf", width = 9, height = 9)
sr_p08_ko_volcano[["plot"]]
## Error:
## ! object 'sr_p08_ko_volcano' not found
plotted <- dev.off()
sr_p08_ko_volcano[["plot"]]
## Error:
## ! object 'sr_p08_ko_volcano' not found
sr_p08_ko_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_retina"]], color_high = colors[["ko_scn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "24location_ma_volcano_strict/sr_p08_ko_ma.pdf", width = 9, height = 9)
sr_p08_ko_ma[["plot"]]
## Error:
## ! object 'sr_p08_ko_ma' not found
plotted <- dev.off()
sr_p08_ko_ma[["plot"]]
## Error:
## ! object 'sr_p08_ko_ma' not found

25.0.5 Also the ko

table_input <- location_tables[["sr_p15_ko"]]
table_name <- "sr_p15_ko"
table <- table_input[["data"]][[table_name]]
sr_p15_ko_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = colors[["ko_retina"]], color_high = colors[["ko_scn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4, min.segment.length = 0, point.padding = 0.2)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "24location_ma_volcano_strict/sr_p15_ko_volcano.pdf", width = 12, height = 12)
sr_p15_ko_volcano[["plot"]]
## Error:
## ! object 'sr_p15_ko_volcano' not found
plotted <- dev.off()
sr_p15_ko_volcano[["plot"]]
## Error:
## ! object 'sr_p15_ko_volcano' not found
sr_p15_ko_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = colors[["ko_retina"]], color_high = colors[["ko_scn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "24location_ma_volcano_strict/sr_p15_ko_ma.pdf", width = 9, height = 9)
sr_p15_ko_ma[["plot"]]
## Error:
## ! object 'sr_p15_ko_ma' not found
plotted <- dev.off()
sr_p15_ko_ma[["plot"]]
## Error:
## ! object 'sr_p15_ko_ma' not found

25.0.6 And time

time_tables_full <- combine_de_tables(
  time_de, keepers = time_keepers,
  label_column = label_column,
  excel = glue("25full_contrasts_time/full_tables-v{ver}.xlsx"))
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
time_sig_full <- extract_significant_genes(
  time_tables_full, according_to = "deseq",
  excel = glue("25full_contrasts_time/full_sig-v{ver}.xlsx"))
time_tables <- list()
time_sig <- list()
time_gp <- list()
time_cp <- list()
for (k in seq_along(time_keepers)) {
  name <- names(time_keepers)[k]
  message("Examining ", name)
  keeper <- time_keepers[name]
  includes <- time_inclusions[[name]]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- time_inclusions[[include_df_name]]
  includes <- time_inclusions[[include_name]]
  summary(rownames(time_sig_full[["deseq"]][["ups"]][[name]]) %in% includes)
  include_filename <- glue("26time_contrasts/{name}_including_wt_{lfc_cutoff}_decreased_table-v{ver}.xlsx")
  include_sig_filename <- glue("26time_contrasts/{name}_including_wt_{lfc_cutoff}_decreased_sig-v{ver}.xlsx")
  time_tables[[name]] <- combine_de_tables(
    time_de, extra_annot = include_df,
    keepers = keeper, label_column = label_column,
    excel = include_filename, wanted_genes = includes)
  print(time_tables[[name]])
  time_sig[[name]] <- extract_significant_genes(
    time_tables[[name]], according_to = "deseq",
    excel = include_sig_filename)
  print(time_sig[[name]])
  num_rows <- nrow(time_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(time_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  if (num_rows > 10) {
    time_gp[[name]] <- all_gprofiler(time_sig[[name]], species = "mmusculus")
    gp_written <- write_all_gp(time_gp[[name]])
    time_cp[[name]] <- all_cprofiler(
      time_sig[[name]], time_tables[[name]], orgdb = "org.Mm.eg.db", organism = "mouse",
      orgdb_from = orgdb_from, go_level = go_level, max_groupsize = max_groupsize)
    cp_written <- write_all_cp(time_cp[[name]], prefix = "27")
    gp_written <- write_all_gp(time_gp[[name]], prefix = "27")
  }
}
## Examining t_het_dlgn
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## A set of combined differential expression results.
##                          table deseq_sigup deseq_sigdown edger_sigup
## 1 p15_het_dlgn_vs_p08_het_dlgn         397            14         431
##   edger_sigdown limma_sigup limma_sigdown
## 1            14         359            13
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## Plot describing unique/shared genes in a differential expression table.

## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##            deseq_up deseq_down
## t_het_dlgn      397         14

## There are 411 significant up and down genes.
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error in `simple_cl[["kegg_universe"]]`:
## ! subscript out of bounds

Send the plots separately.

for (k in seq_along(time_keepers)) {
  name <- names(time_keepers)[k]
  message("Examining ", name)
  keeper <- time_keepers[name]
  includes <- time_inclusions[[name]]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- time_inclusions[[include_df_name]]
  includes <- time_inclusions[[include_name]]
  num_rows <- nrow(time_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(time_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  num_objects <- length(time_cp[[name]])
  if (num_objects == 0) {
    warning("Something failed in all_cprofiler.")
  } else {
    upp <- which(grepl(x = names(time_cp[[name]]), pattern = "_up$"))
    downp <- which(grepl(x = names(time_cp[[name]]), pattern = "_down$"))
    if (length(upp) > 0) {
      mf_sig <- time_cp[[name]][[upp]][["go_data"]][["MF_enrich"]]
      cc_sig <- time_cp[[name]][[upp]][["go_data"]][["CC_enrich"]]
      bp_sig <- time_cp[[name]][[upp]][["go_data"]][["BP_enrich"]]
      mf_plots_up <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_up_filename <- glue("28clusterProfiler_plots/{name}_up_mf_sig_tree.pdf")
      pp(file = mf_tree_up_filename)
      try(print(mf_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_up_filename <- glue("28clusterProfiler_plots/{name}_up_mf_sig_bar.pdf")
      pp(file = mf_bar_up_filename)
      try(print(mf_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_up <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_up_filename <- glue("28clusterProfiler_plots/{name}_up_cc_sig_tree.pdf")
      pp(file = cc_tree_up_filename)
      try(print(cc_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_up_filename <- glue("28clusterProfiler_plots/{name}_up_cc_sig_bar.pdf")
      pp(file = cc_bar_up_filename)
      try(print(cc_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_up_filename <- glue("28clusterProfiler_plots/{name}_up_bp_sig_tree.pdf")
      pp(file = bp_tree_up_filename)
      try(print(bp_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_up_filename <- glue("28clusterProfiler_plots/{name}_up_bp_sig_bar.pdf")
      pp(file = bp_bar_up_filename)
      try(print(bp_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
    if (length(downp) > 0) {
      mf_sig <- time_cp[[name]][[downp]][["go_data"]][["MF_enrich"]]
      cc_sig <- time_cp[[name]][[downp]][["go_data"]][["CC_enrich"]]
      bp_sig <- time_cp[[name]][[downp]][["go_data"]][["BP_enrich"]]
      mf_plots_down <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_down_filename <- glue("28clusterProfiler_plots/{name}_down_mf_sig_tree.pdf")
      pp(file = mf_tree_down_filename)
      try(print(mf_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_down_filename <- glue("28clusterProfiler_plots/{name}_down_mf_sig_bar.pdf")
      pp(file = mf_bar_down_filename)
      try(print(mf_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_down <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_down_filename <- glue("28clusterProfiler_plots/{name}_down_cc_sig_tree.pdf")
      pp(file = cc_tree_down_filename)
      try(print(cc_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_down_filename <- glue("28clusterProfiler_plots/{name}_down_cc_sig_bar.pdf")
      pp(file = cc_bar_down_filename)
      try(print(cc_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_down_filename <- glue("28clusterProfiler_plots/{name}_down_bp_sig_tree.pdf")
      pp(file = bp_tree_down_filename)
      try(print(bp_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_down_filename <- glue("28clusterProfiler_plots/{name}_down_bp_sig_bar.pdf")
      pp(file = bp_bar_down_filename)
      try(print(bp_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
  }
}
## Examining t_het_dlgn
## There are 411 significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_ko_dlgn
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_het_retina
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_ko_retina
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_het_scn
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_ko_scn
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.

25.1 Volcano plots by time

25.1.1 t_het_dlgn

table_name <- "t_het_dlgn"
table_input <- time_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
t_het_dlgn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4, min.segment.length = 0, point.padding = 0.2)
pp(file = "29time_ma_volcano/t_het_dlgn_volcano.pdf", width = 12, height = 12)
## Warning in pp(file = "29time_ma_volcano/t_het_dlgn_volcano.pdf", width = 12, :
## The directory: 29time_ma_volcano does not exist, will attempt to create it.
t_het_dlgn_volcano[["plot"]]
plotted <- dev.off()
t_het_dlgn_volcano[["plot"]]

t_het_dlgn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
pp(file = "29time_ma_volcano/t_het_dlgn_ma.pdf", width = 9, height = 9)
t_het_dlgn_ma[["plot"]]
plotted <- dev.off()
t_het_dlgn_ma[["plot"]]

25.1.2 t_ko_dlgn

table_name <- "t_ko_dlgn"
table_input <- time_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
t_ko_dlgn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4, min.segment.length = 0, point.padding = 0.2)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "29time_ma_volcano/t_ko_dlgn_volcano.pdf", width = 12, height = 12)
t_ko_dlgn_volcano[["plot"]]
## Error:
## ! object 't_ko_dlgn_volcano' not found
plotted <- dev.off()
t_ko_dlgn_volcano[["plot"]]
## Error:
## ! object 't_ko_dlgn_volcano' not found
t_ko_dlgn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "29time_ma_volcano/t_ko_dlgn_ma.pdf", width = 9, height = 9)
t_ko_dlgn_ma[["plot"]]
## Error:
## ! object 't_ko_dlgn_ma' not found
plotted <- dev.off()
t_ko_dlgn_ma[["plot"]]
## Error:
## ! object 't_ko_dlgn_ma' not found

25.1.3 t_het_retina

table_name <- "t_het_retina"
table_input <- time_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
t_het_retina_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = time_colors[["p08_het_retina"]], color_high = time_colors[["p15_het_retina"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4, min.segment.length = 0, point.padding = 0.2)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "29time_ma_volcano/t_het_retina_volcano.pdf", width = 12, height = 12)
t_het_retina_volcano[["plot"]]
## Error:
## ! object 't_het_retina_volcano' not found
plotted <- dev.off()
t_het_retina_volcano[["plot"]]
## Error:
## ! object 't_het_retina_volcano' not found
t_het_retina_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = time_colors[["p08_het_retina"]], color_high = time_colors[["p15_het_retina"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "29time_ma_volcano/t_het_retina_ma.pdf", width = 9, height = 9)
t_het_retina_ma[["plot"]]
## Error:
## ! object 't_het_retina_ma' not found
plotted <- dev.off()
t_het_retina_ma[["plot"]]
## Error:
## ! object 't_het_retina_ma' not found

25.1.4 t_ko_retina

table_name <- "t_ko_retina"
table_input <- time_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
t_ko_retina_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4, min.segment.length = 0, point.padding = 0.2)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "29time_ma_volcano/t_ko_retina_volcano.pdf", width = 12, height = 12)
t_ko_retina_volcano[["plot"]]
## Error:
## ! object 't_ko_retina_volcano' not found
plotted <- dev.off()
t_ko_retina_volcano[["plot"]]
## Error:
## ! object 't_ko_retina_volcano' not found
t_ko_retina_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "29time_ma_volcano/t_ko_retina_ma.pdf", width = 9, height = 9)
t_ko_retina_ma[["plot"]]
## Error:
## ! object 't_ko_retina_ma' not found
plotted <- dev.off()
t_ko_retina_ma[["plot"]]
## Error:
## ! object 't_ko_retina_ma' not found

25.2 t_het_scn

table_name <- "t_het_scn"
table_input <- time_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
t_het_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4, min.segment.length = 0, point.padding = 0.2)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "29time_ma_volcano/t_het_scn_volcano.pdf", width = 12, height = 12)
t_het_scn_volcano[["plot"]]
## Error:
## ! object 't_het_scn_volcano' not found
plotted <- dev.off()
t_het_scn_volcano[["plot"]]
## Error:
## ! object 't_het_scn_volcano' not found
t_het_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "29time_ma_volcano/t_het_scn_ma.pdf", width = 9, height = 9)
t_het_scn_ma[["plot"]]
## Error:
## ! object 't_het_scn_ma' not found
plotted <- dev.off()
t_het_scn_ma[["plot"]]
## Error:
## ! object 't_het_scn_ma' not found

25.3 t_ko_scn

table_name <- "t_ko_scn"
table_input <- time_tables[[table_name]]
table <- table_input[["data"]][[table_name]]
t_ko_scn_volcano <- plot_volcano_condition_de(
  table, table_name, fc_col = "deseq_logfc", p_col = "deseq_adjp",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  label_column = "mgi_symbol", label = interesting_genes, alpha = 1.0,
  size = 4, min.segment.length = 0, point.padding = 0.2)
## Error in `plot_volcano_condition_de()`:
## ! Column: deseq_logfc is not in the table.
pp(file = "29time_ma_volcano/t_ko_scn_volcano.pdf", width = 12, height = 12)
t_ko_scn_volcano[["plot"]]
## Error:
## ! object 't_ko_scn_volcano' not found
plotted <- dev.off()
t_ko_scn_volcano[["plot"]]
## Error:
## ! object 't_ko_scn_volcano' not found
t_ko_scn_ma <- plot_ma_condition_de(
  table, table_name, expr_col = "deseq_basemean", fc_col = "deseq_logfc",
  color_low = time_colors[["p08_het_dlgn"]], color_high = time_colors[["p15_het_dlgn"]],
  p_col = "deseq_adjp", label_column = "mgi_symbol", label = interesting_genes, outline = outline)
## The column: mgi_symbol is not in the data, using rownames.
## Warning in max(newdf[["avg"]]): no non-missing arguments to max; returning -Inf
## Warning in plot_ma_condition_de(table, table_name, expr_col = "deseq_basemean",
## : NAs introduced by coercion
## Error in `[[<-.data.frame`:
## ! replacement has 1 row, data has 0
pp(file = "29time_ma_volcano/t_ko_scn_ma.pdf", width = 9, height = 9)
t_ko_scn_ma[["plot"]]
## Error:
## ! object 't_ko_scn_ma' not found
plotted <- dev.off()
t_ko_scn_ma[["plot"]]
## Error:
## ! object 't_ko_scn_ma' not found

25.3.0.1 Repeat with the strict filter

time_strict_tables <- list()
time_strict_sig <- list()
time_strict_gp <- list()
time_strict_cp <- list()
time_strict_en <- list()
for (k in seq_along(time_keepers)) {
  name <- names(time_keepers)[k]
  message("Examining ", name)
  keeper <- time_keepers[name]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- time_inclusions_strict[[include_df_name]]
  includes <- time_inclusions_strict[[include_name]]
  found_includes <- rownames(time_sig_full[["deseq"]][["ups"]][[name]]) %in% includes
  summary(found_includes)
  if (sum(found_includes) == 0) {
    next
  }
  include_filename <- glue("30time_strict_contrasts_excel/{name}_including_wt_{lfc_cutoff}_decreased_table-v{ver}.xlsx")
  include_sig_filename <- glue("30time_strict_contrasts_excel/{name}_including_wt_{lfc_cutoff}_decreased_sig-v{ver}.xlsx")
  time_strict_tables[[name]] <- combine_de_tables(
    time_de, extra_annot = include_df,
    keepers = keeper, label_column = label_column,
    excel = include_filename, wanted_genes = includes)
  print(time_strict_tables[[name]])
  time_strict_sig[[name]] <- extract_significant_genes(
    time_strict_tables[[name]], according_to = "deseq",
    excel = include_sig_filename)
  print(time_strict_sig[[name]])
  num_rows <- nrow(time_strict_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(time_strict_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  if (num_rows >= 10) {
    message("Performing gprofiler/clusterProfiler.")
    time_strict_gp[[name]] <- all_gprofiler(time_strict_sig[[name]], species = "mmusculus")
    time_strict_cp[[name]] <- all_cprofiler(
      time_strict_sig[[name]], time_strict_tables[[name]],
      orgdb = "org.Mm.eg.db", go_level = go_level, orgdb_from = orgdb_from,
      max_groupsize = max_groupsize, organism = "mouse")
    #if (!is.null(get0("m2_gsc"))) {
    #  time_strict_en[[name]] <- all_enricher(time_strict_sig[[name]], gsc = m2_gsc,
    #                                      orgdb = "org.Mm.eg.db", from = "ENSEMBL", to = "SYMBOL")
    #}
    gp_written <- write_all_gp(time_strict_gp[[name]], prefix = "31", suffix = "strict")
    cp_written <- write_all_cp(time_strict_cp[[name]], prefix = "31", suffix = "strict")
    #en_written <- write_all_en(time_strict_en[[name]])
  } else {
    warning("There are less than 10 genes up and down in the ", name, " comparison.")
    message("There are less than 10 genes up and down in the ", name, " comparison.")
  }
}
## Examining t_het_dlgn
## Looking for subscript invalid names, start of extract_keepers.
## Looking for subscript invalid names, end of extract_keepers.
## A set of combined differential expression results.
##                          table deseq_sigup deseq_sigdown edger_sigup
## 1 p15_het_dlgn_vs_p08_het_dlgn          19             2          20
##   edger_sigdown limma_sigup limma_sigdown
## 1             2          12             3
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## Plot describing unique/shared genes in a differential expression table.

## A set of genes deemed significant according to deseq.
## The parameters defining significant were:
## LFC cutoff: 1 adj P cutoff: 0.05
##            deseq_up deseq_down
## t_het_dlgn       19          2

## There are 21 significant up and down genes.
## Performing gprofiler/clusterProfiler.
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error : unable to find an inherited method for function 'metadata' for signature 'x = "NULL"'
## Error in `simple_cl[["kegg_universe"]]`:
## ! subscript out of bounds

Send the plots separately.

for (k in seq_along(time_keepers)) {
  name <- names(time_keepers)[k]
  message("Examining ", name)
  keeper <- time_keepers[name]
  includes <- time_inclusions[[name]]
  include_name <- paste0("inc_", name)
  include_df_name <- paste0("df_", name)
  include_df <- time_inclusions[[include_df_name]]
  includes <- time_inclusions[[include_name]]
  num_rows <- nrow(time_sig[[name]][["deseq"]][["ups"]][[name]]) +
    nrow(time_sig[[name]][["deseq"]][["downs"]][[name]])
  message("There are ", num_rows, " significant up and down genes.")
  num_objects <- length(time_cp[[name]])
  if (num_objects == 0) {
    warning("Something failed in all_cprofiler.")
  } else {
    upp <- which(grepl(x = names(time_cp[[name]]), pattern = "_up$"))
    downp <- which(grepl(x = names(time_cp[[name]]), pattern = "_down$"))
    if (length(upp) > 0) {
      mf_sig <- time_cp[[name]][[upp]][["go_data"]][["MF_enrich"]]
      cc_sig <- time_cp[[name]][[upp]][["go_data"]][["CC_enrich"]]
      bp_sig <- time_cp[[name]][[upp]][["go_data"]][["BP_enrich"]]
      mf_plots_up <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_up_filename <- glue("32cp_trees/{name}_up_mf_sig_tree.pdf")
      pp(file = mf_tree_up_filename)
      try(print(mf_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_up_filename <- glue("32cp_bar/{name}_up_mf_sig_bar.pdf")
      pp(file = mf_bar_up_filename)
      try(print(mf_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_up <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_up_filename <- glue("32cp_trees/{name}_up_cc_sig_tree.pdf")
      pp(file = cc_tree_up_filename)
      try(print(cc_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_up_filename <- glue("32cp_bar/{name}_up_cc_sig_bar.pdf")
      pp(file = cc_bar_up_filename)
      try(print(cc_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_up_filename <- glue("32cp_trees/{name}_up_bp_sig_tree.pdf")
      pp(file = bp_tree_up_filename)
      try(print(bp_plots_up[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_up <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_up_filename <- glue("32cp_bar/{name}_up_bp_sig_bar.pdf")
      pp(file = bp_bar_up_filename)
      try(print(bp_plots_up[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
    if (length(downp) > 0) {
      mf_sig <- time_cp[[name]][[downp]][["go_data"]][["MF_enrich"]]
      cc_sig <- time_cp[[name]][[downp]][["go_data"]][["CC_enrich"]]
      bp_sig <- time_cp[[name]][[downp]][["go_data"]][["BP_enrich"]]
      mf_plots_down <- plot_enrichresult(mf_sig, showCategory = go_categories)
      mf_tree_down_filename <- glue("32cp_trees/{name}_down_mf_sig_tree.pdf")
      pp(file = mf_tree_down_filename)
      try(print(mf_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      mf_bar_down_filename <- glue("32cp_bar/{name}_down_mf_sig_bar.pdf")
      pp(file = mf_bar_down_filename)
      try(print(mf_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      cc_plots_down <- plot_enrichresult(cc_sig, showCategory = go_categories)
      cc_tree_down_filename <- glue("32cp_trees/{name}_down_cc_sig_tree.pdf")
      pp(file = cc_tree_down_filename)
      try(print(cc_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      cc_bar_down_filename <- glue("32cp_bar/{name}_down_cc_sig_bar.pdf")
      pp(file = cc_bar_down_filename)
      try(print(cc_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_tree_down_filename <- glue("32cp_trees/{name}_down_bp_sig_tree.pdf")
      pp(file = bp_tree_down_filename)
      try(print(bp_plots_down[["tree"]]), silent = TRUE)
      plotted <- dev.off()
      bp_plots_down <- plot_enrichresult(bp_sig, showCategory = go_categories)
      bp_bar_down_filename <- glue("32cp_bar/{name}_down_bp_sig_bar.pdf")
      pp(file = bp_bar_down_filename)
      try(print(bp_plots_down[["bar"]]), silent = TRUE)
      plotted <- dev.off()
    }
  }
}
## Examining t_het_dlgn
## There are 411 significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_ko_dlgn
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_het_retina
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_ko_retina
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_het_scn
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.
## Examining t_ko_scn
## There are  significant up and down genes.
## Warning: Something failed in all_cprofiler.

26 Translatome queries

In conversation with Colenso, he spoke about a series of contrasts which would be interesting to attempt in order to query the changes across both locations and genotypes and/or both locations and time, thus:

(p08_het_scn / p08_het_retina) / (p08_ko_scn / p08_ko_retina)

as an example. We can definitely do these, but they do not work for all methods employed (I think they work best with limma and edgeR).

Lets find out!

26.1 Two scn/retina comparisons

  • (p08_het_scn / p08_het_retina) / (p08_ko_scn / p08_ko_retina)
  • (p15_het_scn / p15_het_retina) / (p15_ko_scn / p15_ko_retina)
scn_extra <- glue("\\
  p08het = (conditionp08_het_scn - conditionp08_het_retina), \\
  p08ko = (conditionp08_ko_scn - conditionp08_ko_retina), \\
  p08het_vs_p08ko = (conditionp08_het_scn - conditionp08_het_retina) - (conditionp08_ko_scn - conditionp08_ko_retina), \\
  p15het = (conditionp15_het_scn - conditionp15_het_retina), \\
  p15ko = (conditionp15_ko_scn - conditionp15_ko_retina), \\
  p15het_vs_p15ko = (conditionp15_het_scn - conditionp15_het_retina) - (conditionp15_ko_scn - conditionp15_ko_retina)")
scn_translatome_de_keepers <- list(
  "p08het" = c("p08_het_scn", "p08_het_retina"),
  "p08ko" = c("p08_ko_scn", "p08_ko_retina"),
  "p15het" = c("p15_het_scn", "p15_het_retina"),
  "p15ko" = c("p15_ko_scn", "p15_ko_retina"))
scn_translatome_keepers <- list(
  "p08het" = c("p08_het_scn", "p08_het_retina"),
  "p08ko" = c("p08_ko_scn", "p08_ko_retina"),
  "p08_scn_translatome" = c("p08het", "p08ko"),
  "p15het" = c("p15_het_scn", "p15_het_retina"),
  "p15ko" = c("p15_ko_scn", "p15_ko_retina"),
  "p15_scn_translatome" = c("p15het", "p15ko"))
filt <- normalize(v3_pairwise_input, filter = TRUE)
## Removing 10162 low-count genes (15263 remaining).
limma_test <- limma_pairwise(filt,
                             keepers = scn_translatome_de_keepers,
                             model_fstring = "~ 0 + condition",
                             model_svs = FALSE, extra_contrastrs = scn_extra)
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
edger_test <- edger_pairwise(filt,
                             keepers = scn_translatome_de_keepers,
                             model_fstring = "~ 0 + condition",
                             model_svs = FALSE, extra_contrasts = scn_extra)
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
scn_translatome_de <- all_pairwise(v3_pairwise_input, filter = TRUE,
                                   keepers = scn_translatome_de_keepers,
                                   model_svs = FALSE,
                                   model_fstring = "~ 0 + condition",
                                   do_basic = FALSE, do_dream = FALSE,
                                   do_noiseq = FALSE, do_ebseq = FALSE,
                                   extra_contrasts = scn_extra)
## Warning in all_pairwise(v3_pairwise_input, filter = TRUE, keepers =
## scn_translatome_de_keepers, : This will likely fail because of how the keepers
## and extra contrasts are evaluated.
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 10162 low-count genes (15263 remaining).
## converting counts to integer mode
## gene-wise dispersion estimates
## mean-dispersion relationship
## final dispersion estimates
## The contrast p08het is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p08ko is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p08het is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p15het is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p15ko is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p15het is not in the results.
## If this is not an extra contrast, then this is an error.
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
scn_combined_test <- combine_de_tables(
  scn_translatome_de, keepers = scn_translatome_keepers,
  excel = glue("33translatome_xlsx/test_scn_translatome_unfiltered_nosva-v{ver}.xlsx"))
## Looking for subscript invalid names, start of extract_keepers.
## coefficient edger did not find conditionp08ko or conditionp08het.
## coefficient limma did not find p08ko or p08het.
## coefficient edger did not find conditionp15ko or conditionp15het.
## coefficient limma did not find p15ko or p15het.
## Looking for subscript invalid names, end of extract_keepers.
scn_translatome_de_sva <- all_pairwise(v3_pairwise_input, filter = TRUE,
                                       keepers = scn_translatome_de_keepers,
                                       model_svs = "svaseq",
                                       model_fstring = "~ 0 + condition",
                                       do_basic = FALSE, do_dream = FALSE,
                                       do_noiseq = FALSE, do_ebseq = FALSE,
                                       extra_contrasts = scn_extra)
## Warning in all_pairwise(v3_pairwise_input, filter = TRUE, keepers =
## scn_translatome_de_keepers, : This will likely fail because of how the keepers
## and extra contrasts are evaluated.
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 10162 low-count genes (15263 remaining).
## Error in `adjuster_svs()`:
## ! unused arguments (do_basic = FALSE, do_dream = FALSE, do_noiseq = FALSE, do_ebseq = FALSE)
scn_combined_test_sva <- combine_de_tables(
  scn_translatome_de_sva, keepers = scn_translatome_keepers,
  excel = glue("33translatome_xlsx/test_scn_translatome_unfiltered_sva-v{ver}.xlsx"))
## Error:
## ! object 'scn_translatome_de_sva' not found

26.1.1 Subtracting DESeq2 results: p08 scn het vs ko

p08_scn_combined_deseq <- subtract_deseq_results(
  first_table = scn_combined_test[["data"]][["p08het"]],
  second_table = scn_combined_test[["data"]][["p08ko"]],
  first_lfc = "deseq_logfc", second_lfc = "deseq_logfc",
  first_p = "deseq_adjp", second_p = "deseq_adjp",
  first_name = "het", second_name = "ko",
  excel = glue("33translatome_xlsx/translatome_p08_scn_combined_deseq-v{ver}.xlsx"))
## Error in `subtract_deseq_results()`:
## ! could not find function "subtract_deseq_results"

26.1.2 Subtracting DESeq2 results: p15 scn het vs ko

p15_scn_combined_deseq <- subtract_deseq_results(
  first_table = scn_combined_test[["data"]][["p15het"]],
  second_table = scn_combined_test[["data"]][["p15ko"]],
  first_lfc = "deseq_logfc", second_lfc = "deseq_logfc",
  first_p = "deseq_adjp", second_p = "deseq_adjp",
  first_name = "het", second_name = "ko",
  excel = glue("34translatome_deseqsub_xlsx/translatome_p15_scn_combined_deseq-v{ver}.xlsx"))
## Error in `subtract_deseq_results()`:
## ! could not find function "subtract_deseq_results"

26.2 One dlgn/retina comparison

  • (p08_het_dlgn / p08_het_retina) / (p08_ko_dlgn / p08_ko_retina)
p08_dlgn_extra <- "p08het_vs_p08ko = (conditionp08_het_dlgn - conditionp08_het_retina) - (conditionp08_ko_dlgn - conditionp08_ko_retina)"
p08_dlgn_translatome_de_keepers <- list(
  "p08het" = c("p08_het_dlgn", "p08_het_retina"),
  "p08ko" = c("p08_ko_dlgn", "p08_ko_retina"))
p08_dlgn_translatome_keepers <- list(
  "p08_het_dlgn_vs_retina" = c("p08_het_dlgn", "p08_het_retina"),
  "p08_ko_dlgn_vs_retina" = c("p08_ko_dlgn", "p08_ko_retina"),
  "p08_dlgn_translatome" = c("p08het", "p08ko"))
p08_dlgn_translatome_de <- all_pairwise(v3_pairwise_input, filter = TRUE,
                                        keepers = p08_dlgn_translatome_de_keepers,
                                        model_svs = FALSE,
                                        model_fstring = "~ 0 + condition",
                                        do_basic = FALSE, do_dream = FALSE,
                                        do_noiseq = FALSE, do_ebseq = FALSE,
                                        extra_contrasts = p08_dlgn_extra)
## Warning in all_pairwise(v3_pairwise_input, filter = TRUE, keepers =
## p08_dlgn_translatome_de_keepers, : This will likely fail because of how the
## keepers and extra contrasts are evaluated.
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 10162 low-count genes (15263 remaining).
## converting counts to integer mode
## gene-wise dispersion estimates
## mean-dispersion relationship
## final dispersion estimates
## The contrast p08het is not in the results.
## If this is not an extra contrast, then this is an error.
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
p08_dlgn_combined_test <- combine_de_tables(
  p08_dlgn_translatome_de, keepers = p08_dlgn_translatome_keepers,
  label_column = label_column,
  excel = glue("33translatome_xlsx/test_p08_dlgn_translatome_unfiltered_nosva-v{ver}.xlsx"))
## Looking for subscript invalid names, start of extract_keepers.
## coefficient edger did not find conditionp08ko or conditionp08het.
## coefficient limma did not find p08ko or p08het.
## Looking for subscript invalid names, end of extract_keepers.
p08_dlgn_translatome_de_sva <- all_pairwise(v3_pairwise_input, filter = TRUE,
                                            keepers = p08_dlgn_translatome_de_keepers,
                                            model_svs = "svaseq",
                                            model_fstring = "~ 0 + condition",
                                            do_basic = FALSE, do_dream = FALSE,
                                            do_noiseq = FALSE, do_ebseq = FALSE,
                                            extra_contrasts = p08_dlgn_extra)
## Warning in all_pairwise(v3_pairwise_input, filter = TRUE, keepers =
## p08_dlgn_translatome_de_keepers, : This will likely fail because of how the
## keepers and extra contrasts are evaluated.
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 10162 low-count genes (15263 remaining).
## Error in `adjuster_svs()`:
## ! unused arguments (do_basic = FALSE, do_dream = FALSE, do_noiseq = FALSE, do_ebseq = FALSE)
p08_dlgn_combined_test_sva <- combine_de_tables(
  p08_dlgn_translatome_de_sva, keepers = p08_dlgn_translatome_keepers,
  label_column = label_column,
  excel = glue("33translatome_xlsx/test_p08_dlgn_translatome_unfiltered_sva-v{ver}.xlsx"))
## Error:
## ! object 'p08_dlgn_translatome_de_sva' not found

26.2.1 Subtracting the DESeq2 results

26.3 Two scn/retina comparisons (p15/p08 across het/ko)

  • (p15_het_scn / p15_het_retina) / (p08_het_scn / p08_het_retina)
  • (p15_ko_scn / p15_ko_retina) / (p08_ko_scn / p08_ko_retina)
time_scn_extra <- glue("\\
  p15het = (conditionp15_het_scn - conditionp15_het_retina), \\
  p08het = (conditionp08_het_scn - conditionp08_het_retina), \\
  p15het_vs_p08het = (conditionp15_het_scn - conditionp15_het_retina) - (conditionp08_het_scn - conditionp08_het_retina),
  p15ko = (conditionp15_ko_scn - conditionp15_ko_retina), \\
  p08ko = (conditionp08_ko_scn - conditionp08_ko_retina), \\
  p15ko_vs_p08ko = (conditionp15_ko_scn - conditionp15_ko_retina) - (conditionp08_ko_scn - conditionp08_ko_retina)")
time_scn_translatome_de_keepers <- list(
  "p15het" = c("p15_het_scn", "p15_het_retina"),
  "p08het" = c("p08_het_scn", "p08_het_retina"),
  "p15ko" = c("p15_ko_scn", "p15_ko_retina"),
  "p08ko" = c("p08_ko_scn", "p08_ko_retina"))
time_scn_translatome_keepers <- list(
  "p15het" = c("p15_het_scn", "p15_het_retina"),
  "p08het" = c("p08_het_scn", "p08_het_retina"),
  "p15ko" = c("p15_ko_scn", "p15_ko_retina"),
  "p08ko" = c("p08_ko_scn", "p08_ko_retina"),
  "p15_het_sc_vs_retina" = c("p15_het_scn", "p15_het_retina"),
  "p08_het_sc_vs_retina" = c("p08_het_scn", "p08_het_retina"),
  "scn_het_translatome" = c("p15het", "p08het"),
  "scn_ko_translatome" = c("p15ko", "p08ko"))
time_scn_translatome_de <- all_pairwise(v3_pairwise_input, filter = TRUE,
                                        keepers = time_scn_translatome_de_keepers,
                                        model_svs = FALSE,
                                        model_fstring = "~ 0 + condition",
                                        do_basic = FALSE, do_dream = FALSE,
                                        do_noiseq = FALSE, do_ebseq = FALSE,
                                        extra_contrasts = time_scn_extra)
## Warning in all_pairwise(v3_pairwise_input, filter = TRUE, keepers =
## time_scn_translatome_de_keepers, : This will likely fail because of how the
## keepers and extra contrasts are evaluated.
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 10162 low-count genes (15263 remaining).
## converting counts to integer mode
## gene-wise dispersion estimates
## mean-dispersion relationship
## final dispersion estimates
## The contrast p15het is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p08het is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p15het is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p15ko is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p08ko is not in the results.
## If this is not an extra contrast, then this is an error.
## The contrast p15ko is not in the results.
## If this is not an extra contrast, then this is an error.
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## conditions
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
time_scn_translatome_test <- combine_de_tables(
  time_scn_translatome_de,
  keepers = time_scn_translatome_keepers,
  label_column = label_column,
  excel = glue("33translatome_xlsx/test_time_scn_translatome_unfiltered_nosva-v{ver}.xlsx"))
## Looking for subscript invalid names, start of extract_keepers.
## coefficient edger did not find conditionp08het or conditionp15het.
## coefficient limma did not find p08het or p15het.
## coefficient edger did not find conditionp08ko or conditionp15ko.
## coefficient limma did not find p08ko or p15ko.
## Looking for subscript invalid names, end of extract_keepers.
time_scn_translatome_de_sva <- all_pairwise(v3_pairwise_input, filter = TRUE,
                                            keepers = time_scn_translatome_de_keepers,
                                            model_svs = "svaseq",
                                            model_fstring = "~ 0 + condition",
                                            do_basic = FALSE, do_dream = FALSE,
                                            do_noiseq = FALSE, do_ebseq = FALSE,
                                            extra_contrasts = time_scn_extra)
## Warning in all_pairwise(v3_pairwise_input, filter = TRUE, keepers =
## time_scn_translatome_de_keepers, : This will likely fail because of how the
## keepers and extra contrasts are evaluated.
##   p08_het_dlgn p08_het_retina    p08_het_scn    p08_ko_dlgn  p08_ko_retina 
##              3              3              3              3              3 
##     p08_ko_scn    p08_wt_dlgn  p08_wt_retina     p08_wt_scn   p15_het_dlgn 
##              3              5              5              3              4 
## p15_het_retina    p15_het_scn    p15_ko_dlgn  p15_ko_retina     p15_ko_scn 
##              4              3              3              3              3 
##    p15_wt_dlgn  p15_wt_retina     p15_wt_scn 
##              5              5              2
## Removing 10162 low-count genes (15263 remaining).
## Error in `adjuster_svs()`:
## ! unused arguments (do_basic = FALSE, do_dream = FALSE, do_noiseq = FALSE, do_ebseq = FALSE)
time_scn_translatome_test_sva <- combine_de_tables(
  time_scn_translatome_de_sva,
  keepers = time_scn_translatome_keepers,
  label_column = label_column,
  excel = glue("33translatome_xlsx/test_time_scn_translatome_unfiltered_sva-v{ver}.xlsx"))
## Error:
## ! object 'time_scn_translatome_de_sva' not found

Next step: Perform the retina filter; need to think about the proper union/intersection of the retina/x expression values

In the previous block, we are making 2 global comparisons, here is one of them:

(p15hetscn/p15hetret)/(p08hetscn/p08hetret)

I therefore want to extract the most logical set of genes higher in some/all of these conditions with respect to the corresponding wt conditions. Previously, in section ‘Extract genes included for each set of contrasts’, I attempted to perform this operation for 2 specific wt conditions. When this was performed, it took the unique(union) of the two sets. Thus it stands to reason that I want to take the unique(union) of all 4 in this instance? e.g.:

(p15hetscn > p15wtscn) | (p15hetret > p15wtret) | (p08hetscn > p08wtscn) | (p08hetret > p08wtret)

I kind of think it should be:

((p15hetscn > p15wtscn) | (p15hetret > p15wtret)) & ((p08hetscn > p08wtscn) | (p08hetret > p08wtret))

gross, perhaps I should just do this manually, given that there are only a few putative translatomes to query?

27 Quick and dirty DESeq2 contrast of contrasts

In a fashion similar to how Hector handled the effect of phagocytosis with Laura and Najib a long time ago, I propose to do a simple subtraction of the results of our two contrasts which comprise the translatome query (I was thinking about this last week, thus the inclusion of them in the de tables above). Similarly to the phagocytosis effect, I will simply take the worst posible adjusted p-value. I will repeat this with limma/EdgeR and see how similar the final results are to what those methods provide in the (a/b)/(c/d) comparisons. I am reasonably certain that DESeq2’s results() function has the ability to perform these odd contrasts, but I have never figured out how; perhaps I will use this as a chance to revisit that…

Let us test this idea with the p08 dlgn query, which seeks to compare:

(p08_het_dlgn / p08_het_retina) / (p08_ko_dlgn / p08_ko_retina)

These are maintained in the de_table with the names ‘p08_het_dlgn_vs_retina’ and ‘p08_ko_dlgn_vs_retina’

27.1 p08 dlgn het vs ko

p08_dlgn_combined_deseq <- subtract_deseq_results(
  first_table = p08_dlgn_combined_test[["data"]][["p08_het_dlgn_vs_retina"]],
  second_table = p08_dlgn_combined_test[["data"]][["p08_ko_dlgn_vs_retina"]],
  first_lfc = "deseq_logfc", second_lfc = "deseq_logfc",
  first_p = "deseq_adjp", second_p = "deseq_adjp",
  first_name = "het", second_name = "ko",
  excel = glue("34translatome_deseqsub_xlsx/translatome_p08_dlgn_combined_deseq-v{ver}.xlsx"))
## Error in `subtract_deseq_results()`:
## ! could not find function "subtract_deseq_results"

See how similar these results are to those obtained from limma/edger.

test_columns <- c("edger_logfc", "limma_logfc", "edger_adjp", "limma_adjp")
test_df <- p08_dlgn_combined_test[["data"]][["p08_dlgn_translatome"]][, test_columns]
test_df <- merge(test_df, p08_dlgn_combined_deseq, by = "row.names")
## Error in `h()`:
## ! error in evaluating the argument 'y' in selecting a method for function 'merge': object 'p08_dlgn_combined_deseq' not found
rownames(test_df) <- test_df[["Row.names"]]
test_df[["Row.names"]] <- NULL
cor.test(test_df[["limma_logfc"]], test_df[["het_vs_ko_logfc"]])
## Error in `cor.test.default()`:
## ! 'y' must be a numeric vector
cor.test(test_df[["edger_logfc"]], test_df[["het_vs_ko_logfc"]])
## Error in `cor.test.default()`:
## ! 'y' must be a numeric vector
tt <- plot_linear_scatter(test_df[, c("limma_logfc", "het_vs_ko_logfc")])
## Error in `h()`:
## ! error in evaluating the argument 'x' in selecting a method for function 'as.data.frame': subscript contains invalid names
tt[["scatter"]]
## NULL
tt <- plot_linear_scatter(test_df[, c("edger_adjp", "het_vs_ko_p")])
## Error in `h()`:
## ! error in evaluating the argument 'x' in selecting a method for function 'as.data.frame': subscript contains invalid names
tt[["scatter"]]
## NULL
## So, using the maximum p-value is a complete failure; but the extreme similarities
## between this and edgeR suggest to me that it is likely possible to use the results
## from edgeR without concern (or limma for that matter, it was also extremely similar)
## Or I can spend a little time and collect the numbers on each side of the division
## and calculate a t statistic myself.

28 Non-Specific filtering of the translatome data

I have on hand

  • p08het_vs_p08ko : (p08_het_scn - p08_het_retina) - (p08_ko_scn - p08_ko_retina)
  • p15het_vs_p15ko : (p15_het_scn - p15_het_retina) - (p15_ko_scn - p15_ko_retina)
  • p08het_vs_p08ko : (p08_het_dlgn - p08_het_retina) - (p08_ko_dlgn - p08_ko_retina)
  • p15het_vs_p08het : (p15_het_scn - p15_het_retina) - (p08_het_scn - p08_het_retina)
  • p15ko_vs_p08ko : (p15_ko_scn - p15_ko_retina) - (p08_ko_scn - p08_ko_retina)

I have gene sets up above which define the genes suitable for each of these pieces. There are only 5 comparisons, let us step through them.

28.1 SCN translatome het/ko at p08

The data for this contrast resides in scn_combined_test\(data\)p08_scn_translatome or the same slot of scn_combined_test_sva

  • p08_het_scn - p08_het_retina) - (p08_ko_scn - p08_ko_retina)

Thus, the inclusion_sig portions to extract are found in: inclusion_sig[[“deseq”]][[“ups”]], and are named exactly as written above!

p08_het_vs_ko_translatome_unfilt <- scn_combined_test[["data"]][["p08_scn_translatome"]]
num_union <- unique(c(rownames(inclusion_sig[["deseq"]][["ups"]][["p08_het_scn"]]),
                      rownames(inclusion_sig[["deseq"]][["ups"]][["p08_het_retina"]])))
length(num_union)
## [1] 1188
den_union <- unique(c(rownames(inclusion_sig[["deseq"]][["ups"]][["p08_ko_scn"]]),
                      rownames(inclusion_sig[["deseq"]][["ups"]][["p08_ko_retina"]])))
length(den_union)
## [1] 1624
both_union <- unique(c(num_union, den_union))
length(both_union)
## [1] 2005
both_inter_idx <- num_union %in% den_union
both_inter <- num_union[both_inter_idx]
length(both_inter)
## [1] 807
keeper <- list("p08_scn_translatome" = c("p08het", "p08ko"))
p08_scn_translatome_union_filtered <- combine_de_tables(
  scn_translatome_de, keepers = keeper,
  label_column = label_column,
  excel = glue("35translatome_union/p08_scn_translatome_union_filtered_nosva-v{ver}.xlsx"),
  wanted_genes = both_union)
## Looking for subscript invalid names, start of extract_keepers.
## coefficient edger did not find conditionp08ko or conditionp08het.
## coefficient limma did not find p08ko or p08het.
## Looking for subscript invalid names, end of extract_keepers.
p08_scn_translatome_inter_filtered <- combine_de_tables(
  scn_translatome_de, keepers = keeper,
  label_column = label_column,
  excel = glue("35translatome_union/p08_scn_translatome_intersect_filtered_nosva-v{ver}.xlsx"),
  wanted_genes = both_inter)
## Looking for subscript invalid names, start of extract_keepers.
## coefficient edger did not find conditionp08ko or conditionp08het.
## coefficient limma did not find p08ko or p08het.
## Looking for subscript invalid names, end of extract_keepers.
p08_scn_translatome_union_filtered_sva <- combine_de_tables(
  scn_translatome_de_sva, keepers = keeper,
  label_column = label_column,
  excel = glue("35translatome_union/p08_scn_translatome_union_filtered_sva-v{ver}.xlsx"),
  wanted_genes = both_union)
## Error:
## ! object 'scn_translatome_de_sva' not found
p08_scn_translatome_union_filtered <- combine_de_tables(
  scn_translatome_de, keepers = keeper,
  label_column = label_column,
  excel = glue("35translatome_union/p08_scn_translatome_intersect_filtered_sva-v{ver}.xlsx"),
  wanted_genes = both_inter)
## Looking for subscript invalid names, start of extract_keepers.
## coefficient edger did not find conditionp08ko or conditionp08het.
## coefficient limma did not find p08ko or p08het.
## Looking for subscript invalid names, end of extract_keepers.

29 Venn/UpSet of Retina, SCN, and dLGN DE Genes

Here is a snippet from Rashmi which expresses nicely the DE-result comparisons she is most interested:

Since, I want to know the number of DEG expressed in Retina, SCN and dLGN with respect to genotype, Location and time. I prepared the venn diagram for these comparison:

  • Genotype: P8 Ret Het vs KO, P15 Ret Het vs KO, P8 SCN Het vs KO, P15 SCN Het vs KO, P8 dLGN Het vs KO, P15 dLGN Het vs KO
  • Location: P8_het Ret vs SCN, P8_KO Ret vs SCN, P15_het Ret vs SCN, P15_KO Ret vs SCN, P8_het Ret vs dLGN, P8_KO Ret vs dLGN, P15_het Ret vs dLGN, P15_KO Ret vs dLGN, P8_het SCN vs dLGN, P8_KO SCN vs dLGN, P15_het SCN vs dLGN, P15_KO SCN vs dLGN.

Since I was interested in understanding the change in local translatome according to Location for different developmental time points for Het and KO. Hence, I tried to generate a venn diagram for Location (Ret and SCN) at developmental time points P8 and P15 for genotype het and KO. So the venn diagram / upset plot will be for location where some genes will be shared/unique for P8_Ret_het, P8_SCN_Het, P15_Ret_HET, P15_SCN_HET. We can prepare an upset plot for P8_Ret_KO, P8_SCN_KO, P15_Ret_KO and P15_SCN_KO also. Or can generate an upset plot by combining both P8_Ret_het, P8_SCN_Het, P15_Ret_HET and P15_SCN_HET and P8_Ret_KO, P8_SCN_KO, P15_Ret_KO and P15_SCN_KO.

Ok, let us see if I can implement this, starting with the genotype query

  • Genotype: P8 Ret Het vs KO, P15 Ret Het vs KO, P8 SCN Het vs KO, P15 SCN Het vs KO, P8 dLGN Het vs KO, P15 dLGN Het vs KO

29.1 ko vs het; all locations and times

## The appropriate data structure is 'genotype_tables',
## and the tables of interest are:
table_names <- c("kh_p08_retina", "kh_p15_retina", "kh_p08_scn",
                 "kh_p15_scn", "kh_p08_dlgn", "kh_p15_dlgn")
table_names %in% names(genotype_sig)
## [1] FALSE FALSE FALSE FALSE  TRUE FALSE
newsig <- genotype_sig[[1]]
for (sig in 2:length(table_names)) {
  name <- table_names[sig]
  newsig[["deseq"]][["ups"]][[name]] <- genotype_sig[[name]][["deseq"]][["ups"]][[name]]
  newsig[["deseq"]][["downs"]][[name]] <- genotype_sig[[name]][["deseq"]][["downs"]][[name]]
}
genotype_upsetr <- upsetr_sig(newsig)
## Error in `1:ncol(data)`:
## ! argument of length 0
genotype_upset_written <- write_upset_groups(genotype_upsetr, excel = "36upset_genotype/genotype_upset_groups.xlsx")
## Error:
## ! object 'genotype_upsetr' not found
genotype_upsetr[["all_plot"]]
## Error:
## ! object 'genotype_upsetr' not found
pp(file = "36upset_genotype/test_genotype_upset.pdf")
## Warning in pp(file = "36upset_genotype/test_genotype_upset.pdf"): The
## directory: 36upset_genotype does not exist, will attempt to create it.
print(genotype_upsetr[["all_plot"]])
## Error:
## ! object 'genotype_upsetr' not found
plotted <- dev.off()

Now let us try the location-specific comparisons

29.2 scn vs retina, p08

## The appropriate data structure is 'genotype_tables',
## and the tables of interest are:
table_names <- c("sr_p08_het", "sr_p08_ko")
table_names %in% names(location_sig)
## [1] FALSE FALSE
location_upset_input <- list()
first_table <- table_names[1]
newsig <- location_sig[[first_table]]
for (sig in 2:length(table_names)) {
  name <- table_names[sig]
  newsig[["deseq"]][["ups"]][[name]] <- location_sig[[name]][["deseq"]][["ups"]][[name]]
  newsig[["deseq"]][["downs"]][[name]] <- location_sig[[name]][["deseq"]][["downs"]][[name]]
}
location_upsetr <- upsetr_sig(newsig)
## Error in `xtfrm.data.frame()`:
## ! cannot xtfrm data frames
location_upset_written <- write_upset_groups(location_upsetr, excel = "36upset_genotype/sr_p08_hetko_upset_groups.xlsx")
## Error:
## ! object 'location_upsetr' not found
location_upsetr[["all_plot"]]
## Error:
## ! object 'location_upsetr' not found
pp(file = "36upset_genotype/test_location_sr_p08_hetko_upset.pdf")
print(location_upsetr[["all_plot"]])
## Error:
## ! object 'location_upsetr' not found
plotted <- dev.off()

I am reasonably certain that Rashmi would like a table of the genes shared among increased scn ko and het in the above plot along with the increased retina (e.g. the 269 and 103 gene sets).

29.3 scn vs retina, p15

table_names <- c("sr_p15_het", "sr_p15_ko")
table_names %in% names(location_sig)
## [1] FALSE FALSE
location_upset_input <- list()
first_table <- table_names[1]
newsig <- location_sig[[first_table]]
for (sig in 2:length(table_names)) {
  name <- table_names[sig]
  newsig[["deseq"]][["ups"]][[name]] <- location_sig[[name]][["deseq"]][["ups"]][[name]]
  newsig[["deseq"]][["downs"]][[name]] <- location_sig[[name]][["deseq"]][["downs"]][[name]]
}
location_upsetr <- upsetr_sig(newsig)
## Error in `xtfrm.data.frame()`:
## ! cannot xtfrm data frames
location_upset_written <- write_upset_groups(location_upsetr, excel = "36upset_genotype/sr_p15_hetko_upset_groups.xlsx")
## Error:
## ! object 'location_upsetr' not found
location_upsetr[["all_plot"]]
## Error:
## ! object 'location_upsetr' not found
print(scn_retina_p15_upset_result)
## Error:
## ! object 'scn_retina_p15_upset_result' not found
pp(file = "36upset_genotype/test_location_sr_p15_hetko_upset.pdf")
print(location_upsetr[["all_plot"]])
## Error:
## ! object 'location_upsetr' not found
plotted <- dev.off()

29.4 dlgn vs retina, p08

## The appropriate data structure is 'genotype_tables',
## and the tables of interest are:
table_names <- c("dr_p08_het", "dr_p08_ko")
location_upset_input <- list()
first_table <- table_names[1]
newsig <- location_sig[[first_table]]
for (sig in 2:length(table_names)) {
  name <- table_names[sig]
  newsig[["deseq"]][["ups"]][[name]] <- location_sig[[name]][["deseq"]][["ups"]][[name]]
  newsig[["deseq"]][["downs"]][[name]] <- location_sig[[name]][["deseq"]][["downs"]][[name]]
}
location_upsetr <- upsetr_sig(newsig)
## Error in `1:ncol(data)`:
## ! argument of length 0
location_upset_written <- write_upset_groups(location_upsetr, excel = "3upset_genotype/dr_p08_hetko_upset_groups.xlsx")
## Error:
## ! object 'location_upsetr' not found
location_upsetr[["all_plot"]]
## Error:
## ! object 'location_upsetr' not found
pp(file = "36upset_genotype/test_location_dr_p08_hetko_upset.pdf")
print(location_upsetr[["all_plot"]])
## Error:
## ! object 'location_upsetr' not found
plotted <- dev.off()

29.5 dlgn vs retina, p15

## The appropriate data structure is 'genotype_tables',
## and the tables of interest are:
table_names <- c("dr_p15_het", "dr_p15_ko")
location_upset_input <- list()
first_table <- table_names[1]
newsig <- location_sig[[first_table]]
for (sig in 2:length(table_names)) {
  name <- table_names[sig]
  newsig[["deseq"]][["ups"]][[name]] <- location_sig[[name]][["deseq"]][["ups"]][[name]]
  newsig[["deseq"]][["downs"]][[name]] <- location_sig[[name]][["deseq"]][["downs"]][[name]]
}
location_upsetr <- upsetr_sig(newsig)
## Error in `xtfrm.data.frame()`:
## ! cannot xtfrm data frames
location_upset_written <- write_upset_groups(location_upsetr, excel = "37upset_location/dr_p15_hetko_upset_groups.xlsx")
## Error:
## ! object 'location_upsetr' not found
location_upsetr[["all_plot"]]
## Error:
## ! object 'location_upsetr' not found
pp(file = "37upset_location/test_location_dr_p15_hetko_upset.pdf")
## Warning in pp(file = "37upset_location/test_location_dr_p15_hetko_upset.pdf"):
## The directory: 37upset_location does not exist, will attempt to create it.
print(location_upsetr[["all_plot"]])
## Error:
## ! object 'location_upsetr' not found
plotted <- dev.off()

29.6 dlgn vs scn, p08

table_names <- c("ds_p08_het", "ds_p08_ko")
location_upset_input <- list()
first_table <- table_names[1]
newsig <- location_sig[[first_table]]
for (sig in 2:length(table_names)) {
  name <- table_names[sig]
  newsig[["deseq"]][["ups"]][[name]] <- location_sig[[name]][["deseq"]][["ups"]][[name]]
  newsig[["deseq"]][["downs"]][[name]] <- location_sig[[name]][["deseq"]][["downs"]][[name]]
}
location_upsetr <- upsetr_sig(newsig)
## Error in `xtfrm.data.frame()`:
## ! cannot xtfrm data frames
location_upset_written <- write_upset_groups(location_upsetr, excel = "37upset_location/ds_p08_hetko_upset_groups.xlsx")
## Error:
## ! object 'location_upsetr' not found
location_upsetr[["all_plot"]]
## Error:
## ! object 'location_upsetr' not found
pp(file = "37upset_location/test_location_ds_p08_hetko_upset.pdf")
print(location_upsetr[["all_plot"]])
## Error:
## ! object 'location_upsetr' not found
plotted <- dev.off()

29.7 dlgn vs scn, p15

table_names <- c("ds_p15_het", "ds_p15_ko")
location_upset_input <- list()
first_table <- table_names[1]
newsig <- location_sig[[first_table]]
for (sig in 2:length(table_names)) {
  name <- table_names[sig]
  newsig[["deseq"]][["ups"]][[name]] <- location_sig[[name]][["deseq"]][["ups"]][[name]]
  newsig[["deseq"]][["downs"]][[name]] <- location_sig[[name]][["deseq"]][["downs"]][[name]]
}
location_upsetr <- upsetr_sig(newsig)
## Error in `xtfrm.data.frame()`:
## ! cannot xtfrm data frames
location_upset_written <- write_upset_groups(location_upsetr, excel = "37upset_location/ds_p15_hetko_upset_groups.xlsx")
## Error:
## ! object 'location_upsetr' not found
location_upsetr[["all_plot"]]
## Error:
## ! object 'location_upsetr' not found
pp(file = "37upset_location/test_location_ds_p15_hetko_upset.pdf")
print(location_upsetr[["all_plot"]])
## Error:
## ! object 'location_upsetr' not found
plotted <- dev.off()

30 Shared and unique gene sets across x/wt

In this block I want to find the unique and shared genes between:

  1. scn p8 het/wt and retina p8 het/wt: hwp08scninc, hwp08retinc, hwp08scndec, hwp08retdec
  2. scn p15 het/wt and retina p15 het/wt: hwp15scninc, hwp15retinc, hwp15scndec, hwp15retdec
  3. #1 and #2 together: 8 catgories above
  4. scn p8 ko/wt and retina p8 ko/wt
  5. scn p15 ko/wt and retina p15 ko/wt
  6. #4 and #5 together

The comparisons of het/wt are found in the ‘inclusion_sig’ dataset; because they are providing our cutoffs for nonspecific binding.

30.1 Number 1 above: p08_het vs wt for scn and retina.

table_names <- c("p08_het_scn", "p08_het_retina")
inclusion_upsetr <- upsetr_sig(inclusion_sig, contrasts = table_names)
inclusion_upset_written <- write_upset_groups(inclusion_upsetr, excel = "37upset_locations/rs_p08_het_inclusion_upset_groups.xlsx")
inclusion_upsetr[["all_plot"]]

pp(file = "37upset_locations/inclusion_sr_p08_upset.pdf")
print(inclusion_upsetr[["all_plot"]])
plotted <- dev.off()

30.2 Number 2 above: p15_het vs wt for scn and retina.

table_names <- c("p15_het_scn", "p15_het_retina")
inclusion_upsetr <- upsetr_sig(inclusion_sig, contrasts = table_names)
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
inclusion_upset_written <- write_upset_groups(inclusion_upsetr, excel = "37upset_locations/rs_p15_het_inclusion_upset_groups.xlsx")
inclusion_upsetr[["all_plot"]]

pp(file = "37upset_locations/inclusion_sr_p15_upset.pdf")
print(inclusion_upsetr[["all_plot"]])
plotted <- dev.off()

30.3 Number 3 above: combination of #1 and #2

table_names <- c("p08_het_scn", "p08_het_retina", "p15_het_scn", "p15_het_retina")
inclusion_upsetr <- upsetr_sig(inclusion_sig, contrasts = table_names)
inclusion_upset_written <- write_upset_groups(inclusion_upsetr, excel = "37upset_locations/rs_p08p15_het_inclusion_upset_groups.xlsx")
inclusion_upsetr[["all_plot"]]

pp(file = "37upset_locations/inclusion_sr_p08p15_upset.pdf")
print(inclusion_upsetr[["all_plot"]])
plotted <- dev.off()

30.4 Number 4 above p08_ko vs wt for scn and retina.

table_names <- c("p08_ko_scn", "p08_ko_retina")
inclusion_upsetr <- upsetr_sig(inclusion_sig, contrasts = table_names)
inclusion_upset_written <- write_upset_groups(inclusion_upsetr, excel = "37upset_locations/rs_p08_ko_inclusion_upset_groups.xlsx")
inclusion_upsetr[["all_plot"]]

pp(file = "37upset_locations/inclusion_sr_p08_ko_upset.pdf")
print(inclusion_upsetr[["all_plot"]])
plotted <- dev.off()

30.5 Number 5 above p15_ko vs wt for scn and retina.

table_names <- c("p15_ko_scn", "p15_ko_retina")
inclusion_upsetr <- upsetr_sig(inclusion_sig, contrasts = table_names)
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
## `geom_line()`: Each group consists of only one observation.
## i Do you need to adjust the group aesthetic?
inclusion_upset_written <- write_upset_groups(inclusion_upsetr, excel = "37upset_locations/rs_p15_ko_inclusion_upset_groups.xlsx")
inclusion_upsetr[["all_plot"]]

pp(file = "37upset_locations/inclusion_sr_p15_ko_upset.pdf")
print(inclusion_upsetr[["all_plot"]])
plotted <- dev.off()

30.6 Number 6 above: Combining #4 and #5

table_names <- c("p08_ko_scn", "p08_ko_retina", "p15_ko_scn", "p15_ko_retina")
inclusion_upsetr <- upsetr_sig(inclusion_sig, contrasts = table_names)
inclusion_upset_written <- write_upset_groups(inclusion_upsetr, excel = "37upset_locations/rs_p08p15_ko_inclusion_upset_groups.xlsx")
inclusion_upsetr[["all_plot"]]

pp(file = "37upset_locations/inclusion_sr_p08p15_ko_upset.pdf")
print(inclusion_upsetr[["all_plot"]])
plotted <- dev.off()

31 GSVA

msigdb <- "reference/msigdb_v2024.1.Mm.db"
if (file.exists(msigdb)) {
  v3_h_gsva <- simple_gsva(v3_pairwise_input, orgdb = "org.Mm.eg.db", signature_category = "mh",
                           signatures = msigdb, id_source = "fdata",
                           required_id = "mgi_symbol")
  v3_h_gsva

  v3_h_gsva_sig <- get_sig_gsva_categories(
    v3_h_gsva, excel = "38msigdb/gsva_sig_hallmark_categories.xlsx")
  v3_h_gsva_sig

  v3_m1_gsva <- simple_gsva(v3_pairwise_input, orgdb = "org.Mm.eg.db", signature_category = "m1",
                            signatures = msigdb, id_source = "fdata",
                            required_id = "mgi_symbol")
  v3_m1_gsva
  v3_m1_gsva_sig <- get_sig_gsva_categories(
    v3_m1_gsva, excel = "38msigdb/gsva_sig_positional_categories.xlsx")
  v3_m1_gsva_sig

  v3_m2_gsva <- simple_gsva(v3_pairwise_input, orgdb = "org.Mm.eg.db", signature_category = "m2",
                            signatures = msigdb, id_source = "fdata",
                            required_id = "mgi_symbol")
  v3_m2_gsva
  v3_m2_gsva_sig <- get_sig_gsva_categories(
    v3_m2_gsva, excel = "38msigdb/gsva_sig_curated_categories.xlsx")
  v3_m2_gsva_sig

  v3_m3_gsva <- simple_gsva(v3_pairwise_input, orgdb = "org.Mm.eg.db", signature_category = "m3",
                            signatures = msigdb, id_source = "fdata",
                            required_id = "mgi_symbol")
  v3_m3_gsva
  v3_m3_gsva_sig <- get_sig_gsva_categories(
    v3_m3_gsva, excel = "38msigdb/gsva_sig_regulatory_categories.xlsx")
  v3_m3_gsva_sig

  v3_m5_gsva <- simple_gsva(v3_pairwise_input, orgdb = "org.Mm.eg.db", signature_category = "m5",
                            signatures = msigdb, id_source = "fdata",
                            required_id = "mgi_symbol")
  v3_m5_gsva
  v3_m5_gsva_sig <- get_sig_gsva_categories(
    v3_m5_gsva, excel = "38msigdb/gsva_sig_ontology_categories.xlsx")
  v3_m5_gsva_sig

  v3_m8_gsva <- simple_gsva(v3_pairwise_input, orgdb = "org.Mm.eg.db", signature_category = "m8",
                            signatures = msigdb, id_source = "fdata",
                            required_id = "mgi_symbol")
  v3_m8_gsva
  v3_m8_gsva_sig <- get_sig_gsva_categories(
    v3_m8_gsva, excel = "38msigdb/gsva_sig_celltype_categories.xlsx")
  v3_m8_gsva_sig
}
## Error:
## ! no such table: gene_set

32 GSEA images

Up above I created a fairly large set of enrichment/GSEA analyses. Let us pull some of the most interesting results here and look at them.

Here are the specific queries from Rashmi:

  • Genotype (het vs ko):
    • P8 het and ko for Ret
    • SCN (P8 het vs KO SCN
    • P8 het vs KO Ret)
    • P15 het and ko for Ret
  • Location (somal vs axonal):
    • SR_P08_KO
    • SR_P08_Het
    • SR_P15_KO
    • SR_P15_Het
  • Time(p8vs p15):
    • t_het_Ret_ po8-p15
    • t_KO_Ret_C po8-p15
    • t_het_SCN_po8-p15
    • t_KO_SCN_po8-p15

32.1 Genotype

Let us take a moment and see for which contrasts I acquired results:

I need to make a little summary for clusterprofiler too so that I can easily see how many hits there are for each contrast.

summary(genotype_full_gp)
##                 Length Class            Mode
## kh_p08_dlgn_up  20     gprofiler_result list
## kh_p15_dlgn_up  27     gprofiler_result list
## kh_p08_scn_up   20     gprofiler_result list
## kh_p08_scn_down 24     gprofiler_result list
## kh_p15_scn_down 25     gprofiler_result list
for (i in names(genotype_full_gp)) {
  print(i)
  print(genotype_full_gp[[i]][["num_hits"]])
}
## [1] "kh_p08_dlgn_up"
##    BP    CC CORUM    HP  KEGG    MF  REAC    TF    WP 
##     0     0     0     0     0     0     0     0     0 
## [1] "kh_p15_dlgn_up"
##    BP    CC CORUM    HP  KEGG    MF  REAC    TF    WP 
##    63    32     0   118     1    21     6     3     0 
## [1] "kh_p08_scn_up"
##    BP    CC CORUM    HP  KEGG    MF  REAC    TF    WP 
##     0     0     0     0     0     0     0     0     0 
## [1] "kh_p08_scn_down"
##    BP    CC CORUM    HP  KEGG    MF  REAC    TF    WP 
##    58    21     0     0     0     3     0   113     0 
## [1] "kh_p15_scn_down"
##    BP    CC CORUM    HP  KEGG    MF  REAC    TF    WP 
##    49     9     0     0     1     4     0     1     0
summary(genotype_full_cp)
## Error:
## ! object 'genotype_full_cp' not found
for (i in names(genotype_full_cp)) {
  print(i)
  print(nrow(genotype_full_cp[[i]][["gse_go"]]))
}
## Error:
## ! object 'genotype_full_cp' not found

32.1.1 p8 het/ko for retina:

This contrast, even before filtering away the high-wt genes, only has 8 genes in the set of up and down genes combined. As a result, my function which performs gProfiler/clusterProfiler skips it, and also skips the p15 het/ko for retina samples.

32.1.2 p8 het/ko for scn:

This has a bunch more genes: 51 up and 128 down. Unfortunately, gProfiler sees no significant over-representation in the up category of genes. The down category has

The up/down sets from clusterProfiler have enrich_go, gse_go, and go_data to look at.

genotype_full_gp$kh_p08_scn_up$num_hits
##    BP    CC CORUM    HP  KEGG    MF  REAC    TF    WP 
##     0     0     0     0     0     0     0     0     0
genotype_full_gp$kh_p08_scn_down$num_hits
##    BP    CC CORUM    HP  KEGG    MF  REAC    TF    WP 
##    58    21     0     0     0     3     0   113     0
plots <- plot_enrichresult(genotype_full_gp$kh_p08_scn_down[["BP_enrich"]])
## Warning in (function (model, data, ...) : Arguments in `...` must be used.
## x Problematic argument:
## * by = "Count"
## i Did you misspell an argument name?
plots[["dot"]]

plots[["tree"]]

Perhaps I should just ask the question: for which categories did I get results back?

summary(genotype_full_gp)
##                 Length Class            Mode
## kh_p08_dlgn_up  20     gprofiler_result list
## kh_p15_dlgn_up  27     gprofiler_result list
## kh_p08_scn_up   20     gprofiler_result list
## kh_p08_scn_down 24     gprofiler_result list
## kh_p15_scn_down 25     gprofiler_result list

kh_p08_dlgn_up: No significant gProfiler results. kh_p15_dlgn_up: Significant BP, HP, KEGG, MF, REAC, TF kh_p08_scn_up: No significant gProfiler results. kh_p08_scn_down: Significant BP, MiRNA, MF, TF kh_p15_scn_down: Significant BP, MF

plots <- plot_enrichresult(genotype_full_gp[["kh_p15_dlgn_up"]][["BP_enrich"]])
## Warning in (function (model, data, ...) : Arguments in `...` must be used.
## x Problematic argument:
## * by = "Count"
## i Did you misspell an argument name?
plots[["dot"]]

32.2 Location

32.2.1 Scn vs retina ko, p08

plots <- plot_enrichresult(location_gp[["sr_p08_ko"]][["sr_p08_ko_up"]][["BP_enrich"]])
plots[["dot"]]
## NULL
plots <- plot_enrichresult(location_gp[["sr_p08_ko"]][["sr_p08_ko_down"]][["BP_enrich"]])
plots[["dot"]]
## NULL

32.2.2 scn vs retina, het, p08

Enriched groups: BP, KEGG, MF, TF, CC

summary(location_gp[["sr_p08_het"]][["sr_p08_het_up"]])
## Length  Class   Mode 
##      0   NULL   NULL
plots <- plot_enrichresult(location_gp[["sr_p08_het"]][["sr_p08_het_up"]][["BP_enrich"]])
plots[["dot"]]
## NULL
plots <- plot_enrichresult(location_gp[["sr_p08_het"]][["sr_p08_het_up"]][["CC_enrich"]])
plots[["dot"]]
## NULL
plots <- plot_enrichresult(location_gp[["sr_p08_het"]][["sr_p08_het_down"]][["BP_enrich"]])
plots[["dot"]]
## NULL
sr_p08_het_topn_gsea <- plot_topn_gsea(location_cp[[""]])
## Error in `if (nrow(gse) < topn) ...`:
## ! argument is of length zero

32.2.3 Scn vs retina ko, p15

plots <- plot_enrichresult(location_gp[["sr_p15_ko"]][["sr_p15_ko_up"]][["BP_enrich"]])
plots[["dot"]]
## NULL
plots <- plot_enrichresult(location_gp[["sr_p15_ko"]][["sr_p15_ko_down"]][["BP_enrich"]])
plots[["dot"]]
## NULL

32.2.4 scn vs retina, het, p15

plots <- plot_enrichresult(location_gp[["sr_p15_het"]][["sr_p15_het_up"]][["BP_enrich"]])
plots[["dot"]]
## NULL
plots <- plot_enrichresult(location_gp[["sr_p15_het"]][["sr_p15_het_down"]][["BP_enrich"]])
plots[["dot"]]
## NULL

32.3 Time

32.3.1 het retina

Ups: significant results for BP, MF, TF Downs: BP, MF, REAC, TF, WP

plots <- plot_enrichresult(time_gp[["t_het_retina"]][["t_het_retina_up"]][["BP_enrich"]])
plots[["dot"]]
## NULL
plots <- plot_enrichresult(time_gp[["t_het_retina"]][["t_het_retina_down"]][["BP_enrich"]])
plots[["dot"]]
## NULL

32.3.2 ko retina

Up: BP, MiRNA, MF Down: BP, MF, REAC, TF

plots <- plot_enrichresult(time_gp[["t_ko_retina"]][["t_ko_retina_up"]][["BP_enrich"]])
plots[["dot"]]
## NULL
plots <- plot_enrichresult(time_gp[["t_ko_retina"]][["t_ko_retina_down"]][["BP_enrich"]])
plots[["dot"]]
## NULL

32.3.3 het scn

Neither of the SCN gProfiler queries provided any results.

33 Bibliography

pander::pander(sessionInfo())
message(paste0("This is hpgltools commit: ", get_git_commit()))
message(paste0("Saving to ", savefile))
tmp <- sm(saveme(filename = savefile))
tmp <- loadme(filename = savefile)
Hänzelmann, Sonja, Robert Castelo, and Justin Guinney. 2013. GSVA: Gene Set Variation Analysis for Microarray and RNA-Seq Data.” BMC Bioinformatics 14 (1): 7. https://doi.org/10.1186/1471-2105-14-7.
Hoffman, Gabriel E, and Panos Roussos. 2020. “Dream: Powerful Differential Expression Analysis for Repeated Measures Designs.” Bioinformatics 37 (2): 192–201. https://doi.org/10.1093/bioinformatics/btaa687.
Hoffman, Gabriel E., and Eric E. Schadt. 2016. variancePartition: Interpreting Drivers of Variation in Complex Gene Expression Studies.” BMC Bioinformatics 17 (1): 483. https://doi.org/10.1186/s12859-016-1323-z.
Leek, Jeffrey T., W. Evan Johnson, Hilary S. Parker, Andrew E. Jaffe, and John D. Storey. 2012. “The SVA Package for Removing Batch Effects and Other Unwanted Variation in High-Throughput Experiments.” Bioinformatics 28 (6): 882–83. https://doi.org/10.1093/bioinformatics/bts034.
Leng, Ning, John A. Dawson, James A. Thomson, Victor Ruotti, Anna I. Rissman, Bart M. G. Smits, Jill D. Haag, Michael N. Gould, Ron M. Stewart, and Christina Kendziorski. 2013. EBSeq: An Empirical Bayes Hierarchical Model for Inference in RNA-seq Experiments.” Bioinformatics 29 (8): 1035–43. https://doi.org/10.1093/bioinformatics/btt087.
Li, Huiling, Yangao Huo, Xi He, Liping Yao, Hao Zhang, Yiqiang Cui, Huijuan Xiao, et al. 2022. “A Male Germ-Cell-Specific Ribosome Controls Male Fertility.” Nature 612 (7941): 725–31. https://doi.org/10.1038/s41586-022-05508-0.
Love, Michael I., Wolfgang Huber, and Simon Anders. 2014. “Moderated Estimation of Fold Change and Dispersion for RNA-Seq Data with DESeq2.” bioRxiv. https://doi.org/10.1101/002832.
Raudvere, Uku, Liis Kolberg, Ivan Kuzmin, Tambet Arak, Priit Adler, Hedi Peterson, and Jaak Vilo. 2019. “G:Profiler: A Web Server for Functional Enrichment Analysis and Conversions of Gene Lists (2019 Update).” Nucleic Acids Research 47 (W1): W191–98. https://doi.org/10.1093/nar/gkz369.
Ritchie, Matthew E., Belinda Phipson, Di Wu, Yifang Hu, Charity W. Law, Wei Shi, and Gordon K. Smyth. 2015. “Limma Powers Differential Expression Analyses for RNA-sequencing and Microarray Studies.” Nucleic Acids Research 43 (7): e47. https://doi.org/10.1093/nar/gkv007.
Robinson, Mark D., Davis J. McCarthy, and Gordon K. Smyth. 2010. edgeR: A Bioconductor Package for Differential Expression Analysis of Digital Gene Expression Data.” Bioinformatics 26 (1): 139–40. https://doi.org/10.1093/bioinformatics/btp616.
Smedley, Damian, Syed Haider, Benoit Ballester, Richard Holland, Darin London, Gudmundur Thorisson, and Arek Kasprzyk. 2009. BioMart – Biological Queries Made Easy.” BMC Genomics 10 (1): 22. https://doi.org/10.1186/1471-2164-10-22.
Tarazona, Sonia, Fernando García, Alberto Ferrer, Joaquín Dopazo, and Ana Conesa. 2011. NOIseq: A RNA-seq Differential Expression Method Robust for Sequencing Depth Biases.” EMBnet.journal 17 (B): 18–19. https://doi.org/10.14806/ej.17.B.265.
Yu, Guangchuang. n.d. 📖 Introduction Biomedical Knowledge Mining Using GOSemSim and clusterProfiler. Accessed June 21, 2024. https://yulab-smu.top/biomedical-knowledge-mining-book/.
LS0tCnRpdGxlOiAiQW5hbHlzZXMgb2YgdGhlIElQUkdDIHJlLXByb2Nlc3NlZCBzYW1wbGVzLiIKYXV0aG9yOiAiYXRiIGFiZWxld0BnbWFpbC5jb20iCmRhdGU6ICJgciBTeXMuRGF0ZSgpYCIKYmlibGlvZ3JhcGh5OiBhdGIuYmliCm91dHB1dDoKICBodG1sX2RvY3VtZW50OgogICAgY29kZV9kb3dubG9hZDogdHJ1ZQogICAgY29kZV9mb2xkaW5nOiBzaG93CiAgICBmaWdfY2FwdGlvbjogdHJ1ZQogICAgZmlnX2hlaWdodDogNwogICAgZmlnX3dpZHRoOiA3CiAgICBoaWdobGlnaHQ6IHplbmJ1cm4KICAgIGtlZXBfbWQ6IGZhbHNlCiAgICBtb2RlOiBzZWxmY29udGFpbmVkCiAgICBudW1iZXJfc2VjdGlvbnM6IHRydWUKICAgIHNlbGZfY29udGFpbmVkOiB0cnVlCiAgICB0aGVtZTogcmVhZGFibGUKICAgIHRvYzogdHJ1ZQogICAgdG9jX2Zsb2F0OgogICAgICBjb2xsYXBzZWQ6IGZhbHNlCiAgICAgIHNtb290aF9zY3JvbGw6IGZhbHNlCiAgICAgIHJtZGZvcm1hdHM6OnJlYWR0aGVkb3duOgogICAgICAgIGNvZGVfZG93bmxvYWQ6IHRydWUKICAgICAgICBjb2RlX2ZvbGRpbmc6IHNob3cKICAgICAgICBkZl9wcmludDogcGFnZWQKICAgICAgICBmaWdfY2FwdGlvbjogdHJ1ZQogICAgICAgIGZpZ19oZWlnaHQ6IDcKICAgICAgICBmaWdfd2lkdGg6IDcKICAgICAgICBoaWdobGlnaHQ6IHplbmJ1cm4KICAgICAgICB3aWR0aDogMzAwCiAgICAgICAga2VlcF9tZDogZmFsc2UKICAgICAgICBtb2RlOiBzZWxmY29udGFpbmVkCiAgICAgICAgdG9jX2Zsb2F0OiB0cnVlCiAgICAgICAgQmlvY1N0eWxlOjpodG1sX2RvY3VtZW50OgogICAgICAgICAgY29kZV9kb3dubG9hZDogdHJ1ZQogICAgICAgICAgY29kZV9mb2xkaW5nOiBzaG93CiAgICAgICAgICBmaWdfY2FwdGlvbjogdHJ1ZQogICAgICAgICAgZmlnX2hlaWdodDogNwogICAgICAgICAgZmlnX3dpZHRoOiA3CiAgICAgICAgICBoaWdobGlnaHQ6IHplbmJ1cm4KICAgICAgICAgIGtlZXBfbWQ6IGZhbHNlCiAgICAgICAgICBtb2RlOiBzZWxmY29udGFpbmVkCiAgICAgICAgICB0b2NfZmxvYXQ6IHRydWUKLS0tCgo8c3R5bGUgdHlwZT0idGV4dC9jc3MiPgpib2R5LCB0ZCB7CmZvbnQtc2l6ZTogMTZweDsKfQpjb2RlLnJ7CmZvbnQtc2l6ZTogMTZweDsKfQpwcmUgewpmb250LXNpemU6IDE2cHgKfQpib2R5IC5tYWluLWNvbnRhaW5lciB7Cm1heC13aWR0aDogMTYwMHB4Owp9Cjwvc3R5bGU+CgpgYGB7ciBvcHRpb25zLCBpbmNsdWRlPUZBTFNFfQpsaWJyYXJ5KGhwZ2x0b29scykKbGlicmFyeShkcGx5cikKbGlicmFyeShlbnJpY2hwbG90KQpsaWJyYXJ5KGdncGxvdDIpCmxpYnJhcnkoZ2dyZXBlbCkKbGlicmFyeShncHJvZmlsZXIyKQp0dCA8LSB0cnkoZGV2dG9vbHM6OmxvYWRfYWxsKCJ+L2hwZ2x0b29scyIpKQprbml0cjo6b3B0c19rbml0JHNldCgKICBwcm9ncmVzcyA9IFRSVUUsIHZlcmJvc2UgPSBUUlVFLCB3aWR0aCA9IDkwLCBlY2hvID0gVFJVRSkKa25pdHI6Om9wdHNfY2h1bmskc2V0KAogIGVycm9yID0gVFJVRSwgZmlnLndpZHRoID0gOCwgZmlnLmhlaWdodCA9IDgsIGZpZy5yZXRpbmEgPSAyLAogIG91dC53aWR0aCA9ICIxMDAlIiwgZGV2ID0gInBuZyIsCiAgZGV2LmFyZ3MgPSBsaXN0KHBuZyA9IGxpc3QodHlwZSA9ICJjYWlyby1wbmciKSkpCm9sZF9vcHRpb25zIDwtIG9wdGlvbnMoZGlnaXRzID0gNCwgc3RyaW5nc0FzRmFjdG9ycyA9IEZBTFNFLCBrbml0ci5kdXBsaWNhdGUubGFiZWwgPSAiYWxsb3ciKQpnZ3Bsb3QyOjp0aGVtZV9zZXQoZ2dwbG90Mjo6dGhlbWVfYncoYmFzZV9zaXplID0gMTIpKQp2ZXIgPC0gIjIwMjQwOCIKcHJldmlvdXNfZmlsZSA8LSAiIgp2ZXIgPC0gZm9ybWF0KFN5cy5EYXRlKCksICIlWSVtJWQiKQoKIyN0bXAgPC0gc20obG9hZG1lKGZpbGVuYW1lPXBhc3RlMChnc3ViKHBhdHRlcm49IlxcLlJtZCIsIHJlcGxhY2U9IiIsIHg9cHJldmlvdXNfZmlsZSksICItdiIsIHZlciwgIi5yZGEueHoiKSkpCnJtZF9maWxlIDwtICJpcHJnY19hbmFseXNlc18yMDI0MDguUm1kIgpzYXZlZmlsZSA8LSBnc3ViKHBhdHRlcm4gPSAiXFwuUm1kIiwgcmVwbGFjZSA9ICJcXC5yZGFcXC54eiIsIHggPSBybWRfZmlsZSkKIyMgTm8gb25lIHdpbGwgZXZlciByZWFkIHRoaXMgSSBzdXNwZWN0LCBidXQgdGhlcmUgd2lsbCBfbmV2ZXJfIGJlIGEgd29yZCBwcm9kdWNlZAojIyBieSBhbiBMTE0gaW4gYSBkb2N1bWVudCB3cml0dGVuIGJ5IG1lLgpgYGAKCk5vdGUgdG8gc2VsZjogSSBjaGFuZ2VkIHRoZSBjbHVzdGVyUHJvZmlsZXIgb3V0cHV0IGxpc3QgdG8gc3RvcmUgdGhlCmVucmljaFJlc3VsdHMgaW4gW1t4eF9kYXRhXV1bW2VucmljaF1dLCBvciBpbiB0aGUgY2FzZSBvZiBnbzoKW1tnb19kYXRhXV1bW0JQX2VucmljaF1dIGluc3RlYWQgb2YgZW5yaWNoX3Jlc3VsdHMuCgojIFNldCBzb21lIHBhcmFtZXRlcnMgd2hpY2ggd2lsbCBiZSB1c2VkIGxhdGVyCgpgYGB7cn0KIyMgTG9nRkMgY3V0b2ZmIHdoZW4gd29ya2luZyBvbiB0aGUgaW5jbHVzaW9uIHNldHMuCmxmY19jdXRvZmYgPC0gMC4xCiMjIEFkanVzdGVkIHAtdmFsdWUgY3V0b2ZmIHdoZW4gd29ya2luZyBvbiB0aGUgaW5jbHVzaW9uIHNldHMuCmFkanBfY3V0b2ZmIDwtIDAuMQojIyBJbmNyZWFzZSB0aGUgbWF4aW11bSBhbGxvd2VkIGdyb3VwIHNpemUgd2hlbiB3b3JraW5nIHdpdGggY2x1c3RlclByb2ZpbGVyCiMjIFRoaXMgc2hvdWxkIGJyaW5nIG91dCBzb21lIG9mIHRoZSBtb3JlIGdlbmVyYWwgZ3JvdXBzIGxpa2UgJ2NlbGxib2R5JwptYXhfZ3JvdXBzaXplIDwtIDIwMDAKIyMgQWxsb3cgZ3JvdXBzIGhpZ2hlciB1cCBpbiB0aGUgdHJlZSBmb3IgY2x1c3RlclByb2ZpbGVyIHJlc3VsdHMuCmdvX2xldmVsIDwtIDIKIyMgQWxsb3cgMTAgR08gY2F0ZWdvcmllcyB0byBiZSBkaXNwbGF5ZWQgd2hlbiBwbG90dGluZy4KZ29fY2F0ZWdvcmllcyA8LSAxNAojIyBNQSBwbG90IHBvaW50IG91dGxpbmVzCm91dGxpbmUgPC0gRkFMU0UKIyMgU3BlZWQgdXAgY2x1c3RlclByb2ZpbGVyIGJ5IGNob29zaW5nIHRoZSBjb3JyZWN0IGtleXR5cGVzCm9yZ2RiX2Zyb20gPC0gIkVOU0VNQkwiCmRlZmF1bHRfZnN0cmluZyA8LSAifiAwICsgY29uZGl0aW9uIgpgYGAKCiMgVE9ET3MKCiogQ2hlY2sgdGhpcyBmb3IgY29ycmVjdG5lc3MuCiogUmVvcmdhbml6ZSBpdAoqIEdTVkEgKEBoYW56ZWxtYW5uR1NWQUdlbmVTZXQyMDEzYSkKKiBDb25zaWRlciBlbWJlZGRpbmcgc29tZS9tYW55L2FsbCBvZiB0aGUgRXhjZWwgb3V0cHV0cyBpbnRvIHRoZSBodG1sCiAgb3V0cHV0IHZpYSB4ZnVuOjplbWJlZF9kaXIoJ2V4Y2VsLycpCgojIE1lZXRpbmcgd2l0aCBUaGVyZXNhCgpQcmV2aW91cyBwYXBlcnMgZGlkIG5vdCBkbyBhbiBleHBsaWNpdCBzdWJ0cmFjdGlvbiwgaW5zdGVhZCBqdXN0CmNvbXBhcmVkIHRvIFdUIGFuZCBrZXB0IHRoZSBnZW5lcyB3aGljaCBhcmUgPiBpbiBkZWx0YS9oZXQgdnMuIHd0LgpUaGVyZSBhcmUgbXVsdGlwbGUgd2F5cyB0byBkZWFsIHdpdGggdGhpcyBhbmQgdGhhdCBxdWVyeSBoYXMgbm90IHlldApiZWVuIGRlZmluZWQuICBMYXRlciwgVGhlcmVzYSBjYW1lIHRvIHRoZSBjb25jbHVzaW9uIHRoYXQgdGhlCnN1YnRyYWN0aW9uIG1ldGhvZCBpcyBub3QgYXBwcm9wcmlhdGUuCgojIEludHJvZHVjdGlvbgoKSW4gdGhpcyBkb2N1bWVudCBJIGhvcGUgdG8gZXhwbG9yZSB0aGUgZnJlc2hseSBwcm9jZXNzZWQgc2FtcGxlcyBhbmQKcGVyZm9ybSBzb21lIGNvbXBhcmlzb25zIHRvIHNlZSB0aGF0IHdlIGhhdmUgdGhlIGV4cGVjdGVkIHNpbWlsYXJpdGllcwphbmQgZGlmZmVyZW5jZXMgZnJvbSB0aGUgcHJpb3IgYW5hbHlzaXMgcGVyZm9ybWVkIGJ5IFRoZXJlc2EuCgpUaGVyZSBpcyBvbmUgd2F5IGluIHdoaWNoIEkgZXhwZWN0IGFueS9hbGwgb2YgdGhlc2UgYW5hbHlzZXMgdG8gYmUKZXhwbGljaXRseSBkaWZmZXJlbnQ6IHRoaXMgc2hvdWxkIGluY2x1ZGUgdGhlIGNoYW5nZXMgcHJvZHVjZWQgYnkKQXByaWwncyByZW5hbWluZyBvZiBzb21lIHNhbXBsZXMuCgpNeSBpbnRlbnRpb24gaXMgdG8gcHJvZHVjZSBhIHNhbXBsZSBzaGVldCB3aGljaCBpbmNsdWRlcyBvbmUgY29sdW1uCndpdGggbm9uLXVtaS1kZWR1cGxpY2F0ZWQgcmVzdWx0cyBhbmQgb25lIHdpdGggZGVkdXBsaWNhdGVkIHJlc3VsdHMuCldpdGggdGhlIGV4Y2VwdGlvbiBvZiB0aGUgcHJldmlvdXMgcG9pbnQsIEkgaG9wZSB0aGF0IHRoZSBmaXJzdCB3aWxsCmJlIGlkZW50aWNhbCAob3IgYXQgbGVhc3QgdmVyeSBjbG9zZSB0byBpZGVudGljYWwpIHRvIFRoZXJlc2EncyByZXN1bHQKd2hpbGUgdGhlIHNlY29uZCBJIGV4cGVjdCB3aWxsIGJlIHN1YnRseSBkaWZmZXJlbnQgLS0gYnV0IEkgYW0gaG9waW5nCnN1YnRseSBlbm91Z2ggdGhhdCBpdCB3aWxsIG5vdCBzaWduaWZpY2FudGx5IGNoYW5nZSB0aGUgaW50ZXJwcmV0YXRpb24KYnV0IGJlIGEgbGl0dGxlIG1vcmUgcHJlY2lzZS4KCkxldHMgc2VlISAgSSBuZWVkIHRoZXJlZm9yZSB0byBtYWtlIGEgY2hhbmdlIHRvIG15IG1ldGFkYXRhIGdhdGhlcmluZwpmdW5jdGlvbiB0byBpbmNsdWRlIHRoZSB1bWkgZGVkdXBsaWNhdGVkIHJlc3VsdC4gIEkgYW0gdGhpbmtpbmcKdGhlcmVmb3JlIHRvIGNyZWF0ZSBhIHNlcGFyYXRlIHNwZWNpZmljYXRpb24gZm9yIHVtaS1iYXJjb2RlZCBzYW1wbGVzCmJlY2F1c2UgbG9va2luZyB0aHJvdWdoIHRoZSBsb2dzIGZvciB1bWkgc3R1ZmYgd2hlbiB0aGV5IGFyZSBub3QgdXNlZAp3aWxsIGJlIHRvbyBtdWNoIG9mIGEgcGFpbi4uLgoKIyMgU21hbGwgcmFuZG9tIHJlbWluZGVyCgpJIGhhdmUgYSBjb3VwbGUgcGljdHVyZXMgb2YgUlBMMjIgdG8gaGVscCBtZSByZW1lbWJlciB0aGUgZXhwZXJpbWVudGFsCmRlc2lnbjoKCiogVGhlIGh1bWFuIHJpYm9zb21lIHdpdGggUlBMMjIgaW4gcmVkOiAhW2h1bWFuX3JwbDIyX3JlZF0oaHVtYW5fcmlib3NvbWVfTDIyX3JlZC5wbmcpCiogVGhlIG1vdXNlIHJpYm9zb21lIHdpdGggUlBMMjIgaW4gZ3JlZW4gYXQgdGhlIGNlbnRlcjoKIVttb3VzZV9ycGwyMl0obW11c2N1bHVzX1JQTDIyX2dyZWVuX2NlbnRlci5wbmcpCgpUaGF0IHNlY29uZCBwaWN0dXJlIGNhbWUgZnJvbTogKEBsaU1hbGVHZXJtY2VsbHNwZWNpZmljUmlib3NvbWUyMDIyKQoKIyBBIG5vdGUgYWJvdXQgaW1wbGVtZW50YXRpb24KCkkgd291bGQgbGlrZSB0byBpbXByb3ZlIHRoaXMgZG9jdW1lbnQgYnkgY29tcGFyaW5nL2NvbnRyYXN0aW5nIHRoZQptZXRob2RvbG9naWVzIHBlcmZvcm1lZCBieSBvdGhlciBncm91cHMgYW5kIHRob3NlIHBlcmZvcm1lZCBieSBtZSBpbgppdC4gIEkgbmV2ZXIgZnVsbHkgYXBwcmVjaWF0ZWQgdGhlIHN1aXRlIG9mIGNvbXB1dGF0aW9uYWwgbWV0aG9kcwphcHBsaWVkIGJ5IHByZXZpb3VzIGdyb3VwcyB3aGVuIGV4YW1pbmluZyBUUkFQIGRhdGE7IEkgaW5zdGVhZCBzaW1wbHkKZm9sbG93ZWQgVGhlcmVzYSdzIG5vdGVib29rIHdpdGhvdXQgY29uc2lkZXJpbmcgb3RoZXIgcG9zc2liaWxpdGllcy4KCkkgdGhlcmVmb3JlIHNwZW50IGEgbGl0dGxlIHRpbWUgc3RlcHBpbmcgdGhyb3VnaCBoZXIgdGhlc2lzIGFuZApwdWxsaW5nIG91dCB0aGUgcmVsZXZhbnQgcGFwZXJzIGluIHRoZSBob3BlcyBvZiBsZWFybmluZyB0aGVzZSB2YXJpb3VzCm1ldGhvZHMuICBJIHNob3VsZCB0aGVyZWZvcmUgYmUgYWJsZSBzb29uIHRvIGNvbXBhcmUvY29udHJhc3QgdGhlCnZhcmlvdXMgbWV0aG9kcyBlbXBsb3llZCBieSBvdGhlciBsYWJzIGluIGFkZGl0aW9uIHRvIGNvcHlpbmcKVGhlcmVzYSdzIGxvZ2ljLgoKIyMgVGhlIGZvbGxvd2luZyBibG9jayBjYW5ub3Qgd29yayBpbiB0aGUgY29udGFpbmVyCgpUaGUgZm9sbG93aW5nIGJsb2NrIGFzc3VtZXMgdGhlIGZ1bGwgdHJlZSBvZiBwcmVwcm9jZXNzZWQgZGF0YSB3aXRoCnRoZSBsb2dzIGZyb20gdGhlIHRyaW1tZXIsIG1hcHBpbmcsIHVtaSBkZWR1cGxpY2F0aW9uLCBjb3VudGluZywgZXRjLgpBcyBhIHJlc3VsdCBpdCBjYW5ub3Qgd29yayBpbiB0aGUgY29udGFpbmVyIHdoaWNoIGhhcyBvbmx5IHRoZSB2YXJpb3VzCmNvdW50IHRhYmxlcy4KCkFzIGEgcmVzdWx0LCBJIGFtIGluY2x1ZGluZyBhIGNvcHkgb2YgdGhpcyBzaGVldCBhZnRlciBydW5uaW5nIHRoZQpmb2xsb3dpbmcgYmxvY2sgaW4gbXkgd29ya2luZyB0cmVlLiAgSSBzdXBwb3NlIGZvciB0aGUgbW9tZW50IHlvdSB3aWxsCmhhdmUgdG8gdHJ1c3QgdGhhdCBpdCB3b3JrZWQuICAoZm9yIHJpZ2h0IG5vdywgd2hlbiB0ZXN0aW5nIG91dCB0aGlzCmNvbnRhaW5lciwgSSBhbSBqdXN0IHNlbmRpbmcgdGhlIFIgd29ya2luZyBkaXJlY3RvcnkgdG8gbXkgdHJlZSBmb3IKdGhpcyBibG9jaywgdGhlbiBtb3ZpbmcgaXQgYmFjay4KCkkgd2lsbCBuZWVkIHRvIG1hbnVhbGx5IGVkaXQgb25lIGNvbHVtbiB0aG91Z2gsIHRoZSBzeW1saW5rIGNvbHVtbgpmcm9tIFRoZXJlc2EgaGFzIGEgc2VyaWVzIG9mIHBhdGhzIHdoaWNoIGRvIG5vdCB3b3JrIGluIHRoZSBjb250YWluZXIuCgoKYGBge3IsIGV2YWw9RkFMU0V9CnVtaV9zcGVjIDwtIG1ha2Vfcm5hc2VxX3NwZWModW1pID0gVFJVRSkKaXByZ2NfMjAyMl9tZXRhIDwtIGdhdGhlcl9wcmVwcm9jZXNzaW5nX21ldGFkYXRhKCJzYW1wbGVfc2hlZXRzLzIwMjQwNjA2X29ubHlfdW1kX3NlcXVlbmNlZC54bHN4IiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNwZWMgPSB1bWlfc3BlYywgc3BlY2llcyA9ICJtbTM5XzExMiIsIHZlcmJvc2UgPSBGQUxTRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJhc2VkaXIgPSAicHJlcHJvY2Vzc2luZy91bWRfc2VxdWVuY2VkIikKY29sbmFtZXMoaXByZ2NfMjAyMl9tZXRhW1sibmV3X21ldGEiXV0pCmhlYWQoaXByZ2NfMjAyMl9tZXRhW1sibmV3X21ldGEiXV0pCmBgYAoKYGBge3J9CnNhbXBsZV9zaGVldCA8LSAic2FtcGxlX3NoZWV0cy8yMDI0MDYwNl9vbmx5X3VtZF9zZXF1ZW5jZWRfbW9kaWZpZWQueGxzeCIKbXNpZ2RiIDwtICJyZWZlcmVuY2UvbXNpZ2RiX3YyMDI0LjEuTW0uZGIiCm1zaWdfZGF0YSA8LSBOVUxMCm1ha2VfdHJhbnNwYXJlbnQgPC0gZnVuY3Rpb24oKSB7CiAgZ2dwbG90Mjo6dGhlbWUoCiAgICBwYW5lbC5iYWNrZ3JvdW5kID0gZWxlbWVudF9yZWN0KGZpbGwgPSAndHJhbnNwYXJlbnQnKSwKICAgIHBsb3QuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsID0gJ3RyYW5zcGFyZW50JywgY29sb3IgPSBOQSksCiAgICBwYW5lbC5ncmlkLm1ham9yID0gZWxlbWVudF9ibGFuaygpLAogICAgcGFuZWwuZ3JpZC5taW5vciA9IGVsZW1lbnRfYmxhbmsoKSwKICAgIGxlZ2VuZC5iYWNrZ3JvdW5kID0gZWxlbWVudF9yZWN0KGZpbGwgPSAndHJhbnNwYXJlbnQnKSwKICAgIGxlZ2VuZC5ib3guYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsID0gJ3RyYW5zcGFyZW50JykpCn0KYGBgCgpJIHdpbGwgZmlndXJlIG91dCBpZiBJIGNhbiBsZWF2ZSBtU2lnREIgTTIgaW4gdGhpcyBpbWFnZTsgaWYgbm90LCB0aGVuCmFueSBhbmFseXNlcyBkZXBlbmRpbmcgb24gdGhvc2UgZ2VuZSBzZXRzIHdpbGwgZmFpbCBhIHByaW9yaS4KCmBgYHtyfQptMl9nc2MgPC0gdHJ5KGxvYWRfZ210X3NpZ25hdHVyZXMoc2lnbmF0dXJlcyA9IG1zaWdkYiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpZ25hdHVyZV9jYXRlZ29yeSA9ICJNMiIpLCBzaWxlbnQgPSBUUlVFKQojIyBJIGRvIG5vdCB0aGluayBJIGhhdmUgcGVybWlzc2lvbiB0byBsb2FkIHRoZSBtc2lnZGIgaW4gdGhlIGNvbnRhaW5lcgojIyBTbywgaWYgdGhpcyBmYWlscywganVzdCBsb2FkIGl0IGZyb20gR1NWQURhdGEsIG9oIHdhaXQgbm8sIEdTVkFkYXRhIGlzIGh1bWFuLgppZiAoInRyeS1lcnJvciIgJWluJSBjbGFzcyhtMl9nc2MpKSB7CiAgd2FybmluZygiVW5hYmxlIHRvIGxvYWQgdGhlIE0yIE1zaWdEQiBkYXRhLiIpCn0KYGBgCgpGcm9tIHRoaXMgcG9pbnQgb24sIEkgYW0gaG9waW5nL2ludGVuZGluZyB0byBwdWxsIGxpYmVyYWxseSBmcm9tClRoZXJlc2EncyBub3RlYm9vayB3aXRoIGEgZGl2ZXJzaW9uIHRvIGNvbXBhcmUgdGhlIHRocmVlIGRhdGFzZXRzOgoKKiBQcmUtQXByaWwgcmVuYW1pbmc6IEUuZy4gVGhlcmVzYSdzIGN1cnJlbnQgZGF0YXNldAoqIFBvc3QgcmVuYW1pbmc6IFVubGVzcyBJIGFtIG1pc3Rha2VuLCB0aGlzIHNob3VsZCBiZSB2ZXJ5IHNpbWlsYXIgdG8KICB0aGUgYWJvdmUuCiogUG9zdCBkZWR1cGxpY2F0aW9uOiBHaXZlbiB3aGF0IEkgc2F3IGZyb20gdGhlIGV4dHJhY3RlZCBsb2dzIGluIHRoZQogIHNhbXBsZSBzaGVldCwgSSBleHBlY3QgdGhpcyB0byBiZSBzaW1pbGFyIGJ1dCBub3QgaWRlbnRpY2FsIHRvIHRoZQogIHByZXZpb3VzIHR3by4KCkxldHMgZmluZCBvdXQhICAgQnV0IGZpcnN0LCBhbm5vdGF0aW9ucyEKCiMgQW5ub3RhdGlvbiBkYXRhCgpJIGFtIHB1bGxpbmcgdGhpcyBmcm9tIFRoZXJlc2EncyBhbnhvbnRyYXBSX3BpcGVsaW5lLlJtZCwgcHJpbWFyaWx5CmJlY2F1c2UgaXQgbG9va3Mgc2ltaWxhciB0byB0aGUgb3RoZXIgZG9jdW1lbnRzLCBidXQgd2FzIG1vZGlmaWVkIG1vcmUKcmVjZW50bHkuICBJIHdpbGwgY2hhbmdlIGl0IHNsaWdodGx5LCBwcmltYXJpbHkgYmVjYXVzZSBJIGdyYWJiZWQgYQpuZXcgbW11c2N1bHVzIGFzc2VtYmx5IGFuZCB0aGVyZWZvcmUgSSB3aWxsIHB1bGwgdGhlIG1tdXNjdWx1cwphbm5vdGF0aW9ucyBmcm9tIGEgc3BlY2lmaWMgYmlvbWFydAooQHNtZWRsZXlCaW9NYXJ0QmlvbG9naWNhbFF1ZXJpZXMyMDA5KSBhcmNoaXZlIHRoYXQgc2hvdWxkIG1hdGNoIGl0LgoKQSBub3RlIGZyb20gdGhlIGZ1dHVyZTogbXVsdGlwbGUgZW5zZW1ibCBhcmNoaXZlIHNlcnZlcnMgaGF2ZSBiZWVuCnRha2VuIG9mZmxpbmUgc2luY2UgbGFzdCBJIHJhbiB0aGlzLiAgTGV0IHVzIHNlZSBpZiBGZWIuIDIwMjMgc3RpbGwKd29ya3MuCgojIyBBbiBpbXBvcnRhbnQgbm90ZSEKCkluIHRoZSByZWNlbnQgcGFzdCwgZW5zZW1ibCBxdWVyaWVzIGhhdmUgYmVjb21lIGluY29uc2lzdGVudCwgZmFpbGluZwptdWNoIG1vcmUgb2Z0ZW4gdGhhbiBldmVyIGluIHRoZSBwYXN0LiAgSSBkbyBub3QgdGhpbmsgdGhpcyBpcyB0aGUKZmF1bHQgb2YgZW5zZW1ibDsgYnV0IEkgdGhpbmsgSSBuZWVkIGEgZmFsbGJhY2sgbWVjaGFuaXNtIGZvcgpjb2xsZWN0aW5nIGFubm90YXRpb24gaW5mb3JtYXRpb24uCgpJbiB0aGUgY2FzZSBvZiBlbnNlbWJsLCBpdCBzaG91bGQgYmUgdHJpdmlhbCAoYnV0IGxlc3MgZnVuKSB0byB1c2UgYQpjb21iaW5hdGlvbiBvZiB0aGUgbG9jYWxseSBpbnN0YWxsZWQgb3JnZGIgYW5kIHR4ZGIgZGF0YWJhc2VzLgoKVGhpcyBkb2VzIG9wZW4gYSByaXNrIHRoYXQgdGhlIHNldCBvZiBnZW5lcyB3aXRoIGFubm90YXRpb25zIHdpbGwgYmUKZGlmZmVyZW50IGRlcGVuZGluZyBvbiB3aGVuIHRoZSBjb250YWluZXIgaXMgcnVuIGR1ZSB0byBkaWZmZXJlbmNlcwpiZXR3ZWVuIHRoZSBvcmdkYi90eGRiIGluc3RhbmNlIGFuZCB0aGUgRmViIDIwMjMgYmlvbWFydC4gIEkgYW0gbm90CnN1cmUgdGhlcmUgaXMgbXVjaCBJIGNhbiBkbyBhYm91dCB0aGF0IGV4Y2VwdCB0byBidW5kbGUgdGhlIHNldCBvZgphbm5vdGF0aW9ucyBJIGRvd25sb2FkZWQgaW4gdGhlIGNvbnRhaW5lciAtLSBzaW5jZQpsb2FkX2Jpb21hcnRfYW5ub3RhdGlvbnMoKSBkb2VzIHNhdmUgYSByZGEgY29weSBvZiBpdHMgZG93bmxvYWQuCgpvaywgSSBkaWQgYm90aC4gIElmIHlvdSwgZGVhciByZWFkZXIsIHdpc2ggdG8gZG93bmxvYWQgeW91ciBvd24KYW5ub3RhdGlvbnMsIGFuZCBlbnNlbWJsIGlzIGhhdmluZyB0cm91YmxlcywgdGhlIGZvbGxvd2luZyBzaG91bGQgd29yawp3aXRob3V0IGEgcHJvYmxlbTsgaW4gYWRkaXRpb24gdGhlIHJkYSBhbm5vdGF0aW9ucyBhcmUgaW4gL2RhdGEgb2YgdGhlCmNvbnRhaW5lciBhbmQgc2hvdWxkIGdldCBsb2FkZWQuCgpgYGB7cn0KdHhfZ2VuZV9tYXAgPC0gZGF0YS5mcmFtZSgpCm1tX2Fubm90IDwtIHRyeShsb2FkX2Jpb21hcnRfYW5ub3RhdGlvbnMoc3BlY2llcyA9ICJtbXVzY3VsdXMiLCB5ZWFyID0gIjIwMjMiLCBtb250aCA9ICIwMiIpKQppZiAoInRyeS1lcnJvciIgJWluJSBjbGFzcyhtbV9hbm5vdCkpIHsKICBmaWVsZHMgPC0gYygiQUNDTlVNIiwgIkVOU0VNQkwiLCAiRU5TRU1CTFRSQU5TIiwgIkVOVEVaSUQiLCAiR0VORU5BTUUiLCAiU1lNQk9MIikKICBvcmdkYl9hbm5vdCA8LSBsb2FkX29yZ2RiX2Fubm90YXRpb25zKCJvcmcuTW0uZWcuZGIiLCBmaWVsZHMgPSBmaWVsZHMpCiAgZ2VuZV9pbmZvIDwtIG9yZ2RiX2Fubm90W1siZ2VuZXMiXV0KICAjIyBOb3RlLCB0aGVyZSBhcmUgYSBidW5jaCBvZiB2YXJpYW50cyBvZiB0aGUgdHhkYiBwYWNrYWdlIG9uZSBtaWdodCB1c2UuCiAgIyMgSSBkbyBub3QgdGhpbmsgaXQgbWF0dGVycyBhIGxvdCBmb3Igb3VyIHB1cnBvc2VzLCBidXQgSSBzdXNwZWN0IHRoYXQgaWYgd2UgdXNlZAogICMjIGEgbWlzbWF0Y2hlZCBCU2dlbm9tZSBhbmQgdHJpZWQgdG8gcHVsbCBDRFMgc2VxdWVuY2VzLCB0aGF0IG1pZ2h0IGVuZCBiYWRseS4KICBwa2cgPC0gIlR4RGIuTW11c2N1bHVzLlVDU0MubW0xMC5rbm93bkdlbmUiCiAgdHhfYW5ub3QgPC0gbG9hZF90eGRiX2Fubm90YXRpb25zKHBrZykKICB0cmFuc2NyaXB0cyA8LSB0eF9hbm5vdFtbIlRYIl1dCiAgdHJhbnNjcmlwdHNbWyJ0eCJdXSA8LSBnc3ViKHggPSB0cmFuc2NyaXB0c1tbIlRYTkFNRSJdXSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcGF0dGVybiA9ICJcXC5cXGQrJCIsIHJlcGxhY2VtZW50ID0gIiIpCiAgbW1fYW5ub3QgPC0gbWVyZ2UoZ2VuZV9pbmZvLCB0cmFuc2NyaXB0cywgYnkueCA9ICJlbnNlbWJsdHJhbnMiLCBieS55ID0gInR4IikKICByb3duYW1lcyhtbV9hbm5vdCkgPC0gbWFrZS5uYW1lcyhtbV9hbm5vdFtbImVuc2VtYmwiXV0sIHVuaXF1ZSA9IFRSVUUpCn0gZWxzZSB7CiAgbW1fYW5ub3QgPC0gbW1fYW5ub3RbWyJhbm5vdGF0aW9uIl1dCiAgbW1fYW5ub3RbWyJ0eGlkIl1dIDwtIHBhc3RlMChtbV9hbm5vdFtbImVuc2VtYmxfdHJhbnNjcmlwdF9pZCJdXSwgIi4iLCBtbV9hbm5vdFtbInZlcnNpb24iXV0pCiAgcm93bmFtZXMobW1fYW5ub3QpIDwtIG1ha2UubmFtZXMobW1fYW5ub3RbWyJlbnNlbWJsX2dlbmVfaWQiXV0sIHVuaXF1ZT1UUlVFKQogIHR4X2dlbmVfbWFwIDwtIG1tX2Fubm90WywgYygidHhpZCIsICJlbnNlbWJsX2dlbmVfaWQiKV0KfQpgYGAKCiMgSGlzYXQyIHN1bW1hcml6ZWRFeHBlcmltZW50cwoKVGhlIHByaW1hcnkgZGlmZmVyZW5jZSBiZXR3ZWVuIG15IGJsb2NrIGFuZCBUaGVyZXNhJ3MgYXJlOgoKMS4gIEkgYW0gcHVsbGluZyB0aGUgbWV0YWRhdGEgZGlyZWN0bHkgZnJvbSB0aGUKICAgIGdhdGhlcl9wcmVwcm9jZXNzaW5nX21ldGFkYXRhKCkgYWJvdmUuCjIuICBJIGFtIHVzaW5nIHRoZSBjb2x1bW4gJ3N5bWxpbmsnIHdoaWNoIGlzIGp1c3QgYSBjb3B5IG9mIHRoZQogICAgZXhpc3RpbmcgJ2ZpbGUnIGNvbHVtbiB3aXRoIGEgc21hbGwgY2hhbmdlIHNvIEkgY2FuIGxvYWQgaXQgZnJvbQogICAgbXkgZGlyZWN0b3J5IHdpdGhvdXQgaGF2aW5nIHRvIGNvcHkgZXZlcnl0aGluZy4KMy4gIEkgYW0gdXNpbmcgdGhlIGVuc2VtYmwgZ2Vub21lIHJlbGVhc2UgMzksIHZlcnNpb24gMTEyIGFuZCBzbwogICAgcHVsbGVkIGEgc29tZXdoYXQgbmV3ZXIgY29weSBvZiB0aGUgYW5ub3RhdGlvbiBkYXRhLgo0LiAgVGhlIG9yaWdpbmFsIGlzIG5hbWVkICd2MScsIGZvbGxvd2VkIGJ5ICd2MicgYW5kICd2MycgZm9yIHRoZQogICAgb3RoZXIgdHdvIHRyZWF0bWVudHMgSSBwZXJmb3JtZWQuCgojIyBDb2xvciBjaG9pY2VzIGFuZCByZXVzZWQgcGFyYW1ldGVycwoKR2l2ZW4gdGhhdCB3ZSBhcmUgZXhjbHVkaW5nIGEgYnVuY2ggb2YgdGhlIG9sZGVyIHNhbXBsZXMsIHRoZSBzZXQgb2YKY29sb3JzIEkgZXhwZWN0IHRvIGZpbmQgaXMgZGlmZmVyZW50OyBzbyBJIHdpbGwgbWFrZSBleHBsaWNpdCBoZXJlIHRoZQp2YXJpb3VzIGNvbG9ycyB1c2VkIHRvIGRlbm90ZSBsb2NhdGlvbi9nZW5vdHlwZS90aW1lL2V0Yy4KCkFwcmlsIHR1cm5lZCBtZSBvbnRvIHRoaXMgd2Vic2l0ZSAncGFsZXR0b24uY29tJyBmb3IgdGhpcyBraW5kIG9mCnN0dWZmIGFuZCBJIHdpbGwgdHJ5IGFuZCBwaWNrIG91dCBwYWxldHRlcyB3aGljaCBiYXNpY2FsbHkgbWF0Y2ggd2hhdApJIGFtIGdldHRpbmcgd2l0aCB0aGUgb3JpZ2luYWwgY29sb3JzLgoKYGBge3J9CmNvbG9yX2Nob2ljZXMgPC0gbGlzdCgKICAiYWxsIiA9IGxpc3QoCiAgICAicDA4X2hldF9kbGduIiA9ICIjRTcyOThBIiwKICAgICJwMTVfaGV0X2RsZ24iID0gIiNFNzI5OEEiLAogICAgInAwOF9oZXRfcmV0aW5hIiA9ICIjMjM4QjQ1IiwKICAgICJwMTVfaGV0X3JldGluYSIgPSAiIzIzOEI0NSIsCiAgICAicDA4X2hldF9zY24iID0gIiM0MjkyQzYiLAogICAgInAxNV9oZXRfc2NuIiA9ICIjNDI5MkM2IiwKICAgICJwMDhfa29fZGxnbiIgPSAiI0M5OTRDNyIsCiAgICAicDE1X2tvX2RsZ24iID0gIiNDOTk0QzciLAogICAgInAwOF9rb19yZXRpbmEiID0gIiM3NGM0NzYiLAogICAgInAxNV9rb19yZXRpbmEiID0gIiM3NGM0NzYiLAogICAgInAwOF9rb19zY24iID0gIiM5QkNBRTEiLAogICAgInAxNV9rb19zY24iID0gIiM5QkNBRTEiLAogICAgInAwOF93dF9kbGduIiA9ICIjOTgwMDQzIiwKICAgICJwMTVfd3RfZGxnbiIgPSAiIzk4MDA0MyIsCiAgICAicDA4X3d0X3JldGluYSIgPSAiIzAwNDAwOCIsCiAgICAicDE1X3d0X3JldGluYSIgPSAiIzAwNDAwOCIsCiAgICAicDA4X3d0X3NjbiIgPSAiIzA4NTE5QyIsCiAgICAicDE1X3d0X3NjbiIgPSAiIzA4NTE5QyIsCiAgICAicDYwX3d0X2RsZ24iID0gIiMzMzMzMzMiLAogICAgInA2MF93dF9yZXRpbmEiID0gIiMyMjIyMjIiLAogICAgInA2MF93dF9zY24iID0gIiMxMTExMTEiKSwKICAiZ2Vub19sb2MiID0gbGlzdCgKICAgICJoZXRfZGxnbiIgPSAiI0U3Mjk4QSIsCiAgICAiaGV0X3JldGluYSIgPSAiIzIzOEI0NSIsCiAgICAiaGV0X3NjbiIgPSAiIzQyOTJDNiIsCiAgICAia29fZGxnbiIgPSAiI0M5OTRDNyIsCiAgICAia29fcmV0aW5hIiA9ICIjNzRjNDc2IiwKICAgICJrb19zY24iID0gIiM5QkNBRTEiLAogICAgInd0X2RsZ24iID0gIiM5ODAwNDMiLAogICAgInd0X3JldGluYSIgPSAiIzAwNDAwOCIsCiAgICAid3Rfc2NuIiA9ICIjMDg1MTlDIiksCiAgImxvY2F0aW9uIiA9IGxpc3QoCiAgICAicmV0aW5hIiA9ICIjMDA0MDA4IiwKICAgICJkbGduIiA9ICIjOTgwMDQzIiwKICAgICJzY24iID0gIiMwODUxOUMiKSwKICAiZ2Vub3R5cGUiID0gbGlzdCgKICAgICJ3dCIgPSAiIzc0YzQ3NiIsCiAgICAiaGV0IiA9ICIjMjM4QjQ1IiwKICAgICJrbyIgPSAiIzAwNkQyQyIpLAogICJ0aW1lIiA9IGxpc3QoCiAgICAicDA4IiA9ICIjNUUxMDRCIiwKICAgICJwMTUiID0gIiM0RTkyMzEiKSkKbGFiZWxfY29sdW1uIDwtICJtZ2lfc3ltYm9sIiAjIyBTZXQgdGhlIGNvbHVtbiB1c2VkIHRvIGV4dHJhY3QgZ2VuZSBzeW1ib2xzIHJhdGhlciB0aGFuIEVOU0cuLi4uLgpjb2xvcnMgPC0gY29sb3JfY2hvaWNlc1tbImdlbm9fbG9jIl1dCnRpbWVfY29sb3JzIDwtIGxpc3QoCiAgInAwOF9oZXRfZGxnbiIgPSAiI0U3Mjk4QSIsCiAgInAxNV9oZXRfZGxnbiIgPSAiIzhhMTg1MiIsCiAgInAwOF9oZXRfcmV0aW5hIiA9ICIjMjM4QjQ1IiwKICAicDE1X2hldF9yZXRpbmEiID0gIiMxNTUzMjkiLAogICJwMDhfaGV0X3NjbiIgPSAiIzQyOTJDNiIsCiAgInAxNV9oZXRfc2NuIiA9ICIjMjc1Nzc2IiwKICAicDA4X2tvX2RsZ24iID0gIiNDOTk0QzciLAogICJwMTVfa29fZGxnbiIgPSAiIzc4NTg3NyIsCiAgInAwOF9rb19yZXRpbmEiID0gIiM3NEM0NzYiLAogICJwMTVfa29fcmV0aW5hIiA9ICIjNDU3NTQ2IiwKICAicDA4X2tvX3NjbiIgPSAiIzlCQ0FFMSIsCiAgInAxNV9rb19zY24iID0gIiM1ZDc5ODciKQpgYGAKClRoZXJlIGlzIG9uZSBub3Rld29ydGh5IHNhbXBsZTogaXByZ2NfMTAzLCBpdCB3YXMgZWZmZWN0aXZlbHkgcmVwbGFjZWQKd2hlbiBBcHJpbCByZW5hbWVkIHRoZSBzYW1wbGVzIGFuZCBzbyBleGlzdHMgaW4gdGhlIHYxIGRhdGEsIGJ1dCBub3QKdjIvdjM7IHRoZXkgaW5zdGVhZCBoYXZlIHRoZSBuZXdseSBuYW1lZCBzYW1wbGVzIHdoaWNoIEkgY2FsbGVkCmlwcmdjXzEyMyB0byBpcHJnY18xMzAuICBBcyBhIHJlc3VsdCwgSSBjb3BpZWQgdGhlIGFubm90YXRpb25zIGZvcgppcHJnY18xMjMgdG8gbXkgY29sdW1uIHNvIHRoYXQgdGhlcmUgaXMgbm8gZGlzY3JlcGVuY3kgaW4gdGVybXMgb2YKZ2Vub3R5cGUvbG9jYXRpb24vdGltZS4KCiMjIFRoZSBvcmlnaW5hbCBjb3VudCB0YWJsZXMKCkF0IHRoZSBtb21lbnQgSSBoYXZlIG5vdCBpbmNsdWRlZCB0aGUgb3JpZ2luYWwgY291bnRzIGluIHRoaXMKY29udGFpbmVyIGJlY2F1c2Ugd2UgbWFkZSBzb21lIGNoYW5nZXMgdG8gdGhlIG1hcHBpbmcgc3RyYXRlZ3kgYW5kCmFsc28gZm91bmQgdGhhdCBhIGNvdXBsZSBzYW1wbGVzIHdlcmUgbWl4ZWQgdXAgaW4gc2VxdWVuY2luZzsgYXMgYQpyZXN1bHQgSSBkb2N1bWVudGVkIGFsbCBvZiB0aGUgY2hhbmdlcyBpbiB0aGUgc2FtcGxlIHNoZWV0cyBhbmQKcHJlcHJvY2Vzc2luZyBkb2N1bWVudHMgYW5kIGV4Y2x1ZGVkIHRoZSBvcmlnaW5hbCBmaWxlcy4KClRoaXMgaXMgYWxzbyB3aHkgc29tZSBjb2x1bW5zIGluIHRoZSBzYW1wbGUgc2hlZXQgaGF2ZSBzdWZmaXhlcyBsaWtlCidhZGgnIGFuZCAnYXRiJywgdGhvc2UgZGVub3RlIGZyb20gd2hvbSB0aGUgcmVsZXZhbnQgbWV0YWRhdGEgY29sdW1ucwpjYW1lIGZyb20uCgpgYGB7ciwgZXZhbD1GQUxTRX0KbW0zOF9oaXNhdF92MSA8LSBjcmVhdGVfc2Uoc2FtcGxlX3NoZWV0LAogICAgICAgICAgICAgICAgICAgICAgICAgICBnZW5lX2luZm8gPSBtbV9hbm5vdCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgZmlsZV9jb2x1bW4gPSAic3ltbGluayIpICU+JQogIHNldF9jb25kaXRpb25zKGZhY3QgPSAiZ2Vub19sb2NfYXRiIikgJT4lCiAgc2V0X2JhdGNoZXMoZmFjdCA9ICJ0aW1lX2F0YiIpICU+JQogIHNldF9jb2xvcnMoY29sb3JfY2hvaWNlc1tbImdlbm9fbG9jIl1dKQptbTM4X2hpc2F0X3YxCmBgYAoKIyMgUmVjb3VudGVkIHRhYmxlcyBhbmQgdGhlIGRlZHVwbGljYXRlZCByZXN1bHQKCkluIHRoZSBmb2xsb3dpbmcgSSBtYWtlIHR3byBtb3JlIHZlcnNpb25zIG9mIHRoZSBkYXRhLCBvbmUgcmVtYXBwZWQKd2l0aCB0aGUgY2hhbmdlcyB0byB0aGUgc2FtcGxlIGlkZW50aXRpZXMsIGFuZCBvbmUgd2l0aCBkZWR1cGxpY2F0aW9uCmFwcGxpZWQuCgpgYGB7cn0KbW0zOF9oaXNhdF92MiA8LSBjcmVhdGVfc2Uoc2FtcGxlX3NoZWV0LCBnZW5lX2luZm8gPSBtbV9hbm5vdCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgZmlsZV9jb2x1bW4gPSAiaGlzYXRfY291bnRfdGFibGUiKSAlPiUKICBzZXRfY29uZGl0aW9ucyhmYWN0ID0gImdlbm9fbG9jX2F0YiIpICU+JQogIHNldF9iYXRjaGVzKGZhY3QgPSAidGltZV9hdGIiKSAlPiUKICBzZXRfY29sb3JzKGNvbG9yX2Nob2ljZXNbWyJnZW5vX2xvYyJdXSkKbW0zOF9oaXNhdF92MgptbTM4X2hpc2F0X3YzIDwtIGNyZWF0ZV9zZShzYW1wbGVfc2hlZXQsIGdlbmVfaW5mbyA9IG1tX2Fubm90LAogICAgICAgICAgICAgICAgICAgICAgICAgICBmaWxlX2NvbHVtbiA9ICJ1bWlfZGVkdXBfb3V0cHV0X2NvdW50IikgJT4lCiAgc2V0X2NvbmRpdGlvbnMoZmFjdCA9ICJnZW5vX2xvY19hdGIiKSAlPiUKICBzZXRfYmF0Y2hlcyhmYWN0ID0gInRpbWVfYXRiIikgJT4lCiAgc2V0X2NvbG9ycyhjb2xvcl9jaG9pY2VzW1siZ2Vub19sb2MiXV0pCm1tMzhfaGlzYXRfdjMKCmFsbF9mYWN0IDwtIHBhc3RlMChjb2xEYXRhKG1tMzhfaGlzYXRfdjMpW1sidGltZV9hdGIiXV0sICJfIiwKICAgICAgICAgICAgICAgICAgIGNvbERhdGEobW0zOF9oaXNhdF92MylbWyJnZW5vX2xvY19hdGIiXV0pCmNvbERhdGEobW0zOF9oaXNhdF92MylbWyJ0aW1lX2dlbm9fbG9jIl1dIDwtIGFsbF9mYWN0CmBgYAoKTm90ZSB0aGUgZW5kIG9mIHRoZSBwcmV2aW91cyBibG9jaywgSSBjcmVhdGVkIGEgZmFjdG9yIG91dCBvZiB0aGUKY29tYmluYXRpb24gb2YgdGltZSwgZ2Vub3R5cGUsIGFuZCBsb2NhdGlvbi4gIEluIGEgZnV0dXJlIGludm9jYXRpb24Kb2YgdGhpcyBub3RlYm9vaywgSSB3aWxsIGNoYW5nZSB0aGUgcGFpcndpc2UgY29tcGFyaXNvbnMgdG8gYWRkIGVhY2gKb2YgdGhlc2UgdGhyZWUgZmFjdG9ycyB0byB0aGUgc3RhdGlzdGljYWwgbW9kZWwgaW5zdGVhZCBvZiB0aGlzLiAgVGhlCmNvZGUgdG8gZG8gdGhhdCBpcyBub3QgX3F1aXRlXyByZWFkeSB5ZXQuCgojIE5vbi16ZXJvIENvdW50cyBwZXIgU2FtcGxlCgpMZXQncyBsb29rIGF0IHRoZSBudW1iZXIgb2Ygbm9uLXplcm8gZ2VuZXMgZm9yIGFsbCBzYW1wbGVzIHZlcnN1cyB0aGUKY292ZXJhZ2UuCgpBcyBhYm92ZSwgdGhpcyBkb2VzIG5vdCBnZXQgcnVuIGJlY2F1c2UgSSBkaWQgbm90IGNvcHkgdGhlIGNvdW50IHRhYmxlcy4KCmBgYHtyIGV2YWw9RkFMU0V9CnYxX25vbnplcm8gPC0gcGxvdF9ub256ZXJvKG1tMzhfaGlzYXRfdjEpCnYxX25vbnplcm8KYGBgCgpCdXQgdGhlc2UgZG8hCgpgYGB7cn0KcGxvdF9sZWdlbmQobW0zOF9oaXNhdF92MikKdjJfbm9uemVybyAgPC0gcGxvdF9ub256ZXJvKG1tMzhfaGlzYXRfdjIsIHlfaW50ZXJjZXB0ID0gMC42NSkKdjJfbm9uemVybwpwcChmaWxlID0gIjAxZGlhZ25vc3RpY19pbWFnZXMvbm9uemVyb192Ml91bmZpbHRlcmVkLnBkZiIpCnYyX25vbnplcm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCgp2M19ub256ZXJvICA8LSBwbG90X25vbnplcm8obW0zOF9oaXNhdF92MywgeV9pbnRlcmNlcHQgPSAwLjY1KQp2M19ub256ZXJvCnBwKGZpbGUgPSAiMDFkaWFnbm9zdGljX2ltYWdlcy9ub256ZXJvX3YzX3VuZmlsdGVyZWQucGRmIikKdjNfbm9uemVyb1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgpPaCB3b3csIEkgZGlkIG5vdCBleHBlY3Qgc3VjaCBhIHByb2ZvdW5kIGVmZmVjdCBvbiB0aGUgY3BtIHZhbHVlcyBvbgp0aGUgbW9yZSBzYXR1cmF0ZWQgbGlicmFyaWVzLiAgSSBndWVzcyBpbiByZXRyb3NwZWN0IEkgc2hvdWxkIGhhdmU/CgpBbHNvIG5vdGUgdG8gc2VsZiwgd2UgYXJlIG5vdCBtZXNzaW5nIHdpdGggcDYwLgoKIyMgRXhjbHVkZSBwNjAKCmBgYHtyfQptbTM4X2hpc2F0X3YyIDwtIHN1YnNldF9zZShtbTM4X2hpc2F0X3YyLCBzdWJzZXQgPSAidGltZV9hdGIhPSdwNjAnIikKbW0zOF9oaXNhdF92MyA8LSBzdWJzZXRfc2UobW0zOF9oaXNhdF92Mywgc3Vic2V0ID0gInRpbWVfYXRiIT0ncDYwJyIpCmBgYAoKIyMgUmVwbG90IHRoZSBub256ZXJvIGdlbmUgcGxvdHMKCmBgYHtyfQp2Ml9ub256ZXJvX2ZpbHQgPC0gcGxvdF9ub256ZXJvKG1tMzhfaGlzYXRfdjIsIHBsb3RfbGFiZWxzID0gRkFMU0UpCnBwKGZpbGUgPSAiMDFkaWFnbm9zdGljX2ltYWdlcy9ub256ZXJvX3YyX2ZpbHQucGRmIikKdjJfbm9uemVyb19maWx0W1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQoKdjNfbm9uemVyb19maWx0IDwtIHBsb3Rfbm9uemVybyhtbTM4X2hpc2F0X3YzLCBwbG90X2xhYmVscyA9IEZBTFNFKQpwcChmaWxlID0gIjAxZGlhZ25vc3RpY19pbWFnZXMvbm9uemVyb192M19maWx0LnBkZiIpCnYzX25vbnplcm9fZmlsdFtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgpPbmNlIGFnYWluLCBJIGRvIG5vdCB3YW50IHRvIGxvc2UgdGhlIHByZXZpb3VzIGNvZGUsIHNvIGhlcmUgaXMgdGhlIHYxIGludm9jYXRpb24KCmBgYHtyLCBldmFsPUZBTFNFfQptbTM4X2hpc2F0X3YxIDwtIHN1YnNldF9zZShtbTM4X2hpc2F0X3YxLCBzdWJzZXQgPSAidGltZV9hdGIhPSdwNjAnIikKYGBgCgojIFF1aWNrIFBDQSwgdGhlbiByZXR1cm4gdG8gVGhlcmVzYSdzIGRvY3VtZW50CgpgYGB7cn0KdjJfbm9ybSA8LSBub3JtYWxpemUobW0zOF9oaXNhdF92MiwgdHJhbnNmb3JtID0gImxvZzIiLCBjb252ZXJ0ID0gImNwbSIsCiAgICAgICAgICAgICAgICAgICAgIG5vcm0gPSAicXVhbnQiLCBmaWx0ZXIgPSBUUlVFKQp2Ml9ub3JtX3BjYSA8LSBwbG90X3BjYSh2Ml9ub3JtKQp2Ml9ub3JtX3BjYQpwcChmaWxlID0gIjAxZGlhZ25vc3RpY19pbWFnZXMvdjJfbm9ybV9wY2EucGRmIikKdjJfbm9ybV9wY2FbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCgp2M19ub3JtIDwtIG5vcm1hbGl6ZShtbTM4X2hpc2F0X3YzLCB0cmFuc2Zvcm0gPSAibG9nMiIsIGNvbnZlcnQgPSAiY3BtIiwKICAgICAgICAgICAgICAgICAgICAgbm9ybSA9ICJxdWFudCIsIGZpbHRlciA9IFRSVUUpCnYzX25vcm1fcGNhIDwtIHBsb3RfcGNhKHYzX25vcm0pCnYzX25vcm1fcGNhCnBwKGZpbGUgPSAiMDFkaWFnbm9zdGljX2ltYWdlcy92M19ub3JtX3BjYS5wZGYiKQp2M19ub3JtX3BjYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgpJYmlkLgoKYGBge3IsIGV2YWw9RkFMU0V9CnYxX25vcm0gPC0gbm9ybWFsaXplKG1tMzhfaGlzYXRfdjEsIHRyYW5zZm9ybSA9ICJsb2cyIiwgY29udmVydCA9ICJjcG0iLAogICAgICAgICAgICAgICAgICAgICBub3JtID0gInF1YW50IiwgZmlsdGVyID0gVFJVRSkKcGxvdF9wY2EodjFfbm9ybSkKYGBgCgpUbyBteSBleWVzIGl0IGxvb2tzIGxpa2Ugd2UganVzdCBoYXZlIDEgd2VpcmRvIHAxNSBzYW1wbGU/CkRlZHVwbGljYXRpb24gaGFkIGEgbWlub3IgYnV0IHNpZ25pZmljYW50IGVmZmVjdCBvbiB0aGUgUENBLgoKV2l0aCB0aGF0IGluIG1pbmQsIGxldCB1cyBsb29rIGF0IFRoZXJlc2EncyBXT1JLSU5HIGRvY3VtZW50IGFuZCBzZWUKd2hhdCB3ZSBjYW4gcmVjYXBpdHVsYXRlLgoKVGhlcmVzYSdzIGRvY3VtZW50OiBUaGUgVFJBUCBwcm90b2NvbCBoYXMgc29tZSB2YXJpYWJpbGl0eSB3aGljaCBpcwppbnRyb2R1Y2VkIGF0IGRpZmZlcmVudCBzdHBkZiBpbmNsdWRpbmcgaG9tb2dlbml6YXRpb24sIGFudGlib2R5CmxhYmVsaW5nLCBwdWxsZG93biBlZmZpY2llbmN5L3NwZWNpZmljaXR5LCBzYW1wbGUgaGFuZGxpbmcgZHVyaW5nCmNsZWFudXAsIGFuZCBsaWJyYXJ5IHByZXAvc2VxdWVuY2luZy4gV2Uga25vdyBmcm9tIFJhc2htaSdzIFFDCnRoYXQgdGhlcmUgaXMgdmFyaWFiaWxpdHkgYXQgdGhlIGxldmVsIG9mIHB1bGxkb3duIGVmZmljaWVuY3kgKGFtb3VudApvZiBSTkEgaXNvbGF0ZWQpLiBTaGUgaXMgZG9pbmcgYSBnb29kIGpvYiBvZiBrZWVwaW5nIHRyYWNrIG9mIHRoaXMgZm9yCmFsbCBoZXIgc2FtcGxlcyBhbmQgd2UgaGF2ZSB2YWxpZGF0ZWQgaGVyIFA4IHJlc3VsdHMgKGF0dGFjaGVkCnN1cHBsZW1lbnRhcnkgZmlndXJlIDNEKS4gV2UgY29uc2lzdGVudGx5IHNlZSBjbGVhciBkaWZmZXJlbmNlcwpiZXR3ZWVuIGNvbnRyb2wgYW5kIGNyZSBzYW1wbGVzIGZvciB0aGUgcmV0aW5hLCB3aGljaCBtYWtlcyBzZW5zZQpiZWNhdXNlIHRoZSBjZWxsIGJvZGllcyBhcmUgaW4gdGhlIHJldGluYS4gVGhlIHRhcmdldCB0aXNzdWUKZGlmZmVyZW5jZXMgYXJlIHNtYWxsZXIsIHdoaWNoIGFsc28gbWFrZXMgc2Vuc2UgZm9yIGF4b24tVFJBUC4gV2UKdGhpbmsgdGhhdCBzb21lIG9mIGhlciBQMTUgc2FtcGxlcyBhcmUgbm90IGdvb2QgYmFzZWQgb24gbG93IGFtb3VudHMKb2YgaXNvbGF0ZWQgUk5BIGZyb20gY3JlKCspIHJldGluYSBzYW1wbGVzLiBXZSBwbGFuIHRvIGRyb3AgdGhlc2UKc2FtcGxlcyBhbmQgbm90IHBlcmZvcm0gYWRkaXRpb25hbCBpc29sYXRpb25zIGF0IHRoaXMgdGltZQpwb2ludC4gQmFzZWQgb24gdGhpcyAoYW5kIHRoZSBnZW5lcmFsIGxhY2sgb2YgbGFyZ2UgZGV2ZWxvcG1lbnRhbAplZmZlY3RzKSwgd2Ugd2VyZSBwbGFubmluZyB0byBmb2N1cyBvbiBwcmVzZW50aW5nIHRoZSBQOCBkYXRhIG9ubHkgaW4KdGhlIHBhcGVyLiBJbnRlcmVzdGVkIHRvIGhlYXIgeW91ciB0aG91Z2h0cyBpbiB0aGlzLi4uCgpNeSBub3RlczogVGhlcmVzYSdzIGZpcnN0IG9wZXJhdGlvbnMgaW4gdGhpcyBub3RlYm9vayB3ZXJlIHRvOgoKMS4gIFNldCBsb2NhdGlvbiBhcyBjb25kaXRpb24sIGdlbm90eXBlIGFzIGJhdGNoLgoyLiAgUGVyZm9ybSBQQ0EgYmVmb3JlL2FmdGVyIHN2YS4KCmBgYHtyfQp2M19sb2NfZ2VubyA8LSBzZXRfY29uZGl0aW9ucyhtbTM4X2hpc2F0X3YzLCBmYWN0ID0gImxvY2F0aW9uX2F0YiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG9ycyA9IGNvbG9yX2Nob2ljZXNbWyJsb2NhdGlvbiJdXSkgJT4lCiAgc2V0X2JhdGNoZXMoZmFjdCA9ICJnZW5vdHlwZV9hdGIiKQpgYGAKCiMjIFRoZSBhc3NvY2lhdGVkIFBDQQoKQXQgZGlmZmVyZW50IHRpbWVzLCBpdCBhcHBlYXJzIHRvIG1lIHRoYXQgVGhlcmVzYSBoYXMgcHJlZmVycmVkCnNsaWdodGx5IGRpZmZlcmVudCBub3JtYWxpemF0aW9uIG1ldGhvZHMsIHByaW1hcmlseSBhIG1peCBvZiBUTU0gYW5kCnF1YW50aWxlLgoKVGh1cyBJIHdpbGwgdXNlIGRpZmZlcmVudCBzdWZmaXggbGV0dGVycyB0byBkZW5vdGUgdmFyaW91cwpub3JtYWxpemF0aW9ucyBlbXBsb3llZCwgYW5kIGlmIHRoZXkgdHVybiBvdXQgdGhlIHNhbWUgSSB3aWxsIHBpY2sgb25lIGFyYml0cmFyaWx5LgoKYGBge3J9CmxvY19nZW5vX25xIDwtIG5vcm1hbGl6ZSh2M19sb2NfZ2VubywgdHJhbnNmb3JtID0gImxvZzIiLCBjb252ZXJ0ID0gImNwbSIsCiAgICAgICAgICAgICAgICAgICAgICAgICBmaWx0ZXIgPSBUUlVFLCBub3JtID0gInF1YW50IikKbG9jYXRpb25fZ2Vub3R5cGVfcGNhIDwtIHBsb3RfcGNhKGxvY19nZW5vX25xKQpwcChmaWxlID0gIjAxZGlhZ25vc3RpY19pbWFnZXMvbG9jYXRpb25fZ2Vub3R5cGVfbm9ybV9wY2EucGRmIikKbG9jYXRpb25fZ2Vub3R5cGVfcGNhW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpsb2NhdGlvbl9nZW5vdHlwZV9wY2EKIyMgb2ssIEkgaGF2ZSB0d28gd2VpcmRvIHNhbXBsZXMgd2hpY2ggbG9vayB2ZXJ5IG11Y2ggbGlrZSB0aGV5IGFyZSBhY3R1YWxseSBkbGduLgojIyBUaGVzZSBhcmUgc2FtcGxlIElEcyBpcHJnY182NiBhbmQgaXByZ2NfMTMwCgpsb2NfZ2Vub19udCA8LSBub3JtYWxpemUodjNfbG9jX2dlbm8sIHRyYW5zZm9ybSA9ICJsb2cyIiwgY29udmVydCA9ICJjcG0iLAogICAgICAgICAgICAgICAgICAgICAgICAgZmlsdGVyID0gVFJVRSwgbm9ybSA9ICJ0bW0iKQpsb2NhdGlvbl9nZW5vdHlwZV90bW1fcGNhIDwtIHBsb3RfcGNhKGxvY19nZW5vX250KQpwcChmaWxlID0gIjAxZGlhZ25vc3RpY19pbWFnZXMvbG9jYXRpb25fZ2Vub3R5cGVfdG1tX3BjYS5wZGYiKQpsb2NhdGlvbl9nZW5vdHlwZV90bW1fcGNhW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpsb2NhdGlvbl9nZW5vdHlwZV90bW1fcGNhCmBgYAoKQSByYW5kb20gdGhvdWdodCBhYm91dCB0aGVzZSBQQ0EgcGxvdHMsIGl0IG1pZ2h0IGJlIHdvcnRoIHdoaWxlIHRvIGFkZAphIHBhbmVsIGJlbG93IHRoZSBsZWdlbmQgd2l0aCB0aGUgc2FtcGxlIG51bWJlcnMgcGVyIGNvbmRpdGlvbi9iYXRjaC4KCk9mIGNvdXJzZSwgdGhlIHNhbWUgaW5mb3JtYXRpb24gaXMgcHJvdmlkZWQgaW4gYSBtb3JlIGZ1biBmYXNoaW9uIHZpYQpteSBzaWxseSBzYW5rZXkgZnVuY3Rpb246CgpgYGB7cn0Kc2FtcGxlX3NhbmtleSA8LSBwbG90X21ldGFfc2Fua2V5KHYzX2xvY19nZW5vLCBjb2xvcl9jaG9pY2VzID0gY29sb3JfY2hvaWNlcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZhY3RvcnMgPSBjKCJnZW5vdHlwZV9hdGIiLCAibG9jYXRpb25fYXRiIiwgInRpbWVfYXRiIikpCnBwKGZpbGUgPSAiMDFkaWFnbm9zdGljX2ltYWdlcy9kZXNpZ25fc2Fua2V5LnBkZiIpCnNhbXBsZV9zYW5rZXlbWyJnZ3Bsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKc2FtcGxlX3NhbmtleQpgYGAKCiMgQSBTaG9ydCBjb252ZXJzYXRpb24gd2l0aCBSYXNobWkKClJhc2htaSBjYW1lIGJ5IGFuZCB3ZSBkaXNjdXNzZWQgdGhlIHNhbXBsZXMgYSBsaXR0bGUuICBTaGUgc3VnZ2VzdGVkCnRoYXQgaXMgbGlrZWx5IHRoYXQgd2Ugd2lsbCBuZWVkIHRvIGV4Y2x1ZGUgdGhlIDIwMjIwNSBzYW1wbGVzLCB0aGVzZQptYXkgYmUgaWRlbnRpZmllZCBieSBhIGZldyB3YXlzLCBtb3N0IGVhc2lseSBJIHRoaW5rIHZpYSB0aGUKJ3Byb2plY3RfYWgnIGNvbHVtbiwgdGhleSBhcmUgdGhlIDAyMV8xIHNhbXBsZXMuCgpNeSBzZW5zZSB3YXMgdGhhdCBzaGUgY29uY3VycmVkIHdpdGggbXkgaW50ZXJwcmV0YXRpb24gb2YgdGhlIHVtaQpkZWR1cGxpY2F0aW9uLCBzbyBJIHdpbGwgY29udGludWUgdXNpbmcgdGhlIGRlZHVwbGljYXRlZCByZXN1bHRzCmV4Y2x1c2l2ZWx5LCBhdCBsZWFzdCBmb3Igbm93LgoKIyBNZWxhbm9wc2luIFNhbml0eSBDaGVjawoKT25lIG9mIFRoZXJlc2EncyBmaXJzdCBjaGVja3Mgd2FzIHdpc2VseSBmb3IgbWVsYW5vcHNpbi4gIExldCB1cwpyZXBlYXQgYSB2ZXJzaW9uIG9mIHRoaXM6CgpBbiBpbXBvcnRhbnQgbm90ZTogSW5kcmFqZWV0IFBhdGlsIHJlbW92ZWQgdGhlIGdyb3VwZWRzdGF0cyBhbmQgaXRzCmFzc29jaWF0ZWQgcGxvdHRpbmcgbGlicmFyeSBmcm9tIENSQU4vZ2l0aHViL2V0Yy4gIEkgYW0gbm90IGNlcnRhaW4Kd2hhdCBoYXBwZW5lZCwgYnV0IHRoYXQgbmVjZXNzaXRhdGVzIGEgY2hhbmdlIGluIGhvdyBJIHBsb3QgdGhpcy4KCmBgYHtyfQpvcG40X2V4cHJzIDwtIGRhdGEuZnJhbWUoY29tYmluZWQgPSBjb2xEYXRhKGxvY19nZW5vX250KVtbImdlbm9fbG9jX2F0YiJdXSwKICAgICAgICAgICAgICAgICAgICAgICAgIGxvY2F0aW9uID0gY29sRGF0YShsb2NfZ2Vub19udClbWyJsb2NhdGlvbl9hdGIiXV0sCiAgICAgICAgICAgICAgICAgICAgICAgICBnZW5vdHlwZSA9IGNvbERhdGEobG9jX2dlbm9fbnQpW1siZ2Vub3R5cGVfYXRiIl1dLAogICAgICAgICAgICAgICAgICAgICAgICAgb3BuID0gYXNzYXkobG9jX2dlbm9fbnQpWyJFTlNNVVNHMDAwMDAwMjE3OTkiLCBdKQoKIyMgZ3JvdXBlZHN0YXRzOjpncm91cGVkX3N1bW1hcnkob3BuNF9leHBycywgbG9jYXRpb24sIG9wbikKIyMgb3BuNF9sb2NhdGlvbiA8LSBnZ2JldHdlZW5zdGF0cyhkYXRhID0gb3BuNF9leHBycywgeCA9IGxvY2F0aW9uLCB5ID0gb3BuKQojIyBwcChmaWxlID0gImltYWdlcy9nZ2JldHdlZW5fbG9jYXRpb24ucGRmIikKIyMgb3BuNF9sb2NhdGlvbgojIyBwbG90dGVkIDwtIGRldi5vZmYoKQojIyBvcG40X2xvY2F0aW9uCgojIyBvcG40X2dlbm90eXBlIDwtIGdnYmV0d2VlbnN0YXRzKGRhdGEgPSBvcG40X2V4cHJzLCB4ID0gZ2Vub3R5cGUsIHkgPSBvcG4pCiMjIHBwKGZpbGUgPSAiaW1hZ2VzL2dnYmV0d2Vlbl9sb2NhdGlvbi5wZGYiKQojIyBvcG40X2dlbm90eXBlCiMjIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiMjIG9wbjRfZ2Vub3R5cGUKCiMjIG9wbjRfY29tYmluZWQgPC0gZ2diZXR3ZWVuc3RhdHMoZGF0YSA9IG9wbjRfZXhwcnMsIHggPSBjb21iaW5lZCwgeSA9IG9wbikKIyMgcHAoZmlsZSA9ICJpbWFnZXMvZ2diZXR3ZWVuX2NvbWJpbmVkLnBkZiIpCiMjIG9wbjRfY29tYmluZWQKIyMgcGxvdHRlZCA8LSBkZXYub2ZmKCkKIyMgb3BuNF9jb21iaW5lZApgYGAKCm9rLCBzbyBJIHBsb3R0ZWQgdGhlIHF1ZXN0aW9uIGEgYml0IGRpZmZlcmVudGx5LCBidXQgZ290IHRoZSBzYW1lCmFuc3dlci4KCkhlcmUgaXMgdGhlIHRleHQgb2YgVGhlcmVzYSdzIG5vdGVib29rIGZvbGxvd2luZyB0aGlzIGFuYWx5c2lzOgoKIlVnaCBvaCwgbG9va3MgbGlrZSB0aGVyZSBpcyBhdCBsZWFzdCBvbmUgcmV0aW5hIEtPIHNhbXBsZSB0aGF0IGhhcwpzb21lIG1lbGFub3BzaW4gZXhwcmVzc2lvbiBpbiBpdC4gVHVybnMgb3V0IGlwUkdDXzA3IGlzIGEgYmFkIGVnZwp3aGljaCBpcyBzdXBwb3NlZCB0byBiZSBhIEtPIGJ1dCBoYXMgbWVsYW5vcHNpbiBleHByZXNzaW9uLiBJdOKAmXMKZnJpZW5kcyB3aGljaCB3ZXJlIHBvb2xlZCBmcm9tIHRoZSBzYW1lIG1pY2UgYXJlIGlwcmdjXzA2IGFuZAppcHJnY18wOCwgc28gd2UgbmVlZCB0byBleGNsdWRlIGFsbCB0aGVzZSBzYW1wbGVzLiIKCkkgYW0gYWxzbyBzZWVpbmcgc29tZSBrbm9ja291dCBleHByZXNzaW9uIHdpdGggc29tZSBjYXZlYXRzOiBJIGRvIG5vdApoYXZlIHRoZSBhZmZlY3RlZCBzYW1wbGVzIGluIG15IGRhdGFzZXQgKGlwcmdjXzA3KSBhbmQgdGhlIGxldmVscyBJIGFtCnNlZWluZyBhcmUgcXVpdGUgbG93IC0tIEkgd2lsbCBsb29rIGluIElHViB0byBkb3VibGUgY2hlY2ssIGJ1dCBJCnN0cm9uZ2x5IHN1c3BlY3QgdGhhdCB0aGVzZSBhcmUgc29tZSBwaWRkbHkgcmVhZHMgbmVhciB0aGUgVVRScy4KCk9ud2FyZCEKCiMgUENBIHBsb3RzCgojIyBQQ0Egb2YgYWxsIGdlbmVzIGJ5IGxvY2F0aW9uCgpUaGVyZXNhJ3MgZmlyc3QgcGNhIHdhcyBvZiBsb2cyIGNwbSB2YWx1ZXMuICBJIG1pZ2h0IGFkZCBxdWFudGlsZS90bW0KdG8gdGhpcz8KCmBgYHtyfQp2M19sb2NhdGlvbiA8LSBzZXRfY29uZGl0aW9ucyhtbTM4X2hpc2F0X3YzLCBmYWN0ID0gImxvY2F0aW9uX2F0YiIpICU+JQogIHNldF9iYXRjaGVzKGZhY3QgPSAiZ2Vub3R5cGVfYXRiIikgJT4lCiAgc2V0X2NvbG9ycyhjb2xvcl9jaG9pY2VzW1sibG9jYXRpb24iXV0pCnYzX2xvY2F0aW9uX25vcm0gPC0gbm9ybWFsaXplKHYzX2xvY2F0aW9uLCBmaWx0ZXIgPSBUUlVFLCBub3JtID0gInF1YW50IiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0cmFuc2Zvcm0gPSAibG9nMiIsIGNvbnZlcnQgPSAiY3BtIikKdjNfbG9jYXRpb25fcGNhIDwtIHBsb3RfcGNhKHYzX2xvY2F0aW9uX25vcm0pCnBwKGZpbGUgPSAiMDFkaWFnbm9zdGljX2ltYWdlcy92M19sb2NhdGlvbl9ub3JtX3BjYS5wZGYiKQp2M19sb2NhdGlvbl9wY2EKZGV2Lm9mZigpCnYzX2xvY2F0aW9uX3BjYQpgYGAKCk9uY2UgYWdhaW4gd2Ugc2VlIHRoYXQgc2FtcGxlcyBpcHJnY182NiBhbmQgaXByZ2NfMTMwIGFyZSBsaWtlbHkKYWN0dWFsbHkgRExHTiBhbmQgbm90IFNDTi4gIEkgYW0gdGhlcmVmb3JlIGdvaW5nIHRvIGFkZCBhIGNvbHVtbiB0bwp0aGUgc2FtcGxlIHNoZWV0IG5vdGluZyB0aGlzLCBhbmQgcmVtb3ZlIHRoZW0gZnJvbSB0aGUgZXhwcmVzc2lvbnNldC4KCkkgd2lsbCB0aHVzIHJlcGxvdCB0aGUgZGF0YSBhZnRlciByZW1vdmluZyB0aG9zZSB0d28uICBJZiB3ZSB3YW50IHRvCnNlZSB3aGF0IGl0IGxvb2tzIGxpa2Ugd2l0aCB0aGUgcmUtYXR0cmlidXRlZCBsb2NhdGlvbnMsIHdlIGNhbiBkbyBzby4KClRoZXJlc2EgaGFzIGEgbmljZSBjaGFuZ2UgdG8gdGhlIFBDQSBwbG90dGVyIGluIHdoaWNoIHNoZSBzZXRzIHRoZQphbHBoYSBjaGFubmVsIGFzIGFuIGFkZGl0aW9uYWwgdmlzdWFsIHF1ZXVlIGZvciBhIG1ldGFkYXRhIGZhY3Rvci4uLgoKYGBge3J9Cm1tMzhfaGlzYXRfdjMgPC0gc3Vic2V0X3NlKG1tMzhfaGlzYXRfdjMsIHN1YnNldD0ic2FtcGxlaWQhPSdpcHJnY18xMzAnIikgJT4lCiAgc3Vic2V0X3NlKHN1YnNldD0ic2FtcGxlaWQhPSdpcHJnY182NiciKQp2M19sb2NhdGlvbiA8LSBzZXRfY29uZGl0aW9ucyhtbTM4X2hpc2F0X3YzLCBmYWN0ID0gImxvY2F0aW9uX2F0YiIpICU+JQogIHNldF9iYXRjaGVzKGZhY3QgPSAiZ2Vub3R5cGVfYXRiIikgJT4lCiAgc2V0X2NvbG9ycyhjb2xvcl9jaG9pY2VzW1sibG9jYXRpb24iXV0pCgp2M19sb2NhdGlvbl9ub3JtIDwtIG5vcm1hbGl6ZSh2M19sb2NhdGlvbiwgZmlsdGVyID0gVFJVRSwgbm9ybSA9ICJxdWFudCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdHJhbnNmb3JtID0gImxvZzIiLCBjb252ZXJ0ID0gImNwbSIpCmZpbHRlcmVkX2xvY2F0aW9uX3BjYSA8LSBwbG90X3BjYSh2M19sb2NhdGlvbl9ub3JtKQpwcChmaWxlID0gIjAyZmlsdGVyZWRfaW1hZ2VzL2ZpbHRlcmVkX2xvY2F0aW9uX3BjYS5wZGYiKQpmaWx0ZXJlZF9sb2NhdGlvbl9wY2FbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmZpbHRlcmVkX2xvY2F0aW9uX3BjYQoKcmVtb3ZlZF9zYW5rZXkgPC0gcGxvdF9tZXRhX3NhbmtleSh2M19sb2NhdGlvbiwgY29sb3JfY2hvaWNlcyA9IGNvbG9yX2Nob2ljZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmFjdG9ycyA9IGMoImdlbm90eXBlX2F0YiIsICJsb2NhdGlvbl9hdGIiLCAidGltZV9hdGIiKSkKcHAoZmlsZSA9ICIwMmZpbHRlcmVkX2ltYWdlcy9maWx0ZXJlZF9zYW5rZXkucGRmIikKcmVtb3ZlZF9zYW5rZXlbWyJnZ3Bsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKcmVtb3ZlZF9zYW5rZXkKYGBgCgpIZXJlIGlzIFRoZXJlc2EncyB0ZXh0LCByZWNhbGwgb25jZSBhZ2FpbiB0aGF0IEkgZG8gbm90IGhhdmUgc29tZSBvZgp0aGVzZSBvbGRlciBzYW1wbGVzIChpcHJnY182Mik6CgpQQzEgdnMgUEMyIGlkZW50aWZpZXMgcmV0aW5hIHZzIGF4b24gaXMgc3RpbGwgdGhlIG1haW4gY29tcG9uZW50IG9mCnZhcmlhdGlvbi4gV2UgZG8gc2VlIHRob3VnaCB0aGF0IGluIHRoZSBQQzIgZGlyZWN0aW9uLCB3ZSBzZWUgd2l0aCB0aGUKbmV3IHNhbXBsZXMgYWRkZWQsIHdlIGRvbuKAmXQgc2VlIHNlcGFyYXRpb24gYmFzZWQgb24gYXhvbmFsIHRhcmdldHMKKGRMR04gdnMgU0NOKS4gSW4gdGhlIFBDMSB2cyBQQzMgcGxvdCwgd2Ugc2VlIHRoYXQgaXTigJlzIFBDMyB3aGVyZSB3ZQpzdGFydCB0byBzZWUgdmFyaWF0aW9uIGNvcnJlbGF0ZWQgd2l0aCBheG9uYWwgY29tcGFydG1lbnQuIExldOKAmXMgbG9vawphdCBQQzEgdnMgUEMyIGNvbG9yZWQgYnkgYmF0Y2ggKHdoZW4gdGhleSB3ZXJlIHByb2Nlc3NlZC9zZXF1ZW5jZWQpIHRvCnNlZSBpZiB0aGF0IGlzIHdoYXQgaXMgY29udHJpYnV0aW5nIHNvIG11Y2ggdmFyaWF0aW9uIGluIFBDMi4KClNpZGUgbm90ZTogaXBSR0MgNjIgc2VlbXMgbGlrZSBhbiBvZGQgYmFsbC4gVGhpcyBzZWVtcyB0byBtZSBsaWtlIGl0CnNob3VsZCBoYXZlIGJlZW4gYSBkTEdOIFAwOCBzYW1wbGUuIElzIHRoZXJlIGFueSBwb3NzaWJpbGl0eSB0aGlzIGdvdAptaXNsYWJlbGVkIGVhcmx5IG9uPyBJIHdlbnQgYmFjayBhbmQgZG91YmxlIGNoZWNrZWQgdG8gc2VlIGlmIGFsbCBteQpwcm9jZXNzaW5nIGlzIGNvcnJlY3QgYW5kIGl0IGluZGVlZCB3YXMgbGFiZWxlZCBhbiBTQ04gUDE1IGZyb20gdGhlCnRpbWUgSSBnb3QgdGhlIHNhbXBsZXMsIGFuZCBpdCBpcyBpbmRlZWQuCgojIERFCgpJIG5vdyBzd2l0Y2hlZCB0byBUaGVyZXNhJ3MgZG9jdW1lbnQgJ1dPUktJTkdfYXhvblRSQVAuLi4nIGFuZCB3aWxsCnN0YXJ0IHB1bGxpbmcgc2VjdGlvbnMgZnJvbSBpdC4gIEkgYW0gcmVhc29uYWJseSBjZXJ0YWluIEkgaGF2ZQpyZWFzb25hYmx5IHNpbWlsYXIgc2FtcGxlIGRpc3RyaWJ1dGlvbnMsIHNvIEkgcHJlc3VtZSBJIGNhbiBpbnZva2UKc2ltaWxhci9pZGVudGljYWwgY2FsbHMgZm9yIERFU2VxIGFuZCBmcmllbmRzLgoKIyMgcDggcmV0aW5hcwoKSW4gdGhlIGJsb2NrIGltbWVkaWF0ZWx5IGJlZm9yZSB0aGUgREUgYW5hbHlzZXMsIFRoZXJlc2EgY3JlYXRlZCBhCnN1YnNldCBleHByZXNzaW9uc2V0IG9mIG9ubHkgcDA4IHJldGluYXMuICBUaHVzIHRoaXMgaW5pdGlhbCBERSBJCmFzc3VtZSB3aWxsIGJlIHVzZWQgdG8gc3VidHJhY3QgZm9yIHRoZSBTQ04vRExHTiBhbmFseXNlcyB0aGF0IGZvbGxvdy4KKEkgZ3Vlc3MgSSBjb3VsZCByZWFkIGFoZWFkIGFuZCBmaW5kIG91dCwgYnV0IG5vISBJIHdhbnQgdG8gYmUgYQpibGFuayBzbGF0ZSkKClRoZXJlc2EncyBwcmltYXJ5IHdvcmtmbG93IG1ha2VzIGhlYXZ5IHVzZSBvZiBERVNlcTIKKEBsb3ZlTW9kZXJhdGVkRXN0aW1hdGlvbkZvbGQyMDE0KSBhbmQgc3ZhCihAbGVla1NWQVBhY2thZ2VSZW1vdmluZzIwMTIpLiAgSW4gc29tZShtb3N0Pykgb2YgVGhlcmVzYSdzCmludm9jYXRpb25zIG9mIHRoZSBhbGxfcGFpcndpc2UoKSBmdW5jdGlvbiwgc2hlIGV4Y2x1ZGVzIHRoZSBvdGhlcgptZXRob2RzIHRoYXQgaXQgcGVyZm9ybXMuICBJbiB0aGlzIHdvcmtib29rLCBJIGxlZnQgdGhvc2UgbWV0aG9kcyBvbiwKdGh1cyB3ZSBjYW4gZXZhbHVhdGUgdGhlIHJlbGF0aXZlIHBlcmZvcm1hbmNlIERFU2VxMiB2cy4gc29tZSAoYWxsPyBJCm1heSBoYXZlIGRpc2FibGVkIEVCU2VxL2RyZWFtIGJlY2F1c2UgdGhleSB3ZXJlIHRha2luZyB0b28gbG9uZykKb2YgdGhlIGZvbGxvd2luZzoKCiogbGltbWE6IChAcml0Y2hpZUxpbW1hUG93ZXJzRGlmZmVyZW50aWFsMjAxNSkgKGFtb25nIG90aGVyCiAgcmVmZXJlbmNlcykgb3JpZ2luYWxseSB3cml0dGVuIGZvciBtaWNyb2FycmF5cy4KKiBFZGdlUjogKEByb2JpbnNvbkVkZ2VSQmlvY29uZHVjdG9yUGFja2FnZTIwMTApLCB3aGljaCBzaGFyZXMgbWFueQogIGFzc3VtcHRpb25zIHdpdGggREVTZXEyLgoqIEVCU2VxOiAoQGxlbmdFQlNlcUVtcGlyaWNhbEJheWVzMjAxMyksIGJlY2F1c2UgSSBoYXZlIGEgc29mdCBzcG90CiAgZm9yIGFueSBCYXllc2lhbiBtZXRob2QuCiogTm9pc2VxOiAoQHRhcmF6b25hTk9Jc2VxUk5Bc2VxRGlmZmVyZW50aWFsMjAxMSksIHdoaWNoIHNlZWtzIHRvCiAgZGlyZWN0bHkgbW9kZWwgdmFyaWFuY2UgaW4gYW4gUk5BU2VxIGRhdGFzZXQgYW5kIHVzZSB0aGF0IHRvIGltcHJvdmUKICB0aGUgc2Vuc2l0aXZpdHkgb2YgdGhlIHJlc3VsdCwgbXVjaCBsaWtlOgoqIERyZWFtOiAoQGhvZmZtYW5EcmVhbVBvd2VyZnVsRGlmZmVyZW50aWFsMjAyMCksIHdyaXR0ZW4gYnkgdGhlIHNhbWUKICBhdXRob3JzIChhbmQgdXNlcyB2ZXJ5IHNpbWlsYXIgbG9naWMpIGFzIG9uZSBvZiBteSBmYXZvcml0ZSB0b29scywKICB2YXJpYW5jZVBhcnRpdGlvbihAaG9mZm1hblZhcmlhbmNlUGFydGl0aW9uSW50ZXJwcmV0aW5nRHJpdmVyczIwMTYpLgoKYGBge3J9Cm1tMzhfcDhfcmV0aW5hIDwtIHN1YnNldF9zZShtbTM4X2hpc2F0X3YzLCBzdWJzZXQgPSAidGltZV9hdGI9PSdwMDgnICYgbG9jYXRpb25fYXRiPT0ncmV0aW5hJyIpCm1tX25vcm1hbF9wOF9yZXRfZGUgPC0gYWxsX3BhaXJ3aXNlKG1tMzhfcDhfcmV0aW5hLCBtb2RlbF9zdnMgPSAic3Zhc2VxIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbW9kZWxfZnN0cmluZyA9ICJ+IDAgKyBjb25kaXRpb24iLCBmaWx0ZXIgPSBUUlVFKQptbV9ub3JtYWxfcDhfcmV0X2RlCmBgYAoKVGhlcmUgc2VlbXMgdG8gYmUgYSBkaXNjcmVwZW5jeSB3aXRoIHByZXZpb3VzIGl0ZXJhdGlvbnMgb2YgdGhpcy4KTGV0IHVzIHNpbXBsaWZ5IHRvIGp1c3QgZG9pbmcgZGVzZXEgYW5kIGZpbmQgd2hhdCBpcyBjYXVzaW5nIGl0LgpJbiBteSBwcmV2aW91cyBpdGVyYXRpb24sIEkgZ290IDM2MzIgZ2VuZXMgaW4gdGhlIHVuaXF1ZShjKCkpIG9yIGhldCtrby4KCmBgYHtyfQpkZXNlcV9vbmx5IDwtIGRlc2VxX3BhaXJ3aXNlKG1tMzhfcDhfcmV0aW5hLCBtb2RlbF9zdnMgPSAic3Zhc2VxIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtb2RlbF9mc3RyaW5nID0gZGVmYXVsdF9mc3RyaW5nLCBmaWx0ZXIgPSBUUlVFKQpkZXNlcV9oZXRrZWVwZXJfZ2VuZXMgPC0gZGVzZXFfb25seSRhbGxfdGFibGVzJHd0X3JldGluYV92c19oZXRfcmV0aW5hICU+JQogIGZpbHRlcihsb2dGQyA8PSAtMC4yNSAmIGFkai5QLlZhbCA8PSAwLjA1KQpkZXNlcV9rb2tlZXBlcl9nZW5lcyA8LSBkZXNlcV9vbmx5JGFsbF90YWJsZXMkd3RfcmV0aW5hX3ZzX2tvX3JldGluYSAlPiUKICBmaWx0ZXIobG9nRkMgPD0gLTAuMjUgJiBhZGouUC5WYWwgPD0gMC4wNSkKZGVzZXFfa2VlcGVyZ2VuZXMgPC0gdW5pcXVlKGMocm93bmFtZXMoZGVzZXFfaGV0a2VlcGVyX2dlbmVzKSwKICAgICAgICAgICAgICAgICAgICAgICAgcm93bmFtZXMoZGVzZXFfa29rZWVwZXJfZ2VuZXMpKSkKbGVuZ3RoKGRlc2VxX2tlZXBlcmdlbmVzKQoKZGVzZXFfcGFpcl9oZXRrZWVwZXJfZ2VuZXMgPC0gbW1fbm9ybWFsX3A4X3JldF9kZSRkZXNlcSRhbGxfdGFibGVzJHd0X3JldGluYV92c19oZXRfcmV0aW5hICU+JQogIGZpbHRlcihsb2dGQyA8PSAtMC4yNSAmIGFkai5QLlZhbCA8PSAwLjA1KQpkZXNlcV9wYWlyX2tva2VlcGVyX2dlbmVzIDwtIG1tX25vcm1hbF9wOF9yZXRfZGUkZGVzZXEkYWxsX3RhYmxlcyR3dF9yZXRpbmFfdnNfa29fcmV0aW5hICU+JQogIGZpbHRlcihsb2dGQyA8PSAtMC4yNSAmIGFkai5QLlZhbCA8PSAwLjA1KQpkZXNlcV9wYWlyX2tlZXBlcmdlbmVzIDwtIHVuaXF1ZShjKHJvd25hbWVzKGRlc2VxX3BhaXJfaGV0a2VlcGVyX2dlbmVzKSwKICAgICAgICAgICAgICAgICAgICAgICAgcm93bmFtZXMoZGVzZXFfcGFpcl9rb2tlZXBlcl9nZW5lcykpKQpsZW5ndGgoZGVzZXFfcGFpcl9rZWVwZXJnZW5lcykKYGBgCgpUaGUgZm9sbG93aW5nIGludm9jYXRpb24gcGVyZm9ybWVkIGJ5IFRoZXJlc2EgZmlsdGVycyB0aGUgd3QvaGV0CmNvbXBhcmlzb24gZm9yIG9ubHkgdGhvc2UgZ2VuZXMgd2hpY2ggaW5jcmVhc2VkIGJ5IGF0IGxlYXN0IDAuMjUgbG9nRkMKd2l0aCBhIHNpZ25pZmljYW50IGFkanVzdGVkIHAtdmFsdWUuICBJIGFzc3VtZSB0aGF0IHRoaXMgaXMgdG8gdXNlIHRoZQp3dCBzYW1wbGVzIGFzIGEgdHJhbnNsYXRpb25hbCBjb250cm9sIGZvciB0aGUga2V0L2tvIGNvbXBhcmlzb25zOyBJIGFtCnRoZXJlZm9yZSB0aGlua2luZyB0aGF0IGZvciBteSBwdXJwb3NlcywgSSB3aWxsIHRoZXJlZm9yZSBzZXBhcmF0ZSB0aGUKY29udHJhc3RzIGZyb20gYWxsX3BhaXJ3aXNlIGRvIHRoaXMgaW4gYSBzdGVwd2lzZSBmYXNoaW9uLi4uCgpUaGUgYmxvY2sgb2YgY29kZSBpbW1lZGlhdGVseSBmb2xsb3dpbmcgVGhlcmVzYSdzIGFsbF9wYWlyd2lzZSgpCmludm9jYXRpb24gaXMgYSBsaXR0bGUgY29uZnVzaW5nIGZvciBtZSBhbmQgd2FycmFudHMgc29tZSBleHBsYW5hdGlvbgpieSBtZSB0byBtZSBpbiB0aGUgaG9wZXMgdGhhdCBJIGRvIG5vdCBtaXN1bmRlcnN0YW5kIHdoYXQgaXMgaGFwcGVuaW5nCmFuZCB0aGUgZ29hbHMgdGhlcmVpbi4KCkkgdGhpbmsgSSBjYW4gc2FmZWx5IGFzc3VtZSB0aGF0IHRoZSBnb2FsIGhlcmUgaXMgdG8gcHVsbCBvdXQgdGhlIElEcwp3aGljaCBpbmNyZWFzZWQgaW4gaGV0IHdpdGggcmVzcGVjdCB0byB3aWxkIHR5cGU7IGV2ZW4gaWYgYnkgYSBzbWFsbAptYXJnaW4sIGFzIGxvbmcgYXMgaXQgaXMgc3RhdGlzdGljYWxseSBzaWduaWZpY2FudCB2aXMgYSB2aXMgdGhlCmFkanVzdGVkIHAtdmFsdWUuCgpJIGFtIGdvaW5nIHRvIHBlcmZvcm0gd2hhdCBJIHRoaW5rIGlzIHRoZSBzYW1lIHRoaW5nIGluIGEgc2xpZ2h0bHkKZGlmZmVyZW50IGZhc2hpb24gc28gdGhhdCBJIGNhbiBzaGFyZSBhIGNvcHkgb2YgdGhlIHJlc3VsdHMgd2l0aAp3aG9tZXZlciBpcyBpbnRlcmVzdGVkLiAgSSB3aWxsIGFsc28gcmVwZWF0IFRoZXJlc2EncyBpbnZvY2F0aW9uIGFuZApwcm92ZSB0byBteXNlbGYgdGhhdCBJIHVuZGVyc3Rvb2QgYW5kIGdvdCB0aGUgc2FtZSBhbnN3ZXIuCgpgYGB7cn0Kd3RfaGV0X2tlZXBlciA8LSBsaXN0KCJoZXRfdnNfd3QiID0gYygiaGV0X3JldGluYSIsICJ3dF9yZXRpbmEiKSkKaGV0X3d0X3RhYmxlIDwtIGNvbWJpbmVfZGVfdGFibGVzKAogIG1tX25vcm1hbF9wOF9yZXRfZGUsIGtlZXBlcnMgPSB3dF9oZXRfa2VlcGVyLCBsYWJlbF9jb2x1bW4gPSBsYWJlbF9jb2x1bW4sCiAgZXhjZWwgPSAiMDN0aGVyZXNhX2NvbXBhcmlzb25fZXhjZWwvaGV0X3JldGluYV9jb250cm9sLnhsc3giKQp3YW50ZWRfc2lnIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCiAgaGV0X3d0X3RhYmxlLCBsZmMgPSAwLjI1LCBhY2NvcmRpbmdfdG8gPSAiZGVzZXEiLAogIGV4Y2VsID0gIjAzdGhlcmVzYV9jb21wYXJpc29uX2V4Y2VsL2hldF9yZXRpbmFfY29udHJvbC1zaWcueGxzeCIpCndhbnRlZF9oZXRfaW5jcmVhc2VkIDwtIHdhbnRlZF9zaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbImhldF92c193dCJdXQppbmNyZWFzZWRfaGV0X2dlbmVzIDwtIHJvd25hbWVzKHdhbnRlZF9oZXRfaW5jcmVhc2VkKQpgYGAKCkhlcmUgYXJlIFRoZXJlc2EncyBuZXh0IGxpbmVzOgoKYGBge3J9Cm1tX2RlX25vcm1hbF9wOF9yZXQgPC0gbW1fbm9ybWFsX3A4X3JldF9kZQpoZXRrZWVwZXJfZ2VuZXMgPC0gbW1fZGVfbm9ybWFsX3A4X3JldCRkZXNlcSRhbGxfdGFibGVzJHd0X3JldGluYV92c19oZXRfcmV0aW5hICU+JQogIGZpbHRlcihsb2dGQyA8PSAtMC4yNSAmIGFkai5QLlZhbCA8PSAwLjA1KQprb2tlZXBlcl9nZW5lcyA8LSBtbV9kZV9ub3JtYWxfcDhfcmV0JGRlc2VxJGFsbF90YWJsZXMkd3RfcmV0aW5hX3ZzX2tvX3JldGluYSAlPiUKICBmaWx0ZXIobG9nRkMgPD0gLTAuMjUgJiBhZGouUC5WYWwgPD0gMC4wNSkKa2VlcGVyZ2VuZXMgPC0gdW5pcXVlKGMocm93bmFtZXMoaGV0a2VlcGVyX2dlbmVzKSwKICAgICAgICAgICAgICAgICAgICAgICAgcm93bmFtZXMoa29rZWVwZXJfZ2VuZXMpKSkKIyMgV2Uga25vdyBhIHByaW9yaSB0aGF0IE9wbjQgaXMgRU5TTVVTRzAwMDAwMDIxNzk5CiMjIEkgZG8gbm90IGV4cGVjdCB0byBzZWUgaXQgaW4gdGhpcyBzZXQsIGl0IHNob3VsZCBiZSBoaWdoZXIgaW4gd3QKIyMgcmV0aW5hIHZzIGtvIHJldGluYSBieSBhIHNpZ25pZmljYW50IG1hcmdpbi4KIkVOU01VU0cwMDAwMDAyMTc5OSIgJWluJSBrZWVwZXJnZW5lcwojIyBPb29vaGhoIGJ1dCBpdCBfaXNfIGhpZ2hlciBpbiBoZXQgdnMuIHd0LCBhcyB3ZSBzYXcgaW4KIyMgdGhlIHZpb2xpbiBwbG90IGVhcmxpZXIuCmBgYAoKSSB0aGluayBSYXNobWkgbWFkZSBhIGNvbXBlbGxpbmcgcG9pbnQgd2hpY2ggaWxsdXN0cmF0ZXMgd2h5IHdlIGxpa2VseQpzaG91bGQgZXhwZWN0IHRoZSBleHByZXNzaW9uIG9mIE9wbjQgdG8gc2lnbmlmaWNhbnRseSBoaWdoZXIgaW4gdGhlCmhldGVyb3p5Z290ZXMgdnMgd2lsZC10eXBlOgoKMS4gIFJlY2FsbCB0aGF0IHRoZSBhc3NheSBpcyB1c2luZyB0aGUgaW1tdW5vcHVyaWZpY2F0aW9uIHRvIGV4dHJhY3QKICAgIHRoZSBSTkFzLgoyLiAgVGhlIHd0IHNhbXBsZXMgZG8gbm90IGhhdmUgdGhlIGNyZSByZWNvbWJpbmFzZSBhbmQgdGhlcmVmb3JlIG5vIEhBCiAgICBhbmQgdGhlcmVmb3JlIGV2ZXJ5dGhpbmcgd2Ugb2JzZXJ2ZSBpcyBkdWUgdG8gbm9uLXNwZWNpZmljCiAgICBiaW5kaW5nLgozLiAgVGhlIHNldCBvZiBnZW5lcyBvYnNlcnZlZCBkdWUgdG8gbm9uLXNwZWNpZmljIGJpbmRpbmcgaXMgZGlmZmVyZW50CiAgICB0aGFuIGhldC9rbyAocHJlc3VtYWJseSBhIGxhcmdlciBudW1iZXIgb2YgcmVsYXRpdmVseSBzbWFsbAogICAgdmFsdWVzKSwgdGhlcmVmb3JlIHRoZSBkaXZpc29yIHBlcmZvcm1lZCBpbiB0aGUgY3BtIGlzIGxpa2VseQogICAgcmVsYXRpdmx5IGxhcmdlIHJlc3VsdGluZyBpbiBub3JtYWxpemVkIHZhbHVlcyBnZXR0aW5nIHNoaWZ0ZWQKICAgIGRvd24gdG8gc29tZSBkZWdyZWUuCjQuICBPbiB0aGUgb3RoZXIgaGFuZCwgdGhlIHNldCBvZiBnZW5lcyBvYnNlcnZlZCBpbiBoZXQva28gYXJlIG1vcmUKICAgIGxpa2VseSB0byBiZSBvbmx5IHRoZSBzcGVjaWZpYyBiaW5kZXJzIGFuZCB0aGVyZWZvcmUgc21hbGxlciAoSQogICAgY2FuIHRlc3QgdGhpcykgcmVzdWx0aW5nIGluIGEgc21hbGxlciBkaXZpc29yIGFuZCBzbGlnaHQgc2hpZnRpbmcKICAgIHVwIGluIHRoZSBjcG0gdmFsdWVzLgoKVGhpcyBtYWtlcyBtZSB3b25kZXIgaWYgYW55IG5vcm1hbGl6YXRpb24gbWV0aG9kcyBleGlzdCB3aGljaCBkbwpzb21ldGhpbmcgbGlrZSBtdWx0aXBseSB0aGUgdmFsdWVzIGJ5IHNvbWUgdmFsdWUgcmVsYXRlZCB0byB0aGUKcHJvcG9ydGlvbiBvZiBvYnNlcnZlZCBnZW5lczsgYW5kL29yIGlmIHRoaXMgaXMgYSBnb29kL2JhZC9pbmRpZmZlcmVudAppZGVhLgoKQWxzbywganVzdCBhIG5vdGUgZm9yIG1lIHRvIHJlbWVtYmVyOiBSUEwyMiwgbm90IFJQUzIyLCBmb3Igc29tZQpyZWFzb24gSSBrZWVwIHRoaW5raW5nIHRoZSBzbWFsbCBzdWJ1bml0LgoKIyMgUHJvdmUgSSB1bmRlcnN0b29kCgpgYGB7cn0KaGV0a2VlcGVyX2dlbmVzIDwtIG1tX25vcm1hbF9wOF9yZXRfZGUkZGVzZXEkYWxsX3RhYmxlcyR3dF9yZXRpbmFfdnNfaGV0X3JldGluYSAlPiUKICBmaWx0ZXIobG9nRkMgPD0gLTAuMjUgJiBhZGouUC5WYWwgPD0gMC4wNSkKdGVzdHRoYXQ6OmV4cGVjdF90cnVlKG5yb3coaGV0a2VlcGVyX2dlbmVzKSA9PSBsZW5ndGgoaW5jcmVhc2VkX2hldF9nZW5lcykpCnRhYV9rZWVwZXJzIDwtIHNvcnQocm93bmFtZXMoaGV0a2VlcGVyX2dlbmVzKSkKYXRiX2tlZXBlcnMgPC0gc29ydChpbmNyZWFzZWRfaGV0X2dlbmVzKQp0ZXN0dGhhdDo6ZXhwZWN0X2VxdWFsKHRhYV9rZWVwZXJzLCBhdGJfa2VlcGVycykKYGBgCgpZYXkhIEkgY2FuIHJlYWQhICBOb3cgbGV0IHVzIHJlcGVhdCBmb3IgdGhlIEtPIHZzIHd0CgpgYGB7cn0Kd3Rfa29fa2VlcGVyIDwtIGxpc3QoImtvX3ZzX3d0IiA9IGMoImtvX3JldGluYSIsICJ3dF9yZXRpbmEiKSkKa29fd3RfdGFibGUgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgbW1fbm9ybWFsX3A4X3JldF9kZSwga2VlcGVycyA9IHd0X2tvX2tlZXBlciwgbGFiZWxfY29sdW1uID0gbGFiZWxfY29sdW1uLAogIGV4Y2VsID0gIjAzdGhlcmVzYV9jb21wYXJpc29uX2V4Y2VsL2tvX3JldGluYV9jb250cm9sLnhsc3giKQp3YW50ZWRfc2lnIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCiAga29fd3RfdGFibGUsIGxmYyA9IDAuMjUsIGFjY29yZGluZ190byA9ICJkZXNlcSIsCiAgZXhjZWwgPSAiMDN0aGVyZXNhX2NvbXBhcmlzb25fZXhjZWwva29fcmV0aW5hX2NvbnRyb2wtc2lnLnhsc3giKQoKd2FudGVkX2tvX2luY3JlYXNlZCA8LSB3YW50ZWRfc2lnW1siZGVzZXEiXV1bWyJ1cHMiXV1bWyJrb192c193dCJdXQppbmNyZWFzZWRfa29fZ2VuZXMgPC0gcm93bmFtZXMod2FudGVkX2tvX2luY3JlYXNlZCkKYGBgCgpUaGUgbmV4dCB0aGluZyBwZXJmb3JtZWQgaW4gVGhlcmVzYSdzIGRvY3VtZW50IGlzIGEgdW5pcXVlKGNvbmNhdGVuYXRpb24gb2YKdGhlc2UgdHdvIGdlbmUgZ3JvdXBzKSwgdGh1cyBzdWNraW5nIHVwIGV2ZXJ5IGdlbmUgd2hpY2ggd2FzCnNpZ25pZmljYW50bHkgaGlnaGVyIGluIGVpdGhlciB0aGUga25vY2tvdXQgX29yXyBoZXRlcnp5b3VzIHNhbXBsZXMKd2l0aCByZXNwZWN0IHRvIHdpbGQtdHlwZS4KClRoaXMgd2FzIGZvbGxvd2VkIGJ5IGEgY291cGxlIG9mIG1lcmdlIG9wZXJhdGlvbnMgb2YgYSBsaXR0bGUgYml0IG9mCnRoZSBhbm5vdGF0aW9uIGRhdGE7IEkgYW0gbm90IHN1cmUgSSB1bmRlcnN0YW5kIHRoZSBnb2FsIHlldC4uLgoKSGVyZSBpcyBoZXIgY29kZS4gSSBjb3BpZWQgdGhlIGFubm90YXRpb24gJ21naV9zeW1ib2wnIGNvbHVtbiB0bwonZXh0ZXJuYWxfZ2VuZV9uYW1lJyBzbyB0aGF0IEkgbmVlZCBub3QgY2hhbmdlIGFueSBvZiBoZXIgY29kZS4gIEkgYW0KYXNzdW1pbmcgdGhpcyBpcyB0aGUgYXBwcm9wcmlhdGUgY29sdW1uIG9mIGludGVyZXN0LCBJIGRvIG5vdCBrbm93CnRoaXMgZm9yIGNlcnRhaW4sIGJ1dCBpdCBzZWVtcyBxdWl0ZSBsaWtlbHkuCgpXaGlsZSBJIGFtIGF0IGl0LCBoZXJlIGlzIHRoZSBzZXRfc2lnX2xpbW1hKCkgZnVuY3Rpb24gZnJvbSBUaGVyZXNhJ3MgaGVscGVycy5SCgpgYGB7cn0Kc2V0X3NpZ19saW1tYSA8LSBmdW5jdGlvbihsaW1tYV90YmwsIGZhY3RvcnMgPSBOVUxMKSB7CiAgaWYgKGlzLm51bGwoZmFjdG9ycykpIHsKICAgICNzZXQgc2lnbmlmaWNhbmNlIGZvciBwbG90dGluZyBjb2xvcnMKICAgIGxpbW1hX3RibCRTaWduaWZpY2FuY2UgPC0gTkEKICAgIGxpbW1hX3RibFthYnMobGltbWFfdGJsJGxvZ0ZDKSA8IDEgfCBsaW1tYV90YmwkYWRqLlAuVmFsID4gLjA1LCAiU2lnbmlmaWNhbmNlIl0gPC0gIk5vdCBcbkVucmljaGVkIgogICAgbGltbWFfdGJsW2xpbW1hX3RibCRsb2dGQyA+PSAxICAmIGxpbW1hX3RibCRhZGouUC5WYWwgPD0gLjA1LCBdW1siU2lnbmlmaWNhbmNlIl1dIDwtICJEaXNlYXNlIFxuVXByZWd1bGF0ZWQiCiAgICBsaW1tYV90YmxbbGltbWFfdGJsJGxvZ0ZDIDw9IC0xICAmIGxpbW1hX3RibCRhZGouUC5WYWwgPD0gLjA1LCBdW1siU2lnbmlmaWNhbmNlIl1dIDwtICJEaXNlYXNlIFxuRG93bnJlZ3VsYXRlZCIKICAgIGxpbW1hX3RibCRTaWduaWZpY2FuY2UgPC0gZmFjdG9yKGxpbW1hX3RibCRTaWduaWZpY2FuY2UsIGxldmVscyA9IGMoIlVwcmVndWxhdGVkIiwgIkRvd25yZWd1bGF0ZWQiLCAgIk5vdCBcbkVucmljaGVkIikpCiAgfSBlbHNlIHsKICAgIGxpbW1hX3RibCRTaWduaWZpY2FuY2UgPC0gTkEKICAgIGxpbW1hX3RibFthYnMobGltbWFfdGJsJGxvZ0ZDKSA8IDEgfCBsaW1tYV90YmwkYWRqLlAuVmFsID4gLjA1LCAiU2lnbmlmaWNhbmNlIl0gPC0gIk5vdCBcbkVucmljaGVkIgogICAgaWYobnJvdyhsaW1tYV90YmxbbGltbWFfdGJsJGxvZ0ZDID49IDEgICYgbGltbWFfdGJsJGFkai5QLlZhbCA8PSAuMDUsIF0pICE9IDApIHsKICAgICAgbGltbWFfdGJsW2xpbW1hX3RibCRsb2dGQyA+PSAxICAmIGxpbW1hX3RibCRhZGouUC5WYWwgPD0gLjA1LCBdW1siU2lnbmlmaWNhbmNlIl1dIDwtIGZhY3RvcnNbMV0KICAgIH0KICAgIGlmIChucm93KGxpbW1hX3RibFtsaW1tYV90YmwkbG9nRkMgPD0gLTEgICYgbGltbWFfdGJsJGFkai5QLlZhbCA8PSAuMDUsIF0pICE9IDApIHsKICAgICAgbGltbWFfdGJsW2xpbW1hX3RibCRsb2dGQyA8PSAtMSAgJiBsaW1tYV90YmwkYWRqLlAuVmFsIDw9IC4wNSwgXVtbIlNpZ25pZmljYW5jZSJdXSA8LSBmYWN0b3JzWzJdCiAgICB9CiAgICBsaW1tYV90YmwkU2lnbmlmaWNhbmNlIDwtIGZhY3RvcihsaW1tYV90YmwkU2lnbmlmaWNhbmNlLCBsZXZlbHMgPSBjKGZhY3RvcnMsICAiTm90IFxuRW5yaWNoZWQiKSkKICB9CiAgcmV0dXJuKGxpbW1hX3RibCkKfQpgYGAKCiMjIyBDb21iaW5pbmcgaGV0L3d0IGFuZCBrby93dAoKYGBge3J9Cm1tX2Fubm90W1siZXh0ZXJuYWxfZ2VuZV9uYW1lIl1dIDwtIG1tX2Fubm90W1sibWdpX3N5bWJvbCJdXQprZWVwZXJnZW5lcyA8LSB1bmlxdWUoYyhyb3duYW1lcyhoZXRrZWVwZXJfZ2VuZXMpLCByb3duYW1lcyhrb2tlZXBlcl9nZW5lcykpKQpsZW5ndGgoa2VlcGVyZ2VuZXMpCmFubm90c190b19tZXJnZSA8LSBtbV9hbm5vdCAlPiUKICBzZWxlY3QoZW5zZW1ibF9nZW5lX2lkLCBleHRlcm5hbF9nZW5lX25hbWUpICU+JQogIGZpbHRlcihlbnNlbWJsX2dlbmVfaWQgJWluJQogICAgICAgICAgIHJvd25hbWVzKG1tX2RlX25vcm1hbF9wOF9yZXQkZGVzZXEkYWxsX3RhYmxlcyRrb19yZXRpbmFfdnNfaGV0X3JldGluYSkpICU+JQogIGRpc3RpbmN0KCkKbW1fZGVfbm9ybWFsX3A4X3JldCRkZXNlcSRhbGxfdGFibGVzJGtvX3JldGluYV92c19oZXRfcmV0aW5hIDwtIG1lcmdlKAogIG1tX2RlX25vcm1hbF9wOF9yZXQkZGVzZXEkYWxsX3RhYmxlcyRrb19yZXRpbmFfdnNfaGV0X3JldGluYSwgYW5ub3RzX3RvX21lcmdlLAogIGJ5LnggPSAwLCBieS55ID0gImVuc2VtYmxfZ2VuZV9pZCIsIGFsbC54ID0gVFJVRSkKZGYgPC0gbW1fZGVfbm9ybWFsX3A4X3JldCRkZXNlcSRhbGxfdGFibGVzJGtvX3JldGluYV92c19oZXRfcmV0aW5hICU+JQogIGRwbHlyOjptdXRhdGUobG9nRkMgPSAtbG9nRkMpICU+JQogIHNldF9zaWdfbGltbWEoZmFjdG9ycyA9IGMoIkhldCBFbnJpY2hlZCIsICJLTyBFbnJpY2hlZCIpKQpgYGAKCk15IHZlcnNpb24gb2YgdGhlIGFib3ZlIHRhc2sgbWFrZXMgdXNlIG9mIHRoZSBleGNsdWRlcyBvcHRpb24gb2YKY29tYmluZV9kZV90YWJlcy4gIEdpdmVuIHRoZSBzZXQgb2YgdW5pcXVlIGdlbmUgSURzIGluY3JlYXNlZCBpbiB0aGUKaGV0L2tvLCBJIGNhbiBhc2sgdG8gZXhsdWRlIGFueXRoaW5nIG5vdCBpbiB0aGF0IHNldC4gIEkgY291bGQgYWxzbwpoYXZlIG1vcmUgcGFyc2ltb25pb3VzbHkgZGlyZWN0bHkgZXhjbHVkZWQgYW55IGdlbmUgSUQgaW5jcmVhc2VkIGluCnRoZSB3dCBzYW1wbGVzLiAgQnV0LCBUaGVyZXNhIGFscmVhZHkgcHJvdmlkZWQgdGhlIGNvZGUgdG8gZG8gdGhlCmZvcm1lciwgc28gaXQgd2lsbCBiZSBsZXNzIHR5cGluZy9vcHBvcnR1bml0eSBmb3Igc2lsbHkgbWlzdGFrZXMgdG8KanVzdCBkbyB0aGF0LgoKYGBge3J9CmJvdGhfaW5jcmVhc2VkX2dlbmVzIDwtIHVuaXF1ZShjKGluY3JlYXNlZF9oZXRfZ2VuZXMsIGluY3JlYXNlZF9rb19nZW5lcykpCiMjIGFyYml0cmFpcmx5IGdyYWIgYWxsIGdlbmVzIGZyb20gb25lIG9mIG15IGRhdGEgc3RydWN0dXJlcy4KYWxsX2dlbmVzIDwtIHJvd25hbWVzKGV4cHJzKG1tMzhfaGlzYXRfdjMpKQpleGNsdWRlX2lkeCA8LSBhbGxfZ2VuZXMgJWluJSBib3RoX2luY3JlYXNlZF9nZW5lcwpzdW1tYXJ5KGV4Y2x1ZGVfaWR4KQpgYGAKCk15IEFwcmlsIDIwMjUgdmVyc2lvbiBvZiB0aGlzIHNob3dzIDIxLDc5MyBhbmQgMyw2MzIgZ2VuZXMgaW4gdGhpcwpzZXQuICBJcyB0aGF0IHN0aWxsIHRydWU/ICAoQXMgb2YgMjAyNjAzMTEsIGl0IGlzISkKCmBgYHtyfQpleGNsdWRlX2luY3JlYXNlZF9nZW5lcyA8LSBhbGxfZ2VuZXNbZXhjbHVkZV9pZHhdCnJldGluYV9rZWVwZXJzIDwtIGxpc3QoCiAgImhldF92c193dCIgPSBjKCJoZXRfcmV0aW5hIiwgInd0X3JldGluYSIpLAogICJrb192c193dCIgPSBjKCJrb19yZXRpbmEiLCAid3RfcmV0aW5hIiksCiAgImtvX3ZzX2hldCIgPSBjKCJrb19yZXRpbmEiLCAiaGV0X3JldGluYSIpKQojIyBBIHJlbWluZGVyIHRvIG15c2VsZjogdGhlcmUgaXMgYWxzbyBhIHBhcmFtZXRlciAnd2FudGVkX2dlbmVzJwojIyB3aGljaCBkb2VzIGVmZmVjdGl2ZWx5IHRoZSBzYW1lIHRoaW5nIGFzIGV4Y2x1ZGVzIGluIHRoaXMgY29udGV4dDsKIyMgZXhjbHVkZXMgd2FzIG9yaWdpbmFsbHkgd3JpdHRlbiB0byBhbGxvdyBmbGV4aWJsZSwga2V5d29yZC1iYXNlZAojIyBleGNsdXNpb24uCnA4X3JldGluYV90YWJsZXMgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgbW1fbm9ybWFsX3A4X3JldF9kZSwga2VlcGVycyA9IHJldGluYV9rZWVwZXJzLAogIHdhbnRlZF9nZW5lcyA9IGJvdGhfaW5jcmVhc2VkX2dlbmVzLCBsYWJlbF9jb2x1bW4gPSBsYWJlbF9jb2x1bW4sCiAgZXhjZWwgPSBnbHVlKCIwM3RoZXJlc2FfY29tcGFyaXNvbl9leGNlbC9wOF9yZXRpbmFfa2VwdF9nZW5lc19pbmNyZWFzZWRfaW5fd3RfdGFibGVzLXZ7dmVyfS54bHN4IikpCgpwOF9yZXRpbmFfc2lnIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCiAgcDhfcmV0aW5hX3RhYmxlcywgYWNjb3JkaW5nX3RvID0gImRlc2VxIiwKICBleGNlbCA9IGdsdWUoIjAzdGhlcmVzYV9jb21wYXJpc29uX2V4Y2VsL3A4X3JldGluYV9rZXB0X2dlbmVzX2luY3JlYXNlZF9pbl93dF9zaWctdnt2ZXJ9Lnhsc3giKSkKCm9wcG9zaXRlX3A4X3JldGluYV90YWJsZXMgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgbW1fbm9ybWFsX3A4X3JldF9kZSwga2VlcGVycyA9IHJldGluYV9rZWVwZXJzLAogIGV4Y2x1ZGVzID0gYm90aF9pbmNyZWFzZWRfZ2VuZXMsIGxhYmVsX2NvbHVtbiA9IGxhYmVsX2NvbHVtbiwKICBleGNlbCA9IGdsdWUoIjAzdGhlcmVzYV9jb21wYXJpc29uX2V4Y2VsL3A4X3JldGluYV9yZW1vdmVkX2dlbmVzX2luY3JlYXNlZF9pbl93dF90YWJsZXMtdnt2ZXJ9Lnhsc3giKSkKCm9wcG9zaXRlX3A4X3JldGluYV9zaWcgPC0gZXh0cmFjdF9zaWduaWZpY2FudF9nZW5lcygKICBwOF9yZXRpbmFfdGFibGVzLCBhY2NvcmRpbmdfdG8gPSAiZGVzZXEiLAogIGV4Y2VsID0gZ2x1ZSgiMDN0aGVyZXNhX2NvbXBhcmlzb25fZXhjZWwvcDhfcmV0aW5hX3JlbW92ZWRfZ2VuZXNfaW5jcmVhc2VkX2luX3d0X3NpZy12e3Zlcn0ueGxzeCIpKQpgYGAKCiMgRmlsdGVyaW5nIG91dCBub24tc3BlY2lmaWMgZ2VuZXMgYW5kIGV4YW1pbmluZyB0aGUgcmVzdWx0cwoKVGhlIGZvbGxvd2luZyBpcyBhIGNvcHkvcGFzdGUgZnJvbSBUaGVyZXNhIGNvbnRhaW5pbmcgdGhlIHJlbWFpbmluZwp0YXNrcyBzaGUgcGVyZm9ybWVkIGFuZCB3aWxsIHByb3ZpZGUgdGhlIHRlbXBsYXRlIGZvciBpbXBsZW1lbnRhdGlvbgpvZiB0aGUgZmluYWwgdGFza3MuCgpUaGlzIHBpY2tzIHVwIHdpdGggdGhlIGxpbmVzIGZyb20gaGVyIG5vdGVib29rIGltbWVkaWF0ZWx5IGZvbGxvd2luZwp0aGUgaW52b2NhdGlvbiBvZiAnc2V0X3NpZ19saW1tYShmYWN0b3JzID0gYygiSGV0IEVucmljaGVkIiAuLi4nLgoKRm9yIGFsbCBvZiB0aGUgcmVtYWluaW5nIGJsb2NrcyBJIHdpbGwgY29weSBpbiBoZXIgY29kZSwgdHVybiBvZmYgaXRzCmV2YWx1YXRpb24sIHJ1biB0aGUgYmxvY2tzIG1hbnVhbGx5LCBjb21wYXJlIHRoZW0gdG8gaGVyIG5vdGVib29rCm91dHB1dCwgdGhlbiBlbmFibGUgZWFjaCBibG9jayBhcyBJIGVuc3VyZSBJIHVuZGVyc3RhbmQgaXQuCgpJIHdpbGwgbGlrZWx5IHRoZXJlZm9yZSBpbnRyb2R1Y2Ugc29tZSBzbWFsbCBmb3JtYXR0aW5nIGNoYW5nZXMgYW5kCmFkZCBzb21lIGFkZGl0aW9uYWwgR1NFQS9lbnJpY2htZW50IHRhc2tzIG9uY2UgdGhlIG5vbi1zcGVjaWZpYwpmaWx0ZXJpbmcgaXMgY29tcGxldGUuCgpgYGB7cn0KZGYgPC0gZGYgJT4lCiAgZmlsdGVyKFJvdy5uYW1lcyAlaW4lIGtlZXBlcmdlbmVzKQpsYWJlbHNfdXBzIDwtIGRmICU+JQogIGZpbHRlcihhZGouUC5WYWwgPD0gMC4wNSAmIGFicyhsb2dGQykgPiAxKSAlPiUKICBhcnJhbmdlKGxvZ0ZDKSAlPiUKICBoZWFkKG4gPSA5KQpsYWJlbHNfZG93bnMgPC0gZGYgJT4lCiAgZmlsdGVyKGFkai5QLlZhbCA8PSAwLjA1ICYgYWJzKGxvZ0ZDKSA+IDEpICU+JQogIGFycmFuZ2UoLWxvZ0ZDKSAlPiUKICBoZWFkKG4gPSAxMSkKbGFiZWxzIDwtIHJiaW5kKGxhYmVsc191cHMsIGxhYmVsc19kb3ducykKcmVzX3RibCA8LSBkZgpERXBsb3QgPC0gZ2dwbG90KHJlc190YmwsIGFlcyh4ID0gbG9nRkMsIHkgPSAtbG9nMTAoYWRqLlAuVmFsKSwgbGFiZWwgPSBleHRlcm5hbF9nZW5lX25hbWUpKSArCiAgZ2VvbV9wb2ludChhZXMoY29sb3VyID0gU2lnbmlmaWNhbmNlKSwgc2l6ZSA9IDQpICsKICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBjKC0xLCAxKSkgKwogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IC1sb2cxMCgwLjA1KSkgKwogIHRoZW1lX2NsYXNzaWMoYmFzZV9zaXplID0gMjApICsKICB4bGFiKCJsb2cyKEZDKSIpICsKICB5bGFiKCItbG9nMTAocC12YWx1ZSkiKSArCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikgKwogIHNjYWxlX2NvbG9yX21hbnVhbCh2YWx1ZXMgPSBjKCIjRjg3NjZEIiwgIiMwMEJGQzQiLCAiR3JleSIpKSArCiAgZ2VvbV9sYWJlbF9yZXBlbCgKICAgIGRhdGEgPSBmaWx0ZXIoZGYsCiAgICAgICAgICAgICAgICAgICMjIGMoJ3M1X2hldF9kbGduJywgJ3M1X2hldF9yZXQnLCAnczVfaGV0X3NjbicpKSwKICAgICAgICAgICAgICAgICAgZXh0ZXJuYWxfZ2VuZV9uYW1lICVpbiUgbGFiZWxzJGV4dGVybmFsX2dlbmVfbmFtZSksCiAgICAjIyBudWRnZV94ID0gLTAuNSwKICAgIG51ZGdlX3kgPSAzLCBtYXgub3ZlcmxhcHMgPSAxNSkgKwogIHhsaW0oYygtMywgNikpCgpwcChmaWxlID0gIjAzdGhlcmVzYV9jb21wYXJpc29uX2ltYWdlcy9wMDhfcmV0aW5hX0RFXzEzMTIwMjQucGRmIikKREVwbG90CnBsb3R0ZWQgPC0gZGV2Lm9mZigpCkRFcGxvdAp3cml0ZV94bHN4KGRmLCBleGNlbCA9ICJleGNlbC9yZXRpbmFoZXRfdnNfcmV0aW5ha29fV1RmaWx0ZXJlZC54bHN4IikKYGBgCgojIyBIb3cgbWFueSB1cHMvZG93bnMKCmBgYHtyfQprb19lbnJpY2hlZCA8LSBkZiAlPiUKICBmaWx0ZXIoU2lnbmlmaWNhbmNlID09ICJLTyBFbnJpY2hlZCIpCm5yb3coa29fZW5yaWNoZWQpCmhldF9lbnJpY2hlZCA8LSAgZGYgJT4lCiAgZmlsdGVyKFNpZ25pZmljYW5jZSA9PSAiSGV0IEVucmljaGVkIikKbnJvdyhoZXRfZW5yaWNoZWQpCmBgYAoKIyMgY2F0ZWdvcnkgZW5yaWNobWVudC9HU0VBCgpgYGB7cn0KcmVndWxhdGVkX2dlbmVzIDwtIHJlc190YmwgJT4lCiAgZmlsdGVyKGFkai5QLlZhbCA8PSAwLjA1KSAlPiUKICBhcnJhbmdlKGxvZ0ZDKSAlPiUKICBzZWxlY3QoUm93Lm5hbWVzLCBsb2dGQywgYWRqLlAuVmFsLCBleHRlcm5hbF9nZW5lX25hbWUsIFNpZ25pZmljYW5jZSkgJT4lCiAgZmlsdGVyKGFicyhsb2dGQykgPj0gMSkKIyMgZ3NlYV9yZXN1bHRfa28gPC0gZ29zdChxdWVyeSA9IGtvX2dlbmVzJGV4dGVybmFsX2dlbmVfbmFtZSwKIyMgICAgICAgICAgICAgICAgICAgICAgICBvcmdhbmlzbSA9ICJtbXVzY3VsdXMiLAojIyAgICAgICAgICAgICAgICAgICAgICAgIGV2Y29kZXMgPSBUUlVFLAojIyAgICAgICAgICAgICAgICAgICAgICAgIG9yZGVyZWRfcXVlcnkgPSBUUlVFKQpnc2VhX3Jlc3VsdF9oZXQgPC0gZ29zdChxdWVyeSA9IGhldF9lbnJpY2hlZCRleHRlcm5hbF9nZW5lX25hbWUsCiAgICAgICAgICAgICAgICAgICAgICAgIG9yZ2FuaXNtID0gIm1tdXNjdWx1cyIsCiAgICAgICAgICAgICAgICAgICAgICAgIGV2Y29kZXMgPSBUUlVFLAogICAgICAgICAgICAgICAgICAgICAgICBvcmRlcmVkX3F1ZXJ5ID0gVFJVRSkKIyNnc2VhX3Jlc3VsdF9hbGxkeXNyZWd1bGF0ZWQgPC0gZ29zdChxdWVyeSA9IGFsbGR5c3JlZ3VsYXRlZF9nZW5lcyRleHRlcm5hbF9nZW5lX25hbWUsCiMjICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgb3JnYW5pc20gPSAibW11c2N1bHVzIiwKIyMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBldmNvZGVzID0gVFJVRSwKIyMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvcmRlcmVkX3F1ZXJ5ID0gVFJVRSkKYGBgCgpJIGhhdmUgYSBmdW5jdGlvbiBpbiBteSBwYWNrYWdlIHdoaWNoIHNlZWtzIHRvIG1ha2UgZ1Byb2ZpbGVyIHF1ZXJpZXMKYSBiaXQgbW9yZSBjb21wbGV0ZSBhbmQgZWFzeS4gIExldCB1cyBzZWUgaG93IHNpbWlsYXIgdGhlIHJlc3VsdCBpcy4uLgoKYGBge3IsIGV2YWw9RkFMU0V9CnJvd25hbWVzKGFsbGR5c3JlZ3VsYXRlZF9nZW5lcykgPC0gYWxsZHlzcmVndWxhdGVkX2dlbmVzW1siUm93Lm5hbWVzIl1dCmFsbGR5c3JlZ3VsYXRlZF9nZW5lc1tbIlJvdy5uYW1lcyJdXSA8LSBOVUxMCgpoZXRfZ3AgPC0gc2ltcGxlX2dwcm9maWxlcihyb3duYW1lcyhhbGxkeXNyZWd1bGF0ZWRfZ2VuZXMpLAogICAgICAgICAgICAgICAgICAgICAgICAgICBzcGVjaWVzID0gIm1tdXNjdWx1cyIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgIGV4Y2VsID0gZ2x1ZSgiZXhjZWwvaGV0X2dwcm9maWxlci12e3Zlcn0ueGxzeCIpKQpoZXRfZ3AKZW5yaWNocGxvdDo6ZG90cGxvdChoZXRfZ3BbWyJCUF9lbnJpY2giXV0pCmdwX3BhaXIgPC0gZW5yaWNocGxvdDo6cGFpcndpc2VfdGVybXNpbShoZXRfZ3BbWyJCUF9lbnJpY2giXV0pCmVucmljaHBsb3Q6OmVtYXBwbG90KGdwX3BhaXIpCmVucmljaHBsb3Q6OnNzcGxvdChncF9wYWlyKQplbnJpY2hwbG90Ojp0cmVlcGxvdChncF9wYWlyKQp1cHNldHBsb3QoaGV0X2dwW1siQlBfZW5yaWNoIl1dKQoKZW5yaWNocGxvdDo6ZG90cGxvdChoZXRfZ3BbWyJSRUFDX2VucmljaCJdXSkKZ3BfcGFpciA8LSBlbnJpY2hwbG90OjpwYWlyd2lzZV90ZXJtc2ltKGhldF9ncFtbIlJFQUNfZW5yaWNoIl1dKQplbnJpY2hwbG90OjplbWFwcGxvdChncF9wYWlyKQplbnJpY2hwbG90Ojpzc3Bsb3QoZ3BfcGFpcikKZW5yaWNocGxvdDo6dHJlZXBsb3QoZ3BfcGFpcikKdXBzZXRwbG90KGhldF9ncFtbIlJFQUNfZW5yaWNoIl1dKQpgYGAKCkkgbWFrZSBhIHNvbWV3aGF0IGFyYml0cmFyeSBkaXN0aW5jdGlvbiBiZXR3ZWVuIHRoZSBjb25jZXB0cyBvZgpvdmVyLWVucmljaG1lbnQgYW5hbHlzZXMgYW5kIEdTRUE6IHRoZSBmb3JtZXIgKGFzIHBlcmZvcm1lZCBieQpncHJvZmlsZXIpIChAcmF1ZHZlcmVQcm9maWxlcldlYlNlcnZlcjIwMTkpIHNlZWtzIHRvIGZpbmQgZ3JvdXBzIG9mCmdlbmVzIG92ZXJyZXByZXNlbnRlZCBpbiBHTy9yZWFjdG9tZS9ldGMuICBUaGVzZSBncm91cHMgb2YgZ2VuZXMgYXJlCnRha2VuIGV4Y2x1c2l2ZWx5IGZyb20gdGhlIHRvcC1uL2JvdHRvbS1uIGdlbmVzIHdpdGggcmVzcGVjdCB0bwpmb2xkLWNoYW5nZSBiZXR3ZWVuIGNvbmRpdGlvbnMgb2YgaW50ZXJlc3Q7IGluIHRoaXMgY2FzZSBtb3N0CmRpZmZlcmVudCB0aGFuIHd0IGluIHRoZSBwMDggcmV0aW5hIGtvIG9yIGhldCBzYW1wbGVzLgoKV2l0aCB0aGF0IGluIG1pbmQsIEkgY2FuIGludm9rZSBhIHNpbWlsYXIgZnVuY3Rpb24gdXNpbmcgdGhlIGZ1bGwKdGFibGUgb2YgREUgcmVzdWx0cyB0byBnZXQgd2hhdCBJIGNhbGwgdGhlIEdTRUEgcmVzdWx0IHVzaW5nCmNsdXN0ZXJQcm9maWxlciAoQHl1SW50cm9kdWN0aW9uQmlvbWVkaWNhbEtub3dsZWRnZSkuICBJbiB0aGUKZm9sbG93aW5nIGJsb2NrIEkgd2lsbCB1c2UgdGhlICdhbGxfY3Byb2ZpbGVyJyBmdW5jdGlvbiBvbiB0aGUgZGF0YQpzdHJ1Y3R1cmVzIG5hbWVkICdwOF9yZXRpbmFfdGFibGVzJyBhbmQgJ29wcG9zaXRlX3A4X3JldGluYV90YWJsZXMnIGluCm9yZGVyIHRvIGdldCB0aGVzZSBHU0VBIHJlc3VsdHMgZm9yIGVhY2ggY29udHJhc3QgcGVyZm9ybWVkIChoZXQvd3QsCmtvL3d0LCBoZXQva28pLiAgSSB3aWxsIGZvbGxvdyB0aGF0IHVwIHdpdGggJ2FsbF9ncHJvZmlsZXInIHdoaWNoIGRvZXMKdGhlIHNhbWUsIGJ1dCB1c2VzIGdQcm9maWxlcidzIGVucmljaG1lbnQgYW5hbHlzZXMgKGl0IHdpbGwgdGhlcmVmb3JlCmluY2x1ZGUgd2hhdCB3ZSBqdXN0IGxvb2tlZCBhdCkuCgpgYGB7cn0KcDA4X3JldGluYV9hbGxfY3AgPC0gYWxsX2Nwcm9maWxlcigKICBwOF9yZXRpbmFfc2lnLCBwOF9yZXRpbmFfdGFibGVzLCBvcmdkYiA9ICJvcmcuTW0uZWcuZGIiLCBvcmdkYl9mcm9tID0gb3JnZGJfZnJvbSwKICBleGNlbCA9ICIwM3RoZXJlc2FfY29tcGFyaXNvbl9leGNlbC9jcHJvZmlsZXJfcDA4X3JldGluYS54bHN4IikKCmVucmljaHBsb3Q6OmRvdHBsb3QocDA4X3JldGluYV9hbGxfY3BbWyJrb192c19oZXRfdXAiXV1bWyJnb19kYXRhIl1dW1siTUZfZW5yaWNoIl1dKQpwMDhfdG9wbl9nc2VhIDwtIHBsb3RfdG9wbl9nc2VhKHAwOF9yZXRpbmFfYWxsX2NwKQoKcHAoZmlsZSA9ICIwM3RoZXJlc2FfY29tcGFyaXNvbl9pbWFnZXMvZ3NlYV9wMDhfcmV0aW5hX2tvX3ZzX2hldF90b3BfaGl0LnBkZiIpCnAwOF90b3BuX2dzZWFbWyJHT19rb192c19oZXRfdXAiXV1bWzFdXQpwbG90dGVkIDwtIGRldi5vZmYoKQoKcDA4X3RvcG5fZ3NlYVtbIkdPX2tvX3ZzX2hldF91cCJdXVtbMV1dCnAwOF90b3BuX2dzZWFbWyJHT19rb192c19oZXRfdXAiXV1bWzJdXQpwMDhfdG9wbl9nc2VhW1siR09fa29fdnNfaGV0X3VwIl1dW1szXV0KcDA4X3RvcG5fZ3NlYVtbIkdPX2tvX3ZzX2hldF91cCJdXVtbNF1dCnAwOF90b3BuX2dzZWFbWyJHT19rb192c19oZXRfdXAiXV1bWzVdXQoKcHAoZmlsZSA9ICIwM3RoZXJlc2FfY29tcGFyaXNvbl9pbWFnZXMvZ3NlYV9wMDhfcmV0aW5hX2hldF92c193dF90b3BfaGl0LnBkZiIpCnAwOF90b3BuX2dzZWFbWyJHT19oZXRfdnNfd3RfdXAiXV1bWzFdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpgYGAKCmBgYHtyfQojZ3NlYV9rbyA8LSAgZ3NlYV9yZXN1bHRfa29bWyJyZXN1bHQiXV0gJT4lCiMgICAgc2VsZWN0KHRlcm1fbmFtZSwgcF92YWx1ZSwgdGVybV9zaXplLCBpbnRlcnNlY3Rpb25fc2l6ZSwgcmVjYWxsLCBzb3VyY2UsIGludGVyc2VjdGlvbikgJT4lCiMgICAgYXJyYW5nZShkZXNjKHJlY2FsbCkpICU+JQojICAgIGhlYWQobiA9IDEwKQojICBnc2VhX3Bsb3RzX2tvIDwtIGdncGxvdChnc2VhX2tvLCBhZXMoeCA9IHJlY2FsbCwgeSA9IHJlb3JkZXIodGVybV9uYW1lLCByZWNhbGwpLCBmaWxsID0gcF92YWx1ZSkpICsKIyAgZ2VvbV9iYXIoc3RhdCA9ICJpZGVudGl0eSIpKwojICBzY2FsZV9maWxsX2NvbnRpbnVvdXMobG93ID0gImJsdWUiLCBoaWdoID0gInJlZCIpICsKIyAgdGhlbWVfYncoKSsKIyAgeWxhYigiIikgKwojICB4bGFiKCJHU0VBIFNjb3JlIikKZ3NlYV9oZXQgPC0gIGdzZWFfcmVzdWx0X2hldFtbInJlc3VsdCJdXSAlPiUKICBzZWxlY3QodGVybV9uYW1lLCBwX3ZhbHVlLCB0ZXJtX3NpemUsIGludGVyc2VjdGlvbl9zaXplLCByZWNhbGwsIHNvdXJjZSwgaW50ZXJzZWN0aW9uKSAlPiUKICBhcnJhbmdlKGRlc2MocmVjYWxsKSkgJT4lCiAgaGVhZChuID0gMTApCmdzZWFfcGxvdHNfaGV0IDwtIGdncGxvdChnc2VhX2hldCwgYWVzKHggPSByZWNhbGwsIHkgPSByZW9yZGVyKHRlcm1fbmFtZSwgcmVjYWxsKSwgZmlsbCA9IHBfdmFsdWUpKSArCiAgZ2VvbV9iYXIoc3RhdCA9ICJpZGVudGl0eSIpICsKICBzY2FsZV9maWxsX2NvbnRpbnVvdXMobG93ID0gImJsdWUiLCBoaWdoID0gInJlZCIpICsKICB0aGVtZV9idygpICsKICB5bGFiKCIiKSArCiAgeGxhYigiT3ZlciBSZXByZXNlbnRhdGlvbiBTY29yZSIpCnBwKGZpbGUgPSAiMDN0aGVyZXNhX2NvbXBhcmlzb25faW1hZ2VzL0dTRUFfcDA4X2F4b250cmFwX3JldGluYWhldF91cHJlZ3VsYXRlZF92c19yZXRpbmFrby5wZGYiKQpnc2VhX3Bsb3RzX2hldApwbG90dGVkIDwtIGRldi5vZmYoKQpnc2VhX3Bsb3RzX2hldApgYGAKCmBgYHtyLCBldmFsPUZBTFNFfQpnc2VhX2FsbCA8LSAgZ3NlYV9yZXN1bHRfYWxsZHlzcmVndWxhdGVkW1sicmVzdWx0Il1dICU+JQogIHNlbGVjdCh0ZXJtX25hbWUsIHBfdmFsdWUsIHRlcm1fc2l6ZSwgaW50ZXJzZWN0aW9uX3NpemUsIHJlY2FsbCwgc291cmNlLCBpbnRlcnNlY3Rpb24pICU+JQogIGFycmFuZ2UoZGVzYyhyZWNhbGwpKSAlPiUKICBoZWFkKG4gPSAxMCkKZ3NlYV9wbG90c19hbGwgPC0gZ2dwbG90KGdzZWFfYWxsLCBhZXMoeCA9IHJlY2FsbCwgeSA9IHJlb3JkZXIodGVybV9uYW1lLCByZWNhbGwpLCBmaWxsID0gcF92YWx1ZSkpICsKICBnZW9tX2JhcihzdGF0ID0gImlkZW50aXR5IikgKwogIHNjYWxlX2ZpbGxfY29udGludW91cyhsb3cgPSAiYmx1ZSIsIGhpZ2ggPSAicmVkIikgKwogIHRoZW1lX2J3KCkgKwogIHlsYWIoIiIpICsKICB4bGFiKCJPdmVyIFJlcHJlc2VudGF0aW9uIFNjb3JlIikKCnBwKGZpbGUgPSAiaW1hZ2VzL0dTRUFfcDA4X3JldGluYV9heG9udHJhcF9hbGxkeXNyZWd1bGF0ZWRnZW5lcy5wZGYiKQpnc2VhX3Bsb3RzX2FsbApwbG90dGVkIDwtIGRldi5vZmYoKQpgYGAKCiMgU0NOIEhldCB2cyBLTwoKSXQgaXMgb25seSBub3cgdGhhdCBJIHJlYWxpemVkIHdlIGFyZSBzcGxpdHRpbmcgdGhlIGRhdGEgYnkgbG9jYXRpb24KZm9yIGVhY2ggc2V0IG9mIGNvbXBhcmlzb25zLiAgSSB0aGluayB0aGF0LCBsZWZ0IHRvIG15IG93biBkZXZpY2VzLCBJCndvdWxkIHByZWZlciB0byBrZWVwIHRoZSBpbnB1dCBkYXRhIHN0cnVjdHVyZSBpbnRhY3QsIHBlcmZvcm0gdGhlCnNvbWV3aGF0IGxhcmdlciBudW1iZXIgb2YgY29udHJhc3RzLCBhbmQgdGhlbiBzcGxpdCB1cCB0aGUgcmVzdWx0cy4KSWRlYWxseSB0aGlzIHdpbGwgc2xpZ2h0bHkgaW1wcm92ZSB0aGUgZmlkZWxpdHkgb2YgdGhlIHJlc3VsdHMKcmV0dXJuZWQgYnkgREVTZXEyIGFuZCBmcmllbmRzLiAgQnV0LCBJIHdpbGwgcnVuIHRoZSBzdGF0ZSBvZgpUaGVyZXNhJ3Mgbm90ZWJvb2sgd2l0aCBhcyBmZXcgY2hhbmdlcyBhcyBwb3NzaWJsZSBmaXJzdCwgdGhlbiBhZGQKdGhpcy4KCiMjIFBDQQoKSSBhbSBnb2luZyB0byBza2lwIHRoaXMgUENBIHBsb3QgZm9yIGEgY291cGxlIG9mIHJlYXNvbnM6IEkgYWxyZWFkeQpkaWQgYSBzdXBlcnNldCBvZiBpdCwgYW5kIHRoZSBzdWJzZXQgVGhlcmVzYSBwZXJmb3JtZWQgaXMgbm90IHZhbGlkCmdpdmVuIHRoZSBzZXQgb2Ygc2FtcGxlcyBpbmNsdWRlZCBpbiBteSBzYW1wbGUgc2hlZXQsIGFuZCBmaWd1cmluZyBvdXQKdGhlIGFjdHVhbGx5IGNvcnJlc3BvbmRpbmcgc3Vic2V0IHdpbGwgdGFrZSBtZSBmb3JldmVyLi4uICBJbgphZGRpdGlvbiwgSSB3YW50IHRvIHVzZSBteSBtbTM4X2hpc2F0X3YzIGZvciBldmVyeXRoaW5nLi4uCgpgYGB7ciwgZXZhbD1GQUxTRX0KbW0zOF9zdWJzZXQgPC0gc3Vic2V0X3NlKAogIG1tMzhfaGlzYXQsCiAgc3Vic2V0ID0gIihiYXRjaCA9PSAnNCcgfCBiYXRjaCA9PSAnNScgfCBiYXRjaCA9PSAnNicpICYgdGltZSA9PSAncDA4JyAmIGxvY2F0aW9uID09ICdzY24nIHwgc2FtcGxlaWQgPT0gJ2lwcmdjXzAzJyIpCm1tMzhfbm9ybSA8LSBub3JtYWxpemUobW0zOF9zdWJzZXQsIGZpbHRlciA9IFRSVUUsIGNvbnZlcnQgPSAiY3BtIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRyYW5zZm9ybSA9ICJsb2cyIiwgYmF0Y2ggPSAic3Zhc2VxIikKbW0zOF9ub3JtIDwtIHNldF9iYXRjaGVzKG1tMzhfbm9ybSwgZmFjdCA9ICJsb2NhdGlvbiIpCm1tMzhfbm9ybSA8LSBzZXRfY29uZGl0aW9ucyhtbTM4X25vcm0sIGZhY3QgPSAiZ2Vub3R5cGUiKQpwY2Ffbm9ybSA8LSBwbG90X3BjYShtbTM4X25vcm0sIG1heF9vdmVybGFwcyA9IDcwKQpwY2Ffbm9ybSRwbG90CmBgYAoKSW5zdGVhZCBJIHdpbGwgc2ltcGxpZnkgdGhlIHN1YnNldCBhbmQgc2VlIHdoYXQgaGFwcGVucy4uLgoKYGBge3J9CnNjbl9zYW1wbGVzIDwtIHN1YnNldF9zZShtbTM4X2hpc2F0X3YzLAogICAgICAgICAgICAgICAgICAgICAgICAgICBzdWJzZXQgPSAibG9jYXRpb25fYXRiID09ICdzY24nIikgJT4lCiAgc2V0X2JhdGNoZXMoZmFjdCA9ICJsb2NhdGlvbl9hdGIiKSAlPiUKICBzZXRfY29uZGl0aW9ucyhmYWN0ID0gImdlbm90eXBlX2F0YiIsIGNvbG9ycyA9IGNvbG9yX2Nob2ljZXNbWyJnZW5vdHlwZSJdXSkKc2NuX25vcm0gPC0gbm9ybWFsaXplKHNjbl9zYW1wbGVzLCBmaWx0ZXIgPSBUUlVFLCBjb252ZXJ0ID0gImNwbSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgIHRyYW5zZm9ybSA9ICJsb2cyIiwgYmF0Y2ggPSAic3Zhc2VxIikKc2NuX25vcm1fcGNhIDwtIHBsb3RfcGNhKHNjbl9ub3JtKQpzY25fbm9ybV9wY2EKYGBgCgojIExpYnJhcnkgc2l6ZXMgcG9zdC1kZWR1cGxpY2F0aW9uCgpUaGVyZXNhJ3MgbmV4dCBvcGVyYXRpb24gd2FzIHRvIHBlcmZvcm0gbGlic2l6ZS9ub256ZXJvIHBsb3RzLiAgSQphbHJlYWR5IGRpZCB0aGUgcHJlL3Bvc3QgZGVkdXBsaWNhdGlvbiBub256ZXJvLCBoZXJlIGlzIHRoZSBhbmFsYWdvdXMKbGlic2l6ZS4KCnYyIGlzIHByZS1kZWR1cGxpY2F0aW9uIGFuZCB2MyBpcyBwb3N0LgoKYGBge3J9CnBsb3RfbGlic2l6ZShtbTM4X2hpc2F0X3YyKQpwb3N0X2ZpbHRlcl9ub256ZXJvIDwtIHBsb3RfbGlic2l6ZShtbTM4X2hpc2F0X3YzLCB0ZXh0ID0gRkFMU0UpCnBwKGZpbGUgPSAiMDFkaWFnbm9zdGljX2ltYWdlcy9wb3N0X2FsbF9maWx0ZXJlc19ub256ZXJvLnBkZiIpCnBvc3RfZmlsdGVyX25vbnplcm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnBvc3RfZmlsdGVyX25vbnplcm8KYGBgCgpJIGFtIGEgYml0IGNvbmNlcm5lZCBhYm91dCBzb21lIG9mIHRoZXNlIGxpYnJhcnkgc2l6ZXMKcG9zdC1kZWR1cGxpY2F0aW9uLgoKTGV0IHVzIGxvb2sgYXQgdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIHJlYWRzIGFuZCBkdXBsaWNhdGlvbiwgd2hpY2ggSQphc3N1bWUgd2lsbCBiZSByZWxhdGl2ZWx5IGxpbmVhci4KCmBgYHtyfQp0ZXN0IDwtIGNvbERhdGEobW0zOF9oaXNhdF92MylbLCBjKCJoaXNhdF9nZW5vbWVfc2luZ2xlX2FsbCIsICJ1bWlfZGVkdXBfcGN0X3JlYWRzIildCnRlc3RfcGxvdCA8LSBwbG90X2xpbmVhcl9zY2F0dGVyKHRlc3QsIGxvZXNzID0gVFJVRSkKdGVzdF9wbG90W1sic2NhdHRlciJdXQp0ZXN0X3Bsb3RbWyJzY2F0dGVyIl1dCnBwKGZpbGUgPSAiMDFkaWFnbm9zdGljX2ltYWdlcy9kZWR1cGxpY2F0aW9uX3ZzX2hpc2F0X3NjYXR0ZXIucGRmIikKdGVzdF9wbG90W1sic2NhdHRlciJdXQpkZXYub2ZmKCkKYGBgCgpUaGVyZXNhIGFsc28gcHJvZHVjZWQgYSBkZW5zaXR5L3NhbXBsZSBwbG90LCB0aGF0IG1pZ2h0IHByb3ZlIHF1aXRlCnVzZWZ1bCBmb3IgdGhlc2UgZHVlIHRvIHRoZWlyIHNpZ25pZmljYW50bHkgbGFyZ2VyIHZhcmlhbmNlIGFjcm9zcwpzYW1wbGVzIChkdWUgdG8gZGVkdXBsaWNhdGlvbikuCgpgYGB7cn0KcHAoZmlsZSA9ICIwMWRpYWdub3N0aWNfaW1hZ2VzL3NhbXBsZV9kZW5zaXR5LnBkZiIpCm1tMzhfZGVuc2l0eSA8LSBwbG90X2RlbnNpdHkobG9jX2dlbm9fbnQpCm1tMzhfZGVuc2l0eVtbInBsb3QiXV0gKwogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKcGxvdF9ib3hwbG90KGxvY19nZW5vX250KQpkZXYub2ZmKCkKbW0zOF9kZW5zaXR5W1sicGxvdCJdXSArCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQpib3ggPC0gcGxvdF9ib3hwbG90KGxvY19nZW5vX250KQpwcChmaWxlID0gIjAxZGlhZ25vc3RpY19pbWFnZXMvc2FtcGxlX2JveHBsb3QucGRmIikKYm94CmRldi5vZmYoKQpib3gKYGBgCgpUaGVyZSBpcyBzb21lIGRpZmZlcmVuY2UgYWNyb3NzIHNhbXBsZSBkZW5zaXRpZXMsIGJ1dCBpdCBpcyBub3QgdG9vCmNyYXp5dG93bi4KCiMgRGl2ZXJnaW5nIGEgbGl0dGxlCgpBdCB0aGlzIHBvaW50IGluIHRoZSBkb2N1bWVudCBJIHJlYWQgYWhlYWQgYSBiaXQgYW5kIGNhbWUgdG8gdGhlCmNvbmNsdXNpb24gdGhhdCBpdCByZXBlYXRzIHRoZSBhYm92ZSBsb2dpYyBvZiB0YWtpbmcgdGhlIHVuaW9uIG9mIHd0CmNvbXBhcmlzb25zIHRvIHJlbW92ZSBnZW5lcyBmcm9tIHRoZSBhcHByb3ByaWF0ZSBoZXQva28gb3IgcDE1L3AwOCBvcgpsb2NhdGlvbiBjb21wYXJpc29ucy4gIFRoaXMgc2VlbXMgcXVpdGUgcmVhc29uYWJsZSB0byBtZSwgYnV0IEkgd291bGQKcHJlZmVyIHRvIG5vdCBzZXBhcmF0ZSBhbGwgdGhlIGRhdGEsIHNvIEkgd2lsbCBhdHRlbXB0IHRvIGR1cGxpY2F0ZQphbmQgc2xpZ2h0bHkgc3RyZWFtbGluZSB0aGlzIGxvZ2ljIG9uIHRoZSBmdWxsIGRhdGFzZXQuICBUaHVzIEkgYW0KZ29pbmcgdG8gc2tpcCBkb3duIHRvIHRoZSBlbmQgYW5kIGF0dGVtcHQgdG8gaW1wbGVtZW50IHRoaXMuCgojIyBERQoKTm90ZTogVGhlIGZvbGxvd2luZyBmZXcgYmxvY2tzIGFyZSBhbGwgY29weS9wYXN0ZWQgZGlyZWN0bHkgZnJvbQpUaGVyZXNhJ3Mgbm90ZWJvb2sgYW5kIGFyZSBub3QgZXZhbHVhdGVkIGJlY2F1c2UgdGhleSBhcmUgcGVyZm9ybWVkCmFsbW9zdCBpZGVudGljYWxseSBsYXRlciBidXQgd2l0aCBzbGlnaHRseSBkaWZmZXJlbnQgbG9naWMvb3JkZXJzLgoKYGBge3IsIGV2YWw9RkFMU0V9Cm1tX2RlX25vcm1hbF9wOF9zY24gPC0gYWxsX3BhaXJ3aXNlKG1tMzhfc3Vic2V0LCBtb2RlbF9iYXRjaCA9ICJzdmFzZXEiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwYXJhbGxlbCA9IEZBTFNFLCBkb19lYnNlcSA9IEZBTFNFLCBkb19iYXNpYyA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb19kcmVhbSA9IEZBTFNFLCBkb19ub2lzZXEgPSBGQUxTRSwgZG9fZWRnZXIgPSBGQUxTRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmlsdGVyID0gVFJVRSkKYW5ub3RzX3RvX21lcmdlIDwtIG1tX2Fubm90ICU+JQogIHNlbGVjdChlbnNlbWJsX2dlbmVfaWQsIGV4dGVybmFsX2dlbmVfbmFtZSkgJT4lCiAgZmlsdGVyKGVuc2VtYmxfZ2VuZV9pZCAlaW4lIHJvd25hbWVzKG1tX2RlX25vcm1hbF9wOF9zY24kZGVzZXEkYWxsX3RhYmxlcyRrb19zY25fdnNfaGV0X3NjbikpICU+JQogIGRpc3RpbmN0KCkKbW1fZGVfbm9ybWFsX3A4X3NjbiRkZXNlcSRhbGxfdGFibGVzJGtvX3Njbl92c19oZXRfc2NuIDwtIG1lcmdlKAogIG1tX2RlX25vcm1hbF9wOF9zY24kZGVzZXEkYWxsX3RhYmxlcyRrb19zY25fdnNfaGV0X3NjbiwKICBhbm5vdHNfdG9fbWVyZ2UsIGJ5LnggPSAwLCBieS55ID0gImVuc2VtYmxfZ2VuZV9pZCIsIGFsbC54ID0gVFJVRSkKYGBgCgpgYGB7ciwgZXZhbD1GQUxTRX0KaGV0a2VlcGVyX2dlbmVzIDwtIG1tX2RlX25vcm1hbF9wOF9zY24kZGVzZXEkYWxsX3RhYmxlcyR3dF9zY25fdnNfaGV0X3NjbiAlPiUKICBmaWx0ZXIobG9nRkMgPD0gLTAuMSAmIGFkai5QLlZhbCA8PSAwLjA1KQprb2tlZXBlcl9nZW5lcyA8LSBtbV9kZV9ub3JtYWxfcDhfc2NuJGRlc2VxJGFsbF90YWJsZXMkd3Rfc2NuX3ZzX2tvX3NjbiAlPiUKICBmaWx0ZXIobG9nRkMgPD0gLTAuMSAmIGFkai5QLlZhbCA8PSAwLjA1KQprZWVwZXJnZW5lcyA8LSB1bmlxdWUoYyhyb3duYW1lcyhoZXRrZWVwZXJfZ2VuZXMpLCByb3duYW1lcyhrb2tlZXBlcl9nZW5lcykpKQpkZiA8LSBtbV9kZV9ub3JtYWxfcDhfc2NuJGRlc2VxJGFsbF90YWJsZXMka29zY25fdnNfaGV0c2NuICU+JQogIGRwbHlyOjptdXRhdGUobG9nRkMgPSAtbG9nRkMpICU+JQogIHNldF9zaWdfbGltbWEoZmFjdG9ycyA9IGMoIkhldCBFbnJpY2hlZCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAiS08gRW5yaWNoZWQiKSkKZGYgPC0gZGYgJT4lCiAgZmlsdGVyKFJvdy5uYW1lcyAlaW4lIGtlZXBlcmdlbmVzKQpsYWJlbHNfdXBzIDwtIGRmICU+JQogIGZpbHRlcihhYnMobG9nRkMpID4gMSkgJT4lCiAgYXJyYW5nZShsb2dGQykgJT4lCiAgaGVhZChuID0gMSkKbGFiZWxzX2Rvd25zIDwtIGRmICU+JQogIGZpbHRlcihhYnMobG9nRkMpID4gMSkgJT4lCiAgYXJyYW5nZSgtbG9nRkMpICU+JQogIGhlYWQobiA9IDEpCmxhYmVscyA8LSByYmluZChsYWJlbHNfdXBzLCBsYWJlbHNfZG93bnMpCnJlc190YmwgPC0gZGYKREVwbG90IDwtIGdncGxvdChyZXNfdGJsLCBhZXMoeCA9IGxvZ0ZDLCB5ID0gLWxvZzEwKGFkai5QLlZhbCksIGxhYmVsID0gZXh0ZXJuYWxfZ2VuZV9uYW1lKSkgKwogIGdlb21fcG9pbnQoYWVzKGNvbG91ciA9IFNpZ25pZmljYW5jZSksIHNpemUgPSA0KSArCiAgZ2VvbV92bGluZSh4aW50ZXJjZXB0ID0gYygtMSwgMSkpICsKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSAtbG9nMTAoMC4wNSkpICsKICB0aGVtZV9jbGFzc2ljKGJhc2Vfc2l6ZSA9IDIwKSArCiAgeGxhYigibG9nMihGQykiKSArCiAgeWxhYigiLWxvZzEwKHAtdmFsdWUpIikgKwogICMjIGdndGl0bGUodGl0bGUsIHN1YnRpdGxlID0gc3VidGl0bGUpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb249InJpZ2h0IikgKwogIHNjYWxlX2NvbG9yX21hbnVhbCh2YWx1ZXM9YygiSGV0IEVucmljaGVkIiA9ICIjRjg3NjZEIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIktPIEVucmljaGVkIiA9ICIjMDBCRkM0IiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIk5vdFxuIEVucmljaGVkIiA9ICJHcmV5IikpICsKICBnZW9tX2xhYmVsX3JlcGVsKGRhdGE9ZmlsdGVyKGRmLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIyMgYygnczVfaGV0X2RsZ24nLCAnczVfaGV0X3JldCcsICdzNV9oZXRfc2NuJykpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZXh0ZXJuYWxfZ2VuZV9uYW1lICVpbiUgbGFiZWxzJGV4dGVybmFsX2dlbmVfbmFtZSksCiAgICAgICAgICAgICAgICAgICAjIyBudWRnZV94ID0gLTAuNSwKICAgICAgICAgICAgICAgICAgIG51ZGdlX3kgPSAzLCBtYXgub3ZlcmxhcHMgPSAxNSkgKwogIGdndGl0bGUoIlNDTiBIZXQgdnMgS08gVHJhbnNsYXRvbWUiKQoKcHAoZmlsZSA9ICJpbWFnZXMvcDA4X3Njbl9ERV8xMzEyMDI0LnBkZiIpCkRFcGxvdApwbG90dGVkIDwtIGRldi5vZmYoKQoKd3JpdGVfeGxzeChkZiwgZXhjZWwgPSAiZXhjZWwvc2NuaGV0X3ZzX3NjbmtvX1dUZmlsdGVyZWQueGxzeCIpCmBgYAoKIyMgSG93IG1hbnkgdXBzL2Rvd25zCgpgYGB7ciwgZXZhbD1GQUxTRX0Ka29fZW5yaWNoZWQgPC0gZGYgJT4lCiAgZmlsdGVyKFNpZ25pZmljYW5jZSA9PSAiS08gRW5yaWNoZWQiKQoKaGV0X2VucmljaGVkIDwtIGRmICU+JQogIGZpbHRlcihTaWduaWZpY2FuY2UgPT0gIkhldCBFbnJpY2hlZCIpCmBgYAoKIyMgR1NFQQoKYGBge3IsIGV2YWw9RkFMU0V9CmtvX2dlbmVzIDwtIHJlc190YmwgJT4lCiAgZmlsdGVyKGFkai5QLlZhbCA8PSAwLjA1KSAlPiUKICBhcnJhbmdlKC1hYnMobG9nRkMpKSAlPiUKICBzZWxlY3QoUm93Lm5hbWVzLCBsb2dGQywgYWRqLlAuVmFsLCBleHRlcm5hbF9nZW5lX25hbWUsIFNpZ25pZmljYW5jZSkgJT4lCiAgZmlsdGVyKGxvZ0ZDIDw9IC0xKQoKaGV0X2dlbmVzIDwtIHJlc190YmwgJT4lCiAgZmlsdGVyKGFkai5QLlZhbCA8PSAwLjA1KSAlPiUKICBhcnJhbmdlKC1hYnMobG9nRkMpKSAlPiUKICBzZWxlY3QoUm93Lm5hbWVzLCBsb2dGQywgYWRqLlAuVmFsLCBleHRlcm5hbF9nZW5lX25hbWUsIFNpZ25pZmljYW5jZSkgJT4lCiAgZmlsdGVyKGxvZ0ZDID49IDEpCgphbGxkeXNyZWd1bGF0ZWRfZ2VuZXMgPC0gcmVzX3RibCAlPiUKICBmaWx0ZXIoYWRqLlAuVmFsIDw9IDAuMDUpICU+JQogIGFycmFuZ2UobG9nRkMpICU+JQogIHNlbGVjdChSb3cubmFtZXMsIGxvZ0ZDLCBhZGouUC5WYWwsIGV4dGVybmFsX2dlbmVfbmFtZSwgU2lnbmlmaWNhbmNlKSAlPiUKICBmaWx0ZXIoYWJzKGxvZ0ZDKSA+PSAxKQoKZ3NlYV9yZXN1bHRfa28gPC0gZ29zdChxdWVyeSA9IGtvX2dlbmVzJGV4dGVybmFsX2dlbmVfbmFtZSwKICAgICAgICAgICAgICAgICAgICAgICBvcmdhbmlzbSA9ICJtbXVzY3VsdXMiLAogICAgICAgICAgICAgICAgICAgICAgIGV2Y29kZXMgPSBUUlVFLAogICAgICAgICAgICAgICAgICAgICAgIG9yZGVyZWRfcXVlcnkgPSBUUlVFKQoKZ3NlYV9yZXN1bHRfaGV0IDwtIGdvc3QocXVlcnkgPSBoZXRfZ2VuZXMkZXh0ZXJuYWxfZ2VuZV9uYW1lLAogICAgICAgICAgICAgICAgICAgICAgICBvcmdhbmlzbSA9ICJtbXVzY3VsdXMiLAogICAgICAgICAgICAgICAgICAgICAgICBldmNvZGVzID0gVFJVRSwKICAgICAgICAgICAgICAgICAgICAgICAgb3JkZXJlZF9xdWVyeSA9IFRSVUUpCgpnc2VhX3Jlc3VsdF9hbGxkeXNyZWd1bGF0ZWQgPC0gZ29zdChxdWVyeSA9IGFsbGR5c3JlZ3VsYXRlZF9nZW5lcyRleHRlcm5hbF9nZW5lX25hbWUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9yZ2FuaXNtID0gIm1tdXNjdWx1cyIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV2Y29kZXMgPSBUUlVFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvcmRlcmVkX3F1ZXJ5ID0gVFJVRSkKYGBgCgpgYGB7ciwgZXZhbD1GQUxTRX0KZ3NlYV9rbyA8LSAgZ3NlYV9yZXN1bHRfa29bWyJyZXN1bHQiXV0gJT4lCiAgc2VsZWN0KHRlcm1fbmFtZSwgcF92YWx1ZSwgdGVybV9zaXplLCBpbnRlcnNlY3Rpb25fc2l6ZSwgcmVjYWxsLCBzb3VyY2UsIGludGVyc2VjdGlvbikgJT4lCiAgYXJyYW5nZShkZXNjKHJlY2FsbCkpICU+JQogIGhlYWQobiA9IDEwKQpnc2VhX3Bsb3RzX2tvIDwtIGdncGxvdChnc2VhX2tvLCBhZXMoeCA9IHJlY2FsbCwgeSA9IHJlb3JkZXIodGVybV9uYW1lLCByZWNhbGwpLCBmaWxsID0gcF92YWx1ZSkpICsKICBnZW9tX2JhcihzdGF0ID0gImlkZW50aXR5IikgKwogIHNjYWxlX2ZpbGxfY29udGludW91cyhsb3cgPSAiYmx1ZSIsIGhpZ2ggPSAicmVkIikgKwogIHRoZW1lX2J3KCkgKwogIHlsYWIoIiIpICsKICB4bGFiKCJPdmVyIGVucmljaG1lbnQgU2NvcmUiKQoKZ3NlYV9oZXQgPC0gIGdzZWFfcmVzdWx0X2hldFtbInJlc3VsdCJdXSAlPiUKICBzZWxlY3QodGVybV9uYW1lLCBwX3ZhbHVlLCB0ZXJtX3NpemUsIGludGVyc2VjdGlvbl9zaXplLCByZWNhbGwsIHNvdXJjZSwgaW50ZXJzZWN0aW9uKSAlPiUKICBhcnJhbmdlKGRlc2MocmVjYWxsKSkgJT4lCiAgaGVhZChuID0gMTApCmdzZWFfcGxvdHNfaGV0IDwtIGdncGxvdChnc2VhX2hldCwgYWVzKHggPSByZWNhbGwsIHkgPSByZW9yZGVyKHRlcm1fbmFtZSwgcmVjYWxsKSwgZmlsbCA9IHBfdmFsdWUpKSArCiAgZ2VvbV9iYXIoc3RhdCA9ICJpZGVudGl0eSIpICsKICBzY2FsZV9maWxsX2NvbnRpbnVvdXMobG93ID0gImJsdWUiLCBoaWdoID0gInJlZCIpICsKICB0aGVtZV9idygpICsKICB5bGFiKCIiKSArCiAgeGxhYigiT3ZlciBlbnJpY2htZW50IFNjb3JlIikKCmdzZWFfYWxsIDwtICBnc2VhX3Jlc3VsdF9hbGxkeXNyZWd1bGF0ZWRbWyJyZXN1bHQiXV0gJT4lCiAgc2VsZWN0KHRlcm1fbmFtZSwgcF92YWx1ZSwgdGVybV9zaXplLCBpbnRlcnNlY3Rpb25fc2l6ZSwgcmVjYWxsLCBzb3VyY2UsIGludGVyc2VjdGlvbikgJT4lCiAgYXJyYW5nZShkZXNjKHJlY2FsbCkpICU+JQogIGhlYWQobiA9IDEwKQpnc2VhX3Bsb3RzX2FsbCA8LSBnZ3Bsb3QoZ3NlYV9hbGwsIGFlcyh4ID0gcmVjYWxsLCB5ID0gcmVvcmRlcih0ZXJtX25hbWUsIHJlY2FsbCksIGZpbGwgPSBwX3ZhbHVlKSkgKwogIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiKSArCiAgc2NhbGVfZmlsbF9jb250aW51b3VzKGxvdyA9ICJibHVlIiwgaGlnaCA9ICJyZWQiKSArCiAgdGhlbWVfYncoKSArCiAgeWxhYigiIikgKwogIHhsYWIoIk92ZXIgZW5yaWNobWVudCBTY29yZSIpCgpwcChmaWxlID0gImltYWdlcy9HU0VBX3AwOF9yZXRpbmFfYXhvbnRyYXBfYWxsZHlzcmVndWxhdGVkZ2VuZXMucGRmIikKZ3NlYV9wbG90c19hbGwKcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgojIEhldCBSZXRpbmEgdnMgSGV0IFNDTgoKYGBge3IsIGV2YWw9RkFMU0V9Cm1tMzhfc3Vic2V0MiA8LSBzdWJzZXRfc2UoCiAgbW0zOF9oaXNhdCwKICBzdWJzZXQgPSAiKGJhdGNoID09ICc0JyB8IGJhdGNoID09ICc1JyB8IGJhdGNoID09ICc2JykgJiB0aW1lID09ICdwMDgnICYgZ2Vub3R5cGUgIT0gJ2tvJyAmIGxvY2F0aW9uICE9ICdkbGduJyB8IHNhbXBsZWlkID09ICdpcHJnY18wMyciKQoKbW0zOF9zdWJzZXQyIDwtIHN1YnNldF9zZShtbTM4X3N1YnNldDIsIHN1YnNldCA9ICJzYW1wbGVpZCAhPSAnaXByZ2NfODknIikKbW0zOF9zdWJzZXQyJGRlc2lnbiAlPiUKICBzZWxlY3QoZ2Vub3R5cGUsIGxvY2F0aW9uKSAlPiUKICB0YWJsZSgpCm1tMzhfbm9ybTIgPC0gbm9ybWFsaXplKG1tMzhfc3Vic2V0MiwgZmlsdGVyPVRSVUUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29udmVydD0iY3BtIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0cmFuc2Zvcm09ImxvZzIiLCBiYXRjaCA9ICJzdmFzZXEiKQpgYGAKCiMjIFBDQQoKYGBge3IsIGV2YWw9RkFMU0V9Cm1tMzhfbm9ybTIgPC0gc2V0X2JhdGNoZXMobW0zOF9ub3JtMiwgZmFjdCA9ICJsb2NhdGlvbiIpCm1tMzhfbm9ybTIgPC0gc2V0X2NvbmRpdGlvbnMobW0zOF9ub3JtMiwgZmFjdCA9ICJnZW5vdHlwZSIpCnBjYV9ub3JtMiA8LSBwbG90X3BjYShtbTM4X25vcm0yLCBtYXhfb3ZlcmxhcHMgPSA3MCkKcGNhX25vcm0yJHBsb3QKYGBgCgojIyBERQoKYGBge3IsIGV2YWw9RkFMU0V9Cm1tX2RlX3N1YnNldDIgPC0gYWxsX3BhaXJ3aXNlKG1tMzhfc3Vic2V0MiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbW9kZWxfYmF0Y2g9InN2YXNlcSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBhcmFsbGVsPUZBTFNFLCBkb19lYnNlcT1GQUxTRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG9fYmFzaWMgPSBGQUxTRSwgZG9fZHJlYW0gPSBGQUxTRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG9fbm9pc2VxID0gRkFMU0UsIGRvX2VkZ2VyID0gRkFMU0UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZpbHRlciA9IFRSVUUpCmBgYAoKYGBge3IsIGV2YWw9RkFMU0V9CnJldGluYWtlZXBlcl9nZW5lcyA8LSBtbV9kZV9zdWJzZXQyJGRlc2VxJGFsbF90YWJsZXMkd3RfcmV0aW5hX3ZzX2hldF9yZXRpbmEgJT4lCiAgZmlsdGVyKGxvZ0ZDIDw9IC0wLjEgJiBhZGouUC5WYWwgPD0gMC4wNSkKCnNjbmtlZXBlcl9nZW5lcyA8LSBtbV9kZV9zdWJzZXQyJGRlc2VxJGFsbF90YWJsZXMkd3Rfc2NuX3ZzX2hldF9zY24gJT4lCiAgZmlsdGVyKGxvZ0ZDIDw9IC0wLjEgJiBhZGouUC5WYWwgPD0gMC4wNSkKCmtlZXBlcmdlbmVzIDwtIHVuaXF1ZShjKHJvd25hbWVzKHJldGluYWtlZXBlcl9nZW5lcyksIHJvd25hbWVzKHNjbmtlZXBlcl9nZW5lcykpKQoKYW5ub3RzX3RvX21lcmdlIDwtIG1tX2Fubm90ICU+JQogIHNlbGVjdChlbnNlbWJsX2dlbmVfaWQsIGV4dGVybmFsX2dlbmVfbmFtZSkgJT4lCiAgZmlsdGVyKGVuc2VtYmxfZ2VuZV9pZCAlaW4lIHJvd25hbWVzKG1tX2RlX3N1YnNldDIkZGVzZXEkYWxsX3RhYmxlcyRoZXRfc2NuX3ZzX2hldF9yZXRpbmEpKSAlPiUKICBkaXN0aW5jdCgpCgptbV9kZV9zdWJzZXQyJGRlc2VxJGFsbF90YWJsZXMkaGV0X3Njbl92c19oZXRfcmV0aW5hIDwtIG1lcmdlKAogIG1tX2RlX3N1YnNldDIkZGVzZXEkYWxsX3RhYmxlcyRoZXRfc2NuX3ZzX2hldF9yZXRpbmEsCiAgYW5ub3RzX3RvX21lcmdlLCBieS54ID0gMCwKICBieS55ID0gImVuc2VtYmxfZ2VuZV9pZCIsIGFsbC54ID0gVFJVRSkKCmRmIDwtIG1tX2RlX3N1YnNldDIkZGVzZXEkYWxsX3RhYmxlcyRoZXRfc2NuX3ZzX2hldF9yZXRpbmEgJT4lCiAgbXV0YXRlKFNpZ25pZmljYW5jZSA9IGNhc2Vfd2hlbihsb2dGQyA8PSAtMS4wIH4gIlJldGluYSBFbnJpY2hlZCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsb2dGQyA+PSAxLjAgfiAiU0NOIEVucmljaGVkIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxvZ0ZDID4gLTEuMCAmIGxvZ0ZDIDwgMS4wIH4gIk5vdFxuIEVucmljaGVkIikpCgpkZiA8LSBkZiAlPiUKICBmaWx0ZXIoUm93Lm5hbWVzICVpbiUga2VlcGVyZ2VuZXMpCgpzY25fZW5yaWNoZWQgPC0gZGYgJT4lCiAgZmlsdGVyKGFkai5QLlZhbCA8PSAwLjA1ICYgbG9nRkMgPj0gMS4wKSAlPiUKICBhcnJhbmdlKC1sb2dGQykgJT4lCiAgc2VsZWN0KFJvdy5uYW1lcywgZXh0ZXJuYWxfZ2VuZV9uYW1lLCBsb2dGQywgYWRqLlAuVmFsKSAlPiUKICBtdXRhdGUoU2lnbmlmaWNhbmNlID0gIlNDTiBFbnJpY2hlZCIpICU+JQogIGZpbHRlcihSb3cubmFtZXMgJWluJSByb3duYW1lcyhzY25rZWVwZXJfZ2VuZXMpKQoKcmV0aW5hX2VucmljaGVkIDwtIGRmICU+JQogIGZpbHRlcihhZGouUC5WYWwgPD0gMC4wNSAmIGxvZ0ZDIDw9IC0xLjApICU+JQogIGFycmFuZ2UobG9nRkMpICAlPiUKICBzZWxlY3QoUm93Lm5hbWVzLCBleHRlcm5hbF9nZW5lX25hbWUsIGxvZ0ZDLCBhZGouUC5WYWwpICU+JQogIG11dGF0ZShTaWduaWZpY2FuY2UgPSAiUmV0aW5hIEVucmljaGVkIikgJT4lCiAgZmlsdGVyKFJvdy5uYW1lcyAlaW4lIHJvd25hbWVzKHJldGluYWtlZXBlcl9nZW5lcykpCgpub3RlbnJpY2hlZCA8LSBkZiAlPiUKICBzZWxlY3QoUm93Lm5hbWVzLCBleHRlcm5hbF9nZW5lX25hbWUsIGxvZ0ZDLCBhZGouUC5WYWwsIFNpZ25pZmljYW5jZSkgJT4lCiAgZmlsdGVyKFJvdy5uYW1lcyAlaW4lIGMocm93bmFtZXMocmV0aW5ha2VlcGVyX2dlbmVzKSwKICAgICAgICAgICAgICAgICAgICAgICAgICByb3duYW1lcyhzY25rZWVwZXJfZ2VuZXMpKVtkdXBsaWNhdGVkKGMocm93bmFtZXMocmV0aW5ha2VlcGVyX2dlbmVzKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcm93bmFtZXMoc2Nua2VlcGVyX2dlbmVzKSkpXSkgJT4lCiAgZmlsdGVyKFNpZ25pZmljYW5jZSA9PSAiTm90XG4gRW5yaWNoZWQiKQoKZGYgPC0gcmJpbmQoc2NuX2VucmljaGVkLCByZXRpbmFfZW5yaWNoZWQsIG5vdGVucmljaGVkKQpkZiA8LSBkZiAlPiUKICBkaXN0aW5jdCgpCgojIyB3cml0ZXhsOjp3cml0ZV94bHN4KGRmLCBwYXRoID0gImF4b25UUkFQX0RFX3Jlc3VsdHNfMjAyNDAyMDIvcmV0aW5haGV0X3ZzX3Njbl9oZXRfV1RmaWx0ZXJlZC54bHN4IikKYGBgCgpgYGB7ciwgZXZhbD1GQUxTRX0KbGFiZWxzX3VwcyA8LSBkZiAlPiUKICBmaWx0ZXIoYWRqLlAuVmFsIDw9IDAuMDUgJiBhYnMobG9nRkMpID4gMS4wKSAlPiUKICBhcnJhbmdlKGxvZ0ZDKSAlPiUKICBoZWFkKG4gPSAxMCkKCmxhYmVsc19kb3ducyA8LSBkZiAlPiUKICBmaWx0ZXIoYWRqLlAuVmFsIDw9IDAuMDUgJiBhYnMobG9nRkMpID4gMS4wKSAlPiUKICBhcnJhbmdlKC1sb2dGQykgJT4lCiAgaGVhZChuID0gMTApCgpsYWJlbHMgPC0gcmJpbmQobGFiZWxzX3VwcywgbGFiZWxzX2Rvd25zKQoKbGFiZWxzX3JlcXVlc3RlZCA8LSBjKCJDZGgxMCIsIkNkaDEyIiwiQ2RoMTMiLCJDZGgxOCIsCiAgICAgICAgICAgICAgICAgICAgICAiQ2RoNyIsIkNkaDgiLCJDZGg5IiwiQ250bjMiLAogICAgICAgICAgICAgICAgICAgICAgIkNudG40IiwiQ250bjUiLCJDbnRuNiIsIktpcnJlbDMiLAogICAgICAgICAgICAgICAgICAgICAgIk5yeG4xIiwiTnJ4bjMiLCJTZW1hM2MiLCJTZW1hNmQiLAogICAgICAgICAgICAgICAgICAgICAgIlRlbm0xIiwiVGVubTIiLCJUZW5tNCIpCnJlc190YmwgPC0gZGYKREVwbG90IDwtIGdncGxvdChyZXNfdGJsLCBhZXMoeCA9IGxvZ0ZDLCB5ID0gLWxvZzEwKGFkai5QLlZhbCksIGxhYmVsID0gZXh0ZXJuYWxfZ2VuZV9uYW1lKSkgKwogIGdlb21fcG9pbnQoYWVzKGNvbG91ciA9IFNpZ25pZmljYW5jZSksIHNpemUgPSA0KSArCiAgZ2VvbV92bGluZSh4aW50ZXJjZXB0ID0gYygtMSwgMSkpICsKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSAtbG9nMTAoMC4wNSkpICsKICB0aGVtZV9jbGFzc2ljKGJhc2Vfc2l6ZSA9IDIwKSArCiAgeGxhYigibG9nMihGQykiKSArCiAgeWxhYigiLWxvZzEwKHAtdmFsdWUpIikgKwogICMjIGdndGl0bGUodGl0bGUsIHN1YnRpdGxlID0gc3VidGl0bGUpICsKICB0aGVtZShsZWdlbmQucG9zaXRpb249InJpZ2h0IikgKwogIHNjYWxlX2NvbG9yX21hbnVhbCh2YWx1ZXM9YygiR3JleSIsICIjRjg3NjZEIiwgIiMwMEJGQzQiKSkgKwogIGdlb21fbGFiZWxfcmVwZWwoZGF0YT1maWx0ZXIoZGYsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBleHRlcm5hbF9nZW5lX25hbWUgJWluJSBsYWJlbHNfcmVxdWVzdGVkKSwKICAgICAgICAgICAgICAgICAgICMjIGMobGFiZWxzJGV4dGVybmFsX2dlbmVfbmFtZSwgIk9wbjQiKSksICNjKCdzNV9oZXRfZGxnbicsICdzNV9oZXRfcmV0JywgJ3M1X2hldF9zY24nKSksCiAgICAgICAgICAgICAgICAgICAjIyBudWRnZV94ID0gLTAuNSwKICAgICAgICAgICAgICAgICAgIG51ZGdlX3kgPSAxNSwgbWF4Lm92ZXJsYXBzID0gMjUpCgojcHAoZmlsZSA9ICJheG9uVFJBUF9Wb2xjYW5vcGxvdHNfMjAyNDAyMDIvcDA4X3JldGluYXZzc2NuaGV0X0RFX3JlcXVlc3RlZF9nZW5lbGFiZWxzXzAyMDUyMDI0LnBkZiIpCkRFcGxvdAojZGV2Lm9mZigpCmBgYAoKIyMgSG93IG1hbnkgdXBzL2Rvd25zCgpgYGB7ciwgZXZhbD1GQUxTRX0Kc2NuX2VucmljaGVkIDwtIGRmICU+JQogIGZpbHRlcihhZGouUC5WYWwgPD0gMC4wNSAmIGxvZ0ZDID49IDEuMCkgJT4lCiAgYXJyYW5nZSgtbG9nRkMpICU+JQogIHNlbGVjdChSb3cubmFtZXMsIGV4dGVybmFsX2dlbmVfbmFtZSwgbG9nRkMsIGFkai5QLlZhbCwgU2lnbmlmaWNhbmNlKQoKcmV0aW5hX2VucmljaGVkIDwtIGRmICU+JQogIGZpbHRlcihhZGouUC5WYWwgPD0gMC4wNSAmIGxvZ0ZDIDw9IC0xLjApICU+JQogIGFycmFuZ2UobG9nRkMpICAlPiUKICBzZWxlY3QoUm93Lm5hbWVzLCBleHRlcm5hbF9nZW5lX25hbWUsIGxvZ0ZDLCBhZGouUC5WYWwsIFNpZ25pZmljYW5jZSkKCnNjbl9lbnJpY2hlZApyZXRpbmFfZW5yaWNoZWQKCmRmICU+JQogIGZpbHRlcihTaWduaWZpY2FuY2UgPT0gIk5vdFxuIEVucmljaGVkIikKYGBgCgojIyBHU0VBCgpgYGB7ciwgZXZhbD1GQUxTRX0KZ3NlYV9yZXN1bHRfc2NuIDwtIGdvc3QocXVlcnkgPSBzY25fZW5yaWNoZWQkZXh0ZXJuYWxfZ2VuZV9uYW1lLAogICAgICAgICAgICAgICAgICAgICAgICBvcmdhbmlzbSA9ICJtbXVzY3VsdXMiLCBldmNvZGVzID0gVFJVRSwKICAgICAgICAgICAgICAgICAgICAgICAgb3JkZXJlZF9xdWVyeSA9IFRSVUUsIHNvdXJjZSA9IGMoIkdPIikpCgpnc2VhX3Jlc3VsdF9yZXQgPC0gZ29zdChxdWVyeSA9IHJldGluYV9lbnJpY2hlZCRleHRlcm5hbF9nZW5lX25hbWUsCiAgICAgICAgICAgICAgICAgICAgICAgIG9yZ2FuaXNtID0gIm1tdXNjdWx1cyIsIGV2Y29kZXMgPSBUUlVFLAogICAgICAgICAgICAgICAgICAgICAgICBvcmRlcmVkX3F1ZXJ5ID0gVFJVRSwgc291cmNlID0gYygiR08iKSkKYGBgCgpgYGB7ciwgZXZhbD1GQUxTRX0KZ3NlYV9zY24gPC0gIGdzZWFfcmVzdWx0X3NjbltbInJlc3VsdCJdXSAlPiUKICBzZWxlY3QodGVybV9uYW1lLCBwX3ZhbHVlLCB0ZXJtX3NpemUsIGludGVyc2VjdGlvbl9zaXplLCByZWNhbGwsIHNvdXJjZSwgaW50ZXJzZWN0aW9uKSAlPiUKICBhcnJhbmdlKGRlc2MocmVjYWxsKSkgJT4lCiAgaGVhZChuID0gMjApCmdzZWFfcGxvdHNfc2NuIDwtIGdncGxvdChnc2VhX3NjbiwgYWVzKHggPSByZWNhbGwsIHkgPSByZW9yZGVyKHRlcm1fbmFtZSwgcmVjYWxsKSwgZmlsbCA9IHBfdmFsdWUpKSArCiAgZ2VvbV9iYXIoc3RhdCA9ICJpZGVudGl0eSIpICsKICBzY2FsZV9maWxsX2NvbnRpbnVvdXMobG93ID0gImJsdWUiLCBoaWdoID0gInJlZCIpICsKICB0aGVtZV9idygpICsKICB5bGFiKCIiKSArCiAgeGxhYigiT3ZlciBlbnJpY2htZW50IFNjb3JlIikKCnBwKGZpbGUgPSAiaW1hZ2VzL0dTRUFfU0NOaGV0X3ZzX3JldGluYV9lbnJpY2hlZF9QMDgucGRmIikKZ3NlYV9wbG90c19zY24KcGxvdHRlZCA8LSBkZXYub2ZmKCkKCmdzZWFfcmV0IDwtICBnc2VhX3Jlc3VsdF9yZXRbWyJyZXN1bHQiXV0gJT4lCiAgc2VsZWN0KHRlcm1fbmFtZSwgcF92YWx1ZSwgdGVybV9zaXplLCBpbnRlcnNlY3Rpb25fc2l6ZSwgcmVjYWxsLCBzb3VyY2UsIGludGVyc2VjdGlvbikgJT4lCiAgYXJyYW5nZShkZXNjKHJlY2FsbCkpICU+JQogIGhlYWQobiA9IDIwKQpnc2VhX3Bsb3RzX3JldCA8LSBnZ3Bsb3QoZ3NlYV9yZXQsIGFlcyh4ID0gcmVjYWxsLCB5ID0gcmVvcmRlcih0ZXJtX25hbWUsIHJlY2FsbCksIGZpbGwgPSBwX3ZhbHVlKSkgKwogIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiKSArCiAgc2NhbGVfZmlsbF9jb250aW51b3VzKGxvdyA9ICJibHVlIiwgaGlnaCA9ICJyZWQiKSArCiAgdGhlbWVfYncoKSArCiAgeWxhYigiIikgKwogIHhsYWIoIk92ZXIgZW5yaWNobWVudCBTY29yZSIpCgpwcChmaWxlID0gImltYWdlcy9HU0VBX1JldGluYWhldF92c19TQ05fZW5yaWNoZWRfUDA4LnBkZiIpCmdzZWFfcGxvdHNfcmV0CnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmBgYAoKIyBLTyBSZXRpbmEgdnMgS08gU0NOCgpgYGB7ciwgZXZhbD1GQUxTRX0KbW0zOF9zdWJzZXQzIDwtIHN1YnNldF9zZSgKICBtbTM4X2hpc2F0LAogIHN1YnNldCA9ICIoYmF0Y2ggPT0gJzQnIHwgYmF0Y2ggPT0gJzUnIHwgYmF0Y2ggPT0gJzYnKSAmIHRpbWUgPT0gJ3AwOCcgICYgZ2Vub3R5cGUgIT0gJ2hldCcgJiBsb2NhdGlvbiAhPSAnZGxnbicgfCBzYW1wbGVpZCA9PSAnaXByZ2NfMDMnIikKCm1tMzhfc3Vic2V0MyA8LSBzdWJzZXRfc2UobW0zOF9zdWJzZXQzLCBzdWJzZXQgPSAic2FtcGxlaWQgIT0gJ2lwcmdjXzg2JyIpCm1tMzhfc3Vic2V0MyRkZXNpZ24gJT4lCiAgc2VsZWN0KGdlbm90eXBlLCBsb2NhdGlvbikgJT4lCiAgdGFibGUoKQptbTM4X25vcm0zIDwtIG5vcm1hbGl6ZShtbTM4X3N1YnNldDMsIGZpbHRlcj1UUlVFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnZlcnQ9ImNwbSIsIHRyYW5zZm9ybT0ibG9nMiIsIGJhdGNoID0gInN2YXNlcSIpCmBgYAoKIyMgUENBCgpgYGB7ciwgZXZhbD1GQUxTRX0KbW0zOF9ub3JtMyA8LSBzZXRfYmF0Y2hlcyhtbTM4X25vcm0zLCBmYWN0ID0gImxvY2F0aW9uIikKbW0zOF9ub3JtMyA8LSBzZXRfY29uZGl0aW9ucyhtbTM4X25vcm0zLCBmYWN0ID0gImdlbm90eXBlIikKcGNhX25vcm0zIDwtIHBsb3RfcGNhKG1tMzhfbm9ybTMsIG1heF9vdmVybGFwcyA9IDcwKQpwY2Ffbm9ybTMkcGxvdApgYGAKCiMjIERFCgpgYGB7ciwgZXZhbD1GQUxTRX0KbW1fZGVfc3Vic2V0MyA8LSBhbGxfcGFpcndpc2UobW0zOF9zdWJzZXQzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtb2RlbF9iYXRjaD0ic3Zhc2VxIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcGFyYWxsZWw9RkFMU0UsIGRvX2Vic2VxPUZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb19iYXNpYyA9IEZBTFNFLCBkb19kcmVhbSA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb19ub2lzZXEgPSBGQUxTRSwgZG9fZWRnZXIgPSBGQUxTRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmlsdGVyID0gVFJVRSkKCnJldGluYWtlZXBlcl9nZW5lcyA8LSBtbV9kZV9zdWJzZXQzJGRlc2VxJGFsbF90YWJsZXMkd3RyZXRpbmFfdnNfa29yZXRpbmEgJT4lCiAgZmlsdGVyKGxvZ0ZDIDw9IC0xLjAgJiBhZGouUC5WYWwgPD0gMC4wNSkKCnNjbmtlZXBlcl9nZW5lcyA8LSBtbV9kZV9zdWJzZXQzJGRlc2VxJGFsbF90YWJsZXMkd3RzY25fdnNfa29zY24gJT4lCiAgZmlsdGVyKGxvZ0ZDIDw9IC0xLjAgJiBhZGouUC5WYWwgPD0gMC4wNSkKCmtlZXBlcmdlbmVzIDwtIHVuaXF1ZShjKHJvd25hbWVzKHJldGluYWtlZXBlcl9nZW5lcyksIHJvd25hbWVzKHNjbmtlZXBlcl9nZW5lcykpKQoKYW5ub3RzX3RvX21lcmdlIDwtIG1tX2Fubm90ICU+JQogIHNlbGVjdChlbnNlbWJsX2dlbmVfaWQsIGV4dGVybmFsX2dlbmVfbmFtZSkgJT4lCiAgZmlsdGVyKGVuc2VtYmxfZ2VuZV9pZCAlaW4lIHJvd25hbWVzKG1tX2RlX3N1YnNldDMkZGVzZXEkYWxsX3RhYmxlcyRrb19zY25fdnNfa29fcmV0aW5hKSkgJT4lCiAgZGlzdGluY3QoKQoKbW1fZGVfc3Vic2V0MyRkZXNlcSRhbGxfdGFibGVzJGtvX3Njbl92c19rb19yZXRpbmEgPC0gbWVyZ2UoCiAgbW1fZGVfc3Vic2V0MyRkZXNlcSRhbGxfdGFibGVzJGtvX3Njbl92c19rb19yZXRpbmEsCiAgYW5ub3RzX3RvX21lcmdlLCBieS54ID0gMCwKICBieS55ID0gImVuc2VtYmxfZ2VuZV9pZCIsIGFsbC54ID0gVFJVRSkKCmRmIDwtIG1tX2RlX3N1YnNldDMkZGVzZXEkYWxsX3RhYmxlcyRrb19zY25fdnNfa29fcmV0aW5hICU+JQogIG11dGF0ZShTaWduaWZpY2FuY2UgPSBjYXNlX3doZW4obG9nRkMgPD0gLTEgfiAiUmV0aW5hIEVucmljaGVkIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxvZ0ZDID49IDEgfiAiU0NOIEVucmljaGVkIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxvZ0ZDID4gLTEgJiBsb2dGQyA8IDEgfiAiTm90XG4gRW5yaWNoZWQiKSkKCmRmIDwtIGRmICU+JQogIGZpbHRlcihSb3cubmFtZXMgJWluJSBrZWVwZXJnZW5lcykKCnNjbl9lbnJpY2hlZCA8LSBkZiAlPiUKICBmaWx0ZXIoYWRqLlAuVmFsIDw9IDAuMDUgJiBsb2dGQyA+PSAxKSAlPiUKICBhcnJhbmdlKC1sb2dGQykgJT4lCiAgc2VsZWN0KFJvdy5uYW1lcywgZXh0ZXJuYWxfZ2VuZV9uYW1lLCBsb2dGQywgYWRqLlAuVmFsKSAlPiUKICBtdXRhdGUoU2lnbmlmaWNhbmNlID0gIlNDTiBFbnJpY2hlZCIpICU+JQogIGZpbHRlcihSb3cubmFtZXMgJWluJSByb3duYW1lcyhzY25rZWVwZXJfZ2VuZXMpKQoKZGYgJT4lCiAgZmlsdGVyKGFkai5QLlZhbCA8PSAwLjA1ICYgbG9nRkMgPD0gLTEpICU+JQogIGFycmFuZ2UobG9nRkMpICAlPiUKICBzZWxlY3QoUm93Lm5hbWVzLCBleHRlcm5hbF9nZW5lX25hbWUsIGxvZ0ZDLCBhZGouUC5WYWwpICU+JQogIG11dGF0ZShTaWduaWZpY2FuY2UgPSAiUmV0aW5hIEVucmljaGVkIikgJT4lCiAgZmlsdGVyKFJvdy5uYW1lcyAlaW4lIHJvd25hbWVzKHJldGluYWtlZXBlcl9nZW5lcykpIC0+IHJldGluYV9lbnJpY2hlZAoKbm90ZW5yaWNoZWQgPC0gZGYgJT4lCiAgc2VsZWN0KFJvdy5uYW1lcywgZXh0ZXJuYWxfZ2VuZV9uYW1lLCBsb2dGQywgYWRqLlAuVmFsLCBTaWduaWZpY2FuY2UpICU+JQogIGZpbHRlcihSb3cubmFtZXMgJWluJSBjKHJvd25hbWVzKHJldGluYWtlZXBlcl9nZW5lcyksCiAgICAgICAgICAgICAgICAgICAgICAgICAgcm93bmFtZXMoc2Nua2VlcGVyX2dlbmVzKSlbZHVwbGljYXRlZChjKHJvd25hbWVzKHJldGluYWtlZXBlcl9nZW5lcyksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJvd25hbWVzKHNjbmtlZXBlcl9nZW5lcykpKV0pCgpkZiA8LSByYmluZChzY25fZW5yaWNoZWQsIHJldGluYV9lbnJpY2hlZCwgbm90ZW5yaWNoZWQpCmBgYAoKYGBge3IsIGV2YWw9RkFMU0V9CmxhYmVsc191cHMgPC0gZGYgJT4lCiAgZmlsdGVyKGFkai5QLlZhbCA8PSAwLjA1ICYgYWJzKGxvZ0ZDKSA+IDEpICU+JQogIGFycmFuZ2UobG9nRkMpICU+JQogIGhlYWQobiA9IDEwKQoKbGFiZWxzX2Rvd25zIDwtIGRmICU+JQogIGZpbHRlcihhZGouUC5WYWwgPD0gMC4wNSAmIGFicyhsb2dGQykgPiAxKSAlPiUKICBhcnJhbmdlKC1sb2dGQykgJT4lCiAgaGVhZChuID0gMTApCgpsYWJlbHMgPC0gcmJpbmQobGFiZWxzX3VwcywgbGFiZWxzX2Rvd25zKQojIyB3YW50ZWRfY29sdW1uIDwtICJTaWduaWZpY2FuY2UiCgpyZXNfdGJsIDwtIGRmCkRFcGxvdCA8LSBnZ3Bsb3QocmVzX3RibCwgYWVzKHggPSBsb2dGQywgeSA9IC1sb2cxMChhZGouUC5WYWwpLCBsYWJlbCA9IGV4dGVybmFsX2dlbmVfbmFtZSkpICsKICBnZW9tX3BvaW50KGFlcyhjb2xvdXIgPSBTaWduaWZpY2FuY2UpLCBzaXplID0gNCkgKwogICMjIGdlb21fcG9pbnQoYWVzKGNvbG91ciA9ICEhc3ltKHdhbnRlZF9jb2x1bW4pKSwgc2l6ZSA9IDQpICsKICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBjKC0xLCAxKSkgKwogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IC1sb2cxMCgwLjA1KSkgKwogIHRoZW1lX2NsYXNzaWMoYmFzZV9zaXplID0gMjApICsKICB4bGFiKCJsb2cyKEZDKSIpICsKICB5bGFiKCItbG9nMTAocC12YWx1ZSkiKSArCiAgIyMgZ2d0aXRsZSh0aXRsZSwgc3VidGl0bGUgPSBzdWJ0aXRsZSkgKwogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpICsKICBzY2FsZV9jb2xvcl9tYW51YWwodmFsdWVzID0gYygiR3JleSIsICIjRjg3NjZEIiwgIiMwMEJGQzQiKSkgKwogIGdlb21fbGFiZWxfcmVwZWwoZGF0YSA9IGZpbHRlcigKICAgIGRmLCBleHRlcm5hbF9nZW5lX25hbWUgJWluJSBjKGxhYmVscyRleHRlcm5hbF9nZW5lX25hbWUsICJPcG40IikpLAogICAgIyMgYygnczVfaGV0X2RsZ24nLCAnczVfaGV0X3JldCcsICdzNV9oZXRfc2NuJykpLAogICAgIyMgbnVkZ2VfeCA9IC0wLjUsCiAgICBudWRnZV95ID0gMTAsIG1heC5vdmVybGFwcyA9IDI1KQoKcHAoZmlsZSA9ICJpbWFnZXMvcDA4X3JldGluYXZzc2Nua29fREVfMTMxMjAyNC5wZGYiKQpERXBsb3QKcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgojIyBIb3cgbWFueSB1cHMvZG93bnMKCmBgYHtyLCBldmFsPUZBTFNFfQpzY25fZW5yaWNoZWQKcmV0aW5hX2VucmljaGVkCm5vdGVucmljaGVkICU+JQogIGZpbHRlcihTaWduaWZpY2FuY2UgPT0gIk5vdFxuIEVucmljaGVkIikKYGBgCgojIyBHU0VBCgpgYGB7ciwgZXZhbD1GQUxTRX0KZ3NlYV9yZXN1bHRfc2NuIDwtIGdvc3QocXVlcnkgPSBzY25fZW5yaWNoZWQkZXh0ZXJuYWxfZ2VuZV9uYW1lLAogICAgICAgICAgICAgICAgICAgICAgICBvcmdhbmlzbSA9ICJtbXVzY3VsdXMiLAogICAgICAgICAgICAgICAgICAgICAgICBldmNvZGVzID0gVFJVRSwKICAgICAgICAgICAgICAgICAgICAgICAgb3JkZXJlZF9xdWVyeSA9IFRSVUUsCiAgICAgICAgICAgICAgICAgICAgICAgIHNvdXJjZSA9IGMoIkdPIikpCgpnc2VhX3Jlc3VsdF9yZXQgPC0gZ29zdChxdWVyeSA9IHJldGluYV9lbnJpY2hlZCRleHRlcm5hbF9nZW5lX25hbWUsCiAgICAgICAgICAgICAgICAgICAgICAgIG9yZ2FuaXNtID0gIm1tdXNjdWx1cyIsCiAgICAgICAgICAgICAgICAgICAgICAgIGV2Y29kZXMgPSBUUlVFLAogICAgICAgICAgICAgICAgICAgICAgICBvcmRlcmVkX3F1ZXJ5ID0gVFJVRSwKICAgICAgICAgICAgICAgICAgICAgICAgc291cmNlID0gYygiR08iKSkKYGBgCgpgYGB7ciwgZXZhbD1GQUxTRX0KZ3NlYV9zY24gPC0gIGdzZWFfcmVzdWx0X3NjbltbInJlc3VsdCJdXSAlPiUKICBzZWxlY3QodGVybV9uYW1lLCBwX3ZhbHVlLCB0ZXJtX3NpemUsIGludGVyc2VjdGlvbl9zaXplLCByZWNhbGwsIHNvdXJjZSwgaW50ZXJzZWN0aW9uKSAlPiUKICBhcnJhbmdlKGRlc2MocmVjYWxsKSkgJT4lCiAgaGVhZChuID0gMjApCmdzZWFfcGxvdHNfc2NuIDwtIGdncGxvdChnc2VhX3NjbiwgYWVzKHggPSByZWNhbGwsIHkgPSByZW9yZGVyKHRlcm1fbmFtZSwgcmVjYWxsKSwgZmlsbCA9IHBfdmFsdWUpKSArCiAgZ2VvbV9iYXIoc3RhdCA9ICJpZGVudGl0eSIpICsKICBzY2FsZV9maWxsX2NvbnRpbnVvdXMobG93ID0gImJsdWUiLCBoaWdoID0gInJlZCIpICsKICB0aGVtZV9idygpICsKICB5bGFiKCIiKSArCiAgeGxhYigiR1NFQSBTY29yZSIpCgpwcChmaWxlID0gImltYWdlcy9HU0VBX1NDTmtvX2VucmljaGVkX3ZzX3JldGluYV9QMDgucGRmIikKZ3NlYV9wbG90c19zY24KcGxvdHRlZCA8LSBkZXYub2ZmKCkKCmdzZWFfcmV0IDwtICBnc2VhX3Jlc3VsdF9yZXRbWyJyZXN1bHQiXV0gJT4lCiAgc2VsZWN0KHRlcm1fbmFtZSwgcF92YWx1ZSwgdGVybV9zaXplLCBpbnRlcnNlY3Rpb25fc2l6ZSwgcmVjYWxsLCBzb3VyY2UsIGludGVyc2VjdGlvbikgJT4lCiAgYXJyYW5nZShkZXNjKHJlY2FsbCkpICU+JQogIGhlYWQobiA9IDIwKQpnc2VhX3Bsb3RzX3JldCA8LSBnZ3Bsb3QoZ3NlYV9yZXQsIGFlcyh4ID0gcmVjYWxsLCB5ID0gcmVvcmRlcih0ZXJtX25hbWUsIHJlY2FsbCksIGZpbGwgPSBwX3ZhbHVlKSkgKwogIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiKSArCiAgc2NhbGVfZmlsbF9jb250aW51b3VzKGxvdyA9ICJibHVlIiwgaGlnaCA9ICJyZWQiKSArCiAgdGhlbWVfYncoKSArCiAgeWxhYigiIikgKwogIHhsYWIoIkdTRUEgU2NvcmUiKQoKcHAoZmlsZSA9ICJpbWFnZXMvR1NFQV9SZXRpbmFrb19lbnJpY2hlZF92c19TQ05fUDA4LnBkZiIpCmdzZWFfcGxvdHNfcmV0CnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmBgYAoKIyBNeSB2ZXJzaW9uIG9mIHRoZSBnbG9iYWwgYW5hbHlzaXMKCkkgd2FudCB0byBoYXZlIGFuIGludm9jYXRpb24gb2YgYWxsX3BhaXJ3aXNlKCkgd2hpY2ggdXNlcyBhbGwgc2FtcGxlcywKaW4gdGhlIGZvbGxvd2luZyBibG9jayBJIHdpbGwgc2V0IHRoYXQgdXAgdXNpbmcgYSBzZXQgb2YgJ2tlZXBlcnMnCndoaWNoIHdpbGwgYmUgbmFtZWQgYnkgdGltZSwgbG9jYXRpb24sIHRoZW4gMiBsZXR0ZXJzIGZvciB0aGUKbnVtZXJhdG9yL2Rlbm9taW5hdG9yOiB3IGZvciBXVCwgaCBmb3IgaGV0LCBkIGZvciBkZWx0YTsgdGh1cwoicDA4X3JldGluYV9odyIgaXMgY29tcGFyaW5nIHRoZSBoZXQvd3QgZm9yIHRoZSBwMDggcmV0aW5hIHNhbXBsZXMuCgpJZiB0aGV5IGFyZSBvZiBpbnRlcmVzdCwgSSB3aWxsIGhhdmUgYSBzZXBhcmF0ZSBzZXQgd2hpY2ggZm9sbG93cyB0aGUKc2FtZSBjb252ZW50aW9uIHdpdGggbmFtZXMgbGlrZSAicDA4X2tvX3NyIiB0byBjb21wYXJlIHAwOCBkZWx0YXMgd2l0aApTQ04gYXMgdGhlIG51bWVyYXRvciBhbmQgcmV0aW5hIGFzIHRoZSBkZW5vbWluYXRvci4KCiMjIFNldCB1cCB0aGUgZXhjbHVzaW9uIGRhdGFzZXQKClRoZSBtb3N0IHBlY3VsaWFyIGFzcGVjdCBvZiB0aGlzIGFuYWx5c2lzIHJlc2lkZXMgaW4gdGhlIGNob2ljZXMKYXJvdW5kIGNob29zaW5nIHdoaWNoIGdlbmVzIHRvIGNvbnNpZGVyIHdoZW4gY29tcGFyaW5nIHRoZQpnZW5vdHlwZXMvbG9jYXRpb25zL3RpbWVzLiAgVGhlIGdlbmVyYWwgaWRlYSBpcyBwcmV0dHkgY2xlYXI6IGZpbmQgdGhlCmdlbmVzIHdoaWNoIGFyZSBub24tc3BlY2lmaWNhbGx5IGJlaW5nIHB1bGxlZCBkb3duIGluIHRoZSBXVCBzYW1wbGVzCmFuZCBlaXRoZXIgZXhjbHVkZSBvciBkaXNjb3VudCB0aGVtLiAgVGhlIHZhcmlvdXMgcG90ZW50aWFsIG1ldGhvZHMKZm9yIHBlcmZvcm1pbmcgdGhpcyBhcmUgY29uZnVzaW5nOgoKMS4gIFdoaWNoIHNldCBvZiBjb21wYXJpc29ucyBvZiB3dC9rbyB3dC9oZXQgZG8gd2UgdXNlIHRvCiAgICBleGNsdWRlL2Rpc2NvdW50IGdlbmVzPwogICAgYS4gIFNob3VsZCBpdCBiZSBhIGNvbWJpbmF0aW9uIG9mIGFsbCBzYW1wbGVzIHd0IHZzLiB4PwogICAgYi4gIFNob3VsZCBpdCBiZSBvbmx5IHRoZSAncmVsZXZhbnQnIGNvbXBhcmlzb24sIGUuZy4gaWYgd2UgYXJlCiAgICBjb21wYXJpbmcgcDA4X2RsZ25faGV0IHZzLiBwMDhfc2NuX2hldDsgZG8gd2UgcmVtb3ZlIGdlbmVzCiAgICBvYnNlcnZlZCBpbiAocDA4X2RsZ25faGV0L3d0ICYmIHAwOF9zY25faGV0L3d0KQoyLiAgRG8gd2UgaW5zdGVhZCBhdHRlbXB0IHRvIHVzZSB0aGlzIHgvd3QgaW5mb3JtYXRpb24gdG8gbm9ybWFsaXplCiAgICB0aGUgZXhwcmVzc2lvbiB2YWx1ZXMgaW4gdGhlIG90aGVyIGNvbmRpdGlvbnMgYW5kIGtlZXAgdGhvc2UKICAgIGdlbmVzPwoKVGhlcmVzYSdzIGN1cnJlbnQgd29ya3NoZWV0IGltcGxlbWVudHMgYSB2ZXJzaW9uIG9mIDFiIGluIHdoaWNoIHNoZQpzZXBhcmF0ZWQgdGhlIHZhcmlvdXMgaW5wdXQgZ2VuZSBzZXRzIHRvIGRlZmluZSB0aGUgZXhjbHVzaW9uIGdlbmVzLgpJIGFtIGdvaW5nIHRvIHJlcGVhdCB0aGlzLCBidXQgbGVhdmUgdGhlIHN0YXJ0aW5nIGRhdGEgc3RydWN0dXJlCmludGFjdC4KCkluIHRoaXMgZmlyc3QgaXRlcmF0aW9uLCBJIHdpbGwgZG8gdGhhdCBieSBjcmVhdGluZyBhIHNpbXBsaWZpZWQgbW9kZWwKb2YgdGhlIGRhdGEgd2hpY2ggY29tYmluZXMgdGhlIHRpbWUvZ2Vub3R5cGUvbG9jYXRpb24gYW5kIHVzaW5nIHN2YS4KSW4gbXkgbmV4dCBpdGVyYXRpb24gSSB3aWxsIHVzZSBhIGZ1bGwgc3RhdGlzdGljYWwgbW9kZWwgY29udGFpbmluZwplYWNoIG9mIHRob3NlIGZhY3RvcnMgKGFuZCBwcm9iYWJseSBhbHNvIHVzaW5nIHN2YSkuCgpOb3RlOiBteSBjb2xvciBjaG9pY2VzIGFyZSBraW5kIG9mIGdhcmJhZ2UuCgpJbiBhZGRpdGlvbiwgdGhlIGV4Y2x1c2lvbiBkYXRhc2V0IGlzIHRoZSBzYW1lIGFzIHRoZSBhbmFseXNpcwpkYXRhc2V0LCBpdCBpcyByZWFsbHkgb25seSB0aGUgY29udHJhc3RzIHdoaWNoIHdpbGwgYmUgZGlmZmVyZW50LgoKYGBge3J9CnYzX3BhaXJ3aXNlX2lucHV0IDwtIHNldF9jb25kaXRpb25zKG1tMzhfaGlzYXRfdjMsIGZhY3QgPSAidGltZV9nZW5vX2xvYyIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG9ycyA9IGNvbG9yX2Nob2ljZXNbWyJhbGwiXV0pCmBgYAoKIyMgQSBoZWF0bWFwIG9mIHNwZWNpZmljIGdlbmVzIGFjcm9zcyBhbGwgY29uZGl0aW9ucwoKYGBge3J9CmFsbF9jb25kX2dlbmVfaGVhdG1hcF9zdGFydCA8LSBub3JtYWxpemUodjNfcGFpcndpc2VfaW5wdXQsIGZpbHRlciA9ICJzaW1wbGUiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxlbmd0aF9jb2x1bW4gPSAiY2RzX2xlbmd0aCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29udmVydCA9ICJycGttIiwgdHJhbnNmb3JtID0gImxvZzIiKQphbGxfY29uZF9nZW5lX2hlYXRtYXBfaW5wdXQgPC0gbWVkaWFuX2J5X2ZhY3RvcihhbGxfY29uZF9nZW5lX2hlYXRtYXBfc3RhcnQpCmFsbF9jb25kX210cnggPC0gYWxsX2NvbmRfZ2VuZV9oZWF0bWFwX2lucHV0W1sibWVkaWFucyJdXQpjb2xvcl9vcmRlciA8LSBjb2xuYW1lcyhhbGxfY29uZF9tdHJ4KQpuYV9pZHggPC0gaXMubmEoYWxsX2NvbmRfbXRyeCkKYWxsX2NvbmRfbXRyeFtuYV9pZHhdIDwtIDAKdmFyaWFuY2VzIDwtIG1hdHJpeFN0YXRzOjpyb3dWYXJzKGFzLm1hdHJpeChhbGxfY29uZF9tdHJ4KSkKdmFyaWFudF9nZW5lcyA8LSB2YXJpYW5jZXMgPiA2LjIKaW5wdXRfbXRyeCA8LSBhbGxfY29uZF9tdHJ4W3ZhcmlhbnRfZ2VuZXMsIF0KY29uZF9jb2xvcnMgPC0gZ2V0X2NvbG9yc19ieV9jb25kaXRpb24odjNfcGFpcndpc2VfaW5wdXQsIGxldmVscyA9IGNvbG9yX29yZGVyKQpkaW0oaW5wdXRfbXRyeCkKcHAoZmlsZSA9ICIwNGluY2x1c2lvbl9jb21wYXJpc29ucy90b3BfMTA0X3ZhcmlhbnRfcnBrbV9nZW5lc19oZWF0bWFwLnBkZiIpCmdwbG90czo6aGVhdG1hcC4yKGFzLm1hdHJpeChpbnB1dF9tdHJ4KSwgc2NhbGUgPSAibm9uZSIsIHRyYWNlID0gIm5vbmUiLAogICAgICAgICAgICAgICAgICBDb2xTaWRlQ29sb3JzID0gY29uZF9jb2xvcnMpCmRldi5vZmYoKQpgYGAKCiMgVE9ETzogQ2hhbmdlIHRoZSBhYm92ZSB0byBzdWJ0cmFjdCB3dAoKUmFzaG1pIHN1Z2dlc3RlZCB3ZSBzaG91bGQgZG8gdGhlIGFib3ZlIHBsb3QgYWZ0ZXIgc3VidHJhY3RpbmcgdGhlIHd0CmNvdW50cy4gIFRoaXMgaXMgYSBnb29kIGlkZWEsIGJ1dCBpdCB3aWxsIGhhdmUgdG8gd2FpdCB1bnRpbCB3ZSBmaW5pc2gKdGhlIGN1cnJlbnQgc2V0LgoKIyMgU2V0IHVwIHRoZSBjb250cmFzdHMKCkluIHRoZSBmb2xsb3dpbmcgZmV3IGJsb2NrcyBJIHdpbGwgc2V0IHVwIHRoZSB2YXJpb3VzIGNvbXBhcmlzb25zIG9mCmludGVyZXN0LiAgU3RhcnRpbmcgd2l0aCB0aGUgc2V0IG9mIGdlbmVzIHRvIGV4Y2x1ZGUgYmVjYXVzZSB0aGV5IHdlcmUKb2JzZXJ2ZWQgdG8gYmluZCBub24tc3BlY2lmaWNhbGx5IGluIHRoZSB3dCBzYW1wbGVzLgoKIyMjIEluY2x1c2lvbiBjb250cmFzdHMKCkluIGVhY2ggZXhjbHVzaW9uIEkgd2lsbCBoYXZlIHRoZSBjb250cmFzdCBmaXJzdCBmb2xsb3dlZCBieSB0aGUgcGFpcgpvZiBjb250cmFzdHMgd2hpY2ggd2lsbCBiZSB1c2VkIHRvIGRlZmluZSB0aGUgZ2VuZSBzZXQgdG8gZXhjbHVkZS4KCiogcDE1X2hldF9kbGduL3AwOF9oZXRfZGxnbjogcDE1X3d0X2RsZ24vcDE1X2hldF9kbGduLAogIHAwOF93dF9kbGduL3AwOF9oZXRfZGxnbjsgcmVtb3ZlIHRoZSBnZW5lcyBpbmNyZWFzZWQgaW4gd3QuCiogcDE1X2tvX3Njbi9wMDhfa29fc2NuOiBwMTVfd3Rfc2NuL3AxNV9rb19zY24sIHAwOF93dF9zY24vcDE1X2tvX3NjbgoqIHAxNV9oZXRfcmV0aW5hL3AwOF9oZXRfcmV0aW5hOiBJIHRoaW5rIHlvdSBnZXQgaXQsIHd0L2hldCBmb3IgYm90aAogIHAxNSByZXRpbmFzIGFuZCBwMDggcmV0aW5hcy4uLgoKUHV0IHNsaWdodGx5IGRpZmZlcmVudGx5LCBmb3IgZXZlcnkgdGVybSBvZiBpbnRlcmVzdCBJIHdpbGwgY3JlYXRlIGEKY29udHJhc3Qgd2l0aCB0aGUgd3QgYXMgbnVtZXJhdG9yIGFuZCB0aGUgZGVzaXJlZCB0ZXJtIGFzIGRlbm9taW5hdG9yLAp0aGVuIHB1bGwgb3V0IHRoZSBnZW5lcyBpbmNyZWFzZWQgaW4gd3QuCgpgYGB7cn0KaW5jbHVzaW9ucyA8LSBsaXN0KAogICMjIEkgbGlrZSBhbHBoYWJldGl6aW5nIHRoaW5ncywgc3RhcnQgd2l0aCBkbGduCiAgInAxNV9oZXRfZGxnbiIgPSBjKCJwMTVfaGV0X2RsZ24iLCAicDE1X3d0X2RsZ24iKSwKICAicDA4X2hldF9kbGduIiA9IGMoInAwOF9oZXRfZGxnbiIsICJwMDhfd3RfZGxnbiIpLAogICJwMTVfa29fZGxnbiIgPSBjKCJwMTVfa29fZGxnbiIsICJwMTVfd3RfZGxnbiIpLAogICJwMDhfa29fZGxnbiIgPSBjKCJwMDhfa29fZGxnbiIsICJwMDhfd3RfZGxnbiIpLAogICMjIFRoZW4gcmV0aW5hcwogICJwMTVfaGV0X3JldGluYSIgPSBjKCJwMTVfaGV0X3JldGluYSIsICJwMTVfd3RfcmV0aW5hIiksCiAgInAwOF9oZXRfcmV0aW5hIiA9IGMoInAwOF9oZXRfcmV0aW5hIiwgInAwOF93dF9yZXRpbmEiKSwKICAicDE1X2tvX3JldGluYSIgPSBjKCJwMTVfa29fcmV0aW5hIiwgInAxNV93dF9yZXRpbmEiKSwKICAicDA4X2tvX3JldGluYSIgPSBjKCJwMDhfa29fcmV0aW5hIiwgInAwOF93dF9yZXRpbmEiKSwKICAjIyBUaGVuIHNjbgogICJwMTVfaGV0X3NjbiIgPSBjKCJwMTVfaGV0X3NjbiIsICJwMTVfd3Rfc2NuIiksCiAgInAwOF9oZXRfc2NuIiA9IGMoInAwOF9oZXRfc2NuIiwgInAwOF93dF9zY24iKSwKICAicDE1X2tvX3NjbiIgPSBjKCJwMTVfa29fc2NuIiwgInAxNV93dF9zY24iKSwKICAicDA4X2tvX3NjbiIgPSBjKCJwMDhfa29fc2NuIiwgInAwOF93dF9zY24iKSkKYGBgCgojIyMgVGltZSBjb250cmFzdHMKCkZvciBlYWNoIGxvY2F0aW9uL2dlbm90eXBlIG9mIGludGVyZXN0LCBsZXQgdXMgY29tcGFyZSBwMTUvcDA4CgpgYGB7cn0KdGltZV9rZWVwZXJzIDwtIGxpc3QoCiAgIyMgRExHTgogICJ0X2hldF9kbGduIiA9IGMoInAxNV9oZXRfZGxnbiIsICJwMDhfaGV0X2RsZ24iKSwKICAidF9rb19kbGduIiA9IGMoInAxNV9rb19kbGduIiwgInAwOF9rb19kbGduIiksCiAgIyMgUmV0aW5hCiAgInRfaGV0X3JldGluYSIgPSBjKCJwMTVfaGV0X3JldGluYSIsICJwMDhfaGV0X3JldGluYSIpLAogICJ0X2tvX3JldGluYSIgPSBjKCJwMTVfa29fcmV0aW5hIiwgInAwOF9rb19yZXRpbmEiKSwKICAjIyBTQ04KICAidF9oZXRfc2NuIiA9IGMoInAxNV9oZXRfc2NuIiwgInAwOF9oZXRfc2NuIiksCiAgInRfa29fc2NuIiA9IGMoInAxNV9rb19zY24iLCAicDA4X2tvX3NjbiIpKQpgYGAKCiMjIyBMb2NhdGlvbiBjb250cmFzdHMKCkNvbXBhcmUgbG9jYXRpb25zIGFuZCBrZWVwIHRpbWUvZ2Vub3R5cGUgY29uc2lzdGVudC4gIEkgd2lsbCB1c2UgdGhlCmxvY2F0aW9uIGluaXRpYWxzIHRvIGRlZmluZSBudW1lcmF0b3IvZGVub21pbmF0b3IuCgpgYGB7cn0KbG9jYXRpb25fa2VlcGVycyA8LSBsaXN0KAogICMjIGRsZ24vcmV0aW5hCiAgImRyX3AwOF9oZXQiID0gYygicDA4X2hldF9kbGduIiwgInAwOF9oZXRfcmV0aW5hIiksCiAgImRyX3AxNV9oZXQiID0gYygicDE1X2hldF9kbGduIiwgInAxNV9oZXRfcmV0aW5hIiksCiAgImRyX3AwOF9rbyIgPSBjKCJwMDhfa29fZGxnbiIsICJwMDhfa29fcmV0aW5hIiksCiAgImRyX3AxNV9rbyIgPSBjKCJwMTVfa29fZGxnbiIsICJwMTVfa29fcmV0aW5hIiksCiAgIyMgc2NuL3JldGluYQogICJzcl9wMDhfaGV0IiA9IGMoInAwOF9oZXRfc2NuIiwgInAwOF9oZXRfcmV0aW5hIiksCiAgInNyX3AxNV9oZXQiID0gYygicDE1X2hldF9zY24iLCAicDE1X2hldF9yZXRpbmEiKSwKICAic3JfcDA4X2tvIiA9IGMoInAwOF9rb19zY24iLCAicDA4X2tvX3JldGluYSIpLAogICJzcl9wMTVfa28iID0gYygicDE1X2tvX3NjbiIsICJwMTVfa29fcmV0aW5hIiksCiAgIyMgZGxnbi9zY24KICAiZHNfcDA4X2hldCIgPSBjKCJwMDhfaGV0X2RsZ24iLCAicDA4X2hldF9zY24iKSwKICAiZHNfcDE1X2hldCIgPSBjKCJwMTVfaGV0X2RsZ24iLCAicDE1X2hldF9zY24iKSwKICAiZHNfcDA4X2tvIiA9IGMoInAwOF9rb19kbGduIiwgInAwOF9rb19zY24iKSwKICAiZHNfcDE1X2tvIiA9IGMoInAxNV9rb19kbGduIiwgInAxNV9rb19zY24iKSkKYGBgCgojIyMgR2Vub3R5cGUgY29udHJhc3RzCgpDb21wYXJlIGtvL2hldCB3aGlsZSBrZWVwaW5nIHRpbWUvbG9jYXRpb24gY29uc3RhbnQuICBTaW1pbGFybHksIHVzZQp0aGUgaW5pdGlhbHMgdG8gZGVub3RlIG51bWVyYXRvci9kZW5vbWluYXRvciwgd2hpY2ggd2lsbCBhbHdheXMgYmUga2guCgpgYGB7cn0KZ2Vub3R5cGVfa2VlcGVycyA8LSBsaXN0KAogICMjIERMR04KICAia2hfcDA4X2RsZ24iID0gYygicDA4X2tvX2RsZ24iLCAicDA4X2hldF9kbGduIiksCiAgImtoX3AxNV9kbGduIiA9IGMoInAxNV9rb19kbGduIiwgInAxNV9oZXRfZGxnbiIpLAogICMjIFJldGluYQogICJraF9wMDhfcmV0aW5hIiA9IGMoInAwOF9rb19yZXRpbmEiLCAicDA4X2hldF9yZXRpbmEiKSwKICAia2hfcDE1X3JldGluYSIgPSBjKCJwMTVfa29fcmV0aW5hIiwgInAxNV9oZXRfcmV0aW5hIiksCiAgIyMgU0NOCiAgImtoX3AwOF9zY24iID0gYygicDA4X2tvX3NjbiIsICJwMDhfaGV0X3NjbiIpLAogICJraF9wMTVfc2NuIiA9IGMoInAxNV9rb19zY24iLCAicDE1X2hldF9zY24iKSkKYGBgCgojIyBQZXJmb3JtIHRoZSBleGNsdXNpb24gY29tcGFyaXNvbgoKTXkgYWxsX3BhaXJ3aXNlKCkgZnVuY3Rpb24gbm93IGhhcyBhIHBhcmFtZXRlciB3aGljaCBhbGxvd3MgbWUgdG8KY2hvb3NlIHdoaWNoIGNvbnRyYXN0cyB0byBwZXJmb3JtIGluc3RlYWQgb2YgbGl0ZXJhbGx5IGRvaW5nIGV2ZXJ5CnBvc3NpYmxlIGNvbXBhcmlzb24uICBUaGF0IGlzIHdlbGwgc3VpdGVkIGZvciB0aGVzZSBvcGVyYXRpb25zOgoKSW4gYSBjb250YWluZXIsIHRoZSBmb2xsb3dpbmcgYXBwZWFycyB0byBmYWlsIHdpdGg6CgoiZXJyb3IgY29kZSAxIGZyb20gTGFwYWNrIHJvdXRpbmUgJ2RnZXNkZCciCgpSdW5uaW5nIGl0IG1hbnVhbGx5IG91dHNpZGUgdGhlIGNvbnRhaW5lciByZXN1bHRzIGluIGl0IHdvcmtpbmcKd2l0aG91dCBlcnJvci4gIEkgYXNzdW1lIHRoZXJlZm9yZSB0aGF0IHRoZSBwcm9ibGVtIGxpZXMgaW4gdGhlCmNvbXBpbGF0aW9uIGZsYWdzIG9mIExBUEFDSyBpbiB0aGUgY29udGFpbmVyLgoKTm90ZTogVGhpcyBwcm9ibGVtIHdhcyBmaXhlZCBieSByZW1vdmluZyBzb21lIHBhcmFsbGVsaXphdGlvbi4KCmBgYHtyfQppbmNsdXNpb25fZGUgPC0gYWxsX3BhaXJ3aXNlKAogIHYzX3BhaXJ3aXNlX2lucHV0LCBmaWx0ZXIgPSAic2ltcGxlIiwgbW9kZWxfZnN0cmluZyA9IGRlZmF1bHRfZnN0cmluZywKICBrZWVwZXJzID0gaW5jbHVzaW9ucywgbW9kZWxfc3ZzID0gInN2YXNlcSIpCmluY2x1c2lvbl9kZQoKaW5jbHVzaW9uX3RhYmxlcyA8LSBjb21iaW5lX2RlX3RhYmxlcygKICBpbmNsdXNpb25fZGUsIGtlZXBlcnMgPSBpbmNsdXNpb25zLCBsYWJlbF9jb2x1bW4gPSBsYWJlbF9jb2x1bW4sCiAgZXhjZWwgPSBnbHVlKCIwNGluY2x1c2lvbl9jb21wYXJpc29ucy9pbmNsdXNpb25fdGFibGVzLXZ7dmVyfS54bHN4IikpCmluY2x1c2lvbl90YWJsZXMKIyMgMjAyNjAzOiBJIHN1Y2Nlc3NmdWxseSByZWNhcGl0dWxhdGVkIHByZXZpb3VzIG5vbi1jb250YWluZXIgcmVzdWx0LgoKaW5jbHVzaW9uX3NpZyA8LSBleHRyYWN0X3NpZ25pZmljYW50X2dlbmVzKAogIGluY2x1c2lvbl90YWJsZXMsIGxmYyA9IGxmY19jdXRvZmYsIHAgPSBhZGpwX2N1dG9mZiwgYWNjb3JkaW5nX3RvID0gImRlc2VxIiwKICBleGNlbCA9IGdsdWUoIjA0aW5jbHVzaW9uX2NvbXBhcmlzb25zL2luY2x1c2lvbl9zaWctdnt2ZXJ9Lnhsc3giKSkKaW5jbHVzaW9uX3NpZwpgYGAKCjIwMjUwNTogQSBzdHJhbmdlIHRoaW5nIGhhcHBlbmVkIGhlcmUgaW4gdGhpcyBpdGVyYXRpb246IHRoZSBwbG90IG9mCnRoZSBzaWduaWZpY2FudCBnZW5lcyBpcyB0aGUgZXhhY3Qgc2FtZSBhcyB0aGUgcHJldmlvdXMgaXRlcmF0aW9uOyBidXQKdGhlIHRhYmxlIG9mIG51bWJlcnMgb2YgZ2VuZXMgbG9va3MgZGlmZmVyZW50LgoKRm9yIGV4YW1wbGUsIHRoZSBwcmV2aW91cyB0YWJsZSBzaG93ZWQ6IHAxNV9oZXRfZGxnbiB3aXRoIDIwNjcgdXAgYW5kCjIzODEgZG93bi4gIFRoZSBwbG90IHNob3dzIGV4YWN0bHkgdGhhdDsgYnV0IHRoZSBuZXcgdGFibGUgc2hvd3MgNjA3CnVwIGFuZCAxMjI5IGRvd24uICBMZXQgdXMgY2hlY2sgdGhlIGFjdHVhbCBkYXRhIHN0cnVjdHVyZSBhbmQgc2VlIHdoYXQKaXMgdXA/CgpJIHRoaW5rIEkgZ2V0IGl0OiB3aGVuIHdlIGRvIHRoZSBleHRyYWN0X3NpZ25pZmljYW50X2dlbmVzIGFib3ZlLCB3ZQpleHBsaWNpdGx5IHNldCBhIG5vbi1zdGFuZGFyZCBwLXZhbHVlIGFuZCBsb2dGQyBiZWNhdXNlIHdlIGFyZQpleHBsaWNpdGx5IGF0dGVtcHRpbmcgdG8gdXNlIGEgdmVyeSBsb29zZSBkZWZpbml0aW9uIG9mIHRoZSBzZXQgb2YKZ2VuZXMgd2hpY2ggYXJlIGluIGdyZWF0ZXIgYWJ1bmRhbmNlIHRoYW4gdGhlaXIgbW9zdCBzaW1pbGFyCndpbGQtdHlwZS4gSG93ZXZlciwgd2hlbiBJIGNyZWF0ZSB0aGUgYmFycGxvdCBvZiBzaWduaWZpY2FudCBnZW5lczsKdGhvc2UgdmFsdWVzIGFyZSBleHBsaWNpdGx5IHNldCB0byAwLDEsMiBsb2dGQ3MgYW5kIHAtdmFsdWUgMC4wNS4KVGhlcmVmb3JlLCB3aGF0IEkgbmVlZCB0byBkbywgaW4gb3JkZXIgdG8gY2hlY2sgY29uc2lzdGVuY3ksIGlzIHRvCnJlcGVhdCB0aGlzIGNhbGwgYnV0IHdpdGggdGhlIGRlZmF1bHQgRkMvcCB2YWx1ZXMgYW5kIHNlZSB3aGF0IHRoZQpudW1iZXJzIGxvb2sgbGlrZS4KCmBgYHtyfQp0ZXN0X2luY2x1c2lvbiA8LSBleHRyYWN0X3NpZ25pZmljYW50X2dlbmVzKAogIGluY2x1c2lvbl90YWJsZXMsIGFjY29yZGluZ190byA9ICJkZXNlcSIsIGV4Y2VsID0gImV4Y2VsL2RlZmF1bHRfaW5jbHVzaW9uX3NpZy54bHN4IikKdGVzdF9pbmNsdXNpb24KYGBgCgpZZWFoLCBJIHRoaW5rIHRoaXMgbWFrZXMgc2Vuc2U7IHdoYXQgSSBuZWVkIHRvIGRvOiBjaGFuZ2UgdGhlCnNpZ25pZmljYW50IGJhciBwbG90IHNvIHRoYXQgaXQgdXNlcyB0aGUgbGZjIGN1dG9mZiBhcmd1bWVudCBhcyB0aGUKc2Vjb25kIG9mIGl0cyAzIGN1dG9mZnMuICBUaGF0IHNob3VsZCBlbnN1cmUgdGhhdCB0aGVzZSBudW1iZXJzIGFyZQpjb25zaXN0ZW50IGFjcm9zcyBhbmFseXNlcyBhbmQgcGFyYW1ldGVycyBwcm92aWRlZC4KCmBgYHtyfQpkaW0oaW5jbHVzaW9uX3NpZyRkZXNlcSR1cHMkcDE1X2hldF9kbGduKQp0ZXN0X2FsbF91cCA8LSBpbmNsdXNpb25fdGFibGVzJGRhdGEkcDE1X2hldF9kbGduW1siZGVzZXFfbG9nZmMiXV0gPiAwLjEgJgogIGluY2x1c2lvbl90YWJsZXMkZGF0YSRwMTVfaGV0X2RsZ25bWyJkZXNlcV9hZGpwIl1dIDw9IDAuMQpzdW1tYXJ5KHRlc3RfYWxsX3VwKQpgYGAKCk9oaCwgSSBnZXQgaXQsIHdoZW4gSSB3YXMgdGVzdGluZyB0aGlzIG91dCBtYW51YWxseSwgSSBzZXQgdGhlIGxvZ0ZDCnRvIDEuMCBpbnN0ZWFkIG9mIHRoZSB2ZXJ5IG1pbmltYWwgMC4xIHdlIGhhdmUgYmVlbiB1c2luZyBmb3IgdGhpcyEKCiMgTUEvVm9sY2FubyBwbG90cyBvZiB4IHZzIHd0CgpSYXNobWkgYXNrZWQgdG8gc2VlIHRoZSBjb21wYXJpc29ucyBhZ2FpbnN0IHd0OyBJIHdpbGwgbmFtZSBlYWNoIGZpbGUKeHcgdG8gc2hvdyB0aGF0IGl0IGlzIHggdnMgd3QuIGZvciB3aGF0ZXZlciBvdGhlciBwYXJhbWV0ZXJzIGFyZSBiZWluZwpleGFtaW5lZC4gIEl0IGlzIGxpa2VseSB0aGF0IHNvbWUgY29sb3JzIHdpbGwgYmUgd3JvbmcgYmVjYXVzZSB0aGlzIGlzCm15IGZpcnN0IHRpbWUgY3JlYXRpbmcgdGhlc2UgcGxvdHMgYW5kIHdlIGFyZSBkb2luZyB0aGVtIG1hbnVhbGx5LgoKIyMgcDE1X2hldF9kbGduCgpgYGB7cn0KYWxsYyA8LSBjb2xvcl9jaG9pY2VzW1siYWxsIl1dCnRhYmxlX25hbWUgPC0gInAxNV9oZXRfZGxnbiIKdGFibGUgPC0gaW5jbHVzaW9uX3RhYmxlc1tbImRhdGEiXV1bW3RhYmxlX25hbWVdXQpudW0gPC0gInAxNV9oZXRfZGxnbiIKZGVub20gPC0gInAxNV93dF9kbGduIgpod19wMTVfZGxnbl92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLCBmaWxsID0gImJsYWNrIiwKICBjb2xvcl9sb3cgPSBhbGxjW1tkZW5vbV1dLCBjb2xvcl9oaWdoID0gYWxsY1tbbnVtXV0sCiAgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IDEwLCBhbHBoYSA9IDEuMCwKICBzaXplID0gNCkKCnBwKGZpbGUgPSAiMDVpbmNsdXNpb25fdm9sY2Fub19tYS9od19wMTVfZGxnbl92b2xjYW5vLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKaHdfcDE1X2RsZ25fdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKaHdfcDE1X2RsZ25fdm9sY2Fub1tbInBsb3QiXV0KCmh3X3AxNV9kbGduX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gYWxsY1tbZGVub21dXSwgY29sb3JfaGlnaCA9IGFsbGNbW251bV1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIG91dGxpbmUgPSBvdXRsaW5lLAogIGxhYmVsID0gMTApCgpwcChmaWxlID0gIjA1aW5jbHVzaW9uX3ZvbGNhbm9fbWEvaHdfcDE1X2RsZ25fbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpod19wMTVfZGxnbl9tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKaHdfcDE1X2RsZ25fbWFbWyJwbG90Il1dCmBgYAoKIyMgcDA4X2hldF9kbGduCgpgYGB7cn0KdGFibGVfbmFtZSA8LSAicDA4X2hldF9kbGduIgp0YWJsZSA8LSBpbmNsdXNpb25fdGFibGVzW1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCm51bSA8LSAicDA4X2hldF9kbGduIgpkZW5vbSA8LSAicDA4X3d0X2RsZ24iCmh3X3AwOF9kbGduX3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGZpbGwgPSAiYmxhY2siLAogIGNvbG9yX2xvdyA9IGFsbGNbW2Rlbm9tXV0sIGNvbG9yX2hpZ2ggPSBhbGxjW1tudW1dXSwKICBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gMTAsIGFscGhhID0gMS4wLAogIHNpemUgPSA0KQpwcChmaWxlID0gIjA1aW5jbHVzaW9uX3ZvbGNhbm9fbWEvaHdfcDA4X2RsZ25fdm9sY2Fuby5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmh3X3AwOF9kbGduX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmh3X3AwOF9kbGduX3ZvbGNhbm9bWyJwbG90Il1dCmh3X3AwOF9kbGduX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gYWxsY1tbZGVub21dXSwgY29sb3JfaGlnaCA9IGFsbGNbW251bV1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIG91dGxpbmUgPSBvdXRsaW5lLAogIGxhYmVsID0gMTApCnBwKGZpbGUgPSAiMDVpbmNsdXNpb25fdm9sY2Fub19tYS9od19wMDhfZGxnbl9tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmh3X3AwOF9kbGduX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpod19wMDhfZGxnbl9tYVtbInBsb3QiXV0KYGBgCgojIyBwMTVfa29fZGxnbgoKYGBge3J9CnRhYmxlX25hbWUgPC0gInAxNV9rb19kbGduIgp0YWJsZSA8LSBpbmNsdXNpb25fdGFibGVzW1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCm51bSA8LSAicDE1X2tvX2RsZ24iCmRlbm9tIDwtICJwMTVfd3RfZGxnbiIKa3dfcDE1X2RsZ25fdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgZmlsbCA9ICJibGFjayIsCiAgY29sb3JfbG93ID0gYWxsY1tbZGVub21dXSwgY29sb3JfaGlnaCA9IGFsbGNbW251bV1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSAxMCwgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQpCnBwKGZpbGUgPSAiMDVpbmNsdXNpb25fdm9sY2Fub19tYS9rd19wMTVfZGxnbl92b2xjYW5vLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKa3dfcDE1X2RsZ25fdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa3dfcDE1X2RsZ25fdm9sY2Fub1tbInBsb3QiXV0Ka3dfcDE1X2RsZ25fbWEgPC0gcGxvdF9tYV9jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGV4cHJfY29sID0gImRlc2VxX2Jhc2VtZWFuIiwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwKICBjb2xvcl9sb3cgPSBhbGxjW1tkZW5vbV1dLCBjb2xvcl9oaWdoID0gYWxsY1tbbnVtXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgb3V0bGluZSA9IG91dGxpbmUsCiAgbGFiZWwgPSAxMCkKcHAoZmlsZSA9ICIwNWluY2x1c2lvbl92b2xjYW5vX21hL2t3X3AxNV9kbGduX21hLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKa3dfcDE1X2RsZ25fbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmt3X3AxNV9kbGduX21hW1sicGxvdCJdXQpgYGAKCiMjIHAwOF9rb19kbGduCgpgYGB7cn0KdGFibGVfbmFtZSA8LSAicDA4X2tvX2RsZ24iCnRhYmxlIDwtIGluY2x1c2lvbl90YWJsZXNbWyJkYXRhIl1dW1t0YWJsZV9uYW1lXV0KbnVtIDwtICJwMDhfa29fZGxnbiIKZGVub20gPC0gInAwOF93dF9kbGduIgprd19wMDhfZGxnbl92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLCBmaWxsID0gImJsYWNrIiwKICBjb2xvcl9sb3cgPSBhbGxjW1tkZW5vbV1dLCBjb2xvcl9oaWdoID0gYWxsY1tbbnVtXV0sCiAgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IDEwLCBhbHBoYSA9IDEuMCwKICBzaXplID0gNCkKcHAoZmlsZSA9ICIwNWluY2x1c2lvbl92b2xjYW5vX21hL2t3X3AwOF9kbGduX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQprd19wMDhfZGxnbl92b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQprd19wMDhfZGxnbl92b2xjYW5vW1sicGxvdCJdXQprd19wMDhfZGxnbl9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGFsbGNbW2Rlbm9tXV0sIGNvbG9yX2hpZ2ggPSBhbGxjW1tudW1dXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBvdXRsaW5lID0gb3V0bGluZSwKICBsYWJlbCA9IDEwKQpwcChmaWxlID0gIjA1aW5jbHVzaW9uX3ZvbGNhbm9fbWEva3dfcDA4X2RsZ25fbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQprd19wMDhfZGxnbl9tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa3dfcDA4X2RsZ25fbWFbWyJwbG90Il1dCmBgYAoKIyMgcDE1X2hldF9yZXRpbmEKCmBgYHtyfQp0YWJsZV9uYW1lIDwtICJwMTVfaGV0X3JldGluYSIKdGFibGUgPC0gaW5jbHVzaW9uX3RhYmxlc1tbImRhdGEiXV1bW3RhYmxlX25hbWVdXQpudW0gPC0gInAxNV9oZXRfcmV0aW5hIgpkZW5vbSA8LSAicDE1X3d0X3JldGluYSIKaHdfcDE1X3JldGluYV92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLCBmaWxsID0gImJsYWNrIiwKICBjb2xvcl9sb3cgPSBhbGxjW1tkZW5vbV1dLCBjb2xvcl9oaWdoID0gYWxsY1tbbnVtXV0sCiAgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IDEwLCBhbHBoYSA9IDEuMCwKICBzaXplID0gNCkKcHAoZmlsZSA9ICIwNWluY2x1c2lvbl92b2xjYW5vX21hL2h3X3AxNV9yZXRpbmFfdm9sY2Fuby5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmh3X3AxNV9yZXRpbmFfdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKaHdfcDE1X3JldGluYV92b2xjYW5vW1sicGxvdCJdXQpod19wMTVfcmV0aW5hX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gYWxsY1tbZGVub21dXSwgY29sb3JfaGlnaCA9IGFsbGNbW251bV1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIG91dGxpbmUgPSBvdXRsaW5lLAogIGxhYmVsID0gMTApCnBwKGZpbGUgPSAiMDVpbmNsdXNpb25fdm9sY2Fub19tYS9od19wMTVfcmV0aW5hX21hLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKaHdfcDE1X3JldGluYV9tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKaHdfcDE1X3JldGluYV9tYVtbInBsb3QiXV0KYGBgCgojIyBwMDhfaGV0X3JldGluYQoKYGBge3J9CnRhYmxlX25hbWUgPC0gInAwOF9oZXRfcmV0aW5hIgp0YWJsZSA8LSBpbmNsdXNpb25fdGFibGVzW1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCm51bSA8LSAicDA4X2hldF9yZXRpbmEiCmRlbm9tIDwtICJwMDhfd3RfcmV0aW5hIgpod19wMDhfcmV0aW5hX3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGZpbGwgPSAiYmxhY2siLAogIGNvbG9yX2xvdyA9IGFsbGNbW2Rlbm9tXV0sIGNvbG9yX2hpZ2ggPSBhbGxjW1tudW1dXSwKICBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gMTAsIGFscGhhID0gMS4wLAogIHNpemUgPSA0KQpwcChmaWxlID0gIjA1aW5jbHVzaW9uX3ZvbGNhbm9fbWEvaHdfcDA4X3JldGluYV92b2xjYW5vLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKaHdfcDA4X3JldGluYV92b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpod19wMDhfcmV0aW5hX3ZvbGNhbm9bWyJwbG90Il1dCmh3X3AwOF9yZXRpbmFfbWEgPC0gcGxvdF9tYV9jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGV4cHJfY29sID0gImRlc2VxX2Jhc2VtZWFuIiwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwKICBjb2xvcl9sb3cgPSBhbGxjW1tkZW5vbV1dLCBjb2xvcl9oaWdoID0gYWxsY1tbbnVtXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgb3V0bGluZSA9IG91dGxpbmUsCiAgbGFiZWwgPSAxMCkKcHAoZmlsZSA9ICIwNWluY2x1c2lvbl92b2xjYW5vX21hL2h3X3AwOF9yZXRpbmFfbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpod19wMDhfcmV0aW5hX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpod19wMDhfcmV0aW5hX21hW1sicGxvdCJdXQpgYGAKCiMjIHAxNV9rb19yZXRpbmEKCmBgYHtyfQp0YWJsZV9uYW1lIDwtICJwMTVfa29fcmV0aW5hIgp0YWJsZSA8LSBpbmNsdXNpb25fdGFibGVzW1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCm51bSA8LSAicDE1X2tvX3JldGluYSIKZGVub20gPC0gInAxNV93dF9yZXRpbmEiCmt3X3AxNV9yZXRpbmFfdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgZmlsbCA9ICJibGFjayIsCiAgY29sb3JfbG93ID0gYWxsY1tbZGVub21dXSwgY29sb3JfaGlnaCA9IGFsbGNbW251bV1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSAxMCwgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQpCnBwKGZpbGUgPSAiMDVpbmNsdXNpb25fdm9sY2Fub19tYS9rd19wMTVfcmV0aW5hX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQprd19wMTVfcmV0aW5hX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmt3X3AxNV9yZXRpbmFfdm9sY2Fub1tbInBsb3QiXV0Ka3dfcDE1X3JldGluYV9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGFsbGNbW2Rlbm9tXV0sIGNvbG9yX2hpZ2ggPSBhbGxjW1tudW1dXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBvdXRsaW5lID0gb3V0bGluZSwKICBsYWJlbCA9IDEwKQpwcChmaWxlID0gIjA1aW5jbHVzaW9uX3ZvbGNhbm9fbWEva3dfcDE1X3JldGluYV9tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmt3X3AxNV9yZXRpbmFfbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmt3X3AxNV9yZXRpbmFfbWFbWyJwbG90Il1dCmBgYAoKIyMgcDA4X2tvX3JldGluYQoKYGBge3J9CnRhYmxlX25hbWUgPC0gInAwOF9rb19yZXRpbmEiCnRhYmxlIDwtIGluY2x1c2lvbl90YWJsZXNbWyJkYXRhIl1dW1t0YWJsZV9uYW1lXV0KbnVtIDwtICJwMDhfa29fcmV0aW5hIgpkZW5vbSA8LSAicDA4X3d0X3JldGluYSIKa3dfcDA4X3JldGluYV92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLCBmaWxsID0gImJsYWNrIiwKICBjb2xvcl9sb3cgPSBhbGxjW1tkZW5vbV1dLCBjb2xvcl9oaWdoID0gYWxsY1tbbnVtXV0sCiAgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IDEwLCBhbHBoYSA9IDEuMCwKICBzaXplID0gNCkKcHAoZmlsZSA9ICIwNWluY2x1c2lvbl92b2xjYW5vX21hL2t3X3AwOF9yZXRpbmFfdm9sY2Fuby5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmt3X3AwOF9yZXRpbmFfdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa3dfcDA4X3JldGluYV92b2xjYW5vW1sicGxvdCJdXQprd19wMDhfcmV0aW5hX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gYWxsY1tbZGVub21dXSwgY29sb3JfaGlnaCA9IGFsbGNbW251bV1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIG91dGxpbmUgPSBvdXRsaW5lLAogIGxhYmVsID0gMTApCnBwKGZpbGUgPSAiMDVpbmNsdXNpb25fdm9sY2Fub19tYV9wMDhfcmV0aW5hX21hLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKa3dfcDA4X3JldGluYV9tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa3dfcDA4X3JldGluYV9tYVtbInBsb3QiXV0KYGBgCgojIyBwMTVfaGV0X3NjbgoKYGBge3J9CnRhYmxlX25hbWUgPC0gInAxNV9oZXRfc2NuIgp0YWJsZSA8LSBpbmNsdXNpb25fdGFibGVzW1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCm51bSA8LSAicDE1X2hldF9zY24iCmRlbm9tIDwtICJwMTVfd3Rfc2NuIgpod19wMTVfc2NuX3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGZpbGwgPSAiYmxhY2siLAogIGNvbG9yX2xvdyA9IGFsbGNbW2Rlbm9tXV0sIGNvbG9yX2hpZ2ggPSBhbGxjW1tudW1dXSwKICBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gMTAsIGFscGhhID0gMS4wLAogIHNpemUgPSA0KQpwcChmaWxlID0gIjA1aW5jbHVzaW9uX3ZvbGNhbm9fbWEvaHdfcDE1X3Njbl92b2xjYW5vLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKaHdfcDE1X3Njbl92b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpod19wMTVfc2NuX3ZvbGNhbm9bWyJwbG90Il1dCmh3X3AxNV9zY25fbWEgPC0gcGxvdF9tYV9jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGV4cHJfY29sID0gImRlc2VxX2Jhc2VtZWFuIiwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwKICBjb2xvcl9sb3cgPSBhbGxjW1tkZW5vbV1dLCBjb2xvcl9oaWdoID0gYWxsY1tbbnVtXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgb3V0bGluZSA9IG91dGxpbmUsCiAgbGFiZWwgPSAxMCkKcHAoZmlsZSA9ICIwNWluY2x1c2lvbl92b2xjYW5vX21hL2h3X3AxNV9zY25fbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpod19wMTVfc2NuX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpod19wMTVfc2NuX21hW1sicGxvdCJdXQpgYGAKCiMjIHAwOF9oZXRfc2NuCgpgYGB7cn0KdGFibGVfbmFtZSA8LSAicDA4X2hldF9zY24iCnRhYmxlIDwtIGluY2x1c2lvbl90YWJsZXNbWyJkYXRhIl1dW1t0YWJsZV9uYW1lXV0KbnVtIDwtICJwMDhfaGV0X3NjbiIKZGVub20gPC0gInAwOF93dF9zY24iCmh3X3AwOF9zY25fdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgZmlsbCA9ICJibGFjayIsCiAgY29sb3JfbG93ID0gYWxsY1tbZGVub21dXSwgY29sb3JfaGlnaCA9IGFsbGNbW251bV1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSAxMCwgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQpCnBwKGZpbGUgPSAiMDVpbmNsdXNpb25fdm9sY2Fub19tYS9od19wMDhfc2NuX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpod19wMDhfc2NuX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmh3X3AwOF9zY25fdm9sY2Fub1tbInBsb3QiXV0KaHdfcDA4X3Njbl9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGFsbGNbW2Rlbm9tXV0sIGNvbG9yX2hpZ2ggPSBhbGxjW1tudW1dXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBvdXRsaW5lID0gb3V0bGluZSwKICBsYWJlbCA9IDEwKQpwcChmaWxlID0gIjA1aW5jbHVzaW9uX3ZvbGNhbm9fbWEvaHdfcDA4X3Njbl9tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmh3X3AwOF9zY25fbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmh3X3AwOF9zY25fbWFbWyJwbG90Il1dCmBgYAoKIyMgcDE1X2tvX3NjbgoKYGBge3J9CnRhYmxlX25hbWUgPC0gInAxNV9rb19zY24iCnRhYmxlIDwtIGluY2x1c2lvbl90YWJsZXNbWyJkYXRhIl1dW1t0YWJsZV9uYW1lXV0KbnVtIDwtICJwMTVfa29fc2NuIgpkZW5vbSA8LSAicDE1X3d0X3NjbiIKa3dfcDE1X3Njbl92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLCBmaWxsID0gImJsYWNrIiwKICBjb2xvcl9sb3cgPSBhbGxjW1tkZW5vbV1dLCBjb2xvcl9oaWdoID0gYWxsY1tbbnVtXV0sCiAgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IDEwLCBhbHBoYSA9IDEuMCwKICBzaXplID0gNCkKcHAoZmlsZSA9ICIwNWluY2x1c2lvbl92b2xjYW5vX21hL2t3X3AxNV9zY25fdm9sY2Fuby5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmt3X3AxNV9zY25fdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa3dfcDE1X3Njbl92b2xjYW5vW1sicGxvdCJdXQprd19wMTVfc2NuX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gYWxsY1tbZGVub21dXSwgY29sb3JfaGlnaCA9IGFsbGNbW251bV1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIG91dGxpbmUgPSBvdXRsaW5lLAogIGxhYmVsID0gMTApCnBwKGZpbGUgPSAiMDVpbmNsdXNpb25fdm9sY2Fub19tYS9rd19wMTVfc2NuX21hLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKa3dfcDE1X3Njbl9tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa3dfcDE1X3Njbl9tYVtbInBsb3QiXV0KYGBgCgojIyBwMDhfa29fc2NuCgpgYGB7cn0KdGFibGVfbmFtZSA8LSAicDA4X2tvX3NjbiIKdGFibGUgPC0gaW5jbHVzaW9uX3RhYmxlc1tbImRhdGEiXV1bW3RhYmxlX25hbWVdXQpudW0gPC0gInAwOF9rb19zY24iCmRlbm9tIDwtICJwMDhfd3Rfc2NuIgprd19wMDhfc2NuX3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGZpbGwgPSAiYmxhY2siLAogIGNvbG9yX2xvdyA9IGFsbGNbW2Rlbm9tXV0sIGNvbG9yX2hpZ2ggPSBhbGxjW1tudW1dXSwKICBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gMTAsIGFscGhhID0gMS4wLAogIHNpemUgPSA0KQpwcChmaWxlID0gIjA1aW5jbHVzaW9uX3ZvbGNhbm9fbWEva3dfcDA4X3Njbl92b2xjYW5vLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKa3dfcDA4X3Njbl92b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQprd19wMDhfc2NuX3ZvbGNhbm9bWyJwbG90Il1dCmt3X3AwOF9zY25fbWEgPC0gcGxvdF9tYV9jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGV4cHJfY29sID0gImRlc2VxX2Jhc2VtZWFuIiwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwKICBjb2xvcl9sb3cgPSBhbGxjW1tkZW5vbV1dLCBjb2xvcl9oaWdoID0gYWxsY1tbbnVtXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgb3V0bGluZSA9IG91dGxpbmUsCiAgbGFiZWwgPSAxMCkKcHAoZmlsZSA9ICIwNWluY2x1c2lvbl92b2xjYW5vX21hL2t3X3AwOF9zY25fbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQprd19wMDhfc2NuX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQprd19wMDhfc2NuX21hW1sicGxvdCJdXQpgYGAKClNlZSB0aGUgc2hhcmVkL3VuaXF1ZSBnZW5lcyBpbiB0aGVzZSBzZXRzLgoKYGBge3J9CmluY2x1c2lvbl91cHNldHMgPC0gdXBzZXRyX3NpZyhpbmNsdXNpb25fc2lnKQppbmNsdXNpb25faW50ZXJzZWN0cyA8LSB3cml0ZV91cHNldF9ncm91cHMoCiAgaW5jbHVzaW9uX3Vwc2V0cywgZXhjZWwgPSAiMDRpbmNsdXNpb25fY29tcGFyaXNvbi9pbmNsdXNpb25fZ2VuZV9ncm91cHMueGxzeCIpCmBgYAoKIyMjIEV4dHJhY3QgZ2VuZXMgaW5jbHVkZWQgZm9yIGVhY2ggc2V0IG9mIGNvbnRyYXN0cwoKTm93LCB1c2luZyB0aGF0IGZ1bmN0aW9uLCBwdWxsIG91dCB0aGUgZ2VuZSBJRHMgb2YgZ2VuZXMgd2UgZG8gbm90CnRydXN0IGJlY2F1c2UgdGhleSB3ZXJlIHRvbyBoaWdoIGluIHd0IGZvciBldmVyeSBjb250cmFzdCB3ZSBhcmUKbGlrZWx5IHRvIHBlcmZvcm0uCgpUaGUgZm9sbG93aW5nIHdhcyBhIG1vZGlmaWVkIHZlcnNpb24gb2YgdGhlIGluY2x1c2lvbiBmdW5jdGlvbiB3aGljaCBpcyBzb21ld2hhdCBtb3JlIHJlc3RyaWN0aXZlLgoKYGBge3J9CmV4dHJhY3RfaW5jbHVzaW9uc19zdHJpY3QgPC0gZnVuY3Rpb24oaW5jbHVzaW9uX3NpZywgaW5jbHVzaW9uX3RhYmxlcywgaW5jbHVzaW9ucywga2VlcGVycywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhbGxfZ2VuZXMsIGFjY29yZGluZ190byA9ICJkZXNlcSIsIHdoaWNoID0gInVwcyIpIHsKICByZXRsaXN0IDwtIGxpc3QoKQogIHRhYmxlX25hbWVzIDwtIG5hbWVzKGluY2x1c2lvbl9zaWdbW2FjY29yZGluZ190b11dW1t3aGljaF1dKQogIGZvciAoY19udW0gaW4gc2VxX2Fsb25nKGtlZXBlcnMpKSB7CiAgICBjb250cmFzdCA8LSBuYW1lcyhrZWVwZXJzKVtjX251bV0KICAgIG51bWVyYXRvcl9uYW1lIDwtIGtlZXBlcnNbW2NfbnVtXV1bMV0KICAgIGRlbm9taW5hdG9yX25hbWUgPC0ga2VlcGVyc1tbY19udW1dXVsyXQogICAgIyMgSW4gbXkgbmV3IGJyYW5jaCBJIGNsZWFuZWQgdXAgdGhlIHNhbml0aXplciBmdW5jdGlvbiBmb3IgY29udHJhc3RzIHNvIHRoaXMgaXMgbm90IG5lZWRlZC4KICAgICMjIFRoZSBmb2xsb3dpbmcgdHdvIGxpbmVzIGFyZSBubyBsb25nZXIgbmVlZGVkIGJlY2F1c2Ugb2YgdGhlIGNsZWFudXBzIEkgcGVyZm9ybWVkLgogICAgIyNudW1lcmF0b3JfbmFtZSA8LSBnc3ViKHggPSBudW1lcmF0b3JfbmFtZSwgcGF0dGVybiA9ICIoaGV0fGtvfHd0KSIsIHJlcGxhY2VtZW50ID0gIl9cXDFfIikKICAgICMjZGVub21pbmF0b3JfbmFtZSA8LSBnc3ViKHggPSBkZW5vbWluYXRvcl9uYW1lLCBwYXR0ZXJuID0gIihoZXR8a298d3QpIiwgcmVwbGFjZW1lbnQgPSAiX1xcMV8iKQogICAgbnVtZXJhdG9yX3RhYmxlIDwtIGluY2x1c2lvbl9zaWdbW2FjY29yZGluZ190b11dW1t3aGljaF1dW1tudW1lcmF0b3JfbmFtZV1dCiAgICBudW1lcmF0b3JfZ2VuZXMgPC0gcm93bmFtZXMobnVtZXJhdG9yX3RhYmxlKQogICAgZGVub21pbmF0b3JfdGFibGUgPC0gaW5jbHVzaW9uX3NpZ1tbYWNjb3JkaW5nX3RvXV1bW3doaWNoXV1bW2Rlbm9taW5hdG9yX25hbWVdXQogICAgZGVub21pbmF0b3JfZ2VuZXMgPC0gcm93bmFtZXMoZGVub21pbmF0b3JfdGFibGUpCiAgICBkZl9jb2x1bW5zIDwtIHBhc3RlMCgiZGVzZXFfIiwgYygibG9nZmMiLCAiYWRqcCIsICJkZW4iKSkKICAgIGluY2x1ZGVkX251bSA8LSBpbmNsdXNpb25fdGFibGVzW1siZGF0YSJdXVtbbnVtZXJhdG9yX25hbWVdXVssIGRmX2NvbHVtbnNdCiAgICBjb2xuYW1lcyhpbmNsdWRlZF9udW0pIDwtIGMoIm51bWVyYXRvcl92c193dF9sb2dmYyIsICJudW1lcmF0b3JfdnNfd3RfYWRqcCIsICJudW1fd3RfbWVhbl9leHBycyIpCiAgICBpbmNsdWRlZF9kZW4gPC0gaW5jbHVzaW9uX3RhYmxlc1tbImRhdGEiXV1bW2Rlbm9taW5hdG9yX25hbWVdXVssIGRmX2NvbHVtbnNdCiAgICBjb2xuYW1lcyhpbmNsdWRlZF9kZW4pIDwtIGMoImRlbm9taW5hdG9yX3ZzX3d0X2xvZ2ZjIiwgImRlbm9taW5hdG9yX3ZzX3d0X2FkanAiLCAiZGVuX3d0X21lYW5fZXhwcnMiKQogICAgIyMgSSB0aGluayB0aGlzIGlzIHdoZXJlIHRoaW5ncyB3ZW50IHdyb25nLAogICAgIyMgY29tcGFyZSB0aGlzIG1vZGlmaWVkIGxpbmUgdG8gdGhlIG9yaWdpbmFsIHRvIHByb3ZlIGl0LgogICAgaW5jbHVkZWRfZGYgPC0gbWVyZ2UoaW5jbHVkZWRfbnVtLCBpbmNsdWRlZF9kZW4sIGJ5ID0gInJvdy5uYW1lcyIpCiAgICAjIyBQcmV2aW91c2x5LCBJIGRpZCBub3Qgc3BlY2lmeSB0aGUgbWVyZ2UgYWN0aW9uLCBhbGwgPSBGQUxTRSBieSBkZWZhdWx0LgogICAgIyMgVGhpcyB0aGVuIHdpbGwgcmVzdWx0IGluIGEgZGlmZmVyZW5jZSBpbiB0aGUgcm93cyBvYnNlcnZlZAogICAgIyMgaW5jbHVkZWRfZGYgPC0gbWVyZ2UoaW5jbHVkZWRfbnVtLCBpbmNsdWRlZF9kZW4sIGJ5ID0gInJvdy5uYW1lcyIsIGFsbCA9IEZBTFNFKQogICAgcm93bmFtZXMoaW5jbHVkZWRfZGYpIDwtIGluY2x1ZGVkX2RmW1siUm93Lm5hbWVzIl1dCiAgICBpbmNsdWRlZF9kZltbIlJvdy5uYW1lcyJdXSA8LSBOVUxMCiAgICBjb25jYXRlbmF0ZWRfZ2VuZXMgPC0gYyhudW1lcmF0b3JfZ2VuZXMsIGRlbm9taW5hdG9yX2dlbmVzKQogICAgYm90aF9nZW5lX2lkeCA8LSBkdXBsaWNhdGVkKGNvbmNhdGVuYXRlZF9nZW5lcykKICAgIGdlbmVzX2luX2JvdGggPC0gY29uY2F0ZW5hdGVkX2dlbmVzW2JvdGhfZ2VuZV9pZHhdCiAgICBtZXNzYWdlKCJUaGUgc2V0IG9mIHVuaXF1ZSBnZW5lcyBoaWdoZXIgaW4gIiwgbnVtZXJhdG9yX25hbWUsCiAgICAgICAgICAgICIgdnMuIHd0IGlzICIsIGxlbmd0aChudW1lcmF0b3JfZ2VuZXMpLCAiLiIpCiAgICBtZXNzYWdlKCJUaGUgc2V0IG9mIHVuaXF1ZSBnZW5lcyBoaWdoZXIgaW4gIiwgZGVub21pbmF0b3JfbmFtZSwKICAgICAgICAgICAgIiB2cy4gd3QgaXMgIiwgbGVuZ3RoKGRlbm9taW5hdG9yX2dlbmVzKSwgIi4iKQogICAgbWVzc2FnZSgiVGhlIGludGVyc2VjdGlvbiBvZiB0aGVtIGlzICIsIGxlbmd0aChnZW5lc19pbl9ib3RoKSwgIiBnZW5lcy4iKQogICAgaW5jbHVkZV9uYW1lIDwtIHBhc3RlMCgiaW5jXyIsIGNvbnRyYXN0KQogICAgaW5jbHVkZV9pZHggPC0gYWxsX2dlbmVzICVpbiUgZ2VuZXNfaW5fYm90aAogICAgaW5jbHVkZV9nZW5lcyA8LSBhbGxfZ2VuZXNbaW5jbHVkZV9pZHhdCiAgICBkZl9uYW1lIDwtIHBhc3RlMCgiZGZfIiwgY29udHJhc3QpCiAgICByZXRsaXN0W1tkZl9uYW1lXV0gPC0gaW5jbHVkZWRfZGYKICAgIHdyaXR0ZW5faW5jbHVzaW9uIDwtIHdyaXRlX3hsc3goCiAgICAgIGRhdGEgPSBpbmNsdWRlZF9kZiwKICAgICAgZXhjZWwgPSBnbHVlKCIwN2luY2x1ZGVkX3N0cmljdF9nZW5lc19leGNlbC97aW5jbHVkZV9uYW1lfS12e3Zlcn0ueGxzeCIpKQogICAgcmV0bGlzdFtbaW5jbHVkZV9uYW1lXV0gPC0gaW5jbHVkZV9nZW5lcwogICAgcmV0bGlzdFtbY29udHJhc3RdXSA8LSBpbmNsdWRlX2dlbmVzCiAgfQogIHJldHVybihyZXRsaXN0KQp9CmBgYAoKVGhpcyBpcyB0aGUgcHJlLTIwMjUwNSB2ZXJzaW9uIG9mIHRoaXMgZnVuY3Rpb24uCgpgYGB7cn0KZXh0cmFjdF9pbmNsdXNpb25zIDwtIGZ1bmN0aW9uKGluY2x1c2lvbl9zaWcsIGluY2x1c2lvbl90YWJsZXMsIGluY2x1c2lvbnMsIGtlZXBlcnMsIGFsbF9nZW5lcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFjY29yZGluZ190byA9ICJkZXNlcSIsIHdoaWNoID0gInVwcyIpIHsKICByZXRsaXN0IDwtIGxpc3QoKQogIHRhYmxlX25hbWVzIDwtIG5hbWVzKGluY2x1c2lvbl9zaWdbW2FjY29yZGluZ190b11dW1t3aGljaF1dKQogIGZvciAoY19udW0gaW4gc2VxX2Fsb25nKGtlZXBlcnMpKSB7CiAgICBjb250cmFzdCA8LSBuYW1lcyhrZWVwZXJzKVtjX251bV0KICAgIG51bWVyYXRvcl9uYW1lIDwtIGtlZXBlcnNbW2NfbnVtXV1bMV0KICAgIGRlbm9taW5hdG9yX25hbWUgPC0ga2VlcGVyc1tbY19udW1dXVsyXQogICAgIyMgSW4gbXkgbmV3IGJyYW5jaCBJIGNsZWFuZWQgdXAgdGhlIHNhbml0aXplciBmdW5jdGlvbiBmb3IgY29udHJhc3RzIHNvIHRoaXMgaXMgbm90IG5lZWRlZC4KICAgICMjIFRoZSBmb2xsb3dpbmcgdHdvIGxpbmVzIGFyZSBubyBsb25nZXIgbmVlZGVkIGJlY2F1c2Ugb2YgdGhlIGNsZWFudXBzIEkgcGVyZm9ybWVkLgogICAgIyNudW1lcmF0b3JfbmFtZSA8LSBnc3ViKHggPSBudW1lcmF0b3JfbmFtZSwgcGF0dGVybiA9ICIoaGV0fGtvfHd0KSIsIHJlcGxhY2VtZW50ID0gIl9cXDFfIikKICAgICMjZGVub21pbmF0b3JfbmFtZSA8LSBnc3ViKHggPSBkZW5vbWluYXRvcl9uYW1lLCBwYXR0ZXJuID0gIihoZXR8a298d3QpIiwgcmVwbGFjZW1lbnQgPSAiX1xcMV8iKQogICAgbnVtZXJhdG9yX3RhYmxlIDwtIGluY2x1c2lvbl9zaWdbW2FjY29yZGluZ190b11dW1t3aGljaF1dW1tudW1lcmF0b3JfbmFtZV1dCiAgICBudW1lcmF0b3JfZ2VuZXMgPC0gcm93bmFtZXMobnVtZXJhdG9yX3RhYmxlKQogICAgZGVub21pbmF0b3JfdGFibGUgPC0gaW5jbHVzaW9uX3NpZ1tbYWNjb3JkaW5nX3RvXV1bW3doaWNoXV1bW2Rlbm9taW5hdG9yX25hbWVdXQogICAgZGVub21pbmF0b3JfZ2VuZXMgPC0gcm93bmFtZXMoZGVub21pbmF0b3JfdGFibGUpCiAgICBkZl9jb2x1bW5zIDwtIHBhc3RlMCgiZGVzZXFfIiwgYygibG9nZmMiLCAiYWRqcCIsICJkZW4iKSkKICAgIGluY2x1ZGVkX251bSA8LSBpbmNsdXNpb25fdGFibGVzW1siZGF0YSJdXVtbbnVtZXJhdG9yX25hbWVdXVssIGRmX2NvbHVtbnNdCiAgICBjb2xuYW1lcyhpbmNsdWRlZF9udW0pIDwtIGMoIm51bWVyYXRvcl92c193dF9sb2dmYyIsICJudW1lcmF0b3JfdnNfd3RfYWRqcCIsICJudW1fd3RfbWVhbl9leHBycyIpCiAgICBpbmNsdWRlZF9kZW4gPC0gaW5jbHVzaW9uX3RhYmxlc1tbImRhdGEiXV1bW2Rlbm9taW5hdG9yX25hbWVdXVssIGRmX2NvbHVtbnNdCiAgICBjb2xuYW1lcyhpbmNsdWRlZF9kZW4pIDwtIGMoImRlbm9taW5hdG9yX3ZzX3d0X2xvZ2ZjIiwgImRlbm9taW5hdG9yX3ZzX3d0X2FkanAiLCAiZGVuX3d0X21lYW5fZXhwcnMiKQogICAgaW5jbHVkZWRfZGYgPC0gbWVyZ2UoaW5jbHVkZWRfbnVtLCBpbmNsdWRlZF9kZW4sIGJ5ID0gInJvdy5uYW1lcyIpCiAgICByb3duYW1lcyhpbmNsdWRlZF9kZikgPC0gaW5jbHVkZWRfZGZbWyJSb3cubmFtZXMiXV0KICAgIGluY2x1ZGVkX2RmW1siUm93Lm5hbWVzIl1dIDwtIE5VTEwKICAgIGluY2x1ZGVfZ2VuZXMgPC0gdW5pcXVlKGMobnVtZXJhdG9yX2dlbmVzLCBkZW5vbWluYXRvcl9nZW5lcykpCiAgICBtZXNzYWdlKCJUaGUgc2V0IG9mIHVuaXF1ZSBnZW5lcyBoaWdoZXIgaW4gIiwgbnVtZXJhdG9yX25hbWUsCiAgICAgICAgICAgICIgdnMuIHd0IGlzICIsIGxlbmd0aChudW1lcmF0b3JfZ2VuZXMpLCAiLiIpCiAgICBtZXNzYWdlKCJUaGUgc2V0IG9mIHVuaXF1ZSBnZW5lcyBoaWdoZXIgaW4gIiwgZGVub21pbmF0b3JfbmFtZSwKICAgICAgICAgICAgIiB2cy4gd3QgaXMgIiwgbGVuZ3RoKGRlbm9taW5hdG9yX2dlbmVzKSwgIi4iKQogICAgbWVzc2FnZSgiVGhlIHVuaXF1ZSB1bmlvbiBvZiB0aGVtIGlzICIsIGxlbmd0aChpbmNsdWRlX2dlbmVzKSwgIiBnZW5lcy4iKQogICAgaW5jbHVkZV9uYW1lIDwtIHBhc3RlMCgiaW5jXyIsIGNvbnRyYXN0KQogICAgaW5jbHVkZV9pZHggPC0gYWxsX2dlbmVzICVpbiUgaW5jbHVkZV9nZW5lcwogICAgaW5jbHVkZV9nZW5lcyA8LSBhbGxfZ2VuZXNbaW5jbHVkZV9pZHhdCiAgICBkZl9uYW1lIDwtIHBhc3RlMCgiZGZfIiwgY29udHJhc3QpCiAgICByZXRsaXN0W1tkZl9uYW1lXV0gPC0gaW5jbHVkZWRfZGYKICAgIHdyaXR0ZW5faW5jbHVzaW9uIDwtIHdyaXRlX3hsc3goZGF0YSA9IGluY2x1ZGVkX2RmLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBleGNlbCA9IGdsdWUoImluY2x1ZGVkX2dlbmVzL3tpbmNsdWRlX25hbWV9LXZ7dmVyfS54bHN4IikpCiAgICByZXRsaXN0W1tpbmNsdWRlX25hbWVdXSA8LSBpbmNsdWRlX2dlbmVzCiAgICByZXRsaXN0W1tjb250cmFzdF1dIDwtIGluY2x1ZGVfZ2VuZXMKICB9CiAgcmV0dXJuKHJldGxpc3QpCn0KYGBgCgojICdOb3JtYWwnIEluY2x1c2lvbiBleHRyYWN0aW9uCgpIZXJlIGlzIHRoZSBmdWxsIHNldCBvZiBnZW5lIElEcwoKYGBge3J9CmFsbF9nZW5lcyA8LSByb3duYW1lcyhhc3NheSh2M19wYWlyd2lzZV9pbnB1dCkpCmBgYAoKSW4gdGhlIGZvbGxvd2luZyBibG9ja3MgSSBhbSBpbmNsdWRpbmcgdGhlIHVuaW9uIG9mIGdlbmVzIG9ic2VydmVkCmhpZ2hlciB0aGFuIHd0IGluIGVpdGhlciBvZiB0aGUgbnVtZXJhdG9yIG9yIGRlbm9taW5hdG9yIGZvciBlYWNoCmNvbnRyYXN0LgoKIyMgVGltZQoKYGBge3J9CnRpbWVfaW5jbHVzaW9ucyA8LSBleHRyYWN0X2luY2x1c2lvbnMoaW5jbHVzaW9uX3NpZywgaW5jbHVzaW9uX3RhYmxlcywgaW5jbHVzaW9ucywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aW1lX2tlZXBlcnMsIGFsbF9nZW5lcykKYGBgCgojIyBMb2NhdGlvbgoKYGBge3J9CmxvY2F0aW9uX2luY2x1c2lvbnMgPC0gZXh0cmFjdF9pbmNsdXNpb25zKGluY2x1c2lvbl9zaWcsIGluY2x1c2lvbl90YWJsZXMsIGluY2x1c2lvbnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxvY2F0aW9uX2tlZXBlcnMsIGFsbF9nZW5lcykKYGBgCgojIyBHZW5vdHlwZQoKYGBge3J9Cmdlbm90eXBlX2luY2x1c2lvbnMgPC0gZXh0cmFjdF9pbmNsdXNpb25zKGluY2x1c2lvbl9zaWcsIGluY2x1c2lvbl90YWJsZXMsIGluY2x1c2lvbnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdlbm90eXBlX2tlZXBlcnMsIGFsbF9nZW5lcykKYGBgCgojIFRoZSBzdHJpY3QgJ2luY2x1c2lvbicgc2V0CgojIyBUaW1lCgpgYGB7cn0KdGltZV9pbmNsdXNpb25zX3N0cmljdCA8LSBleHRyYWN0X2luY2x1c2lvbnNfc3RyaWN0KGluY2x1c2lvbl9zaWcsIGluY2x1c2lvbl90YWJsZXMsIGluY2x1c2lvbnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aW1lX2tlZXBlcnMsIGFsbF9nZW5lcykKYGBgCgojIyBMb2NhdGlvbgoKYGBge3J9CmxvY2F0aW9uX2luY2x1c2lvbnNfc3RyaWN0IDwtIGV4dHJhY3RfaW5jbHVzaW9uc19zdHJpY3QoaW5jbHVzaW9uX3NpZywgaW5jbHVzaW9uX3RhYmxlcywgaW5jbHVzaW9ucywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsb2NhdGlvbl9rZWVwZXJzLCBhbGxfZ2VuZXMpCmBgYAoKIyMgR2Vub3R5cGUKCmBgYHtyfQpnZW5vdHlwZV9pbmNsdXNpb25zX3N0cmljdCA8LSBleHRyYWN0X2luY2x1c2lvbnNfc3RyaWN0KGluY2x1c2lvbl9zaWcsIGluY2x1c2lvbl90YWJsZXMsIGluY2x1c2lvbnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2Vub3R5cGVfa2VlcGVycywgYWxsX2dlbmVzKQpgYGAKCiMjIyBDaGVjayB2cyBUaGVyZXNhJ3MgZmlsdGVyCgpVcCBhYm92ZSBUaGVyZXNhIHBlcmZvcm1lZCBhIDAuMjUgbG9nMkZDIGFuZCAwLjA1IGFkanAgZmlsdGVyIHdoaWNoCnByb3ZpZGVkIGEgc2V0IG9mIDIsNjQwIGdlbmVzIG9ic2VydmVkIGhpZ2hlciBpbiB0aGUgcDA4IGhldCByZXRpbmFzCnZzLiB3dCByZXRpbmFzLiAgSSBzaG91bGQgc2VlIHRoYXQgaW4gdGhpcyBpbmNsdXNpb25fc2lnIGRhdGEgc3RydWN0dXJlLgoKVGhlcmUgaXMgYW4gaW1wb3J0YW50IGNhdmVhdCB0aG91Z2g6IGluIFRoZXJlc2EncyBmaWx0ZXIgYWJvdmUsIHNoZQpkaWQgYSBERSBvZiBfb25seV8gdGhlIHJldGluYSBzYW1wbGVzIGJ1dCBJIGRpZCBhbGwgc2FtcGxlcy4gIEkKZXhwZWN0ZWQgdGhhdCB0aGlzIHdvdWxkIHJlc3VsdCBpbiBiYXNpY2FsbHkgdGhlIHNhbWUgcmVzdWx0IChJCmFjdHVhbGx5IGFzc3VtZWQgSSB3b3VsZCBnZXQgYSBmZXcgbW9yZSBnZW5lcyksIGJ1dCBpbnN0ZWFkIGl0IGFwcGVhcnMKdG8gaGF2ZSByZXRyaWV2ZWQgYSBzaWduaWZpY2FudGx5IHNtYWxsZXIgbnVtYmVyIG9mIGdlbmVzIChhYm91dCAxLzIsCmhhcHBpbHkgdGhleSBwcmV0dHkgbXVjaCBhbGwgYXBwZWFyIGluIHRoZSBwcmV2aW91cyBmaWx0ZXIpLiAgQXMgYQpyZXN1bHQsIEkgYW0gZ29pbmcgdG8gdHJ5IHJlbGF4aW5nIG15IGNvbnN0cmFpbnRzIHNsaWdodGx5IHRvIHNlZSBpZiBJCmNhbiByZWNhcGl0dWxhdGUgaGVyIGZpbHRlciAod2hpY2ggd291bGQgbWF0Y2ggVGhlcmVzYSdzIGxhdGVyIGZpbHRlciwKdGhvdWdoIEkgZ3Vlc3MgdGhhdCBpbiB0dXJuIHdpbGwgbGVhZCB0byBhIHNtYWxsZXIgc2V0IG9mIGdlbmVzCmNvbXBhcmVkIHRvIGhlciBsYXRlciwgcmVsYXhlZCAwLjEgZmlsdGVyKS4KCmBgYHtyfQpjb21wYXJpc29uIDwtIGluY2x1c2lvbl9zaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbInAwOF9oZXRfcmV0aW5hIl1dCmNvbXAgPC0gbGlzdCgKICAidGFhIiA9IHRhYV9rZWVwZXJzLAogICJuZXciID0gcm93bmFtZXMoY29tcGFyaXNvbikpCnRlc3RfY29tcGFyaXNvbiA8LSBWZW5uZXJhYmxlOjpWZW5uKGNvbXApClZlbm5lcmFibGU6OnBsb3QodGVzdF9jb21wYXJpc29uKQpgYGAKCkkgd2FudCB0byBoYXZlIGEgbGl0dGxlIGZ1bmN0aW9uIHdoaWNoLCBnaXZlbiBhIGNvbnRyYXN0IG9mIGludGVyZXN0LAp3aWxsIGV4dHJhY3QgdGhlIGdlbmUgc2V0cyB3aGljaCBzaG91bGQgYmUgaW5jbHVkZWQvZXhjbHVkZWQgZ2l2ZW4gdGhlCmFib3ZlLgoKYGBge3J9CndyaXRlX2FsbF9jcCA8LSBmdW5jdGlvbihhbGxfY3AsIHByZWZpeCA9ICIxMiIsIHN1ZmZpeCA9ICIiKSB7CiAgYWxsX3dyaXR0ZW4gPC0gbGlzdCgpCiAgZm9yIChnIGluIHNlcV9sZW4obGVuZ3RoKGFsbF9jcCkpKSB7CiAgICBuYW1lIDwtIG5hbWVzKGFsbF9jcClbZ10KICAgIGRhdHVtIDwtIGFsbF9jcFtbbmFtZV1dCiAgICBmaWxlbmFtZSA8LSBnbHVlKCJ7cHJlZml4fWVucmljaG1lbnRfZXhjZWwve25hbWV9X2Nwcm9maWxlcntzdWZmaXh9LXZ7dmVyfS54bHN4IikKICAgIHdyaXR0ZW4gPC0gc20od3JpdGVfY3BfZGF0YShkYXR1bSwgZXhjZWwgPSBmaWxlbmFtZSkpCiAgICBhbGxfd3JpdHRlbltbZ11dIDwtIHdyaXR0ZW4KICB9CiAgcmV0dXJuKGFsbF93cml0dGVuKQp9CndyaXRlX2FsbF9ncCA8LSBmdW5jdGlvbihhbGxfZ3AsIHByZWZpeCA9ICIxMyIsIHN1ZmZpeCA9ICIiKSB7CiAgYWxsX3dyaXR0ZW4gPC0gbGlzdCgpCiAgZm9yIChnIGluIHNlcV9sZW4obGVuZ3RoKGFsbF9ncCkpKSB7CiAgICBuYW1lIDwtIG5hbWVzKGFsbF9ncClbZ10KICAgIGRhdHVtIDwtIGFsbF9ncFtbbmFtZV1dCiAgICBmaWxlbmFtZSA8LSBnbHVlKCJ7cHJlZml4fWVucmljaG1lbnRfZXhjZWwve25hbWV9X2dwcm9maWxlcntzdWZmaXh9LXZ7dmVyfS54bHN4IikKICAgIHdyaXR0ZW4gPC0gc20od3JpdGVfZ3Byb2ZpbGVyX2RhdGEoZGF0dW0sIGV4Y2VsID0gZmlsZW5hbWUpKQogICAgYWxsX3dyaXR0ZW5bW2ddXSA8LSB3cml0dGVuCiAgfQogIHJldHVybihhbGxfd3JpdHRlbikKfQp3cml0ZV9hbGxfZW4gPC0gZnVuY3Rpb24oYWxsX2VuLCBwcmVmaXggPSAiMTQiLCBzdWZmaXggPSAiIikgewogIGFsbF93cml0dGVuIDwtIGxpc3QoKQogIGZvciAoZSBpbiBzZXFfbGVuKGxlbmd0aChhbGxfZW4pKSkgewogICAgbmFtZSA8LSBuYW1lcyhhbGxfZW4pW2VdCiAgICBkYXR1bSA8LSBhbGxfZW5bW25hbWVdXQogICAgZmlsZW5hbWUgPC0gZ2x1ZSgie3ByZWZpeH1lbnJpY2htZW50X2V4Y2VsL3tuYW1lfV9lbnJpY2hlcntzdWZmaXh9LXZ7dmVyfS54bHN4IikKICAgIHdyaXR0ZW4gPC0gc20od3JpdGVfZW5yaWNoZXJfZGF0YShkYXR1bSwgZXhjZWwgPSBmaWxlbmFtZSkpCiAgICBhbGxfd3JpdHRlbltbZV1dIDwtIHdyaXR0ZW4KICB9CiAgcmV0dXJuKGFsbF93cml0dGVuKQp9CmBgYAoKIyMjIEV4dHJhY3QgZ2VuZXMgaW5jbHVkZWQgZm9yIGVhY2ggc2V0IG9mIGNvbnRyYXN0cwoKTm93LCB1c2luZyB0aGF0IGZ1bmN0aW9uLCBwdWxsIG91dCB0aGUgZ2VuZSBJRHMgb2YgZ2VuZXMgd2UgZG8gbm90CnRydXN0IGJlY2F1c2UgdGhleSB3ZXJlIHRvbyBoaWdoIGluIHd0IGZvciBldmVyeSBjb250cmFzdCB3ZSBhcmUKbGlrZWx5IHRvIHBlcmZvcm0uCgpgYGB7cn0KYWxsX2dlbmVzIDwtIHJvd25hbWVzKGFzc2F5KHYzX3BhaXJ3aXNlX2lucHV0KSkKdGltZV9pbmNsdXNpb25zIDwtIGV4dHJhY3RfaW5jbHVzaW9ucyhpbmNsdXNpb25fc2lnLCBpbmNsdXNpb25fdGFibGVzLCBpbmNsdXNpb25zLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRpbWVfa2VlcGVycywgYWxsX2dlbmVzKQpsb2NhdGlvbl9pbmNsdXNpb25zIDwtIGV4dHJhY3RfaW5jbHVzaW9ucyhpbmNsdXNpb25fc2lnLCBpbmNsdXNpb25fdGFibGVzLCBpbmNsdXNpb25zLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsb2NhdGlvbl9rZWVwZXJzLCBhbGxfZ2VuZXMpCmdlbm90eXBlX2luY2x1c2lvbnMgPC0gZXh0cmFjdF9pbmNsdXNpb25zKGluY2x1c2lvbl9zaWcsIGluY2x1c2lvbl90YWJsZXMsIGluY2x1c2lvbnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdlbm90eXBlX2tlZXBlcnMsIGFsbF9nZW5lcykKYGBgCgojIyBQZXJmb3JtIHRoZSBERSBhbmFseXNlcyBhbmQgZXhjbHVkZSB0aGUgdGFyZ2V0IGdlbmVzCgoKIyMjIEdlbm90eXBlCgpgYGB7cn0KZ2Vub3R5cGVfZGUgPC0gYWxsX3BhaXJ3aXNlKHYzX3BhaXJ3aXNlX2lucHV0LCBmaWx0ZXIgPSBUUlVFLCBtb2RlbF9mc3RyaW5nID0gZGVmYXVsdF9mc3RyaW5nLAogICAgICAgICAgICAgICAgICAgICAgICAgICAga2VlcGVycyA9IGdlbm90eXBlX2tlZXBlcnMsIG1vZGVsX3N2cyA9ICJzdmFzZXEiKQpnZW5vdHlwZV9kZQpgYGAKCiMjIyBMb2NhdGlvbgoKYGBge3J9CmxvY2F0aW9uX2RlIDwtIGFsbF9wYWlyd2lzZSh2M19wYWlyd2lzZV9pbnB1dCwgZmlsdGVyID0gVFJVRSwgbW9kZWxfZnN0cmluZyA9IGRlZmF1bHRfZnN0cmluZywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGtlZXBlcnMgPSBsb2NhdGlvbl9rZWVwZXJzLCBtb2RlbF9zdnMgPSAic3Zhc2VxIikKbG9jYXRpb25fZGUKYGBgCgojIyMgVGltZQoKYGBge3J9CnRpbWVfZGUgPC0gYWxsX3BhaXJ3aXNlKHYzX3BhaXJ3aXNlX2lucHV0LCBmaWx0ZXIgPSBUUlVFLCBtb2RlbF9mc3RyaW5nID0gZGVmYXVsdF9mc3RyaW5nLAogICAgICAgICAgICAgICAgICAgICAgICBrZWVwZXJzID0gdGltZV9rZWVwZXJzLCBtb2RlbF9zdnMgPSAic3Zhc2VxIikKdGltZV9kZQpgYGAKCkl0IGlzIG5lYXIgaGVyZSB3aGVuIHRoZSBjb21wdXRlciBzb21ldGltZXMgZmFpbHMgd2l0aCBubyBtb3JlCnRlbXBmaWxlcy4gIEluIGFub3RoZXIgd2luZG93IEkgYW0gbWVzc2luZyB3aXRoIHRlbXBmaWxlKCkgaW4gUiB0byB0cnkKdG8gdW5kZXJzdGFuZCB3aGVyZSBpdCBpcyBnb2luZyBvZmYgdGhlIHJhaWxzLi4uCgojIyBFeHRyYWN0IHRoZSByZWxldmFudCB0YWJsZXMgYW5kIGluY2x1ZGUgZ2VuZXMgbG93ZXIgaW4gd3QKCiMjIyBHZW5vdHlwZSBjb250cmFzdHMKCkkgd2lsbCBzdGFydCB3aXRoIHRoZSB0YWJsZXMgYW5kIG5vIGluY2x1c2lvbnMgc28gSSBjYW4gY2hlY2sgbXkgd29yay4KCkluIHRoaXMgZmlyc3QgYmxvY2sgSSB3aWxsIGV4cGxhaW4gYSBsaXR0bGUgbW9yZSB0aG9yb3VnaGx5IHdoYXQgaXMKZ29pbmcgb246CgoxLiAgRHVtcCB0aGUgZnVsbCB0YWJsZSBvZiB0aGUgY29udHJhc3RzIEkgZGVmaW5lZCBhYm92ZSBjb21wYXJpbmcgdGhlCiAgICAzIGdlbm90eXBlcyBhY3Jvc3MgdGltZS9sb2NhdGlvbi4KMi4gIEl0ZXJhdGUgb3ZlciBlYWNoIG9mIHRob3NlIGNvbnRyYXN0cyBhbmQgZG8gdGhlIGZvbGxvd2luZzoKICAgIGEuICBFeHRyYWN0IHRoZSBuYW1lIG9mIHRoZSBjb250cmFzdCwgJ2toX3AwOF9kbGduJyBmb3IgZXhhbXBsZQogICAgYi4gIFlhbmsgb3V0IHRoYXQgc3BlY2lmaWMgZW50cnkgZnJvbSB0aGUga2VlcGVyIGxpc3QgYW5kIGl0cyBuYW1lCiAgICBjLiAgWWFuayBvdXQgdGhlIGNvcnJlc3BvbmRpbmcgc2V0IG9mIGdlbmVzIHRvIGluY2x1ZGUgZnJvbSB0aGUKICAgICAgICBpbmNsdXNpb25zIGRhdGEgc3RydWN0dXJlLgogICAgZC4gIENyZWF0ZSBhIGZpbGVuYW1lIGdpdmVuIHRoZSBuYW1lIGluIChhKSBhYm92ZSBhbmQgdGhlIGxvZ0ZDCiAgICAgICAgY3V0b2ZmIGNob3NlbiBmb3IgdGhlIGluY2x1c2lvbnMgKEkgYW0gYXNzdW1pbmcgd2UgbWF5IGNoYW5nZQogICAgICAgIHRoaXMpCiAgICBlLiAgR2l2ZW4gKGIpLCAoYyksIGFuZCAoZCksIGV4dHJhY3QgdGhlIGNvcnJlc3BvbmRpbmcgdGFibGUgZnJvbQogICAgICAgIHRoZSBkaWZmZXJlbnRpYWwgZXhwcmVzc2lvbiBhbmFseXNpcyBhbmQgaW5jbHVkZSB0aGUgYXBwcm9wcmlhdGUKICAgICAgICBnZW5lcy4KCiAgICBGSVhNRTogbXkgZ3Byb2ZpbGVyIGZ1bmN0aW9uIGp1c3QgYXNzdW1lcyBodW1hbiBhbmQgc28gaWYgcGFzc2VkIG1tdXNjdWx1cyB3aWxsIGluY29ycmVjdGx5CiAgICBhdHRlbXB0IHRvIGNvbm5lY3QgdG8gbm9uLWV4aXN0YW50IGRhdGFiYXNlcy4gIExldCB1cyBmaXggdGhhdCBub3cuCgpgYGB7cn0KZ2Vub3R5cGVfdGFibGVzX2Z1bGwgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgZ2Vub3R5cGVfZGUsIGtlZXBlcnMgPSBnZW5vdHlwZV9rZWVwZXJzLCBsYWJlbF9jb2x1bW4gPSBsYWJlbF9jb2x1bW4sCiAgZmFuY3kgPSBUUlVFLAogIGV4Y2VsID0gZ2x1ZSgiMDhmdWxsX2NvbnRyYXN0c19leGNlbC9nZW5vdHlwZV9mdWxsX3RhYmxlcy12e3Zlcn0ueGxzeCIpKQpnZW5vdHlwZV90YWJsZXNfZnVsbApnZW5vdHlwZV9zaWdfZnVsbCA8LSBleHRyYWN0X3NpZ25pZmljYW50X2dlbmVzKAogIGdlbm90eXBlX3RhYmxlc19mdWxsLCBhY2NvcmRpbmdfdG8gPSAiZGVzZXEiLAogIGV4Y2VsID0gZ2x1ZSgiMDhmdWxsX2NvbnRyYXN0c19leGNlbC9nZW5vdHlwZV9mdWxsX3NpZy12e3Zlcn0ueGxzeCIpKQpnZW5vdHlwZV9zaWdfZnVsbApgYGAKCiMjIyBTZWFyY2ggRW5yaWNobWVudCBvZiB0aGVzZSBzZXRzCgpJbiB0aGlzIHJ1biwgd2Ugd2lsbCBzZWFyY2ggdGhlIGZ1bGwgc2V0IG9mIGdlbmVzLCBuZXh0IHdlIHdpbGwgb25seQpkbyB0aGUgaW5jbHVzaW9ucy4KCmBgYHtyfQpnZW5vdHlwZV9mdWxsX2dwIDwtIGFsbF9ncHJvZmlsZXIoZ2Vub3R5cGVfc2lnX2Z1bGwsIHNwZWNpZXMgPSAibW11c2N1bHVzIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV4Y2VsID0gIjA5ZnVsbF9jb250cmFzdHNfZW5yaWNoL2dlbm90eXBlX2Z1bGxfZ3Byb2ZpbGVyLnhsc3giKQpnZW5vdHlwZV9mdWxsX2NwIDwtIGFsbF9jcHJvZmlsZXIoZ2Vub3R5cGVfc2lnX2Z1bGwsIGdlbm90eXBlX3RhYmxlc19mdWxsLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgb3JnZGIgPSAib3JnLk1tLmVnLmRiIiwgZ29fbGV2ZWwgPSBnb19sZXZlbCwgb3JnYW5pc20gPSAibW91c2UiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgb3JnZGJfZnJvbSA9IG9yZ2RiX2Zyb20sIG1heF9ncm91cHNpemUgPSBtYXhfZ3JvdXBzaXplLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZXhjZWwgPSAiMDlmdWxsX2NvbnRyYXN0c19lbnJpY2gvZ2Vub3R5cGVfZnVsbF9jcHJvZmlsZXIueGxzeCIpCmdlbm90eXBlX2Z1bGxfdXBzZXQgPC0gdXBzZXRyX3NpZyhnZW5vdHlwZV9zaWdfZnVsbCkKZ2Vub3R5cGVfZnVsbF9pbnRlcnNlY3RzIDwtIHdyaXRlX3Vwc2V0X2dyb3VwcyhnZW5vdHlwZV9mdWxsX3Vwc2V0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV4Y2VsID0gIjA5ZnVsbF9jb250cmFzdHNfaW50ZXJzZWN0aW9ucy9nZW5vdHlwZV9mdWxsX2dlbmVfZ3JvdXBzLnhsc3giKQpgYGAKCk5vdyBzZXBhcmF0ZSB0aGUgdmFyaW91cyBnZW5vdHlwZSB0YWJsZXMgYW5kIHBlcmZvcm0gdGhlIGluY2x1c2lvbnMgb2YKdGhlIGdlbmVzIHdpdGggcmVsYXRpdmVseSBsb3cgd3QgdmFsdWVzLgoKIyMjIyBFeHRyYWN0IGZpbHRlcmVkIGdlbmVzIGFuZCBwYXNzIHRvIGdwcm9maWxlcjIvY2x1c3RlcnByb2ZpbGVyCgpgYGB7cn0KZ2Vub3R5cGVfdGFibGVzIDwtIGxpc3QoKQpnZW5vdHlwZV9zaWcgPC0gbGlzdCgpCmdlbm90eXBlX2dwIDwtIGxpc3QoKQpnZW5vdHlwZV9jcCA8LSBsaXN0KCkKZ2Vub3R5cGVfZW4gPC0gbGlzdCgpCmZvciAoayBpbiBzZXFfYWxvbmcoZ2Vub3R5cGVfa2VlcGVycykpIHsKICBuYW1lIDwtIG5hbWVzKGdlbm90eXBlX2tlZXBlcnMpW2tdCiAgbWVzc2FnZSgiRXhhbWluaW5nICIsIG5hbWUpCiAga2VlcGVyIDwtIGdlbm90eXBlX2tlZXBlcnNbbmFtZV0KICBpbmNsdWRlX25hbWUgPC0gcGFzdGUwKCJpbmNfIiwgbmFtZSkKICBpbmNsdWRlX2RmX25hbWUgPC0gcGFzdGUwKCJkZl8iLCBuYW1lKQogIGluY2x1ZGVfZGYgPC0gZ2Vub3R5cGVfaW5jbHVzaW9uc1tbaW5jbHVkZV9kZl9uYW1lXV0KICBpbmNsdWRlcyA8LSBnZW5vdHlwZV9pbmNsdXNpb25zW1tpbmNsdWRlX25hbWVdXQogIHN1bW1hcnkocm93bmFtZXMoZ2Vub3R5cGVfc2lnX2Z1bGxbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dKSAlaW4lIGluY2x1ZGVzKQogIGluY2x1ZGVfZmlsZW5hbWUgPC0gZ2x1ZSgiMTBnZW5vdHlwZV9jb250cmFzdHNfZXhjZWwve25hbWV9X2luY2x1ZGluZ193dF97bGZjX2N1dG9mZn1fZGVjcmVhc2VkX3RhYmxlLXZ7dmVyfS54bHN4IikKICBpbmNsdWRlX3NpZ19maWxlbmFtZSA8LSBnbHVlKCIxMGdlbm90eXBlX2NvbnRyYXN0c19leGNlbC97bmFtZX1faW5jbHVkaW5nX3d0X3tsZmNfY3V0b2ZmfV9kZWNyZWFzZWRfc2lnLXZ7dmVyfS54bHN4IikKICBnZW5vdHlwZV90YWJsZXNbW25hbWVdXSA8LSBjb21iaW5lX2RlX3RhYmxlcygKICAgIGdlbm90eXBlX2RlLCBleHRyYV9hbm5vdCA9IGluY2x1ZGVfZGYsCiAgICBrZWVwZXJzID0ga2VlcGVyLCBsYWJlbF9jb2x1bW4gPSBsYWJlbF9jb2x1bW4sCiAgICBleGNlbCA9IGluY2x1ZGVfZmlsZW5hbWUsIHdhbnRlZF9nZW5lcyA9IGluY2x1ZGVzKQogIHByaW50KGdlbm90eXBlX3RhYmxlc1tbbmFtZV1dKQogIGdlbm90eXBlX3NpZ1tbbmFtZV1dIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCiAgICBnZW5vdHlwZV90YWJsZXNbW25hbWVdXSwgYWNjb3JkaW5nX3RvID0gImRlc2VxIiwKICAgIGV4Y2VsID0gaW5jbHVkZV9zaWdfZmlsZW5hbWUpCiAgcHJpbnQoZ2Vub3R5cGVfc2lnW1tuYW1lXV0pCiAgbnVtX3Jvd3MgPC0gbnJvdyhnZW5vdHlwZV9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1sidXBzIl1dW1tuYW1lXV0pICsKICAgIG5yb3coZ2Vub3R5cGVfc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbImRvd25zIl1dW1tuYW1lXV0pCiAgbWVzc2FnZSgiVGhlcmUgYXJlICIsIG51bV9yb3dzLCAiIHNpZ25pZmljYW50IHVwIGFuZCBkb3duIGdlbmVzLiIpCiAgaWYgKG51bV9yb3dzID49IDEwKSB7CiAgICBtZXNzYWdlKCJQZXJmb3JtaW5nIGdwcm9maWxlci9jbHVzdGVyUHJvZmlsZXIuIikKICAgIGdlbm90eXBlX2dwW1tuYW1lXV0gPC0gYWxsX2dwcm9maWxlcihnZW5vdHlwZV9zaWdbW25hbWVdXSwgc3BlY2llcyA9ICJtbXVzY3VsdXMiKQogICAgZ2Vub3R5cGVfY3BbW25hbWVdXSA8LSBhbGxfY3Byb2ZpbGVyKAogICAgICBnZW5vdHlwZV9zaWdbW25hbWVdXSwgZ2Vub3R5cGVfdGFibGVzW1tuYW1lXV0sCiAgICAgIG9yZ2RiID0gIm9yZy5NbS5lZy5kYiIsIG9yZ2RiX2Zyb20gPSBvcmdkYl9mcm9tLAogICAgICBnb19sZXZlbCA9IGdvX2xldmVsLCBtYXhfZ3JvdXBzaXplID0gbWF4X2dyb3Vwc2l6ZSwgb3JnYW5pc20gPSAibW91c2UiKQogICAgI2lmICghaXMubnVsbChnZXQwKCJtMl9nc2MiKSkpIHsKICAgICMgIGdlbm90eXBlX2VuW1tuYW1lXV0gPC0gYWxsX2VucmljaGVyKGdlbm90eXBlX3NpZ1tbbmFtZV1dLCBnc2MgPSBtMl9nc2MsCiAgICAjICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvcmdkYiA9ICJvcmcuTW0uZWcuZGIiLCBmcm9tID0gIkVOU0VNQkwiLCB0byA9ICJTWU1CT0wiKQogICAgI30KICAgIGdwX3dyaXR0ZW4gPC0gd3JpdGVfYWxsX2dwKGdlbm90eXBlX2dwW1tuYW1lXV0sIHByZWZpeCA9ICIxMSIpCiAgICBjcF93cml0dGVuIDwtIHdyaXRlX2FsbF9jcChnZW5vdHlwZV9jcFtbbmFtZV1dLCBwcmVmaXggPSAiMTEiKQogICAgI2VuX3dyaXR0ZW4gPC0gd3JpdGVfYWxsX2VuKGdlbm90eXBlX2VuW1tuYW1lXV0pCiAgfSBlbHNlIHsKICAgIHdhcm5pbmcoIlRoZXJlIGFyZSBsZXNzIHRoYW4gMTAgZ2VuZXMgdXAgYW5kIGRvd24gaW4gdGhlICIsIG5hbWUsICIgY29tcGFyaXNvbi4iKQogICAgbWVzc2FnZSgiVGhlcmUgYXJlIGxlc3MgdGhhbiAxMCBnZW5lcyB1cCBhbmQgZG93biBpbiB0aGUgIiwgbmFtZSwgIiBjb21wYXJpc29uLiIpCiAgfQp9CmBgYAoKUGxvdCB0aGUgcmVzdWx0cyBzZXBhcmF0ZWx5LgoKYGBge3J9CmZvciAoayBpbiBzZXFfYWxvbmcoZ2Vub3R5cGVfa2VlcGVycykpIHsKICBuYW1lIDwtIG5hbWVzKGdlbm90eXBlX2tlZXBlcnMpW2tdCiAgbWVzc2FnZSgiRXhhbWluaW5nICIsIG5hbWUpCiAga2VlcGVyIDwtIGdlbm90eXBlX2tlZXBlcnNbbmFtZV0KICBpbmNsdWRlX25hbWUgPC0gcGFzdGUwKCJpbmNfIiwgbmFtZSkKICBpbmNsdWRlX2RmX25hbWUgPC0gcGFzdGUwKCJkZl8iLCBuYW1lKQogIGluY2x1ZGVfZGYgPC0gZ2Vub3R5cGVfaW5jbHVzaW9uc1tbaW5jbHVkZV9kZl9uYW1lXV0KICBpbmNsdWRlcyA8LSBnZW5vdHlwZV9pbmNsdXNpb25zW1tpbmNsdWRlX25hbWVdXQogIHN1bW1hcnkocm93bmFtZXMoZ2Vub3R5cGVfc2lnX2Z1bGxbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dKSAlaW4lIGluY2x1ZGVzKQogIG51bV9yb3dzIDwtIG5yb3coZ2Vub3R5cGVfc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dKSArCiAgICBucm93KGdlbm90eXBlX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJkb3ducyJdXVtbbmFtZV1dKQogIG5yb3coZ2Vub3R5cGVfc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dKQogIG5yb3coZ2Vub3R5cGVfc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbImRvd25zIl1dW1tuYW1lXV0pCiAgbWVzc2FnZSgiVGhlcmUgYXJlICIsIG51bV9yb3dzLCAiIHNpZ25pZmljYW50IHVwIGFuZCBkb3duIGdlbmVzLiIpCiAgIyMgIzEgaXMgdXAgYW5kICMyIGlzIGRvd24sIGF2b2lkaW5nIHR5cGVvcyBoZXJlLgogIG51bV9vYmplY3RzIDwtIGxlbmd0aChnZW5vdHlwZV9jcFtbbmFtZV1dKQogIGlmIChudW1fb2JqZWN0cyA9PSAwKSB7CiAgICB3YXJuaW5nKCJTb21ldGhpbmcgd2VudCB3cm9uZyBpbiBhbGxfY3Byb2ZpbGVyLiIpCiAgfSBlbHNlIHsKICAgIHVwcCA8LSB3aGljaChncmVwbCh4ID0gbmFtZXMoZ2Vub3R5cGVfY3BbW25hbWVdXSksIHBhdHRlcm4gPSAiX3VwJCIpKQogICAgZG93bnAgPC0gd2hpY2goZ3JlcGwoeCA9IG5hbWVzKGdlbm90eXBlX2NwW1tuYW1lXV0pLCBwYXR0ZXJuID0gIl9kb3duJCIpKQogICAgaWYgKGxlbmd0aCh1cHApID4gMCkgewogICAgICBtZl9zaWcgPC0gZ2Vub3R5cGVfY3BbW25hbWVdXVtbdXBwXV1bWyJnb19kYXRhIl1dW1siTUZfZW5yaWNoIl1dCiAgICAgIGNjX3NpZyA8LSBnZW5vdHlwZV9jcFtbbmFtZV1dW1t1cHBdXVtbImdvX2RhdGEiXV1bWyJDQ19lbnJpY2giXV0KICAgICAgYnBfc2lnIDwtIGdlbm90eXBlX2NwW1tuYW1lXV1bW3VwcF1dW1siZ29fZGF0YSJdXVtbIkJQX2VucmljaCJdXQogICAgICBtZl9wbG90c191cCA8LSBwbG90X2VucmljaHJlc3VsdChtZl9zaWcpCiAgICAgIG1mX3RyZWVfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMTJjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X3VwX21mX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBtZl90cmVlX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQobWZfcGxvdHNfdXBbWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgbWZfYmFyX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjEyY2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9tZl9zaWdfYmFyLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBtZl9iYXJfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChtZl9wbG90c191cFtbImJhciJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGNjX3Bsb3RzX3VwIDwtIHBsb3RfZW5yaWNocmVzdWx0KGNjX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgY2NfdHJlZV91cF9maWxlbmFtZSA8LSBnbHVlKCIxMmNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfY2Nfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IGNjX3RyZWVfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c191cFtbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBjY19iYXJfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMTJjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X3VwX2NjX3NpZ19iYXIucGRmIikKICAgICAgcHAoZmlsZSA9IGNjX2Jhcl91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGNjX3Bsb3RzX3VwW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgYnBfcGxvdHNfdXAgPC0gcGxvdF9lbnJpY2hyZXN1bHQoYnBfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBicF90cmVlX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjEyY2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9icF9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gYnBfdHJlZV91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGJwX3Bsb3RzX3VwW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGJwX3Bsb3RzX3VwIDwtIHBsb3RfZW5yaWNocmVzdWx0KGJwX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgYnBfYmFyX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjEyY2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9icF9zaWdfYmFyLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBicF9iYXJfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChicF9wbG90c191cFtbImJhciJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICB9CiAgICBpZiAobGVuZ3RoKGRvd25wKSA+IDApIHsKICAgICAgbWZfc2lnIDwtIGdlbm90eXBlX2NwW1tuYW1lXV1bW2Rvd25wXV1bWyJnb19kYXRhIl1dW1siTUZfZW5yaWNoIl1dCiAgICAgIGNjX3NpZyA8LSBnZW5vdHlwZV9jcFtbbmFtZV1dW1tkb3ducF1dW1siZ29fZGF0YSJdXVtbIkNDX2VucmljaCJdXQogICAgICBicF9zaWcgPC0gZ2Vub3R5cGVfY3BbW25hbWVdXVtbZG93bnBdXVtbImdvX2RhdGEiXV1bWyJCUF9lbnJpY2giXV0KICAgICAgbWZfcGxvdHNfZG93biA8LSBwbG90X2VucmljaHJlc3VsdChtZl9zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIG1mX3RyZWVfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIxMmNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fZG93bl9tZl9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gbWZfdHJlZV9kb3duX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQobWZfcGxvdHNfZG93bltbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBtZl9iYXJfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIxMmNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fZG93bl9tZl9zaWdfYmFyLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBtZl9iYXJfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX2Rvd25bWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBjY19wbG90c19kb3duIDwtIHBsb3RfZW5yaWNocmVzdWx0KGNjX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgY2NfdHJlZV9kb3duX2ZpbGVuYW1lIDwtIGdsdWUoIjEyY2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV9kb3duX2NjX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBjY190cmVlX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c19kb3duW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGNjX2Jhcl9kb3duX2ZpbGVuYW1lIDwtIGdsdWUoIjEyY2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV9kb3duX2NjX3NpZ19iYXIucGRmIikKICAgICAgcHAoZmlsZSA9IGNjX2Jhcl9kb3duX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoY2NfcGxvdHNfZG93bltbImJhciJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGJwX3Bsb3RzX2Rvd24gPC0gcGxvdF9lbnJpY2hyZXN1bHQoYnBfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBicF90cmVlX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMTJjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fYnBfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IGJwX3RyZWVfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGJwX3Bsb3RzX2Rvd25bWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgYnBfcGxvdHNfZG93biA8LSBwbG90X2VucmljaHJlc3VsdChicF9zaWcsIGNhdGVvZ29yaWVzID0gZ29fY2F0ZWdvcmllcykKICAgICAgYnBfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMTJjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fYnBfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gYnBfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChicF9wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgIH0KICB9Cn0KYGBgCgpBIGZldyBzcGVjaWZpYyBwbG90cyBvZiBpbnRlcmVzdDogQ29sZW5zbyBhc2tlZCB0byBsYWJlbCBhIGZldyBnZW5lcwpmb3IgdGhlIGtub2Nrb3V0L2hldCBwMDhfcmV0aW5hcywgcDA4X3NjbiwgYW5kIHAwOF9kbGduOiBlaXRoZXIgdGhlCnRvcC0xNSBvciBhbGwgc2lnbmlmaWNhbnQuICBJIGFtIHByZXR0eSBzdXJlIGlmIEkgdGVsbCBpdCAxNSBhbmQgdGhlcmUKYXJlIG5vdCB0aGF0IG1hbnksIGl0IHdpbGwganVzdCBkbyB0aGUgc2lnbmlmaWNhbnQ/ICBMZXQgdXMgZmluZCBvdXQhCgojIyMjIGtvL2hldCBmb3IgcDA4IHJldGluYXMKCmBgYHtyfQp0YWJsZV9uYW1lIDwtICJraF9wMDhfcmV0aW5hIgp0YWJsZV9pbnB1dCA8LSBnZW5vdHlwZV90YWJsZXNbW3RhYmxlX25hbWVdXQp0YWJsZSA8LSB0YWJsZV9pbnB1dFtbImRhdGEiXV1bW3RhYmxlX25hbWVdXQppbnRlcmVzdGluZyA8LSBjKCJPcG40IiwgIkdtOTAwOCIsICJMcnIxIiwgIkNuYmQxIikKa2hfcDA4X3JldGluYV92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLCBmaWxsID0gImJsYWNrIiwKICBjb2xvcl9sb3cgPSBjb2xvcnNbWyJoZXRfcmV0aW5hIl1dLCBjb2xvcl9oaWdoID0gY29sb3JzW1sia29fcmV0aW5hIl1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZywgYWxwaGEgPSAxLjAsCiAgb3V0bGluZSA9IG91dGxpbmUsIHNpemUgPSA0KQpwcChmaWxlID0gIjEzZ2Vub3R5cGVfbWFfdm9sY2Fuby9raF9wMDhfcmV0aW5hX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpraF9wMDhfcmV0aW5hX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmtoX3AwOF9yZXRpbmFfdm9sY2Fub1tbInBsb3QiXV0Ka2hfcDA4X3JldGluYV9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX3JldGluYSJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImhldF9yZXRpbmEiXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgb3V0bGluZSA9IG91dGxpbmUsCiAgbGFiZWwgPSBpbnRlcmVzdGluZykKcHAoZmlsZSA9ICIxM2dlbm90eXBlX21hX3ZvbGNhbm8va2hfcDA4X3JldGluYV9tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmtoX3AwOF9yZXRpbmFfbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmtoX3AwOF9yZXRpbmFfbWFbWyJwbG90Il1dCmBgYAoKIyMjIyBrby9oZXQgcDA4IFNDTgoKSSBhbSBnb2luZyB0byBtYWtlIGFuIGV4ZWN1dGl2ZSBkZWNpc2lvbiBmb3IgdGhpcyBwbG90LCAxNSBpcyB0b28gbWFueQphbmQgbWFrZXMgaXQgY3JhenkgY2x1dHRlcmVkLgoKIyMjIFJlcGVhdCB0aGlzIHdpdGggdHdvIHNldHMgb2YgZ2VuZXMKCmBgYHtyfQp0YWJsZV9uYW1lIDwtICJraF9wMDhfc2NuIgp0YWJsZV9pbnB1dCA8LSBnZW5vdHlwZV90YWJsZXNbW3RhYmxlX25hbWVdXQp0YWJsZSA8LSB0YWJsZV9pbnB1dFtbImRhdGEiXV1bW3RhYmxlX25hbWVdXQppbnRlcmVzdGluZ19nZW5lcyA8LSBjKCJGaWduIiwgIk5ybjEiLCAiRHB5c2wyIiwgIkFjdGIiLCAiRmdmOSIsICJPdHgyIiwgIlNlYzIzIiwKICAgICAgICAgICAgICAgICAgICAgICAiTmNhbTEiLCAiTWFwNCIsICJTZWMyMmIiLCAiTmxnbjMiLCAiTWFyY2tzIiwgIkNkNDciLAogICAgICAgICAgICAgICAgICAgICAgICJEcHlzbDMiLCAiTGluN2MiLCAiQ2FkbTEiLCAiU254MTIiLCAiUmhvYSIsICJJbnBwNWYiLAogICAgICAgICAgICAgICAgICAgICAgICJBdGcxMiIsICJTZXQiLCAiR3NrM2IiLCAiUGRjZDQiLCAiR2FicmEyIiwgIlRtY28xIiwgIkFuYXBjMTYiKQpraF9wMDhfc2NuX3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXEKX2FkanAiLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgc2l6ZSA9IDQsIGFscGhhID0gMS4wLAogIG91dGxpbmUgPSBvdXRsaW5lLCBjb2xvcl9sb3cgPSBjb2xvcnNbWyJrb19zY24iXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJoZXRfc2NuIl1dKQpwcChmaWxlID0gIjEzZ2Vub3R5cGVfbWFfdm9sY2Fuby9raF9wMDhfc2NuX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpraF9wMDhfc2NuX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmtoX3AwOF9zY25fdm9sY2Fub1tbInBsb3QiXV0KIyMgd2h5IGluIHRoZSBjcmFwIGlzIGl0IGRvdWJsZS1sYWJlbGxpbmchPwojIyBNeSBNQSBwbG90dGVyIGlzbid0IGFzIHNtYXJ0IGFzIHRoZSB2b2xjYW5vIHBsb3R0ZXIsIHRoZSBnZW5lcyBhcmU6CmtoX3AwOF9zY25fbWEgPC0gcGxvdF9tYV9jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGV4cHJfY29sID0gImRlc2VxX2Jhc2VtZWFuIiwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwKICBjb2xvcl9sb3cgPSBjb2xvcnNbWyJrb19zY24iXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJoZXRfc2NuIl1dLAogIG91dGxpbmUgPSBvdXRsaW5lLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzKQpwcChmaWxlID0gIjEzZ2Vub3R5cGVfbWFfdm9sY2Fuby9raF9wMDhfc2NuX21hLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKa2hfcDA4X3Njbl9tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa2hfcDA4X3Njbl9tYVtbInBsb3QiXV0KYGBgCgojIyMjIFNhbWUgcGxvdCBidXQgYSBkaWZmZXJlbnQgc2V0IG9mIGxhYmVsZWQgZ2VuZXMKCmBgYHtyfQp0YWJsZV9uYW1lIDwtICJraF9wMDhfc2NuIgp0YWJsZV9pbnB1dCA8LSBnZW5vdHlwZV90YWJsZXNbW3RhYmxlX25hbWVdXQp0YWJsZSA8LSB0YWJsZV9pbnB1dFtbImRhdGEiXV1bW3RhYmxlX25hbWVdXQppbnRlcmVzdGluZ19nZW5lcyA8LSBjKAogICJBbmFwYzE2IiwgIkdhYnJhMiIsICJUbWNvMSIsICJTb2QyIiwgIkZnZjkiLCAiUGRjZDQiLCAiUmhvYSIsICJHc2szYiIsICJGb3hwMSIsCiAgIk5jYW0xIiwgIk1hcmNrcyIsICJGaWduIiwgIkRweXNsMyIsICJJbnBwNWYiLCAiQ2FkbTEiLCAiTWFwNCIsICJVZ2NnIiwgIkVsb3ZsNCIsCiAgIkVsYXZsMSIsICJDZmwyIiwgIlRubnQxIiwgIkduYjEiLCAiSW1wYWN0IiwgIk5ybjEiLCAiTmxnbjMiLCAiQWN0YiIsICJDZDQ3IiwKICAiU2VjMjJiIiwgIlNsYzE3YTciLCAiVmdsdXQxIiwgIkFjdGIiLCAiQjRnYWx0NSIsICJGb3hwMSIsICJPdHgyIiwgIkxpbjdjIiwKICAiU254MTIiLCAiQXRnMTIiLCAiU2V0IikKa2hfcDA4X3Njbl92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX3NjbiJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImhldF9zY24iXV0sCiAgb3V0bGluZSA9IG91dGxpbmUsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgc2l6ZSA9IDQsIGFscGhhID0gMS4wKQpwcChmaWxlID0gIjEzZ2Vub3R5cGVfbWFfdm9sY2Fuby9raF9wMDhfc2NuX3ZvbGNhbm9fdjIucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpraF9wMDhfc2NuX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmtoX3AwOF9zY25fdm9sY2Fub1tbInBsb3QiXV0KIyMgd2h5IGluIHRoZSBjcmFwIGlzIGl0IGRvdWJsZS1sYWJlbGxpbmchPwojIyBNeSBNQSBwbG90dGVyIGlzbid0IGFzIHNtYXJ0IGFzIHRoZSB2b2xjYW5vIHBsb3R0ZXIsIHRoZSBnZW5lcyBhcmU6CmtoX3AwOF9zY25fbWEgPC0gcGxvdF9tYV9jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGV4cHJfY29sID0gImRlc2VxX2Jhc2VtZWFuIiwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwKICBjb2xvcl9sb3cgPSBjb2xvcnNbWyJrb19zY24iXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJoZXRfc2NuIl1dLAogIG91dGxpbmUgPSBvdXRsaW5lLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzKQpwcChmaWxlID0gIjEzZ2Vub3R5cGVfbWFfdm9sY2Fuby9raF9wMDhfc2NuX21hX3YyLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKa2hfcDA4X3Njbl9tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa2hfcDA4X3Njbl9tYVtbInBsb3QiXV0KYGBgCgojIyMjIGtvL2hldCBwMDggZExHTgoKYGBge3J9CnRhYmxlX25hbWUgPC0gImtoX3AwOF9kbGduIgp0YWJsZV9pbnB1dCA8LSBnZW5vdHlwZV90YWJsZXNbW3RhYmxlX25hbWVdXQp0YWJsZSA8LSB0YWJsZV9pbnB1dFtbImRhdGEiXV1bW3RhYmxlX25hbWVdXQpraF9wMDhfZGxnbl92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX2RsZ24iXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJoZXRfZGxnbiJdXSwKICBvdXRsaW5lID0gb3V0bGluZSwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IDEwLCBzaXplID0gNCwgYWxwaGEgPSAxLjApCnBwKGZpbGUgPSAiMTNnZW5vdHlwZV9tYV92b2xjYW5vL2toX3AwOF9kbGduX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpraF9wMDhfZGxnbl92b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpraF9wMDhfZGxnbl92b2xjYW5vW1sicGxvdCJdXQojIyBNeSBNQSBwbG90dGVyIGlzbid0IGFzIHNtYXJ0IGFzIHRoZSB2b2xjYW5vIHBsb3R0ZXIsIHRoZSBnZW5lcyBhcmU6CmtoX3AwOF9kbGduX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gY29sb3JzW1sia29fZGxnbiJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImhldF9kbGduIl1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gMTAsIG91dGxpbmUgPSBvdXRsaW5lKQpwcChmaWxlID0gIjEzZ2Vub3R5cGVfbWFfdm9sY2Fuby9raF9wMDhfZGxnbl9tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmtoX3AwOF9kbGduX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpraF9wMDhfZGxnbl9tYVtbInBsb3QiXV0KCmBgYAoKIyMjIyBrby9oZXQgZm9yIHAxNSByZXRpbmFzCgpXaGVuIGxhc3QgSSByYW4gdGhpcyBtYW51YWxseSwgaXQgZGlkIG5vdCBkb3VibGUtbGFiZWwsIGhvcGVmdWxseSB0aGF0CnJlbWFpbnMgdHJ1ZSBpbiB0aGUgY29udGFpbmVyLgoKYGBge3J9CnRhYmxlX25hbWUgPC0gImtoX3AxNV9yZXRpbmEiCnRhYmxlX2lucHV0IDwtIGdlbm90eXBlX3RhYmxlc1tbdGFibGVfbmFtZV1dCnRhYmxlIDwtIHRhYmxlX2lucHV0W1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCmludGVyZXN0aW5nIDwtIGMoIk9wbjQiLCAiR205MDA4IiwgIkxycjEiLCAiQ25iZDEiKQpraF9wMTVfcmV0aW5hX3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGZpbGwgPSAiYmxhY2siLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX3JldGluYSJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImhldF9yZXRpbmEiXV0sCiAgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nLCBhbHBoYSA9IDEuMCwKICBvdXRsaW5lID0gb3V0bGluZSwgc2l6ZSA9IDQpCnBwKGZpbGUgPSAiMTNnZW5vdHlwZV9tYV92b2xjYW5vL2toX3AxNV9yZXRpbmFfdm9sY2Fuby5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmtoX3AxNV9yZXRpbmFfdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa2hfcDE1X3JldGluYV92b2xjYW5vW1sicGxvdCJdXQojIyB3aHkgaW4gdGhlIGNyYXAgaXMgaXQgZG91YmxlLWxhYmVsbGluZyE/CiMjIE15IE1BIHBsb3R0ZXIgaXNuJ3QgYXMgc21hcnQgYXMgdGhlIHZvbGNhbm8gcGxvdHRlciwgdGhlIGdlbmVzIGFyZToKa2hfcDE1X3JldGluYV9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX3JldGluYSJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImhldF9yZXRpbmEiXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZywgb3V0bGluZSA9IG91dGxpbmUpCnBwKGZpbGUgPSAiMTNnZW5vdHlwZV9tYV92b2xjYW5vL2toX3AxNV9yZXRpbmFfbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpraF9wMTVfcmV0aW5hX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpraF9wMTVfcmV0aW5hX21hW1sicGxvdCJdXQpgYGAKCiMjIyMga28vaGV0IHAxNSBTQ04KCiMjIyBSZXBlYXQgdGhpcyB3aXRoIHR3byBzZXRzIG9mIGdlbmVzCgpgYGB7cn0KdGFibGVfbmFtZSA8LSAia2hfcDE1X3NjbiIKdGFibGVfaW5wdXQgPC0gZ2Vub3R5cGVfdGFibGVzW1t0YWJsZV9uYW1lXV0KdGFibGUgPC0gdGFibGVfaW5wdXRbWyJkYXRhIl1dW1t0YWJsZV9uYW1lXV0KaW50ZXJlc3RpbmdfZ2VuZXMgPC0gYygiRmlnbiIsICJOcm4xIiwgIkRweXNsMiIsICJBY3RiIiwgIkZnZjkiLCAiT3R4MiIsICJTZWMyMyIsCiAgICAgICAgICAgICAgICAgICAgICAgIk5jYW0xIiwgIk1hcDQiLCAiU2VjMjJiIiwgIk5sZ24zIiwgIk1hcmNrcyIsICJDZDQ3IiwKICAgICAgICAgICAgICAgICAgICAgICAiRHB5c2wzIiwgIkxpbjdjIiwgIkNhZG0xIiwgIlNueDEyIiwgIlJob2EiLCAiSW5wcDVmIiwKICAgICAgICAgICAgICAgICAgICAgICAiQXRnMTIiLCAiU2V0IiwgIkdzazNiIiwgIlBkY2Q0IiwgIkdhYnJhMiIsICJUbWNvMSIsICJBbmFwYzE2IikKa2hfcDE1X3Njbl92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgc2l6ZSA9IDQsIGFscGhhID0gMS4wLAogIG91dGxpbmUgPSBvdXRsaW5lLCBjb2xvcl9sb3cgPSBjb2xvcnNbWyJrb19zY24iXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJoZXRfc2NuIl1dKQpwcChmaWxlID0gIjEzZ2Vub3R5cGVfbWFfdm9sY2Fuby9raF9wMTVfc2NuX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpraF9wMTVfc2NuX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmtoX3AxNV9zY25fdm9sY2Fub1tbInBsb3QiXV0Ka2hfcDE1X3Njbl9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX3NjbiJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImhldF9zY24iXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgb3V0bGluZSA9IG91dGxpbmUpCnBwKGZpbGUgPSAiMTNnZW5vdHlwZV9tYV92b2xjYW5vL2toX3AxNV9zY25fbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpraF9wMTVfc2NuX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpraF9wMTVfc2NuX21hW1sicGxvdCJdXQpgYGAKClJvdW5kIDIgd2l0aCBhIHNlcGFyYXRlIGdlbmUgc2V0LgoKYGBge3J9CnRhYmxlX25hbWUgPC0gImtoX3AxNV9zY24iCnRhYmxlX2lucHV0IDwtIGdlbm90eXBlX3RhYmxlc1tbdGFibGVfbmFtZV1dCnRhYmxlIDwtIHRhYmxlX2lucHV0W1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCmludGVyZXN0aW5nX2dlbmVzIDwtIGMoCiAgIkFuYXBjMTYiLCAiR2FicmEyIiwgIlRtY28xIiwgIlNvZDIiLCAiRmdmOSIsICJQZGNkNCIsICJSaG9hIiwgIkdzazNiIiwgIkZveHAxIiwKICAiTmNhbTEiLCAiTWFyY2tzIiwgIkZpZ24iLCAiRHB5c2wzIiwgIklucHA1ZiIsICJDYWRtMSIsICJNYXA0IiwgIlVnY2ciLCAiRWxvdmw0IiwKICAiRWxhdmwxIiwgIkNmbDIiLCAiVG5udDEiLCAiR25iMSIsICJJbXBhY3QiLCAiTnJuMSIsICJObGduMyIsICJBY3RiIiwgIkNkNDciLAogICJTZWMyMmIiLCAiU2xjMTdhNyIsICJWZ2x1dDEiLCAiQWN0YiIsICJCNGdhbHQ1IiwgIkZveHAxIiwgIk90eDIiLCAiTGluN2MiLAogICJTbngxMiIsICJBdGcxMiIsICJTZXQiKQpraF9wMTVfc2NuX3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXFfYWRqcCIsCiAgY29sb3JfbG93ID0gY29sb3JzW1sia29fc2NuIl1dLCBjb2xvcl9oaWdoID0gY29sb3JzW1siaGV0X3NjbiJdXSwKICBvdXRsaW5lID0gb3V0bGluZSwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzLCBzaXplID0gNCwgYWxwaGEgPSAxLjApCnBwKGZpbGUgPSAiMTNnZW5vdHlwZV9tYV92b2xjYW5vL2toX3AxNV9zY25fdm9sY2Fub192Mi5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmtoX3AxNV9zY25fdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa2hfcDE1X3Njbl92b2xjYW5vW1sicGxvdCJdXQpraF9wMTVfc2NuX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gY29sb3JzW1sia29fc2NuIl1dLCBjb2xvcl9oaWdoID0gY29sb3JzW1siaGV0X3NjbiJdXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzLCBvdXRsaW5lID0gb3V0bGluZSkKcHAoZmlsZSA9ICIxM2dlbm90eXBlX21hX3ZvbGNhbm8va2hfcDE1X3Njbl9tYV92Mi5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCmtoX3AxNV9zY25fbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmtoX3AxNV9zY25fbWFbWyJwbG90Il1dCmBgYAoKIyMjIyBrby9oZXQgcDE1IGRMR04KCmBgYHtyfQp0YWJsZV9uYW1lIDwtICJraF9wMTVfZGxnbiIKdGFibGVfaW5wdXQgPC0gZ2Vub3R5cGVfdGFibGVzW1t0YWJsZV9uYW1lXV0KdGFibGUgPC0gdGFibGVfaW5wdXRbWyJkYXRhIl1dW1t0YWJsZV9uYW1lXV0Ka2hfcDE1X2RsZ25fdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwKICBjb2xvcl9sb3cgPSBjb2xvcnNbWyJrb19kbGduIl1dLCBjb2xvcl9oaWdoID0gY29sb3JzW1siaGV0X2RsZ24iXV0sCiAgb3V0bGluZSA9IG91dGxpbmUsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSAxMCwgc2l6ZSA9IDQsIGFscGhhID0gMS4wKQpwcChmaWxlID0gIjEzZ2Vub3R5cGVfbWFfdm9sY2Fuby9raF9wMTVfZGxnbl92b2xjYW5vLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKa2hfcDE1X2RsZ25fdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa2hfcDE1X2RsZ25fdm9sY2Fub1tbInBsb3QiXV0KIyMgTXkgTUEgcGxvdHRlciBpc24ndCBhcyBzbWFydCBhcyB0aGUgdm9sY2FubyBwbG90dGVyLCB0aGUgZ2VuZXMgYXJlOgpraF9wMTVfZGxnbl9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX2RsZ24iXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJoZXRfZGxnbiJdXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IDEwLCBvdXRsaW5lID0gb3V0bGluZSkKcHAoZmlsZSA9ICIxM2dlbm90eXBlX21hX3ZvbGNhbm8va2hfcDE1X2RsZ25fbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpraF9wMTVfZGxnbl9tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKa2hfcDE1X2RsZ25fbWFbWyJwbG90Il1dCmBgYAoKQSBxdWVyeSBmcm9tIFJhc2htaToKCiJJIHdhcyBkaXNjdXNzaW5nIHdpdGggRHIuIFNwZWVyIGFib3V0IHRoZSBkTEdOIGRhdGEgYW5kIHdlIGZvdW5kIG1vc3RseQpyZXRpbmFsIGdlbmVzIGluIGRMR04gSGV0L0tPIG9yIHRpbWUgcG9pbnQgY29tcGFyaXNvbi4gUGxlYXNlIGNoZWNrIGlmCnRob3NlIGFyZSBub3QgcmV0aW5hIHNhbXBsZXMuIgoKSSBjaGVja2VkIHRoZSBzYW1wbGVzIGV0YyBhbmQgZXZlcnl0aGluZyBsb29rcyBvayB0byBtZTsgcGVyaGFwcyBJIGNhbgp1c2UgdGhlIHJlc3VsdHMgdG8gbG9vayBhdCB0aGlzIHF1ZXN0aW9uIGluIGFub3RoZXIgd2F5OgoKSSB3aWxsIHRoZXJlZm9yZSBsb2FkIHRoZSBwMDhfaGV0X2RsZ24vcDA4X2tvX2RsZ24gdGFibGUgYW5kIGNvbXBhcmUKaXQgdG8gdGhlIHAwOF9oZXRfcmV0aW5hL3AwOF9rb19yZXRpbmEgdGFibGUgZGlyZWN0bHkuICBJIHRoaW5rIHRoYXQKaWYgdGhlc2UgdHVybiBvdXQgdG8gYmUgaWRlbnRpY2FsLCB0aGVuIHRoZSBoeXBvdGhlc2lzIHN1Z2dlc3RlZCBieQp0aGlzIHF1ZXJ5IGlzIGNvcnJlY3QuCgpOb3RlLCBpbiBvcmRlciB0byBkbyB0aGlzLCBJIG11c3QgdXNlIHRoZSBmdWxsIHRhYmxlcywgbm90IHRoZQpwb3N0LWluY2x1c2lvbiB0YWJsZXMgYmVjYXVzZSBJIGNhbm5vdCBndWFyYW50ZWUgdGhhdCB0aGV5IHdpbGwgaGF2ZQppZGVudGljYWwgZ2VuZSBJRHMuCgpgYGB7cn0KcmV0aW5hX3RhYmxlIDwtIGdlbm90eXBlX3RhYmxlc19mdWxsW1siZGF0YSJdXVtbImtoX3AwOF9yZXRpbmEiXV0KZGxnbl90YWJsZSA8LSBnZW5vdHlwZV90YWJsZXNfZnVsbFtbImRhdGEiXV1bWyJraF9wMDhfZGxnbiJdXQpyZXRpbmFfc3Vic2V0IDwtIHJldGluYV90YWJsZVssIGMoImVuc2VtYmxfZ2VuZV9pZCIsICJkZXNlcV9sb2dmYyIpXQpjb2xuYW1lcyhyZXRpbmFfc3Vic2V0KSA8LSBjKCJJRCIsICJyZXRpbmFfbG9nZmMiKQpkbGduX3N1YnNldCA8LSBkbGduX3RhYmxlWywgYygiZW5zZW1ibF9nZW5lX2lkIiwgImRlc2VxX2xvZ2ZjIildCmNvbG5hbWVzKGRsZ25fc3Vic2V0KSA8LSBjKCJJRCIsICJkbGduX2xvZ2ZjIikKbWVyZ2VkIDwtIG1lcmdlKHJldGluYV9zdWJzZXQsIGRsZ25fc3Vic2V0LCBieSA9ICJJRCIpCnJvd25hbWVzKG1lcmdlZCkgPC0gbWFrZS5uYW1lcyhtZXJnZWRbWyJJRCJdXSwgdW5pcXVlID0gVFJVRSkKbWVyZ2VkW1siSUQiXV0gPC0gTlVMTApwbG90dGVkIDwtIHBsb3RfbGluZWFyX3NjYXR0ZXIobWVyZ2VkKQpwcChmaWxlID0gImltYWdlcy9raF9wMDhfcmV0aW5hX3ZzX2RsZ25fZGVzZXFfbG9nZmNfdmFsdWVzLnBuZyIpCnBsb3R0ZWRbWyJzY2F0dGVyIl1dCmRldi5vZmYoKQpwbG90dGVkW1sic2NhdHRlciJdXQpgYGAKClJhc2htaSBhc2tlZCBpZiB3ZSBjb3VsZCBhbHNvIGRvIHRoZSBwMTUgZm9yIHRoaXMgY29tcGFyaXNvbjoKCmBgYHtyfQpyZXRpbmFfdGFibGUgPC0gZ2Vub3R5cGVfdGFibGVzX2Z1bGxbWyJkYXRhIl1dW1sia2hfcDE1X3JldGluYSJdXQpkbGduX3RhYmxlIDwtIGdlbm90eXBlX3RhYmxlc19mdWxsW1siZGF0YSJdXVtbImtoX3AxNV9kbGduIl1dCnJldGluYV9zdWJzZXQgPC0gcmV0aW5hX3RhYmxlWywgYygiZW5zZW1ibF9nZW5lX2lkIiwgImRlc2VxX2xvZ2ZjIildCmNvbG5hbWVzKHJldGluYV9zdWJzZXQpIDwtIGMoIklEIiwgInJldGluYV9sb2dmYyIpCmRsZ25fc3Vic2V0IDwtIGRsZ25fdGFibGVbLCBjKCJlbnNlbWJsX2dlbmVfaWQiLCAiZGVzZXFfbG9nZmMiKV0KY29sbmFtZXMoZGxnbl9zdWJzZXQpIDwtIGMoIklEIiwgImRsZ25fbG9nZmMiKQptZXJnZWQgPC0gbWVyZ2UocmV0aW5hX3N1YnNldCwgZGxnbl9zdWJzZXQsIGJ5ID0gIklEIikKcm93bmFtZXMobWVyZ2VkKSA8LSBtYWtlLm5hbWVzKG1lcmdlZFtbIklEIl1dLCB1bmlxdWUgPSBUUlVFKQptZXJnZWRbWyJJRCJdXSA8LSBOVUxMCnBsb3R0ZWQgPC0gcGxvdF9saW5lYXJfc2NhdHRlcihtZXJnZWQpCnBwKGZpbGUgPSAiaW1hZ2VzL2toX3AxNV9yZXRpbmFfdnNfZGxnbl9kZXNlcV9sb2dmY192YWx1ZXMucG5nIikKcGxvdHRlZFtbInNjYXR0ZXIiXV0KZGV2Lm9mZigpCnBsb3R0ZWRbWyJzY2F0dGVyIl1dCmBgYAoKIyMjIyBSZXBlYXQgd2l0aCB0aGUgc3RyaWN0IGZpbHRlcgoKYGBge3J9Cmdlbm90eXBlX3N0cmljdF90YWJsZXMgPC0gbGlzdCgpCmdlbm90eXBlX3N0cmljdF9zaWcgPC0gbGlzdCgpCmdlbm90eXBlX3N0cmljdF9ncCA8LSBsaXN0KCkKZ2Vub3R5cGVfc3RyaWN0X2NwIDwtIGxpc3QoKQpnZW5vdHlwZV9zdHJpY3RfZW4gPC0gbGlzdCgpCmZvciAoayBpbiBzZXFfYWxvbmcoZ2Vub3R5cGVfa2VlcGVycykpIHsKICBuYW1lIDwtIG5hbWVzKGdlbm90eXBlX2tlZXBlcnMpW2tdCiAgbWVzc2FnZSgiRXhhbWluaW5nICIsIG5hbWUpCiAga2VlcGVyIDwtIGdlbm90eXBlX2tlZXBlcnNbbmFtZV0KICBpbmNsdWRlX25hbWUgPC0gcGFzdGUwKCJpbmNfIiwgbmFtZSkKICBpbmNsdWRlX2RmX25hbWUgPC0gcGFzdGUwKCJkZl8iLCBuYW1lKQogIGluY2x1ZGVfc3RyaWN0X2RmIDwtIGdlbm90eXBlX2luY2x1c2lvbnNfc3RyaWN0W1tpbmNsdWRlX2RmX25hbWVdXQogIGluY2x1ZGVzX3N0cmljdCA8LSBnZW5vdHlwZV9pbmNsdXNpb25zX3N0cmljdFtbaW5jbHVkZV9uYW1lXV0KICBzdW1tYXJ5KHJvd25hbWVzKGdlbm90eXBlX3NpZ19mdWxsW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSkgJWluJSBpbmNsdWRlc19zdHJpY3QpCiAgaW5jbHVkZV9maWxlbmFtZSA8LSBnbHVlKCIxNGdlbm90eXBlX3N0cmljdF9jb250cmFzdHNfZXhjZWwve25hbWV9X2luY2x1ZGluZ193dF97bGZjX2N1dG9mZn1fZGVjcmVhc2VkX3RhYmxlLXZ7dmVyfS54bHN4IikKICBpbmNsdWRlX3NpZ19maWxlbmFtZSA8LSBnbHVlKCIxNGdlbm90eXBlX3N0cmljdF9jb250cmFzdHNfZXhjZWwve25hbWV9X2luY2x1ZGluZ193dF97bGZjX2N1dG9mZn1fZGVjcmVhc2VkX3NpZy12e3Zlcn0ueGxzeCIpCiAgZ2Vub3R5cGVfc3RyaWN0X3RhYmxlc1tbbmFtZV1dIDwtIGNvbWJpbmVfZGVfdGFibGVzKAogICAgZ2Vub3R5cGVfZGUsIGV4dHJhX2Fubm90ID0gaW5jbHVkZV9zdHJpY3RfZGYsCiAgICBrZWVwZXJzID0ga2VlcGVyLCBsYWJlbF9jb2x1bW4gPSBsYWJlbF9jb2x1bW4sCiAgICBleGNlbCA9IGluY2x1ZGVfZmlsZW5hbWUsIHdhbnRlZF9nZW5lcyA9IGluY2x1ZGVzX3N0cmljdCkKICBwcmludChnZW5vdHlwZV9zdHJpY3RfdGFibGVzW1tuYW1lXV0pCiAgZ2Vub3R5cGVfc3RyaWN0X3NpZ1tbbmFtZV1dIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCiAgICBnZW5vdHlwZV9zdHJpY3RfdGFibGVzW1tuYW1lXV0sIGFjY29yZGluZ190byA9ICJkZXNlcSIsCiAgICBleGNlbCA9IGluY2x1ZGVfc2lnX2ZpbGVuYW1lKQogIHByaW50KGdlbm90eXBlX3N0cmljdF9zaWdbW25hbWVdXSkKICBudW1fcm93cyA8LSBucm93KGdlbm90eXBlX3N0cmljdF9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1sidXBzIl1dW1tuYW1lXV0pICsKICAgIG5yb3coZ2Vub3R5cGVfc3RyaWN0X3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJkb3ducyJdXVtbbmFtZV1dKQogIG1lc3NhZ2UoIlRoZXJlIGFyZSAiLCBudW1fcm93cywgIiBzaWduaWZpY2FudCB1cCBhbmQgZG93biBnZW5lcy4iKQogIGlmIChudW1fcm93cyA+PSAxMCkgewogICAgbWVzc2FnZSgiUGVyZm9ybWluZyBncHJvZmlsZXIvY2x1c3RlclByb2ZpbGVyLiIpCiAgICBnZW5vdHlwZV9zdHJpY3RfZ3BbW25hbWVdXSA8LSBhbGxfZ3Byb2ZpbGVyKGdlbm90eXBlX3N0cmljdF9zaWdbW25hbWVdXSwgc3BlY2llcyA9ICJtbXVzY3VsdXMiKQogICAgZ2Vub3R5cGVfc3RyaWN0X2NwW1tuYW1lXV0gPC0gYWxsX2Nwcm9maWxlcigKICAgICAgZ2Vub3R5cGVfc3RyaWN0X3NpZ1tbbmFtZV1dLCBnZW5vdHlwZV9zdHJpY3RfdGFibGVzW1tuYW1lXV0sCiAgICAgIG9yZ2RiID0gIm9yZy5NbS5lZy5kYiIsIGdvX2xldmVsID0gZ29fbGV2ZWwsCiAgICAgIG9yZ2RiX2Zyb20gPSBvcmdkYl9mcm9tLCBtYXhfZ3JvdXBzaXplID0gbWF4X2dyb3Vwc2l6ZSwgb3JnYW5pc20gPSAibW91c2UiKQogICAgI2lmICghaXMubnVsbChnZXQwKCJtMl9nc2MiKSkpIHsKICAgICMgIGdlbm90eXBlX3N0cmljdF9lbltbbmFtZV1dIDwtIGFsbF9lbnJpY2hlcihnZW5vdHlwZV9zdHJpY3Rfc2lnW1tuYW1lXV0sIGdzYyA9IG0yX2dzYywKICAgICMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9yZ2RiID0gIm9yZy5NbS5lZy5kYiIsIGZyb20gPSAiRU5TRU1CTCIsIHRvID0gIlNZTUJPTCIpCiAgICAjfQogICAgZ3Bfd3JpdHRlbiA8LSB3cml0ZV9hbGxfZ3AoZ2Vub3R5cGVfc3RyaWN0X2dwW1tuYW1lXV0sIHByZWZpeCA9ICIxNSIsIHN1ZmZpeCA9ICJzdHJpY3QiKQogICAgY3Bfd3JpdHRlbiA8LSB3cml0ZV9hbGxfY3AoZ2Vub3R5cGVfc3RyaWN0X2NwW1tuYW1lXV0sIHByZWZpeCA9ICIxNSIsIHN1ZmZpeCA9ICJzdHJpY3QiKQogICAgI2VuX3dyaXR0ZW4gPC0gd3JpdGVfYWxsX2VuKGdlbm90eXBlX3N0cmljdF9lbltbbmFtZV1dKQogIH0gZWxzZSB7CiAgICB3YXJuaW5nKCJUaGVyZSBhcmUgbGVzcyB0aGFuIDEwIGdlbmVzIHVwIGFuZCBkb3duIGluIHRoZSAiLCBuYW1lLCAiIGNvbXBhcmlzb24uIikKICAgIG1lc3NhZ2UoIlRoZXJlIGFyZSBsZXNzIHRoYW4gMTAgZ2VuZXMgdXAgYW5kIGRvd24gaW4gdGhlICIsIG5hbWUsICIgY29tcGFyaXNvbi4iKQogIH0KfQpgYGAKCiMgR2Vub3R5cGUgU3RyaWN0IHBsb3RzCgpgYGB7cn0KZm9yIChrIGluIHNlcV9hbG9uZyhnZW5vdHlwZV9rZWVwZXJzKSkgewogIG5hbWUgPC0gbmFtZXMoZ2Vub3R5cGVfa2VlcGVycylba10KICBtZXNzYWdlKCJFeGFtaW5pbmcgIiwgbmFtZSkKICBrZWVwZXIgPC0gZ2Vub3R5cGVfa2VlcGVyc1tuYW1lXQogIGluY2x1ZGVfbmFtZSA8LSBwYXN0ZTAoImluY18iLCBuYW1lKQogIGluY2x1ZGVfZGZfbmFtZSA8LSBwYXN0ZTAoImRmXyIsIG5hbWUpCiAgaW5jbHVkZV9zdHJpY3RfZGYgPC0gZ2Vub3R5cGVfaW5jbHVzaW9uc19zdHJpY3RbW2luY2x1ZGVfZGZfbmFtZV1dCiAgaW5jbHVkZXNfc3RyaWN0IDwtIGdlbm90eXBlX2luY2x1c2lvbnNfc3RyaWN0W1tpbmNsdWRlX25hbWVdXQogIHN1bW1hcnkocm93bmFtZXMoZ2Vub3R5cGVfc2lnX2Z1bGxbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dKSAlaW4lIGluY2x1ZGVzX3N0cmljdCkKICBudW1fcm93cyA8LSBucm93KGdlbm90eXBlX3N0cmljdF9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1sidXBzIl1dW1tuYW1lXV0pICsKICAgIG5yb3coZ2Vub3R5cGVfc3RyaWN0X3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJkb3ducyJdXVtbbmFtZV1dKQogIG5yb3coZ2Vub3R5cGVfc3RyaWN0X3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSkKICBucm93KGdlbm90eXBlX3N0cmljdF9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSkKICBtZXNzYWdlKCJUaGVyZSBhcmUgIiwgbnVtX3Jvd3MsICIgc2lnbmlmaWNhbnQgdXAgYW5kIGRvd24gZ2VuZXMuIikKICAjIyAjMSBpcyB1cCBhbmQgIzIgaXMgZG93biwgYXZvaWRpbmcgdHlwZW9zIGhlcmUuCiAgbnVtX29iamVjdHMgPC0gbGVuZ3RoKGdlbm90eXBlX3N0cmljdF9jcFtbbmFtZV1dKQogIGlmIChudW1fb2JqZWN0cyA9PSAwKSB7CiAgICB3YXJuaW5nKCJTb21ldGhpbmcgd2VudCB3cm9uZyBpbiBhbGxfY3Byb2ZpbGVyLiIpCiAgfSBlbHNlIHsKICAgIHVwcCA8LSB3aGljaChncmVwbCh4ID0gbmFtZXMoZ2Vub3R5cGVfc3RyaWN0X2NwW1tuYW1lXV0pLCBwYXR0ZXJuID0gIl91cCQiKSkKICAgIGRvd25wIDwtIHdoaWNoKGdyZXBsKHggPSBuYW1lcyhnZW5vdHlwZV9zdHJpY3RfY3BbW25hbWVdXSksIHBhdHRlcm4gPSAiX2Rvd24kIikpCiAgICBpZiAobGVuZ3RoKHVwcCkgPiAwKSB7CiAgICAgIG1mX3NpZyA8LSBnZW5vdHlwZV9zdHJpY3RfY3BbW25hbWVdXVtbdXBwXV1bWyJnb19kYXRhIl1dW1siTUZfZW5yaWNoIl1dCiAgICAgIGNjX3NpZyA8LSBnZW5vdHlwZV9zdHJpY3RfY3BbW25hbWVdXVtbdXBwXV1bWyJnb19kYXRhIl1dW1siQ0NfZW5yaWNoIl1dCiAgICAgIGJwX3NpZyA8LSBnZW5vdHlwZV9zdHJpY3RfY3BbW25hbWVdXVtbdXBwXV1bWyJnb19kYXRhIl1dW1siQlBfZW5yaWNoIl1dCiAgICAgIG1mX3Bsb3RzX3VwIDwtIHBsb3RfZW5yaWNocmVzdWx0KG1mX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgbWZfdHJlZV91cF9maWxlbmFtZSA8LSBnbHVlKCIxNmNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfbWZfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IG1mX3RyZWVfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChtZl9wbG90c191cFtbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBtZl9iYXJfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMTZjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X3VwX21mX3NpZ19iYXIucGRmIikKICAgICAgcHAoZmlsZSA9IG1mX2Jhcl91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX3VwW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfcGxvdHNfdXAgPC0gcGxvdF9lbnJpY2hyZXN1bHQoY2Nfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBjY190cmVlX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjE2Y2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9jY19zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gY2NfdHJlZV91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGNjX3Bsb3RzX3VwW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGNjX2Jhcl91cF9maWxlbmFtZSA8LSBnbHVlKCIxNmNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfY2Nfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gY2NfYmFyX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoY2NfcGxvdHNfdXBbWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBicF9wbG90c191cCA8LSBwbG90X2VucmljaHJlc3VsdChicF9zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGJwX3RyZWVfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMTZjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X3VwX2JwX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBicF90cmVlX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoYnBfcGxvdHNfdXBbWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgYnBfcGxvdHNfdXAgPC0gcGxvdF9lbnJpY2hyZXN1bHQoYnBfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBicF9iYXJfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMTZjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X3VwX2JwX3NpZ19iYXIucGRmIikKICAgICAgcHAoZmlsZSA9IGJwX2Jhcl91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGJwX3Bsb3RzX3VwW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgIH0KICAgIGlmIChsZW5ndGgoZG93bnApID4gMCkgewogICAgICBtZl9zaWcgPC0gZ2Vub3R5cGVfc3RyaWN0X2NwW1tuYW1lXV1bW2Rvd25wXV1bWyJnb19kYXRhIl1dW1siTUZfZW5yaWNoIl1dCiAgICAgIGNjX3NpZyA8LSBnZW5vdHlwZV9zdHJpY3RfY3BbW25hbWVdXVtbZG93bnBdXVtbImdvX2RhdGEiXV1bWyJDQ19lbnJpY2giXV0KICAgICAgYnBfc2lnIDwtIGdlbm90eXBlX3N0cmljdF9jcFtbbmFtZV1dW1tkb3ducF1dW1siZ29fZGF0YSJdXVtbIkJQX2VucmljaCJdXQogICAgICBtZl9wbG90c19kb3duIDwtIHBsb3RfZW5yaWNocmVzdWx0KG1mX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgbWZfdHJlZV9kb3duX2ZpbGVuYW1lIDwtIGdsdWUoIjE2Y2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV9kb3duX21mX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBtZl90cmVlX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChtZl9wbG90c19kb3duW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIG1mX2Jhcl9kb3duX2ZpbGVuYW1lIDwtIGdsdWUoIjE2Y2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV9kb3duX21mX3NpZ19iYXIucGRmIikKICAgICAgcHAoZmlsZSA9IG1mX2Jhcl9kb3duX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQobWZfcGxvdHNfZG93bltbImJhciJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGNjX3Bsb3RzX2Rvd24gPC0gcGxvdF9lbnJpY2hyZXN1bHQoY2Nfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBjY190cmVlX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMTZjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fY2Nfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IGNjX3RyZWVfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGNjX3Bsb3RzX2Rvd25bWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMTZjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fY2Nfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gY2NfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgYnBfcGxvdHNfZG93biA8LSBwbG90X2VucmljaHJlc3VsdChicF9zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGJwX3RyZWVfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIxNmNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fZG93bl9icF9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gYnBfdHJlZV9kb3duX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoYnBfcGxvdHNfZG93bltbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBicF9wbG90c19kb3duIDwtIHBsb3RfZW5yaWNocmVzdWx0KGJwX3NpZywgY2F0ZW9nb3JpZXMgPSBnb19jYXRlZ29yaWVzKQogICAgICBicF9iYXJfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIxNmNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fZG93bl9icF9zaWdfYmFyLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBicF9iYXJfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGJwX3Bsb3RzX2Rvd25bWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgfQogIH0KfQpgYGAKCkdpdmVuIHRoaXMgc3RyaWN0ZXIgZmlsdGVyLCBJIHRoaW5rIG5vIGdlbmVzIHBhc3MgaW4gdGhlIGdlbm9tZQpjb21wYXJpc29ucy4KCiMjIyBMb2NhdGlvbiBjb250cmFzdHMgd2l0aCBnZW5lcyByZW1vdmVkL2tlcHQKCldlIHdpbGwgbm93IHJlcGVhdCB0aGUgYWJvdmUgdGFza3Mgc2Vla2luZyBsb2NhdGlvbiBkaWZmZXJlbmNlcwppbnN0ZWFkIG9mIGdlbm90eXBlOyBlc3NlbnRpYWxseSBJIGNvcHkvcGFzdGVkIHRoZSBhYm92ZSB3aXRoCnMvZ2Vub3R5cGUvbG9jYXRpb24vZy4KCmBgYHtyfQpsb2NhdGlvbl90YWJsZXNfZnVsbCA8LSBjb21iaW5lX2RlX3RhYmxlcygKICBsb2NhdGlvbl9kZSwga2VlcGVycyA9IGxvY2F0aW9uX2tlZXBlcnMsIGxhYmVsX2NvbHVtbiA9IGxhYmVsX2NvbHVtbiwKICBleGNlbCA9IGdsdWUoIjE3ZnVsbF9sb2NhdGlvbl9jb250cmFzdHMvbG9jYXRpb25fZnVsbF90YWJsZXMtdnt2ZXJ9Lnhsc3giKSkKbG9jYXRpb25fdGFibGVzX2Z1bGwKbG9jYXRpb25fc2lnX2Z1bGwgPC0gZXh0cmFjdF9zaWduaWZpY2FudF9nZW5lcygKICBsb2NhdGlvbl90YWJsZXNfZnVsbCwgYWNjb3JkaW5nX3RvID0gImRlc2VxIiwKICBleGNlbCA9IGdsdWUoIjE3ZnVsbF9sb2NhdGlvbl9jb250cmFzdHMvbG9jYXRpb25fZnVsbF9zaWctdnt2ZXJ9Lnhsc3giKSkKbG9jYXRpb25fc2lnX2Z1bGwKbG9jYXRpb25fZnVsbF91cHNldCA8LSB1cHNldHJfc2lnKGxvY2F0aW9uX3NpZ19mdWxsKQpsb2NhdGlvbl9mdWxsX2ludGVyc2VjdHMgPC0gd3JpdGVfdXBzZXRfZ3JvdXBzKAogIGxvY2F0aW9uX2Z1bGxfdXBzZXQsCiAgZXhjZWwgPSAiZXhjZWwvMTdmdWxsX2xvY2F0aW9uX2NvbnRyYXN0cy9sb2NhdGlvbl9mdWxsX2dlbmVfZ3JvdXBzLnhsc3giKQpgYGAKCmBgYHtyfQpsb2NhdGlvbl90YWJsZXMgPC0gbGlzdCgpCmxvY2F0aW9uX3NpZyA8LSBsaXN0KCkKbG9jYXRpb25fZ3AgPC0gbGlzdCgpCmxvY2F0aW9uX2NwIDwtIGxpc3QoKQpmb3IgKGsgaW4gc2VxX2Fsb25nKGxvY2F0aW9uX2tlZXBlcnMpKSB7CiAgbmFtZSA8LSBuYW1lcyhsb2NhdGlvbl9rZWVwZXJzKVtrXQogIG1lc3NhZ2UoIkV4YW1pbmluZyAiLCBuYW1lKQogIGtlZXBlciA8LSBsb2NhdGlvbl9rZWVwZXJzW25hbWVdCiAgaW5jbHVkZXMgPC0gbG9jYXRpb25faW5jbHVzaW9uc1tbbmFtZV1dCiAgaW5jbHVkZV9uYW1lIDwtIHBhc3RlMCgiaW5jXyIsIG5hbWUpCiAgaW5jbHVkZV9kZl9uYW1lIDwtIHBhc3RlMCgiZGZfIiwgbmFtZSkKICBpbmNsdWRlX2RmIDwtIGxvY2F0aW9uX2luY2x1c2lvbnNbW2luY2x1ZGVfZGZfbmFtZV1dCiAgaW5jbHVkZXMgPC0gbG9jYXRpb25faW5jbHVzaW9uc1tbaW5jbHVkZV9uYW1lXV0KICBzdW1tYXJ5KHJvd25hbWVzKGxvY2F0aW9uX3NpZ19mdWxsW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSkgJWluJSBpbmNsdWRlcykKICBpbmNsdWRlX2ZpbGVuYW1lIDwtIGdsdWUoIjE4bG9jYXRpb25fY29udHJhc3RzL3tuYW1lfV9pbmNsdWRpbmdfd3Rfe2xmY19jdXRvZmZ9X2RlY3JlYXNlZF90YWJsZS12e3Zlcn0ueGxzeCIpCiAgaW5jbHVkZV9zaWdfZmlsZW5hbWUgPC0gZ2x1ZSgiMThsb2NhdGlvbl9jb250cmFzdHMve25hbWV9X2luY2x1ZGluZ193dF97bGZjX2N1dG9mZn1fZGVjcmVhc2VkX3NpZy12e3Zlcn0ueGxzeCIpCiAgbG9jYXRpb25fdGFibGVzW1tuYW1lXV0gPC0gY29tYmluZV9kZV90YWJsZXMoCiAgICBsb2NhdGlvbl9kZSwgZXh0cmFfYW5ub3QgPSBpbmNsdWRlX2RmLAogICAga2VlcGVycyA9IGtlZXBlciwgbGFiZWxfY29sdW1uID0gbGFiZWxfY29sdW1uLAogICAgZXhjZWwgPSBpbmNsdWRlX2ZpbGVuYW1lLCB3YW50ZWRfZ2VuZXMgPSBpbmNsdWRlcykKICBwcmludChsb2NhdGlvbl90YWJsZXNbW25hbWVdXSkKICBsb2NhdGlvbl9zaWdbW25hbWVdXSA8LSBleHRyYWN0X3NpZ25pZmljYW50X2dlbmVzKAogICAgbG9jYXRpb25fdGFibGVzW1tuYW1lXV0sIGFjY29yZGluZ190byA9ICJkZXNlcSIsCiAgICBleGNlbCA9IGluY2x1ZGVfc2lnX2ZpbGVuYW1lKQogIHByaW50KGxvY2F0aW9uX3NpZ1tbbmFtZV1dKQogIG51bV9yb3dzIDwtIG5yb3cobG9jYXRpb25fc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dKSArCiAgICBucm93KGxvY2F0aW9uX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJkb3ducyJdXVtbbmFtZV1dKQogIG1lc3NhZ2UoIlRoZXJlIGFyZSAiLCBudW1fcm93cywgIiBzaWduaWZpY2FudCB1cCBhbmQgZG93biBnZW5lcy4iKQogIGlmIChudW1fcm93cyA+IDEwKSB7CiAgICBsb2NhdGlvbl9ncFtbbmFtZV1dIDwtIGFsbF9ncHJvZmlsZXIobG9jYXRpb25fc2lnW1tuYW1lXV0sIHNwZWNpZXMgPSAibW11c2N1bHVzIikKICAgIGxvY2F0aW9uX2NwW1tuYW1lXV0gPC0gYWxsX2Nwcm9maWxlcigKICAgICAgbG9jYXRpb25fc2lnW1tuYW1lXV0sIGxvY2F0aW9uX3RhYmxlc1tbbmFtZV1dLAogICAgICBvcmdkYiA9ICJvcmcuTW0uZWcuZGIiLCBnb19sZXZlbCA9IGdvX2xldmVsLCBvcmdkYl9mcm9tID0gb3JnZGJfZnJvbSwKICAgICAgbWF4X2dyb3Vwc2l6ZSA9IG1heF9ncm91cHNpemUsIG9yZ2FuaXNtID0gIm1vdXNlIikKICAgIGNwX3dyaXR0ZW4gPC0gd3JpdGVfYWxsX2NwKGxvY2F0aW9uX2NwW1tuYW1lXV0sIHByZWZpeCA9ICIxOSIpCiAgICBncF93cml0dGVuIDwtIHdyaXRlX2FsbF9ncChsb2NhdGlvbl9ncFtbbmFtZV1dLCBwcmVmaXggPSAiMTkiKQogIH0KfQpgYGAKClByaW50IG91dCBhbGwgdGhlIHBsb3RzIGluIGEgc2VwYXJhdGUgYmxvY2suCgpgYGB7cn0KZm9yIChrIGluIHNlcV9hbG9uZyhsb2NhdGlvbl9rZWVwZXJzKSkgewogIG5hbWUgPC0gbmFtZXMobG9jYXRpb25fa2VlcGVycylba10KICBtZXNzYWdlKCJFeGFtaW5pbmcgIiwgbmFtZSkKICBrZWVwZXIgPC0gbG9jYXRpb25fa2VlcGVyc1tuYW1lXQogIGluY2x1ZGVzIDwtIGxvY2F0aW9uX2luY2x1c2lvbnNbW25hbWVdXQogIGluY2x1ZGVfbmFtZSA8LSBwYXN0ZTAoImluY18iLCBuYW1lKQogIGluY2x1ZGVfZGZfbmFtZSA8LSBwYXN0ZTAoImRmXyIsIG5hbWUpCiAgaW5jbHVkZV9kZiA8LSBsb2NhdGlvbl9pbmNsdXNpb25zW1tpbmNsdWRlX2RmX25hbWVdXQogIGluY2x1ZGVzIDwtIGxvY2F0aW9uX2luY2x1c2lvbnNbW2luY2x1ZGVfbmFtZV1dCiAgc3VtbWFyeShyb3duYW1lcyhsb2NhdGlvbl9zaWdfZnVsbFtbImRlc2VxIl1dW1sidXBzIl1dW1tuYW1lXV0pICVpbiUgaW5jbHVkZXMpCiAgbnVtX3Jvd3MgPC0gbnJvdyhsb2NhdGlvbl9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1sidXBzIl1dW1tuYW1lXV0pICsKICAgIG5yb3cobG9jYXRpb25fc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbImRvd25zIl1dW1tuYW1lXV0pCiAgbnJvdyhsb2NhdGlvbl9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1sidXBzIl1dW1tuYW1lXV0pCiAgbnJvdyhsb2NhdGlvbl9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSkKICBtZXNzYWdlKCJUaGVyZSBhcmUgIiwgbnVtX3Jvd3MsICIgc2lnbmlmaWNhbnQgdXAgYW5kIGRvd24gZ2VuZXMuIikKICBudW1fb2JqZWN0cyA8LSBsZW5ndGgobG9jYXRpb25fY3BbW25hbWVdXSkKICBpZiAobnVtX29iamVjdHMgPT0gMCkgewogICAgd2FybmluZygiU29tZXRoaW5nIHdlbnQgd3JvbmcgaW4gYWxsX2Nwcm9maWxlci4iKQogIH0gZWxzZSB7CiAgICB1cHAgPC0gd2hpY2goZ3JlcGwoeCA9IG5hbWVzKGxvY2F0aW9uX2NwW1tuYW1lXV0pLCBwYXR0ZXJuID0gIl91cCQiKSkKICAgIGRvd25wIDwtIHdoaWNoKGdyZXBsKHggPSBuYW1lcyhsb2NhdGlvbl9jcFtbbmFtZV1dKSwgcGF0dGVybiA9ICJfZG93biQiKSkKICAgIGlmIChsZW5ndGgodXBwKSA+IDApIHsKICAgICAgbWZfc2lnIDwtIGxvY2F0aW9uX2NwW1tuYW1lXV1bW3VwcF1dW1siZ29fZGF0YSJdXVtbIk1GX2VucmljaCJdXQogICAgICBjY19zaWcgPC0gbG9jYXRpb25fY3BbW25hbWVdXVtbdXBwXV1bWyJnb19kYXRhIl1dW1siQ0NfZW5yaWNoIl1dCiAgICAgIGJwX3NpZyA8LSBsb2NhdGlvbl9jcFtbbmFtZV1dW1t1cHBdXVtbImdvX2RhdGEiXV1bWyJCUF9lbnJpY2giXV0KICAgICAgbWZfcGxvdHNfdXAgPC0gcGxvdF9lbnJpY2hyZXN1bHQobWZfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBtZl90cmVlX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjE5Y2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9tZl9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gbWZfdHJlZV91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX3VwW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIG1mX2Jhcl91cF9maWxlbmFtZSA8LSBnbHVlKCIxOWNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfbWZfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gbWZfYmFyX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQobWZfcGxvdHNfdXBbWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBjY19wbG90c191cCA8LSBwbG90X2VucmljaHJlc3VsdChjY19zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGNjX3RyZWVfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMTljbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X3VwX2NjX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBjY190cmVlX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoY2NfcGxvdHNfdXBbWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfYmFyX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjE5Y2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9jY19zaWdfYmFyLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBjY19iYXJfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c191cFtbImJhciJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGJwX3Bsb3RzX3VwIDwtIHBsb3RfZW5yaWNocmVzdWx0KGJwX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgYnBfdHJlZV91cF9maWxlbmFtZSA8LSBnbHVlKCIxOWNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfYnBfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IGJwX3RyZWVfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChicF9wbG90c191cFtbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBicF9wbG90c191cCA8LSBwbG90X2VucmljaHJlc3VsdChicF9zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGJwX2Jhcl91cF9maWxlbmFtZSA8LSBnbHVlKCIxOWNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfYnBfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gYnBfYmFyX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoYnBfcGxvdHNfdXBbWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgfQogICAgaWYgKGxlbmd0aChkb3ducCkgPiAwKSB7CiAgICAgIG1mX3NpZyA8LSBsb2NhdGlvbl9jcFtbbmFtZV1dW1tkb3ducF1dW1siZ29fZGF0YSJdXVtbIk1GX2VucmljaCJdXQogICAgICBjY19zaWcgPC0gbG9jYXRpb25fY3BbW25hbWVdXVtbZG93bnBdXVtbImdvX2RhdGEiXV1bWyJDQ19lbnJpY2giXV0KICAgICAgYnBfc2lnIDwtIGxvY2F0aW9uX2NwW1tuYW1lXV1bW2Rvd25wXV1bWyJnb19kYXRhIl1dW1siQlBfZW5yaWNoIl1dCiAgICAgIG1mX3Bsb3RzX2Rvd24gPC0gcGxvdF9lbnJpY2hyZXN1bHQobWZfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBtZl90cmVlX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMTljbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fbWZfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IG1mX3RyZWVfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX2Rvd25bWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgbWZfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMTljbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fbWZfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gbWZfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChtZl9wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfcGxvdHNfZG93biA8LSBwbG90X2VucmljaHJlc3VsdChjY19zaWcsIHNob3dDYXRlZ29yeSA9IDEyKQogICAgICBjY190cmVlX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMTljbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fY2Nfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IGNjX3RyZWVfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGNjX3Bsb3RzX2Rvd25bWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMTljbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fY2Nfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gY2NfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgYnBfcGxvdHNfZG93biA8LSBwbG90X2VucmljaHJlc3VsdChicF9zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGJwX3RyZWVfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIxOWNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fZG93bl9icF9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gYnBfdHJlZV9kb3duX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoYnBfcGxvdHNfZG93bltbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBicF9wbG90c19kb3duIDwtIHBsb3RfZW5yaWNocmVzdWx0KGJwX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgYnBfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMTljbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fYnBfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gYnBfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChicF9wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgIH0KICB9Cn0KYGBgCgpDb2xlbnNvIHNlbnQgYSBzcGVjaWZpYyBxdWVyeSBvZiBpbnRlcmVzdCwgY29tcGFyaW5nIFNDTiB2cy4gUmV0aW5hcwphdCBwMDggaW4gdGhlIGhldGVyb3p5Z290ZXMgaW5jbHVkaW5nIGEgc2V0IG9mIGdlbmVzIG9mIHBhcnRpY3VsYXIKaW50ZXJlc3QuICBQZXJoYXBzIEkgY2FuIHVzZSBzb21lIG9mIHRoZXNlIGFzIG1hcmtlcnMgdG8gcXVhbGl0eQpjb250cm9sIG15IHdvcmsgaW4gdGhlIGZ1dHVyZT8KCkhlcmUgYXJlIHRoZSBnZW5lczoKCk9wbjQsIEVvbWVzLCBUcnBjNywgT3BybTEsIE5yNGEzLCBUYngyMCwgSXJ4NiwgQVc1NTE5ODQsIFBjZGgxOSwKQWRjeWFwMSwgQmFpYXAzLCBDaGwxLCBHcmluM2EsIElnZjEsIEdyaWExLCBHcmluMmQsIEdyaW4zYSwgQ2hybmE2LApDaHJuYTMsIEh0cjVhLCBIdHIyYSwgSHRyNywgSXJ4NCwgUGx4bkMxLCBTZW1hNmQsIFNlbWE0ZiwgU2VtYTRhLApTZW1hNmIsIExycmM0YiwgTHJyYzU4LCBMcnJjM2IsIFdudDQsIFdudDliLCBDdHhuMywgVGVubTEsIEduYTE0LApSZ3M0LCBSZ3M2LCBSZ3M1CgpgYGB7cn0KdGFibGVfaW5wdXQgPC0gbG9jYXRpb25fdGFibGVzW1sic3JfcDA4X2hldCJdXQp0YWJsZV9uYW1lIDwtICJzcl9wMDhfaGV0Igp0YWJsZSA8LSB0YWJsZV9pbnB1dFtbImRhdGEiXV1bW3RhYmxlX25hbWVdXQppbnRlcmVzdGluZ19nZW5lcyA8LSBjKCJPcG40IiwgIkVvbWVzIiwgIlRycGM3IiwgIk9wcm0xIiwgIk5yNGEzIiwgIlRieDIwIiwKICAgICAgICAgICAgICAgICAgICAgICAiSXJ4NiIsICJBVzU1MTk4NCIsICJQY2RoMTkiLCAiQWRjeWFwMXIxIiwgIkJhaWFwMyIsCiAgICAgICAgICAgICAgICAgICAgICAgIkNobDEiLCAiR3JpbjNhIiwgIklnZjEiLCAiR3JpYTEiLCAiR3JpbjJkIiwgIkdyaW4zYSIsCiAgICAgICAgICAgICAgICAgICAgICAgIkNocm5hNiIsICJDaHJuYTMiLCAiSHRyNWEiLCAiSHRyMmEiLCAiSHRyNyIsICJJcng0IiwKICAgICAgICAgICAgICAgICAgICAgICAiUGx4bkMxIiwgIlNlbWE2ZCIsICJTZW1hNGYiLCAiU2VtYTRhIiwgIlNlbWE2YiIsICJMcnJjNGIiLAogICAgICAgICAgICAgICAgICAgICAgICJMcnJjNTgiLCAiTHJyYzNiIiwgIldudDQiLCAiV250OWIiLCAiQ3R4bjMiLCAiVGVubTEiLCAiR25hMTQiLAogICAgICAgICAgICAgICAgICAgICAgICJSZ3M0IiwgIlJnczYiLCAiUmdzNSIsICJQb3U0ZjIiLCAiQ2hybmIzIiwgIkJjYW4iKQpzcl9wMDhfaGV0X3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXFfYWRqcCIsCiAgY29sb3JfbG93ID0gY29sb3JzW1siaGV0X3JldGluYSJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImhldF9zY24iXV0sCiAgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzLCBhbHBoYSA9IDEuMCwKICBzaXplID0gNCkKcHAoZmlsZSA9ICIyMGxvY2F0aW9uX21hX3ZvbGNhbm8vc3JfcDA4X2hldF92b2xjYW5vLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKc3JfcDA4X2hldF92b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpzcl9wMDhfaGV0X3ZvbGNhbm9bWyJwbG90Il1dCnNyX3AwOF9oZXRfbWEgPC0gcGxvdF9tYV9jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGV4cHJfY29sID0gImRlc2VxX2Jhc2VtZWFuIiwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwKICBjb2xvcl9sb3cgPSBjb2xvcnNbWyJoZXRfcmV0aW5hIl1dLCBjb2xvcl9oaWdoID0gY29sb3JzW1siaGV0X3NjbiJdXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzLCBvdXRsaW5lID0gb3V0bGluZSkKcHAoZmlsZSA9ICIyMGxvY2F0aW9uX21hX3ZvbGNhbm8vc3JfcDA4X2hldF9tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCnNyX3AwOF9oZXRfbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnNyX3AwOF9oZXRfbWFbWyJwbG90Il1dCmBgYAoKIyMjIEFsc28gdGhlIGtvCgpgYGB7cn0KdGFibGVfaW5wdXQgPC0gbG9jYXRpb25fdGFibGVzW1sic3JfcDA4X2tvIl1dCnRhYmxlX25hbWUgPC0gInNyX3AwOF9rbyIKdGFibGUgPC0gdGFibGVfaW5wdXRbWyJkYXRhIl1dW1t0YWJsZV9uYW1lXV0Kc3JfcDA4X2tvX3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXFfYWRqcCIsCiAgY29sb3JfbG93ID0gY29sb3JzW1sia29fcmV0aW5hIl1dLCBjb2xvcl9oaWdoID0gY29sb3JzW1sia29fc2NuIl1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQpCnBwKGZpbGUgPSAiMjBsb2NhdGlvbl9tYV92b2xjYW5vL3NyX3AwOF9rb192b2xjYW5vLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKc3JfcDA4X2tvX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnNyX3AwOF9rb192b2xjYW5vW1sicGxvdCJdXQpzcl9wMDhfa29fbWEgPC0gcGxvdF9tYV9jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGV4cHJfY29sID0gImRlc2VxX2Jhc2VtZWFuIiwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwKICBjb2xvcl9sb3cgPSBjb2xvcnNbWyJrb19yZXRpbmEiXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJrb19zY24iXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgb3V0bGluZSA9IG91dGxpbmUpCnBwKGZpbGUgPSAiMjBsb2NhdGlvbl9tYV92b2xjYW5vL3NyX3AwOF9rb19tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCnNyX3AwOF9rb19tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKc3JfcDA4X2tvX21hW1sicGxvdCJdXQpgYGAKCmBgYHtyfQp0YWJsZV9pbnB1dCA8LSBsb2NhdGlvbl90YWJsZXNbWyJzcl9wMTVfaGV0Il1dCnRhYmxlX25hbWUgPC0gInNyX3AxNV9oZXQiCnRhYmxlIDwtIHRhYmxlX2lucHV0W1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCmludGVyZXN0aW5nX2dlbmVzIDwtIGMoIk9wbjQiLCAiRW9tZXMiLCAiVHJwYzciLCAiT3BybTEiLCAiTnI0YTMiLCAiVGJ4MjAiLAogICAgICAgICAgICAgICAgICAgICAgICJJcng2IiwgIkFXNTUxOTg0IiwgIlBjZGgxOSIsICJBZGN5YXAxcjEiLCAiQmFpYXAzIiwKICAgICAgICAgICAgICAgICAgICAgICAiQ2hsMSIsICJHcmluM2EiLCAiSWdmMSIsICJHcmlhMSIsICJHcmluMmQiLCAiR3JpbjNhIiwKICAgICAgICAgICAgICAgICAgICAgICAiQ2hybmE2IiwgIkNocm5hMyIsICJIdHI1YSIsICJIdHIyYSIsICJIdHI3IiwgIklyeDQiLAogICAgICAgICAgICAgICAgICAgICAgICJQbHhuQzEiLCAiU2VtYTZkIiwgIlNlbWE0ZiIsICJTZW1hNGEiLCAiU2VtYTZiIiwgIkxycmM0YiIsCiAgICAgICAgICAgICAgICAgICAgICAgIkxycmM1OCIsICJMcnJjM2IiLCAiV250NCIsICJXbnQ5YiIsICJDdHhuMyIsICJUZW5tMSIsICJHbmExNCIsCiAgICAgICAgICAgICAgICAgICAgICAgIlJnczQiLCAiUmdzNiIsICJSZ3M1IiwgIlBvdTRmMiIsICJDaHJuYjMiLCAiQmNhbiIpCnNyX3AxNV9oZXRfdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwKICBjb2xvcl9sb3cgPSBjb2xvcnNbWyJoZXRfcmV0aW5hIl1dLCBjb2xvcl9oaWdoID0gY29sb3JzW1siaGV0X3NjbiJdXSwKICBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gaW50ZXJlc3RpbmdfZ2VuZXMsIGFscGhhID0gMS4wLAogIHNpemUgPSA0KQpwcChmaWxlID0gIjIwbG9jYXRpb25fbWFfdm9sY2Fuby9zcl9wMTVfaGV0X3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpzcl9wMTVfaGV0X3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnNyX3AxNV9oZXRfdm9sY2Fub1tbInBsb3QiXV0Kc3JfcDE1X2hldF9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImhldF9yZXRpbmEiXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJoZXRfc2NuIl1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gaW50ZXJlc3RpbmdfZ2VuZXMsIG91dGxpbmUgPSBvdXRsaW5lKQpwcChmaWxlID0gIjIwbG9jYXRpb25fbWFfdm9sY2Fuby9zcl9wMTVfaGV0X21hLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKc3JfcDE1X2hldF9tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKc3JfcDE1X2hldF9tYVtbInBsb3QiXV0KYGBgCgojIyMgQWxzbyB0aGUga28KCmBgYHtyfQp0YWJsZV9pbnB1dCA8LSBsb2NhdGlvbl90YWJsZXNbWyJzcl9wMTVfa28iXV0KdGFibGVfbmFtZSA8LSAic3JfcDE1X2tvIgp0YWJsZSA8LSB0YWJsZV9pbnB1dFtbImRhdGEiXV1bW3RhYmxlX25hbWVdXQpzcl9wMTVfa29fdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwKICBjb2xvcl9sb3cgPSBjb2xvcnNbWyJrb19yZXRpbmEiXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJrb19zY24iXV0sCiAgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzLCBhbHBoYSA9IDEuMCwKICBzaXplID0gNCwgbWluLnNlZ21lbnQubGVuZ3RoID0gMCwgcG9pbnQucGFkZGluZyA9IDAuMikKcHAoZmlsZSA9ICIyMGxvY2F0aW9uX21hX3ZvbGNhbm8vc3JfcDE1X2tvX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSAxMiwgaGVpZ2h0ID0gMTIpCnNyX3AxNV9rb192b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpzcl9wMTVfa29fdm9sY2Fub1tbInBsb3QiXV0Kc3JfcDE1X2tvX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gY29sb3JzW1sia29fcmV0aW5hIl1dLCBjb2xvcl9oaWdoID0gY29sb3JzW1sia29fc2NuIl1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gaW50ZXJlc3RpbmdfZ2VuZXMsIG91dGxpbmUgPSBvdXRsaW5lKQpwcChmaWxlID0gIjIwbG9jYXRpb25fbWFfdm9sY2Fuby9zcl9wMTVfa29fbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpzcl9wMTVfa29fbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnNyX3AxNV9rb19tYVtbInBsb3QiXV0KYGBgCgojIyMjIFRlc3QgYSBzcGVjaWZpYyBsb2NhdGlvbiBxdWVyeSBmb3IgZHVwbGljYXRlZCBJRHMKCkxldCB1cyBzZWUgaWYgYW55IEVuc2VtYmwgZ2VuZSBJRHMgYW5kL29yIE1HSSBJRHMgYXJlIHNoYXJlZCBpbiB0aGUKd29ya3NoZWV0IGxvY2F0aW9uX3NyX3AwOF9rb19pbmNsdWRpbmdfd3RfMC4xX2RlY3JlYXNlZF9zaWcgdXAvZG93bi4KCmBgYHtyfQp0ZXN0X3RhYmxlX3VwIDwtIGxvY2F0aW9uX3NpZ1tbInNyX3AwOF9rbyJdXVtbImRlc2VxIl1dW1sidXBzIl1dW1sxXV0KdGVzdF90YWJsZV9kb3duIDwtIGxvY2F0aW9uX3NpZ1tbInNyX3AwOF9rbyJdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bWzFdXQoKcXVlcnkgPC0gbGlzdCgidXAiID0gcm93bmFtZXModGVzdF90YWJsZV91cCksCiAgICAgICAgICAgICAgImRvd24iID0gcm93bmFtZXModGVzdF90YWJsZV9kb3duKSkKcXVlcnlfdXBzZXQgPC0gVXBTZXRSOjpmcm9tTGlzdChxdWVyeSkKVXBTZXRSOjp1cHNldChxdWVyeV91cHNldCkKCnF1ZXJ5IDwtIGxpc3QoInVwIiA9IHRlc3RfdGFibGVfdXBbWyJtZ2lfc3ltYm9sIl1dLAogICAgICAgICAgICAgICJkb3duIiA9IHRlc3RfdGFibGVfZG93bltbIm1naV9zeW1ib2wiXV0pCnF1ZXJ5X3Vwc2V0IDwtIFVwU2V0Ujo6ZnJvbUxpc3QocXVlcnkpClVwU2V0Ujo6dXBzZXQocXVlcnlfdXBzZXQpCiMjIG9rLCBnb29kLgpgYGAKCiMjIyMgUmVwZWF0IHdpdGggdGhlIHN0cmljdCBmaWx0ZXIKCmBgYHtyfQpsb2NhdGlvbl9zdHJpY3RfdGFibGVzIDwtIGxpc3QoKQpsb2NhdGlvbl9zdHJpY3Rfc2lnIDwtIGxpc3QoKQpsb2NhdGlvbl9zdHJpY3RfZ3AgPC0gbGlzdCgpCmxvY2F0aW9uX3N0cmljdF9jcCA8LSBsaXN0KCkKZm9yIChrIGluIHNlcV9hbG9uZyhsb2NhdGlvbl9rZWVwZXJzKSkgewogIG5hbWUgPC0gbmFtZXMobG9jYXRpb25fa2VlcGVycylba10KICBtZXNzYWdlKCJFeGFtaW5pbmcgIiwgbmFtZSkKICBrZWVwZXIgPC0gbG9jYXRpb25fa2VlcGVyc1tuYW1lXQogIGluY2x1ZGVzIDwtIGxvY2F0aW9uX2luY2x1c2lvbnNfc3RyaWN0W1tuYW1lXV0KICBpbmNsdWRlX25hbWUgPC0gcGFzdGUwKCJpbmNfIiwgbmFtZSkKICBpbmNsdWRlX2RmX25hbWUgPC0gcGFzdGUwKCJkZl8iLCBuYW1lKQogIGluY2x1ZGVfZGYgPC0gbG9jYXRpb25faW5jbHVzaW9uc19zdHJpY3RbW2luY2x1ZGVfZGZfbmFtZV1dCiAgaW5jbHVkZXMgPC0gbG9jYXRpb25faW5jbHVzaW9uc19zdHJpY3RbW2luY2x1ZGVfbmFtZV1dCiAgZm91bmRfaW5jbHVkZXMgPC0gcm93bmFtZXMobG9jYXRpb25fc2lnX2Z1bGxbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dKSAlaW4lIGluY2x1ZGVzCiAgc3VtbWFyeShmb3VuZF9pbmNsdWRlcykKICBpZiAoc3VtKGZvdW5kX2luY2x1ZGVzKSA9PSAwKSB7CiAgICBuZXh0CiAgfQogIGluY2x1ZGVfZmlsZW5hbWUgPC0gZ2x1ZSgiMjFsb2NhdGlvbl9zdHJpY3RfY29udHJhc3RzL3tuYW1lfV9pbmNsdWRpbmdfd3Rfe2xmY19jdXRvZmZ9X2RlY3JlYXNlZF90YWJsZS12e3Zlcn0ueGxzeCIpCiAgaW5jbHVkZV9zaWdfZmlsZW5hbWUgPC0gZ2x1ZSgiMjFsb2NhdGlvbl9zdHJpY3RfY29udHJhc3RzL3tuYW1lfV9pbmNsdWRpbmdfd3Rfe2xmY19jdXRvZmZ9X2RlY3JlYXNlZF9zaWctdnt2ZXJ9Lnhsc3giKQogIGxvY2F0aW9uX3N0cmljdF90YWJsZXNbW25hbWVdXSA8LSBjb21iaW5lX2RlX3RhYmxlcygKICAgIGxvY2F0aW9uX2RlLCBleHRyYV9hbm5vdCA9IGluY2x1ZGVfZGYsCiAgICBrZWVwZXJzID0ga2VlcGVyLCBsYWJlbF9jb2x1bW4gPSBsYWJlbF9jb2x1bW4sCiAgICBleGNlbCA9IGluY2x1ZGVfZmlsZW5hbWUsIHdhbnRlZF9nZW5lcyA9IGluY2x1ZGVzKQogIHByaW50KGxvY2F0aW9uX3N0cmljdF90YWJsZXNbW25hbWVdXSkKICBsb2NhdGlvbl9zdHJpY3Rfc2lnW1tuYW1lXV0gPC0gZXh0cmFjdF9zaWduaWZpY2FudF9nZW5lcygKICAgIGxvY2F0aW9uX3N0cmljdF90YWJsZXNbW25hbWVdXSwgYWNjb3JkaW5nX3RvID0gImRlc2VxIiwKICAgIGV4Y2VsID0gaW5jbHVkZV9zaWdfZmlsZW5hbWUpCiAgcHJpbnQobG9jYXRpb25fc3RyaWN0X3NpZ1tbbmFtZV1dKQogIG51bV9yb3dzIDwtIG5yb3cobG9jYXRpb25fc3RyaWN0X3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSkgKwogICAgbnJvdyhsb2NhdGlvbl9zdHJpY3Rfc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbImRvd25zIl1dW1tuYW1lXV0pCiAgbWVzc2FnZSgiVGhlcmUgYXJlICIsIG51bV9yb3dzLCAiIHNpZ25pZmljYW50IHVwIGFuZCBkb3duIGdlbmVzLiIpCiAgaWYgKG51bV9yb3dzID4gMTApIHsKICAgIGxvY2F0aW9uX3N0cmljdF9ncFtbbmFtZV1dIDwtIGFsbF9ncHJvZmlsZXIobG9jYXRpb25fc3RyaWN0X3NpZ1tbbmFtZV1dLCBzcGVjaWVzID0gIm1tdXNjdWx1cyIpCiAgICBsb2NhdGlvbl9zdHJpY3RfY3BbW25hbWVdXSA8LSBhbGxfY3Byb2ZpbGVyKAogICAgICBsb2NhdGlvbl9zdHJpY3Rfc2lnW1tuYW1lXV0sIGxvY2F0aW9uX3N0cmljdF90YWJsZXNbW25hbWVdXSwKICAgICAgb3JnZGIgPSAib3JnLk1tLmVnLmRiIiwgZ29fbGV2ZWwgPSBnb19sZXZlbCwgb3JnZGJfZnJvbSA9IG9yZ2RiX2Zyb20sCiAgICAgIG1heF9ncm91cHNpemUgPSBtYXhfZ3JvdXBzaXplLCBvcmdhbmlzbSA9ICJtb3VzZSIpCiAgICBjcF93cml0dGVuIDwtIHdyaXRlX2FsbF9jcChsb2NhdGlvbl9zdHJpY3RfY3BbW25hbWVdXSwgcHJlZml4ID0gIjIyIiwgc3VmZml4ID0gInN0cmljdCIpCiAgICBncF93cml0dGVuIDwtIHdyaXRlX2FsbF9ncChsb2NhdGlvbl9zdHJpY3RfZ3BbW25hbWVdXSwgcHJlZml4ID0gIjIyIiwgc3VmZml4ID0gInN0cmljdCIpCiAgfQp9CmBgYAoKUHJpbnQgb3V0IGFsbCB0aGUgcGxvdHMgaW4gYSBzZXBhcmF0ZSBibG9jay4KCmBgYHtyfQpmb3IgKGsgaW4gc2VxX2Fsb25nKGxvY2F0aW9uX2tlZXBlcnMpKSB7CiAgbmFtZSA8LSBuYW1lcyhsb2NhdGlvbl9rZWVwZXJzKVtrXQogIG1lc3NhZ2UoIkV4YW1pbmluZyAiLCBuYW1lKQogIGtlZXBlciA8LSBsb2NhdGlvbl9rZWVwZXJzW25hbWVdCiAgaW5jbHVkZXMgPC0gbG9jYXRpb25faW5jbHVzaW9uc19zdHJpY3RbW25hbWVdXQogIGluY2x1ZGVfbmFtZSA8LSBwYXN0ZTAoImluY18iLCBuYW1lKQogIGluY2x1ZGVfZGZfbmFtZSA8LSBwYXN0ZTAoImRmXyIsIG5hbWUpCiAgaW5jbHVkZV9kZiA8LSBsb2NhdGlvbl9pbmNsdXNpb25zX3N0cmljdFtbaW5jbHVkZV9kZl9uYW1lXV0KICBpbmNsdWRlcyA8LSBsb2NhdGlvbl9pbmNsdXNpb25zX3N0cmljdFtbaW5jbHVkZV9uYW1lXV0KICBzdW1tYXJ5KHJvd25hbWVzKGxvY2F0aW9uX3NpZ19mdWxsW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSkgJWluJSBpbmNsdWRlcykKICBudW1fcm93cyA8LSBucm93KGxvY2F0aW9uX3N0cmljdF9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1sidXBzIl1dW1tuYW1lXV0pICsKICAgIG5yb3cobG9jYXRpb25fc3RyaWN0X3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJkb3ducyJdXVtbbmFtZV1dKQogIG5yb3cobG9jYXRpb25fc3RyaWN0X3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSkKICBucm93KGxvY2F0aW9uX3N0cmljdF9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSkKICBtZXNzYWdlKCJUaGVyZSBhcmUgIiwgbnVtX3Jvd3MsICIgc2lnbmlmaWNhbnQgdXAgYW5kIGRvd24gZ2VuZXMuIikKICBudW1fb2JqZWN0cyA8LSBsZW5ndGgobG9jYXRpb25fc3RyaWN0X2NwW1tuYW1lXV0pCiAgaWYgKG51bV9vYmplY3RzID09IDApIHsKICAgIHdhcm5pbmcoIlNvbWV0aGluZyB3ZW50IHdyb25nIGluIGFsbF9jcHJvZmlsZXIuIikKICB9IGVsc2UgewogICAgdXBwIDwtIHdoaWNoKGdyZXBsKHggPSBuYW1lcyhsb2NhdGlvbl9zdHJpY3RfY3BbW25hbWVdXSksIHBhdHRlcm4gPSAiX3VwJCIpKQogICAgZG93bnAgPC0gd2hpY2goZ3JlcGwoeCA9IG5hbWVzKGxvY2F0aW9uX3N0cmljdF9jcFtbbmFtZV1dKSwgcGF0dGVybiA9ICJfZG93biQiKSkKICAgIGlmIChsZW5ndGgodXBwKSA+IDApIHsKICAgICAgbWZfc2lnIDwtIGxvY2F0aW9uX3N0cmljdF9jcFtbbmFtZV1dW1t1cHBdXVtbImdvX2RhdGEiXV1bWyJNRl9lbnJpY2giXV0KICAgICAgY2Nfc2lnIDwtIGxvY2F0aW9uX3N0cmljdF9jcFtbbmFtZV1dW1t1cHBdXVtbImdvX2RhdGEiXV1bWyJDQ19lbnJpY2giXV0KICAgICAgYnBfc2lnIDwtIGxvY2F0aW9uX3N0cmljdF9jcFtbbmFtZV1dW1t1cHBdXVtbImdvX2RhdGEiXV1bWyJCUF9lbnJpY2giXV0KICAgICAgbWZfcGxvdHNfdXAgPC0gcGxvdF9lbnJpY2hyZXN1bHQobWZfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBtZl90cmVlX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjIzY2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9tZl9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gbWZfdHJlZV91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX3VwW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIG1mX2Jhcl91cF9maWxlbmFtZSA8LSBnbHVlKCIyM2NsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfbWZfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gbWZfYmFyX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQobWZfcGxvdHNfdXBbWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBjY19wbG90c191cCA8LSBwbG90X2VucmljaHJlc3VsdChjY19zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGNjX3RyZWVfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMjNjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X3VwX2NjX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBjY190cmVlX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoY2NfcGxvdHNfdXBbWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfYmFyX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjIzY2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9jY19zaWdfYmFyLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBjY19iYXJfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c191cFtbImJhciJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGJwX3Bsb3RzX3VwIDwtIHBsb3RfZW5yaWNocmVzdWx0KGJwX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgYnBfdHJlZV91cF9maWxlbmFtZSA8LSBnbHVlKCIyM2NsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfYnBfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IGJwX3RyZWVfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChicF9wbG90c191cFtbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBicF9wbG90c191cCA8LSBwbG90X2VucmljaHJlc3VsdChicF9zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGJwX2Jhcl91cF9maWxlbmFtZSA8LSBnbHVlKCIyM2NsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfYnBfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gYnBfYmFyX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoYnBfcGxvdHNfdXBbWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgfQogICAgaWYgKGxlbmd0aChkb3ducCkgPiAwKSB7CiAgICAgIG1mX3NpZyA8LSBsb2NhdGlvbl9zdHJpY3RfY3BbW25hbWVdXVtbZG93bnBdXVtbImdvX2RhdGEiXV1bWyJNRl9lbnJpY2giXV0KICAgICAgY2Nfc2lnIDwtIGxvY2F0aW9uX3N0cmljdF9jcFtbbmFtZV1dW1tkb3ducF1dW1siZ29fZGF0YSJdXVtbIkNDX2VucmljaCJdXQogICAgICBicF9zaWcgPC0gbG9jYXRpb25fc3RyaWN0X2NwW1tuYW1lXV1bW2Rvd25wXV1bWyJnb19kYXRhIl1dW1siQlBfZW5yaWNoIl1dCiAgICAgIG1mX3Bsb3RzX2Rvd24gPC0gcGxvdF9lbnJpY2hyZXN1bHQobWZfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBtZl90cmVlX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMjNjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fbWZfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IG1mX3RyZWVfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX2Rvd25bWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgbWZfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMjNjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fbWZfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gbWZfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChtZl9wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfcGxvdHNfZG93biA8LSBwbG90X2VucmljaHJlc3VsdChjY19zaWcsIHNob3dDYXRlZ29yeSA9IDEyKQogICAgICBjY190cmVlX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMjNjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fY2Nfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IGNjX3RyZWVfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGNjX3Bsb3RzX2Rvd25bWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMjNjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fY2Nfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gY2NfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgYnBfcGxvdHNfZG93biA8LSBwbG90X2VucmljaHJlc3VsdChicF9zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGJwX3RyZWVfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIyM2NsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fZG93bl9icF9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gYnBfdHJlZV9kb3duX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoYnBfcGxvdHNfZG93bltbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBicF9wbG90c19kb3duIDwtIHBsb3RfZW5yaWNocmVzdWx0KGJwX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgYnBfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMjNjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fYnBfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gYnBfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChicF9wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgIH0KICB9Cn0KYGBgCgpDb2xlbnNvIHNlbnQgYSBzcGVjaWZpYyBxdWVyeSBvZiBpbnRlcmVzdCwgY29tcGFyaW5nIFNDTiB2cy4gUmV0aW5hcwphdCBwMDggaW4gdGhlIGhldGVyb3p5Z290ZXMgaW5jbHVkaW5nIGEgc2V0IG9mIGdlbmVzIG9mIHBhcnRpY3VsYXIKaW50ZXJlc3QuICBQZXJoYXBzIEkgY2FuIHVzZSBzb21lIG9mIHRoZXNlIGFzIG1hcmtlcnMgdG8gcXVhbGl0eQpjb250cm9sIG15IHdvcmsgaW4gdGhlIGZ1dHVyZT8KCkhlcmUgYXJlIHRoZSBnZW5lczoKCk9wbjQsIEVvbWVzLCBUcnBjNywgT3BybTEsIE5yNGEzLCBUYngyMCwgSXJ4NiwgQVc1NTE5ODQsIFBjZGgxOSwKQWRjeWFwMSwgQmFpYXAzLCBDaGwxLCBHcmluM2EsIElnZjEsIEdyaWExLCBHcmluMmQsIEdyaW4zYSwgQ2hybmE2LApDaHJuYTMsIEh0cjVhLCBIdHIyYSwgSHRyNywgSXJ4NCwgUGx4bkMxLCBTZW1hNmQsIFNlbWE0ZiwgU2VtYTRhLApTZW1hNmIsIExycmM0YiwgTHJyYzU4LCBMcnJjM2IsIFdudDQsIFdudDliLCBDdHhuMywgVGVubTEsIEduYTE0LApSZ3M0LCBSZ3M2LCBSZ3M1CgpgYGB7cn0KdGFibGVfaW5wdXQgPC0gbG9jYXRpb25fc3RyaWN0X3RhYmxlc1tbInNyX3AwOF9oZXQiXV0KdGFibGVfbmFtZSA8LSAic3JfcDA4X2hldCIKdGFibGUgPC0gdGFibGVfaW5wdXRbWyJkYXRhIl1dW1t0YWJsZV9uYW1lXV0KaW50ZXJlc3RpbmdfZ2VuZXMgPC0gYygiT3BuNCIsICJFb21lcyIsICJUcnBjNyIsICJPcHJtMSIsICJOcjRhMyIsICJUYngyMCIsCiAgICAgICAgICAgICAgICAgICAgICAgIklyeDYiLCAiQVc1NTE5ODQiLCAiUGNkaDE5IiwgIkFkY3lhcDFyMSIsICJCYWlhcDMiLAogICAgICAgICAgICAgICAgICAgICAgICJDaGwxIiwgIkdyaW4zYSIsICJJZ2YxIiwgIkdyaWExIiwgIkdyaW4yZCIsICJHcmluM2EiLAogICAgICAgICAgICAgICAgICAgICAgICJDaHJuYTYiLCAiQ2hybmEzIiwgIkh0cjVhIiwgIkh0cjJhIiwgIkh0cjciLCAiSXJ4NCIsCiAgICAgICAgICAgICAgICAgICAgICAgIlBseG5DMSIsICJTZW1hNmQiLCAiU2VtYTRmIiwgIlNlbWE0YSIsICJTZW1hNmIiLCAiTHJyYzRiIiwKICAgICAgICAgICAgICAgICAgICAgICAiTHJyYzU4IiwgIkxycmMzYiIsICJXbnQ0IiwgIldudDliIiwgIkN0eG4zIiwgIlRlbm0xIiwgIkduYTE0IiwKICAgICAgICAgICAgICAgICAgICAgICAiUmdzNCIsICJSZ3M2IiwgIlJnczUiLCAiUG91NGYyIiwgIkNocm5iMyIsICJCY2FuIikKc3JfcDA4X2hldF92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImhldF9yZXRpbmEiXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJoZXRfc2NuIl1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQpCnBwKGZpbGUgPSAiMjRsb2NhdGlvbl9tYV92b2xjYW5vX3N0cmljdC9zcl9wMDhfaGV0X3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpzcl9wMDhfaGV0X3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnNyX3AwOF9oZXRfdm9sY2Fub1tbInBsb3QiXV0Kc3JfcDA4X2hldF9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImhldF9yZXRpbmEiXV0sIGNvbG9yX2hpZ2ggPSBjb2xvcnNbWyJoZXRfc2NuIl1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gaW50ZXJlc3RpbmdfZ2VuZXMsIG91dGxpbmUgPSBvdXRsaW5lKQpwcChmaWxlID0gIjI0bG9jYXRpb25fbWFfdm9sY2Fub19zdHJpY3Qvc3JfcDA4X2hldF9tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCnNyX3AwOF9oZXRfbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnNyX3AwOF9oZXRfbWFbWyJwbG90Il1dCmBgYAoKIyMjIEFsc28gdGhlIGtvCgpgYGB7cn0KdGFibGVfaW5wdXQgPC0gbG9jYXRpb25fc3RyaWN0X3RhYmxlc1tbInNyX3AwOF9rbyJdXQp0YWJsZV9uYW1lIDwtICJzcl9wMDhfa28iCnRhYmxlIDwtIHRhYmxlX2lucHV0W1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCnNyX3AwOF9rb192b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX3JldGluYSJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImtvX3NjbiJdXSwKICBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gaW50ZXJlc3RpbmdfZ2VuZXMsIGFscGhhID0gMS4wLAogIHNpemUgPSA0KQpwcChmaWxlID0gIjI0bG9jYXRpb25fbWFfdm9sY2Fub19zdHJpY3Qvc3JfcDA4X2tvX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQpzcl9wMDhfa29fdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKc3JfcDA4X2tvX3ZvbGNhbm9bWyJwbG90Il1dCnNyX3AwOF9rb19tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX3JldGluYSJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImtvX3NjbiJdXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzLCBvdXRsaW5lID0gb3V0bGluZSkKcHAoZmlsZSA9ICIyNGxvY2F0aW9uX21hX3ZvbGNhbm9fc3RyaWN0L3NyX3AwOF9rb19tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCnNyX3AwOF9rb19tYVtbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKc3JfcDA4X2tvX21hW1sicGxvdCJdXQpgYGAKCiMjIyBBbHNvIHRoZSBrbwoKYGBge3J9CnRhYmxlX2lucHV0IDwtIGxvY2F0aW9uX3RhYmxlc1tbInNyX3AxNV9rbyJdXQp0YWJsZV9uYW1lIDwtICJzcl9wMTVfa28iCnRhYmxlIDwtIHRhYmxlX2lucHV0W1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCnNyX3AxNV9rb192b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLAogIGNvbG9yX2xvdyA9IGNvbG9yc1tbImtvX3JldGluYSJdXSwgY29sb3JfaGlnaCA9IGNvbG9yc1tbImtvX3NjbiJdXSwKICBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gaW50ZXJlc3RpbmdfZ2VuZXMsIGFscGhhID0gMS4wLAogIHNpemUgPSA0LCBtaW4uc2VnbWVudC5sZW5ndGggPSAwLCBwb2ludC5wYWRkaW5nID0gMC4yKQpwcChmaWxlID0gIjI0bG9jYXRpb25fbWFfdm9sY2Fub19zdHJpY3Qvc3JfcDE1X2tvX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSAxMiwgaGVpZ2h0ID0gMTIpCnNyX3AxNV9rb192b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpzcl9wMTVfa29fdm9sY2Fub1tbInBsb3QiXV0Kc3JfcDE1X2tvX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gY29sb3JzW1sia29fcmV0aW5hIl1dLCBjb2xvcl9oaWdoID0gY29sb3JzW1sia29fc2NuIl1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gaW50ZXJlc3RpbmdfZ2VuZXMsIG91dGxpbmUgPSBvdXRsaW5lKQpwcChmaWxlID0gIjI0bG9jYXRpb25fbWFfdm9sY2Fub19zdHJpY3Qvc3JfcDE1X2tvX21hLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKc3JfcDE1X2tvX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQpzcl9wMTVfa29fbWFbWyJwbG90Il1dCmBgYAoKIyMjIEFuZCB0aW1lCgpgYGB7cn0KdGltZV90YWJsZXNfZnVsbCA8LSBjb21iaW5lX2RlX3RhYmxlcygKICB0aW1lX2RlLCBrZWVwZXJzID0gdGltZV9rZWVwZXJzLAogIGxhYmVsX2NvbHVtbiA9IGxhYmVsX2NvbHVtbiwKICBleGNlbCA9IGdsdWUoIjI1ZnVsbF9jb250cmFzdHNfdGltZS9mdWxsX3RhYmxlcy12e3Zlcn0ueGxzeCIpKQp0aW1lX3NpZ19mdWxsIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCiAgdGltZV90YWJsZXNfZnVsbCwgYWNjb3JkaW5nX3RvID0gImRlc2VxIiwKICBleGNlbCA9IGdsdWUoIjI1ZnVsbF9jb250cmFzdHNfdGltZS9mdWxsX3NpZy12e3Zlcn0ueGxzeCIpKQpgYGAKCmBgYHtyfQp0aW1lX3RhYmxlcyA8LSBsaXN0KCkKdGltZV9zaWcgPC0gbGlzdCgpCnRpbWVfZ3AgPC0gbGlzdCgpCnRpbWVfY3AgPC0gbGlzdCgpCmZvciAoayBpbiBzZXFfYWxvbmcodGltZV9rZWVwZXJzKSkgewogIG5hbWUgPC0gbmFtZXModGltZV9rZWVwZXJzKVtrXQogIG1lc3NhZ2UoIkV4YW1pbmluZyAiLCBuYW1lKQogIGtlZXBlciA8LSB0aW1lX2tlZXBlcnNbbmFtZV0KICBpbmNsdWRlcyA8LSB0aW1lX2luY2x1c2lvbnNbW25hbWVdXQogIGluY2x1ZGVfbmFtZSA8LSBwYXN0ZTAoImluY18iLCBuYW1lKQogIGluY2x1ZGVfZGZfbmFtZSA8LSBwYXN0ZTAoImRmXyIsIG5hbWUpCiAgaW5jbHVkZV9kZiA8LSB0aW1lX2luY2x1c2lvbnNbW2luY2x1ZGVfZGZfbmFtZV1dCiAgaW5jbHVkZXMgPC0gdGltZV9pbmNsdXNpb25zW1tpbmNsdWRlX25hbWVdXQogIHN1bW1hcnkocm93bmFtZXModGltZV9zaWdfZnVsbFtbImRlc2VxIl1dW1sidXBzIl1dW1tuYW1lXV0pICVpbiUgaW5jbHVkZXMpCiAgaW5jbHVkZV9maWxlbmFtZSA8LSBnbHVlKCIyNnRpbWVfY29udHJhc3RzL3tuYW1lfV9pbmNsdWRpbmdfd3Rfe2xmY19jdXRvZmZ9X2RlY3JlYXNlZF90YWJsZS12e3Zlcn0ueGxzeCIpCiAgaW5jbHVkZV9zaWdfZmlsZW5hbWUgPC0gZ2x1ZSgiMjZ0aW1lX2NvbnRyYXN0cy97bmFtZX1faW5jbHVkaW5nX3d0X3tsZmNfY3V0b2ZmfV9kZWNyZWFzZWRfc2lnLXZ7dmVyfS54bHN4IikKICB0aW1lX3RhYmxlc1tbbmFtZV1dIDwtIGNvbWJpbmVfZGVfdGFibGVzKAogICAgdGltZV9kZSwgZXh0cmFfYW5ub3QgPSBpbmNsdWRlX2RmLAogICAga2VlcGVycyA9IGtlZXBlciwgbGFiZWxfY29sdW1uID0gbGFiZWxfY29sdW1uLAogICAgZXhjZWwgPSBpbmNsdWRlX2ZpbGVuYW1lLCB3YW50ZWRfZ2VuZXMgPSBpbmNsdWRlcykKICBwcmludCh0aW1lX3RhYmxlc1tbbmFtZV1dKQogIHRpbWVfc2lnW1tuYW1lXV0gPC0gZXh0cmFjdF9zaWduaWZpY2FudF9nZW5lcygKICAgIHRpbWVfdGFibGVzW1tuYW1lXV0sIGFjY29yZGluZ190byA9ICJkZXNlcSIsCiAgICBleGNlbCA9IGluY2x1ZGVfc2lnX2ZpbGVuYW1lKQogIHByaW50KHRpbWVfc2lnW1tuYW1lXV0pCiAgbnVtX3Jvd3MgPC0gbnJvdyh0aW1lX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSkgKwogICAgbnJvdyh0aW1lX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJkb3ducyJdXVtbbmFtZV1dKQogIG1lc3NhZ2UoIlRoZXJlIGFyZSAiLCBudW1fcm93cywgIiBzaWduaWZpY2FudCB1cCBhbmQgZG93biBnZW5lcy4iKQogIGlmIChudW1fcm93cyA+IDEwKSB7CiAgICB0aW1lX2dwW1tuYW1lXV0gPC0gYWxsX2dwcm9maWxlcih0aW1lX3NpZ1tbbmFtZV1dLCBzcGVjaWVzID0gIm1tdXNjdWx1cyIpCiAgICBncF93cml0dGVuIDwtIHdyaXRlX2FsbF9ncCh0aW1lX2dwW1tuYW1lXV0pCiAgICB0aW1lX2NwW1tuYW1lXV0gPC0gYWxsX2Nwcm9maWxlcigKICAgICAgdGltZV9zaWdbW25hbWVdXSwgdGltZV90YWJsZXNbW25hbWVdXSwgb3JnZGIgPSAib3JnLk1tLmVnLmRiIiwgb3JnYW5pc20gPSAibW91c2UiLAogICAgICBvcmdkYl9mcm9tID0gb3JnZGJfZnJvbSwgZ29fbGV2ZWwgPSBnb19sZXZlbCwgbWF4X2dyb3Vwc2l6ZSA9IG1heF9ncm91cHNpemUpCiAgICBjcF93cml0dGVuIDwtIHdyaXRlX2FsbF9jcCh0aW1lX2NwW1tuYW1lXV0sIHByZWZpeCA9ICIyNyIpCiAgICBncF93cml0dGVuIDwtIHdyaXRlX2FsbF9ncCh0aW1lX2dwW1tuYW1lXV0sIHByZWZpeCA9ICIyNyIpCiAgfQp9CmBgYAoKU2VuZCB0aGUgcGxvdHMgc2VwYXJhdGVseS4KCmBgYHtyfQpmb3IgKGsgaW4gc2VxX2Fsb25nKHRpbWVfa2VlcGVycykpIHsKICBuYW1lIDwtIG5hbWVzKHRpbWVfa2VlcGVycylba10KICBtZXNzYWdlKCJFeGFtaW5pbmcgIiwgbmFtZSkKICBrZWVwZXIgPC0gdGltZV9rZWVwZXJzW25hbWVdCiAgaW5jbHVkZXMgPC0gdGltZV9pbmNsdXNpb25zW1tuYW1lXV0KICBpbmNsdWRlX25hbWUgPC0gcGFzdGUwKCJpbmNfIiwgbmFtZSkKICBpbmNsdWRlX2RmX25hbWUgPC0gcGFzdGUwKCJkZl8iLCBuYW1lKQogIGluY2x1ZGVfZGYgPC0gdGltZV9pbmNsdXNpb25zW1tpbmNsdWRlX2RmX25hbWVdXQogIGluY2x1ZGVzIDwtIHRpbWVfaW5jbHVzaW9uc1tbaW5jbHVkZV9uYW1lXV0KICBudW1fcm93cyA8LSBucm93KHRpbWVfc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dKSArCiAgICBucm93KHRpbWVfc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbImRvd25zIl1dW1tuYW1lXV0pCiAgbWVzc2FnZSgiVGhlcmUgYXJlICIsIG51bV9yb3dzLCAiIHNpZ25pZmljYW50IHVwIGFuZCBkb3duIGdlbmVzLiIpCiAgbnVtX29iamVjdHMgPC0gbGVuZ3RoKHRpbWVfY3BbW25hbWVdXSkKICBpZiAobnVtX29iamVjdHMgPT0gMCkgewogICAgd2FybmluZygiU29tZXRoaW5nIGZhaWxlZCBpbiBhbGxfY3Byb2ZpbGVyLiIpCiAgfSBlbHNlIHsKICAgIHVwcCA8LSB3aGljaChncmVwbCh4ID0gbmFtZXModGltZV9jcFtbbmFtZV1dKSwgcGF0dGVybiA9ICJfdXAkIikpCiAgICBkb3ducCA8LSB3aGljaChncmVwbCh4ID0gbmFtZXModGltZV9jcFtbbmFtZV1dKSwgcGF0dGVybiA9ICJfZG93biQiKSkKICAgIGlmIChsZW5ndGgodXBwKSA+IDApIHsKICAgICAgbWZfc2lnIDwtIHRpbWVfY3BbW25hbWVdXVtbdXBwXV1bWyJnb19kYXRhIl1dW1siTUZfZW5yaWNoIl1dCiAgICAgIGNjX3NpZyA8LSB0aW1lX2NwW1tuYW1lXV1bW3VwcF1dW1siZ29fZGF0YSJdXVtbIkNDX2VucmljaCJdXQogICAgICBicF9zaWcgPC0gdGltZV9jcFtbbmFtZV1dW1t1cHBdXVtbImdvX2RhdGEiXV1bWyJCUF9lbnJpY2giXV0KICAgICAgbWZfcGxvdHNfdXAgPC0gcGxvdF9lbnJpY2hyZXN1bHQobWZfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBtZl90cmVlX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjI4Y2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9tZl9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gbWZfdHJlZV91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX3VwW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIG1mX2Jhcl91cF9maWxlbmFtZSA8LSBnbHVlKCIyOGNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfbWZfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gbWZfYmFyX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQobWZfcGxvdHNfdXBbWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBjY19wbG90c191cCA8LSBwbG90X2VucmljaHJlc3VsdChjY19zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGNjX3RyZWVfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMjhjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X3VwX2NjX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBjY190cmVlX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoY2NfcGxvdHNfdXBbWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfYmFyX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjI4Y2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV91cF9jY19zaWdfYmFyLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBjY19iYXJfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c191cFtbImJhciJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGJwX3Bsb3RzX3VwIDwtIHBsb3RfZW5yaWNocmVzdWx0KGJwX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgYnBfdHJlZV91cF9maWxlbmFtZSA8LSBnbHVlKCIyOGNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfYnBfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IGJwX3RyZWVfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChicF9wbG90c191cFtbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBicF9wbG90c191cCA8LSBwbG90X2VucmljaHJlc3VsdChicF9zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGJwX2Jhcl91cF9maWxlbmFtZSA8LSBnbHVlKCIyOGNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fdXBfYnBfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gYnBfYmFyX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoYnBfcGxvdHNfdXBbWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgfQogICAgaWYgKGxlbmd0aChkb3ducCkgPiAwKSB7CiAgICAgIG1mX3NpZyA8LSB0aW1lX2NwW1tuYW1lXV1bW2Rvd25wXV1bWyJnb19kYXRhIl1dW1siTUZfZW5yaWNoIl1dCiAgICAgIGNjX3NpZyA8LSB0aW1lX2NwW1tuYW1lXV1bW2Rvd25wXV1bWyJnb19kYXRhIl1dW1siQ0NfZW5yaWNoIl1dCiAgICAgIGJwX3NpZyA8LSB0aW1lX2NwW1tuYW1lXV1bW2Rvd25wXV1bWyJnb19kYXRhIl1dW1siQlBfZW5yaWNoIl1dCiAgICAgIG1mX3Bsb3RzX2Rvd24gPC0gcGxvdF9lbnJpY2hyZXN1bHQobWZfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBtZl90cmVlX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMjhjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fbWZfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IG1mX3RyZWVfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX2Rvd25bWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgbWZfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMjhjbHVzdGVyUHJvZmlsZXJfcGxvdHMve25hbWV9X2Rvd25fbWZfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gbWZfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChtZl9wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfcGxvdHNfZG93biA8LSBwbG90X2VucmljaHJlc3VsdChjY19zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGNjX3RyZWVfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIyOGNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fZG93bl9jY19zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gY2NfdHJlZV9kb3duX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoY2NfcGxvdHNfZG93bltbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBjY19iYXJfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIyOGNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fZG93bl9jY19zaWdfYmFyLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBjY19iYXJfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGNjX3Bsb3RzX2Rvd25bWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBicF9wbG90c19kb3duIDwtIHBsb3RfZW5yaWNocmVzdWx0KGJwX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgYnBfdHJlZV9kb3duX2ZpbGVuYW1lIDwtIGdsdWUoIjI4Y2x1c3RlclByb2ZpbGVyX3Bsb3RzL3tuYW1lfV9kb3duX2JwX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBicF90cmVlX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChicF9wbG90c19kb3duW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGJwX3Bsb3RzX2Rvd24gPC0gcGxvdF9lbnJpY2hyZXN1bHQoYnBfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBicF9iYXJfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIyOGNsdXN0ZXJQcm9maWxlcl9wbG90cy97bmFtZX1fZG93bl9icF9zaWdfYmFyLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBicF9iYXJfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGJwX3Bsb3RzX2Rvd25bWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgfQogIH0KfQpgYGAKCiMjIFZvbGNhbm8gcGxvdHMgYnkgdGltZQoKIyMjIHRfaGV0X2RsZ24KCmBgYHtyfQp0YWJsZV9uYW1lIDwtICJ0X2hldF9kbGduIgp0YWJsZV9pbnB1dCA8LSB0aW1lX3RhYmxlc1tbdGFibGVfbmFtZV1dCnRhYmxlIDwtIHRhYmxlX2lucHV0W1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCnRfaGV0X2RsZ25fdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwKICBjb2xvcl9sb3cgPSB0aW1lX2NvbG9yc1tbInAwOF9oZXRfZGxnbiJdXSwgY29sb3JfaGlnaCA9IHRpbWVfY29sb3JzW1sicDE1X2hldF9kbGduIl1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQsIG1pbi5zZWdtZW50Lmxlbmd0aCA9IDAsIHBvaW50LnBhZGRpbmcgPSAwLjIpCnBwKGZpbGUgPSAiMjl0aW1lX21hX3ZvbGNhbm8vdF9oZXRfZGxnbl92b2xjYW5vLnBkZiIsIHdpZHRoID0gMTIsIGhlaWdodCA9IDEyKQp0X2hldF9kbGduX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnRfaGV0X2RsZ25fdm9sY2Fub1tbInBsb3QiXV0KdF9oZXRfZGxnbl9tYSA8LSBwbG90X21hX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZXhwcl9jb2wgPSAiZGVzZXFfYmFzZW1lYW4iLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLAogIGNvbG9yX2xvdyA9IHRpbWVfY29sb3JzW1sicDA4X2hldF9kbGduIl1dLCBjb2xvcl9oaWdoID0gdGltZV9jb2xvcnNbWyJwMTVfaGV0X2RsZ24iXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgb3V0bGluZSA9IG91dGxpbmUpCnBwKGZpbGUgPSAiMjl0aW1lX21hX3ZvbGNhbm8vdF9oZXRfZGxnbl9tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCnRfaGV0X2RsZ25fbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnRfaGV0X2RsZ25fbWFbWyJwbG90Il1dCmBgYAoKIyMjIHRfa29fZGxnbgoKYGBge3J9CnRhYmxlX25hbWUgPC0gInRfa29fZGxnbiIKdGFibGVfaW5wdXQgPC0gdGltZV90YWJsZXNbW3RhYmxlX25hbWVdXQp0YWJsZSA8LSB0YWJsZV9pbnB1dFtbImRhdGEiXV1bW3RhYmxlX25hbWVdXQp0X2tvX2RsZ25fdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwKICBjb2xvcl9sb3cgPSB0aW1lX2NvbG9yc1tbInAwOF9oZXRfZGxnbiJdXSwgY29sb3JfaGlnaCA9IHRpbWVfY29sb3JzW1sicDE1X2hldF9kbGduIl1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQsIG1pbi5zZWdtZW50Lmxlbmd0aCA9IDAsIHBvaW50LnBhZGRpbmcgPSAwLjIpCnBwKGZpbGUgPSAiMjl0aW1lX21hX3ZvbGNhbm8vdF9rb19kbGduX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSAxMiwgaGVpZ2h0ID0gMTIpCnRfa29fZGxnbl92b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQp0X2tvX2RsZ25fdm9sY2Fub1tbInBsb3QiXV0KdF9rb19kbGduX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gdGltZV9jb2xvcnNbWyJwMDhfaGV0X2RsZ24iXV0sIGNvbG9yX2hpZ2ggPSB0aW1lX2NvbG9yc1tbInAxNV9oZXRfZGxnbiJdXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzLCBvdXRsaW5lID0gb3V0bGluZSkKcHAoZmlsZSA9ICIyOXRpbWVfbWFfdm9sY2Fuby90X2tvX2RsZ25fbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQp0X2tvX2RsZ25fbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnRfa29fZGxnbl9tYVtbInBsb3QiXV0KYGBgCgojIyMgdF9oZXRfcmV0aW5hCgpgYGB7cn0KdGFibGVfbmFtZSA8LSAidF9oZXRfcmV0aW5hIgp0YWJsZV9pbnB1dCA8LSB0aW1lX3RhYmxlc1tbdGFibGVfbmFtZV1dCnRhYmxlIDwtIHRhYmxlX2lucHV0W1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCnRfaGV0X3JldGluYV92b2xjYW5vIDwtIHBsb3Rfdm9sY2Fub19jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsIHBfY29sID0gImRlc2VxX2FkanAiLAogIGNvbG9yX2xvdyA9IHRpbWVfY29sb3JzW1sicDA4X2hldF9yZXRpbmEiXV0sIGNvbG9yX2hpZ2ggPSB0aW1lX2NvbG9yc1tbInAxNV9oZXRfcmV0aW5hIl1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQsIG1pbi5zZWdtZW50Lmxlbmd0aCA9IDAsIHBvaW50LnBhZGRpbmcgPSAwLjIpCnBwKGZpbGUgPSAiMjl0aW1lX21hX3ZvbGNhbm8vdF9oZXRfcmV0aW5hX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSAxMiwgaGVpZ2h0ID0gMTIpCnRfaGV0X3JldGluYV92b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQp0X2hldF9yZXRpbmFfdm9sY2Fub1tbInBsb3QiXV0KdF9oZXRfcmV0aW5hX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gdGltZV9jb2xvcnNbWyJwMDhfaGV0X3JldGluYSJdXSwgY29sb3JfaGlnaCA9IHRpbWVfY29sb3JzW1sicDE1X2hldF9yZXRpbmEiXV0sCiAgcF9jb2wgPSAiZGVzZXFfYWRqcCIsIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgb3V0bGluZSA9IG91dGxpbmUpCnBwKGZpbGUgPSAiMjl0aW1lX21hX3ZvbGNhbm8vdF9oZXRfcmV0aW5hX21hLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKdF9oZXRfcmV0aW5hX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQp0X2hldF9yZXRpbmFfbWFbWyJwbG90Il1dCmBgYAoKIyMjIHRfa29fcmV0aW5hCgpgYGB7cn0KdGFibGVfbmFtZSA8LSAidF9rb19yZXRpbmEiCnRhYmxlX2lucHV0IDwtIHRpbWVfdGFibGVzW1t0YWJsZV9uYW1lXV0KdGFibGUgPC0gdGFibGVfaW5wdXRbWyJkYXRhIl1dW1t0YWJsZV9uYW1lXV0KdF9rb19yZXRpbmFfdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwKICBjb2xvcl9sb3cgPSB0aW1lX2NvbG9yc1tbInAwOF9oZXRfZGxnbiJdXSwgY29sb3JfaGlnaCA9IHRpbWVfY29sb3JzW1sicDE1X2hldF9kbGduIl1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQsIG1pbi5zZWdtZW50Lmxlbmd0aCA9IDAsIHBvaW50LnBhZGRpbmcgPSAwLjIpCnBwKGZpbGUgPSAiMjl0aW1lX21hX3ZvbGNhbm8vdF9rb19yZXRpbmFfdm9sY2Fuby5wZGYiLCB3aWR0aCA9IDEyLCBoZWlnaHQgPSAxMikKdF9rb19yZXRpbmFfdm9sY2Fub1tbInBsb3QiXV0KcGxvdHRlZCA8LSBkZXYub2ZmKCkKdF9rb19yZXRpbmFfdm9sY2Fub1tbInBsb3QiXV0KdF9rb19yZXRpbmFfbWEgPC0gcGxvdF9tYV9jb25kaXRpb25fZGUoCiAgdGFibGUsIHRhYmxlX25hbWUsIGV4cHJfY29sID0gImRlc2VxX2Jhc2VtZWFuIiwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwKICBjb2xvcl9sb3cgPSB0aW1lX2NvbG9yc1tbInAwOF9oZXRfZGxnbiJdXSwgY29sb3JfaGlnaCA9IHRpbWVfY29sb3JzW1sicDE1X2hldF9kbGduIl1dLAogIHBfY29sID0gImRlc2VxX2FkanAiLCBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gaW50ZXJlc3RpbmdfZ2VuZXMsIG91dGxpbmUgPSBvdXRsaW5lKQpwcChmaWxlID0gIjI5dGltZV9tYV92b2xjYW5vL3Rfa29fcmV0aW5hX21hLnBkZiIsIHdpZHRoID0gOSwgaGVpZ2h0ID0gOSkKdF9rb19yZXRpbmFfbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnRfa29fcmV0aW5hX21hW1sicGxvdCJdXQpgYGAKCiMjIHRfaGV0X3NjbgoKYGBge3J9CnRhYmxlX25hbWUgPC0gInRfaGV0X3NjbiIKdGFibGVfaW5wdXQgPC0gdGltZV90YWJsZXNbW3RhYmxlX25hbWVdXQp0YWJsZSA8LSB0YWJsZV9pbnB1dFtbImRhdGEiXV1bW3RhYmxlX25hbWVdXQp0X2hldF9zY25fdm9sY2FubyA8LSBwbG90X3ZvbGNhbm9fY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBmY19jb2wgPSAiZGVzZXFfbG9nZmMiLCBwX2NvbCA9ICJkZXNlcV9hZGpwIiwKICBjb2xvcl9sb3cgPSB0aW1lX2NvbG9yc1tbInAwOF9oZXRfZGxnbiJdXSwgY29sb3JfaGlnaCA9IHRpbWVfY29sb3JzW1sicDE1X2hldF9kbGduIl1dLAogIGxhYmVsX2NvbHVtbiA9ICJtZ2lfc3ltYm9sIiwgbGFiZWwgPSBpbnRlcmVzdGluZ19nZW5lcywgYWxwaGEgPSAxLjAsCiAgc2l6ZSA9IDQsIG1pbi5zZWdtZW50Lmxlbmd0aCA9IDAsIHBvaW50LnBhZGRpbmcgPSAwLjIpCnBwKGZpbGUgPSAiMjl0aW1lX21hX3ZvbGNhbm8vdF9oZXRfc2NuX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSAxMiwgaGVpZ2h0ID0gMTIpCnRfaGV0X3Njbl92b2xjYW5vW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQp0X2hldF9zY25fdm9sY2Fub1tbInBsb3QiXV0KdF9oZXRfc2NuX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gdGltZV9jb2xvcnNbWyJwMDhfaGV0X2RsZ24iXV0sIGNvbG9yX2hpZ2ggPSB0aW1lX2NvbG9yc1tbInAxNV9oZXRfZGxnbiJdXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzLCBvdXRsaW5lID0gb3V0bGluZSkKcHAoZmlsZSA9ICIyOXRpbWVfbWFfdm9sY2Fuby90X2hldF9zY25fbWEucGRmIiwgd2lkdGggPSA5LCBoZWlnaHQgPSA5KQp0X2hldF9zY25fbWFbWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnRfaGV0X3Njbl9tYVtbInBsb3QiXV0KYGBgCgojIyB0X2tvX3NjbgoKYGBge3J9CnRhYmxlX25hbWUgPC0gInRfa29fc2NuIgp0YWJsZV9pbnB1dCA8LSB0aW1lX3RhYmxlc1tbdGFibGVfbmFtZV1dCnRhYmxlIDwtIHRhYmxlX2lucHV0W1siZGF0YSJdXVtbdGFibGVfbmFtZV1dCnRfa29fc2NuX3ZvbGNhbm8gPC0gcGxvdF92b2xjYW5vX2NvbmRpdGlvbl9kZSgKICB0YWJsZSwgdGFibGVfbmFtZSwgZmNfY29sID0gImRlc2VxX2xvZ2ZjIiwgcF9jb2wgPSAiZGVzZXFfYWRqcCIsCiAgY29sb3JfbG93ID0gdGltZV9jb2xvcnNbWyJwMDhfaGV0X2RsZ24iXV0sIGNvbG9yX2hpZ2ggPSB0aW1lX2NvbG9yc1tbInAxNV9oZXRfZGxnbiJdXSwKICBsYWJlbF9jb2x1bW4gPSAibWdpX3N5bWJvbCIsIGxhYmVsID0gaW50ZXJlc3RpbmdfZ2VuZXMsIGFscGhhID0gMS4wLAogIHNpemUgPSA0LCBtaW4uc2VnbWVudC5sZW5ndGggPSAwLCBwb2ludC5wYWRkaW5nID0gMC4yKQpwcChmaWxlID0gIjI5dGltZV9tYV92b2xjYW5vL3Rfa29fc2NuX3ZvbGNhbm8ucGRmIiwgd2lkdGggPSAxMiwgaGVpZ2h0ID0gMTIpCnRfa29fc2NuX3ZvbGNhbm9bWyJwbG90Il1dCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCnRfa29fc2NuX3ZvbGNhbm9bWyJwbG90Il1dCnRfa29fc2NuX21hIDwtIHBsb3RfbWFfY29uZGl0aW9uX2RlKAogIHRhYmxlLCB0YWJsZV9uYW1lLCBleHByX2NvbCA9ICJkZXNlcV9iYXNlbWVhbiIsIGZjX2NvbCA9ICJkZXNlcV9sb2dmYyIsCiAgY29sb3JfbG93ID0gdGltZV9jb2xvcnNbWyJwMDhfaGV0X2RsZ24iXV0sIGNvbG9yX2hpZ2ggPSB0aW1lX2NvbG9yc1tbInAxNV9oZXRfZGxnbiJdXSwKICBwX2NvbCA9ICJkZXNlcV9hZGpwIiwgbGFiZWxfY29sdW1uID0gIm1naV9zeW1ib2wiLCBsYWJlbCA9IGludGVyZXN0aW5nX2dlbmVzLCBvdXRsaW5lID0gb3V0bGluZSkKcHAoZmlsZSA9ICIyOXRpbWVfbWFfdm9sY2Fuby90X2tvX3Njbl9tYS5wZGYiLCB3aWR0aCA9IDksIGhlaWdodCA9IDkpCnRfa29fc2NuX21hW1sicGxvdCJdXQpwbG90dGVkIDwtIGRldi5vZmYoKQp0X2tvX3Njbl9tYVtbInBsb3QiXV0KYGBgCgojIyMjIFJlcGVhdCB3aXRoIHRoZSBzdHJpY3QgZmlsdGVyCgpgYGB7cn0KdGltZV9zdHJpY3RfdGFibGVzIDwtIGxpc3QoKQp0aW1lX3N0cmljdF9zaWcgPC0gbGlzdCgpCnRpbWVfc3RyaWN0X2dwIDwtIGxpc3QoKQp0aW1lX3N0cmljdF9jcCA8LSBsaXN0KCkKdGltZV9zdHJpY3RfZW4gPC0gbGlzdCgpCmZvciAoayBpbiBzZXFfYWxvbmcodGltZV9rZWVwZXJzKSkgewogIG5hbWUgPC0gbmFtZXModGltZV9rZWVwZXJzKVtrXQogIG1lc3NhZ2UoIkV4YW1pbmluZyAiLCBuYW1lKQogIGtlZXBlciA8LSB0aW1lX2tlZXBlcnNbbmFtZV0KICBpbmNsdWRlX25hbWUgPC0gcGFzdGUwKCJpbmNfIiwgbmFtZSkKICBpbmNsdWRlX2RmX25hbWUgPC0gcGFzdGUwKCJkZl8iLCBuYW1lKQogIGluY2x1ZGVfZGYgPC0gdGltZV9pbmNsdXNpb25zX3N0cmljdFtbaW5jbHVkZV9kZl9uYW1lXV0KICBpbmNsdWRlcyA8LSB0aW1lX2luY2x1c2lvbnNfc3RyaWN0W1tpbmNsdWRlX25hbWVdXQogIGZvdW5kX2luY2x1ZGVzIDwtIHJvd25hbWVzKHRpbWVfc2lnX2Z1bGxbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dKSAlaW4lIGluY2x1ZGVzCiAgc3VtbWFyeShmb3VuZF9pbmNsdWRlcykKICBpZiAoc3VtKGZvdW5kX2luY2x1ZGVzKSA9PSAwKSB7CiAgICBuZXh0CiAgfQogIGluY2x1ZGVfZmlsZW5hbWUgPC0gZ2x1ZSgiMzB0aW1lX3N0cmljdF9jb250cmFzdHNfZXhjZWwve25hbWV9X2luY2x1ZGluZ193dF97bGZjX2N1dG9mZn1fZGVjcmVhc2VkX3RhYmxlLXZ7dmVyfS54bHN4IikKICBpbmNsdWRlX3NpZ19maWxlbmFtZSA8LSBnbHVlKCIzMHRpbWVfc3RyaWN0X2NvbnRyYXN0c19leGNlbC97bmFtZX1faW5jbHVkaW5nX3d0X3tsZmNfY3V0b2ZmfV9kZWNyZWFzZWRfc2lnLXZ7dmVyfS54bHN4IikKICB0aW1lX3N0cmljdF90YWJsZXNbW25hbWVdXSA8LSBjb21iaW5lX2RlX3RhYmxlcygKICAgIHRpbWVfZGUsIGV4dHJhX2Fubm90ID0gaW5jbHVkZV9kZiwKICAgIGtlZXBlcnMgPSBrZWVwZXIsIGxhYmVsX2NvbHVtbiA9IGxhYmVsX2NvbHVtbiwKICAgIGV4Y2VsID0gaW5jbHVkZV9maWxlbmFtZSwgd2FudGVkX2dlbmVzID0gaW5jbHVkZXMpCiAgcHJpbnQodGltZV9zdHJpY3RfdGFibGVzW1tuYW1lXV0pCiAgdGltZV9zdHJpY3Rfc2lnW1tuYW1lXV0gPC0gZXh0cmFjdF9zaWduaWZpY2FudF9nZW5lcygKICAgIHRpbWVfc3RyaWN0X3RhYmxlc1tbbmFtZV1dLCBhY2NvcmRpbmdfdG8gPSAiZGVzZXEiLAogICAgZXhjZWwgPSBpbmNsdWRlX3NpZ19maWxlbmFtZSkKICBwcmludCh0aW1lX3N0cmljdF9zaWdbW25hbWVdXSkKICBudW1fcm93cyA8LSBucm93KHRpbWVfc3RyaWN0X3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSkgKwogICAgbnJvdyh0aW1lX3N0cmljdF9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSkKICBtZXNzYWdlKCJUaGVyZSBhcmUgIiwgbnVtX3Jvd3MsICIgc2lnbmlmaWNhbnQgdXAgYW5kIGRvd24gZ2VuZXMuIikKICBpZiAobnVtX3Jvd3MgPj0gMTApIHsKICAgIG1lc3NhZ2UoIlBlcmZvcm1pbmcgZ3Byb2ZpbGVyL2NsdXN0ZXJQcm9maWxlci4iKQogICAgdGltZV9zdHJpY3RfZ3BbW25hbWVdXSA8LSBhbGxfZ3Byb2ZpbGVyKHRpbWVfc3RyaWN0X3NpZ1tbbmFtZV1dLCBzcGVjaWVzID0gIm1tdXNjdWx1cyIpCiAgICB0aW1lX3N0cmljdF9jcFtbbmFtZV1dIDwtIGFsbF9jcHJvZmlsZXIoCiAgICAgIHRpbWVfc3RyaWN0X3NpZ1tbbmFtZV1dLCB0aW1lX3N0cmljdF90YWJsZXNbW25hbWVdXSwKICAgICAgb3JnZGIgPSAib3JnLk1tLmVnLmRiIiwgZ29fbGV2ZWwgPSBnb19sZXZlbCwgb3JnZGJfZnJvbSA9IG9yZ2RiX2Zyb20sCiAgICAgIG1heF9ncm91cHNpemUgPSBtYXhfZ3JvdXBzaXplLCBvcmdhbmlzbSA9ICJtb3VzZSIpCiAgICAjaWYgKCFpcy5udWxsKGdldDAoIm0yX2dzYyIpKSkgewogICAgIyAgdGltZV9zdHJpY3RfZW5bW25hbWVdXSA8LSBhbGxfZW5yaWNoZXIodGltZV9zdHJpY3Rfc2lnW1tuYW1lXV0sIGdzYyA9IG0yX2dzYywKICAgICMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9yZ2RiID0gIm9yZy5NbS5lZy5kYiIsIGZyb20gPSAiRU5TRU1CTCIsIHRvID0gIlNZTUJPTCIpCiAgICAjfQogICAgZ3Bfd3JpdHRlbiA8LSB3cml0ZV9hbGxfZ3AodGltZV9zdHJpY3RfZ3BbW25hbWVdXSwgcHJlZml4ID0gIjMxIiwgc3VmZml4ID0gInN0cmljdCIpCiAgICBjcF93cml0dGVuIDwtIHdyaXRlX2FsbF9jcCh0aW1lX3N0cmljdF9jcFtbbmFtZV1dLCBwcmVmaXggPSAiMzEiLCBzdWZmaXggPSAic3RyaWN0IikKICAgICNlbl93cml0dGVuIDwtIHdyaXRlX2FsbF9lbih0aW1lX3N0cmljdF9lbltbbmFtZV1dKQogIH0gZWxzZSB7CiAgICB3YXJuaW5nKCJUaGVyZSBhcmUgbGVzcyB0aGFuIDEwIGdlbmVzIHVwIGFuZCBkb3duIGluIHRoZSAiLCBuYW1lLCAiIGNvbXBhcmlzb24uIikKICAgIG1lc3NhZ2UoIlRoZXJlIGFyZSBsZXNzIHRoYW4gMTAgZ2VuZXMgdXAgYW5kIGRvd24gaW4gdGhlICIsIG5hbWUsICIgY29tcGFyaXNvbi4iKQogIH0KfQpgYGAKClNlbmQgdGhlIHBsb3RzIHNlcGFyYXRlbHkuCgpgYGB7cn0KZm9yIChrIGluIHNlcV9hbG9uZyh0aW1lX2tlZXBlcnMpKSB7CiAgbmFtZSA8LSBuYW1lcyh0aW1lX2tlZXBlcnMpW2tdCiAgbWVzc2FnZSgiRXhhbWluaW5nICIsIG5hbWUpCiAga2VlcGVyIDwtIHRpbWVfa2VlcGVyc1tuYW1lXQogIGluY2x1ZGVzIDwtIHRpbWVfaW5jbHVzaW9uc1tbbmFtZV1dCiAgaW5jbHVkZV9uYW1lIDwtIHBhc3RlMCgiaW5jXyIsIG5hbWUpCiAgaW5jbHVkZV9kZl9uYW1lIDwtIHBhc3RlMCgiZGZfIiwgbmFtZSkKICBpbmNsdWRlX2RmIDwtIHRpbWVfaW5jbHVzaW9uc1tbaW5jbHVkZV9kZl9uYW1lXV0KICBpbmNsdWRlcyA8LSB0aW1lX2luY2x1c2lvbnNbW2luY2x1ZGVfbmFtZV1dCiAgbnVtX3Jvd3MgPC0gbnJvdyh0aW1lX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSkgKwogICAgbnJvdyh0aW1lX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJkb3ducyJdXVtbbmFtZV1dKQogIG1lc3NhZ2UoIlRoZXJlIGFyZSAiLCBudW1fcm93cywgIiBzaWduaWZpY2FudCB1cCBhbmQgZG93biBnZW5lcy4iKQogIG51bV9vYmplY3RzIDwtIGxlbmd0aCh0aW1lX2NwW1tuYW1lXV0pCiAgaWYgKG51bV9vYmplY3RzID09IDApIHsKICAgIHdhcm5pbmcoIlNvbWV0aGluZyBmYWlsZWQgaW4gYWxsX2Nwcm9maWxlci4iKQogIH0gZWxzZSB7CiAgICB1cHAgPC0gd2hpY2goZ3JlcGwoeCA9IG5hbWVzKHRpbWVfY3BbW25hbWVdXSksIHBhdHRlcm4gPSAiX3VwJCIpKQogICAgZG93bnAgPC0gd2hpY2goZ3JlcGwoeCA9IG5hbWVzKHRpbWVfY3BbW25hbWVdXSksIHBhdHRlcm4gPSAiX2Rvd24kIikpCiAgICBpZiAobGVuZ3RoKHVwcCkgPiAwKSB7CiAgICAgIG1mX3NpZyA8LSB0aW1lX2NwW1tuYW1lXV1bW3VwcF1dW1siZ29fZGF0YSJdXVtbIk1GX2VucmljaCJdXQogICAgICBjY19zaWcgPC0gdGltZV9jcFtbbmFtZV1dW1t1cHBdXVtbImdvX2RhdGEiXV1bWyJDQ19lbnJpY2giXV0KICAgICAgYnBfc2lnIDwtIHRpbWVfY3BbW25hbWVdXVtbdXBwXV1bWyJnb19kYXRhIl1dW1siQlBfZW5yaWNoIl1dCiAgICAgIG1mX3Bsb3RzX3VwIDwtIHBsb3RfZW5yaWNocmVzdWx0KG1mX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgbWZfdHJlZV91cF9maWxlbmFtZSA8LSBnbHVlKCIzMmNwX3RyZWVzL3tuYW1lfV91cF9tZl9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gbWZfdHJlZV91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX3VwW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIG1mX2Jhcl91cF9maWxlbmFtZSA8LSBnbHVlKCIzMmNwX2Jhci97bmFtZX1fdXBfbWZfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gbWZfYmFyX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQobWZfcGxvdHNfdXBbWyJiYXIiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBjY19wbG90c191cCA8LSBwbG90X2VucmljaHJlc3VsdChjY19zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGNjX3RyZWVfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMzJjcF90cmVlcy97bmFtZX1fdXBfY2Nfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IGNjX3RyZWVfdXBfZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c191cFtbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBjY19iYXJfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMzJjcF9iYXIve25hbWV9X3VwX2NjX3NpZ19iYXIucGRmIikKICAgICAgcHAoZmlsZSA9IGNjX2Jhcl91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGNjX3Bsb3RzX3VwW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgYnBfcGxvdHNfdXAgPC0gcGxvdF9lbnJpY2hyZXN1bHQoYnBfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBicF90cmVlX3VwX2ZpbGVuYW1lIDwtIGdsdWUoIjMyY3BfdHJlZXMve25hbWV9X3VwX2JwX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBicF90cmVlX3VwX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoYnBfcGxvdHNfdXBbWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgYnBfcGxvdHNfdXAgPC0gcGxvdF9lbnJpY2hyZXN1bHQoYnBfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBicF9iYXJfdXBfZmlsZW5hbWUgPC0gZ2x1ZSgiMzJjcF9iYXIve25hbWV9X3VwX2JwX3NpZ19iYXIucGRmIikKICAgICAgcHAoZmlsZSA9IGJwX2Jhcl91cF9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KGJwX3Bsb3RzX3VwW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgIH0KICAgIGlmIChsZW5ndGgoZG93bnApID4gMCkgewogICAgICBtZl9zaWcgPC0gdGltZV9jcFtbbmFtZV1dW1tkb3ducF1dW1siZ29fZGF0YSJdXVtbIk1GX2VucmljaCJdXQogICAgICBjY19zaWcgPC0gdGltZV9jcFtbbmFtZV1dW1tkb3ducF1dW1siZ29fZGF0YSJdXVtbIkNDX2VucmljaCJdXQogICAgICBicF9zaWcgPC0gdGltZV9jcFtbbmFtZV1dW1tkb3ducF1dW1siZ29fZGF0YSJdXVtbIkJQX2VucmljaCJdXQogICAgICBtZl9wbG90c19kb3duIDwtIHBsb3RfZW5yaWNocmVzdWx0KG1mX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgbWZfdHJlZV9kb3duX2ZpbGVuYW1lIDwtIGdsdWUoIjMyY3BfdHJlZXMve25hbWV9X2Rvd25fbWZfc2lnX3RyZWUucGRmIikKICAgICAgcHAoZmlsZSA9IG1mX3RyZWVfZG93bl9maWxlbmFtZSkKICAgICAgdHJ5KHByaW50KG1mX3Bsb3RzX2Rvd25bWyJ0cmVlIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgbWZfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMzJjcF9iYXIve25hbWV9X2Rvd25fbWZfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gbWZfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChtZl9wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgICAgY2NfcGxvdHNfZG93biA8LSBwbG90X2VucmljaHJlc3VsdChjY19zaWcsIHNob3dDYXRlZ29yeSA9IGdvX2NhdGVnb3JpZXMpCiAgICAgIGNjX3RyZWVfZG93bl9maWxlbmFtZSA8LSBnbHVlKCIzMmNwX3RyZWVzL3tuYW1lfV9kb3duX2NjX3NpZ190cmVlLnBkZiIpCiAgICAgIHBwKGZpbGUgPSBjY190cmVlX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChjY19wbG90c19kb3duW1sidHJlZSJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGNjX2Jhcl9kb3duX2ZpbGVuYW1lIDwtIGdsdWUoIjMyY3BfYmFyL3tuYW1lfV9kb3duX2NjX3NpZ19iYXIucGRmIikKICAgICAgcHAoZmlsZSA9IGNjX2Jhcl9kb3duX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoY2NfcGxvdHNfZG93bltbImJhciJdXSksIHNpbGVudCA9IFRSVUUpCiAgICAgIHBsb3R0ZWQgPC0gZGV2Lm9mZigpCiAgICAgIGJwX3Bsb3RzX2Rvd24gPC0gcGxvdF9lbnJpY2hyZXN1bHQoYnBfc2lnLCBzaG93Q2F0ZWdvcnkgPSBnb19jYXRlZ29yaWVzKQogICAgICBicF90cmVlX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMzJjcF90cmVlcy97bmFtZX1fZG93bl9icF9zaWdfdHJlZS5wZGYiKQogICAgICBwcChmaWxlID0gYnBfdHJlZV9kb3duX2ZpbGVuYW1lKQogICAgICB0cnkocHJpbnQoYnBfcGxvdHNfZG93bltbInRyZWUiXV0pLCBzaWxlbnQgPSBUUlVFKQogICAgICBwbG90dGVkIDwtIGRldi5vZmYoKQogICAgICBicF9wbG90c19kb3duIDwtIHBsb3RfZW5yaWNocmVzdWx0KGJwX3NpZywgc2hvd0NhdGVnb3J5ID0gZ29fY2F0ZWdvcmllcykKICAgICAgYnBfYmFyX2Rvd25fZmlsZW5hbWUgPC0gZ2x1ZSgiMzJjcF9iYXIve25hbWV9X2Rvd25fYnBfc2lnX2Jhci5wZGYiKQogICAgICBwcChmaWxlID0gYnBfYmFyX2Rvd25fZmlsZW5hbWUpCiAgICAgIHRyeShwcmludChicF9wbG90c19kb3duW1siYmFyIl1dKSwgc2lsZW50ID0gVFJVRSkKICAgICAgcGxvdHRlZCA8LSBkZXYub2ZmKCkKICAgIH0KICB9Cn0KYGBgCgojIFRyYW5zbGF0b21lIHF1ZXJpZXMKCkluIGNvbnZlcnNhdGlvbiB3aXRoIENvbGVuc28sIGhlIHNwb2tlIGFib3V0IGEgc2VyaWVzIG9mIGNvbnRyYXN0cwp3aGljaCB3b3VsZCBiZSBpbnRlcmVzdGluZyB0byBhdHRlbXB0IGluIG9yZGVyIHRvIHF1ZXJ5IHRoZSBjaGFuZ2VzCmFjcm9zcyBib3RoIGxvY2F0aW9ucyBhbmQgZ2Vub3R5cGVzIGFuZC9vciBib3RoIGxvY2F0aW9ucyBhbmQgdGltZSwKdGh1czoKCihwMDhfaGV0X3NjbiAvIHAwOF9oZXRfcmV0aW5hKSAvIChwMDhfa29fc2NuIC8gcDA4X2tvX3JldGluYSkKCmFzIGFuIGV4YW1wbGUuICBXZSBjYW4gZGVmaW5pdGVseSBkbyB0aGVzZSwgYnV0IHRoZXkgZG8gbm90IHdvcmsgZm9yCmFsbCBtZXRob2RzIGVtcGxveWVkIChJIHRoaW5rIHRoZXkgd29yayBiZXN0IHdpdGggbGltbWEgYW5kIGVkZ2VSKS4KCkxldHMgZmluZCBvdXQhCgojIyBUd28gc2NuL3JldGluYSBjb21wYXJpc29ucwoKKiAocDA4X2hldF9zY24gLyBwMDhfaGV0X3JldGluYSkgLyAocDA4X2tvX3NjbiAvIHAwOF9rb19yZXRpbmEpCiogKHAxNV9oZXRfc2NuIC8gcDE1X2hldF9yZXRpbmEpIC8gKHAxNV9rb19zY24gLyBwMTVfa29fcmV0aW5hKQoKYGBge3J9CnNjbl9leHRyYSA8LSBnbHVlKCJcXAogIHAwOGhldCA9IChjb25kaXRpb25wMDhfaGV0X3NjbiAtIGNvbmRpdGlvbnAwOF9oZXRfcmV0aW5hKSwgXFwKICBwMDhrbyA9IChjb25kaXRpb25wMDhfa29fc2NuIC0gY29uZGl0aW9ucDA4X2tvX3JldGluYSksIFxcCiAgcDA4aGV0X3ZzX3AwOGtvID0gKGNvbmRpdGlvbnAwOF9oZXRfc2NuIC0gY29uZGl0aW9ucDA4X2hldF9yZXRpbmEpIC0gKGNvbmRpdGlvbnAwOF9rb19zY24gLSBjb25kaXRpb25wMDhfa29fcmV0aW5hKSwgXFwKICBwMTVoZXQgPSAoY29uZGl0aW9ucDE1X2hldF9zY24gLSBjb25kaXRpb25wMTVfaGV0X3JldGluYSksIFxcCiAgcDE1a28gPSAoY29uZGl0aW9ucDE1X2tvX3NjbiAtIGNvbmRpdGlvbnAxNV9rb19yZXRpbmEpLCBcXAogIHAxNWhldF92c19wMTVrbyA9IChjb25kaXRpb25wMTVfaGV0X3NjbiAtIGNvbmRpdGlvbnAxNV9oZXRfcmV0aW5hKSAtIChjb25kaXRpb25wMTVfa29fc2NuIC0gY29uZGl0aW9ucDE1X2tvX3JldGluYSkiKQpzY25fdHJhbnNsYXRvbWVfZGVfa2VlcGVycyA8LSBsaXN0KAogICJwMDhoZXQiID0gYygicDA4X2hldF9zY24iLCAicDA4X2hldF9yZXRpbmEiKSwKICAicDA4a28iID0gYygicDA4X2tvX3NjbiIsICJwMDhfa29fcmV0aW5hIiksCiAgInAxNWhldCIgPSBjKCJwMTVfaGV0X3NjbiIsICJwMTVfaGV0X3JldGluYSIpLAogICJwMTVrbyIgPSBjKCJwMTVfa29fc2NuIiwgInAxNV9rb19yZXRpbmEiKSkKc2NuX3RyYW5zbGF0b21lX2tlZXBlcnMgPC0gbGlzdCgKICAicDA4aGV0IiA9IGMoInAwOF9oZXRfc2NuIiwgInAwOF9oZXRfcmV0aW5hIiksCiAgInAwOGtvIiA9IGMoInAwOF9rb19zY24iLCAicDA4X2tvX3JldGluYSIpLAogICJwMDhfc2NuX3RyYW5zbGF0b21lIiA9IGMoInAwOGhldCIsICJwMDhrbyIpLAogICJwMTVoZXQiID0gYygicDE1X2hldF9zY24iLCAicDE1X2hldF9yZXRpbmEiKSwKICAicDE1a28iID0gYygicDE1X2tvX3NjbiIsICJwMTVfa29fcmV0aW5hIiksCiAgInAxNV9zY25fdHJhbnNsYXRvbWUiID0gYygicDE1aGV0IiwgInAxNWtvIikpCmZpbHQgPC0gbm9ybWFsaXplKHYzX3BhaXJ3aXNlX2lucHV0LCBmaWx0ZXIgPSBUUlVFKQpsaW1tYV90ZXN0IDwtIGxpbW1hX3BhaXJ3aXNlKGZpbHQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAga2VlcGVycyA9IHNjbl90cmFuc2xhdG9tZV9kZV9rZWVwZXJzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1vZGVsX2ZzdHJpbmcgPSAifiAwICsgY29uZGl0aW9uIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtb2RlbF9zdnMgPSBGQUxTRSwgZXh0cmFfY29udHJhc3RycyA9IHNjbl9leHRyYSkKZWRnZXJfdGVzdCA8LSBlZGdlcl9wYWlyd2lzZShmaWx0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIGtlZXBlcnMgPSBzY25fdHJhbnNsYXRvbWVfZGVfa2VlcGVycywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtb2RlbF9mc3RyaW5nID0gIn4gMCArIGNvbmRpdGlvbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbW9kZWxfc3ZzID0gRkFMU0UsIGV4dHJhX2NvbnRyYXN0cyA9IHNjbl9leHRyYSkKc2NuX3RyYW5zbGF0b21lX2RlIDwtIGFsbF9wYWlyd2lzZSh2M19wYWlyd2lzZV9pbnB1dCwgZmlsdGVyID0gVFJVRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBrZWVwZXJzID0gc2NuX3RyYW5zbGF0b21lX2RlX2tlZXBlcnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbW9kZWxfc3ZzID0gRkFMU0UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbW9kZWxfZnN0cmluZyA9ICJ+IDAgKyBjb25kaXRpb24iLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvX2Jhc2ljID0gRkFMU0UsIGRvX2RyZWFtID0gRkFMU0UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG9fbm9pc2VxID0gRkFMU0UsIGRvX2Vic2VxID0gRkFMU0UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZXh0cmFfY29udHJhc3RzID0gc2NuX2V4dHJhKQpzY25fY29tYmluZWRfdGVzdCA8LSBjb21iaW5lX2RlX3RhYmxlcygKICBzY25fdHJhbnNsYXRvbWVfZGUsIGtlZXBlcnMgPSBzY25fdHJhbnNsYXRvbWVfa2VlcGVycywKICBleGNlbCA9IGdsdWUoIjMzdHJhbnNsYXRvbWVfeGxzeC90ZXN0X3Njbl90cmFuc2xhdG9tZV91bmZpbHRlcmVkX25vc3ZhLXZ7dmVyfS54bHN4IikpCnNjbl90cmFuc2xhdG9tZV9kZV9zdmEgPC0gYWxsX3BhaXJ3aXNlKHYzX3BhaXJ3aXNlX2lucHV0LCBmaWx0ZXIgPSBUUlVFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBrZWVwZXJzID0gc2NuX3RyYW5zbGF0b21lX2RlX2tlZXBlcnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1vZGVsX3N2cyA9ICJzdmFzZXEiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtb2RlbF9mc3RyaW5nID0gIn4gMCArIGNvbmRpdGlvbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvX2Jhc2ljID0gRkFMU0UsIGRvX2RyZWFtID0gRkFMU0UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvX25vaXNlcSA9IEZBTFNFLCBkb19lYnNlcSA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBleHRyYV9jb250cmFzdHMgPSBzY25fZXh0cmEpCnNjbl9jb21iaW5lZF90ZXN0X3N2YSA8LSBjb21iaW5lX2RlX3RhYmxlcygKICBzY25fdHJhbnNsYXRvbWVfZGVfc3ZhLCBrZWVwZXJzID0gc2NuX3RyYW5zbGF0b21lX2tlZXBlcnMsCiAgZXhjZWwgPSBnbHVlKCIzM3RyYW5zbGF0b21lX3hsc3gvdGVzdF9zY25fdHJhbnNsYXRvbWVfdW5maWx0ZXJlZF9zdmEtdnt2ZXJ9Lnhsc3giKSkKYGBgCgojIyMgU3VidHJhY3RpbmcgREVTZXEyIHJlc3VsdHM6IHAwOCBzY24gaGV0IHZzIGtvCgpgYGB7cn0KcDA4X3Njbl9jb21iaW5lZF9kZXNlcSA8LSBzdWJ0cmFjdF9kZXNlcV9yZXN1bHRzKAogIGZpcnN0X3RhYmxlID0gc2NuX2NvbWJpbmVkX3Rlc3RbWyJkYXRhIl1dW1sicDA4aGV0Il1dLAogIHNlY29uZF90YWJsZSA9IHNjbl9jb21iaW5lZF90ZXN0W1siZGF0YSJdXVtbInAwOGtvIl1dLAogIGZpcnN0X2xmYyA9ICJkZXNlcV9sb2dmYyIsIHNlY29uZF9sZmMgPSAiZGVzZXFfbG9nZmMiLAogIGZpcnN0X3AgPSAiZGVzZXFfYWRqcCIsIHNlY29uZF9wID0gImRlc2VxX2FkanAiLAogIGZpcnN0X25hbWUgPSAiaGV0Iiwgc2Vjb25kX25hbWUgPSAia28iLAogIGV4Y2VsID0gZ2x1ZSgiMzN0cmFuc2xhdG9tZV94bHN4L3RyYW5zbGF0b21lX3AwOF9zY25fY29tYmluZWRfZGVzZXEtdnt2ZXJ9Lnhsc3giKSkKYGBgCgojIyMgU3VidHJhY3RpbmcgREVTZXEyIHJlc3VsdHM6IHAxNSBzY24gaGV0IHZzIGtvCgpgYGB7cn0KcDE1X3Njbl9jb21iaW5lZF9kZXNlcSA8LSBzdWJ0cmFjdF9kZXNlcV9yZXN1bHRzKAogIGZpcnN0X3RhYmxlID0gc2NuX2NvbWJpbmVkX3Rlc3RbWyJkYXRhIl1dW1sicDE1aGV0Il1dLAogIHNlY29uZF90YWJsZSA9IHNjbl9jb21iaW5lZF90ZXN0W1siZGF0YSJdXVtbInAxNWtvIl1dLAogIGZpcnN0X2xmYyA9ICJkZXNlcV9sb2dmYyIsIHNlY29uZF9sZmMgPSAiZGVzZXFfbG9nZmMiLAogIGZpcnN0X3AgPSAiZGVzZXFfYWRqcCIsIHNlY29uZF9wID0gImRlc2VxX2FkanAiLAogIGZpcnN0X25hbWUgPSAiaGV0Iiwgc2Vjb25kX25hbWUgPSAia28iLAogIGV4Y2VsID0gZ2x1ZSgiMzR0cmFuc2xhdG9tZV9kZXNlcXN1Yl94bHN4L3RyYW5zbGF0b21lX3AxNV9zY25fY29tYmluZWRfZGVzZXEtdnt2ZXJ9Lnhsc3giKSkKYGBgCgojIyBPbmUgZGxnbi9yZXRpbmEgY29tcGFyaXNvbgoKKiAocDA4X2hldF9kbGduIC8gcDA4X2hldF9yZXRpbmEpIC8gKHAwOF9rb19kbGduIC8gcDA4X2tvX3JldGluYSkKCmBgYHtyfQpwMDhfZGxnbl9leHRyYSA8LSAicDA4aGV0X3ZzX3AwOGtvID0gKGNvbmRpdGlvbnAwOF9oZXRfZGxnbiAtIGNvbmRpdGlvbnAwOF9oZXRfcmV0aW5hKSAtIChjb25kaXRpb25wMDhfa29fZGxnbiAtIGNvbmRpdGlvbnAwOF9rb19yZXRpbmEpIgpwMDhfZGxnbl90cmFuc2xhdG9tZV9kZV9rZWVwZXJzIDwtIGxpc3QoCiAgInAwOGhldCIgPSBjKCJwMDhfaGV0X2RsZ24iLCAicDA4X2hldF9yZXRpbmEiKSwKICAicDA4a28iID0gYygicDA4X2tvX2RsZ24iLCAicDA4X2tvX3JldGluYSIpKQpwMDhfZGxnbl90cmFuc2xhdG9tZV9rZWVwZXJzIDwtIGxpc3QoCiAgInAwOF9oZXRfZGxnbl92c19yZXRpbmEiID0gYygicDA4X2hldF9kbGduIiwgInAwOF9oZXRfcmV0aW5hIiksCiAgInAwOF9rb19kbGduX3ZzX3JldGluYSIgPSBjKCJwMDhfa29fZGxnbiIsICJwMDhfa29fcmV0aW5hIiksCiAgInAwOF9kbGduX3RyYW5zbGF0b21lIiA9IGMoInAwOGhldCIsICJwMDhrbyIpKQpwMDhfZGxnbl90cmFuc2xhdG9tZV9kZSA8LSBhbGxfcGFpcndpc2UodjNfcGFpcndpc2VfaW5wdXQsIGZpbHRlciA9IFRSVUUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBrZWVwZXJzID0gcDA4X2RsZ25fdHJhbnNsYXRvbWVfZGVfa2VlcGVycywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1vZGVsX3N2cyA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbW9kZWxfZnN0cmluZyA9ICJ+IDAgKyBjb25kaXRpb24iLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG9fYmFzaWMgPSBGQUxTRSwgZG9fZHJlYW0gPSBGQUxTRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvX25vaXNlcSA9IEZBTFNFLCBkb19lYnNlcSA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZXh0cmFfY29udHJhc3RzID0gcDA4X2RsZ25fZXh0cmEpCnAwOF9kbGduX2NvbWJpbmVkX3Rlc3QgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgcDA4X2RsZ25fdHJhbnNsYXRvbWVfZGUsIGtlZXBlcnMgPSBwMDhfZGxnbl90cmFuc2xhdG9tZV9rZWVwZXJzLAogIGxhYmVsX2NvbHVtbiA9IGxhYmVsX2NvbHVtbiwKICBleGNlbCA9IGdsdWUoIjMzdHJhbnNsYXRvbWVfeGxzeC90ZXN0X3AwOF9kbGduX3RyYW5zbGF0b21lX3VuZmlsdGVyZWRfbm9zdmEtdnt2ZXJ9Lnhsc3giKSkKcDA4X2RsZ25fdHJhbnNsYXRvbWVfZGVfc3ZhIDwtIGFsbF9wYWlyd2lzZSh2M19wYWlyd2lzZV9pbnB1dCwgZmlsdGVyID0gVFJVRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBrZWVwZXJzID0gcDA4X2RsZ25fdHJhbnNsYXRvbWVfZGVfa2VlcGVycywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtb2RlbF9zdnMgPSAic3Zhc2VxIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtb2RlbF9mc3RyaW5nID0gIn4gMCArIGNvbmRpdGlvbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG9fYmFzaWMgPSBGQUxTRSwgZG9fZHJlYW0gPSBGQUxTRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb19ub2lzZXEgPSBGQUxTRSwgZG9fZWJzZXEgPSBGQUxTRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBleHRyYV9jb250cmFzdHMgPSBwMDhfZGxnbl9leHRyYSkKcDA4X2RsZ25fY29tYmluZWRfdGVzdF9zdmEgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgcDA4X2RsZ25fdHJhbnNsYXRvbWVfZGVfc3ZhLCBrZWVwZXJzID0gcDA4X2RsZ25fdHJhbnNsYXRvbWVfa2VlcGVycywKICBsYWJlbF9jb2x1bW4gPSBsYWJlbF9jb2x1bW4sCiAgZXhjZWwgPSBnbHVlKCIzM3RyYW5zbGF0b21lX3hsc3gvdGVzdF9wMDhfZGxnbl90cmFuc2xhdG9tZV91bmZpbHRlcmVkX3N2YS12e3Zlcn0ueGxzeCIpKQpgYGAKCiMjIyBTdWJ0cmFjdGluZyB0aGUgREVTZXEyIHJlc3VsdHMKCiMjIFR3byBzY24vcmV0aW5hIGNvbXBhcmlzb25zIChwMTUvcDA4IGFjcm9zcyBoZXQva28pCgoqIChwMTVfaGV0X3NjbiAvIHAxNV9oZXRfcmV0aW5hKSAvIChwMDhfaGV0X3NjbiAvIHAwOF9oZXRfcmV0aW5hKQoqIChwMTVfa29fc2NuIC8gcDE1X2tvX3JldGluYSkgLyAocDA4X2tvX3NjbiAvIHAwOF9rb19yZXRpbmEpCgpgYGB7cn0KdGltZV9zY25fZXh0cmEgPC0gZ2x1ZSgiXFwKICBwMTVoZXQgPSAoY29uZGl0aW9ucDE1X2hldF9zY24gLSBjb25kaXRpb25wMTVfaGV0X3JldGluYSksIFxcCiAgcDA4aGV0ID0gKGNvbmRpdGlvbnAwOF9oZXRfc2NuIC0gY29uZGl0aW9ucDA4X2hldF9yZXRpbmEpLCBcXAogIHAxNWhldF92c19wMDhoZXQgPSAoY29uZGl0aW9ucDE1X2hldF9zY24gLSBjb25kaXRpb25wMTVfaGV0X3JldGluYSkgLSAoY29uZGl0aW9ucDA4X2hldF9zY24gLSBjb25kaXRpb25wMDhfaGV0X3JldGluYSksCiAgcDE1a28gPSAoY29uZGl0aW9ucDE1X2tvX3NjbiAtIGNvbmRpdGlvbnAxNV9rb19yZXRpbmEpLCBcXAogIHAwOGtvID0gKGNvbmRpdGlvbnAwOF9rb19zY24gLSBjb25kaXRpb25wMDhfa29fcmV0aW5hKSwgXFwKICBwMTVrb192c19wMDhrbyA9IChjb25kaXRpb25wMTVfa29fc2NuIC0gY29uZGl0aW9ucDE1X2tvX3JldGluYSkgLSAoY29uZGl0aW9ucDA4X2tvX3NjbiAtIGNvbmRpdGlvbnAwOF9rb19yZXRpbmEpIikKdGltZV9zY25fdHJhbnNsYXRvbWVfZGVfa2VlcGVycyA8LSBsaXN0KAogICJwMTVoZXQiID0gYygicDE1X2hldF9zY24iLCAicDE1X2hldF9yZXRpbmEiKSwKICAicDA4aGV0IiA9IGMoInAwOF9oZXRfc2NuIiwgInAwOF9oZXRfcmV0aW5hIiksCiAgInAxNWtvIiA9IGMoInAxNV9rb19zY24iLCAicDE1X2tvX3JldGluYSIpLAogICJwMDhrbyIgPSBjKCJwMDhfa29fc2NuIiwgInAwOF9rb19yZXRpbmEiKSkKdGltZV9zY25fdHJhbnNsYXRvbWVfa2VlcGVycyA8LSBsaXN0KAogICJwMTVoZXQiID0gYygicDE1X2hldF9zY24iLCAicDE1X2hldF9yZXRpbmEiKSwKICAicDA4aGV0IiA9IGMoInAwOF9oZXRfc2NuIiwgInAwOF9oZXRfcmV0aW5hIiksCiAgInAxNWtvIiA9IGMoInAxNV9rb19zY24iLCAicDE1X2tvX3JldGluYSIpLAogICJwMDhrbyIgPSBjKCJwMDhfa29fc2NuIiwgInAwOF9rb19yZXRpbmEiKSwKICAicDE1X2hldF9zY192c19yZXRpbmEiID0gYygicDE1X2hldF9zY24iLCAicDE1X2hldF9yZXRpbmEiKSwKICAicDA4X2hldF9zY192c19yZXRpbmEiID0gYygicDA4X2hldF9zY24iLCAicDA4X2hldF9yZXRpbmEiKSwKICAic2NuX2hldF90cmFuc2xhdG9tZSIgPSBjKCJwMTVoZXQiLCAicDA4aGV0IiksCiAgInNjbl9rb190cmFuc2xhdG9tZSIgPSBjKCJwMTVrbyIsICJwMDhrbyIpKQp0aW1lX3Njbl90cmFuc2xhdG9tZV9kZSA8LSBhbGxfcGFpcndpc2UodjNfcGFpcndpc2VfaW5wdXQsIGZpbHRlciA9IFRSVUUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBrZWVwZXJzID0gdGltZV9zY25fdHJhbnNsYXRvbWVfZGVfa2VlcGVycywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1vZGVsX3N2cyA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbW9kZWxfZnN0cmluZyA9ICJ+IDAgKyBjb25kaXRpb24iLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG9fYmFzaWMgPSBGQUxTRSwgZG9fZHJlYW0gPSBGQUxTRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvX25vaXNlcSA9IEZBTFNFLCBkb19lYnNlcSA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZXh0cmFfY29udHJhc3RzID0gdGltZV9zY25fZXh0cmEpCnRpbWVfc2NuX3RyYW5zbGF0b21lX3Rlc3QgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgdGltZV9zY25fdHJhbnNsYXRvbWVfZGUsCiAga2VlcGVycyA9IHRpbWVfc2NuX3RyYW5zbGF0b21lX2tlZXBlcnMsCiAgbGFiZWxfY29sdW1uID0gbGFiZWxfY29sdW1uLAogIGV4Y2VsID0gZ2x1ZSgiMzN0cmFuc2xhdG9tZV94bHN4L3Rlc3RfdGltZV9zY25fdHJhbnNsYXRvbWVfdW5maWx0ZXJlZF9ub3N2YS12e3Zlcn0ueGxzeCIpKQp0aW1lX3Njbl90cmFuc2xhdG9tZV9kZV9zdmEgPC0gYWxsX3BhaXJ3aXNlKHYzX3BhaXJ3aXNlX2lucHV0LCBmaWx0ZXIgPSBUUlVFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGtlZXBlcnMgPSB0aW1lX3Njbl90cmFuc2xhdG9tZV9kZV9rZWVwZXJzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1vZGVsX3N2cyA9ICJzdmFzZXEiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1vZGVsX2ZzdHJpbmcgPSAifiAwICsgY29uZGl0aW9uIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb19iYXNpYyA9IEZBTFNFLCBkb19kcmVhbSA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvX25vaXNlcSA9IEZBTFNFLCBkb19lYnNlcSA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV4dHJhX2NvbnRyYXN0cyA9IHRpbWVfc2NuX2V4dHJhKQp0aW1lX3Njbl90cmFuc2xhdG9tZV90ZXN0X3N2YSA8LSBjb21iaW5lX2RlX3RhYmxlcygKICB0aW1lX3Njbl90cmFuc2xhdG9tZV9kZV9zdmEsCiAga2VlcGVycyA9IHRpbWVfc2NuX3RyYW5zbGF0b21lX2tlZXBlcnMsCiAgbGFiZWxfY29sdW1uID0gbGFiZWxfY29sdW1uLAogIGV4Y2VsID0gZ2x1ZSgiMzN0cmFuc2xhdG9tZV94bHN4L3Rlc3RfdGltZV9zY25fdHJhbnNsYXRvbWVfdW5maWx0ZXJlZF9zdmEtdnt2ZXJ9Lnhsc3giKSkKYGBgCgpOZXh0IHN0ZXA6IFBlcmZvcm0gdGhlIHJldGluYSBmaWx0ZXI7IG5lZWQgdG8gdGhpbmsgYWJvdXQgdGhlIHByb3Blcgp1bmlvbi9pbnRlcnNlY3Rpb24gb2YgdGhlIHJldGluYS94IGV4cHJlc3Npb24gdmFsdWVzCgpJbiB0aGUgcHJldmlvdXMgYmxvY2ssIHdlIGFyZSBtYWtpbmcgMiBnbG9iYWwgY29tcGFyaXNvbnMsIGhlcmUgaXMgb25lCm9mIHRoZW06CgoocDE1aGV0c2NuL3AxNWhldHJldCkvKHAwOGhldHNjbi9wMDhoZXRyZXQpCgpJIHRoZXJlZm9yZSB3YW50IHRvIGV4dHJhY3QgdGhlIG1vc3QgbG9naWNhbCBzZXQgb2YgZ2VuZXMgaGlnaGVyIGluCnNvbWUvYWxsIG9mIHRoZXNlIGNvbmRpdGlvbnMgd2l0aCByZXNwZWN0IHRvIHRoZSBjb3JyZXNwb25kaW5nIHd0CmNvbmRpdGlvbnMuICBQcmV2aW91c2x5LCBpbiBzZWN0aW9uICdFeHRyYWN0IGdlbmVzIGluY2x1ZGVkIGZvciBlYWNoCnNldCBvZiBjb250cmFzdHMnLCBJIGF0dGVtcHRlZCB0byBwZXJmb3JtIHRoaXMgb3BlcmF0aW9uIGZvciAyCnNwZWNpZmljIHd0IGNvbmRpdGlvbnMuICBXaGVuIHRoaXMgd2FzIHBlcmZvcm1lZCwgaXQgdG9vayB0aGUKdW5pcXVlKHVuaW9uKSBvZiB0aGUgdHdvIHNldHMuICBUaHVzIGl0IHN0YW5kcyB0byByZWFzb24gdGhhdCBJIHdhbnQKdG8gdGFrZSB0aGUgdW5pcXVlKHVuaW9uKSBvZiBhbGwgNCBpbiB0aGlzIGluc3RhbmNlPyAgZS5nLjoKCihwMTVoZXRzY24gPiBwMTV3dHNjbikgfCAocDE1aGV0cmV0ID4gcDE1d3RyZXQpIHwKICAocDA4aGV0c2NuID4gcDA4d3RzY24pIHwgKHAwOGhldHJldCA+IHAwOHd0cmV0KQoKSSBraW5kIG9mIHRoaW5rIGl0IHNob3VsZCBiZToKCigocDE1aGV0c2NuID4gcDE1d3RzY24pIHwgKHAxNWhldHJldCA+IHAxNXd0cmV0KSkgJgogICgocDA4aGV0c2NuID4gcDA4d3RzY24pIHwgKHAwOGhldHJldCA+IHAwOHd0cmV0KSkKCmdyb3NzLCBwZXJoYXBzIEkgc2hvdWxkIGp1c3QgZG8gdGhpcyBtYW51YWxseSwgZ2l2ZW4gdGhhdCB0aGVyZSBhcmUKb25seSBhIGZldyBwdXRhdGl2ZSB0cmFuc2xhdG9tZXMgdG8gcXVlcnk/CgojIFF1aWNrIGFuZCBkaXJ0eSBERVNlcTIgY29udHJhc3Qgb2YgY29udHJhc3RzCgpJbiBhIGZhc2hpb24gc2ltaWxhciB0byBob3cgSGVjdG9yIGhhbmRsZWQgdGhlIGVmZmVjdCBvZiBwaGFnb2N5dG9zaXMKd2l0aCBMYXVyYSBhbmQgTmFqaWIgYSBsb25nIHRpbWUgYWdvLCBJIHByb3Bvc2UgdG8gZG8gYSBzaW1wbGUKc3VidHJhY3Rpb24gb2YgdGhlIHJlc3VsdHMgb2Ygb3VyIHR3byBjb250cmFzdHMgd2hpY2ggY29tcHJpc2UgdGhlCnRyYW5zbGF0b21lIHF1ZXJ5IChJIHdhcyB0aGlua2luZyBhYm91dCB0aGlzIGxhc3Qgd2VlaywgdGh1cyB0aGUKaW5jbHVzaW9uIG9mIHRoZW0gaW4gdGhlIGRlIHRhYmxlcyBhYm92ZSkuICBTaW1pbGFybHkgdG8gdGhlCnBoYWdvY3l0b3NpcyBlZmZlY3QsIEkgd2lsbCBzaW1wbHkgdGFrZSB0aGUgd29yc3QgcG9zaWJsZSBhZGp1c3RlZApwLXZhbHVlLiAgSSB3aWxsIHJlcGVhdCB0aGlzIHdpdGggbGltbWEvRWRnZVIgYW5kIHNlZSBob3cgc2ltaWxhciB0aGUKZmluYWwgcmVzdWx0cyBhcmUgdG8gd2hhdCB0aG9zZSBtZXRob2RzIHByb3ZpZGUgaW4gdGhlIChhL2IpLyhjL2QpCmNvbXBhcmlzb25zLiAgSSBhbSByZWFzb25hYmx5IGNlcnRhaW4gdGhhdCBERVNlcTIncyByZXN1bHRzKCkgZnVuY3Rpb24KaGFzIHRoZSBhYmlsaXR5IHRvIHBlcmZvcm0gdGhlc2Ugb2RkIGNvbnRyYXN0cywgYnV0IEkgaGF2ZSBuZXZlcgpmaWd1cmVkIG91dCBob3c7IHBlcmhhcHMgSSB3aWxsIHVzZSB0aGlzIGFzIGEgY2hhbmNlIHRvIHJldmlzaXQgdGhhdC4uLgoKTGV0IHVzIHRlc3QgdGhpcyBpZGVhIHdpdGggdGhlIHAwOCBkbGduIHF1ZXJ5LCB3aGljaCBzZWVrcyB0byBjb21wYXJlOgoKKHAwOF9oZXRfZGxnbiAvIHAwOF9oZXRfcmV0aW5hKSAvIChwMDhfa29fZGxnbiAvIHAwOF9rb19yZXRpbmEpCgpUaGVzZSBhcmUgbWFpbnRhaW5lZCBpbiB0aGUgZGVfdGFibGUgd2l0aCB0aGUgbmFtZXMKJ3AwOF9oZXRfZGxnbl92c19yZXRpbmEnIGFuZCAncDA4X2tvX2RsZ25fdnNfcmV0aW5hJwoKIyMgcDA4IGRsZ24gaGV0IHZzIGtvCgpgYGB7cn0KcDA4X2RsZ25fY29tYmluZWRfZGVzZXEgPC0gc3VidHJhY3RfZGVzZXFfcmVzdWx0cygKICBmaXJzdF90YWJsZSA9IHAwOF9kbGduX2NvbWJpbmVkX3Rlc3RbWyJkYXRhIl1dW1sicDA4X2hldF9kbGduX3ZzX3JldGluYSJdXSwKICBzZWNvbmRfdGFibGUgPSBwMDhfZGxnbl9jb21iaW5lZF90ZXN0W1siZGF0YSJdXVtbInAwOF9rb19kbGduX3ZzX3JldGluYSJdXSwKICBmaXJzdF9sZmMgPSAiZGVzZXFfbG9nZmMiLCBzZWNvbmRfbGZjID0gImRlc2VxX2xvZ2ZjIiwKICBmaXJzdF9wID0gImRlc2VxX2FkanAiLCBzZWNvbmRfcCA9ICJkZXNlcV9hZGpwIiwKICBmaXJzdF9uYW1lID0gImhldCIsIHNlY29uZF9uYW1lID0gImtvIiwKICBleGNlbCA9IGdsdWUoIjM0dHJhbnNsYXRvbWVfZGVzZXFzdWJfeGxzeC90cmFuc2xhdG9tZV9wMDhfZGxnbl9jb21iaW5lZF9kZXNlcS12e3Zlcn0ueGxzeCIpKQpgYGAKClNlZSBob3cgc2ltaWxhciB0aGVzZSByZXN1bHRzIGFyZSB0byB0aG9zZSBvYnRhaW5lZCBmcm9tIGxpbW1hL2VkZ2VyLgoKYGBge3J9CnRlc3RfY29sdW1ucyA8LSBjKCJlZGdlcl9sb2dmYyIsICJsaW1tYV9sb2dmYyIsICJlZGdlcl9hZGpwIiwgImxpbW1hX2FkanAiKQp0ZXN0X2RmIDwtIHAwOF9kbGduX2NvbWJpbmVkX3Rlc3RbWyJkYXRhIl1dW1sicDA4X2RsZ25fdHJhbnNsYXRvbWUiXV1bLCB0ZXN0X2NvbHVtbnNdCnRlc3RfZGYgPC0gbWVyZ2UodGVzdF9kZiwgcDA4X2RsZ25fY29tYmluZWRfZGVzZXEsIGJ5ID0gInJvdy5uYW1lcyIpCnJvd25hbWVzKHRlc3RfZGYpIDwtIHRlc3RfZGZbWyJSb3cubmFtZXMiXV0KdGVzdF9kZltbIlJvdy5uYW1lcyJdXSA8LSBOVUxMCmNvci50ZXN0KHRlc3RfZGZbWyJsaW1tYV9sb2dmYyJdXSwgdGVzdF9kZltbImhldF92c19rb19sb2dmYyJdXSkKY29yLnRlc3QodGVzdF9kZltbImVkZ2VyX2xvZ2ZjIl1dLCB0ZXN0X2RmW1siaGV0X3ZzX2tvX2xvZ2ZjIl1dKQp0dCA8LSBwbG90X2xpbmVhcl9zY2F0dGVyKHRlc3RfZGZbLCBjKCJsaW1tYV9sb2dmYyIsICJoZXRfdnNfa29fbG9nZmMiKV0pCnR0W1sic2NhdHRlciJdXQp0dCA8LSBwbG90X2xpbmVhcl9zY2F0dGVyKHRlc3RfZGZbLCBjKCJlZGdlcl9hZGpwIiwgImhldF92c19rb19wIildKQp0dFtbInNjYXR0ZXIiXV0KIyMgU28sIHVzaW5nIHRoZSBtYXhpbXVtIHAtdmFsdWUgaXMgYSBjb21wbGV0ZSBmYWlsdXJlOyBidXQgdGhlIGV4dHJlbWUgc2ltaWxhcml0aWVzCiMjIGJldHdlZW4gdGhpcyBhbmQgZWRnZVIgc3VnZ2VzdCB0byBtZSB0aGF0IGl0IGlzIGxpa2VseSBwb3NzaWJsZSB0byB1c2UgdGhlIHJlc3VsdHMKIyMgZnJvbSBlZGdlUiB3aXRob3V0IGNvbmNlcm4gKG9yIGxpbW1hIGZvciB0aGF0IG1hdHRlciwgaXQgd2FzIGFsc28gZXh0cmVtZWx5IHNpbWlsYXIpCiMjIE9yIEkgY2FuIHNwZW5kIGEgbGl0dGxlIHRpbWUgYW5kIGNvbGxlY3QgdGhlIG51bWJlcnMgb24gZWFjaCBzaWRlIG9mIHRoZSBkaXZpc2lvbgojIyBhbmQgY2FsY3VsYXRlIGEgdCBzdGF0aXN0aWMgbXlzZWxmLgpgYGAKCiMgTm9uLVNwZWNpZmljIGZpbHRlcmluZyBvZiB0aGUgdHJhbnNsYXRvbWUgZGF0YQoKSSBoYXZlIG9uIGhhbmQKCiogcDA4aGV0X3ZzX3AwOGtvIDogKHAwOF9oZXRfc2NuIC0gcDA4X2hldF9yZXRpbmEpIC0gKHAwOF9rb19zY24gLSBwMDhfa29fcmV0aW5hKQoqIHAxNWhldF92c19wMTVrbyA6IChwMTVfaGV0X3NjbiAtIHAxNV9oZXRfcmV0aW5hKSAtIChwMTVfa29fc2NuIC0gcDE1X2tvX3JldGluYSkKKiBwMDhoZXRfdnNfcDA4a28gOiAocDA4X2hldF9kbGduIC0gcDA4X2hldF9yZXRpbmEpIC0gKHAwOF9rb19kbGduIC0gcDA4X2tvX3JldGluYSkKKiBwMTVoZXRfdnNfcDA4aGV0IDogKHAxNV9oZXRfc2NuIC0gcDE1X2hldF9yZXRpbmEpIC0gKHAwOF9oZXRfc2NuIC0gcDA4X2hldF9yZXRpbmEpCiogcDE1a29fdnNfcDA4a28gOiAocDE1X2tvX3NjbiAtIHAxNV9rb19yZXRpbmEpIC0gKHAwOF9rb19zY24gLSBwMDhfa29fcmV0aW5hKQoKSSBoYXZlIGdlbmUgc2V0cyB1cCBhYm92ZSB3aGljaCBkZWZpbmUgdGhlIGdlbmVzIHN1aXRhYmxlIGZvciBlYWNoIG9mCnRoZXNlIHBpZWNlcy4gIFRoZXJlIGFyZSBvbmx5IDUgY29tcGFyaXNvbnMsIGxldCB1cyBzdGVwIHRocm91Z2ggdGhlbS4KCiMjIFNDTiB0cmFuc2xhdG9tZSBoZXQva28gYXQgcDA4CgpUaGUgZGF0YSBmb3IgdGhpcyBjb250cmFzdCByZXNpZGVzIGluCnNjbl9jb21iaW5lZF90ZXN0JGRhdGEkcDA4X3Njbl90cmFuc2xhdG9tZSBvciB0aGUgc2FtZSBzbG90IG9mCnNjbl9jb21iaW5lZF90ZXN0X3N2YQoKKiBwMDhfaGV0X3NjbiAtIHAwOF9oZXRfcmV0aW5hKSAtIChwMDhfa29fc2NuIC0gcDA4X2tvX3JldGluYSkKClRodXMsIHRoZSBpbmNsdXNpb25fc2lnIHBvcnRpb25zIHRvIGV4dHJhY3QgYXJlIGZvdW5kIGluOgppbmNsdXNpb25fc2lnW1siZGVzZXEiXV1bWyJ1cHMiXV0sIGFuZCBhcmUgbmFtZWQgZXhhY3RseSBhcyB3cml0dGVuIGFib3ZlIQoKYGBge3J9CnAwOF9oZXRfdnNfa29fdHJhbnNsYXRvbWVfdW5maWx0IDwtIHNjbl9jb21iaW5lZF90ZXN0W1siZGF0YSJdXVtbInAwOF9zY25fdHJhbnNsYXRvbWUiXV0KbnVtX3VuaW9uIDwtIHVuaXF1ZShjKHJvd25hbWVzKGluY2x1c2lvbl9zaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbInAwOF9oZXRfc2NuIl1dKSwKICAgICAgICAgICAgICAgICAgICAgIHJvd25hbWVzKGluY2x1c2lvbl9zaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbInAwOF9oZXRfcmV0aW5hIl1dKSkpCmxlbmd0aChudW1fdW5pb24pCmRlbl91bmlvbiA8LSB1bmlxdWUoYyhyb3duYW1lcyhpbmNsdXNpb25fc2lnW1siZGVzZXEiXV1bWyJ1cHMiXV1bWyJwMDhfa29fc2NuIl1dKSwKICAgICAgICAgICAgICAgICAgICAgIHJvd25hbWVzKGluY2x1c2lvbl9zaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbInAwOF9rb19yZXRpbmEiXV0pKSkKbGVuZ3RoKGRlbl91bmlvbikKYm90aF91bmlvbiA8LSB1bmlxdWUoYyhudW1fdW5pb24sIGRlbl91bmlvbikpCmxlbmd0aChib3RoX3VuaW9uKQpib3RoX2ludGVyX2lkeCA8LSBudW1fdW5pb24gJWluJSBkZW5fdW5pb24KYm90aF9pbnRlciA8LSBudW1fdW5pb25bYm90aF9pbnRlcl9pZHhdCmxlbmd0aChib3RoX2ludGVyKQprZWVwZXIgPC0gbGlzdCgicDA4X3Njbl90cmFuc2xhdG9tZSIgPSBjKCJwMDhoZXQiLCAicDA4a28iKSkKcDA4X3Njbl90cmFuc2xhdG9tZV91bmlvbl9maWx0ZXJlZCA8LSBjb21iaW5lX2RlX3RhYmxlcygKICBzY25fdHJhbnNsYXRvbWVfZGUsIGtlZXBlcnMgPSBrZWVwZXIsCiAgbGFiZWxfY29sdW1uID0gbGFiZWxfY29sdW1uLAogIGV4Y2VsID0gZ2x1ZSgiMzV0cmFuc2xhdG9tZV91bmlvbi9wMDhfc2NuX3RyYW5zbGF0b21lX3VuaW9uX2ZpbHRlcmVkX25vc3ZhLXZ7dmVyfS54bHN4IiksCiAgd2FudGVkX2dlbmVzID0gYm90aF91bmlvbikKcDA4X3Njbl90cmFuc2xhdG9tZV9pbnRlcl9maWx0ZXJlZCA8LSBjb21iaW5lX2RlX3RhYmxlcygKICBzY25fdHJhbnNsYXRvbWVfZGUsIGtlZXBlcnMgPSBrZWVwZXIsCiAgbGFiZWxfY29sdW1uID0gbGFiZWxfY29sdW1uLAogIGV4Y2VsID0gZ2x1ZSgiMzV0cmFuc2xhdG9tZV91bmlvbi9wMDhfc2NuX3RyYW5zbGF0b21lX2ludGVyc2VjdF9maWx0ZXJlZF9ub3N2YS12e3Zlcn0ueGxzeCIpLAogIHdhbnRlZF9nZW5lcyA9IGJvdGhfaW50ZXIpCnAwOF9zY25fdHJhbnNsYXRvbWVfdW5pb25fZmlsdGVyZWRfc3ZhIDwtIGNvbWJpbmVfZGVfdGFibGVzKAogIHNjbl90cmFuc2xhdG9tZV9kZV9zdmEsIGtlZXBlcnMgPSBrZWVwZXIsCiAgbGFiZWxfY29sdW1uID0gbGFiZWxfY29sdW1uLAogIGV4Y2VsID0gZ2x1ZSgiMzV0cmFuc2xhdG9tZV91bmlvbi9wMDhfc2NuX3RyYW5zbGF0b21lX3VuaW9uX2ZpbHRlcmVkX3N2YS12e3Zlcn0ueGxzeCIpLAogIHdhbnRlZF9nZW5lcyA9IGJvdGhfdW5pb24pCnAwOF9zY25fdHJhbnNsYXRvbWVfdW5pb25fZmlsdGVyZWQgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgc2NuX3RyYW5zbGF0b21lX2RlLCBrZWVwZXJzID0ga2VlcGVyLAogIGxhYmVsX2NvbHVtbiA9IGxhYmVsX2NvbHVtbiwKICBleGNlbCA9IGdsdWUoIjM1dHJhbnNsYXRvbWVfdW5pb24vcDA4X3Njbl90cmFuc2xhdG9tZV9pbnRlcnNlY3RfZmlsdGVyZWRfc3ZhLXZ7dmVyfS54bHN4IiksCiAgd2FudGVkX2dlbmVzID0gYm90aF9pbnRlcikKYGBgCgojIFZlbm4vVXBTZXQgb2YgUmV0aW5hLCBTQ04sIGFuZCBkTEdOIERFIEdlbmVzCgpIZXJlIGlzIGEgc25pcHBldCBmcm9tIFJhc2htaSB3aGljaCBleHByZXNzZXMgbmljZWx5IHRoZSBERS1yZXN1bHQKY29tcGFyaXNvbnMgc2hlIGlzIG1vc3QgaW50ZXJlc3RlZDoKClNpbmNlLCBJIHdhbnQgdG8ga25vdyB0aGUgbnVtYmVyIG9mIERFRyBleHByZXNzZWQgaW4gUmV0aW5hLCBTQ04gYW5kCmRMR04gd2l0aCByZXNwZWN0IHRvIGdlbm90eXBlLCBMb2NhdGlvbiBhbmQgdGltZS4gSSBwcmVwYXJlZCB0aGUgdmVubgpkaWFncmFtIGZvciB0aGVzZSBjb21wYXJpc29uOgoKKiBHZW5vdHlwZTogUDggUmV0IEhldCB2cyBLTywgUDE1IFJldCBIZXQgdnMgS08sIFA4IFNDTiBIZXQgdnMgS08sCiAgICAgICAgICAgIFAxNSBTQ04gSGV0IHZzIEtPLCBQOCBkTEdOIEhldCB2cyBLTywgUDE1IGRMR04gSGV0IHZzIEtPCiogTG9jYXRpb246IFA4X2hldCBSZXQgdnMgU0NOLCBQOF9LTyBSZXQgdnMgU0NOLCBQMTVfaGV0IFJldCB2cyBTQ04sCiAgICAgICAgICAgIFAxNV9LTyBSZXQgdnMgU0NOLCBQOF9oZXQgUmV0IHZzIGRMR04sIFA4X0tPIFJldCB2cyBkTEdOLAogICAgICAgICAgICBQMTVfaGV0IFJldCB2cyBkTEdOLCBQMTVfS08gUmV0IHZzIGRMR04sIFA4X2hldCBTQ04gdnMgZExHTiwKICAgICAgICAgICAgUDhfS08gU0NOIHZzIGRMR04sIFAxNV9oZXQgU0NOIHZzIGRMR04sIFAxNV9LTyBTQ04gdnMgZExHTi4KClNpbmNlIEkgd2FzIGludGVyZXN0ZWQgaW4gdW5kZXJzdGFuZGluZyB0aGUgY2hhbmdlIGluIGxvY2FsIHRyYW5zbGF0b21lCmFjY29yZGluZyB0byBMb2NhdGlvbiBmb3IgZGlmZmVyZW50IGRldmVsb3BtZW50YWwgdGltZSBwb2ludHMgZm9yIEhldAphbmQgS08uIEhlbmNlLCBJIHRyaWVkIHRvIGdlbmVyYXRlIGEgdmVubiBkaWFncmFtIGZvciBMb2NhdGlvbiAoUmV0CmFuZCBTQ04pIGF0IGRldmVsb3BtZW50YWwgdGltZSBwb2ludHMgUDggYW5kIFAxNSBmb3IgZ2Vub3R5cGUgaGV0IGFuZApLTy4gIFNvIHRoZSB2ZW5uIGRpYWdyYW0gLyB1cHNldCBwbG90IHdpbGwgYmUgZm9yIGxvY2F0aW9uIHdoZXJlIHNvbWUKZ2VuZXMgd2lsbCBiZSBzaGFyZWQvdW5pcXVlIGZvciBQOF9SZXRfaGV0LCBQOF9TQ05fSGV0LCBQMTVfUmV0X0hFVCwKUDE1X1NDTl9IRVQuICBXZSBjYW4gcHJlcGFyZSBhbiB1cHNldCBwbG90IGZvciBQOF9SZXRfS08sClA4X1NDTl9LTywgUDE1X1JldF9LTyBhbmQgUDE1X1NDTl9LTyBhbHNvLiBPciBjYW4gZ2VuZXJhdGUgYW4gdXBzZXQKcGxvdCBieSBjb21iaW5pbmcgYm90aCBQOF9SZXRfaGV0LCBQOF9TQ05fSGV0LCBQMTVfUmV0X0hFVCBhbmQKUDE1X1NDTl9IRVQgYW5kIFA4X1JldF9LTywgUDhfU0NOX0tPLCBQMTVfUmV0X0tPIGFuZCBQMTVfU0NOX0tPLgoKT2ssIGxldCB1cyBzZWUgaWYgSSBjYW4gaW1wbGVtZW50IHRoaXMsIHN0YXJ0aW5nIHdpdGggdGhlIGdlbm90eXBlIHF1ZXJ5CgoqIEdlbm90eXBlOiBQOCBSZXQgSGV0IHZzIEtPLCBQMTUgUmV0IEhldCB2cyBLTywgUDggU0NOIEhldCB2cyBLTywKICAgICAgICAgICAgUDE1IFNDTiBIZXQgdnMgS08sIFA4IGRMR04gSGV0IHZzIEtPLCBQMTUgZExHTiBIZXQgdnMgS08KCiMjIGtvIHZzIGhldDsgYWxsIGxvY2F0aW9ucyBhbmQgdGltZXMKCmBgYHtyfQojIyBUaGUgYXBwcm9wcmlhdGUgZGF0YSBzdHJ1Y3R1cmUgaXMgJ2dlbm90eXBlX3RhYmxlcycsCiMjIGFuZCB0aGUgdGFibGVzIG9mIGludGVyZXN0IGFyZToKdGFibGVfbmFtZXMgPC0gYygia2hfcDA4X3JldGluYSIsICJraF9wMTVfcmV0aW5hIiwgImtoX3AwOF9zY24iLAogICAgICAgICAgICAgICAgICJraF9wMTVfc2NuIiwgImtoX3AwOF9kbGduIiwgImtoX3AxNV9kbGduIikKdGFibGVfbmFtZXMgJWluJSBuYW1lcyhnZW5vdHlwZV9zaWcpCm5ld3NpZyA8LSBnZW5vdHlwZV9zaWdbWzFdXQpmb3IgKHNpZyBpbiAyOmxlbmd0aCh0YWJsZV9uYW1lcykpIHsKICBuYW1lIDwtIHRhYmxlX25hbWVzW3NpZ10KICBuZXdzaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dIDwtIGdlbm90eXBlX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXQogIG5ld3NpZ1tbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSA8LSBnZW5vdHlwZV9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXQp9Cmdlbm90eXBlX3Vwc2V0ciA8LSB1cHNldHJfc2lnKG5ld3NpZykKZ2Vub3R5cGVfdXBzZXRfd3JpdHRlbiA8LSB3cml0ZV91cHNldF9ncm91cHMoZ2Vub3R5cGVfdXBzZXRyLCBleGNlbCA9ICIzNnVwc2V0X2dlbm90eXBlL2dlbm90eXBlX3Vwc2V0X2dyb3Vwcy54bHN4IikKZ2Vub3R5cGVfdXBzZXRyW1siYWxsX3Bsb3QiXV0KcHAoZmlsZSA9ICIzNnVwc2V0X2dlbm90eXBlL3Rlc3RfZ2Vub3R5cGVfdXBzZXQucGRmIikKcHJpbnQoZ2Vub3R5cGVfdXBzZXRyW1siYWxsX3Bsb3QiXV0pCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmBgYAoKTm93IGxldCB1cyB0cnkgdGhlIGxvY2F0aW9uLXNwZWNpZmljIGNvbXBhcmlzb25zCgojIyBzY24gdnMgcmV0aW5hLCBwMDgKCmBgYHtyfQojIyBUaGUgYXBwcm9wcmlhdGUgZGF0YSBzdHJ1Y3R1cmUgaXMgJ2dlbm90eXBlX3RhYmxlcycsCiMjIGFuZCB0aGUgdGFibGVzIG9mIGludGVyZXN0IGFyZToKdGFibGVfbmFtZXMgPC0gYygic3JfcDA4X2hldCIsICJzcl9wMDhfa28iKQp0YWJsZV9uYW1lcyAlaW4lIG5hbWVzKGxvY2F0aW9uX3NpZykKbG9jYXRpb25fdXBzZXRfaW5wdXQgPC0gbGlzdCgpCmZpcnN0X3RhYmxlIDwtIHRhYmxlX25hbWVzWzFdCm5ld3NpZyA8LSBsb2NhdGlvbl9zaWdbW2ZpcnN0X3RhYmxlXV0KZm9yIChzaWcgaW4gMjpsZW5ndGgodGFibGVfbmFtZXMpKSB7CiAgbmFtZSA8LSB0YWJsZV9uYW1lc1tzaWddCiAgbmV3c2lnW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXSA8LSBsb2NhdGlvbl9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1sidXBzIl1dW1tuYW1lXV0KICBuZXdzaWdbWyJkZXNlcSJdXVtbImRvd25zIl1dW1tuYW1lXV0gPC0gbG9jYXRpb25fc2lnW1tuYW1lXV1bWyJkZXNlcSJdXVtbImRvd25zIl1dW1tuYW1lXV0KfQpsb2NhdGlvbl91cHNldHIgPC0gdXBzZXRyX3NpZyhuZXdzaWcpCmxvY2F0aW9uX3Vwc2V0X3dyaXR0ZW4gPC0gd3JpdGVfdXBzZXRfZ3JvdXBzKGxvY2F0aW9uX3Vwc2V0ciwgZXhjZWwgPSAiMzZ1cHNldF9nZW5vdHlwZS9zcl9wMDhfaGV0a29fdXBzZXRfZ3JvdXBzLnhsc3giKQpsb2NhdGlvbl91cHNldHJbWyJhbGxfcGxvdCJdXQpwcChmaWxlID0gIjM2dXBzZXRfZ2Vub3R5cGUvdGVzdF9sb2NhdGlvbl9zcl9wMDhfaGV0a29fdXBzZXQucGRmIikKcHJpbnQobG9jYXRpb25fdXBzZXRyW1siYWxsX3Bsb3QiXV0pCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmBgYAoKSSBhbSByZWFzb25hYmx5IGNlcnRhaW4gdGhhdCBSYXNobWkgd291bGQgbGlrZSBhIHRhYmxlIG9mIHRoZSBnZW5lcwpzaGFyZWQgYW1vbmcgaW5jcmVhc2VkIHNjbiBrbyBhbmQgaGV0IGluIHRoZSBhYm92ZSBwbG90IGFsb25nIHdpdGggdGhlCmluY3JlYXNlZCByZXRpbmEgKGUuZy4gdGhlIDI2OSBhbmQgMTAzIGdlbmUgc2V0cykuCgojIyBzY24gdnMgcmV0aW5hLCBwMTUKCmBgYHtyfQp0YWJsZV9uYW1lcyA8LSBjKCJzcl9wMTVfaGV0IiwgInNyX3AxNV9rbyIpCnRhYmxlX25hbWVzICVpbiUgbmFtZXMobG9jYXRpb25fc2lnKQpsb2NhdGlvbl91cHNldF9pbnB1dCA8LSBsaXN0KCkKZmlyc3RfdGFibGUgPC0gdGFibGVfbmFtZXNbMV0KbmV3c2lnIDwtIGxvY2F0aW9uX3NpZ1tbZmlyc3RfdGFibGVdXQpmb3IgKHNpZyBpbiAyOmxlbmd0aCh0YWJsZV9uYW1lcykpIHsKICBuYW1lIDwtIHRhYmxlX25hbWVzW3NpZ10KICBuZXdzaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dIDwtIGxvY2F0aW9uX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXQogIG5ld3NpZ1tbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSA8LSBsb2NhdGlvbl9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXQp9CmxvY2F0aW9uX3Vwc2V0ciA8LSB1cHNldHJfc2lnKG5ld3NpZykKbG9jYXRpb25fdXBzZXRfd3JpdHRlbiA8LSB3cml0ZV91cHNldF9ncm91cHMobG9jYXRpb25fdXBzZXRyLCBleGNlbCA9ICIzNnVwc2V0X2dlbm90eXBlL3NyX3AxNV9oZXRrb191cHNldF9ncm91cHMueGxzeCIpCmxvY2F0aW9uX3Vwc2V0cltbImFsbF9wbG90Il1dCnByaW50KHNjbl9yZXRpbmFfcDE1X3Vwc2V0X3Jlc3VsdCkKcHAoZmlsZSA9ICIzNnVwc2V0X2dlbm90eXBlL3Rlc3RfbG9jYXRpb25fc3JfcDE1X2hldGtvX3Vwc2V0LnBkZiIpCnByaW50KGxvY2F0aW9uX3Vwc2V0cltbImFsbF9wbG90Il1dKQpwbG90dGVkIDwtIGRldi5vZmYoKQpgYGAKCiMjIGRsZ24gdnMgcmV0aW5hLCBwMDgKCmBgYHtyfQojIyBUaGUgYXBwcm9wcmlhdGUgZGF0YSBzdHJ1Y3R1cmUgaXMgJ2dlbm90eXBlX3RhYmxlcycsCiMjIGFuZCB0aGUgdGFibGVzIG9mIGludGVyZXN0IGFyZToKdGFibGVfbmFtZXMgPC0gYygiZHJfcDA4X2hldCIsICJkcl9wMDhfa28iKQpsb2NhdGlvbl91cHNldF9pbnB1dCA8LSBsaXN0KCkKZmlyc3RfdGFibGUgPC0gdGFibGVfbmFtZXNbMV0KbmV3c2lnIDwtIGxvY2F0aW9uX3NpZ1tbZmlyc3RfdGFibGVdXQpmb3IgKHNpZyBpbiAyOmxlbmd0aCh0YWJsZV9uYW1lcykpIHsKICBuYW1lIDwtIHRhYmxlX25hbWVzW3NpZ10KICBuZXdzaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dIDwtIGxvY2F0aW9uX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXQogIG5ld3NpZ1tbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSA8LSBsb2NhdGlvbl9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXQp9CmxvY2F0aW9uX3Vwc2V0ciA8LSB1cHNldHJfc2lnKG5ld3NpZykKbG9jYXRpb25fdXBzZXRfd3JpdHRlbiA8LSB3cml0ZV91cHNldF9ncm91cHMobG9jYXRpb25fdXBzZXRyLCBleGNlbCA9ICIzdXBzZXRfZ2Vub3R5cGUvZHJfcDA4X2hldGtvX3Vwc2V0X2dyb3Vwcy54bHN4IikKbG9jYXRpb25fdXBzZXRyW1siYWxsX3Bsb3QiXV0KcHAoZmlsZSA9ICIzNnVwc2V0X2dlbm90eXBlL3Rlc3RfbG9jYXRpb25fZHJfcDA4X2hldGtvX3Vwc2V0LnBkZiIpCnByaW50KGxvY2F0aW9uX3Vwc2V0cltbImFsbF9wbG90Il1dKQpwbG90dGVkIDwtIGRldi5vZmYoKQpgYGAKCiMjIGRsZ24gdnMgcmV0aW5hLCBwMTUKCmBgYHtyfQojIyBUaGUgYXBwcm9wcmlhdGUgZGF0YSBzdHJ1Y3R1cmUgaXMgJ2dlbm90eXBlX3RhYmxlcycsCiMjIGFuZCB0aGUgdGFibGVzIG9mIGludGVyZXN0IGFyZToKdGFibGVfbmFtZXMgPC0gYygiZHJfcDE1X2hldCIsICJkcl9wMTVfa28iKQpsb2NhdGlvbl91cHNldF9pbnB1dCA8LSBsaXN0KCkKZmlyc3RfdGFibGUgPC0gdGFibGVfbmFtZXNbMV0KbmV3c2lnIDwtIGxvY2F0aW9uX3NpZ1tbZmlyc3RfdGFibGVdXQpmb3IgKHNpZyBpbiAyOmxlbmd0aCh0YWJsZV9uYW1lcykpIHsKICBuYW1lIDwtIHRhYmxlX25hbWVzW3NpZ10KICBuZXdzaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dIDwtIGxvY2F0aW9uX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXQogIG5ld3NpZ1tbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSA8LSBsb2NhdGlvbl9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXQp9CmxvY2F0aW9uX3Vwc2V0ciA8LSB1cHNldHJfc2lnKG5ld3NpZykKbG9jYXRpb25fdXBzZXRfd3JpdHRlbiA8LSB3cml0ZV91cHNldF9ncm91cHMobG9jYXRpb25fdXBzZXRyLCBleGNlbCA9ICIzN3Vwc2V0X2xvY2F0aW9uL2RyX3AxNV9oZXRrb191cHNldF9ncm91cHMueGxzeCIpCmxvY2F0aW9uX3Vwc2V0cltbImFsbF9wbG90Il1dCnBwKGZpbGUgPSAiMzd1cHNldF9sb2NhdGlvbi90ZXN0X2xvY2F0aW9uX2RyX3AxNV9oZXRrb191cHNldC5wZGYiKQpwcmludChsb2NhdGlvbl91cHNldHJbWyJhbGxfcGxvdCJdXSkKcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgojIyBkbGduIHZzIHNjbiwgcDA4CgpgYGB7cn0KdGFibGVfbmFtZXMgPC0gYygiZHNfcDA4X2hldCIsICJkc19wMDhfa28iKQpsb2NhdGlvbl91cHNldF9pbnB1dCA8LSBsaXN0KCkKZmlyc3RfdGFibGUgPC0gdGFibGVfbmFtZXNbMV0KbmV3c2lnIDwtIGxvY2F0aW9uX3NpZ1tbZmlyc3RfdGFibGVdXQpmb3IgKHNpZyBpbiAyOmxlbmd0aCh0YWJsZV9uYW1lcykpIHsKICBuYW1lIDwtIHRhYmxlX25hbWVzW3NpZ10KICBuZXdzaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dIDwtIGxvY2F0aW9uX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXQogIG5ld3NpZ1tbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSA8LSBsb2NhdGlvbl9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXQp9CmxvY2F0aW9uX3Vwc2V0ciA8LSB1cHNldHJfc2lnKG5ld3NpZykKbG9jYXRpb25fdXBzZXRfd3JpdHRlbiA8LSB3cml0ZV91cHNldF9ncm91cHMobG9jYXRpb25fdXBzZXRyLCBleGNlbCA9ICIzN3Vwc2V0X2xvY2F0aW9uL2RzX3AwOF9oZXRrb191cHNldF9ncm91cHMueGxzeCIpCmxvY2F0aW9uX3Vwc2V0cltbImFsbF9wbG90Il1dCnBwKGZpbGUgPSAiMzd1cHNldF9sb2NhdGlvbi90ZXN0X2xvY2F0aW9uX2RzX3AwOF9oZXRrb191cHNldC5wZGYiKQpwcmludChsb2NhdGlvbl91cHNldHJbWyJhbGxfcGxvdCJdXSkKcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgojIyBkbGduIHZzIHNjbiwgcDE1CgpgYGB7cn0KdGFibGVfbmFtZXMgPC0gYygiZHNfcDE1X2hldCIsICJkc19wMTVfa28iKQpsb2NhdGlvbl91cHNldF9pbnB1dCA8LSBsaXN0KCkKZmlyc3RfdGFibGUgPC0gdGFibGVfbmFtZXNbMV0KbmV3c2lnIDwtIGxvY2F0aW9uX3NpZ1tbZmlyc3RfdGFibGVdXQpmb3IgKHNpZyBpbiAyOmxlbmd0aCh0YWJsZV9uYW1lcykpIHsKICBuYW1lIDwtIHRhYmxlX25hbWVzW3NpZ10KICBuZXdzaWdbWyJkZXNlcSJdXVtbInVwcyJdXVtbbmFtZV1dIDwtIGxvY2F0aW9uX3NpZ1tbbmFtZV1dW1siZGVzZXEiXV1bWyJ1cHMiXV1bW25hbWVdXQogIG5ld3NpZ1tbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXSA8LSBsb2NhdGlvbl9zaWdbW25hbWVdXVtbImRlc2VxIl1dW1siZG93bnMiXV1bW25hbWVdXQp9CmxvY2F0aW9uX3Vwc2V0ciA8LSB1cHNldHJfc2lnKG5ld3NpZykKbG9jYXRpb25fdXBzZXRfd3JpdHRlbiA8LSB3cml0ZV91cHNldF9ncm91cHMobG9jYXRpb25fdXBzZXRyLCBleGNlbCA9ICIzN3Vwc2V0X2xvY2F0aW9uL2RzX3AxNV9oZXRrb191cHNldF9ncm91cHMueGxzeCIpCmxvY2F0aW9uX3Vwc2V0cltbImFsbF9wbG90Il1dCnBwKGZpbGUgPSAiMzd1cHNldF9sb2NhdGlvbi90ZXN0X2xvY2F0aW9uX2RzX3AxNV9oZXRrb191cHNldC5wZGYiKQpwcmludChsb2NhdGlvbl91cHNldHJbWyJhbGxfcGxvdCJdXSkKcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgojIFNoYXJlZCBhbmQgdW5pcXVlIGdlbmUgc2V0cyBhY3Jvc3MgeC93dAoKSW4gdGhpcyBibG9jayBJIHdhbnQgdG8gZmluZCB0aGUgdW5pcXVlIGFuZCBzaGFyZWQgZ2VuZXMgYmV0d2VlbjoKCjEuICBzY24gcDggaGV0L3d0IGFuZCByZXRpbmEgcDggaGV0L3d0OiBod3AwOHNjbmluYywgaHdwMDhyZXRpbmMsIGh3cDA4c2NuZGVjLCBod3AwOHJldGRlYwoyLiAgc2NuIHAxNSBoZXQvd3QgYW5kIHJldGluYSBwMTUgaGV0L3d0OiBod3AxNXNjbmluYywgaHdwMTVyZXRpbmMsIGh3cDE1c2NuZGVjLCBod3AxNXJldGRlYwozLiAgIzEgYW5kICMyIHRvZ2V0aGVyOiAgOCBjYXRnb3JpZXMgYWJvdmUKNC4gIHNjbiBwOCBrby93dCBhbmQgcmV0aW5hIHA4IGtvL3d0CjUuICBzY24gcDE1IGtvL3d0IGFuZCByZXRpbmEgcDE1IGtvL3d0CjYuICAjNCBhbmQgIzUgdG9nZXRoZXIKClRoZSBjb21wYXJpc29ucyBvZiBoZXQvd3QgYXJlIGZvdW5kIGluIHRoZSAnaW5jbHVzaW9uX3NpZycgZGF0YXNldDsKYmVjYXVzZSB0aGV5IGFyZSBwcm92aWRpbmcgb3VyIGN1dG9mZnMgZm9yIG5vbnNwZWNpZmljIGJpbmRpbmcuCgojIyBOdW1iZXIgMSBhYm92ZTogcDA4X2hldCB2cyB3dCBmb3Igc2NuIGFuZCByZXRpbmEuCgpgYGB7cn0KdGFibGVfbmFtZXMgPC0gYygicDA4X2hldF9zY24iLCAicDA4X2hldF9yZXRpbmEiKQppbmNsdXNpb25fdXBzZXRyIDwtIHVwc2V0cl9zaWcoaW5jbHVzaW9uX3NpZywgY29udHJhc3RzID0gdGFibGVfbmFtZXMpCmluY2x1c2lvbl91cHNldF93cml0dGVuIDwtIHdyaXRlX3Vwc2V0X2dyb3VwcyhpbmNsdXNpb25fdXBzZXRyLCBleGNlbCA9ICIzN3Vwc2V0X2xvY2F0aW9ucy9yc19wMDhfaGV0X2luY2x1c2lvbl91cHNldF9ncm91cHMueGxzeCIpCmluY2x1c2lvbl91cHNldHJbWyJhbGxfcGxvdCJdXQpwcChmaWxlID0gIjM3dXBzZXRfbG9jYXRpb25zL2luY2x1c2lvbl9zcl9wMDhfdXBzZXQucGRmIikKcHJpbnQoaW5jbHVzaW9uX3Vwc2V0cltbImFsbF9wbG90Il1dKQpwbG90dGVkIDwtIGRldi5vZmYoKQpgYGAKCiMjIE51bWJlciAyIGFib3ZlOiBwMTVfaGV0IHZzIHd0IGZvciBzY24gYW5kIHJldGluYS4KCmBgYHtyfQp0YWJsZV9uYW1lcyA8LSBjKCJwMTVfaGV0X3NjbiIsICJwMTVfaGV0X3JldGluYSIpCmluY2x1c2lvbl91cHNldHIgPC0gdXBzZXRyX3NpZyhpbmNsdXNpb25fc2lnLCBjb250cmFzdHMgPSB0YWJsZV9uYW1lcykKaW5jbHVzaW9uX3Vwc2V0X3dyaXR0ZW4gPC0gd3JpdGVfdXBzZXRfZ3JvdXBzKGluY2x1c2lvbl91cHNldHIsIGV4Y2VsID0gIjM3dXBzZXRfbG9jYXRpb25zL3JzX3AxNV9oZXRfaW5jbHVzaW9uX3Vwc2V0X2dyb3Vwcy54bHN4IikKaW5jbHVzaW9uX3Vwc2V0cltbImFsbF9wbG90Il1dCnBwKGZpbGUgPSAiMzd1cHNldF9sb2NhdGlvbnMvaW5jbHVzaW9uX3NyX3AxNV91cHNldC5wZGYiKQpwcmludChpbmNsdXNpb25fdXBzZXRyW1siYWxsX3Bsb3QiXV0pCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmBgYAoKIyMgTnVtYmVyIDMgYWJvdmU6IGNvbWJpbmF0aW9uIG9mICMxIGFuZCAjMgoKYGBge3J9CnRhYmxlX25hbWVzIDwtIGMoInAwOF9oZXRfc2NuIiwgInAwOF9oZXRfcmV0aW5hIiwgInAxNV9oZXRfc2NuIiwgInAxNV9oZXRfcmV0aW5hIikKaW5jbHVzaW9uX3Vwc2V0ciA8LSB1cHNldHJfc2lnKGluY2x1c2lvbl9zaWcsIGNvbnRyYXN0cyA9IHRhYmxlX25hbWVzKQppbmNsdXNpb25fdXBzZXRfd3JpdHRlbiA8LSB3cml0ZV91cHNldF9ncm91cHMoaW5jbHVzaW9uX3Vwc2V0ciwgZXhjZWwgPSAiMzd1cHNldF9sb2NhdGlvbnMvcnNfcDA4cDE1X2hldF9pbmNsdXNpb25fdXBzZXRfZ3JvdXBzLnhsc3giKQppbmNsdXNpb25fdXBzZXRyW1siYWxsX3Bsb3QiXV0KcHAoZmlsZSA9ICIzN3Vwc2V0X2xvY2F0aW9ucy9pbmNsdXNpb25fc3JfcDA4cDE1X3Vwc2V0LnBkZiIpCnByaW50KGluY2x1c2lvbl91cHNldHJbWyJhbGxfcGxvdCJdXSkKcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgojIyBOdW1iZXIgNCBhYm92ZSBwMDhfa28gdnMgd3QgZm9yIHNjbiBhbmQgcmV0aW5hLgoKYGBge3J9CnRhYmxlX25hbWVzIDwtIGMoInAwOF9rb19zY24iLCAicDA4X2tvX3JldGluYSIpCmluY2x1c2lvbl91cHNldHIgPC0gdXBzZXRyX3NpZyhpbmNsdXNpb25fc2lnLCBjb250cmFzdHMgPSB0YWJsZV9uYW1lcykKaW5jbHVzaW9uX3Vwc2V0X3dyaXR0ZW4gPC0gd3JpdGVfdXBzZXRfZ3JvdXBzKGluY2x1c2lvbl91cHNldHIsIGV4Y2VsID0gIjM3dXBzZXRfbG9jYXRpb25zL3JzX3AwOF9rb19pbmNsdXNpb25fdXBzZXRfZ3JvdXBzLnhsc3giKQppbmNsdXNpb25fdXBzZXRyW1siYWxsX3Bsb3QiXV0KcHAoZmlsZSA9ICIzN3Vwc2V0X2xvY2F0aW9ucy9pbmNsdXNpb25fc3JfcDA4X2tvX3Vwc2V0LnBkZiIpCnByaW50KGluY2x1c2lvbl91cHNldHJbWyJhbGxfcGxvdCJdXSkKcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgojIyBOdW1iZXIgNSBhYm92ZSBwMTVfa28gdnMgd3QgZm9yIHNjbiBhbmQgcmV0aW5hLgoKYGBge3J9CnRhYmxlX25hbWVzIDwtIGMoInAxNV9rb19zY24iLCAicDE1X2tvX3JldGluYSIpCmluY2x1c2lvbl91cHNldHIgPC0gdXBzZXRyX3NpZyhpbmNsdXNpb25fc2lnLCBjb250cmFzdHMgPSB0YWJsZV9uYW1lcykKaW5jbHVzaW9uX3Vwc2V0X3dyaXR0ZW4gPC0gd3JpdGVfdXBzZXRfZ3JvdXBzKGluY2x1c2lvbl91cHNldHIsIGV4Y2VsID0gIjM3dXBzZXRfbG9jYXRpb25zL3JzX3AxNV9rb19pbmNsdXNpb25fdXBzZXRfZ3JvdXBzLnhsc3giKQppbmNsdXNpb25fdXBzZXRyW1siYWxsX3Bsb3QiXV0KcHAoZmlsZSA9ICIzN3Vwc2V0X2xvY2F0aW9ucy9pbmNsdXNpb25fc3JfcDE1X2tvX3Vwc2V0LnBkZiIpCnByaW50KGluY2x1c2lvbl91cHNldHJbWyJhbGxfcGxvdCJdXSkKcGxvdHRlZCA8LSBkZXYub2ZmKCkKYGBgCgojIyBOdW1iZXIgNiBhYm92ZTogQ29tYmluaW5nICM0IGFuZCAjNQoKYGBge3J9CnRhYmxlX25hbWVzIDwtIGMoInAwOF9rb19zY24iLCAicDA4X2tvX3JldGluYSIsICJwMTVfa29fc2NuIiwgInAxNV9rb19yZXRpbmEiKQppbmNsdXNpb25fdXBzZXRyIDwtIHVwc2V0cl9zaWcoaW5jbHVzaW9uX3NpZywgY29udHJhc3RzID0gdGFibGVfbmFtZXMpCmluY2x1c2lvbl91cHNldF93cml0dGVuIDwtIHdyaXRlX3Vwc2V0X2dyb3VwcyhpbmNsdXNpb25fdXBzZXRyLCBleGNlbCA9ICIzN3Vwc2V0X2xvY2F0aW9ucy9yc19wMDhwMTVfa29faW5jbHVzaW9uX3Vwc2V0X2dyb3Vwcy54bHN4IikKaW5jbHVzaW9uX3Vwc2V0cltbImFsbF9wbG90Il1dCnBwKGZpbGUgPSAiMzd1cHNldF9sb2NhdGlvbnMvaW5jbHVzaW9uX3NyX3AwOHAxNV9rb191cHNldC5wZGYiKQpwcmludChpbmNsdXNpb25fdXBzZXRyW1siYWxsX3Bsb3QiXV0pCnBsb3R0ZWQgPC0gZGV2Lm9mZigpCmBgYAoKIyBHU1ZBCgpgYGB7cn0KbXNpZ2RiIDwtICJyZWZlcmVuY2UvbXNpZ2RiX3YyMDI0LjEuTW0uZGIiCmlmIChmaWxlLmV4aXN0cyhtc2lnZGIpKSB7CiAgdjNfaF9nc3ZhIDwtIHNpbXBsZV9nc3ZhKHYzX3BhaXJ3aXNlX2lucHV0LCBvcmdkYiA9ICJvcmcuTW0uZWcuZGIiLCBzaWduYXR1cmVfY2F0ZWdvcnkgPSAibWgiLAogICAgICAgICAgICAgICAgICAgICAgICAgICBzaWduYXR1cmVzID0gbXNpZ2RiLCBpZF9zb3VyY2UgPSAiZmRhdGEiLAogICAgICAgICAgICAgICAgICAgICAgICAgICByZXF1aXJlZF9pZCA9ICJtZ2lfc3ltYm9sIikKICB2M19oX2dzdmEKCiAgdjNfaF9nc3ZhX3NpZyA8LSBnZXRfc2lnX2dzdmFfY2F0ZWdvcmllcygKICAgIHYzX2hfZ3N2YSwgZXhjZWwgPSAiMzhtc2lnZGIvZ3N2YV9zaWdfaGFsbG1hcmtfY2F0ZWdvcmllcy54bHN4IikKICB2M19oX2dzdmFfc2lnCgogIHYzX20xX2dzdmEgPC0gc2ltcGxlX2dzdmEodjNfcGFpcndpc2VfaW5wdXQsIG9yZ2RiID0gIm9yZy5NbS5lZy5kYiIsIHNpZ25hdHVyZV9jYXRlZ29yeSA9ICJtMSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaWduYXR1cmVzID0gbXNpZ2RiLCBpZF9zb3VyY2UgPSAiZmRhdGEiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVxdWlyZWRfaWQgPSAibWdpX3N5bWJvbCIpCiAgdjNfbTFfZ3N2YQogIHYzX20xX2dzdmFfc2lnIDwtIGdldF9zaWdfZ3N2YV9jYXRlZ29yaWVzKAogICAgdjNfbTFfZ3N2YSwgZXhjZWwgPSAiMzhtc2lnZGIvZ3N2YV9zaWdfcG9zaXRpb25hbF9jYXRlZ29yaWVzLnhsc3giKQogIHYzX20xX2dzdmFfc2lnCgogIHYzX20yX2dzdmEgPC0gc2ltcGxlX2dzdmEodjNfcGFpcndpc2VfaW5wdXQsIG9yZ2RiID0gIm9yZy5NbS5lZy5kYiIsIHNpZ25hdHVyZV9jYXRlZ29yeSA9ICJtMiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaWduYXR1cmVzID0gbXNpZ2RiLCBpZF9zb3VyY2UgPSAiZmRhdGEiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVxdWlyZWRfaWQgPSAibWdpX3N5bWJvbCIpCiAgdjNfbTJfZ3N2YQogIHYzX20yX2dzdmFfc2lnIDwtIGdldF9zaWdfZ3N2YV9jYXRlZ29yaWVzKAogICAgdjNfbTJfZ3N2YSwgZXhjZWwgPSAiMzhtc2lnZGIvZ3N2YV9zaWdfY3VyYXRlZF9jYXRlZ29yaWVzLnhsc3giKQogIHYzX20yX2dzdmFfc2lnCgogIHYzX20zX2dzdmEgPC0gc2ltcGxlX2dzdmEodjNfcGFpcndpc2VfaW5wdXQsIG9yZ2RiID0gIm9yZy5NbS5lZy5kYiIsIHNpZ25hdHVyZV9jYXRlZ29yeSA9ICJtMyIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaWduYXR1cmVzID0gbXNpZ2RiLCBpZF9zb3VyY2UgPSAiZmRhdGEiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVxdWlyZWRfaWQgPSAibWdpX3N5bWJvbCIpCiAgdjNfbTNfZ3N2YQogIHYzX20zX2dzdmFfc2lnIDwtIGdldF9zaWdfZ3N2YV9jYXRlZ29yaWVzKAogICAgdjNfbTNfZ3N2YSwgZXhjZWwgPSAiMzhtc2lnZGIvZ3N2YV9zaWdfcmVndWxhdG9yeV9jYXRlZ29yaWVzLnhsc3giKQogIHYzX20zX2dzdmFfc2lnCgogIHYzX201X2dzdmEgPC0gc2ltcGxlX2dzdmEodjNfcGFpcndpc2VfaW5wdXQsIG9yZ2RiID0gIm9yZy5NbS5lZy5kYiIsIHNpZ25hdHVyZV9jYXRlZ29yeSA9ICJtNSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaWduYXR1cmVzID0gbXNpZ2RiLCBpZF9zb3VyY2UgPSAiZmRhdGEiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVxdWlyZWRfaWQgPSAibWdpX3N5bWJvbCIpCiAgdjNfbTVfZ3N2YQogIHYzX201X2dzdmFfc2lnIDwtIGdldF9zaWdfZ3N2YV9jYXRlZ29yaWVzKAogICAgdjNfbTVfZ3N2YSwgZXhjZWwgPSAiMzhtc2lnZGIvZ3N2YV9zaWdfb250b2xvZ3lfY2F0ZWdvcmllcy54bHN4IikKICB2M19tNV9nc3ZhX3NpZwoKICB2M19tOF9nc3ZhIDwtIHNpbXBsZV9nc3ZhKHYzX3BhaXJ3aXNlX2lucHV0LCBvcmdkYiA9ICJvcmcuTW0uZWcuZGIiLCBzaWduYXR1cmVfY2F0ZWdvcnkgPSAibTgiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgc2lnbmF0dXJlcyA9IG1zaWdkYiwgaWRfc291cmNlID0gImZkYXRhIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlcXVpcmVkX2lkID0gIm1naV9zeW1ib2wiKQogIHYzX204X2dzdmEKICB2M19tOF9nc3ZhX3NpZyA8LSBnZXRfc2lnX2dzdmFfY2F0ZWdvcmllcygKICAgIHYzX204X2dzdmEsIGV4Y2VsID0gIjM4bXNpZ2RiL2dzdmFfc2lnX2NlbGx0eXBlX2NhdGVnb3JpZXMueGxzeCIpCiAgdjNfbThfZ3N2YV9zaWcKfQpgYGAKCiMgR1NFQSBpbWFnZXMKClVwIGFib3ZlIEkgY3JlYXRlZCBhIGZhaXJseSBsYXJnZSBzZXQgb2YgZW5yaWNobWVudC9HU0VBIGFuYWx5c2VzLgpMZXQgdXMgcHVsbCBzb21lIG9mIHRoZSBtb3N0IGludGVyZXN0aW5nIHJlc3VsdHMgaGVyZSBhbmQgbG9vayBhdAp0aGVtLgoKSGVyZSBhcmUgdGhlIHNwZWNpZmljIHF1ZXJpZXMgZnJvbSBSYXNobWk6CgoqIEdlbm90eXBlIChoZXQgdnMga28pOgogICogUDggaGV0IGFuZCBrbyBmb3IgUmV0CiAgKiBTQ04gKFA4IGhldCB2cyBLTyBTQ04KICAqIFA4IGhldCB2cyBLTyBSZXQpCiAgKiBQMTUgaGV0IGFuZCBrbyBmb3IgUmV0CiogTG9jYXRpb24gKHNvbWFsIHZzIGF4b25hbCk6CiAgKiBTUl9QMDhfS08KICAqIFNSX1AwOF9IZXQKICAqIFNSX1AxNV9LTwogICogU1JfUDE1X0hldAoqIFRpbWUocDh2cyBwMTUpOgogICogdF9oZXRfUmV0XyBwbzgtcDE1CiAgKiB0X0tPX1JldF9DIHBvOC1wMTUKICAqIHRfaGV0X1NDTl9wbzgtcDE1CiAgKiB0X0tPX1NDTl9wbzgtcDE1CgojIyBHZW5vdHlwZQoKTGV0IHVzIHRha2UgYSBtb21lbnQgYW5kIHNlZSBmb3Igd2hpY2ggY29udHJhc3RzIEkgYWNxdWlyZWQgcmVzdWx0czoKCkkgbmVlZCB0byBtYWtlIGEgbGl0dGxlIHN1bW1hcnkgZm9yIGNsdXN0ZXJwcm9maWxlciB0b28gc28gdGhhdCBJIGNhbiBlYXNpbHkgc2VlIGhvdyBtYW55IGhpdHMgdGhlcmUKYXJlIGZvciBlYWNoIGNvbnRyYXN0LgoKYGBge3J9CnN1bW1hcnkoZ2Vub3R5cGVfZnVsbF9ncCkKZm9yIChpIGluIG5hbWVzKGdlbm90eXBlX2Z1bGxfZ3ApKSB7CiAgcHJpbnQoaSkKICBwcmludChnZW5vdHlwZV9mdWxsX2dwW1tpXV1bWyJudW1faGl0cyJdXSkKfQpzdW1tYXJ5KGdlbm90eXBlX2Z1bGxfY3ApCmZvciAoaSBpbiBuYW1lcyhnZW5vdHlwZV9mdWxsX2NwKSkgewogIHByaW50KGkpCiAgcHJpbnQobnJvdyhnZW5vdHlwZV9mdWxsX2NwW1tpXV1bWyJnc2VfZ28iXV0pKQp9CmBgYAoKIyMjIHA4IGhldC9rbyBmb3IgcmV0aW5hOgoKVGhpcyBjb250cmFzdCwgZXZlbiBiZWZvcmUgZmlsdGVyaW5nIGF3YXkgdGhlIGhpZ2gtd3QgZ2VuZXMsIG9ubHkgaGFzIDggZ2VuZXMgaW4gdGhlIHNldCBvZiB1cCBhbmQKZG93biBnZW5lcyBjb21iaW5lZC4gIEFzIGEgcmVzdWx0LCBteSBmdW5jdGlvbiB3aGljaCBwZXJmb3JtcyBnUHJvZmlsZXIvY2x1c3RlclByb2ZpbGVyIHNraXBzIGl0LAphbmQgYWxzbyBza2lwcyB0aGUgcDE1IGhldC9rbyBmb3IgcmV0aW5hIHNhbXBsZXMuCgojIyMgcDggaGV0L2tvIGZvciBzY246CgpUaGlzIGhhcyBhIGJ1bmNoIG1vcmUgZ2VuZXM6IDUxIHVwIGFuZCAxMjggZG93bi4KVW5mb3J0dW5hdGVseSwgZ1Byb2ZpbGVyIHNlZXMgbm8gc2lnbmlmaWNhbnQgb3Zlci1yZXByZXNlbnRhdGlvbiBpbiB0aGUgdXAgY2F0ZWdvcnkgb2YgZ2VuZXMuClRoZSBkb3duIGNhdGVnb3J5IGhhcwoKVGhlIHVwL2Rvd24gc2V0cyBmcm9tIGNsdXN0ZXJQcm9maWxlciBoYXZlIGVucmljaF9nbywgZ3NlX2dvLCBhbmQgZ29fZGF0YSB0byBsb29rIGF0LgoKYGBge3J9Cmdlbm90eXBlX2Z1bGxfZ3Aka2hfcDA4X3Njbl91cCRudW1faGl0cwpnZW5vdHlwZV9mdWxsX2dwJGtoX3AwOF9zY25fZG93biRudW1faGl0cwoKcGxvdHMgPC0gcGxvdF9lbnJpY2hyZXN1bHQoZ2Vub3R5cGVfZnVsbF9ncCRraF9wMDhfc2NuX2Rvd25bWyJCUF9lbnJpY2giXV0pCnBsb3RzW1siZG90Il1dCnBsb3RzW1sidHJlZSJdXQpgYGAKClBlcmhhcHMgSSBzaG91bGQganVzdCBhc2sgdGhlIHF1ZXN0aW9uOiBmb3Igd2hpY2ggY2F0ZWdvcmllcyBkaWQgSSBnZXQgcmVzdWx0cyBiYWNrPwoKYGBge3J9CnN1bW1hcnkoZ2Vub3R5cGVfZnVsbF9ncCkKYGBgCgpraF9wMDhfZGxnbl91cDogTm8gc2lnbmlmaWNhbnQgZ1Byb2ZpbGVyIHJlc3VsdHMuCmtoX3AxNV9kbGduX3VwOiBTaWduaWZpY2FudCBCUCwgSFAsIEtFR0csIE1GLCBSRUFDLCBURgpraF9wMDhfc2NuX3VwOiBObyBzaWduaWZpY2FudCBnUHJvZmlsZXIgcmVzdWx0cy4Ka2hfcDA4X3Njbl9kb3duOiBTaWduaWZpY2FudCBCUCwgTWlSTkEsIE1GLCBURgpraF9wMTVfc2NuX2Rvd246IFNpZ25pZmljYW50IEJQLCBNRgoKCmBgYHtyfQpwbG90cyA8LSBwbG90X2VucmljaHJlc3VsdChnZW5vdHlwZV9mdWxsX2dwW1sia2hfcDE1X2RsZ25fdXAiXV1bWyJCUF9lbnJpY2giXV0pCnBsb3RzW1siZG90Il1dCmBgYAoKIyMgTG9jYXRpb24KCiMjIyBTY24gdnMgcmV0aW5hIGtvLCBwMDgKCmBgYHtyfQoKcGxvdHMgPC0gcGxvdF9lbnJpY2hyZXN1bHQobG9jYXRpb25fZ3BbWyJzcl9wMDhfa28iXV1bWyJzcl9wMDhfa29fdXAiXV1bWyJCUF9lbnJpY2giXV0pCnBsb3RzW1siZG90Il1dCgpwbG90cyA8LSBwbG90X2VucmljaHJlc3VsdChsb2NhdGlvbl9ncFtbInNyX3AwOF9rbyJdXVtbInNyX3AwOF9rb19kb3duIl1dW1siQlBfZW5yaWNoIl1dKQpwbG90c1tbImRvdCJdXQpgYGAKCiMjIyBzY24gdnMgcmV0aW5hLCBoZXQsIHAwOAoKRW5yaWNoZWQgZ3JvdXBzOiBCUCwgS0VHRywgTUYsIFRGLCBDQwoKYGBge3J9CnN1bW1hcnkobG9jYXRpb25fZ3BbWyJzcl9wMDhfaGV0Il1dW1sic3JfcDA4X2hldF91cCJdXSkKcGxvdHMgPC0gcGxvdF9lbnJpY2hyZXN1bHQobG9jYXRpb25fZ3BbWyJzcl9wMDhfaGV0Il1dW1sic3JfcDA4X2hldF91cCJdXVtbIkJQX2VucmljaCJdXSkKcGxvdHNbWyJkb3QiXV0KCnBsb3RzIDwtIHBsb3RfZW5yaWNocmVzdWx0KGxvY2F0aW9uX2dwW1sic3JfcDA4X2hldCJdXVtbInNyX3AwOF9oZXRfdXAiXV1bWyJDQ19lbnJpY2giXV0pCnBsb3RzW1siZG90Il1dCgpwbG90cyA8LSBwbG90X2VucmljaHJlc3VsdChsb2NhdGlvbl9ncFtbInNyX3AwOF9oZXQiXV1bWyJzcl9wMDhfaGV0X2Rvd24iXV1bWyJCUF9lbnJpY2giXV0pCnBsb3RzW1siZG90Il1dCgpzcl9wMDhfaGV0X3RvcG5fZ3NlYSA8LSBwbG90X3RvcG5fZ3NlYShsb2NhdGlvbl9jcFtbIiJdXSkKCmBgYAoKIyMjIFNjbiB2cyByZXRpbmEga28sIHAxNQoKYGBge3J9CnBsb3RzIDwtIHBsb3RfZW5yaWNocmVzdWx0KGxvY2F0aW9uX2dwW1sic3JfcDE1X2tvIl1dW1sic3JfcDE1X2tvX3VwIl1dW1siQlBfZW5yaWNoIl1dKQpwbG90c1tbImRvdCJdXQoKcGxvdHMgPC0gcGxvdF9lbnJpY2hyZXN1bHQobG9jYXRpb25fZ3BbWyJzcl9wMTVfa28iXV1bWyJzcl9wMTVfa29fZG93biJdXVtbIkJQX2VucmljaCJdXSkKcGxvdHNbWyJkb3QiXV0KYGBgCgojIyMgc2NuIHZzIHJldGluYSwgaGV0LCBwMTUKCmBgYHtyfQpwbG90cyA8LSBwbG90X2VucmljaHJlc3VsdChsb2NhdGlvbl9ncFtbInNyX3AxNV9oZXQiXV1bWyJzcl9wMTVfaGV0X3VwIl1dW1siQlBfZW5yaWNoIl1dKQpwbG90c1tbImRvdCJdXQoKcGxvdHMgPC0gcGxvdF9lbnJpY2hyZXN1bHQobG9jYXRpb25fZ3BbWyJzcl9wMTVfaGV0Il1dW1sic3JfcDE1X2hldF9kb3duIl1dW1siQlBfZW5yaWNoIl1dKQpwbG90c1tbImRvdCJdXQpgYGAKCiMjIFRpbWUKCiMjIyBoZXQgcmV0aW5hCgpVcHM6IHNpZ25pZmljYW50IHJlc3VsdHMgZm9yIEJQLCBNRiwgVEYKRG93bnM6IEJQLCBNRiwgUkVBQywgVEYsIFdQCgpgYGB7cn0KcGxvdHMgPC0gcGxvdF9lbnJpY2hyZXN1bHQodGltZV9ncFtbInRfaGV0X3JldGluYSJdXVtbInRfaGV0X3JldGluYV91cCJdXVtbIkJQX2VucmljaCJdXSkKcGxvdHNbWyJkb3QiXV0KCnBsb3RzIDwtIHBsb3RfZW5yaWNocmVzdWx0KHRpbWVfZ3BbWyJ0X2hldF9yZXRpbmEiXV1bWyJ0X2hldF9yZXRpbmFfZG93biJdXVtbIkJQX2VucmljaCJdXSkKcGxvdHNbWyJkb3QiXV0KYGBgCgojIyMga28gcmV0aW5hCgpVcDogQlAsIE1pUk5BLCBNRgpEb3duOiBCUCwgTUYsIFJFQUMsIFRGCgpgYGB7cn0KcGxvdHMgPC0gcGxvdF9lbnJpY2hyZXN1bHQodGltZV9ncFtbInRfa29fcmV0aW5hIl1dW1sidF9rb19yZXRpbmFfdXAiXV1bWyJCUF9lbnJpY2giXV0pCnBsb3RzW1siZG90Il1dCgpwbG90cyA8LSBwbG90X2VucmljaHJlc3VsdCh0aW1lX2dwW1sidF9rb19yZXRpbmEiXV1bWyJ0X2tvX3JldGluYV9kb3duIl1dW1siQlBfZW5yaWNoIl1dKQpwbG90c1tbImRvdCJdXQpgYGAKCiMjIyBoZXQgc2NuCgpOZWl0aGVyIG9mIHRoZSBTQ04gZ1Byb2ZpbGVyIHF1ZXJpZXMgcHJvdmlkZWQgYW55IHJlc3VsdHMuCgojIEJpYmxpb2dyYXBoeQoKYGBge3Igc2F2ZW1lLCBldmFsPUZBTFNFfQpwYW5kZXI6OnBhbmRlcihzZXNzaW9uSW5mbygpKQptZXNzYWdlKHBhc3RlMCgiVGhpcyBpcyBocGdsdG9vbHMgY29tbWl0OiAiLCBnZXRfZ2l0X2NvbW1pdCgpKSkKbWVzc2FnZShwYXN0ZTAoIlNhdmluZyB0byAiLCBzYXZlZmlsZSkpCnRtcCA8LSBzbShzYXZlbWUoZmlsZW5hbWUgPSBzYXZlZmlsZSkpCmBgYAoKYGBge3IgbG9hZG1lX2FmdGVyLCBldmFsPUZBTFNFfQp0bXAgPC0gbG9hZG1lKGZpbGVuYW1lID0gc2F2ZWZpbGUpCmBgYAo=