Objectives

Upon completion of this lesson, you will be able to

  • describe the difference between R Scripts and R Notebooks
  • choose the correct R code mechanism
  • list the benefits of packages
  • write R Scripts and execute them
  • create R Notebooks and knit them to documents
  • list the most common packages

The Universe of R

R is a language and environment designed specifically for statistical computing, data analysis, data-oriented computing, machine learning, data mining, and visualization. Developed in the early 1990s by Ross Ihaka and Robert Gentleman at the University of Auckland as an opens-source alternative to the S statistical programming language, R has grown to become one of the most widely used languages for data analysis1. It is open-source and available under the GNU General Public License, which means it is freely available for anyone to use and modify.

R is both a programming language and a software environment. This dual nature means that while you can write scripts and programs in R, you can also interact with it directly through a command-line interface to perform calculations and visualize data on the fly. In fact, there are four distinct ways to leverage R:

  1. write R scripts (programs) that can run by themselves
  2. use R interactively as a “calculator”
  3. create dynamic documents that intersperse markdown with R code chunks
  4. build web apps that include interactive elements and visualizations

The R programming language is similar to Python. It is interpreted (and not compiled like Java or C++), weakly typed (i.e., variables do not have to be declared first nor do you have to define the variable’s data type), and very flexible. It is ideal for exploratory data work and experimentation, along with report and script writing. R, like Python and other interpreted languages, can be slow for certain types of computational tasks, so when code must run fast, it is often re-written in C++ and then called from R.

R vs Python

R and Python are both powerful languages used extensively in data analysis, data-oriented computing, statistical analysis, data mining, and machine learning. While they share some similarities, they also have distinct differences that can influence the choice between them based on specific needs and preferences.

Python generally has a broader user base and is considered more versatile as it is a more recently developed language. According to various industry surveys and studies, Python tends to be more commonly used than R in areas like web development and software engineering, while R is more commonly used in statistical analysis, data analytics reporting, and data mining. Both languages have an extensive set of add-on packages that extend the core functionality of the language, including working with databases.

There are some key differences between these two popular languages:

  1. Language Design and Syntax: Specifically designed for statistical computing and data analysis, R’s syntax can be more intuitive for statisticians and data analysts. It offers a vast number of packages tailored for statistical methods. A general-purpose programming language with a simple and readable syntax, Python is easy to learn and versatile, making it suitable for a wide range of applications beyond data science.

  2. Libraries and Packages: R is known for its comprehensive statistical and data analysis libraries, such as ggplot2 for visualization, dplyr for data manipulation, and caret for machine learning. CRAN hosts over 18,000 packages creating a vast ecosystem. Likewise, Python boasts comprehensive libraries for data science work, including pandas for data manipulation, matplotlib and seaborn for visualization, and scikit-learn for machine learning. Like R, it also integrates well with deep learning frameworks like TensorFlow and offers its own deep learning support with PyTorch.

  3. Data Visualization: R is renowned for its advanced data visualization capabilities, particularly with ggplot2, which allows for creating intricate and customizable plots that are reproducible and can be made interactive through shiny web apps. Similarly, Python offers robust visualization libraries such as matplotlib, seaborn, and plotly, which are highly capable and provide a wide range of plotting options.

  4. Community and Support: R has a strong community focused on statistics and data analysis. CRAN and Bioconductor provide extensive resources and packages. Python also benefits from a larger and more diverse community. Resources are abundant for both languages, and support extends beyond data science into software development, automation, and more.

  5. Integration and Deployment: R is primarily used for data analysis, reporting, prototyping, exploration, and research. Integration with production environments can be more challenging compared to Python. Python is excellent for integrating data science workflows into production systems. Its versatility makes it suitable for developing web applications, APIs, and automation scripts.

Python’s popularity has been rising steadily and it is currently more popular than R in some areas, as evidenced by surveys from organizations like Stack Overflow and the TIOBE Index. However, in practice both languages are essential for data work and a practicing data or computer scientist should know both languages, in addition to other essential languages such as Java, C++, JavaScript, and perhaps Swift, Rust, Go, Rackett, among others. In fact, one can never know enough languages and knowing a variety of languages supporting differing programming paradigms makes learning emerging languages easier. Naturally, the set of language popular in any given time change. In the early 1990’s, C++ was the most popular language, then it was followed by Visual Basic, and then later Java and eventually JavaScript in the early 2000’s.

Downloading R

The base language environment of R can be freely obtained for Windows, MacOS, and various version of Linux from r-project.org.

The most common programming environment for R is Posit (formerly RStudio) which can be obtained for free from posit.co. R must be installed prior to installing Posit.

As an alternative to installing R and Posit locally, a cloud hosted version is provided on posit.cloud. A free plan is available, along with various subscriptions plans including one for education. posit.cloud supports both Posit as well as Jupyter Notebook for integrated R and Python.

Tutorial: Installing R

The tutorial below by Prof. Schedlbauer of Khoury Online demonstrates how to install R (the base language), use the R Console for interactive R, write scripts, and execute scripts from the command line.

Tutorial: Installing Posit

The tutorial below shows the process for installing and navigating Posit (formerly RStudio), the most commonly used IDE for R. Note that Posit also supports Python.

Tutorial: posit.cloud

Packages: R’s Secret Sauce

In R, a package is a collection of R functions, data, and compiled code bundled together for easy sharing and reuse. Packages extend the functionality of base R by providing additional tools, methods, and datasets that users can leverage to perform specific tasks more efficiently. Each package serves a particular purpose, such as data manipulation, statistical analysis, machine learning, visualization, or even integrating with databases and web services. Other languages might call packages libraries, but the key is that they provide additional functionality. There are literally thousands of packages freely available for R. They are the “secret sauce” that makes R such a vibrant environment for data work. As an aside, many of the packages are written in C++ for performance reasons and have also been ported to Python.

Packages are so important for R due to these reasons:

  1. Extended Functionality: Packages provide specialized functions that are not available in base R. For example, the ggplot2 package offers advanced data visualization capabilities, while dplyr simplifies data manipulation tasks.

  2. Reusability: By using packages, you can avoid reinventing the wheel. Instead of writing your own functions from scratch, you can utilize well-tested and optimized functions provided by packages.

  3. Community Contribution: The R community is highly active in developing and maintaining packages. This collaborative effort ensures a continuous influx of new tools and improvements to existing packages.

  4. Efficiency: Many packages are optimized for performance, allowing you to perform complex tasks more quickly and efficiently. For example, the data.table package is known for its speed in handling large datasets.

  5. Consistency and Best Practices: Using packages developed by experienced programmers ensures that your code follows best practices and is consistent with standard methods used in the industry.

Availability of Packages

R has a vast ecosystem of packages available through repositories (kind of like “app stores”) such as CRAN (Comprehensive R Archive Network), Bioconductor, and GitHub. CRAN is the primary repository and hosts thousands of packages, each thoroughly tested and documented.

As of 2024, CRAN hosts over 18,000 packages, catering to a wide range of applications. Bioconductor, another significant repository, focuses on bioinformatics and computational biology, offering over 2,000 packages. Additionally, many developers share their packages on GitHub, providing access to cutting-edge tools and the latest advancements in R programming.

The bottom line is that packages are an integral part of the R ecosystem, significantly enhancing its capabilities and making it a powerful tool for data analysis, statistical computing, and beyond. With thousands of packages available across various repositories, R users have access to an extensive library of functions and tools that can address virtually any computational need.

Interpreted vs Compiled Languages

Understanding the difference between interpreted and compiled languages is essential in the context of programming languages like R, Python, Java, and C++. These differences impact how the languages are executed, their performance, and their typical use cases.

Interpreted Languages

R is an interpreted language, meaning its code is executed line-by-line by an interpreter at runtime. This approach offers several advantages and disadvantages. Code can be written and executed interactively, making these languages ideal for rapid prototyping, exploratory data analysis, and scripting. Errors can be detected and fixed on the fly, which is useful for debugging.

Other examples of interpreted languages include JavaScript and PHP.

Interpreted languages are generally more portable across different platforms since the interpreter handles the machine-specific details, but they are typically slower than compiled languages because the interpreter must translate the high-level instructions into machine code every time the code runs. Performance can be mitigated through optimization techniques and the use of libraries written in compiled languages such as C++ (a method that R takes advantage of).

Compiled Languages

C and C++ are examples of compiled languages, meaning their code is translated into machine code by a compiler before execution. This process creates an executable file that can be run independently of the original source code but is specific to a particular CPU and operating systems (e.g., code compiled for an Apple M2 cannot run on a Linux computer with an Intel i7 CPU).

Compiled code generally runs much faster than interpreted code because it is translated directly into machine language, which the processor can execute directly. Optimization performed by the compiler can lead to significant performance improvements. In addition, some errors can be caught at compile-time, which can result in more robust and reliable code. Once compiled, the executable runs independently of the source code, which can be advantageous for distributing software as the source code isn’t revealed.

Hybrid Languages

Python and Java are considered a hybrid between interpreted and compiled languages. Their source code is compiled into bytecode by a compiler. This bytecode is then interpreted or just-in-time (JIT) compiled by a Virtual Machine (.e.g, the Java Virtual machine: JVM) at runtime. This approach combines some of the advantages of both interpreted and compiled languages as the compiled code is portable across operating systems and CPU designs.

The video below illustrates the common differences between these three types of execution paradigms for programming languages:

Common Use Cases for R

R is a powerful tool widely used across various fields due to its extensive capabilities in statistical computing, data analysis, and graphical representation. Below are some common use cases where R excels:

Data Analysis and Statistical Computing

R is specifically designed for data analysis and statistical computing, making it a preferred choice for statisticians and data scientists. It offers a vast array of built-in functions and packages for conducting statistical tests, building predictive models, and performing advanced data analysis. Common statistical tasks include hypothesis testing, regression analysis, ANOVA, time series analysis, and clustering.

Database Access with R

R provides robust tools for accessing and manipulating databases, which is essential for handling large datasets and integrating data from various sources. The DBI package, along with database-specific packages like RMySQL, RPostgres, and RODBC, allows users to connect to relational databases such as MySQL, PostgreSQL, and SQL Server. For example, using the DBI and RSQLite packages, you can connect to an SQLite database, execute SQL queries, and retrieve results directly into R for analysis and visualization:

Data Visualization

One of R’s strengths is its ability to create high-quality graphics and visualizations. Packages like ggplot2 allow users to produce complex and aesthetically pleasing plots with ease. Whether it’s basic charts like bar plots and histograms or advanced visualizations like heatmaps and interactive graphs, R provides powerful tools to visualize data effectively.

Machine Learning and Predictive Modeling

R is widely used in the field of machine learning for building and evaluating predictive models. The caret package, among others, provides a unified interface for training and tuning a variety of machine learning algorithms. R supports both supervised and unsupervised learning techniques, including decision trees, random forests, support vector machines, and neural networks.

Data Mining with R

R is extensively used for data mining, which involves extracting useful information and patterns from large datasets. The rattle package provides a graphical user interface for data mining, making it accessible for users with varying levels of programming expertise. Additionally, packages like arules and arulesViz support association rule mining and visualization, while tm and text2vec facilitate text mining and natural language processing. R’s comprehensive suite of tools for classification, clustering, and regression, combined with its powerful visualization capabilities, make it ideal for uncovering insights and knowledge from complex data.

Bioinformatics and Genomics

R is a popular choice in the bioinformatics community for analyzing and visualizing biological data. Packages like Bioconductor offer tools for the analysis of genomic data, including DNA sequencing, gene expression, and SNP analysis. R’s ability to handle large datasets and perform complex statistical analysis makes it suitable for biological research.

Financial Analysis

R is used extensively in the finance industry for tasks such as risk analysis, portfolio management, and time series forecasting. Packages like quantmod and TTR provide tools for financial modeling and technical analysis. R’s robust statistical capabilities are essential for developing and testing financial models.

Social Science Research

Researchers in social sciences use R for data analysis, survey analysis, and statistical modeling. The ability to manage and analyze large datasets, combined with R’s extensive statistical functions, makes it a valuable tool for conducting research in sociology, psychology, economics, and political science.

Reporting and Reproducible Research

R facilitates the creation of reproducible reports and dynamic documents through packages like knitr and rmarkdown. These tools allow users to integrate R code with Markdown or LaTeX, producing documents that include both analysis and narrative text. This is particularly useful in academic research and industry reporting, ensuring that analyses are transparent and reproducible.

In summary, R’s versatility and powerful statistical capabilities make it an invaluable tool across various domains. From data analysis and visualization to machine learning and bioinformatics, R provides the tools necessary to perform sophisticated analyses and produce high-quality results. Understanding these common use cases will help you leverage R effectively in your specific field of study or work.

Integrated Programming Environments for R

While you really need nothing more than a simple text editor, such as Sublime, although even Notepad or TextEdit would be fine, to write R scripts (aka programs), most developers leverage an integrated development environment (IDE) that offers syntax-aware editors, code completion, AI assistants, github integration, source code control, package control, and source file management with projects.

The base language environment of R can be freely obtained for Windows, MacOS, and various version of Linux from r-project.org.

For R, there are several popular IDEs and tools that enhance the programming experience by offering features like syntax highlighting, code completion, debugging, and direct execution of code. Here are some of the most commonly used development environments for R:

Posit (RStudio)

Posit (formerly RStudio), is arguably the most popular and widely used IDE for R. It is available in both desktop and server versions and offers a user-friendly interface with many features designed to improve productivity and ease of use.

  • Key Features:
    • Syntax highlighting and code completion.
    • Integrated R help and documentation.
    • Interactive graphics with support for multiple plots.
    • Debugging and profiling tools.
    • Version control integration (Git and SVN).
    • Project management capabilities.
    • Support for R Markdown and dynamic report generation.
  • Usage: Posit is suitable for all levels of R users, from beginners to advanced programmers. It is extensively used in academic settings, data analysis, research environments, and for production. It supports the development of R scripts, markdown documents, presentations, and R Notebooks. Posit supports not just R, but also Python, SQL, D3, as well as Java and C++ allowing multi-language programming.

Posit (formerly RStudio) can be obtained for free from posit.co. R must be installed prior to installing Posit.

As an alternative to installing R and Posit locally, a cloud hosted version is provided on posit.cloud. A free plan is available, along with various subscriptions plans including one for education. posit.cloud supports both Posit as well as Jupyter Notebook for integrated R and Python.

Jupyter Notebooks

Jupyter Notebooks are an increasingly popular choice for data scientists and researchers due to their interactive and literate programming approach. While Jupyter is language-agnostic, it supports R through the IRKernel along with Python – Jupyter Notebooks are equivalent to R Notebooks and the new Quarto Documents.

  • Key Features:
    • Interactive, web-based notebook environment.
    • Support for rich media output (plots, images, videos).
    • In-line code execution with immediate feedback.
    • Integration with various data science libraries.
    • Shareable notebooks with narrative text, code, and output.
  • Usage: Jupyter Notebooks are ideal for exploratory data analysis, teaching, and creating reproducible research documents.

Visual Studio Code (VS Code)

Visual Studio Code is a lightweight, powerful code editor developed by Microsoft. With the appropriate extensions, VS Code can be transformed into a robust IDE for R.

  • Key Features:
    • Syntax highlighting and code completion.
    • Integrated terminal and debugger.
    • Extensive extensions marketplace (e.g., R Extension for Visual Studio Code).
    • Git integration for version control.
    • Support for multiple programming languages.
  • Usage: VS Code is suitable for developers who prefer a versatile code editor that can be customized for various programming needs, including R development.

Emacs with ESS (Emacs Speaks Statistics)

Emacs, combined with the ESS (Emacs Speaks Statistics) package, provides a powerful environment for R programming. ESS is an add-on package that enhances Emacs to support interactive statistical programming and data analysis.

  • Key Features:
    • Advanced text editing capabilities.
    • Syntax highlighting and code completion.
    • Direct interaction with the R process.
    • Debugging and profiling tools.
    • Integration with other statistical packages (e.g., SAS, Stata).
  • Usage: Emacs with ESS is favored by users who are comfortable with Emacs and seek a highly customizable and extensible programming environment.

RGui (R Graphical User Interface)

RGui is the standard graphical user interface that comes with the base R installation. It provides a basic environment for writing and executing R code.

  • Key Features:
    • Simple and lightweight.
    • Integrated R console.
    • Basic script editor.
  • Usage: RGui is suitable for quick, simple tasks and is often used by beginners who are just starting with R.

R as a Programming Language

R is a high-level language that primarily follows the paradigm of functional programming but also supports procedural and object-oriented programming styles. Its flexibility in accommodating different programming paradigms makes it a versatile language for a wide range of computational tasks. However, the programming mechanisms for large-scale software engineering are not as robust as those in Java (or even Python) and therefore it is not recommended to build large applications in R. However, R is an ideal language to experimentation, exploration, proof-of-concept development, research, and prototype development. It is also ideal for quick programming tasks.

Functional Programming in R

Functional programming is at the core of R’s design. In this paradigm, computation is treated as the evaluation of mathematical functions and avoids changing-state and mutable data. R’s functions are first-class objects, meaning they can be passed as arguments to other functions, returned as values from other functions, and assigned to variables.

For example, consider the use of the lapply function, which applies a function to each element of a list and returns a list of results:

# Define a simple function to square a number
square <- function(x) {
  return(x * x)
}

# Create a list of numbers
numbers <- list(1, 2, 3, 4, 5)

# Apply the square function to each element of the list
squared_numbers <- lapply(numbers, square)

print(squared_numbers)

Procedural Programming in R

R also supports procedural programming, where you write sequences of instructions to perform tasks. This approach is useful for tasks that involve looping, conditionals, and step-by-step computations. Here’s an example using a for loop to achieve the same result as the previous functional example:

# Create a list of numbers
numbers <- list(1, 2, 3, 4, 5)

# Initialize an empty list to store squared numbers
squared_numbers <- list()

# Use a for loop to square each number
for (i in 1:length(numbers)) {
  squared_numbers[[i]] <- numbers[[i]] * numbers[[i]]
}

print(squared_numbers)

Object-Oriented Programming in R

R also supports object-oriented programming (OOP) through two systems: S3 and S4. The S3 system is more informal and relies on generic functions and method dispatch based on object class. The S4 system is more formal, with explicit class and method definitions.

Here’s a simple example using the S3 system:

# Define a simple S3 class for a point
point <- function(x, y) {
  structure(list(x = x, y = y), class = "point")
}

# Define a method to print a point object
print.point <- function(p) {
  cat("Point(", p$x, ", ", p$y, ")\n", sep = "")
}

# Create a point object
p <- point(2, 3)

# Print the point object
print(p)

Literate Programming

Literate programming is a programming paradigm introduced by Donald Knuth in the early 1980s. The core idea behind literate programming is to write programs that are understandable by humans first and computers second. This is achieved by interspersing natural language explanations with source code, creating a document that can be read and understood like a book or an article. Literate programming emphasizes the importance of clear communication of ideas and logic, making the code more maintainable and easier to understand.

In literate programming, the emphasis is on explaining the logic and rationale behind the code in a way that is accessible to humans. The goal is to produce a document that is as much about conveying ideas as it is about writing executable code.

The document alternates between sections of explanatory text and sections of code. The text explains what the code does, why certain decisions were made, and how different parts of the code interact. This approach helps readers follow the thought process of the programmer, making it easier to understand complex algorithms and systems.

The structure of a literate program follows a logical narrative flow rather than the strict syntactical requirements of a programming language. This means that the order in which the code is presented in the document can differ from the order in which it is executed. Tools used in literate programming will later rearrange the code into a form that can be compiled or interpreted by the computer.

Literate programming tools allow the document to be compiled or interpreted, executing the code segments embedded within the text. This ensures that the documentation is always synchronized with the code and can be run to verify its correctness.

RMarkdown and R Notebooks are the tools in the R ecosystem that support literate programming for data analysis and reporting. Literate programming is also supported in Python with Jupyter Notebooks. Note that R Notebooks and Jupyter Notebooks can actually contain a mix of code blocks in R, Python, SQL, Java, C++, D3, and many others allowing one to use the programming language most suitable for a particular task. There is also support in Haskell and Rackett for literate programming, so it is not restricted to R and Python.

Extending R with Packages

One of the greatest strengths of R is its extensive ecosystem of packages. Packages in R are collections of R functions, data, and compiled code that extend the capabilities of the base R system. The Comprehensive R Archive Network (CRAN) is the primary repository for R packages, hosting thousands of packages that cover a wide array of functionalities. The key packages include:

  1. tidyverse: A collection of packages designed for data science. This includes ggplot2 for data visualization, dplyr for data manipulation, tidyr for data tidying, and others.

  2. data.table: An extension of data.frame that provides an enhanced version with faster data manipulation capabilities.

  3. shiny: For building interactive web applications directly from R.

  4. caret: For machine learning and predictive modeling.

  5. knitr: For dynamic report generation, integrating R code with LaTeX, HTML, and Markdown.

  6. Rcpp: For seamless integration of R with C++ to enhance performance.

  7. RSQLite: For access and integration with SQLite.

  8. ggplot2: For creating reproducible visualizations.

  9. kable: For construction of visually appealing tables.

Scripts vs Notebooks

R code can be written in two general ways: (1) as Literate Programs using R Markdown Notebooks and (2) as Scripts. Scripts are programs that can run within an IDE such as R Studio or be directly executed from the command line through R. Scripts have the benefit that they can be debugged using a debugger, while R Notebook code chunks are more difficult to debug. Another benefit of scripts is that they can be run from the command line and as part of cron jobs, i.e., they can be scheduled to run automatically at a point in time. Finally, R scripts can be included in shell scripts (on Unix and MacOS) and .bat batch programs (on Windows).

R Notebooks are markdown documents that contain embedded code in R and other languages. They are most useful when producing reports, memos, and analytics journals where we need to intersperse narratives and text with code. An R Notebook produces a document (HTML or PDF, most commonly) when it is “run” (aka “knitted”). An R Script, on the other hand, is a program that runs like in any other language.

R Scripts are preferable when you need stand-alone R programs that can be run directly from the command line or within a shell script. One main benefit of using R Scripts is that we can use the debugger within R Studio.

See Also

Conclusion

In this lesson, we explored the concepts of literate programming and its implementation using R Notebooks. Literate programming, introduced by Donald Knuth, integrates natural language explanations with source code, making programs more understandable and maintainable by humans. R Notebooks exemplify this paradigm by combining narrative text, code, and output in a single document, enhancing readability and reproducibility. We contrasted R Notebooks for dynamic document generation with R Scripts that are stand-along programs that can be run from the command line or embedded within shell scripts.

We also compared R and Python, highlighting that while both are powerful tools for data analysis and machine learning, they differ in syntax, libraries, and use cases. R is favored for statistical analysis, data analytics reporting and visualization, whereas Python is more versatile, suitable for general-purpose programming and integration into production environments. Python’s broader user base and extensive libraries make it more popular overall while R has a strong following in research. An educated computer and data scientist must know both languages.

Additionally, we discussed common development environments for R, including RStudio/Posit.


Files & Resources

All Files for Lesson 6.99

References

None.

Errata

Let us know.


  1. As “R” was a close approximation at the time to the popular “S” programming language, the name chosen was the letter prior to “S” in the alphabet. Today, “R” is much broader in scope that “S” and far more popular.↩︎

LS0tCnRpdGxlOiAiVGhlICdSJyBFY29zeXN0ZW06IEEgUXVpY2sgUHJpbWVyIgpwYXJhbXM6CiAgY2F0ZWdvcnk6IDYKICBudW1iZXI6IDk5CiAgdGltZTogNDUKICBsZXZlbDogYmVnaW5uZXIKICB0YWdzOiAicixyIHNjcmlwdCxyIG5vdGVib29rLG1hcmtkb3duLGxpdGVyYXRlIHByb2dyYW1taW5nIgogIGRlc2NyaXB0aW9uOiAiRXhwbGFpbnMgdGhlICdSJyBVbml2ZXJzZSwgdGhlIGxhbmd1YWdlLCBoaXN0b3J5LCBhbmQgdG9vbHMuCiAgICAgICAgICAgICAgICBTaG93cyBob3cgdG8gY3JlYXRlIFIgc2NyaXB0cyBhbmQgZXhlY3V0ZSB0aGVtIGZyb20gdGhlIGNvbW1hbmQKICAgICAgICAgICAgICAgIGxpbmUgYXMgc3RhbmQtYWxvbmUgcHJvZ3JhbXMgYW5kIGNvbnRyYXN0cyB0aGVtIHdpdGggUiBOb3RlYm9vawogICAgICAgICAgICAgICAgYW5kIHRoZSBrbml0dGluZyBwcm9jZXNzIHRvIHByb2R1Y2UgZG9jdW1lbnRzLiBQcm92aWRlcyBhIHNob3J0CiAgICAgICAgICAgICAgICBpbnRyb2R1Y3Rpb24gdG8gY29tbW9uIElERXMgZm9yIFIuIgpkYXRlOiAiPHNtYWxsPmByIFN5cy5EYXRlKClgPC9zbWFsbD4iCmF1dGhvcjogIjxzbWFsbD5NYXJ0aW4gU2NoZWRsYmF1ZXI8L3NtYWxsPiIKZW1haWw6ICJtLnNjaGVkbGJhdWVyQG5ldS5lZHUiCmFmZmlsaXRhdGlvbjogIk5vcnRoZWFzdGVybiBVbml2ZXJzaXR5IgpvdXRwdXQ6IAogIGJvb2tkb3duOjpodG1sX2RvY3VtZW50MjoKICAgIHRvYzogdHJ1ZQogICAgdG9jX2Zsb2F0OiB0cnVlCiAgICBjb2xsYXBzZWQ6IGZhbHNlCiAgICBudW1iZXJfc2VjdGlvbnM6IGZhbHNlCiAgICBjb2RlX2Rvd25sb2FkOiB0cnVlCiAgICB0aGVtZTogc3BhY2VsYWIKICAgIGhpZ2hsaWdodDogdGFuZ28KLS0tCgotLS0KdGl0bGU6ICI8c21hbGw+YHIgcGFyYW1zJGNhdGVnb3J5YC5gciBwYXJhbXMkbnVtYmVyYDwvc21hbGw+PGJyLz48c3BhbiBzdHlsZT0nY29sb3I6ICMyRTQwNTM7IGZvbnQtc2l6ZTogMC45ZW0nPmByIHJtYXJrZG93bjo6bWV0YWRhdGEkdGl0bGVgPC9zcGFuPiIKLS0tCgpgYGB7ciBjb2RlPXhmdW46OnJlYWRfdXRmOChwYXN0ZTAoaGVyZTo6aGVyZSgpLCcvUi9faW5zZXJ0MkRCLlInKSksIGluY2x1ZGUgPSBGQUxTRX0KYGBgCgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMjIE9iamVjdGl2ZXMKClVwb24gY29tcGxldGlvbiBvZiB0aGlzIGxlc3NvbiwgeW91IHdpbGwgYmUgYWJsZSB0bwoKLSAgIGRlc2NyaWJlIHRoZSBkaWZmZXJlbmNlIGJldHdlZW4gUiBTY3JpcHRzIGFuZCBSIE5vdGVib29rcwotICAgY2hvb3NlIHRoZSBjb3JyZWN0IFIgY29kZSBtZWNoYW5pc20KLSAgIGxpc3QgdGhlIGJlbmVmaXRzIG9mIHBhY2thZ2VzCi0gICB3cml0ZSBSIFNjcmlwdHMgYW5kIGV4ZWN1dGUgdGhlbQotICAgY3JlYXRlIFIgTm90ZWJvb2tzIGFuZCBrbml0IHRoZW0gdG8gZG9jdW1lbnRzCi0gICBsaXN0IHRoZSBtb3N0IGNvbW1vbiBwYWNrYWdlcwoKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgojIyBUaGUgVW5pdmVyc2Ugb2YgUgoKUiBpcyBhIGxhbmd1YWdlIGFuZCBlbnZpcm9ubWVudCBkZXNpZ25lZCBzcGVjaWZpY2FsbHkgZm9yIHN0YXRpc3RpY2FsIGNvbXB1dGluZywgZGF0YSBhbmFseXNpcywgZGF0YS1vcmllbnRlZCBjb21wdXRpbmcsIG1hY2hpbmUgbGVhcm5pbmcsIGRhdGEgbWluaW5nLCBhbmQgdmlzdWFsaXphdGlvbi4gRGV2ZWxvcGVkIGluIHRoZSBlYXJseSAxOTkwcyBieSBSb3NzIEloYWthIGFuZCBSb2JlcnQgR2VudGxlbWFuIGF0IHRoZSBVbml2ZXJzaXR5IG9mIEF1Y2tsYW5kIGFzIGFuIG9wZW5zLXNvdXJjZSBhbHRlcm5hdGl2ZSB0byB0aGUgUyBzdGF0aXN0aWNhbCBwcm9ncmFtbWluZyBsYW5ndWFnZSwgUiBoYXMgZ3Jvd24gdG8gYmVjb21lIG9uZSBvZiB0aGUgbW9zdCB3aWRlbHkgdXNlZCBsYW5ndWFnZXMgZm9yIGRhdGEgYW5hbHlzaXNbXmwtNi0wOTktMV0uIEl0IGlzIG9wZW4tc291cmNlIGFuZCBhdmFpbGFibGUgdW5kZXIgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlLCB3aGljaCBtZWFucyBpdCBpcyBmcmVlbHkgYXZhaWxhYmxlIGZvciBhbnlvbmUgdG8gdXNlIGFuZCBtb2RpZnkuCgpbXmwtNi0wOTktMV06IEFzICJSIiB3YXMgYSBjbG9zZSBhcHByb3hpbWF0aW9uIGF0IHRoZSB0aW1lIHRvIHRoZSBwb3B1bGFyICJTIiBwcm9ncmFtbWluZyBsYW5ndWFnZSwgdGhlIG5hbWUgY2hvc2VuIHdhcyB0aGUgbGV0dGVyIHByaW9yIHRvICJTIiBpbiB0aGUgYWxwaGFiZXQuIFRvZGF5LCAiUiIgaXMgbXVjaCBicm9hZGVyIGluIHNjb3BlIHRoYXQgIlMiIGFuZCBmYXIgbW9yZSBwb3B1bGFyLgoKUiBpcyBib3RoIGEgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2UgYW5kIGEgc29mdHdhcmUgZW52aXJvbm1lbnQuIFRoaXMgZHVhbCBuYXR1cmUgbWVhbnMgdGhhdCB3aGlsZSB5b3UgY2FuIHdyaXRlIHNjcmlwdHMgYW5kIHByb2dyYW1zIGluIFIsIHlvdSBjYW4gYWxzbyBpbnRlcmFjdCB3aXRoIGl0IGRpcmVjdGx5IHRocm91Z2ggYSBjb21tYW5kLWxpbmUgaW50ZXJmYWNlIHRvIHBlcmZvcm0gY2FsY3VsYXRpb25zIGFuZCB2aXN1YWxpemUgZGF0YSBvbiB0aGUgZmx5LiBJbiBmYWN0LCB0aGVyZSBhcmUgZm91ciBkaXN0aW5jdCB3YXlzIHRvIGxldmVyYWdlIFI6CgoxLiAgd3JpdGUgUiBzY3JpcHRzIChwcm9ncmFtcykgdGhhdCBjYW4gcnVuIGJ5IHRoZW1zZWx2ZXMKMi4gIHVzZSBSIGludGVyYWN0aXZlbHkgYXMgYSAiY2FsY3VsYXRvciIKMy4gIGNyZWF0ZSBkeW5hbWljIGRvY3VtZW50cyB0aGF0IGludGVyc3BlcnNlIG1hcmtkb3duIHdpdGggUiBjb2RlIGNodW5rcwo0LiAgYnVpbGQgd2ViIGFwcHMgdGhhdCBpbmNsdWRlIGludGVyYWN0aXZlIGVsZW1lbnRzIGFuZCB2aXN1YWxpemF0aW9ucwoKVGhlIFIgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2UgaXMgc2ltaWxhciB0byBQeXRob24uIEl0IGlzIGludGVycHJldGVkIChhbmQgbm90IGNvbXBpbGVkIGxpa2UgSmF2YSBvciBDKyspLCB3ZWFrbHkgdHlwZWQgKCppLmUuKiwgdmFyaWFibGVzIGRvIG5vdCBoYXZlIHRvIGJlIGRlY2xhcmVkIGZpcnN0IG5vciBkbyB5b3UgaGF2ZSB0byBkZWZpbmUgdGhlIHZhcmlhYmxlJ3MgZGF0YSB0eXBlKSwgYW5kIHZlcnkgZmxleGlibGUuIEl0IGlzIGlkZWFsIGZvciBleHBsb3JhdG9yeSBkYXRhIHdvcmsgYW5kIGV4cGVyaW1lbnRhdGlvbiwgYWxvbmcgd2l0aCByZXBvcnQgYW5kIHNjcmlwdCB3cml0aW5nLiBSLCBsaWtlIFB5dGhvbiBhbmQgb3RoZXIgaW50ZXJwcmV0ZWQgbGFuZ3VhZ2VzLCBjYW4gYmUgc2xvdyBmb3IgY2VydGFpbiB0eXBlcyBvZiBjb21wdXRhdGlvbmFsIHRhc2tzLCBzbyB3aGVuIGNvZGUgbXVzdCBydW4gZmFzdCwgaXQgaXMgb2Z0ZW4gcmUtd3JpdHRlbiBpbiBDKysgYW5kIHRoZW4gY2FsbGVkIGZyb20gUi4KCiMjIFIgdnMgUHl0aG9uCgpSIGFuZCBQeXRob24gYXJlIGJvdGggcG93ZXJmdWwgbGFuZ3VhZ2VzIHVzZWQgZXh0ZW5zaXZlbHkgaW4gZGF0YSBhbmFseXNpcywgZGF0YS1vcmllbnRlZCBjb21wdXRpbmcsIHN0YXRpc3RpY2FsIGFuYWx5c2lzLCBkYXRhIG1pbmluZywgYW5kIG1hY2hpbmUgbGVhcm5pbmcuIFdoaWxlIHRoZXkgc2hhcmUgc29tZSBzaW1pbGFyaXRpZXMsIHRoZXkgYWxzbyBoYXZlIGRpc3RpbmN0IGRpZmZlcmVuY2VzIHRoYXQgY2FuIGluZmx1ZW5jZSB0aGUgY2hvaWNlIGJldHdlZW4gdGhlbSBiYXNlZCBvbiBzcGVjaWZpYyBuZWVkcyBhbmQgcHJlZmVyZW5jZXMuCgpQeXRob24gZ2VuZXJhbGx5IGhhcyBhIGJyb2FkZXIgdXNlciBiYXNlIGFuZCBpcyBjb25zaWRlcmVkIG1vcmUgdmVyc2F0aWxlIGFzIGl0IGlzIGEgbW9yZSByZWNlbnRseSBkZXZlbG9wZWQgbGFuZ3VhZ2UuIEFjY29yZGluZyB0byB2YXJpb3VzIGluZHVzdHJ5IHN1cnZleXMgYW5kIHN0dWRpZXMsIFB5dGhvbiB0ZW5kcyB0byBiZSBtb3JlIGNvbW1vbmx5IHVzZWQgdGhhbiBSIGluIGFyZWFzIGxpa2Ugd2ViIGRldmVsb3BtZW50IGFuZCBzb2Z0d2FyZSBlbmdpbmVlcmluZywgd2hpbGUgUiBpcyBtb3JlIGNvbW1vbmx5IHVzZWQgaW4gc3RhdGlzdGljYWwgYW5hbHlzaXMsIGRhdGEgYW5hbHl0aWNzIHJlcG9ydGluZywgYW5kIGRhdGEgbWluaW5nLiBCb3RoIGxhbmd1YWdlcyBoYXZlIGFuIGV4dGVuc2l2ZSBzZXQgb2YgYWRkLW9uIHBhY2thZ2VzIHRoYXQgZXh0ZW5kIHRoZSBjb3JlIGZ1bmN0aW9uYWxpdHkgb2YgdGhlIGxhbmd1YWdlLCBpbmNsdWRpbmcgd29ya2luZyB3aXRoIGRhdGFiYXNlcy4KClRoZXJlIGFyZSBzb21lIGtleSBkaWZmZXJlbmNlcyBiZXR3ZWVuIHRoZXNlIHR3byBwb3B1bGFyIGxhbmd1YWdlczoKCjEuICAqKkxhbmd1YWdlIERlc2lnbiBhbmQgU3ludGF4Kio6IFNwZWNpZmljYWxseSBkZXNpZ25lZCBmb3Igc3RhdGlzdGljYWwgY29tcHV0aW5nIGFuZCBkYXRhIGFuYWx5c2lzLCBSJ3Mgc3ludGF4IGNhbiBiZSBtb3JlIGludHVpdGl2ZSBmb3Igc3RhdGlzdGljaWFucyBhbmQgZGF0YSBhbmFseXN0cy4gSXQgb2ZmZXJzIGEgdmFzdCBudW1iZXIgb2YgcGFja2FnZXMgdGFpbG9yZWQgZm9yIHN0YXRpc3RpY2FsIG1ldGhvZHMuIEEgZ2VuZXJhbC1wdXJwb3NlIHByb2dyYW1taW5nIGxhbmd1YWdlIHdpdGggYSBzaW1wbGUgYW5kIHJlYWRhYmxlIHN5bnRheCwgUHl0aG9uIGlzIGVhc3kgdG8gbGVhcm4gYW5kIHZlcnNhdGlsZSwgbWFraW5nIGl0IHN1aXRhYmxlIGZvciBhIHdpZGUgcmFuZ2Ugb2YgYXBwbGljYXRpb25zIGJleW9uZCBkYXRhIHNjaWVuY2UuCgoyLiAgKipMaWJyYXJpZXMgYW5kIFBhY2thZ2VzKio6IFIgaXMga25vd24gZm9yIGl0cyBjb21wcmVoZW5zaXZlIHN0YXRpc3RpY2FsIGFuZCBkYXRhIGFuYWx5c2lzIGxpYnJhcmllcywgc3VjaCBhcyBgZ2dwbG90MmAgZm9yIHZpc3VhbGl6YXRpb24sIGBkcGx5cmAgZm9yIGRhdGEgbWFuaXB1bGF0aW9uLCBhbmQgYGNhcmV0YCBmb3IgbWFjaGluZSBsZWFybmluZy4gQ1JBTiBob3N0cyBvdmVyIDE4LDAwMCBwYWNrYWdlcyBjcmVhdGluZyBhIHZhc3QgZWNvc3lzdGVtLiBMaWtld2lzZSwgUHl0aG9uIGJvYXN0cyBjb21wcmVoZW5zaXZlIGxpYnJhcmllcyBmb3IgZGF0YSBzY2llbmNlIHdvcmssIGluY2x1ZGluZyBgcGFuZGFzYCBmb3IgZGF0YSBtYW5pcHVsYXRpb24sIGBtYXRwbG90bGliYCBhbmQgYHNlYWJvcm5gIGZvciB2aXN1YWxpemF0aW9uLCBhbmQgYHNjaWtpdC1sZWFybmAgZm9yIG1hY2hpbmUgbGVhcm5pbmcuIExpa2UgUiwgaXQgYWxzbyBpbnRlZ3JhdGVzIHdlbGwgd2l0aCBkZWVwIGxlYXJuaW5nIGZyYW1ld29ya3MgbGlrZSBUZW5zb3JGbG93IGFuZCBvZmZlcnMgaXRzIG93biBkZWVwIGxlYXJuaW5nIHN1cHBvcnQgd2l0aCBQeVRvcmNoLgoKMy4gICoqRGF0YSBWaXN1YWxpemF0aW9uKio6IFIgaXMgcmVub3duZWQgZm9yIGl0cyBhZHZhbmNlZCBkYXRhIHZpc3VhbGl6YXRpb24gY2FwYWJpbGl0aWVzLCBwYXJ0aWN1bGFybHkgd2l0aCBgZ2dwbG90MmAsIHdoaWNoIGFsbG93cyBmb3IgY3JlYXRpbmcgaW50cmljYXRlIGFuZCBjdXN0b21pemFibGUgcGxvdHMgdGhhdCBhcmUgcmVwcm9kdWNpYmxlIGFuZCBjYW4gYmUgbWFkZSBpbnRlcmFjdGl2ZSB0aHJvdWdoICoqc2hpbnkqKiB3ZWIgYXBwcy4gU2ltaWxhcmx5LCBQeXRob24gb2ZmZXJzIHJvYnVzdCB2aXN1YWxpemF0aW9uIGxpYnJhcmllcyBzdWNoIGFzIGBtYXRwbG90bGliYCwgYHNlYWJvcm5gLCBhbmQgYHBsb3RseWAsIHdoaWNoIGFyZSBoaWdobHkgY2FwYWJsZSBhbmQgcHJvdmlkZSBhIHdpZGUgcmFuZ2Ugb2YgcGxvdHRpbmcgb3B0aW9ucy4KCjQuICAqKkNvbW11bml0eSBhbmQgU3VwcG9ydCoqOiBSIGhhcyBhIHN0cm9uZyBjb21tdW5pdHkgZm9jdXNlZCBvbiBzdGF0aXN0aWNzIGFuZCBkYXRhIGFuYWx5c2lzLiBDUkFOIGFuZCBCaW9jb25kdWN0b3IgcHJvdmlkZSBleHRlbnNpdmUgcmVzb3VyY2VzIGFuZCBwYWNrYWdlcy4gUHl0aG9uIGFsc28gYmVuZWZpdHMgZnJvbSBhIGxhcmdlciBhbmQgbW9yZSBkaXZlcnNlIGNvbW11bml0eS4gUmVzb3VyY2VzIGFyZSBhYnVuZGFudCBmb3IgYm90aCBsYW5ndWFnZXMsIGFuZCBzdXBwb3J0IGV4dGVuZHMgYmV5b25kIGRhdGEgc2NpZW5jZSBpbnRvIHNvZnR3YXJlIGRldmVsb3BtZW50LCBhdXRvbWF0aW9uLCBhbmQgbW9yZS4KCjUuICAqKkludGVncmF0aW9uIGFuZCBEZXBsb3ltZW50Kio6IFIgaXMgcHJpbWFyaWx5IHVzZWQgZm9yIGRhdGEgYW5hbHlzaXMsIHJlcG9ydGluZywgcHJvdG90eXBpbmcsIGV4cGxvcmF0aW9uLCBhbmQgcmVzZWFyY2guIEludGVncmF0aW9uIHdpdGggcHJvZHVjdGlvbiBlbnZpcm9ubWVudHMgY2FuIGJlIG1vcmUgY2hhbGxlbmdpbmcgY29tcGFyZWQgdG8gUHl0aG9uLiBQeXRob24gaXMgZXhjZWxsZW50IGZvciBpbnRlZ3JhdGluZyBkYXRhIHNjaWVuY2Ugd29ya2Zsb3dzIGludG8gcHJvZHVjdGlvbiBzeXN0ZW1zLiBJdHMgdmVyc2F0aWxpdHkgbWFrZXMgaXQgc3VpdGFibGUgZm9yIGRldmVsb3Bpbmcgd2ViIGFwcGxpY2F0aW9ucywgQVBJcywgYW5kIGF1dG9tYXRpb24gc2NyaXB0cy4KClB5dGhvbidzIHBvcHVsYXJpdHkgaGFzIGJlZW4gcmlzaW5nIHN0ZWFkaWx5IGFuZCBpdCBpcyBjdXJyZW50bHkgbW9yZSBwb3B1bGFyIHRoYW4gUiBpbiBzb21lIGFyZWFzLCBhcyBldmlkZW5jZWQgYnkgc3VydmV5cyBmcm9tIG9yZ2FuaXphdGlvbnMgbGlrZSBTdGFjayBPdmVyZmxvdyBhbmQgdGhlIFRJT0JFIEluZGV4LiBIb3dldmVyLCBpbiBwcmFjdGljZSBib3RoIGxhbmd1YWdlcyBhcmUgZXNzZW50aWFsIGZvciBkYXRhIHdvcmsgYW5kIGEgcHJhY3RpY2luZyBkYXRhIG9yIGNvbXB1dGVyIHNjaWVudGlzdCBzaG91bGQga25vdyBib3RoIGxhbmd1YWdlcywgaW4gYWRkaXRpb24gdG8gb3RoZXIgZXNzZW50aWFsIGxhbmd1YWdlcyBzdWNoIGFzIEphdmEsIEMrKywgSmF2YVNjcmlwdCwgYW5kIHBlcmhhcHMgU3dpZnQsIFJ1c3QsIEdvLCBSYWNrZXR0LCBhbW9uZyBvdGhlcnMuIEluIGZhY3QsIG9uZSBjYW4gbmV2ZXIga25vdyBlbm91Z2ggbGFuZ3VhZ2VzIGFuZCBrbm93aW5nIGEgdmFyaWV0eSBvZiBsYW5ndWFnZXMgc3VwcG9ydGluZyBkaWZmZXJpbmcgcHJvZ3JhbW1pbmcgcGFyYWRpZ21zIG1ha2VzIGxlYXJuaW5nIGVtZXJnaW5nIGxhbmd1YWdlcyBlYXNpZXIuIE5hdHVyYWxseSwgdGhlIHNldCBvZiBsYW5ndWFnZSBwb3B1bGFyIGluIGFueSBnaXZlbiB0aW1lIGNoYW5nZS4gSW4gdGhlIGVhcmx5IDE5OTAncywgQysrIHdhcyB0aGUgbW9zdCBwb3B1bGFyIGxhbmd1YWdlLCB0aGVuIGl0IHdhcyBmb2xsb3dlZCBieSBWaXN1YWwgQmFzaWMsIGFuZCB0aGVuIGxhdGVyIEphdmEgYW5kIGV2ZW50dWFsbHkgSmF2YVNjcmlwdCBpbiB0aGUgZWFybHkgMjAwMCdzLgoKIyMgRG93bmxvYWRpbmcgUgoKVGhlIGJhc2UgbGFuZ3VhZ2UgZW52aXJvbm1lbnQgb2YgUiBjYW4gYmUgZnJlZWx5IG9idGFpbmVkIGZvciBXaW5kb3dzLCBNYWNPUywgYW5kIHZhcmlvdXMgdmVyc2lvbiBvZiBMaW51eCBmcm9tIFtyLXByb2plY3Qub3JnXShodHRwczovL2NyYW4uci1wcm9qZWN0Lm9yZy9taXJyb3JzLmh0bWwpLgoKVGhlIG1vc3QgY29tbW9uIHByb2dyYW1taW5nIGVudmlyb25tZW50IGZvciBSIGlzIFBvc2l0IChmb3JtZXJseSBSU3R1ZGlvKSB3aGljaCBjYW4gYmUgb2J0YWluZWQgZm9yIGZyZWUgZnJvbSBbcG9zaXQuY29dKGh0dHBzOi8vcG9zaXQuY28vcHJvZHVjdHMvb3Blbi1zb3VyY2UvcnN0dWRpby8pLiBSIG11c3QgYmUgaW5zdGFsbGVkIHByaW9yIHRvIGluc3RhbGxpbmcgUG9zaXQuCgpBcyBhbiBhbHRlcm5hdGl2ZSB0byBpbnN0YWxsaW5nIFIgYW5kIFBvc2l0IGxvY2FsbHksIGEgY2xvdWQgaG9zdGVkIHZlcnNpb24gaXMgcHJvdmlkZWQgb24gW3Bvc2l0LmNsb3VkXShodHRwOi8vcG9zaXQuY2xvdWQpLiBBIGZyZWUgcGxhbiBpcyBhdmFpbGFibGUsIGFsb25nIHdpdGggdmFyaW91cyBzdWJzY3JpcHRpb25zIHBsYW5zIGluY2x1ZGluZyBvbmUgZm9yIGVkdWNhdGlvbi4gW3Bvc2l0LmNsb3VkXShodHRwOi8vcG9zaXQuY2xvdWQpIHN1cHBvcnRzIGJvdGggUG9zaXQgYXMgd2VsbCBhcyBKdXB5dGVyIE5vdGVib29rIGZvciBpbnRlZ3JhdGVkIFIgYW5kIFB5dGhvbi4KCiMjIyBUdXRvcmlhbDogSW5zdGFsbGluZyBSCgpUaGUgdHV0b3JpYWwgYmVsb3cgYnkgUHJvZi4gU2NoZWRsYmF1ZXIgb2YgS2hvdXJ5IE9ubGluZSBkZW1vbnN0cmF0ZXMgaG93IHRvIGluc3RhbGwgUiAodGhlIGJhc2UgbGFuZ3VhZ2UpLCB1c2UgdGhlIFIgQ29uc29sZSBmb3IgaW50ZXJhY3RpdmUgUiwgd3JpdGUgc2NyaXB0cywgYW5kIGV4ZWN1dGUgc2NyaXB0cyBmcm9tIHRoZSBjb21tYW5kIGxpbmUuCgo6Ojoge3N0eWxlPSJwYWRkaW5nOjcyLjA1JSAwIDAgMDtwb3NpdGlvbjpyZWxhdGl2ZTsifQo8aWZyYW1lIHNyYz0iaHR0cHM6Ly9wbGF5ZXIudmltZW8uY29tL3ZpZGVvLzk4OTYxMDQ2MD9iYWRnZT0wJmFtcDthdXRvcGF1c2U9MCZhbXA7cGxheWVyX2lkPTAmYW1wO2FwcF9pZD01ODQ3OSIgZnJhbWVib3JkZXI9IjAiIGFsbG93PSJhdXRvcGxheTsgZnVsbHNjcmVlbjsgcGljdHVyZS1pbi1waWN0dXJlOyBjbGlwYm9hcmQtd3JpdGUiIHN0eWxlPSJwb3NpdGlvbjphYnNvbHV0ZTt0b3A6MDtsZWZ0OjA7d2lkdGg6MTAwJTtoZWlnaHQ6MTAwJTsiIHRpdGxlPSJJbnN0YWxsIFIsIFIgQ29uc29sZSwgU2NyaXB0cyIgZGF0YS1leHRlcm5hbD0iMSI+Cgo8L2lmcmFtZT4KOjo6CgpgYGB7PWh0bWx9CjxzY3JpcHQgc3JjPSJodHRwczovL3BsYXllci52aW1lby5jb20vYXBpL3BsYXllci5qcyI+PC9zY3JpcHQ+CmBgYAojIyMgVHV0b3JpYWw6IEluc3RhbGxpbmcgUG9zaXQKClRoZSB0dXRvcmlhbCBiZWxvdyBzaG93cyB0aGUgcHJvY2VzcyBmb3IgaW5zdGFsbGluZyBhbmQgbmF2aWdhdGluZyBQb3NpdCAoZm9ybWVybHkgUlN0dWRpbyksIHRoZSBtb3N0IGNvbW1vbmx5IHVzZWQgSURFIGZvciBSLiBOb3RlIHRoYXQgUG9zaXQgYWxzbyBzdXBwb3J0cyBQeXRob24uCgojIyMjIDxUVVRPUklBTCBHT0VTIEhFUkU+CgojIyMgVHV0b3JpYWw6IHBvc2l0LmNsb3VkCgojIyBQYWNrYWdlczogUidzIFNlY3JldCBTYXVjZQoKSW4gUiwgYSBwYWNrYWdlIGlzIGEgY29sbGVjdGlvbiBvZiBSIGZ1bmN0aW9ucywgZGF0YSwgYW5kIGNvbXBpbGVkIGNvZGUgYnVuZGxlZCB0b2dldGhlciBmb3IgZWFzeSBzaGFyaW5nIGFuZCByZXVzZS4gUGFja2FnZXMgZXh0ZW5kIHRoZSBmdW5jdGlvbmFsaXR5IG9mIGJhc2UgUiBieSBwcm92aWRpbmcgYWRkaXRpb25hbCB0b29scywgbWV0aG9kcywgYW5kIGRhdGFzZXRzIHRoYXQgdXNlcnMgY2FuIGxldmVyYWdlIHRvIHBlcmZvcm0gc3BlY2lmaWMgdGFza3MgbW9yZSBlZmZpY2llbnRseS4gRWFjaCBwYWNrYWdlIHNlcnZlcyBhIHBhcnRpY3VsYXIgcHVycG9zZSwgc3VjaCBhcyBkYXRhIG1hbmlwdWxhdGlvbiwgc3RhdGlzdGljYWwgYW5hbHlzaXMsIG1hY2hpbmUgbGVhcm5pbmcsIHZpc3VhbGl6YXRpb24sIG9yIGV2ZW4gaW50ZWdyYXRpbmcgd2l0aCBkYXRhYmFzZXMgYW5kIHdlYiBzZXJ2aWNlcy4gT3RoZXIgbGFuZ3VhZ2VzIG1pZ2h0IGNhbGwgcGFja2FnZXMgbGlicmFyaWVzLCBidXQgdGhlIGtleSBpcyB0aGF0IHRoZXkgcHJvdmlkZSBhZGRpdGlvbmFsIGZ1bmN0aW9uYWxpdHkuIFRoZXJlIGFyZSBsaXRlcmFsbHkgdGhvdXNhbmRzIG9mIHBhY2thZ2VzIGZyZWVseSBhdmFpbGFibGUgZm9yIFIuIFRoZXkgYXJlIHRoZSAic2VjcmV0IHNhdWNlIiB0aGF0IG1ha2VzIFIgc3VjaCBhIHZpYnJhbnQgZW52aXJvbm1lbnQgZm9yIGRhdGEgd29yay4gQXMgYW4gYXNpZGUsIG1hbnkgb2YgdGhlIHBhY2thZ2VzIGFyZSB3cml0dGVuIGluIEMrKyBmb3IgcGVyZm9ybWFuY2UgcmVhc29ucyBhbmQgaGF2ZSBhbHNvIGJlZW4gcG9ydGVkIHRvIFB5dGhvbi4KClBhY2thZ2VzIGFyZSBzbyBpbXBvcnRhbnQgZm9yIFIgZHVlIHRvIHRoZXNlIHJlYXNvbnM6CgoxLiAgKipFeHRlbmRlZCBGdW5jdGlvbmFsaXR5Kio6IFBhY2thZ2VzIHByb3ZpZGUgc3BlY2lhbGl6ZWQgZnVuY3Rpb25zIHRoYXQgYXJlIG5vdCBhdmFpbGFibGUgaW4gYmFzZSBSLiBGb3IgZXhhbXBsZSwgdGhlIGBnZ3Bsb3QyYCBwYWNrYWdlIG9mZmVycyBhZHZhbmNlZCBkYXRhIHZpc3VhbGl6YXRpb24gY2FwYWJpbGl0aWVzLCB3aGlsZSBgZHBseXJgIHNpbXBsaWZpZXMgZGF0YSBtYW5pcHVsYXRpb24gdGFza3MuCgoyLiAgKipSZXVzYWJpbGl0eSoqOiBCeSB1c2luZyBwYWNrYWdlcywgeW91IGNhbiBhdm9pZCByZWludmVudGluZyB0aGUgd2hlZWwuIEluc3RlYWQgb2Ygd3JpdGluZyB5b3VyIG93biBmdW5jdGlvbnMgZnJvbSBzY3JhdGNoLCB5b3UgY2FuIHV0aWxpemUgd2VsbC10ZXN0ZWQgYW5kIG9wdGltaXplZCBmdW5jdGlvbnMgcHJvdmlkZWQgYnkgcGFja2FnZXMuCgozLiAgKipDb21tdW5pdHkgQ29udHJpYnV0aW9uKio6IFRoZSBSIGNvbW11bml0eSBpcyBoaWdobHkgYWN0aXZlIGluIGRldmVsb3BpbmcgYW5kIG1haW50YWluaW5nIHBhY2thZ2VzLiBUaGlzIGNvbGxhYm9yYXRpdmUgZWZmb3J0IGVuc3VyZXMgYSBjb250aW51b3VzIGluZmx1eCBvZiBuZXcgdG9vbHMgYW5kIGltcHJvdmVtZW50cyB0byBleGlzdGluZyBwYWNrYWdlcy4KCjQuICAqKkVmZmljaWVuY3kqKjogTWFueSBwYWNrYWdlcyBhcmUgb3B0aW1pemVkIGZvciBwZXJmb3JtYW5jZSwgYWxsb3dpbmcgeW91IHRvIHBlcmZvcm0gY29tcGxleCB0YXNrcyBtb3JlIHF1aWNrbHkgYW5kIGVmZmljaWVudGx5LiBGb3IgZXhhbXBsZSwgdGhlIGBkYXRhLnRhYmxlYCBwYWNrYWdlIGlzIGtub3duIGZvciBpdHMgc3BlZWQgaW4gaGFuZGxpbmcgbGFyZ2UgZGF0YXNldHMuCgo1LiAgKipDb25zaXN0ZW5jeSBhbmQgQmVzdCBQcmFjdGljZXMqKjogVXNpbmcgcGFja2FnZXMgZGV2ZWxvcGVkIGJ5IGV4cGVyaWVuY2VkIHByb2dyYW1tZXJzIGVuc3VyZXMgdGhhdCB5b3VyIGNvZGUgZm9sbG93cyBiZXN0IHByYWN0aWNlcyBhbmQgaXMgY29uc2lzdGVudCB3aXRoIHN0YW5kYXJkIG1ldGhvZHMgdXNlZCBpbiB0aGUgaW5kdXN0cnkuCgojIyMgQXZhaWxhYmlsaXR5IG9mIFBhY2thZ2VzCgpSIGhhcyBhIHZhc3QgZWNvc3lzdGVtIG9mIHBhY2thZ2VzIGF2YWlsYWJsZSB0aHJvdWdoIHJlcG9zaXRvcmllcyAoa2luZCBvZiBsaWtlICJhcHAgc3RvcmVzIikgc3VjaCBhcyAqQ1JBTiogKENvbXByZWhlbnNpdmUgUiBBcmNoaXZlIE5ldHdvcmspLCAqQmlvY29uZHVjdG9yKiwgYW5kICpHaXRIdWIuKiBDUkFOIGlzIHRoZSBwcmltYXJ5IHJlcG9zaXRvcnkgYW5kIGhvc3RzIHRob3VzYW5kcyBvZiBwYWNrYWdlcywgZWFjaCB0aG9yb3VnaGx5IHRlc3RlZCBhbmQgZG9jdW1lbnRlZC4KCkFzIG9mIDIwMjQsIENSQU4gaG9zdHMgb3ZlciAxOCwwMDAgcGFja2FnZXMsIGNhdGVyaW5nIHRvIGEgd2lkZSByYW5nZSBvZiBhcHBsaWNhdGlvbnMuIEJpb2NvbmR1Y3RvciwgYW5vdGhlciBzaWduaWZpY2FudCByZXBvc2l0b3J5LCBmb2N1c2VzIG9uIGJpb2luZm9ybWF0aWNzIGFuZCBjb21wdXRhdGlvbmFsIGJpb2xvZ3ksIG9mZmVyaW5nIG92ZXIgMiwwMDAgcGFja2FnZXMuIEFkZGl0aW9uYWxseSwgbWFueSBkZXZlbG9wZXJzIHNoYXJlIHRoZWlyIHBhY2thZ2VzIG9uIEdpdEh1YiwgcHJvdmlkaW5nIGFjY2VzcyB0byBjdXR0aW5nLWVkZ2UgdG9vbHMgYW5kIHRoZSBsYXRlc3QgYWR2YW5jZW1lbnRzIGluIFIgcHJvZ3JhbW1pbmcuCgpUaGUgYm90dG9tIGxpbmUgaXMgdGhhdCBwYWNrYWdlcyBhcmUgYW4gaW50ZWdyYWwgcGFydCBvZiB0aGUgUiBlY29zeXN0ZW0sIHNpZ25pZmljYW50bHkgZW5oYW5jaW5nIGl0cyBjYXBhYmlsaXRpZXMgYW5kIG1ha2luZyBpdCBhIHBvd2VyZnVsIHRvb2wgZm9yIGRhdGEgYW5hbHlzaXMsIHN0YXRpc3RpY2FsIGNvbXB1dGluZywgYW5kIGJleW9uZC4gV2l0aCB0aG91c2FuZHMgb2YgcGFja2FnZXMgYXZhaWxhYmxlIGFjcm9zcyB2YXJpb3VzIHJlcG9zaXRvcmllcywgUiB1c2VycyBoYXZlIGFjY2VzcyB0byBhbiBleHRlbnNpdmUgbGlicmFyeSBvZiBmdW5jdGlvbnMgYW5kIHRvb2xzIHRoYXQgY2FuIGFkZHJlc3MgdmlydHVhbGx5IGFueSBjb21wdXRhdGlvbmFsIG5lZWQuCgojIyBJbnRlcnByZXRlZCB2cyBDb21waWxlZCBMYW5ndWFnZXMKClVuZGVyc3RhbmRpbmcgdGhlIGRpZmZlcmVuY2UgYmV0d2VlbiBpbnRlcnByZXRlZCBhbmQgY29tcGlsZWQgbGFuZ3VhZ2VzIGlzIGVzc2VudGlhbCBpbiB0aGUgY29udGV4dCBvZiBwcm9ncmFtbWluZyBsYW5ndWFnZXMgbGlrZSBSLCBQeXRob24sIEphdmEsIGFuZCBDKysuIFRoZXNlIGRpZmZlcmVuY2VzIGltcGFjdCBob3cgdGhlIGxhbmd1YWdlcyBhcmUgZXhlY3V0ZWQsIHRoZWlyIHBlcmZvcm1hbmNlLCBhbmQgdGhlaXIgdHlwaWNhbCB1c2UgY2FzZXMuCgojIyMgSW50ZXJwcmV0ZWQgTGFuZ3VhZ2VzCgpSIGlzIGFuIGludGVycHJldGVkIGxhbmd1YWdlLCBtZWFuaW5nIGl0cyBjb2RlIGlzIGV4ZWN1dGVkIGxpbmUtYnktbGluZSBieSBhbiBpbnRlcnByZXRlciBhdCBydW50aW1lLiBUaGlzIGFwcHJvYWNoIG9mZmVycyBzZXZlcmFsIGFkdmFudGFnZXMgYW5kIGRpc2FkdmFudGFnZXMuIENvZGUgY2FuIGJlIHdyaXR0ZW4gYW5kIGV4ZWN1dGVkIGludGVyYWN0aXZlbHksIG1ha2luZyB0aGVzZSBsYW5ndWFnZXMgaWRlYWwgZm9yIHJhcGlkIHByb3RvdHlwaW5nLCBleHBsb3JhdG9yeSBkYXRhIGFuYWx5c2lzLCBhbmQgc2NyaXB0aW5nLiBFcnJvcnMgY2FuIGJlIGRldGVjdGVkIGFuZCBmaXhlZCBvbiB0aGUgZmx5LCB3aGljaCBpcyB1c2VmdWwgZm9yIGRlYnVnZ2luZy4KCk90aGVyIGV4YW1wbGVzIG9mIGludGVycHJldGVkIGxhbmd1YWdlcyBpbmNsdWRlIEphdmFTY3JpcHQgYW5kIFBIUC4KCkludGVycHJldGVkIGxhbmd1YWdlcyBhcmUgZ2VuZXJhbGx5IG1vcmUgcG9ydGFibGUgYWNyb3NzIGRpZmZlcmVudCBwbGF0Zm9ybXMgc2luY2UgdGhlIGludGVycHJldGVyIGhhbmRsZXMgdGhlIG1hY2hpbmUtc3BlY2lmaWMgZGV0YWlscywgYnV0IHRoZXkgYXJlIHR5cGljYWxseSBzbG93ZXIgdGhhbiBjb21waWxlZCBsYW5ndWFnZXMgYmVjYXVzZSB0aGUgaW50ZXJwcmV0ZXIgbXVzdCB0cmFuc2xhdGUgdGhlIGhpZ2gtbGV2ZWwgaW5zdHJ1Y3Rpb25zIGludG8gbWFjaGluZSBjb2RlIGV2ZXJ5IHRpbWUgdGhlIGNvZGUgcnVucy4gUGVyZm9ybWFuY2UgY2FuIGJlIG1pdGlnYXRlZCB0aHJvdWdoIG9wdGltaXphdGlvbiB0ZWNobmlxdWVzIGFuZCB0aGUgdXNlIG9mIGxpYnJhcmllcyB3cml0dGVuIGluIGNvbXBpbGVkIGxhbmd1YWdlcyBzdWNoIGFzIEMrKyAoYSBtZXRob2QgdGhhdCBSIHRha2VzIGFkdmFudGFnZSBvZikuCgojIyMgQ29tcGlsZWQgTGFuZ3VhZ2VzCgpDIGFuZCBDKysgYXJlIGV4YW1wbGVzIG9mIGNvbXBpbGVkIGxhbmd1YWdlcywgbWVhbmluZyB0aGVpciBjb2RlIGlzIHRyYW5zbGF0ZWQgaW50byBtYWNoaW5lIGNvZGUgYnkgYSBjb21waWxlciBiZWZvcmUgZXhlY3V0aW9uLiBUaGlzIHByb2Nlc3MgY3JlYXRlcyBhbiBleGVjdXRhYmxlIGZpbGUgdGhhdCBjYW4gYmUgcnVuIGluZGVwZW5kZW50bHkgb2YgdGhlIG9yaWdpbmFsIHNvdXJjZSBjb2RlIGJ1dCBpcyBzcGVjaWZpYyB0byBhIHBhcnRpY3VsYXIgQ1BVIGFuZCBvcGVyYXRpbmcgc3lzdGVtcyAoKmUuZy4qLCBjb2RlIGNvbXBpbGVkIGZvciBhbiBBcHBsZSBNMiBjYW5ub3QgcnVuIG9uIGEgTGludXggY29tcHV0ZXIgd2l0aCBhbiBJbnRlbCBpNyBDUFUpLgoKQ29tcGlsZWQgY29kZSBnZW5lcmFsbHkgcnVucyBtdWNoIGZhc3RlciB0aGFuIGludGVycHJldGVkIGNvZGUgYmVjYXVzZSBpdCBpcyB0cmFuc2xhdGVkIGRpcmVjdGx5IGludG8gbWFjaGluZSBsYW5ndWFnZSwgd2hpY2ggdGhlIHByb2Nlc3NvciBjYW4gZXhlY3V0ZSBkaXJlY3RseS4gT3B0aW1pemF0aW9uIHBlcmZvcm1lZCBieSB0aGUgY29tcGlsZXIgY2FuIGxlYWQgdG8gc2lnbmlmaWNhbnQgcGVyZm9ybWFuY2UgaW1wcm92ZW1lbnRzLiBJbiBhZGRpdGlvbiwgc29tZSBlcnJvcnMgY2FuIGJlIGNhdWdodCBhdCBjb21waWxlLXRpbWUsIHdoaWNoIGNhbiByZXN1bHQgaW4gbW9yZSByb2J1c3QgYW5kIHJlbGlhYmxlIGNvZGUuIE9uY2UgY29tcGlsZWQsIHRoZSBleGVjdXRhYmxlIHJ1bnMgaW5kZXBlbmRlbnRseSBvZiB0aGUgc291cmNlIGNvZGUsIHdoaWNoIGNhbiBiZSBhZHZhbnRhZ2VvdXMgZm9yIGRpc3RyaWJ1dGluZyBzb2Z0d2FyZSBhcyB0aGUgc291cmNlIGNvZGUgaXNuJ3QgcmV2ZWFsZWQuCgojIyMgSHlicmlkIExhbmd1YWdlcwoKUHl0aG9uIGFuZCBKYXZhIGFyZSBjb25zaWRlcmVkIGEgaHlicmlkIGJldHdlZW4gaW50ZXJwcmV0ZWQgYW5kIGNvbXBpbGVkIGxhbmd1YWdlcy4gVGhlaXIgc291cmNlIGNvZGUgaXMgY29tcGlsZWQgaW50byBieXRlY29kZSBieSBhIGNvbXBpbGVyLiBUaGlzIGJ5dGVjb2RlIGlzIHRoZW4gaW50ZXJwcmV0ZWQgb3IganVzdC1pbi10aW1lIChKSVQpIGNvbXBpbGVkIGJ5IGEgVmlydHVhbCBNYWNoaW5lICgqLmUuZyosIHRoZSBKYXZhIFZpcnR1YWwgbWFjaGluZTogSlZNKSBhdCBydW50aW1lLiBUaGlzIGFwcHJvYWNoIGNvbWJpbmVzIHNvbWUgb2YgdGhlIGFkdmFudGFnZXMgb2YgYm90aCBpbnRlcnByZXRlZCBhbmQgY29tcGlsZWQgbGFuZ3VhZ2VzIGFzIHRoZSBjb21waWxlZCBjb2RlIGlzIHBvcnRhYmxlIGFjcm9zcyBvcGVyYXRpbmcgc3lzdGVtcyBhbmQgQ1BVIGRlc2lnbnMuCgpUaGUgdmlkZW8gYmVsb3cgaWxsdXN0cmF0ZXMgdGhlIGNvbW1vbiBkaWZmZXJlbmNlcyBiZXR3ZWVuIHRoZXNlIHRocmVlIHR5cGVzIG9mIGV4ZWN1dGlvbiBwYXJhZGlnbXMgZm9yIHByb2dyYW1taW5nIGxhbmd1YWdlczoKCjxpZnJhbWUgd2lkdGg9IjQ4MCIgaGVpZ2h0PSIyNzAiIHNyYz0iaHR0cHM6Ly93d3cueW91dHViZS5jb20vZW1iZWQvSTFmNDVSRWkzazQ/c2k9Z01KMFZDb092MEhVRGJQUCIgdGl0bGU9IkludGVycHJldGVkIHZzIENvbXBpbGVkIExhbmd1YWdlcyIgZnJhbWVib3JkZXI9IjEiIGFsbG93PSJhY2NlbGVyb21ldGVyOyBhdXRvcGxheTsgY2xpcGJvYXJkLXdyaXRlOyBlbmNyeXB0ZWQtbWVkaWE7IGd5cm9zY29wZTsgcGljdHVyZS1pbi1waWN0dXJlOyB3ZWItc2hhcmUiIHJlZmVycmVycG9saWN5PSJzdHJpY3Qtb3JpZ2luLXdoZW4tY3Jvc3Mtb3JpZ2luIiBhbGxvd2Z1bGxzY3JlZW4gZGF0YS1leHRlcm5hbD0iMSI+Cgo8L2lmcmFtZT4KCiMjIENvbW1vbiBVc2UgQ2FzZXMgZm9yIFIKClIgaXMgYSBwb3dlcmZ1bCB0b29sIHdpZGVseSB1c2VkIGFjcm9zcyB2YXJpb3VzIGZpZWxkcyBkdWUgdG8gaXRzIGV4dGVuc2l2ZSBjYXBhYmlsaXRpZXMgaW4gc3RhdGlzdGljYWwgY29tcHV0aW5nLCBkYXRhIGFuYWx5c2lzLCBhbmQgZ3JhcGhpY2FsIHJlcHJlc2VudGF0aW9uLiBCZWxvdyBhcmUgc29tZSBjb21tb24gdXNlIGNhc2VzIHdoZXJlIFIgZXhjZWxzOgoKIyMjIERhdGEgQW5hbHlzaXMgYW5kIFN0YXRpc3RpY2FsIENvbXB1dGluZwoKUiBpcyBzcGVjaWZpY2FsbHkgZGVzaWduZWQgZm9yIGRhdGEgYW5hbHlzaXMgYW5kIHN0YXRpc3RpY2FsIGNvbXB1dGluZywgbWFraW5nIGl0IGEgcHJlZmVycmVkIGNob2ljZSBmb3Igc3RhdGlzdGljaWFucyBhbmQgZGF0YSBzY2llbnRpc3RzLiBJdCBvZmZlcnMgYSB2YXN0IGFycmF5IG9mIGJ1aWx0LWluIGZ1bmN0aW9ucyBhbmQgcGFja2FnZXMgZm9yIGNvbmR1Y3Rpbmcgc3RhdGlzdGljYWwgdGVzdHMsIGJ1aWxkaW5nIHByZWRpY3RpdmUgbW9kZWxzLCBhbmQgcGVyZm9ybWluZyBhZHZhbmNlZCBkYXRhIGFuYWx5c2lzLiBDb21tb24gc3RhdGlzdGljYWwgdGFza3MgaW5jbHVkZSBoeXBvdGhlc2lzIHRlc3RpbmcsIHJlZ3Jlc3Npb24gYW5hbHlzaXMsIEFOT1ZBLCB0aW1lIHNlcmllcyBhbmFseXNpcywgYW5kIGNsdXN0ZXJpbmcuCgojIyMgRGF0YWJhc2UgQWNjZXNzIHdpdGggUgoKUiBwcm92aWRlcyByb2J1c3QgdG9vbHMgZm9yIGFjY2Vzc2luZyBhbmQgbWFuaXB1bGF0aW5nIGRhdGFiYXNlcywgd2hpY2ggaXMgZXNzZW50aWFsIGZvciBoYW5kbGluZyBsYXJnZSBkYXRhc2V0cyBhbmQgaW50ZWdyYXRpbmcgZGF0YSBmcm9tIHZhcmlvdXMgc291cmNlcy4gVGhlIGBEQklgIHBhY2thZ2UsIGFsb25nIHdpdGggZGF0YWJhc2Utc3BlY2lmaWMgcGFja2FnZXMgbGlrZSBgUk15U1FMYCwgYFJQb3N0Z3Jlc2AsIGFuZCBgUk9EQkNgLCBhbGxvd3MgdXNlcnMgdG8gY29ubmVjdCB0byByZWxhdGlvbmFsIGRhdGFiYXNlcyBzdWNoIGFzIE15U1FMLCBQb3N0Z3JlU1FMLCBhbmQgU1FMIFNlcnZlci4gRm9yIGV4YW1wbGUsIHVzaW5nIHRoZSBgREJJYCBhbmQgYFJTUUxpdGVgIHBhY2thZ2VzLCB5b3UgY2FuIGNvbm5lY3QgdG8gYW4gU1FMaXRlIGRhdGFiYXNlLCBleGVjdXRlIFNRTCBxdWVyaWVzLCBhbmQgcmV0cmlldmUgcmVzdWx0cyBkaXJlY3RseSBpbnRvIFIgZm9yIGFuYWx5c2lzIGFuZCB2aXN1YWxpemF0aW9uOgoKIyMjIERhdGEgVmlzdWFsaXphdGlvbgoKT25lIG9mIFIncyBzdHJlbmd0aHMgaXMgaXRzIGFiaWxpdHkgdG8gY3JlYXRlIGhpZ2gtcXVhbGl0eSBncmFwaGljcyBhbmQgdmlzdWFsaXphdGlvbnMuIFBhY2thZ2VzIGxpa2UgYGdncGxvdDJgIGFsbG93IHVzZXJzIHRvIHByb2R1Y2UgY29tcGxleCBhbmQgYWVzdGhldGljYWxseSBwbGVhc2luZyBwbG90cyB3aXRoIGVhc2UuIFdoZXRoZXIgaXQncyBiYXNpYyBjaGFydHMgbGlrZSBiYXIgcGxvdHMgYW5kIGhpc3RvZ3JhbXMgb3IgYWR2YW5jZWQgdmlzdWFsaXphdGlvbnMgbGlrZSBoZWF0bWFwcyBhbmQgaW50ZXJhY3RpdmUgZ3JhcGhzLCBSIHByb3ZpZGVzIHBvd2VyZnVsIHRvb2xzIHRvIHZpc3VhbGl6ZSBkYXRhIGVmZmVjdGl2ZWx5LgoKIyMjIE1hY2hpbmUgTGVhcm5pbmcgYW5kIFByZWRpY3RpdmUgTW9kZWxpbmcKClIgaXMgd2lkZWx5IHVzZWQgaW4gdGhlIGZpZWxkIG9mIG1hY2hpbmUgbGVhcm5pbmcgZm9yIGJ1aWxkaW5nIGFuZCBldmFsdWF0aW5nIHByZWRpY3RpdmUgbW9kZWxzLiBUaGUgYGNhcmV0YCBwYWNrYWdlLCBhbW9uZyBvdGhlcnMsIHByb3ZpZGVzIGEgdW5pZmllZCBpbnRlcmZhY2UgZm9yIHRyYWluaW5nIGFuZCB0dW5pbmcgYSB2YXJpZXR5IG9mIG1hY2hpbmUgbGVhcm5pbmcgYWxnb3JpdGhtcy4gUiBzdXBwb3J0cyBib3RoIHN1cGVydmlzZWQgYW5kIHVuc3VwZXJ2aXNlZCBsZWFybmluZyB0ZWNobmlxdWVzLCBpbmNsdWRpbmcgZGVjaXNpb24gdHJlZXMsIHJhbmRvbSBmb3Jlc3RzLCBzdXBwb3J0IHZlY3RvciBtYWNoaW5lcywgYW5kIG5ldXJhbCBuZXR3b3Jrcy4KCiMjIyBEYXRhIE1pbmluZyB3aXRoIFIKClIgaXMgZXh0ZW5zaXZlbHkgdXNlZCBmb3IgZGF0YSBtaW5pbmcsIHdoaWNoIGludm9sdmVzIGV4dHJhY3RpbmcgdXNlZnVsIGluZm9ybWF0aW9uIGFuZCBwYXR0ZXJucyBmcm9tIGxhcmdlIGRhdGFzZXRzLiBUaGUgYHJhdHRsZWAgcGFja2FnZSBwcm92aWRlcyBhIGdyYXBoaWNhbCB1c2VyIGludGVyZmFjZSBmb3IgZGF0YSBtaW5pbmcsIG1ha2luZyBpdCBhY2Nlc3NpYmxlIGZvciB1c2VycyB3aXRoIHZhcnlpbmcgbGV2ZWxzIG9mIHByb2dyYW1taW5nIGV4cGVydGlzZS4gQWRkaXRpb25hbGx5LCBwYWNrYWdlcyBsaWtlIGBhcnVsZXNgIGFuZCBgYXJ1bGVzVml6YCBzdXBwb3J0IGFzc29jaWF0aW9uIHJ1bGUgbWluaW5nIGFuZCB2aXN1YWxpemF0aW9uLCB3aGlsZSBgdG1gIGFuZCBgdGV4dDJ2ZWNgIGZhY2lsaXRhdGUgdGV4dCBtaW5pbmcgYW5kIG5hdHVyYWwgbGFuZ3VhZ2UgcHJvY2Vzc2luZy4gUuKAmXMgY29tcHJlaGVuc2l2ZSBzdWl0ZSBvZiB0b29scyBmb3IgY2xhc3NpZmljYXRpb24sIGNsdXN0ZXJpbmcsIGFuZCByZWdyZXNzaW9uLCBjb21iaW5lZCB3aXRoIGl0cyBwb3dlcmZ1bCB2aXN1YWxpemF0aW9uIGNhcGFiaWxpdGllcywgbWFrZSBpdCBpZGVhbCBmb3IgdW5jb3ZlcmluZyBpbnNpZ2h0cyBhbmQga25vd2xlZGdlIGZyb20gY29tcGxleCBkYXRhLgoKIyMjIEJpb2luZm9ybWF0aWNzIGFuZCBHZW5vbWljcwoKUiBpcyBhIHBvcHVsYXIgY2hvaWNlIGluIHRoZSBiaW9pbmZvcm1hdGljcyBjb21tdW5pdHkgZm9yIGFuYWx5emluZyBhbmQgdmlzdWFsaXppbmcgYmlvbG9naWNhbCBkYXRhLiBQYWNrYWdlcyBsaWtlIGBCaW9jb25kdWN0b3JgIG9mZmVyIHRvb2xzIGZvciB0aGUgYW5hbHlzaXMgb2YgZ2Vub21pYyBkYXRhLCBpbmNsdWRpbmcgRE5BIHNlcXVlbmNpbmcsIGdlbmUgZXhwcmVzc2lvbiwgYW5kIFNOUCBhbmFseXNpcy4gUidzIGFiaWxpdHkgdG8gaGFuZGxlIGxhcmdlIGRhdGFzZXRzIGFuZCBwZXJmb3JtIGNvbXBsZXggc3RhdGlzdGljYWwgYW5hbHlzaXMgbWFrZXMgaXQgc3VpdGFibGUgZm9yIGJpb2xvZ2ljYWwgcmVzZWFyY2guCgojIyMgRmluYW5jaWFsIEFuYWx5c2lzCgpSIGlzIHVzZWQgZXh0ZW5zaXZlbHkgaW4gdGhlIGZpbmFuY2UgaW5kdXN0cnkgZm9yIHRhc2tzIHN1Y2ggYXMgcmlzayBhbmFseXNpcywgcG9ydGZvbGlvIG1hbmFnZW1lbnQsIGFuZCB0aW1lIHNlcmllcyBmb3JlY2FzdGluZy4gUGFja2FnZXMgbGlrZSBgcXVhbnRtb2RgIGFuZCBgVFRSYCBwcm92aWRlIHRvb2xzIGZvciBmaW5hbmNpYWwgbW9kZWxpbmcgYW5kIHRlY2huaWNhbCBhbmFseXNpcy4gUidzIHJvYnVzdCBzdGF0aXN0aWNhbCBjYXBhYmlsaXRpZXMgYXJlIGVzc2VudGlhbCBmb3IgZGV2ZWxvcGluZyBhbmQgdGVzdGluZyBmaW5hbmNpYWwgbW9kZWxzLgoKIyMjIFNvY2lhbCBTY2llbmNlIFJlc2VhcmNoCgpSZXNlYXJjaGVycyBpbiBzb2NpYWwgc2NpZW5jZXMgdXNlIFIgZm9yIGRhdGEgYW5hbHlzaXMsIHN1cnZleSBhbmFseXNpcywgYW5kIHN0YXRpc3RpY2FsIG1vZGVsaW5nLiBUaGUgYWJpbGl0eSB0byBtYW5hZ2UgYW5kIGFuYWx5emUgbGFyZ2UgZGF0YXNldHMsIGNvbWJpbmVkIHdpdGggUidzIGV4dGVuc2l2ZSBzdGF0aXN0aWNhbCBmdW5jdGlvbnMsIG1ha2VzIGl0IGEgdmFsdWFibGUgdG9vbCBmb3IgY29uZHVjdGluZyByZXNlYXJjaCBpbiBzb2Npb2xvZ3ksIHBzeWNob2xvZ3ksIGVjb25vbWljcywgYW5kIHBvbGl0aWNhbCBzY2llbmNlLgoKIyMjIFJlcG9ydGluZyBhbmQgUmVwcm9kdWNpYmxlIFJlc2VhcmNoCgpSIGZhY2lsaXRhdGVzIHRoZSBjcmVhdGlvbiBvZiByZXByb2R1Y2libGUgcmVwb3J0cyBhbmQgZHluYW1pYyBkb2N1bWVudHMgdGhyb3VnaCBwYWNrYWdlcyBsaWtlIGBrbml0cmAgYW5kIGBybWFya2Rvd25gLiBUaGVzZSB0b29scyBhbGxvdyB1c2VycyB0byBpbnRlZ3JhdGUgUiBjb2RlIHdpdGggTWFya2Rvd24gb3IgTGFUZVgsIHByb2R1Y2luZyBkb2N1bWVudHMgdGhhdCBpbmNsdWRlIGJvdGggYW5hbHlzaXMgYW5kIG5hcnJhdGl2ZSB0ZXh0LiBUaGlzIGlzIHBhcnRpY3VsYXJseSB1c2VmdWwgaW4gYWNhZGVtaWMgcmVzZWFyY2ggYW5kIGluZHVzdHJ5IHJlcG9ydGluZywgZW5zdXJpbmcgdGhhdCBhbmFseXNlcyBhcmUgdHJhbnNwYXJlbnQgYW5kIHJlcHJvZHVjaWJsZS4KCkluIHN1bW1hcnksIFIncyB2ZXJzYXRpbGl0eSBhbmQgcG93ZXJmdWwgc3RhdGlzdGljYWwgY2FwYWJpbGl0aWVzIG1ha2UgaXQgYW4gaW52YWx1YWJsZSB0b29sIGFjcm9zcyB2YXJpb3VzIGRvbWFpbnMuIEZyb20gZGF0YSBhbmFseXNpcyBhbmQgdmlzdWFsaXphdGlvbiB0byBtYWNoaW5lIGxlYXJuaW5nIGFuZCBiaW9pbmZvcm1hdGljcywgUiBwcm92aWRlcyB0aGUgdG9vbHMgbmVjZXNzYXJ5IHRvIHBlcmZvcm0gc29waGlzdGljYXRlZCBhbmFseXNlcyBhbmQgcHJvZHVjZSBoaWdoLXF1YWxpdHkgcmVzdWx0cy4gVW5kZXJzdGFuZGluZyB0aGVzZSBjb21tb24gdXNlIGNhc2VzIHdpbGwgaGVscCB5b3UgbGV2ZXJhZ2UgUiBlZmZlY3RpdmVseSBpbiB5b3VyIHNwZWNpZmljIGZpZWxkIG9mIHN0dWR5IG9yIHdvcmsuCgojIyBJbnRlZ3JhdGVkIFByb2dyYW1taW5nIEVudmlyb25tZW50cyBmb3IgUgoKV2hpbGUgeW91IHJlYWxseSBuZWVkIG5vdGhpbmcgbW9yZSB0aGFuIGEgc2ltcGxlIHRleHQgZWRpdG9yLCBzdWNoIGFzIFN1YmxpbWUsIGFsdGhvdWdoIGV2ZW4gTm90ZXBhZCBvciBUZXh0RWRpdCB3b3VsZCBiZSBmaW5lLCB0byB3cml0ZSBSIHNjcmlwdHMgKCpha2EqIHByb2dyYW1zKSwgbW9zdCBkZXZlbG9wZXJzIGxldmVyYWdlIGFuIGludGVncmF0ZWQgZGV2ZWxvcG1lbnQgZW52aXJvbm1lbnQgKElERSkgdGhhdCBvZmZlcnMgc3ludGF4LWF3YXJlIGVkaXRvcnMsIGNvZGUgY29tcGxldGlvbiwgQUkgYXNzaXN0YW50cywgZ2l0aHViIGludGVncmF0aW9uLCBzb3VyY2UgY29kZSBjb250cm9sLCBwYWNrYWdlIGNvbnRyb2wsIGFuZCBzb3VyY2UgZmlsZSBtYW5hZ2VtZW50IHdpdGggcHJvamVjdHMuCgpUaGUgYmFzZSBsYW5ndWFnZSBlbnZpcm9ubWVudCBvZiBSIGNhbiBiZSBmcmVlbHkgb2J0YWluZWQgZm9yIFdpbmRvd3MsIE1hY09TLCBhbmQgdmFyaW91cyB2ZXJzaW9uIG9mIExpbnV4IGZyb20gW3ItcHJvamVjdC5vcmddKGh0dHBzOi8vY3Jhbi5yLXByb2plY3Qub3JnL21pcnJvcnMuaHRtbCkuCgpGb3IgUiwgdGhlcmUgYXJlIHNldmVyYWwgcG9wdWxhciBJREVzIGFuZCB0b29scyB0aGF0IGVuaGFuY2UgdGhlIHByb2dyYW1taW5nIGV4cGVyaWVuY2UgYnkgb2ZmZXJpbmcgZmVhdHVyZXMgbGlrZSBzeW50YXggaGlnaGxpZ2h0aW5nLCBjb2RlIGNvbXBsZXRpb24sIGRlYnVnZ2luZywgYW5kIGRpcmVjdCBleGVjdXRpb24gb2YgY29kZS4gSGVyZSBhcmUgc29tZSBvZiB0aGUgbW9zdCBjb21tb25seSB1c2VkIGRldmVsb3BtZW50IGVudmlyb25tZW50cyBmb3IgUjoKCiMjIyBQb3NpdCAoUlN0dWRpbykKClBvc2l0IChmb3JtZXJseSBSU3R1ZGlvKSwgaXMgYXJndWFibHkgdGhlIG1vc3QgcG9wdWxhciBhbmQgd2lkZWx5IHVzZWQgSURFIGZvciBSLiBJdCBpcyBhdmFpbGFibGUgaW4gYm90aCBkZXNrdG9wIGFuZCBzZXJ2ZXIgdmVyc2lvbnMgYW5kIG9mZmVycyBhIHVzZXItZnJpZW5kbHkgaW50ZXJmYWNlIHdpdGggbWFueSBmZWF0dXJlcyBkZXNpZ25lZCB0byBpbXByb3ZlIHByb2R1Y3Rpdml0eSBhbmQgZWFzZSBvZiB1c2UuCgotICAgKipLZXkgRmVhdHVyZXMqKjoKICAgIC0gICBTeW50YXggaGlnaGxpZ2h0aW5nIGFuZCBjb2RlIGNvbXBsZXRpb24uCiAgICAtICAgSW50ZWdyYXRlZCBSIGhlbHAgYW5kIGRvY3VtZW50YXRpb24uCiAgICAtICAgSW50ZXJhY3RpdmUgZ3JhcGhpY3Mgd2l0aCBzdXBwb3J0IGZvciBtdWx0aXBsZSBwbG90cy4KICAgIC0gICBEZWJ1Z2dpbmcgYW5kIHByb2ZpbGluZyB0b29scy4KICAgIC0gICBWZXJzaW9uIGNvbnRyb2wgaW50ZWdyYXRpb24gKEdpdCBhbmQgU1ZOKS4KICAgIC0gICBQcm9qZWN0IG1hbmFnZW1lbnQgY2FwYWJpbGl0aWVzLgogICAgLSAgIFN1cHBvcnQgZm9yIFIgTWFya2Rvd24gYW5kIGR5bmFtaWMgcmVwb3J0IGdlbmVyYXRpb24uCi0gICAqKlVzYWdlKio6IFBvc2l0IGlzIHN1aXRhYmxlIGZvciBhbGwgbGV2ZWxzIG9mIFIgdXNlcnMsIGZyb20gYmVnaW5uZXJzIHRvIGFkdmFuY2VkIHByb2dyYW1tZXJzLiBJdCBpcyBleHRlbnNpdmVseSB1c2VkIGluIGFjYWRlbWljIHNldHRpbmdzLCBkYXRhIGFuYWx5c2lzLCByZXNlYXJjaCBlbnZpcm9ubWVudHMsIGFuZCBmb3IgcHJvZHVjdGlvbi4gSXQgc3VwcG9ydHMgdGhlIGRldmVsb3BtZW50IG9mIFIgc2NyaXB0cywgbWFya2Rvd24gZG9jdW1lbnRzLCBwcmVzZW50YXRpb25zLCBhbmQgUiBOb3RlYm9va3MuIFBvc2l0IHN1cHBvcnRzIG5vdCBqdXN0IFIsIGJ1dCBhbHNvIFB5dGhvbiwgU1FMLCBEMywgYXMgd2VsbCBhcyBKYXZhIGFuZCBDKysgYWxsb3dpbmcgbXVsdGktbGFuZ3VhZ2UgcHJvZ3JhbW1pbmcuCgpQb3NpdCAoZm9ybWVybHkgUlN0dWRpbykgY2FuIGJlIG9idGFpbmVkIGZvciBmcmVlIGZyb20gW3Bvc2l0LmNvXShodHRwczovL3Bvc2l0LmNvL3Byb2R1Y3RzL29wZW4tc291cmNlL3JzdHVkaW8vKS4gUiBtdXN0IGJlIGluc3RhbGxlZCBwcmlvciB0byBpbnN0YWxsaW5nIFBvc2l0LgoKQXMgYW4gYWx0ZXJuYXRpdmUgdG8gaW5zdGFsbGluZyBSIGFuZCBQb3NpdCBsb2NhbGx5LCBhIGNsb3VkIGhvc3RlZCB2ZXJzaW9uIGlzIHByb3ZpZGVkIG9uIFtwb3NpdC5jbG91ZF0oaHR0cDovL3Bvc2l0LmNsb3VkKS4gQSBmcmVlIHBsYW4gaXMgYXZhaWxhYmxlLCBhbG9uZyB3aXRoIHZhcmlvdXMgc3Vic2NyaXB0aW9ucyBwbGFucyBpbmNsdWRpbmcgb25lIGZvciBlZHVjYXRpb24uIFtwb3NpdC5jbG91ZF0oaHR0cDovL3Bvc2l0LmNsb3VkKSBzdXBwb3J0cyBib3RoIFBvc2l0IGFzIHdlbGwgYXMgSnVweXRlciBOb3RlYm9vayBmb3IgaW50ZWdyYXRlZCBSIGFuZCBQeXRob24uCgojIyMgSnVweXRlciBOb3RlYm9va3MKCkp1cHl0ZXIgTm90ZWJvb2tzIGFyZSBhbiBpbmNyZWFzaW5nbHkgcG9wdWxhciBjaG9pY2UgZm9yIGRhdGEgc2NpZW50aXN0cyBhbmQgcmVzZWFyY2hlcnMgZHVlIHRvIHRoZWlyIGludGVyYWN0aXZlIGFuZCBsaXRlcmF0ZSBwcm9ncmFtbWluZyBhcHByb2FjaC4gV2hpbGUgSnVweXRlciBpcyBsYW5ndWFnZS1hZ25vc3RpYywgaXQgc3VwcG9ydHMgUiB0aHJvdWdoIHRoZSBJUktlcm5lbCBhbG9uZyB3aXRoIFB5dGhvbiAtLSBKdXB5dGVyIE5vdGVib29rcyBhcmUgZXF1aXZhbGVudCB0byBSIE5vdGVib29rcyBhbmQgdGhlIG5ldyBRdWFydG8gRG9jdW1lbnRzLgoKLSAgICoqS2V5IEZlYXR1cmVzKio6CiAgICAtICAgSW50ZXJhY3RpdmUsIHdlYi1iYXNlZCBub3RlYm9vayBlbnZpcm9ubWVudC4KICAgIC0gICBTdXBwb3J0IGZvciByaWNoIG1lZGlhIG91dHB1dCAocGxvdHMsIGltYWdlcywgdmlkZW9zKS4KICAgIC0gICBJbi1saW5lIGNvZGUgZXhlY3V0aW9uIHdpdGggaW1tZWRpYXRlIGZlZWRiYWNrLgogICAgLSAgIEludGVncmF0aW9uIHdpdGggdmFyaW91cyBkYXRhIHNjaWVuY2UgbGlicmFyaWVzLgogICAgLSAgIFNoYXJlYWJsZSBub3RlYm9va3Mgd2l0aCBuYXJyYXRpdmUgdGV4dCwgY29kZSwgYW5kIG91dHB1dC4KLSAgICoqVXNhZ2UqKjogSnVweXRlciBOb3RlYm9va3MgYXJlIGlkZWFsIGZvciBleHBsb3JhdG9yeSBkYXRhIGFuYWx5c2lzLCB0ZWFjaGluZywgYW5kIGNyZWF0aW5nIHJlcHJvZHVjaWJsZSByZXNlYXJjaCBkb2N1bWVudHMuCgojIyMgVmlzdWFsIFN0dWRpbyBDb2RlIChWUyBDb2RlKQoKVmlzdWFsIFN0dWRpbyBDb2RlIGlzIGEgbGlnaHR3ZWlnaHQsIHBvd2VyZnVsIGNvZGUgZWRpdG9yIGRldmVsb3BlZCBieSBNaWNyb3NvZnQuIFdpdGggdGhlIGFwcHJvcHJpYXRlIGV4dGVuc2lvbnMsIFZTIENvZGUgY2FuIGJlIHRyYW5zZm9ybWVkIGludG8gYSByb2J1c3QgSURFIGZvciBSLgoKLSAgICoqS2V5IEZlYXR1cmVzKio6CiAgICAtICAgU3ludGF4IGhpZ2hsaWdodGluZyBhbmQgY29kZSBjb21wbGV0aW9uLgogICAgLSAgIEludGVncmF0ZWQgdGVybWluYWwgYW5kIGRlYnVnZ2VyLgogICAgLSAgIEV4dGVuc2l2ZSBleHRlbnNpb25zIG1hcmtldHBsYWNlIChlLmcuLCBSIEV4dGVuc2lvbiBmb3IgVmlzdWFsIFN0dWRpbyBDb2RlKS4KICAgIC0gICBHaXQgaW50ZWdyYXRpb24gZm9yIHZlcnNpb24gY29udHJvbC4KICAgIC0gICBTdXBwb3J0IGZvciBtdWx0aXBsZSBwcm9ncmFtbWluZyBsYW5ndWFnZXMuCi0gICAqKlVzYWdlKio6IFZTIENvZGUgaXMgc3VpdGFibGUgZm9yIGRldmVsb3BlcnMgd2hvIHByZWZlciBhIHZlcnNhdGlsZSBjb2RlIGVkaXRvciB0aGF0IGNhbiBiZSBjdXN0b21pemVkIGZvciB2YXJpb3VzIHByb2dyYW1taW5nIG5lZWRzLCBpbmNsdWRpbmcgUiBkZXZlbG9wbWVudC4KCiMjIyBFbWFjcyB3aXRoIEVTUyAoRW1hY3MgU3BlYWtzIFN0YXRpc3RpY3MpCgpFbWFjcywgY29tYmluZWQgd2l0aCB0aGUgRVNTIChFbWFjcyBTcGVha3MgU3RhdGlzdGljcykgcGFja2FnZSwgcHJvdmlkZXMgYSBwb3dlcmZ1bCBlbnZpcm9ubWVudCBmb3IgUiBwcm9ncmFtbWluZy4gRVNTIGlzIGFuIGFkZC1vbiBwYWNrYWdlIHRoYXQgZW5oYW5jZXMgRW1hY3MgdG8gc3VwcG9ydCBpbnRlcmFjdGl2ZSBzdGF0aXN0aWNhbCBwcm9ncmFtbWluZyBhbmQgZGF0YSBhbmFseXNpcy4KCi0gICAqKktleSBGZWF0dXJlcyoqOgogICAgLSAgIEFkdmFuY2VkIHRleHQgZWRpdGluZyBjYXBhYmlsaXRpZXMuCiAgICAtICAgU3ludGF4IGhpZ2hsaWdodGluZyBhbmQgY29kZSBjb21wbGV0aW9uLgogICAgLSAgIERpcmVjdCBpbnRlcmFjdGlvbiB3aXRoIHRoZSBSIHByb2Nlc3MuCiAgICAtICAgRGVidWdnaW5nIGFuZCBwcm9maWxpbmcgdG9vbHMuCiAgICAtICAgSW50ZWdyYXRpb24gd2l0aCBvdGhlciBzdGF0aXN0aWNhbCBwYWNrYWdlcyAoZS5nLiwgU0FTLCBTdGF0YSkuCi0gICAqKlVzYWdlKio6IEVtYWNzIHdpdGggRVNTIGlzIGZhdm9yZWQgYnkgdXNlcnMgd2hvIGFyZSBjb21mb3J0YWJsZSB3aXRoIEVtYWNzIGFuZCBzZWVrIGEgaGlnaGx5IGN1c3RvbWl6YWJsZSBhbmQgZXh0ZW5zaWJsZSBwcm9ncmFtbWluZyBlbnZpcm9ubWVudC4KCiMjIyBSR3VpIChSIEdyYXBoaWNhbCBVc2VyIEludGVyZmFjZSkKClJHdWkgaXMgdGhlIHN0YW5kYXJkIGdyYXBoaWNhbCB1c2VyIGludGVyZmFjZSB0aGF0IGNvbWVzIHdpdGggdGhlIGJhc2UgUiBpbnN0YWxsYXRpb24uIEl0IHByb3ZpZGVzIGEgYmFzaWMgZW52aXJvbm1lbnQgZm9yIHdyaXRpbmcgYW5kIGV4ZWN1dGluZyBSIGNvZGUuCgotICAgKipLZXkgRmVhdHVyZXMqKjoKICAgIC0gICBTaW1wbGUgYW5kIGxpZ2h0d2VpZ2h0LgogICAgLSAgIEludGVncmF0ZWQgUiBjb25zb2xlLgogICAgLSAgIEJhc2ljIHNjcmlwdCBlZGl0b3IuCi0gICAqKlVzYWdlKio6IFJHdWkgaXMgc3VpdGFibGUgZm9yIHF1aWNrLCBzaW1wbGUgdGFza3MgYW5kIGlzIG9mdGVuIHVzZWQgYnkgYmVnaW5uZXJzIHdobyBhcmUganVzdCBzdGFydGluZyB3aXRoIFIuCgojIyBSIGFzIGEgUHJvZ3JhbW1pbmcgTGFuZ3VhZ2UKClIgaXMgYSBoaWdoLWxldmVsIGxhbmd1YWdlIHRoYXQgcHJpbWFyaWx5IGZvbGxvd3MgdGhlIHBhcmFkaWdtIG9mICpmdW5jdGlvbmFsIHByb2dyYW1taW5nKiBidXQgYWxzbyBzdXBwb3J0cyAqcHJvY2VkdXJhbCogYW5kICpvYmplY3Qtb3JpZW50ZWQgcHJvZ3JhbW1pbmcqIHN0eWxlcy4gSXRzIGZsZXhpYmlsaXR5IGluIGFjY29tbW9kYXRpbmcgZGlmZmVyZW50IHByb2dyYW1taW5nIHBhcmFkaWdtcyBtYWtlcyBpdCBhIHZlcnNhdGlsZSBsYW5ndWFnZSBmb3IgYSB3aWRlIHJhbmdlIG9mIGNvbXB1dGF0aW9uYWwgdGFza3MuIEhvd2V2ZXIsIHRoZSBwcm9ncmFtbWluZyBtZWNoYW5pc21zIGZvciBsYXJnZS1zY2FsZSBzb2Z0d2FyZSBlbmdpbmVlcmluZyBhcmUgbm90IGFzIHJvYnVzdCBhcyB0aG9zZSBpbiBKYXZhIChvciBldmVuIFB5dGhvbikgYW5kIHRoZXJlZm9yZSBpdCBpcyBub3QgcmVjb21tZW5kZWQgdG8gYnVpbGQgbGFyZ2UgYXBwbGljYXRpb25zIGluIFIuIEhvd2V2ZXIsIFIgaXMgYW4gaWRlYWwgbGFuZ3VhZ2UgdG8gZXhwZXJpbWVudGF0aW9uLCBleHBsb3JhdGlvbiwgcHJvb2Ytb2YtY29uY2VwdCBkZXZlbG9wbWVudCwgcmVzZWFyY2gsIGFuZCBwcm90b3R5cGUgZGV2ZWxvcG1lbnQuIEl0IGlzIGFsc28gaWRlYWwgZm9yIHF1aWNrIHByb2dyYW1taW5nIHRhc2tzLgoKIyMjIEZ1bmN0aW9uYWwgUHJvZ3JhbW1pbmcgaW4gUgoKRnVuY3Rpb25hbCBwcm9ncmFtbWluZyBpcyBhdCB0aGUgY29yZSBvZiBSJ3MgZGVzaWduLiBJbiB0aGlzIHBhcmFkaWdtLCBjb21wdXRhdGlvbiBpcyB0cmVhdGVkIGFzIHRoZSBldmFsdWF0aW9uIG9mIG1hdGhlbWF0aWNhbCBmdW5jdGlvbnMgYW5kIGF2b2lkcyBjaGFuZ2luZy1zdGF0ZSBhbmQgbXV0YWJsZSBkYXRhLiBSJ3MgZnVuY3Rpb25zIGFyZSBmaXJzdC1jbGFzcyBvYmplY3RzLCBtZWFuaW5nIHRoZXkgY2FuIGJlIHBhc3NlZCBhcyBhcmd1bWVudHMgdG8gb3RoZXIgZnVuY3Rpb25zLCByZXR1cm5lZCBhcyB2YWx1ZXMgZnJvbSBvdGhlciBmdW5jdGlvbnMsIGFuZCBhc3NpZ25lZCB0byB2YXJpYWJsZXMuCgpGb3IgZXhhbXBsZSwgY29uc2lkZXIgdGhlIHVzZSBvZiB0aGUgYGxhcHBseWAgZnVuY3Rpb24sIHdoaWNoIGFwcGxpZXMgYSBmdW5jdGlvbiB0byBlYWNoIGVsZW1lbnQgb2YgYSBsaXN0IGFuZCByZXR1cm5zIGEgbGlzdCBvZiByZXN1bHRzOgoKYGBgIHIKIyBEZWZpbmUgYSBzaW1wbGUgZnVuY3Rpb24gdG8gc3F1YXJlIGEgbnVtYmVyCnNxdWFyZSA8LSBmdW5jdGlvbih4KSB7CiAgcmV0dXJuKHggKiB4KQp9CgojIENyZWF0ZSBhIGxpc3Qgb2YgbnVtYmVycwpudW1iZXJzIDwtIGxpc3QoMSwgMiwgMywgNCwgNSkKCiMgQXBwbHkgdGhlIHNxdWFyZSBmdW5jdGlvbiB0byBlYWNoIGVsZW1lbnQgb2YgdGhlIGxpc3QKc3F1YXJlZF9udW1iZXJzIDwtIGxhcHBseShudW1iZXJzLCBzcXVhcmUpCgpwcmludChzcXVhcmVkX251bWJlcnMpCmBgYAoKIyMjIFByb2NlZHVyYWwgUHJvZ3JhbW1pbmcgaW4gUgoKUiBhbHNvIHN1cHBvcnRzIHByb2NlZHVyYWwgcHJvZ3JhbW1pbmcsIHdoZXJlIHlvdSB3cml0ZSBzZXF1ZW5jZXMgb2YgaW5zdHJ1Y3Rpb25zIHRvIHBlcmZvcm0gdGFza3MuIFRoaXMgYXBwcm9hY2ggaXMgdXNlZnVsIGZvciB0YXNrcyB0aGF0IGludm9sdmUgbG9vcGluZywgY29uZGl0aW9uYWxzLCBhbmQgc3RlcC1ieS1zdGVwIGNvbXB1dGF0aW9ucy4gSGVyZeKAmXMgYW4gZXhhbXBsZSB1c2luZyBhIGZvciBsb29wIHRvIGFjaGlldmUgdGhlIHNhbWUgcmVzdWx0IGFzIHRoZSBwcmV2aW91cyBmdW5jdGlvbmFsIGV4YW1wbGU6CgpgYGAgcgojIENyZWF0ZSBhIGxpc3Qgb2YgbnVtYmVycwpudW1iZXJzIDwtIGxpc3QoMSwgMiwgMywgNCwgNSkKCiMgSW5pdGlhbGl6ZSBhbiBlbXB0eSBsaXN0IHRvIHN0b3JlIHNxdWFyZWQgbnVtYmVycwpzcXVhcmVkX251bWJlcnMgPC0gbGlzdCgpCgojIFVzZSBhIGZvciBsb29wIHRvIHNxdWFyZSBlYWNoIG51bWJlcgpmb3IgKGkgaW4gMTpsZW5ndGgobnVtYmVycykpIHsKICBzcXVhcmVkX251bWJlcnNbW2ldXSA8LSBudW1iZXJzW1tpXV0gKiBudW1iZXJzW1tpXV0KfQoKcHJpbnQoc3F1YXJlZF9udW1iZXJzKQpgYGAKCiMjIyBPYmplY3QtT3JpZW50ZWQgUHJvZ3JhbW1pbmcgaW4gUgoKUiBhbHNvIHN1cHBvcnRzIG9iamVjdC1vcmllbnRlZCBwcm9ncmFtbWluZyAoT09QKSB0aHJvdWdoIHR3byBzeXN0ZW1zOiBTMyBhbmQgUzQuIFRoZSBTMyBzeXN0ZW0gaXMgbW9yZSBpbmZvcm1hbCBhbmQgcmVsaWVzIG9uIGdlbmVyaWMgZnVuY3Rpb25zIGFuZCBtZXRob2QgZGlzcGF0Y2ggYmFzZWQgb24gb2JqZWN0IGNsYXNzLiBUaGUgUzQgc3lzdGVtIGlzIG1vcmUgZm9ybWFsLCB3aXRoIGV4cGxpY2l0IGNsYXNzIGFuZCBtZXRob2QgZGVmaW5pdGlvbnMuCgpIZXJl4oCZcyBhIHNpbXBsZSBleGFtcGxlIHVzaW5nIHRoZSBTMyBzeXN0ZW06CgpgYGAgcgojIERlZmluZSBhIHNpbXBsZSBTMyBjbGFzcyBmb3IgYSBwb2ludApwb2ludCA8LSBmdW5jdGlvbih4LCB5KSB7CiAgc3RydWN0dXJlKGxpc3QoeCA9IHgsIHkgPSB5KSwgY2xhc3MgPSAicG9pbnQiKQp9CgojIERlZmluZSBhIG1ldGhvZCB0byBwcmludCBhIHBvaW50IG9iamVjdApwcmludC5wb2ludCA8LSBmdW5jdGlvbihwKSB7CiAgY2F0KCJQb2ludCgiLCBwJHgsICIsICIsIHAkeSwgIilcbiIsIHNlcCA9ICIiKQp9CgojIENyZWF0ZSBhIHBvaW50IG9iamVjdApwIDwtIHBvaW50KDIsIDMpCgojIFByaW50IHRoZSBwb2ludCBvYmplY3QKcHJpbnQocCkKYGBgCgojIyMgTGl0ZXJhdGUgUHJvZ3JhbW1pbmcKCkxpdGVyYXRlIHByb2dyYW1taW5nIGlzIGEgcHJvZ3JhbW1pbmcgcGFyYWRpZ20gaW50cm9kdWNlZCBieSBEb25hbGQgS251dGggaW4gdGhlIGVhcmx5IDE5ODBzLiBUaGUgY29yZSBpZGVhIGJlaGluZCBsaXRlcmF0ZSBwcm9ncmFtbWluZyBpcyB0byB3cml0ZSBwcm9ncmFtcyB0aGF0IGFyZSB1bmRlcnN0YW5kYWJsZSBieSBodW1hbnMgZmlyc3QgYW5kIGNvbXB1dGVycyBzZWNvbmQuIFRoaXMgaXMgYWNoaWV2ZWQgYnkgaW50ZXJzcGVyc2luZyBuYXR1cmFsIGxhbmd1YWdlIGV4cGxhbmF0aW9ucyB3aXRoIHNvdXJjZSBjb2RlLCBjcmVhdGluZyBhIGRvY3VtZW50IHRoYXQgY2FuIGJlIHJlYWQgYW5kIHVuZGVyc3Rvb2QgbGlrZSBhIGJvb2sgb3IgYW4gYXJ0aWNsZS4gTGl0ZXJhdGUgcHJvZ3JhbW1pbmcgZW1waGFzaXplcyB0aGUgaW1wb3J0YW5jZSBvZiBjbGVhciBjb21tdW5pY2F0aW9uIG9mIGlkZWFzIGFuZCBsb2dpYywgbWFraW5nIHRoZSBjb2RlIG1vcmUgbWFpbnRhaW5hYmxlIGFuZCBlYXNpZXIgdG8gdW5kZXJzdGFuZC4KCkluIGxpdGVyYXRlIHByb2dyYW1taW5nLCB0aGUgZW1waGFzaXMgaXMgb24gZXhwbGFpbmluZyB0aGUgbG9naWMgYW5kIHJhdGlvbmFsZSBiZWhpbmQgdGhlIGNvZGUgaW4gYSB3YXkgdGhhdCBpcyBhY2Nlc3NpYmxlIHRvIGh1bWFucy4gVGhlIGdvYWwgaXMgdG8gcHJvZHVjZSBhIGRvY3VtZW50IHRoYXQgaXMgYXMgbXVjaCBhYm91dCBjb252ZXlpbmcgaWRlYXMgYXMgaXQgaXMgYWJvdXQgd3JpdGluZyBleGVjdXRhYmxlIGNvZGUuCgpUaGUgZG9jdW1lbnQgYWx0ZXJuYXRlcyBiZXR3ZWVuIHNlY3Rpb25zIG9mIGV4cGxhbmF0b3J5IHRleHQgYW5kIHNlY3Rpb25zIG9mIGNvZGUuIFRoZSB0ZXh0IGV4cGxhaW5zIHdoYXQgdGhlIGNvZGUgZG9lcywgd2h5IGNlcnRhaW4gZGVjaXNpb25zIHdlcmUgbWFkZSwgYW5kIGhvdyBkaWZmZXJlbnQgcGFydHMgb2YgdGhlIGNvZGUgaW50ZXJhY3QuIFRoaXMgYXBwcm9hY2ggaGVscHMgcmVhZGVycyBmb2xsb3cgdGhlIHRob3VnaHQgcHJvY2VzcyBvZiB0aGUgcHJvZ3JhbW1lciwgbWFraW5nIGl0IGVhc2llciB0byB1bmRlcnN0YW5kIGNvbXBsZXggYWxnb3JpdGhtcyBhbmQgc3lzdGVtcy4KClRoZSBzdHJ1Y3R1cmUgb2YgYSBsaXRlcmF0ZSBwcm9ncmFtIGZvbGxvd3MgYSBsb2dpY2FsIG5hcnJhdGl2ZSBmbG93IHJhdGhlciB0aGFuIHRoZSBzdHJpY3Qgc3ludGFjdGljYWwgcmVxdWlyZW1lbnRzIG9mIGEgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2UuIFRoaXMgbWVhbnMgdGhhdCB0aGUgb3JkZXIgaW4gd2hpY2ggdGhlIGNvZGUgaXMgcHJlc2VudGVkIGluIHRoZSBkb2N1bWVudCBjYW4gZGlmZmVyIGZyb20gdGhlIG9yZGVyIGluIHdoaWNoIGl0IGlzIGV4ZWN1dGVkLiBUb29scyB1c2VkIGluIGxpdGVyYXRlIHByb2dyYW1taW5nIHdpbGwgbGF0ZXIgcmVhcnJhbmdlIHRoZSBjb2RlIGludG8gYSBmb3JtIHRoYXQgY2FuIGJlIGNvbXBpbGVkIG9yIGludGVycHJldGVkIGJ5IHRoZSBjb21wdXRlci4KCkxpdGVyYXRlIHByb2dyYW1taW5nIHRvb2xzIGFsbG93IHRoZSBkb2N1bWVudCB0byBiZSBjb21waWxlZCBvciBpbnRlcnByZXRlZCwgZXhlY3V0aW5nIHRoZSBjb2RlIHNlZ21lbnRzIGVtYmVkZGVkIHdpdGhpbiB0aGUgdGV4dC4gVGhpcyBlbnN1cmVzIHRoYXQgdGhlIGRvY3VtZW50YXRpb24gaXMgYWx3YXlzIHN5bmNocm9uaXplZCB3aXRoIHRoZSBjb2RlIGFuZCBjYW4gYmUgcnVuIHRvIHZlcmlmeSBpdHMgY29ycmVjdG5lc3MuCgpSTWFya2Rvd24gYW5kIFIgTm90ZWJvb2tzIGFyZSB0aGUgdG9vbHMgaW4gdGhlIFIgZWNvc3lzdGVtIHRoYXQgc3VwcG9ydCBsaXRlcmF0ZSBwcm9ncmFtbWluZyBmb3IgZGF0YSBhbmFseXNpcyBhbmQgcmVwb3J0aW5nLiBMaXRlcmF0ZSBwcm9ncmFtbWluZyBpcyBhbHNvIHN1cHBvcnRlZCBpbiBQeXRob24gd2l0aCBKdXB5dGVyIE5vdGVib29rcy4gTm90ZSB0aGF0IFIgTm90ZWJvb2tzIGFuZCBKdXB5dGVyIE5vdGVib29rcyBjYW4gYWN0dWFsbHkgY29udGFpbiBhIG1peCBvZiBjb2RlIGJsb2NrcyBpbiBSLCBQeXRob24sIFNRTCwgSmF2YSwgQysrLCBEMywgYW5kIG1hbnkgb3RoZXJzIGFsbG93aW5nIG9uZSB0byB1c2UgdGhlIHByb2dyYW1taW5nIGxhbmd1YWdlIG1vc3Qgc3VpdGFibGUgZm9yIGEgcGFydGljdWxhciB0YXNrLiBUaGVyZSBpcyBhbHNvIHN1cHBvcnQgaW4gSGFza2VsbCBhbmQgUmFja2V0dCBmb3IgbGl0ZXJhdGUgcHJvZ3JhbW1pbmcsIHNvIGl0IGlzIG5vdCByZXN0cmljdGVkIHRvIFIgYW5kIFB5dGhvbi4KCiMjIyBFeHRlbmRpbmcgUiB3aXRoIFBhY2thZ2VzCgpPbmUgb2YgdGhlIGdyZWF0ZXN0IHN0cmVuZ3RocyBvZiBSIGlzIGl0cyBleHRlbnNpdmUgZWNvc3lzdGVtIG9mIHBhY2thZ2VzLiBQYWNrYWdlcyBpbiBSIGFyZSBjb2xsZWN0aW9ucyBvZiBSIGZ1bmN0aW9ucywgZGF0YSwgYW5kIGNvbXBpbGVkIGNvZGUgdGhhdCBleHRlbmQgdGhlIGNhcGFiaWxpdGllcyBvZiB0aGUgYmFzZSBSIHN5c3RlbS4gVGhlIENvbXByZWhlbnNpdmUgUiBBcmNoaXZlIE5ldHdvcmsgKENSQU4pIGlzIHRoZSBwcmltYXJ5IHJlcG9zaXRvcnkgZm9yIFIgcGFja2FnZXMsIGhvc3RpbmcgdGhvdXNhbmRzIG9mIHBhY2thZ2VzIHRoYXQgY292ZXIgYSB3aWRlIGFycmF5IG9mIGZ1bmN0aW9uYWxpdGllcy4gVGhlIGtleSBwYWNrYWdlcyBpbmNsdWRlOgoKMS4gICoqdGlkeXZlcnNlKio6IEEgY29sbGVjdGlvbiBvZiBwYWNrYWdlcyBkZXNpZ25lZCBmb3IgZGF0YSBzY2llbmNlLiBUaGlzIGluY2x1ZGVzIGBnZ3Bsb3QyYCBmb3IgZGF0YSB2aXN1YWxpemF0aW9uLCBgZHBseXJgIGZvciBkYXRhIG1hbmlwdWxhdGlvbiwgYHRpZHlyYCBmb3IgZGF0YSB0aWR5aW5nLCBhbmQgb3RoZXJzLgoKMi4gICoqZGF0YS50YWJsZSoqOiBBbiBleHRlbnNpb24gb2YgYGRhdGEuZnJhbWVgIHRoYXQgcHJvdmlkZXMgYW4gZW5oYW5jZWQgdmVyc2lvbiB3aXRoIGZhc3RlciBkYXRhIG1hbmlwdWxhdGlvbiBjYXBhYmlsaXRpZXMuCgozLiAgKipzaGlueSoqOiBGb3IgYnVpbGRpbmcgaW50ZXJhY3RpdmUgd2ViIGFwcGxpY2F0aW9ucyBkaXJlY3RseSBmcm9tIFIuCgo0LiAgKipjYXJldCoqOiBGb3IgbWFjaGluZSBsZWFybmluZyBhbmQgcHJlZGljdGl2ZSBtb2RlbGluZy4KCjUuICAqKmtuaXRyKio6IEZvciBkeW5hbWljIHJlcG9ydCBnZW5lcmF0aW9uLCBpbnRlZ3JhdGluZyBSIGNvZGUgd2l0aCBMYVRlWCwgSFRNTCwgYW5kIE1hcmtkb3duLgoKNi4gICoqUmNwcCoqOiBGb3Igc2VhbWxlc3MgaW50ZWdyYXRpb24gb2YgUiB3aXRoIEMrKyB0byBlbmhhbmNlIHBlcmZvcm1hbmNlLgoKNy4gICoqUlNRTGl0ZSoqOiBGb3IgYWNjZXNzIGFuZCBpbnRlZ3JhdGlvbiB3aXRoIFNRTGl0ZS4KCjguICAqKmdncGxvdDIqKjogRm9yIGNyZWF0aW5nIHJlcHJvZHVjaWJsZSB2aXN1YWxpemF0aW9ucy4KCjkuICAqKmthYmxlKio6IEZvciBjb25zdHJ1Y3Rpb24gb2YgdmlzdWFsbHkgYXBwZWFsaW5nIHRhYmxlcy4KCiMjIFNjcmlwdHMgdnMgTm90ZWJvb2tzCgpSIGNvZGUgY2FuIGJlIHdyaXR0ZW4gaW4gdHdvIGdlbmVyYWwgd2F5czogKDEpIGFzIExpdGVyYXRlIFByb2dyYW1zIHVzaW5nIFIgTWFya2Rvd24gTm90ZWJvb2tzIGFuZCAoMikgYXMgU2NyaXB0cy4gU2NyaXB0cyBhcmUgcHJvZ3JhbXMgdGhhdCBjYW4gcnVuIHdpdGhpbiBhbiBJREUgc3VjaCBhcyBSIFN0dWRpbyBvciBiZSBkaXJlY3RseSBleGVjdXRlZCBmcm9tIHRoZSBjb21tYW5kIGxpbmUgdGhyb3VnaCBSLiBTY3JpcHRzIGhhdmUgdGhlIGJlbmVmaXQgdGhhdCB0aGV5IGNhbiBiZSBkZWJ1Z2dlZCB1c2luZyBhIGRlYnVnZ2VyLCB3aGlsZSBSIE5vdGVib29rIGNvZGUgY2h1bmtzIGFyZSBtb3JlIGRpZmZpY3VsdCB0byBkZWJ1Zy4gQW5vdGhlciBiZW5lZml0IG9mIHNjcmlwdHMgaXMgdGhhdCB0aGV5IGNhbiBiZSBydW4gZnJvbSB0aGUgY29tbWFuZCBsaW5lIGFuZCBhcyBwYXJ0IG9mICpjcm9uKiBqb2JzLCAqaS5lLiosIHRoZXkgY2FuIGJlIHNjaGVkdWxlZCB0byBydW4gYXV0b21hdGljYWxseSBhdCBhIHBvaW50IGluIHRpbWUuIEZpbmFsbHksIFIgc2NyaXB0cyBjYW4gYmUgaW5jbHVkZWQgaW4gc2hlbGwgc2NyaXB0cyAob24gVW5peCBhbmQgTWFjT1MpIGFuZCAqLmJhdCogYmF0Y2ggcHJvZ3JhbXMgKG9uIFdpbmRvd3MpLgoKUiBOb3RlYm9va3MgYXJlIG1hcmtkb3duIGRvY3VtZW50cyB0aGF0IGNvbnRhaW4gZW1iZWRkZWQgY29kZSBpbiBSIGFuZCBvdGhlciBsYW5ndWFnZXMuIFRoZXkgYXJlIG1vc3QgdXNlZnVsIHdoZW4gcHJvZHVjaW5nIHJlcG9ydHMsIG1lbW9zLCBhbmQgYW5hbHl0aWNzIGpvdXJuYWxzIHdoZXJlIHdlIG5lZWQgdG8gaW50ZXJzcGVyc2UgbmFycmF0aXZlcyBhbmQgdGV4dCB3aXRoIGNvZGUuIEFuIFIgTm90ZWJvb2sgcHJvZHVjZXMgYSBkb2N1bWVudCAoSFRNTCBvciBQREYsIG1vc3QgY29tbW9ubHkpIHdoZW4gaXQgaXMgInJ1biIgKCpha2EqICJrbml0dGVkIikuIEFuIFIgU2NyaXB0LCBvbiB0aGUgb3RoZXIgaGFuZCwgaXMgYSBwcm9ncmFtIHRoYXQgcnVucyBsaWtlIGluIGFueSBvdGhlciBsYW5ndWFnZS4KClIgU2NyaXB0cyBhcmUgcHJlZmVyYWJsZSB3aGVuIHlvdSBuZWVkIHN0YW5kLWFsb25lIFIgcHJvZ3JhbXMgdGhhdCBjYW4gYmUgcnVuIGRpcmVjdGx5IGZyb20gdGhlIGNvbW1hbmQgbGluZSBvciB3aXRoaW4gYSBzaGVsbCBzY3JpcHQuIE9uZSBtYWluIGJlbmVmaXQgb2YgdXNpbmcgUiBTY3JpcHRzIGlzIHRoYXQgd2UgY2FuIHVzZSB0aGUgZGVidWdnZXIgd2l0aGluIFIgU3R1ZGlvLgoKIyMgU2VlIEFsc28KCi0gICBGb3IgbW9yZSBpbmZvcm1hdGlvbiBvbiBSIFNjcmlwdHMgKCpha2EqIFIgUHJvZ3JhbXMpOgogICAgLSAgIExlc3NvbiBbNi4xMDkgLS0gUiBTY3JpcHRzIGFuZCBQcm9ncmFtc10oaHR0cDovL2FydGlmaWNpdW0udXMvbGVzc29ucy8wNi5yL2wtNi0xMDktci1wcm9ncmFtcy9sLTYtMTA5Lmh0bWwpCi0gICBGb3IgbW9yZSBpbmZvcm1hdGlvbiBvbiBSIE5vdGVib29rcywgbWFya2Rvd24sIGVtYmVkZGVkIGNvZGUgY2h1bmtzLCBhbmQgaW5saW5lIGNvZGUgaW4gdGhlIGZvcm0gb2YgbGl0ZXJhdGUgcHJvZ3JhbXM6CiAgICAtICAgTGVzc29uIFs2LjIwNCAtLSBMaXRlcmF0ZSBQcm9ncmFtbWluZyB3aXRoIFIgTm90ZWJvb2tzXShodHRwOi8vYXJ0aWZpY2l1bS51cy9sZXNzb25zLzA2LnIvbC02LTIwNC1yLW5vdGVib29rcy9sLTYtMjA0Lmh0bWwpCi0gICBUbyBsZWFybiBob3cgdG8gY3JlYXRlIHByb2plY3RzIHdpdGhpbiBQb3NpdCAoUiBTdHVkaW8pOgogICAgLSAgIExlc3NvbiBbNi4yMDIg4pSGIFdvcmtpbmcgd2l0aCBSIFByb2plY3RzXShodHRwOi8vYXJ0aWZpY2l1bS51cy9sZXNzb25zLzA2LnIvbC02LTIwMi1yLXByb2plY3RzL2wtNi0yMDIuaHRtbCkKCiMjIENvbmNsdXNpb24KCkluIHRoaXMgbGVzc29uLCB3ZSBleHBsb3JlZCB0aGUgY29uY2VwdHMgb2YgbGl0ZXJhdGUgcHJvZ3JhbW1pbmcgYW5kIGl0cyBpbXBsZW1lbnRhdGlvbiB1c2luZyBSIE5vdGVib29rcy4gTGl0ZXJhdGUgcHJvZ3JhbW1pbmcsIGludHJvZHVjZWQgYnkgRG9uYWxkIEtudXRoLCBpbnRlZ3JhdGVzIG5hdHVyYWwgbGFuZ3VhZ2UgZXhwbGFuYXRpb25zIHdpdGggc291cmNlIGNvZGUsIG1ha2luZyBwcm9ncmFtcyBtb3JlIHVuZGVyc3RhbmRhYmxlIGFuZCBtYWludGFpbmFibGUgYnkgaHVtYW5zLiBSIE5vdGVib29rcyBleGVtcGxpZnkgdGhpcyBwYXJhZGlnbSBieSBjb21iaW5pbmcgbmFycmF0aXZlIHRleHQsIGNvZGUsIGFuZCBvdXRwdXQgaW4gYSBzaW5nbGUgZG9jdW1lbnQsIGVuaGFuY2luZyByZWFkYWJpbGl0eSBhbmQgcmVwcm9kdWNpYmlsaXR5LiBXZSBjb250cmFzdGVkIFIgTm90ZWJvb2tzIGZvciBkeW5hbWljIGRvY3VtZW50IGdlbmVyYXRpb24gd2l0aCBSIFNjcmlwdHMgdGhhdCBhcmUgc3RhbmQtYWxvbmcgcHJvZ3JhbXMgdGhhdCBjYW4gYmUgcnVuIGZyb20gdGhlIGNvbW1hbmQgbGluZSBvciBlbWJlZGRlZCB3aXRoaW4gc2hlbGwgc2NyaXB0cy4KCldlIGFsc28gY29tcGFyZWQgUiBhbmQgUHl0aG9uLCBoaWdobGlnaHRpbmcgdGhhdCB3aGlsZSBib3RoIGFyZSBwb3dlcmZ1bCB0b29scyBmb3IgZGF0YSBhbmFseXNpcyBhbmQgbWFjaGluZSBsZWFybmluZywgdGhleSBkaWZmZXIgaW4gc3ludGF4LCBsaWJyYXJpZXMsIGFuZCB1c2UgY2FzZXMuIFIgaXMgZmF2b3JlZCBmb3Igc3RhdGlzdGljYWwgYW5hbHlzaXMsIGRhdGEgYW5hbHl0aWNzIHJlcG9ydGluZyBhbmQgdmlzdWFsaXphdGlvbiwgd2hlcmVhcyBQeXRob24gaXMgbW9yZSB2ZXJzYXRpbGUsIHN1aXRhYmxlIGZvciBnZW5lcmFsLXB1cnBvc2UgcHJvZ3JhbW1pbmcgYW5kIGludGVncmF0aW9uIGludG8gcHJvZHVjdGlvbiBlbnZpcm9ubWVudHMuIFB5dGhvbidzIGJyb2FkZXIgdXNlciBiYXNlIGFuZCBleHRlbnNpdmUgbGlicmFyaWVzIG1ha2UgaXQgbW9yZSBwb3B1bGFyIG92ZXJhbGwgd2hpbGUgUiBoYXMgYSBzdHJvbmcgZm9sbG93aW5nIGluIHJlc2VhcmNoLiBBbiBlZHVjYXRlZCBjb21wdXRlciBhbmQgZGF0YSBzY2llbnRpc3QgbXVzdCBrbm93IGJvdGggbGFuZ3VhZ2VzLgoKQWRkaXRpb25hbGx5LCB3ZSBkaXNjdXNzZWQgY29tbW9uIGRldmVsb3BtZW50IGVudmlyb25tZW50cyBmb3IgUiwgaW5jbHVkaW5nIFJTdHVkaW8vUG9zaXQuCgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMjIEZpbGVzICYgUmVzb3VyY2VzCgpgYGB7ciB6aXBGaWxlcywgZWNobz1GQUxTRX0KemlwTmFtZSA9IHNwcmludGYoIkxlc3NvbkZpbGVzLSVzLSVzLnppcCIsIAogICAgICAgICAgICAgICAgIHBhcmFtcyRjYXRlZ29yeSwKICAgICAgICAgICAgICAgICBwYXJhbXMkbnVtYmVyKQoKdGV4dEFMaW5rID0gcGFzdGUwKCJBbGwgRmlsZXMgZm9yIExlc3NvbiAiLCAKICAgICAgICAgICAgICAgcGFyYW1zJGNhdGVnb3J5LCIuIixwYXJhbXMkbnVtYmVyKQoKIyBkb3dubG9hZEZpbGVzTGluaygpIGlzIGluY2x1ZGVkIGZyb20gX2luc2VydDJEQi5SCmtuaXRyOjpyYXdfaHRtbChkb3dubG9hZEZpbGVzTGluaygiLiIsIHppcE5hbWUsIHRleHRBTGluaykpCmBgYAoKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgojIyBSZWZlcmVuY2VzCgpOb25lLgoKIyMgRXJyYXRhCgpbTGV0IHVzIGtub3ddKGh0dHBzOi8vZm9ybS5qb3Rmb3JtLmNvbS8yMTIxODcwNzI3ODQxNTcpe3RhcmdldD0iX2JsYW5rIn0uCg==