Introduction

This lesson explains the importance of XML as a data-interchange language and shows how XML documents are constructed using tags, hierarchy, and grammar. Introduces the Document Type Definition (DTD) language for defining the grammar (structure) of an XML document.

XML in a Nutshell

XML is a markup language that uses elements and attributes to create a text file of structured data. XML is a common data format used by many organizations, systems, and the web to exchange data. Often, data needed for analytics resides in an XML and must be imported. Consequently, data analytics professionals should know how to read (parse) XML and prepare the data for import into a data mining effort or a data warehouse.

One hallmark of XML is that the tags that create elements are created by the person or system that creates the file. As such, there is no specific definition or explanation of the tags and the elements they create. This increased flexibility for use of the files but can make it more difficult to exchange data between systems effectively. For this reason, we use a DTD to describe the tags that are used the the order and quantity in which they must appear in the XML..

A DTD, or Document Type Definition, defines the structure of an XML document (or file). It describes the elements and attributes used in the file and helps us to understand and agree on the contents. It also is possible for special parsers to read the DTD to ensure that the structure of an XML file is valid according to the DTD. Since we are working with large data sets, we cannot check them ourselves; a DTD, along with the restrictions in the XML language, help to ensure that the data is structured correctly and thus can be imported.

A DTD is text-based. It can be internal, part of your XML file, or external, in a stand-alone text file. Either way, the DTD specifies what we should expect in the XML. The DTD contains the following elements:

  • Elements: appear as tags, determine what is being structured
  • Attributes: provide extra information about the element
  • Entities: define shortcuts that have special meaning
  • PCDATA: parsed character data, data that should be parsed
  • CDATA: character data that should not be parsed

The DTD

The DTD (Document Type Definition) defines the structure of an XML document. it is the grammar for the XML tag language. A DTD can be as part of the XML (internal) or in a separate file (external) which is easier for sharing and standards.

Internal DTD

An internal DTD appears at the top of your XML file. It will appear under the declaration for XML and will not be visible in your XML code if displayed. If you view the source code in a text editor, you will see the DTD, followed by the XML elements.

<?xml version="1.0"?>

<!DOCTYPE writing [
<!ELEMENT writing (title,author+,pubdate?)>
<!ATTLIST writing type CDATA #REQUIRED>
<!ELEMENT title (#PCDATA)>
<!ELEMENT author (authorfirstname, authorlastname)>
<!ELEMENT authorfirstname (#PCDATA)>
<!ELEMENT authorlastname (#PCDATA)>
<!ELEMENT pubdate (month,day,year)>
<!ELEMENT month (#PCDATA)>
<!ELEMENT day (#PCDATA)>
<!ELEMENT year (#PCDATA)>
]>

<writing type = 'book'>
  <title> The Joy of Data </title>
  <author>
    <authorfirstname>J.</authorfirstname>
    <authorlastname>Datasmith</authorlastname>
  </author>
  <pubdate>
    <month>12</month>
    <day>18</day>
    <year>2020</year>
  </pubdate>
</writing>

In the above example, the DTD tell us that the document stores writing, with space to capture the book type, title, author (broken into first and last name – which are called child nodes), and publication date (with the child nodes month, day, and year). #PCDATA tells us that the elements title can be parsed. The type of writing is an attribute of writing.

External DTD

If the DTD is external, it is located in a separate file. You will see that the XML code references the DTD in the opening lines of the XML. For example, a DTD with the name “writing.dtd” would be noted as: xml <!DOCTYPE writing SYSTEM "writing.dtd">

An external DTD file will contain similar text to the internal, but will only specify what is necessary for the definitions.

<!ELEMENT writing (title,author+,pubdate?)>
<!ATTLIST writing type CDATA #REQUIRED>
<!ELEMENT title (#PCDATA)>
<!ELEMENT author (authorfirstname, authorlastname)>
<!ELEMENT authorfirstname (#PCDATA)>
<!ELEMENT authorlastname (#PCDATA)>
<!ELEMENT pubdate (month,day,year)>
<!ELEMENT month (#PCDATA)>
<!ELEMENT day (#PCDATA)>
<!ELEMENT year (#PCDATA)>

DTD vs XML Schema Definition (XSD)

You may find that some XML documents do not contain references to a DTD and instead reference a schema. An XML schema is similar to a DTD in that it describes the structure of the XML document.

A DTD is written as a grammar (in fact, it is a BNF – or Bachus-Naur Form – grammar) and tends to be brief in nature. XML schemas are written in XML and are more powerful than a DTD because they allow for more control over the elements. An XML schema supports data types and name spaces. This allows for more granular descriptions, restrictions, and validation.

Both a DTD and schema will allow you to declare your elements and attributes, describe how these elements are groups, nested, and used, and provide a way to restrict the type or format of elements. A DTD may be less complex to write, because it uses a different language and has less code available. It is useful in specific types of data work where an XML schema is often more complex, using XML code.

A way to think about this:

  • A DTD will define and describe the structure of your document.
  • An XML schema will be more descriptive and allow you to have more control of data types.

In data and analytics work, when we access files to prepare them for including at the data warehouse or an OLAP database, we may see either one of these as part of the XML files.

Data Sources

Not all data in an organization is stored in a database managed by a database management system (DBMS) such as Oracle or MySQL. In fact, a whole lot more data is stored in ad-hoc data files, spreadsheets, and on the web. Most web-based data is contained in unstructured HTML files served by web servers while some data is available in structured or semi-structured XML documents.

Structured vs Unstructured Data

Data in relations (tables) stored in a relational database is structured data and the structure is defined through the database schema while data integrity is defined via constraints and enforced by the DBMS. Every row in a table has the same form and each attribute conforms to defined data types.

The same is not necessarily true for data in files, spreadsheets, and HTML, although some may be semi-structured by virtue of the tabular presentation of the data. Unstructured data is much more difficult to work with but is not at all uncommon.

Web Data & Scraping

A significant source of data is the web where data is often displayed on web pages. Such data is unstructured, and interpretation is done through inspection and retrieval via “scraping”. Web scraping is a technique for acquiring (retrieving) the data contained in unstructured HTML and converting it to a structured format that is usable for processing or inclusion in a structured data store. Most programming languages offer web scraping packages and functions. There are also numerous platforms available for scraping, e.g., http://x-tract.io. While the data is often publicly available, many information providers either explicitly forbid web scraping or they implicitly obfuscate data or hinder web scrapers. Some data is copyrighted (e.g., most sports data) and may not be used for commercial purposes or stored in any retrieval system.

Data Interchange

Data must often be interchanged and exported from systems and databases. An “export format” is required:

  • CSV (Comma Separated Values)
  • XML (domain language or ad-hoc)

Other interchange formats include SOAP and JSON.

CSV files either represents full tables or combinations (joins) of tables but are inherently tabular. XML files have both a hierarchical structures and through a DTD or Schema enforcement of structural constraints. XML files are queried queried through XPath and XQuery or via custom programming.

XML formats (elements, tags, attributes, structure) can be converted from one format to another without programming by specifying transformation rules in XSL. XSL itself is an XML language for specifying transformation rules.

Structure of an XML Document

While XML data is often contained in documents (files) that is not always the case. There are databases that are based on XML, such as Apache Axiom, Qizx, BaseX, Sonic XML Server, eXistdb, Sedna, and webMethod’s Tamino. So, rather than using the term XML document, we often prefer the term XML store.

An XML document uses two main constructs: elements and attributes. An element is a start tag and a matching end tag and contents enclosed between them, e.g., <PHONE>617-373-9000</PHONE>. XML is case sensitive, so <PHONE> and <phone> are not the same element tag. Elements can be nested, and an XML store forms a tree or hierarchy.

AN XML document starts with the preamble <?xml version="1.0"?>, followed by an optional DTD or XML Schema, followed by the root element with its child elements. There is only one root element, so all XML stores form a tree.

<?xml version="1.0" encoding="UTF-8" ?>

<root>
  ...
  <!-- child elements -->
</root>

An XML document is well-formed if

  1. it starts with an XML declaration in the form <?xml version="1.0" encoding="UTF-8" ?>
  2. has a single root element, which contains all other elements in the document as child elements, and, finally,
  3. all elements are properly nested in the XML document.

Elements vs Attributes

Elements may have attributes associated with them. An XML element attribute provides information about the element and is in the form of a name/value pair. Elements may have multiple attributes. Attributes names must only be unique within an element and not within the file.

In the XML fragment below, the element <publisher> has an attribute pID and the element <address> has the attribute aType.

<publishers>
  <publisher pID="p1001">
    <name>Murray Hills Publications</name>
    <address aType="US">
      <city>Wilmington</city>
      <state>DE</state>
    </address>
    <phone>647.331.5555</phone>
    <email>r_clochar@murray-hills.com</email>
  </publisher>
  ...
</publishers>

Attribute values should be enclosed in quotes, although for numbers that is not strictly enforced but good practice. All attributes have the data type “character” or text/string even if they are all numbers. Single and double quotes are the same but cannot be mixed, i.e., ‘att-val’ is equivalent to “att-val”. This is useful when we need to enclose one type of quote within another, e.g., “Karl’s Course”.

Comments

XML allows inserting comments anywhere in the document (except within tags). Comments do not become a part of the processed or displayed content. Comments are enclosed within <-- and -->.

<?xml version="1.0" encoding="UTF-8"?>

<!-– List of publishers -->
<publishers>
  <publisher pID="1001">
    <name>Murray</name>
    <address aType="US">
      <city>Wilmington</city>
      <state>DE</state>
    </address>
    <phone>647.331.5555</phone>
  </publisher>
  ...
</publishers>

Special Characters in XML

As some characters have special meaning in XML, e.g., <_ or >, they are a bit more difficult to insert. The table below lists some of the more common characters and their XML representation.

Character XML Representation Example
< &lt; <tag>y &lt; 0</tag>
> &gt; <tag>use &lt;X&gt;</tag>
& &amp; <tag>y &amp; w</tag>
&quot; <tag>y&quot;</tag>
&apos; <tag>Billy&apos;s Boat</tag>

Defining Structure with a DTD

An important issue is the lack of a definition of the structure of the XML elements. How do we know – or how does a receiving information system know – that the XML file is properly structured? Solution: Define the grammar of the XML. Method: Use the Document Type Definition (DTD) meta language.

A DTD is a set of rules that defines the elements and attributes that can appear within an XML document and their sequence and nesting. An XML document is considered a valid document if it conforms to the associated DTD. A DTD can appear within an XML document (internal DTD) or be defined in an external file (external DTD) which allows for sharing. An internal DTD is only visible within the file in which it resides, meaning that other XML files cannot use the DTD. A DTD that applies to several XML document or is used to communicate an XML standard should be placed into an external file rather than embedded.

Assume that the DTD shown below is stored in the file note.dtd.

<!DOCTYPE note [
<!ELEMENT note (to+,from,heading,body)>
<!ELEMENT to (#PCDATA)>
<!ELEMENT from (#PCDATA)>
<!ELEMENT heading (#PCDATA)>
<!ELEMENT body (#PCDATA)>
]>

Then the XML document below can reference that external DTD and ensure (during parsing of the XML) that it conforms to the rules of the DTD. Note the syntax of the referencing. It starts with <!DOCTYPE followed by the name of the root element (<note>) in this example) and the keyword SYSTEM and the path to the file, which can be a local file or a URL.

<?xml version="1.0" encoding="UTF-8"?>

<!DOCTYPE note SYSTEM "Note.dtd">

<note>
  <to>Chris</to>
  <from>Topher</from>
  <heading>Reminder</heading>
  <body>
   Don't forget the meeting!
  </body>
</note> 

To summarize, the Document Type Definition (DTD) can be in a separate file or defined inline at the beginning of the XML document. It is used to validate the structure of the XML document and defines the “grammar” of our XML “language”. The DTD language is used to define the grammar of the XML using Extended Bachus-Naur Form (EBNF). A DTD is optional (but recommended) although required if the document is to be validated by the parser prior to parsing.

DTD: Cardinality

A cardinality adornment may follow an element. If no cardinality adornment is present, then implied cardinality is exactly 1, i.e., the element must appear and cannot appear more than one. There are three adornments that modify cardinality:

  • _*_ indicating a cardinality of zero or more, i.e., the element can appear as often as desired and possibly not at all (optional)
  • + indicating a cardinality of one or more, i.e., the element can appear as often as desired but must appear at least once (mandatory)
  • ? indicating a cardinality of zero or one, i.e., the element may not appear at all or at most once

These adornments will become clearer in the sections that follow when we look at examples.

DTD: Content

An element that has a value is indicated with <!ELEMENT foo (#PCDATA)>. The value is what is between the opening and closing tag, e.g., for the element <foo>content</foo>, the value is “content”.

DTD: Sequence

A sequence in XML is a an element followed by another element. The diagram DTD below shows the different parts of the DTD and what they specify. The comma in the element definition for Author is a sequence of four child elements, which must appear in that order.

The DTD specifies that the “root” element is <root> and that it contains zero or more <Author> elements as indicated by the specification <!ELEMENT root (Author*)>. The parenthesis specifies child elements, so the root element has zero or more (indicated by the _*_ after Author) Author elements.

Each Author element consists of a sequence of child elements, all of which must appear exactly once (no cardinality adornment) in that sequence (the comma indicates a sequence).

Sequence of Elements in a DTD
Sequence of Elements in a DTD

An XML document that conforms to the above DTD would look like this:

<?xml version="1.0" encoding="UTF-8"?>

<root>
  <Author>
    <authorID>3349</authorID>
    <name>Charles Mercer</name>
    <org>Oakton Partners</org>
    <email>cm@oakton.com.ca</email>
  </Author>
  <Author>
    <authorID>1132</authorID>
    <name>Cindy Chen</name>
    <org>Alpen Presse</org>
    <email>cindy@alpen-presse.de</email>
  </Author>
</root>

The child elements directly underneath <Author> must appear in exactly that order and cannot be omitted, i.e., they are mandatory or required. Leaving one of the child elements out or changing the sequence would result in an error when the XML document is validated (perhaps during parsing when it is loaded into an application for processing).

The XML document below is also valid even though it contains no child elements under <root>. That is valid as the cardinality adornment for Author is _*_ which indicates zero or more elements of that type.

<?xml version="1.0" encoding="UTF-8"?>

<root>
</root>

If an element neither contains child elements nor a value, it can be written in short hand using a “self-closing tag”.

<?xml version="1.0" encoding="UTF-8"?>

<root/>

The XML documents below are not valid for the reason listed for each.

Example I: Missing Element

Not valid as the <org> element is missing.

<root>
  <Author>
    <authorID>3349</authorID>
    <name>Charles Mercer</name>
    <email>cm@oakton.com.ca</email>
  </Author>
</root>

Example II: Out of Sequence

Not valid as the child elements for <Author> are not in the order listed in the DTD.

<root>
  <Author>
    <authorID>3349</authorID>
    <name>Charles Mercer</name>
    <email>cm@oakton.com.ca</email>
    <org>Oakton Partners</org>
  </Author>
</root>

Example III: Tag Names Incorrect

Not valid as the <Author> tag is misspelled (wrong capitalization).

<root>
  <Author>
    <authorID>3349</authorID>
    <name>Charles Mercer</name>
    <email>cm@oakton.com.ca</email>
    <org>Oakton Partners</org>
  </author>
</root>

Example IV: Missing Closing Tag

Not valid as the <Author> tag is not closed, i.e., it is missing the closing tag </Author>

<root>
  <Author>
    <authorID>3349</authorID>
    <name>Charles Mercer</name>
    <email>cm@oakton.com.ca</email>
    <org>Oakton Partners</org>
</root>

Example V: Incorrect Nesting

Not valid as the <Author> tag is closed before the <org> tag, so the nesting is incorrect.

<root>
  <Author>
    <authorID>3349</authorID>
    <name>Charles Mercer</name>
    <email>cm@oakton.com.ca</email>
    <org>Oakton Partners
  </Author></org>
</root>

Naturally, most of these errors would not occur in practice if the XML is generated programmatically.

DTD: Mixed Content

The comma (,) operator specifies sequence in a DTD. On the other hand, the or (|) operator specifies a choice. If the DTD contains (foo|bar) it means that either the element foo or the element bar must appear. So, the DTD below specifies that an Author element has a mandatory child element authorID followed by another mandatory child element name followed by either an org or an email element. Actually, since it states _(org|email)*_ it means zero of more of (org|email).

Sequence of Elements in a DTD
Sequence of Elements in a DTD

An XML document that conforms to the above DTD would look like this:

<?xml version="1.0" encoding="UTF-8"?>

<root>
  <Author>
    <authorID>3349</authorID>
    <name>Charles Mercer</name>
    <org>Oakton Partners</org>
    <email>cm@oakton.com.ca</email>
    <email>c443@yahoo.com</email>
  </Author>
  <Author>
    <authorID>1132</authorID>
    <name>Cindy Chen</name>
    <email>cindy@alpen-presse.de</email>
  </Author>
  <Author>
    <authorID>3349</authorID>
    <name>Mark Weill</name>
  </Author>
    <Author>
    <authorID>876</authorID>
    <name>Allison Khan</name>
    <org>AK Associates</org>
    <email>cindy@alpen-presse.de</email>
    <org>Okaton College</org>
  </Author>
</root>

If we wanted at least one org and one email we would have specified the DTD as follows, where it would be one of authorID, name, org, email in that sequence followed by zero or more combinations of org and email.

<!DOCTYPE root [
  <!ELEMENT root (Author*)>
  <!ELEMENT Author
      (authorID,name,org,email,(org|email)*)>
  <!ELEMENT authorID (#PCDATA)>   
  <!ELEMENT name (#PCDATA)>
  <!ELEMENT org (#PCDATA)>
  <!ELEMENT email (#PCDATA)>
]>

DTD: Attributes

Elements may (optionally) have attributes which are key/value pairs set in the tag of the element. Attributes are character strings and should be enclosed in quotes (although that is not strictly required if they are numbers).

The example below uses an attribute for authorID rather than representing it as a child element. This is common for “key values”. An attribute can have a data type of CDATA (character data), ID (unique value), IDREF (reference to an ID), or an enumerated token (see below), among others. AN attribute is specified with <!ATTLIST element attribute default>, so <!ATTLIST Author authorID CDATA #REQUIRED> specifies the (mandatory) attribute authorID for the element Author.

<?xml version="1.0" encoding="UTF-8"?>

<!DOCTYPE root [
  <!ELEMENT root (Author*)>
  <!ELEMENT Author
      (name,org,email)>
  <!ATTLIST Author authorID CDATA #REQUIRED>   
  <!ELEMENT name (#PCDATA)>
  <!ELEMENT org (#PCDATA)>
  <!ELEMENT email (#PCDATA)>
]>

<root>
  <Author authorID = "3349">
    <name>Charles Mercer</name>
    <org>Oakton Partners</org>
    <email>c443@yahoo.com</email>
  </Author>
</root>

Element attributes can have the following constraints:

  • #REQUIRED – a value for the attribute must be provided
  • #FIXED – a fixed value that cannot be changed and all attributes for all element instances have that value; a value must be specified after this keyword
  • #IMPLIED – an optional attribute that may or may not be present

Note that these keywords must be in upper case and only one can be specified.

DTD: Enumerated Tokens

XML recognizes characters as the default type of any value. In addition, XML supports value sets or categorical attributes which draw their values from a predefined set of values. To enforce a value set (enumerated tokens) it must be defined as an attribute as no validation can occur for parsed character data (#PCDATA).

In the XML example below, the LearningAsset element has the attribute difficulty. It must be one of (easy|medium|difficult) and, if not present, defaults to the value “easy”. Setting the attribute to any value other than the ones defined results in a validation error.

<!DOCTYPE root [
<!ELEMENT root (LearningAsset*)>
<!ELEMENT LearningAsset (…)>
<!ATTLIST LearningAsset difficulty 
(easy|medium|difficult) "easy">
...
]>

<root>
  <LearningAsset difficulty = "medium">
    ...
  </LearningAsset>
  <LearningAsset>
    ...
  </LearningAsset>

  ...
</root>

Learning Checkpoint I

LC I.1

As an exercise, build an XML document that conforms to the DTD below.

<!DOCTYPE root [
  <!ELEMENT root (Author*)>
  <!ELEMENT Author
      (authorID,name,(org+|email)+>
  <!ELEMENT authorID (#PCDATA)>   
  <!ELEMENT name (#PCDATA)>
  <!ELEMENT org (#PCDATA)>
  <!ELEMENT email (#PCDATA)>
]>

LC I.2

Identify the two errors present in the DTD and corresponding XML. How would you fix the DTD so that it validates the XML?

<?xml version="1.0" encoding="UTF-8"?>

<!DOCTYPE msg
[<!ELEMENT note (to+,from,heading,body?)>
<!ELEMENT to (#PCDATA)>
<!ELEMENT from (#PCDATA)>
<!ELEMENT heading (#PCDATA)>
<!ELEMENT body (#PCDATA)>
]>

<note>  
  <from>Chris</from>
  <to>Topher</to>
  <heading>Reminder</heading>
  <body>
   Meeting today at 12 in WVH166!
  </body>
</note> 

LC I.2

A | indicates that one of several elements can be included. So, in the example below which elements must be present?

<?xml version="1.0" encoding="UTF-8"?>

<!DOCTYPE notes [
<!ELEMENT notes (note+)>
<!ELEMENT note (recipients,from,(body|image))>
<!ELEMENT recipients (to+)>
<!ELEMENT to (#PCDATA)>
<!ELEMENT from (#PCDATA)>
<!ELEMENT heading (#PCDATA)>
<!ELEMENT body (#PCDATA)>
]>

<notes>
  <note>
    <recipients>
      <to>Topher</to>
      <to>Pat</to>
    </recipients>
    <from>Chris</from>
    <body>
      Meeting today at 12 in WVH166!
    </body>
  </note> 
  
  <note>  
    <recipients>
      <to>Erin</to>
      <to>Barbara</to>
    </recipients>
    <from>Chris</from>
    <image>
      receipt.jpg
    </image>
  </note> 
</notes>

LC I.3

Create a DTD for the XML (courses.xml below assuming the following “rules”:

  • There is at least one section per course but no limit
  • time and location may appear in any order and location is optional
  • courseNumber and crn are required key attributes that must be unique
  • each course may or may not run and whether it’s running should be signaled with a Boolean attribute
  • campus is restricted to {Online,Boston,Vancouver,Seattle} and defaults to “Boston” if not specified
<?xml version="1.0" encoding="UTF-8"?>

<courses>
  <course courseNumber="CS5200">
    <title>Database Management Systems</title>
    <credits>4.0</credits>
    <sections>
      <section crn="377868" modality="Traditional">
        <days>MR</days>
        <time>11:45am - 1:25pm</time>
        <location>RY301</location>
      </section>
      <section crn="556224" modality="Flipped">
        <days>R</days>
        <time>6:00pm - 9:15pm</time>
        <location>103</location>
        <campus>Vancouver</campus>
      </section>
      <section crn="809445" modality="Online" />
    </sections>
  </course>
</courses>

Best Practices

Repeated elements are often best grouped under a common parent element as it makes it easier to parse.

<?xml version="1.0" encoding="UTF-8"?>

<!DOCTYPE note [
<!ELEMENT note (recipients,from,heading,body)>
<!ELEMENTS recipients (to+)>
<!ELEMENT to (#PCDATA)>
<!ELEMENT from (#PCDATA)>
<!ELEMENT heading (#PCDATA)>
<!ELEMENT body (#PCDATA)>
]>

<note>
  <recipients>
    <to>Topher</to>
    <to>Pat</to>
    <to>Maria</to>
  </recipients>
  <from>Chris</from>
  <heading>Reminder</heading>
  <body>
   Meeting today at 12 in WVH166!
  </body>
</note> 

Parsing an XML Document

An XML document must generally be valid in order to be easily parsable by a program, i.e., conform to rules (“grammar”) prescribed by the DTD. An XML document can be well-formed without being valid, i.e., each element has a properly nested start and end tag but may or may not have a DTD not conform to it. In practice, many XML files do not have a DTD which can make parsing (reading) them challenging and much more difficult.

Approaches to Parsing

There are two general approaches to “parsing” an XML document: DOM vs SAX parsing. In DOM (Document Object Model) parsing, the entire XML document is converted to an in-memory tree data structured and thus resides fully in memory where it can be traversed node-by-node or via XPath. In SAX (Simple API for XML) parsing only one element is parsed and loaded into memory at a time and a callback function registered by the processing program is called for each element as it is encountered. SAX parsing is preferable for very large XML documents, but it is more difficult to parse as context is lost and XPath cannot be used.

With SAX parsing, events are triggered when the XML is being parsed. When the parser is parsing the XML, and encounters a start tag (e.g., <something>), then it triggers an event and calls a processing function that is registered for that tag, so different tags may have different event handleing functions. Similarly, when the end of the tag is met while parsing (</something>), it triggers another event and calls a registered callback function. Using a SAX parser implies you need to handle these events and make sense of the data returned with each event, but you do not have “context” unless you track it.

On the other hand, in DOM parsing, there are no events triggered while parsing. The entire XML is parsed and a DOM tree (of the nodes in the XML) is generated and returned. Once parsed, the user can navigate the tree to access the various data previously embedded in the various nodes in the XML. In general, DOM is easier to use but has an overhead of parsing the entire XML before you can start using it and keeping it in memory.

DOM

  • tree model parser converting XML to an internal tree
  • loads the document fully into the memory and then parse the document
  • memory constrained as it loads the entire XML before parsing
  • entire object model is stored as a tree in memory, thus memory intensive
  • object model can be read and written, i.e., can insert or delete elements
  • preferable if the XML content fits into memory
  • can be queried via XPath
  • easy to navigate in any direction
  • slow to process

SAX

  • event based parser using callback functions
  • parses the document as it is read, i.e., parses one element at a time
  • no memory constraints as it does not store the XML in memory.
  • read only, i.e., can’t insert or delete elements
  • preferable when XML is large or memory is limited
  • reads the XML document from beginning to end in one direction
  • fast processing

Inspecting XML Files

XML files are text files and must be viewed (and edited) using a text editor and not a word processor. So, use TextEdit on MacOS, or Notepad on Windows. Don’t use Word or Pages. A programming environment (IDE) such as Visual Studio, Eclipse, or R Studio can also be used.

For example, to inspect the pubmed_sample.xml file, you can use R Studio – it handles XML well. Download the pubmed_sample.xml first, then launch R Studio, and then click File/Open File… and choose pubmed_sample.xml from the folder into which you downloaded the file (likely the Downloads folder). Voila…

The short demonstration below shows how to open an XML file in R Studio for inspection.

In many cases you can open up a file by just double-clicking on it. That launches the app that’s associated with the extension (.xml, .dtd, etc). You can change the association in your preference settings. But often you want to open a file with a different app or there’s no association. You will need to open the app compatible with the file type first, then open the file from within the app.

A note on downloading XML files from a link: this doesn’t generally work, as, by default, XML files are opened in your browser which will attempt to render the XML. So, to download and save an XML file, open the context menu by (generally) right-clicking on the link and selecting “Save As” or a similar choice depending on your operating system.

Summary

XML is an important mechanism for transferring data between parties and information systems.

Tutorial I: Overview of XML and DTD

In this narrated chalk-talk, Khoury Boston’s Prof. Schedlbauer provides an overview of XML, common use, and the role of DTD as a defining grammar of an XML schema1.

Tutorial II: Essential XML with Examples

In these two video tutorials, Khoury Boston’s Prof. Schedlbauer explains first how to build an XML document with an associated DTD followed by a code walk-through in https://www.xmlvalidation.com.

The tutorial uses the XML file:


Files & Resources

All Files for Lesson 80.101

Errata

Let us know.


  1. This chalk-talk was originally prepared for IS2000 – Principles of Information Systems and may contain references to that course↩︎

LS0tDQp0aXRsZTogIkJhc2ljcyBvZiBYTUwgYW5kIERURCINCnBhcmFtczoNCiAgY2F0ZWdvcnk6IDgwDQogIG51bWJlcjogMTAxDQogIHRpbWU6ICI0NSBtaW4iDQogIGxldmVsOiBiZWdpbm5lcg0KICB0YWdzOiAieG1sLGR0ZCINCiAgZGVzY3JpcHRpb246ICJUaGlzIGxlc3NvbiBwcmVzZW50cyBhbiBpbnRyb2R1Y3Rpb24gdG8gWE1MLCBpdHMgY29tbW9uIHVzZSwNCiAgICAgICAgICAgICAgICBzdHJ1Y3R1cmUgb2YgYW4gWE1MIGRvY3VtZW50LCBhbmQgdGhlIHByaW5jaXBsZXMgb2YgWE1MDQogICAgICAgICAgICAgICAgY29uc3RydWN0aW9uLiBEaXN0aW5ndWlzaGVzIGJldHdlZW4gdmFsaWQgYW5kIHdlbGwtZm9ybWVkIFhNTA0KICAgICAgICAgICAgICAgIGFuZCBzaG93cyBob3cgdG8gdXNlIERURCB0byBkZWZpbmUgdGhlIGdyYW1tYXIgb2YgYW4gWE1MDQogICAgICAgICAgICAgICAgbGFuZ3VhZ2UuIg0KZGF0ZTogIjxzbWFsbD5gciBTeXMuRGF0ZSgpYDwvc21hbGw+Ig0KYXV0aG9yOiAiPHNtYWxsPk1hcnRpbiBTY2hlZGxiYXVlcjwvc21hbGw+Ig0KZW1haWw6ICJtLnNjaGVkbGJhdWVyQG5ldS5lZHUiDQphZmZpbGl0YXRpb246ICJOb3J0aGVhc3Rlcm4gVW5pdmVyc2l0eSINCm91dHB1dDogDQogIGJvb2tkb3duOjpodG1sX2RvY3VtZW50MjoNCiAgICB0b2M6IHRydWUNCiAgICB0b2NfZmxvYXQ6IHRydWUNCiAgICBjb2xsYXBzZWQ6IGZhbHNlDQogICAgbnVtYmVyX3NlY3Rpb25zOiBmYWxzZQ0KICAgIGNvZGVfZG93bmxvYWQ6IHRydWUNCiAgICB0aGVtZTogcmVhZGFibGUNCiAgICBoaWdobGlnaHQ6IHRhbmdvDQotLS0NCg0KLS0tDQp0aXRsZTogIjxzbWFsbD5gciBwYXJhbXMkY2F0ZWdvcnlgLmByIHBhcmFtcyRudW1iZXJgPC9zbWFsbD48YnIvPjxzcGFuIHN0eWxlPSdjb2xvcjogIzJFNDA1MzsgZm9udC1zaXplOiAwLjllbSc+YHIgcm1hcmtkb3duOjptZXRhZGF0YSR0aXRsZWA8L3NwYW4+Ig0KLS0tDQoNCmBgYHtyIGNvZGU9eGZ1bjo6cmVhZF91dGY4KHBhc3RlMChoZXJlOjpoZXJlKCksJy9SL19pbnNlcnQyREIuUicpKSwgaW5jbHVkZSA9IEZBTFNFfQ0KYGBgDQoNCiMjIEludHJvZHVjdGlvbg0KDQpUaGlzIGxlc3NvbiBleHBsYWlucyB0aGUgaW1wb3J0YW5jZSBvZiBYTUwgYXMgYSBkYXRhLWludGVyY2hhbmdlIGxhbmd1YWdlIGFuZCBzaG93cyBob3cgWE1MIGRvY3VtZW50cyBhcmUgY29uc3RydWN0ZWQgdXNpbmcgdGFncywgaGllcmFyY2h5LCBhbmQgZ3JhbW1hci4gSW50cm9kdWNlcyB0aGUgRG9jdW1lbnQgVHlwZSBEZWZpbml0aW9uIChEVEQpIGxhbmd1YWdlIGZvciBkZWZpbmluZyB0aGUgZ3JhbW1hciAoc3RydWN0dXJlKSBvZiBhbiBYTUwgZG9jdW1lbnQuDQoNCiMjIFhNTCBpbiBhIE51dHNoZWxsDQoNClhNTCBpcyBhIG1hcmt1cCBsYW5ndWFnZSB0aGF0IHVzZXMgZWxlbWVudHMgYW5kIGF0dHJpYnV0ZXMgdG8gY3JlYXRlIGEgdGV4dCBmaWxlIG9mIHN0cnVjdHVyZWQgZGF0YS4gWE1MIGlzIGEgY29tbW9uIGRhdGEgZm9ybWF0IHVzZWQgYnkgbWFueSBvcmdhbml6YXRpb25zLCBzeXN0ZW1zLCBhbmQgdGhlIHdlYiB0byBleGNoYW5nZSBkYXRhLiBPZnRlbiwgZGF0YSBuZWVkZWQgZm9yIGFuYWx5dGljcyByZXNpZGVzIGluIGFuIFhNTCBhbmQgbXVzdCBiZSBpbXBvcnRlZC4gQ29uc2VxdWVudGx5LCBkYXRhIGFuYWx5dGljcyBwcm9mZXNzaW9uYWxzIHNob3VsZCBrbm93IGhvdyB0byByZWFkIChwYXJzZSkgWE1MIGFuZCBwcmVwYXJlIHRoZSBkYXRhIGZvciBpbXBvcnQgaW50byBhIGRhdGEgbWluaW5nIGVmZm9ydCBvciBhIGRhdGEgd2FyZWhvdXNlLg0KDQpPbmUgaGFsbG1hcmsgb2YgWE1MIGlzIHRoYXQgdGhlIHRhZ3MgdGhhdCBjcmVhdGUgZWxlbWVudHMgYXJlIGNyZWF0ZWQgYnkgdGhlIHBlcnNvbiBvciBzeXN0ZW0gdGhhdCBjcmVhdGVzIHRoZSBmaWxlLiBBcyBzdWNoLCB0aGVyZSBpcyBubyBzcGVjaWZpYyBkZWZpbml0aW9uIG9yIGV4cGxhbmF0aW9uIG9mIHRoZSB0YWdzIGFuZCB0aGUgZWxlbWVudHMgdGhleSBjcmVhdGUuIFRoaXMgaW5jcmVhc2VkIGZsZXhpYmlsaXR5IGZvciB1c2Ugb2YgdGhlIGZpbGVzIGJ1dCBjYW4gbWFrZSBpdCBtb3JlIGRpZmZpY3VsdCB0byBleGNoYW5nZSBkYXRhIGJldHdlZW4gc3lzdGVtcyBlZmZlY3RpdmVseS4gRm9yIHRoaXMgcmVhc29uLCB3ZSB1c2UgYSBEVEQgdG8gZGVzY3JpYmUgdGhlIHRhZ3MgdGhhdCBhcmUgdXNlZCB0aGUgdGhlIG9yZGVyIGFuZCBxdWFudGl0eSBpbiB3aGljaCB0aGV5IG11c3QgYXBwZWFyIGluIHRoZSBYTUwuLg0KDQpBIERURCwgb3IgRG9jdW1lbnQgVHlwZSBEZWZpbml0aW9uLCBkZWZpbmVzIHRoZSBzdHJ1Y3R1cmUgb2YgYW4gWE1MIGRvY3VtZW50IChvciBmaWxlKS4gSXQgZGVzY3JpYmVzIHRoZSBlbGVtZW50cyBhbmQgYXR0cmlidXRlcyB1c2VkIGluIHRoZSBmaWxlIGFuZCBoZWxwcyB1cyB0byB1bmRlcnN0YW5kIGFuZCBhZ3JlZSBvbiB0aGUgY29udGVudHMuIEl0IGFsc28gaXMgcG9zc2libGUgZm9yIHNwZWNpYWwgcGFyc2VycyB0byByZWFkIHRoZSBEVEQgdG8gZW5zdXJlIHRoYXQgdGhlIHN0cnVjdHVyZSBvZiBhbiBYTUwgZmlsZSBpcyB2YWxpZCBhY2NvcmRpbmcgdG8gdGhlIERURC4gU2luY2Ugd2UgYXJlIHdvcmtpbmcgd2l0aCBsYXJnZSBkYXRhIHNldHMsIHdlIGNhbm5vdCBjaGVjayB0aGVtIG91cnNlbHZlczsgYSBEVEQsIGFsb25nIHdpdGggdGhlIHJlc3RyaWN0aW9ucyBpbiB0aGUgWE1MIGxhbmd1YWdlLCBoZWxwIHRvIGVuc3VyZSB0aGF0IHRoZSBkYXRhIGlzIHN0cnVjdHVyZWQgY29ycmVjdGx5IGFuZCB0aHVzIGNhbiBiZSBpbXBvcnRlZC4NCg0KQSBEVEQgaXMgdGV4dC1iYXNlZC4gSXQgY2FuIGJlIGludGVybmFsLCBwYXJ0IG9mIHlvdXIgWE1MIGZpbGUsIG9yIGV4dGVybmFsLCBpbiBhIHN0YW5kLWFsb25lIHRleHQgZmlsZS4gRWl0aGVyIHdheSwgdGhlIERURCBzcGVjaWZpZXMgd2hhdCB3ZSBzaG91bGQgZXhwZWN0IGluIHRoZSBYTUwuIFRoZSBEVEQgY29udGFpbnMgdGhlIGZvbGxvd2luZyBlbGVtZW50czoNCg0KLSAgICpFbGVtZW50cyo6IGFwcGVhciBhcyB0YWdzLCBkZXRlcm1pbmUgd2hhdCBpcyBiZWluZyBzdHJ1Y3R1cmVkDQotICAgKkF0dHJpYnV0ZXMqOiBwcm92aWRlIGV4dHJhIGluZm9ybWF0aW9uIGFib3V0IHRoZSBlbGVtZW50DQotICAgKkVudGl0aWVzKjogZGVmaW5lIHNob3J0Y3V0cyB0aGF0IGhhdmUgc3BlY2lhbCBtZWFuaW5nDQotICAgKlBDREFUQSo6IHBhcnNlZCBjaGFyYWN0ZXIgZGF0YSwgZGF0YSB0aGF0IHNob3VsZCBiZSBwYXJzZWQNCi0gICAqQ0RBVEEqOiBjaGFyYWN0ZXIgZGF0YSB0aGF0IHNob3VsZCBub3QgYmUgcGFyc2VkDQoNCiMjIyBUaGUgRFREDQoNClRoZSBEVEQgKERvY3VtZW50IFR5cGUgRGVmaW5pdGlvbikgZGVmaW5lcyB0aGUgc3RydWN0dXJlIG9mIGFuIFhNTCBkb2N1bWVudC4gaXQgaXMgdGhlIGdyYW1tYXIgZm9yIHRoZSBYTUwgdGFnIGxhbmd1YWdlLiBBIERURCBjYW4gYmUgYXMgcGFydCBvZiB0aGUgWE1MIChpbnRlcm5hbCkgb3IgaW4gYSBzZXBhcmF0ZSBmaWxlIChleHRlcm5hbCkgd2hpY2ggaXMgZWFzaWVyIGZvciBzaGFyaW5nIGFuZCBzdGFuZGFyZHMuDQoNCiMjIyMgSW50ZXJuYWwgRFREDQoNCkFuIGludGVybmFsIERURCBhcHBlYXJzIGF0IHRoZSB0b3Agb2YgeW91ciBYTUwgZmlsZS4gSXQgd2lsbCBhcHBlYXIgdW5kZXIgdGhlIGRlY2xhcmF0aW9uIGZvciBYTUwgYW5kIHdpbGwgbm90IGJlIHZpc2libGUgaW4geW91ciBYTUwgY29kZSBpZiBkaXNwbGF5ZWQuIElmIHlvdSB2aWV3IHRoZSBzb3VyY2UgY29kZSBpbiBhIHRleHQgZWRpdG9yLCB5b3Ugd2lsbCBzZWUgdGhlIERURCwgZm9sbG93ZWQgYnkgdGhlIFhNTCBlbGVtZW50cy4NCg0KYGBgIHhtbA0KPD94bWwgdmVyc2lvbj0iMS4wIj8+DQoNCjwhRE9DVFlQRSB3cml0aW5nIFsNCjwhRUxFTUVOVCB3cml0aW5nICh0aXRsZSxhdXRob3IrLHB1YmRhdGU/KT4NCjwhQVRUTElTVCB3cml0aW5nIHR5cGUgQ0RBVEEgI1JFUVVJUkVEPg0KPCFFTEVNRU5UIHRpdGxlICgjUENEQVRBKT4NCjwhRUxFTUVOVCBhdXRob3IgKGF1dGhvcmZpcnN0bmFtZSwgYXV0aG9ybGFzdG5hbWUpPg0KPCFFTEVNRU5UIGF1dGhvcmZpcnN0bmFtZSAoI1BDREFUQSk+DQo8IUVMRU1FTlQgYXV0aG9ybGFzdG5hbWUgKCNQQ0RBVEEpPg0KPCFFTEVNRU5UIHB1YmRhdGUgKG1vbnRoLGRheSx5ZWFyKT4NCjwhRUxFTUVOVCBtb250aCAoI1BDREFUQSk+DQo8IUVMRU1FTlQgZGF5ICgjUENEQVRBKT4NCjwhRUxFTUVOVCB5ZWFyICgjUENEQVRBKT4NCl0+DQoNCjx3cml0aW5nIHR5cGUgPSAnYm9vayc+DQogIDx0aXRsZT4gVGhlIEpveSBvZiBEYXRhIDwvdGl0bGU+DQogIDxhdXRob3I+DQogICAgPGF1dGhvcmZpcnN0bmFtZT5KLjwvYXV0aG9yZmlyc3RuYW1lPg0KICAgIDxhdXRob3JsYXN0bmFtZT5EYXRhc21pdGg8L2F1dGhvcmxhc3RuYW1lPg0KICA8L2F1dGhvcj4NCiAgPHB1YmRhdGU+DQogICAgPG1vbnRoPjEyPC9tb250aD4NCiAgICA8ZGF5PjE4PC9kYXk+DQogICAgPHllYXI+MjAyMDwveWVhcj4NCiAgPC9wdWJkYXRlPg0KPC93cml0aW5nPg0KYGBgDQoNCkluIHRoZSBhYm92ZSBleGFtcGxlLCB0aGUgRFREIHRlbGwgdXMgdGhhdCB0aGUgZG9jdW1lbnQgc3RvcmVzIHdyaXRpbmcsIHdpdGggc3BhY2UgdG8gY2FwdHVyZSB0aGUgYm9vayB0eXBlLCB0aXRsZSwgYXV0aG9yIChicm9rZW4gaW50byBmaXJzdCBhbmQgbGFzdCBuYW1lIC0tIHdoaWNoIGFyZSBjYWxsZWQgY2hpbGQgbm9kZXMpLCBhbmQgcHVibGljYXRpb24gZGF0ZSAod2l0aCB0aGUgY2hpbGQgbm9kZXMgbW9udGgsIGRheSwgYW5kIHllYXIpLiAjUENEQVRBIHRlbGxzIHVzIHRoYXQgdGhlIGVsZW1lbnRzIHRpdGxlIGNhbiBiZSBwYXJzZWQuIFRoZSB0eXBlIG9mIHdyaXRpbmcgaXMgYW4gYXR0cmlidXRlIG9mIHdyaXRpbmcuDQoNCiMjIyMgRXh0ZXJuYWwgRFREDQoNCklmIHRoZSBEVEQgaXMgZXh0ZXJuYWwsIGl0IGlzIGxvY2F0ZWQgaW4gYSBzZXBhcmF0ZSBmaWxlLiBZb3Ugd2lsbCBzZWUgdGhhdCB0aGUgWE1MIGNvZGUgcmVmZXJlbmNlcyB0aGUgRFREIGluIHRoZSBvcGVuaW5nIGxpbmVzIG9mIHRoZSBYTUwuIEZvciBleGFtcGxlLCBhIERURCB3aXRoIHRoZSBuYW1lICJ3cml0aW5nLmR0ZCIgd291bGQgYmUgbm90ZWQgYXM6IGB4bWwgPCFET0NUWVBFIHdyaXRpbmcgU1lTVEVNICJ3cml0aW5nLmR0ZCI+YA0KDQpBbiBleHRlcm5hbCBEVEQgZmlsZSB3aWxsIGNvbnRhaW4gc2ltaWxhciB0ZXh0IHRvIHRoZSBpbnRlcm5hbCwgYnV0IHdpbGwgb25seSBzcGVjaWZ5IHdoYXQgaXMgbmVjZXNzYXJ5IGZvciB0aGUgZGVmaW5pdGlvbnMuDQoNCmBgYCB4bWwNCjwhRUxFTUVOVCB3cml0aW5nICh0aXRsZSxhdXRob3IrLHB1YmRhdGU/KT4NCjwhQVRUTElTVCB3cml0aW5nIHR5cGUgQ0RBVEEgI1JFUVVJUkVEPg0KPCFFTEVNRU5UIHRpdGxlICgjUENEQVRBKT4NCjwhRUxFTUVOVCBhdXRob3IgKGF1dGhvcmZpcnN0bmFtZSwgYXV0aG9ybGFzdG5hbWUpPg0KPCFFTEVNRU5UIGF1dGhvcmZpcnN0bmFtZSAoI1BDREFUQSk+DQo8IUVMRU1FTlQgYXV0aG9ybGFzdG5hbWUgKCNQQ0RBVEEpPg0KPCFFTEVNRU5UIHB1YmRhdGUgKG1vbnRoLGRheSx5ZWFyKT4NCjwhRUxFTUVOVCBtb250aCAoI1BDREFUQSk+DQo8IUVMRU1FTlQgZGF5ICgjUENEQVRBKT4NCjwhRUxFTUVOVCB5ZWFyICgjUENEQVRBKT4NCmBgYA0KDQojIyMgRFREIHZzIFhNTCBTY2hlbWEgRGVmaW5pdGlvbiAoWFNEKQ0KDQpZb3UgbWF5IGZpbmQgdGhhdCBzb21lIFhNTCBkb2N1bWVudHMgZG8gbm90IGNvbnRhaW4gcmVmZXJlbmNlcyB0byBhIERURCBhbmQgaW5zdGVhZCByZWZlcmVuY2UgYSBzY2hlbWEuIEFuIFhNTCBzY2hlbWEgaXMgc2ltaWxhciB0byBhIERURCBpbiB0aGF0IGl0IGRlc2NyaWJlcyB0aGUgc3RydWN0dXJlIG9mIHRoZSBYTUwgZG9jdW1lbnQuDQoNCkEgRFREIGlzIHdyaXR0ZW4gYXMgYSBncmFtbWFyIChpbiBmYWN0LCBpdCBpcyBhIEJORiAtLSBvciBCYWNodXMtTmF1ciBGb3JtIC0tIGdyYW1tYXIpIGFuZCB0ZW5kcyB0byBiZSBicmllZiBpbiBuYXR1cmUuIFhNTCBzY2hlbWFzIGFyZSB3cml0dGVuIGluIFhNTCBhbmQgYXJlIG1vcmUgcG93ZXJmdWwgdGhhbiBhIERURCBiZWNhdXNlIHRoZXkgYWxsb3cgZm9yIG1vcmUgY29udHJvbCBvdmVyIHRoZSBlbGVtZW50cy4gQW4gWE1MIHNjaGVtYSBzdXBwb3J0cyBkYXRhIHR5cGVzIGFuZCBuYW1lIHNwYWNlcy4gVGhpcyBhbGxvd3MgZm9yIG1vcmUgZ3JhbnVsYXIgZGVzY3JpcHRpb25zLCByZXN0cmljdGlvbnMsIGFuZCB2YWxpZGF0aW9uLg0KDQpCb3RoIGEgRFREIGFuZCBzY2hlbWEgd2lsbCBhbGxvdyB5b3UgdG8gZGVjbGFyZSB5b3VyIGVsZW1lbnRzIGFuZCBhdHRyaWJ1dGVzLCBkZXNjcmliZSBob3cgdGhlc2UgZWxlbWVudHMgYXJlIGdyb3VwcywgbmVzdGVkLCBhbmQgdXNlZCwgYW5kIHByb3ZpZGUgYSB3YXkgdG8gcmVzdHJpY3QgdGhlIHR5cGUgb3IgZm9ybWF0IG9mIGVsZW1lbnRzLiBBIERURCBtYXkgYmUgbGVzcyBjb21wbGV4IHRvIHdyaXRlLCBiZWNhdXNlIGl0IHVzZXMgYSBkaWZmZXJlbnQgbGFuZ3VhZ2UgYW5kIGhhcyBsZXNzIGNvZGUgYXZhaWxhYmxlLiBJdCBpcyB1c2VmdWwgaW4gc3BlY2lmaWMgdHlwZXMgb2YgZGF0YSB3b3JrIHdoZXJlIGFuIFhNTCBzY2hlbWEgaXMgb2Z0ZW4gbW9yZSBjb21wbGV4LCB1c2luZyBYTUwgY29kZS4NCg0KQSB3YXkgdG8gdGhpbmsgYWJvdXQgdGhpczoNCg0KLSAgIEEgRFREIHdpbGwgZGVmaW5lIGFuZCBkZXNjcmliZSB0aGUgc3RydWN0dXJlIG9mIHlvdXIgZG9jdW1lbnQuDQotICAgQW4gWE1MIHNjaGVtYSB3aWxsIGJlIG1vcmUgZGVzY3JpcHRpdmUgYW5kIGFsbG93IHlvdSB0byBoYXZlIG1vcmUgY29udHJvbCBvZiBkYXRhIHR5cGVzLg0KDQpJbiBkYXRhIGFuZCBhbmFseXRpY3Mgd29yaywgd2hlbiB3ZSBhY2Nlc3MgZmlsZXMgdG8gcHJlcGFyZSB0aGVtIGZvciBpbmNsdWRpbmcgYXQgdGhlIGRhdGEgd2FyZWhvdXNlIG9yIGFuIE9MQVAgZGF0YWJhc2UsIHdlIG1heSBzZWUgZWl0aGVyIG9uZSBvZiB0aGVzZSBhcyBwYXJ0IG9mIHRoZSBYTUwgZmlsZXMuDQoNCiMjIERhdGEgU291cmNlcw0KDQpOb3QgYWxsIGRhdGEgaW4gYW4gb3JnYW5pemF0aW9uIGlzIHN0b3JlZCBpbiBhIGRhdGFiYXNlIG1hbmFnZWQgYnkgYSBkYXRhYmFzZSBtYW5hZ2VtZW50IHN5c3RlbSAoREJNUykgc3VjaCBhcyBPcmFjbGUgb3IgTXlTUUwuIEluIGZhY3QsIGEgd2hvbGUgbG90IG1vcmUgZGF0YSBpcyBzdG9yZWQgaW4gYWQtaG9jIGRhdGEgZmlsZXMsIHNwcmVhZHNoZWV0cywgYW5kIG9uIHRoZSB3ZWIuIE1vc3Qgd2ViLWJhc2VkIGRhdGEgaXMgY29udGFpbmVkIGluIHVuc3RydWN0dXJlZCBIVE1MIGZpbGVzIHNlcnZlZCBieSB3ZWIgc2VydmVycyB3aGlsZSBzb21lIGRhdGEgaXMgYXZhaWxhYmxlIGluIHN0cnVjdHVyZWQgb3Igc2VtaS1zdHJ1Y3R1cmVkIFhNTCBkb2N1bWVudHMuDQoNCiMjIyBTdHJ1Y3R1cmVkIHZzIFVuc3RydWN0dXJlZCBEYXRhDQoNCkRhdGEgaW4gcmVsYXRpb25zICh0YWJsZXMpIHN0b3JlZCBpbiBhIHJlbGF0aW9uYWwgZGF0YWJhc2UgaXMgc3RydWN0dXJlZCBkYXRhIGFuZCB0aGUgc3RydWN0dXJlIGlzIGRlZmluZWQgdGhyb3VnaCB0aGUgZGF0YWJhc2Ugc2NoZW1hIHdoaWxlIGRhdGEgaW50ZWdyaXR5IGlzIGRlZmluZWQgdmlhIGNvbnN0cmFpbnRzIGFuZCBlbmZvcmNlZCBieSB0aGUgREJNUy4gRXZlcnkgcm93IGluIGEgdGFibGUgaGFzIHRoZSBzYW1lIGZvcm0gYW5kIGVhY2ggYXR0cmlidXRlIGNvbmZvcm1zIHRvIGRlZmluZWQgZGF0YSB0eXBlcy4NCg0KVGhlIHNhbWUgaXMgbm90IG5lY2Vzc2FyaWx5IHRydWUgZm9yIGRhdGEgaW4gZmlsZXMsIHNwcmVhZHNoZWV0cywgYW5kIEhUTUwsIGFsdGhvdWdoIHNvbWUgbWF5IGJlIHNlbWktc3RydWN0dXJlZCBieSB2aXJ0dWUgb2YgdGhlIHRhYnVsYXIgcHJlc2VudGF0aW9uIG9mIHRoZSBkYXRhLiBVbnN0cnVjdHVyZWQgZGF0YSBpcyBtdWNoIG1vcmUgZGlmZmljdWx0IHRvIHdvcmsgd2l0aCBidXQgaXMgbm90IGF0IGFsbCB1bmNvbW1vbi4NCg0KIyMjIFdlYiBEYXRhICYgU2NyYXBpbmcNCg0KQSBzaWduaWZpY2FudCBzb3VyY2Ugb2YgZGF0YSBpcyB0aGUgd2ViIHdoZXJlIGRhdGEgaXMgb2Z0ZW4gZGlzcGxheWVkIG9uIHdlYiBwYWdlcy4gU3VjaCBkYXRhIGlzIHVuc3RydWN0dXJlZCwgYW5kIGludGVycHJldGF0aW9uIGlzIGRvbmUgdGhyb3VnaCBpbnNwZWN0aW9uIGFuZCByZXRyaWV2YWwgdmlhICJzY3JhcGluZyIuIFdlYiBzY3JhcGluZyBpcyBhIHRlY2huaXF1ZSBmb3IgYWNxdWlyaW5nIChyZXRyaWV2aW5nKSB0aGUgZGF0YSBjb250YWluZWQgaW4gdW5zdHJ1Y3R1cmVkIEhUTUwgYW5kIGNvbnZlcnRpbmcgaXQgdG8gYSBzdHJ1Y3R1cmVkIGZvcm1hdCB0aGF0IGlzIHVzYWJsZSBmb3IgcHJvY2Vzc2luZyBvciBpbmNsdXNpb24gaW4gYSBzdHJ1Y3R1cmVkIGRhdGEgc3RvcmUuIE1vc3QgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2VzIG9mZmVyIHdlYiBzY3JhcGluZyBwYWNrYWdlcyBhbmQgZnVuY3Rpb25zLiBUaGVyZSBhcmUgYWxzbyBudW1lcm91cyBwbGF0Zm9ybXMgYXZhaWxhYmxlIGZvciBzY3JhcGluZywgKmUuZy4qLCA8aHR0cDovL3gtdHJhY3QuaW8+LiBXaGlsZSB0aGUgZGF0YSBpcyBvZnRlbiBwdWJsaWNseSBhdmFpbGFibGUsIG1hbnkgaW5mb3JtYXRpb24gcHJvdmlkZXJzIGVpdGhlciBleHBsaWNpdGx5IGZvcmJpZCB3ZWIgc2NyYXBpbmcgb3IgdGhleSBpbXBsaWNpdGx5IG9iZnVzY2F0ZSBkYXRhIG9yIGhpbmRlciB3ZWIgc2NyYXBlcnMuIFNvbWUgZGF0YSBpcyBjb3B5cmlnaHRlZCAoZS5nLiwgbW9zdCBzcG9ydHMgZGF0YSkgYW5kIG1heSBub3QgYmUgdXNlZCBmb3IgY29tbWVyY2lhbCBwdXJwb3NlcyBvciBzdG9yZWQgaW4gYW55IHJldHJpZXZhbCBzeXN0ZW0uDQoNCiMjIERhdGEgSW50ZXJjaGFuZ2UNCg0KRGF0YSBtdXN0IG9mdGVuIGJlIGludGVyY2hhbmdlZCBhbmQgZXhwb3J0ZWQgZnJvbSBzeXN0ZW1zIGFuZCBkYXRhYmFzZXMuIEFuICJleHBvcnQgZm9ybWF0IiBpcyByZXF1aXJlZDoNCg0KLSAgIENTViAoQ29tbWEgU2VwYXJhdGVkIFZhbHVlcykNCi0gICBYTUwgKGRvbWFpbiBsYW5ndWFnZSBvciBhZC1ob2MpDQoNCk90aGVyIGludGVyY2hhbmdlIGZvcm1hdHMgaW5jbHVkZSBTT0FQIGFuZCBKU09OLg0KDQpDU1YgZmlsZXMgZWl0aGVyIHJlcHJlc2VudHMgZnVsbCB0YWJsZXMgb3IgY29tYmluYXRpb25zIChqb2lucykgb2YgdGFibGVzIGJ1dCBhcmUgaW5oZXJlbnRseSB0YWJ1bGFyLiBYTUwgZmlsZXMgaGF2ZSBib3RoIGEgaGllcmFyY2hpY2FsIHN0cnVjdHVyZXMgYW5kIHRocm91Z2ggYSBEVEQgb3IgU2NoZW1hIGVuZm9yY2VtZW50IG9mIHN0cnVjdHVyYWwgY29uc3RyYWludHMuIFhNTCBmaWxlcyBhcmUgcXVlcmllZCBxdWVyaWVkIHRocm91Z2ggWFBhdGggYW5kIFhRdWVyeSBvciB2aWEgY3VzdG9tIHByb2dyYW1taW5nLg0KDQpYTUwgZm9ybWF0cyAoZWxlbWVudHMsIHRhZ3MsIGF0dHJpYnV0ZXMsIHN0cnVjdHVyZSkgY2FuIGJlIGNvbnZlcnRlZCBmcm9tIG9uZSBmb3JtYXQgdG8gYW5vdGhlciB3aXRob3V0IHByb2dyYW1taW5nIGJ5IHNwZWNpZnlpbmcgdHJhbnNmb3JtYXRpb24gcnVsZXMgaW4gWFNMLiBYU0wgaXRzZWxmIGlzIGFuIFhNTCBsYW5ndWFnZSBmb3Igc3BlY2lmeWluZyB0cmFuc2Zvcm1hdGlvbiBydWxlcy4NCg0KIyMgU3RydWN0dXJlIG9mIGFuIFhNTCBEb2N1bWVudA0KDQpXaGlsZSBYTUwgZGF0YSBpcyBvZnRlbiBjb250YWluZWQgaW4gZG9jdW1lbnRzIChmaWxlcykgdGhhdCBpcyBub3QgYWx3YXlzIHRoZSBjYXNlLiBUaGVyZSBhcmUgZGF0YWJhc2VzIHRoYXQgYXJlIGJhc2VkIG9uIFhNTCwgc3VjaCBhcyBBcGFjaGUgQXhpb20sIFFpengsIEJhc2VYLCBTb25pYyBYTUwgU2VydmVyLCBlWGlzdGRiLCBTZWRuYSwgYW5kIHdlYk1ldGhvZCdzIFRhbWluby4gU28sIHJhdGhlciB0aGFuIHVzaW5nIHRoZSB0ZXJtIFhNTCBkb2N1bWVudCwgd2Ugb2Z0ZW4gcHJlZmVyIHRoZSB0ZXJtIFhNTCBzdG9yZS4NCg0KQW4gWE1MIGRvY3VtZW50IHVzZXMgdHdvIG1haW4gY29uc3RydWN0czogZWxlbWVudHMgYW5kIGF0dHJpYnV0ZXMuIEFuIGVsZW1lbnQgaXMgYSBzdGFydCB0YWcgYW5kIGEgbWF0Y2hpbmcgZW5kIHRhZyBhbmQgY29udGVudHMgZW5jbG9zZWQgYmV0d2VlbiB0aGVtLCAqZS5nLiosIGA8UEhPTkU+NjE3LTM3My05MDAwPC9QSE9ORT5gLiBYTUwgaXMgY2FzZSBzZW5zaXRpdmUsIHNvIGA8UEhPTkU+YCBhbmQgYDxwaG9uZT5gIGFyZSBub3QgdGhlIHNhbWUgZWxlbWVudCB0YWcuIEVsZW1lbnRzIGNhbiBiZSBuZXN0ZWQsIGFuZCBhbiBYTUwgc3RvcmUgZm9ybXMgYSB0cmVlIG9yIGhpZXJhcmNoeS4NCg0KQU4gWE1MIGRvY3VtZW50IHN0YXJ0cyB3aXRoIHRoZSBwcmVhbWJsZSBgPD94bWwgdmVyc2lvbj0iMS4wIj8+YCwgZm9sbG93ZWQgYnkgYW4gb3B0aW9uYWwgRFREIG9yIFhNTCBTY2hlbWEsIGZvbGxvd2VkIGJ5IHRoZSByb290IGVsZW1lbnQgd2l0aCBpdHMgY2hpbGQgZWxlbWVudHMuIFRoZXJlIGlzIG9ubHkgb25lIHJvb3QgZWxlbWVudCwgc28gYWxsIFhNTCBzdG9yZXMgZm9ybSBhIHRyZWUuDQoNCmBgYCB4bWwNCjw/eG1sIHZlcnNpb249IjEuMCIgZW5jb2Rpbmc9IlVURi04IiA/Pg0KDQo8cm9vdD4NCiAgLi4uDQogIDwhLS0gY2hpbGQgZWxlbWVudHMgLS0+DQo8L3Jvb3Q+DQpgYGANCg0KQW4gWE1MIGRvY3VtZW50IGlzICoqd2VsbC1mb3JtZWQqKiBpZg0KDQoxLiAgaXQgc3RhcnRzIHdpdGggYW4gWE1MIGRlY2xhcmF0aW9uIGluIHRoZSBmb3JtIGA8P3htbCB2ZXJzaW9uPSIxLjAiIGVuY29kaW5nPSJVVEYtOCIgPz5gDQoyLiAgaGFzIGEgc2luZ2xlIHJvb3QgZWxlbWVudCwgd2hpY2ggY29udGFpbnMgYWxsIG90aGVyIGVsZW1lbnRzIGluIHRoZSBkb2N1bWVudCBhcyBjaGlsZCBlbGVtZW50cywgYW5kLCBmaW5hbGx5LA0KMy4gIGFsbCBlbGVtZW50cyBhcmUgcHJvcGVybHkgbmVzdGVkIGluIHRoZSBYTUwgZG9jdW1lbnQuDQoNCiMjIyBFbGVtZW50cyB2cyBBdHRyaWJ1dGVzDQoNCkVsZW1lbnRzIG1heSBoYXZlIGF0dHJpYnV0ZXMgYXNzb2NpYXRlZCB3aXRoIHRoZW0uIEFuIFhNTCBlbGVtZW50IGF0dHJpYnV0ZSBwcm92aWRlcyBpbmZvcm1hdGlvbiBhYm91dCB0aGUgZWxlbWVudCBhbmQgaXMgaW4gdGhlIGZvcm0gb2YgYSBuYW1lL3ZhbHVlIHBhaXIuIEVsZW1lbnRzIG1heSBoYXZlIG11bHRpcGxlIGF0dHJpYnV0ZXMuIEF0dHJpYnV0ZXMgbmFtZXMgbXVzdCBvbmx5IGJlIHVuaXF1ZSB3aXRoaW4gYW4gZWxlbWVudCBhbmQgbm90IHdpdGhpbiB0aGUgZmlsZS4NCg0KSW4gdGhlIFhNTCBmcmFnbWVudCBiZWxvdywgdGhlIGVsZW1lbnQgKlw8cHVibGlzaGVyXD4qIGhhcyBhbiBhdHRyaWJ1dGUgKnBJRCogYW5kIHRoZSBlbGVtZW50ICpcPGFkZHJlc3NcPiogaGFzIHRoZSBhdHRyaWJ1dGUgKmFUeXBlKi4NCg0KYGBgIHhtbA0KPHB1Ymxpc2hlcnM+DQogIDxwdWJsaXNoZXIgcElEPSJwMTAwMSI+DQogICAgPG5hbWU+TXVycmF5IEhpbGxzIFB1YmxpY2F0aW9uczwvbmFtZT4NCiAgICA8YWRkcmVzcyBhVHlwZT0iVVMiPg0KICAgICAgPGNpdHk+V2lsbWluZ3RvbjwvY2l0eT4NCiAgICAgIDxzdGF0ZT5ERTwvc3RhdGU+DQogICAgPC9hZGRyZXNzPg0KICAgIDxwaG9uZT42NDcuMzMxLjU1NTU8L3Bob25lPg0KICAgIDxlbWFpbD5yX2Nsb2NoYXJAbXVycmF5LWhpbGxzLmNvbTwvZW1haWw+DQogIDwvcHVibGlzaGVyPg0KICAuLi4NCjwvcHVibGlzaGVycz4NCmBgYA0KDQpBdHRyaWJ1dGUgdmFsdWVzIHNob3VsZCBiZSBlbmNsb3NlZCBpbiBxdW90ZXMsIGFsdGhvdWdoIGZvciBudW1iZXJzIHRoYXQgaXMgbm90IHN0cmljdGx5IGVuZm9yY2VkIGJ1dCBnb29kIHByYWN0aWNlLiBBbGwgYXR0cmlidXRlcyBoYXZlIHRoZSBkYXRhIHR5cGUgImNoYXJhY3RlciIgb3IgdGV4dC9zdHJpbmcgZXZlbiBpZiB0aGV5IGFyZSBhbGwgbnVtYmVycy4gU2luZ2xlIGFuZCBkb3VibGUgcXVvdGVzIGFyZSB0aGUgc2FtZSBidXQgY2Fubm90IGJlIG1peGVkLCAqaS5lLiosICdhdHQtdmFsJyBpcyBlcXVpdmFsZW50IHRvICJhdHQtdmFsIi4gVGhpcyBpcyB1c2VmdWwgd2hlbiB3ZSBuZWVkIHRvIGVuY2xvc2Ugb25lIHR5cGUgb2YgcXVvdGUgd2l0aGluIGFub3RoZXIsICplLmcuKiwgIkthcmwncyBDb3Vyc2UiLg0KDQojIyMgQ29tbWVudHMNCg0KWE1MIGFsbG93cyBpbnNlcnRpbmcgY29tbWVudHMgYW55d2hlcmUgaW4gdGhlIGRvY3VtZW50IChleGNlcHQgd2l0aGluIHRhZ3MpLiBDb21tZW50cyBkbyBub3QgYmVjb21lIGEgcGFydCBvZiB0aGUgcHJvY2Vzc2VkIG9yIGRpc3BsYXllZCBjb250ZW50LiBDb21tZW50cyBhcmUgZW5jbG9zZWQgd2l0aGluIGA8LS1gIGFuZCBgLS0+YC4NCg0KYGBgIHhtbA0KPD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiPz4NCg0KPCEt4oCTIExpc3Qgb2YgcHVibGlzaGVycyAtLT4NCjxwdWJsaXNoZXJzPg0KICA8cHVibGlzaGVyIHBJRD0iMTAwMSI+DQogICAgPG5hbWU+TXVycmF5PC9uYW1lPg0KICAgIDxhZGRyZXNzIGFUeXBlPSJVUyI+DQogICAgICA8Y2l0eT5XaWxtaW5ndG9uPC9jaXR5Pg0KICAgICAgPHN0YXRlPkRFPC9zdGF0ZT4NCiAgICA8L2FkZHJlc3M+DQogICAgPHBob25lPjY0Ny4zMzEuNTU1NTwvcGhvbmU+DQogIDwvcHVibGlzaGVyPg0KICAuLi4NCjwvcHVibGlzaGVycz4NCmBgYA0KDQojIyMgU3BlY2lhbCBDaGFyYWN0ZXJzIGluIFhNTA0KDQpBcyBzb21lIGNoYXJhY3RlcnMgaGF2ZSBzcGVjaWFsIG1lYW5pbmcgaW4gWE1MLCAqZS5nLiwqIFw8XF8gb3IgKlw+KiwgdGhleSBhcmUgYSBiaXQgbW9yZSBkaWZmaWN1bHQgdG8gaW5zZXJ0LiBUaGUgdGFibGUgYmVsb3cgbGlzdHMgc29tZSBvZiB0aGUgbW9yZSBjb21tb24gY2hhcmFjdGVycyBhbmQgdGhlaXIgWE1MIHJlcHJlc2VudGF0aW9uLg0KDQp8IENoYXJhY3RlciB8IFhNTCBSZXByZXNlbnRhdGlvbiB8IEV4YW1wbGUgICAgICAgICAgICAgICAgICAgICAgICB8DQp8Oi0tLS0tLS0tLS18Oi0tLS0tLS0tLS0tLS0tLS0tLS18Oi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18DQp8IFw8ICAgICAgICB8IGAmbHQ7YCAgICAgICAgICAgICB8IGA8dGFnPnkgJmx0OyAwPC90YWc+YCAgICAgICAgICB8DQp8IFw+ICAgICAgICB8IGAmZ3Q7YCAgICAgICAgICAgICB8IGA8dGFnPnVzZSAmbHQ7WCZndDs8L3RhZz5gICAgICB8DQp8ICYgICAgICAgICB8IGAmYW1wO2AgICAgICAgICAgICB8IGA8dGFnPnkgJmFtcDsgdzwvdGFnPmAgICAgICAgICB8DQp8ICIgICAgICAgICB8IGAmcXVvdDtgICAgICAgICAgICB8IGA8dGFnPnkmcXVvdDs8L3RhZz5gICAgICAgICAgICB8DQp8ICcgICAgICAgICB8IGAmYXBvcztgICAgICAgICAgICB8IGA8dGFnPkJpbGx5JmFwb3M7cyBCb2F0PC90YWc+YCB8DQoNCiMjIERlZmluaW5nIFN0cnVjdHVyZSB3aXRoIGEgRFREDQoNCkFuIGltcG9ydGFudCBpc3N1ZSBpcyB0aGUgbGFjayBvZiBhIGRlZmluaXRpb24gb2YgdGhlIHN0cnVjdHVyZSBvZiB0aGUgWE1MIGVsZW1lbnRzLiBIb3cgZG8gd2Uga25vdyAtLSBvciBob3cgZG9lcyBhIHJlY2VpdmluZyBpbmZvcm1hdGlvbiBzeXN0ZW0ga25vdyAtLSB0aGF0IHRoZSBYTUwgZmlsZSBpcyBwcm9wZXJseSBzdHJ1Y3R1cmVkPyBTb2x1dGlvbjogRGVmaW5lIHRoZSBncmFtbWFyIG9mIHRoZSBYTUwuIE1ldGhvZDogVXNlIHRoZSBEb2N1bWVudCBUeXBlIERlZmluaXRpb24gKERURCkgbWV0YSBsYW5ndWFnZS4NCg0KQSBEVEQgaXMgYSBzZXQgb2YgcnVsZXMgdGhhdCBkZWZpbmVzIHRoZSBlbGVtZW50cyBhbmQgYXR0cmlidXRlcyB0aGF0IGNhbiBhcHBlYXIgd2l0aGluIGFuIFhNTCBkb2N1bWVudCBhbmQgdGhlaXIgc2VxdWVuY2UgYW5kIG5lc3RpbmcuIEFuIFhNTCBkb2N1bWVudCBpcyBjb25zaWRlcmVkIGEgdmFsaWQgZG9jdW1lbnQgaWYgaXQgY29uZm9ybXMgdG8gdGhlIGFzc29jaWF0ZWQgRFRELiBBIERURCBjYW4gYXBwZWFyIHdpdGhpbiBhbiBYTUwgZG9jdW1lbnQgKGludGVybmFsIERURCkgb3IgYmUgZGVmaW5lZCBpbiBhbiBleHRlcm5hbCBmaWxlIChleHRlcm5hbCBEVEQpIHdoaWNoIGFsbG93cyBmb3Igc2hhcmluZy4gQW4gaW50ZXJuYWwgRFREIGlzIG9ubHkgdmlzaWJsZSB3aXRoaW4gdGhlIGZpbGUgaW4gd2hpY2ggaXQgcmVzaWRlcywgbWVhbmluZyB0aGF0IG90aGVyIFhNTCBmaWxlcyBjYW5ub3QgdXNlIHRoZSBEVEQuIEEgRFREIHRoYXQgYXBwbGllcyB0byBzZXZlcmFsIFhNTCBkb2N1bWVudCBvciBpcyB1c2VkIHRvIGNvbW11bmljYXRlIGFuIFhNTCBzdGFuZGFyZCBzaG91bGQgYmUgcGxhY2VkIGludG8gYW4gZXh0ZXJuYWwgZmlsZSByYXRoZXIgdGhhbiBlbWJlZGRlZC4NCg0KQXNzdW1lIHRoYXQgdGhlIERURCBzaG93biBiZWxvdyBpcyBzdG9yZWQgaW4gdGhlIGZpbGUgKm5vdGUuZHRkKi4NCg0KYGBgIHhtbA0KPCFET0NUWVBFIG5vdGUgWw0KPCFFTEVNRU5UIG5vdGUgKHRvKyxmcm9tLGhlYWRpbmcsYm9keSk+DQo8IUVMRU1FTlQgdG8gKCNQQ0RBVEEpPg0KPCFFTEVNRU5UIGZyb20gKCNQQ0RBVEEpPg0KPCFFTEVNRU5UIGhlYWRpbmcgKCNQQ0RBVEEpPg0KPCFFTEVNRU5UIGJvZHkgKCNQQ0RBVEEpPg0KXT4NCmBgYA0KDQpUaGVuIHRoZSBYTUwgZG9jdW1lbnQgYmVsb3cgY2FuIHJlZmVyZW5jZSB0aGF0IGV4dGVybmFsIERURCBhbmQgZW5zdXJlIChkdXJpbmcgcGFyc2luZyBvZiB0aGUgWE1MKSB0aGF0IGl0IGNvbmZvcm1zIHRvIHRoZSBydWxlcyBvZiB0aGUgRFRELiBOb3RlIHRoZSBzeW50YXggb2YgdGhlIHJlZmVyZW5jaW5nLiBJdCBzdGFydHMgd2l0aCBgPCFET0NUWVBFYCBmb2xsb3dlZCBieSB0aGUgbmFtZSBvZiB0aGUgcm9vdCBlbGVtZW50ICgqXDxub3RlXD4qKSBpbiB0aGlzIGV4YW1wbGUpIGFuZCB0aGUga2V5d29yZCBgU1lTVEVNYCBhbmQgdGhlIHBhdGggdG8gdGhlIGZpbGUsIHdoaWNoIGNhbiBiZSBhIGxvY2FsIGZpbGUgb3IgYSBVUkwuDQoNCmBgYCB4bWwNCjw/eG1sIHZlcnNpb249IjEuMCIgZW5jb2Rpbmc9IlVURi04Ij8+DQoNCjwhRE9DVFlQRSBub3RlIFNZU1RFTSAiTm90ZS5kdGQiPg0KDQo8bm90ZT4NCiAgPHRvPkNocmlzPC90bz4NCiAgPGZyb20+VG9waGVyPC9mcm9tPg0KICA8aGVhZGluZz5SZW1pbmRlcjwvaGVhZGluZz4NCiAgPGJvZHk+DQogICBEb24ndCBmb3JnZXQgdGhlIG1lZXRpbmchDQogIDwvYm9keT4NCjwvbm90ZT4gDQpgYGANCg0KVG8gc3VtbWFyaXplLCB0aGUgRG9jdW1lbnQgVHlwZSBEZWZpbml0aW9uIChEVEQpIGNhbiBiZSBpbiBhIHNlcGFyYXRlIGZpbGUgb3IgZGVmaW5lZCBpbmxpbmUgYXQgdGhlIGJlZ2lubmluZyBvZiB0aGUgWE1MIGRvY3VtZW50LiBJdCBpcyB1c2VkIHRvIHZhbGlkYXRlIHRoZSBzdHJ1Y3R1cmUgb2YgdGhlIFhNTCBkb2N1bWVudCBhbmQgZGVmaW5lcyB0aGUgImdyYW1tYXIiIG9mIG91ciBYTUwgImxhbmd1YWdlIi4gVGhlIERURCBsYW5ndWFnZSBpcyB1c2VkIHRvIGRlZmluZSB0aGUgZ3JhbW1hciBvZiB0aGUgWE1MIHVzaW5nIEV4dGVuZGVkIEJhY2h1cy1OYXVyIEZvcm0gKEVCTkYpLiBBIERURCBpcyBvcHRpb25hbCAoYnV0IHJlY29tbWVuZGVkKSBhbHRob3VnaCByZXF1aXJlZCBpZiB0aGUgZG9jdW1lbnQgaXMgdG8gYmUgdmFsaWRhdGVkIGJ5IHRoZSBwYXJzZXIgcHJpb3IgdG8gcGFyc2luZy4NCg0KIyMjIERURDogQ2FyZGluYWxpdHkNCg0KQSBjYXJkaW5hbGl0eSBhZG9ybm1lbnQgbWF5IGZvbGxvdyBhbiBlbGVtZW50LiBJZiBubyBjYXJkaW5hbGl0eSBhZG9ybm1lbnQgaXMgcHJlc2VudCwgdGhlbiBpbXBsaWVkIGNhcmRpbmFsaXR5IGlzIGV4YWN0bHkgMSwgKmkuZS4qLCB0aGUgZWxlbWVudCBtdXN0IGFwcGVhciBhbmQgY2Fubm90IGFwcGVhciBtb3JlIHRoYW4gb25lLiBUaGVyZSBhcmUgdGhyZWUgYWRvcm5tZW50cyB0aGF0IG1vZGlmeSBjYXJkaW5hbGl0eToNCg0KLSAgIFxfXCpcXyBpbmRpY2F0aW5nIGEgY2FyZGluYWxpdHkgb2YgemVybyBvciBtb3JlLCAqaS5lLiosIHRoZSBlbGVtZW50IGNhbiBhcHBlYXIgYXMgb2Z0ZW4gYXMgZGVzaXJlZCBhbmQgcG9zc2libHkgbm90IGF0IGFsbCAob3B0aW9uYWwpDQotICAgKisqIGluZGljYXRpbmcgYSBjYXJkaW5hbGl0eSBvZiBvbmUgb3IgbW9yZSwgKmkuZS4qLCB0aGUgZWxlbWVudCBjYW4gYXBwZWFyIGFzIG9mdGVuIGFzIGRlc2lyZWQgYnV0IG11c3QgYXBwZWFyIGF0IGxlYXN0IG9uY2UgKG1hbmRhdG9yeSkNCi0gICA/IGluZGljYXRpbmcgYSBjYXJkaW5hbGl0eSBvZiB6ZXJvIG9yIG9uZSwgKmkuZS4qLCB0aGUgZWxlbWVudCBtYXkgbm90IGFwcGVhciBhdCBhbGwgb3IgYXQgbW9zdCBvbmNlDQoNClRoZXNlIGFkb3JubWVudHMgd2lsbCBiZWNvbWUgY2xlYXJlciBpbiB0aGUgc2VjdGlvbnMgdGhhdCBmb2xsb3cgd2hlbiB3ZSBsb29rIGF0IGV4YW1wbGVzLg0KDQojIyMgRFREOiBDb250ZW50DQoNCkFuIGVsZW1lbnQgdGhhdCBoYXMgYSB2YWx1ZSBpcyBpbmRpY2F0ZWQgd2l0aCBgPCFFTEVNRU5UIGZvbyAoI1BDREFUQSk+YC4gVGhlIHZhbHVlIGlzIHdoYXQgaXMgYmV0d2VlbiB0aGUgb3BlbmluZyBhbmQgY2xvc2luZyB0YWcsICplLmcuKiwgZm9yIHRoZSBlbGVtZW50ICpcPGZvb1w+Y29udGVudFw8L2Zvb1w+KiwgdGhlIHZhbHVlIGlzICJjb250ZW50Ii4NCg0KIyMjIERURDogU2VxdWVuY2UNCg0KQSBzZXF1ZW5jZSBpbiBYTUwgaXMgYSBhbiBlbGVtZW50IGZvbGxvd2VkIGJ5IGFub3RoZXIgZWxlbWVudC4gVGhlIGRpYWdyYW0gRFREIGJlbG93IHNob3dzIHRoZSBkaWZmZXJlbnQgcGFydHMgb2YgdGhlIERURCBhbmQgd2hhdCB0aGV5IHNwZWNpZnkuIFRoZSBjb21tYSBpbiB0aGUgZWxlbWVudCBkZWZpbml0aW9uIGZvciAqQXV0aG9yKiBpcyBhIHNlcXVlbmNlIG9mIGZvdXIgY2hpbGQgZWxlbWVudHMsIHdoaWNoIG11c3QgYXBwZWFyIGluIHRoYXQgb3JkZXIuDQoNClRoZSBEVEQgc3BlY2lmaWVzIHRoYXQgdGhlICJyb290IiBlbGVtZW50IGlzICpcPHJvb3RcPiogYW5kIHRoYXQgaXQgY29udGFpbnMgemVybyBvciBtb3JlICpcPEF1dGhvclw+KiBlbGVtZW50cyBhcyBpbmRpY2F0ZWQgYnkgdGhlIHNwZWNpZmljYXRpb24gYDwhRUxFTUVOVCByb290IChBdXRob3IqKT5gLiBUaGUgcGFyZW50aGVzaXMgc3BlY2lmaWVzIGNoaWxkIGVsZW1lbnRzLCBzbyB0aGUgKnJvb3QqIGVsZW1lbnQgaGFzIHplcm8gb3IgbW9yZSAoaW5kaWNhdGVkIGJ5IHRoZSBcX1wqXF8gYWZ0ZXIgKkF1dGhvciopICpBdXRob3IqIGVsZW1lbnRzLg0KDQpFYWNoICpBdXRob3IqIGVsZW1lbnQgY29uc2lzdHMgb2YgYSBzZXF1ZW5jZSBvZiBjaGlsZCBlbGVtZW50cywgYWxsIG9mIHdoaWNoIG11c3QgYXBwZWFyIGV4YWN0bHkgb25jZSAobm8gY2FyZGluYWxpdHkgYWRvcm5tZW50KSBpbiB0aGF0IHNlcXVlbmNlICh0aGUgY29tbWEgaW5kaWNhdGVzIGEgc2VxdWVuY2UpLg0KDQohW1NlcXVlbmNlIG9mIEVsZW1lbnRzIGluIGEgRFREXShkdGQtc2VxdWVuY2UuanBnKXt3aWR0aD0iNzUlIn0NCg0KQW4gWE1MIGRvY3VtZW50IHRoYXQgY29uZm9ybXMgdG8gdGhlIGFib3ZlIERURCB3b3VsZCBsb29rIGxpa2UgdGhpczoNCg0KYGBgIHhtbA0KPD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiPz4NCg0KPHJvb3Q+DQogIDxBdXRob3I+DQogICAgPGF1dGhvcklEPjMzNDk8L2F1dGhvcklEPg0KICAgIDxuYW1lPkNoYXJsZXMgTWVyY2VyPC9uYW1lPg0KICAgIDxvcmc+T2FrdG9uIFBhcnRuZXJzPC9vcmc+DQogICAgPGVtYWlsPmNtQG9ha3Rvbi5jb20uY2E8L2VtYWlsPg0KICA8L0F1dGhvcj4NCiAgPEF1dGhvcj4NCiAgICA8YXV0aG9ySUQ+MTEzMjwvYXV0aG9ySUQ+DQogICAgPG5hbWU+Q2luZHkgQ2hlbjwvbmFtZT4NCiAgICA8b3JnPkFscGVuIFByZXNzZTwvb3JnPg0KICAgIDxlbWFpbD5jaW5keUBhbHBlbi1wcmVzc2UuZGU8L2VtYWlsPg0KICA8L0F1dGhvcj4NCjwvcm9vdD4NCmBgYA0KDQpUaGUgY2hpbGQgZWxlbWVudHMgZGlyZWN0bHkgdW5kZXJuZWF0aCAqXDxBdXRob3JcPiogbXVzdCBhcHBlYXIgaW4gZXhhY3RseSB0aGF0IG9yZGVyIGFuZCBjYW5ub3QgYmUgb21pdHRlZCwgKmkuZS4qLCB0aGV5IGFyZSBtYW5kYXRvcnkgb3IgcmVxdWlyZWQuIExlYXZpbmcgb25lIG9mIHRoZSBjaGlsZCBlbGVtZW50cyBvdXQgb3IgY2hhbmdpbmcgdGhlIHNlcXVlbmNlIHdvdWxkIHJlc3VsdCBpbiBhbiBlcnJvciB3aGVuIHRoZSBYTUwgZG9jdW1lbnQgaXMgdmFsaWRhdGVkIChwZXJoYXBzIGR1cmluZyBwYXJzaW5nIHdoZW4gaXQgaXMgbG9hZGVkIGludG8gYW4gYXBwbGljYXRpb24gZm9yIHByb2Nlc3NpbmcpLg0KDQpUaGUgWE1MIGRvY3VtZW50IGJlbG93IGlzIGFsc28gdmFsaWQgZXZlbiB0aG91Z2ggaXQgY29udGFpbnMgbm8gY2hpbGQgZWxlbWVudHMgdW5kZXIgKlw8cm9vdFw+Ki4gVGhhdCBpcyB2YWxpZCBhcyB0aGUgY2FyZGluYWxpdHkgYWRvcm5tZW50IGZvciAqQXV0aG9yKiBpcyBcX1wqXF8gd2hpY2ggaW5kaWNhdGVzIHplcm8gb3IgbW9yZSBlbGVtZW50cyBvZiB0aGF0IHR5cGUuDQoNCmBgYCB4bWwNCjw/eG1sIHZlcnNpb249IjEuMCIgZW5jb2Rpbmc9IlVURi04Ij8+DQoNCjxyb290Pg0KPC9yb290Pg0KYGBgDQoNCklmIGFuIGVsZW1lbnQgbmVpdGhlciBjb250YWlucyBjaGlsZCBlbGVtZW50cyBub3IgYSB2YWx1ZSwgaXQgY2FuIGJlIHdyaXR0ZW4gaW4gc2hvcnQgaGFuZCB1c2luZyBhICJzZWxmLWNsb3NpbmcgdGFnIi4NCg0KYGBgIHhtbA0KPD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiPz4NCg0KPHJvb3QvPg0KYGBgDQoNClRoZSBYTUwgZG9jdW1lbnRzIGJlbG93IGFyZSBub3QgdmFsaWQgZm9yIHRoZSByZWFzb24gbGlzdGVkIGZvciBlYWNoLg0KDQojIyMjIEV4YW1wbGUgSTogTWlzc2luZyBFbGVtZW50DQoNCk5vdCB2YWxpZCBhcyB0aGUgKlw8b3JnXD4qIGVsZW1lbnQgaXMgbWlzc2luZy4NCg0KYGBgIHhtbA0KPHJvb3Q+DQogIDxBdXRob3I+DQogICAgPGF1dGhvcklEPjMzNDk8L2F1dGhvcklEPg0KICAgIDxuYW1lPkNoYXJsZXMgTWVyY2VyPC9uYW1lPg0KICAgIDxlbWFpbD5jbUBvYWt0b24uY29tLmNhPC9lbWFpbD4NCiAgPC9BdXRob3I+DQo8L3Jvb3Q+DQpgYGANCg0KIyMjIyBFeGFtcGxlIElJOiBPdXQgb2YgU2VxdWVuY2UNCg0KTm90IHZhbGlkIGFzIHRoZSBjaGlsZCBlbGVtZW50cyBmb3IgKlw8QXV0aG9yXD4qIGFyZSBub3QgaW4gdGhlIG9yZGVyIGxpc3RlZCBpbiB0aGUgRFRELg0KDQpgYGAgeG1sDQo8cm9vdD4NCiAgPEF1dGhvcj4NCiAgICA8YXV0aG9ySUQ+MzM0OTwvYXV0aG9ySUQ+DQogICAgPG5hbWU+Q2hhcmxlcyBNZXJjZXI8L25hbWU+DQogICAgPGVtYWlsPmNtQG9ha3Rvbi5jb20uY2E8L2VtYWlsPg0KICAgIDxvcmc+T2FrdG9uIFBhcnRuZXJzPC9vcmc+DQogIDwvQXV0aG9yPg0KPC9yb290Pg0KYGBgDQoNCiMjIyMgRXhhbXBsZSBJSUk6IFRhZyBOYW1lcyBJbmNvcnJlY3QNCg0KTm90IHZhbGlkIGFzIHRoZSAqXDxBdXRob3JcPiogdGFnIGlzIG1pc3NwZWxsZWQgKHdyb25nIGNhcGl0YWxpemF0aW9uKS4NCg0KYGBgIHhtbA0KPHJvb3Q+DQogIDxBdXRob3I+DQogICAgPGF1dGhvcklEPjMzNDk8L2F1dGhvcklEPg0KICAgIDxuYW1lPkNoYXJsZXMgTWVyY2VyPC9uYW1lPg0KICAgIDxlbWFpbD5jbUBvYWt0b24uY29tLmNhPC9lbWFpbD4NCiAgICA8b3JnPk9ha3RvbiBQYXJ0bmVyczwvb3JnPg0KICA8L2F1dGhvcj4NCjwvcm9vdD4NCmBgYA0KDQojIyMjIEV4YW1wbGUgSVY6IE1pc3NpbmcgQ2xvc2luZyBUYWcNCg0KTm90IHZhbGlkIGFzIHRoZSAqXDxBdXRob3JcPiogdGFnIGlzIG5vdCBjbG9zZWQsICppLmUuKiwgaXQgaXMgbWlzc2luZyB0aGUgY2xvc2luZyB0YWcgKlw8L0F1dGhvclw+Kg0KDQpgYGAgeG1sDQo8cm9vdD4NCiAgPEF1dGhvcj4NCiAgICA8YXV0aG9ySUQ+MzM0OTwvYXV0aG9ySUQ+DQogICAgPG5hbWU+Q2hhcmxlcyBNZXJjZXI8L25hbWU+DQogICAgPGVtYWlsPmNtQG9ha3Rvbi5jb20uY2E8L2VtYWlsPg0KICAgIDxvcmc+T2FrdG9uIFBhcnRuZXJzPC9vcmc+DQo8L3Jvb3Q+DQpgYGANCg0KIyMjIyBFeGFtcGxlIFY6IEluY29ycmVjdCBOZXN0aW5nDQoNCk5vdCB2YWxpZCBhcyB0aGUgKlw8QXV0aG9yXD4qIHRhZyBpcyBjbG9zZWQgYmVmb3JlIHRoZSAqXDxvcmdcPiogdGFnLCBzbyB0aGUgbmVzdGluZyBpcyBpbmNvcnJlY3QuDQoNCmBgYCB4bWwNCjxyb290Pg0KICA8QXV0aG9yPg0KICAgIDxhdXRob3JJRD4zMzQ5PC9hdXRob3JJRD4NCiAgICA8bmFtZT5DaGFybGVzIE1lcmNlcjwvbmFtZT4NCiAgICA8ZW1haWw+Y21Ab2FrdG9uLmNvbS5jYTwvZW1haWw+DQogICAgPG9yZz5PYWt0b24gUGFydG5lcnMNCiAgPC9BdXRob3I+PC9vcmc+DQo8L3Jvb3Q+DQpgYGANCg0KTmF0dXJhbGx5LCBtb3N0IG9mIHRoZXNlIGVycm9ycyB3b3VsZCBub3Qgb2NjdXIgaW4gcHJhY3RpY2UgaWYgdGhlIFhNTCBpcyBnZW5lcmF0ZWQgcHJvZ3JhbW1hdGljYWxseS4NCg0KIyMjIERURDogTWl4ZWQgQ29udGVudA0KDQpUaGUgY29tbWEgKCwpIG9wZXJhdG9yIHNwZWNpZmllcyBzZXF1ZW5jZSBpbiBhIERURC4gT24gdGhlIG90aGVyIGhhbmQsIHRoZSBvciAoXHwpIG9wZXJhdG9yIHNwZWNpZmllcyBhIGNob2ljZS4gSWYgdGhlIERURCBjb250YWlucyAqKGZvb1x8YmFyKSogaXQgbWVhbnMgdGhhdCBlaXRoZXIgdGhlIGVsZW1lbnQgKmZvbyogb3IgdGhlIGVsZW1lbnQgKmJhciogbXVzdCBhcHBlYXIuIFNvLCB0aGUgRFREIGJlbG93IHNwZWNpZmllcyB0aGF0IGFuICpBdXRob3IqIGVsZW1lbnQgaGFzIGEgbWFuZGF0b3J5IGNoaWxkIGVsZW1lbnQgKmF1dGhvcklEKiBmb2xsb3dlZCBieSBhbm90aGVyIG1hbmRhdG9yeSBjaGlsZCBlbGVtZW50ICpuYW1lKiBmb2xsb3dlZCBieSBlaXRoZXIgYW4gKm9yZyogb3IgYW4gKmVtYWlsKiBlbGVtZW50LiBBY3R1YWxseSwgc2luY2UgaXQgc3RhdGVzIFxfKG9yZ1x8ZW1haWwpXCpcXyBpdCBtZWFucyB6ZXJvIG9mIG1vcmUgb2YgKihvcmdcfGVtYWlsKSouDQoNCiFbU2VxdWVuY2Ugb2YgRWxlbWVudHMgaW4gYSBEVERdKGR0ZC1taXhlZC1jb250ZW50LmpwZyl7d2lkdGg9Ijc1JSJ9DQoNCkFuIFhNTCBkb2N1bWVudCB0aGF0IGNvbmZvcm1zIHRvIHRoZSBhYm92ZSBEVEQgd291bGQgbG9vayBsaWtlIHRoaXM6DQoNCmBgYCB4bWwNCjw/eG1sIHZlcnNpb249IjEuMCIgZW5jb2Rpbmc9IlVURi04Ij8+DQoNCjxyb290Pg0KICA8QXV0aG9yPg0KICAgIDxhdXRob3JJRD4zMzQ5PC9hdXRob3JJRD4NCiAgICA8bmFtZT5DaGFybGVzIE1lcmNlcjwvbmFtZT4NCiAgICA8b3JnPk9ha3RvbiBQYXJ0bmVyczwvb3JnPg0KICAgIDxlbWFpbD5jbUBvYWt0b24uY29tLmNhPC9lbWFpbD4NCiAgICA8ZW1haWw+YzQ0M0B5YWhvby5jb208L2VtYWlsPg0KICA8L0F1dGhvcj4NCiAgPEF1dGhvcj4NCiAgICA8YXV0aG9ySUQ+MTEzMjwvYXV0aG9ySUQ+DQogICAgPG5hbWU+Q2luZHkgQ2hlbjwvbmFtZT4NCiAgICA8ZW1haWw+Y2luZHlAYWxwZW4tcHJlc3NlLmRlPC9lbWFpbD4NCiAgPC9BdXRob3I+DQogIDxBdXRob3I+DQogICAgPGF1dGhvcklEPjMzNDk8L2F1dGhvcklEPg0KICAgIDxuYW1lPk1hcmsgV2VpbGw8L25hbWU+DQogIDwvQXV0aG9yPg0KICAgIDxBdXRob3I+DQogICAgPGF1dGhvcklEPjg3NjwvYXV0aG9ySUQ+DQogICAgPG5hbWU+QWxsaXNvbiBLaGFuPC9uYW1lPg0KICAgIDxvcmc+QUsgQXNzb2NpYXRlczwvb3JnPg0KICAgIDxlbWFpbD5jaW5keUBhbHBlbi1wcmVzc2UuZGU8L2VtYWlsPg0KICAgIDxvcmc+T2thdG9uIENvbGxlZ2U8L29yZz4NCiAgPC9BdXRob3I+DQo8L3Jvb3Q+DQpgYGANCg0KSWYgd2Ugd2FudGVkIGF0IGxlYXN0IG9uZSAqb3JnKiBhbmQgb25lICplbWFpbCogd2Ugd291bGQgaGF2ZSBzcGVjaWZpZWQgdGhlIERURCBhcyBmb2xsb3dzLCB3aGVyZSBpdCB3b3VsZCBiZSBvbmUgb2YgKmF1dGhvcklEKiwgKm5hbWUqLCAqb3JnKiwgKmVtYWlsKiBpbiB0aGF0IHNlcXVlbmNlIGZvbGxvd2VkIGJ5IHplcm8gb3IgbW9yZSBjb21iaW5hdGlvbnMgb2YgKm9yZyogYW5kICplbWFpbCouDQoNCmBgYCB4bWwNCjwhRE9DVFlQRSByb290IFsNCiAgPCFFTEVNRU5UIHJvb3QgKEF1dGhvciopPg0KICA8IUVMRU1FTlQgQXV0aG9yDQogICAgICAoYXV0aG9ySUQsbmFtZSxvcmcsZW1haWwsKG9yZ3xlbWFpbCkqKT4NCiAgPCFFTEVNRU5UIGF1dGhvcklEICgjUENEQVRBKT4gICANCiAgPCFFTEVNRU5UIG5hbWUgKCNQQ0RBVEEpPg0KICA8IUVMRU1FTlQgb3JnICgjUENEQVRBKT4NCiAgPCFFTEVNRU5UIGVtYWlsICgjUENEQVRBKT4NCl0+DQpgYGANCg0KIyMjIERURDogQXR0cmlidXRlcw0KDQpFbGVtZW50cyBtYXkgKG9wdGlvbmFsbHkpIGhhdmUgYXR0cmlidXRlcyB3aGljaCBhcmUga2V5L3ZhbHVlIHBhaXJzIHNldCBpbiB0aGUgdGFnIG9mIHRoZSBlbGVtZW50LiBBdHRyaWJ1dGVzIGFyZSBjaGFyYWN0ZXIgc3RyaW5ncyBhbmQgc2hvdWxkIGJlIGVuY2xvc2VkIGluIHF1b3RlcyAoYWx0aG91Z2ggdGhhdCBpcyBub3Qgc3RyaWN0bHkgcmVxdWlyZWQgaWYgdGhleSBhcmUgbnVtYmVycykuDQoNClRoZSBleGFtcGxlIGJlbG93IHVzZXMgYW4gYXR0cmlidXRlIGZvciAqYXV0aG9ySUQqIHJhdGhlciB0aGFuIHJlcHJlc2VudGluZyBpdCBhcyBhIGNoaWxkIGVsZW1lbnQuIFRoaXMgaXMgY29tbW9uIGZvciAia2V5IHZhbHVlcyIuIEFuIGF0dHJpYnV0ZSBjYW4gaGF2ZSBhIGRhdGEgdHlwZSBvZiAqQ0RBVEEqIChjaGFyYWN0ZXIgZGF0YSksICpJRCogKHVuaXF1ZSB2YWx1ZSksICpJRFJFRiogKHJlZmVyZW5jZSB0byBhbiAqSUQqKSwgb3IgYW4gZW51bWVyYXRlZCB0b2tlbiAoc2VlIGJlbG93KSwgYW1vbmcgb3RoZXJzLiBBTiBhdHRyaWJ1dGUgaXMgc3BlY2lmaWVkIHdpdGggYDwhQVRUTElTVCBlbGVtZW50IGF0dHJpYnV0ZSBkZWZhdWx0PmAsIHNvIGA8IUFUVExJU1QgQXV0aG9yIGF1dGhvcklEIENEQVRBICNSRVFVSVJFRD5gIHNwZWNpZmllcyB0aGUgKG1hbmRhdG9yeSkgYXR0cmlidXRlICphdXRob3JJRCogZm9yIHRoZSBlbGVtZW50ICpBdXRob3IuKg0KDQpgYGAgeG1sDQo8P3htbCB2ZXJzaW9uPSIxLjAiIGVuY29kaW5nPSJVVEYtOCI/Pg0KDQo8IURPQ1RZUEUgcm9vdCBbDQogIDwhRUxFTUVOVCByb290IChBdXRob3IqKT4NCiAgPCFFTEVNRU5UIEF1dGhvcg0KICAgICAgKG5hbWUsb3JnLGVtYWlsKT4NCiAgPCFBVFRMSVNUIEF1dGhvciBhdXRob3JJRCBDREFUQSAjUkVRVUlSRUQ+ICAgDQogIDwhRUxFTUVOVCBuYW1lICgjUENEQVRBKT4NCiAgPCFFTEVNRU5UIG9yZyAoI1BDREFUQSk+DQogIDwhRUxFTUVOVCBlbWFpbCAoI1BDREFUQSk+DQpdPg0KDQo8cm9vdD4NCiAgPEF1dGhvciBhdXRob3JJRCA9ICIzMzQ5Ij4NCiAgICA8bmFtZT5DaGFybGVzIE1lcmNlcjwvbmFtZT4NCiAgICA8b3JnPk9ha3RvbiBQYXJ0bmVyczwvb3JnPg0KICAgIDxlbWFpbD5jNDQzQHlhaG9vLmNvbTwvZW1haWw+DQogIDwvQXV0aG9yPg0KPC9yb290Pg0KYGBgDQoNCkVsZW1lbnQgYXR0cmlidXRlcyBjYW4gaGF2ZSB0aGUgZm9sbG93aW5nIGNvbnN0cmFpbnRzOg0KDQotICAgKiNSRVFVSVJFRCogLS0gYSB2YWx1ZSBmb3IgdGhlIGF0dHJpYnV0ZSBtdXN0IGJlIHByb3ZpZGVkDQotICAgKiNGSVhFRCogLS0gYSBmaXhlZCB2YWx1ZSB0aGF0IGNhbm5vdCBiZSBjaGFuZ2VkIGFuZCBhbGwgYXR0cmlidXRlcyBmb3IgYWxsIGVsZW1lbnQgaW5zdGFuY2VzIGhhdmUgdGhhdCB2YWx1ZTsgYSB2YWx1ZSBtdXN0IGJlIHNwZWNpZmllZCBhZnRlciB0aGlzIGtleXdvcmQNCi0gICAqI0lNUExJRUQqIC0tIGFuIG9wdGlvbmFsIGF0dHJpYnV0ZSB0aGF0IG1heSBvciBtYXkgbm90IGJlIHByZXNlbnQNCg0KTm90ZSB0aGF0IHRoZXNlIGtleXdvcmRzIG11c3QgYmUgaW4gdXBwZXIgY2FzZSBhbmQgb25seSBvbmUgY2FuIGJlIHNwZWNpZmllZC4NCg0KIyMjIERURDogRW51bWVyYXRlZCBUb2tlbnMNCg0KWE1MIHJlY29nbml6ZXMgY2hhcmFjdGVycyBhcyB0aGUgZGVmYXVsdCB0eXBlIG9mIGFueSB2YWx1ZS4gSW4gYWRkaXRpb24sIFhNTCBzdXBwb3J0cyB2YWx1ZSBzZXRzIG9yIGNhdGVnb3JpY2FsIGF0dHJpYnV0ZXMgd2hpY2ggZHJhdyB0aGVpciB2YWx1ZXMgZnJvbSBhIHByZWRlZmluZWQgc2V0IG9mIHZhbHVlcy4gVG8gZW5mb3JjZSBhIHZhbHVlIHNldCAoZW51bWVyYXRlZCB0b2tlbnMpIGl0IG11c3QgYmUgZGVmaW5lZCBhcyBhbiBhdHRyaWJ1dGUgYXMgbm8gdmFsaWRhdGlvbiBjYW4gb2NjdXIgZm9yIHBhcnNlZCBjaGFyYWN0ZXIgZGF0YSAoKiNQQ0RBVEEqKS4NCg0KSW4gdGhlIFhNTCBleGFtcGxlIGJlbG93LCB0aGUgKkxlYXJuaW5nQXNzZXQqIGVsZW1lbnQgaGFzIHRoZSBhdHRyaWJ1dGUgKmRpZmZpY3VsdHkqLiBJdCBtdXN0IGJlIG9uZSBvZiAqKGVhc3lcfG1lZGl1bVx8ZGlmZmljdWx0KSogYW5kLCBpZiBub3QgcHJlc2VudCwgZGVmYXVsdHMgdG8gdGhlIHZhbHVlICoiZWFzeSIqLiBTZXR0aW5nIHRoZSBhdHRyaWJ1dGUgdG8gYW55IHZhbHVlIG90aGVyIHRoYW4gdGhlIG9uZXMgZGVmaW5lZCByZXN1bHRzIGluIGEgdmFsaWRhdGlvbiBlcnJvci4NCg0KYGBgIHhtbA0KPCFET0NUWVBFIHJvb3QgWw0KPCFFTEVNRU5UIHJvb3QgKExlYXJuaW5nQXNzZXQqKT4NCjwhRUxFTUVOVCBMZWFybmluZ0Fzc2V0ICjigKYpPg0KPCFBVFRMSVNUIExlYXJuaW5nQXNzZXQgZGlmZmljdWx0eSANCihlYXN5fG1lZGl1bXxkaWZmaWN1bHQpICJlYXN5Ij4NCi4uLg0KXT4NCg0KPHJvb3Q+DQogIDxMZWFybmluZ0Fzc2V0IGRpZmZpY3VsdHkgPSAibWVkaXVtIj4NCiAgICAuLi4NCiAgPC9MZWFybmluZ0Fzc2V0Pg0KICA8TGVhcm5pbmdBc3NldD4NCiAgICAuLi4NCiAgPC9MZWFybmluZ0Fzc2V0Pg0KDQogIC4uLg0KPC9yb290Pg0KYGBgDQoNCiMjIyBMZWFybmluZyBDaGVja3BvaW50IEkNCg0KIyMjIyBMQyBJLjENCg0KQXMgYW4gZXhlcmNpc2UsIGJ1aWxkIGFuIFhNTCBkb2N1bWVudCB0aGF0IGNvbmZvcm1zIHRvIHRoZSBEVEQgYmVsb3cuDQoNCmBgYCB4bWwNCjwhRE9DVFlQRSByb290IFsNCiAgPCFFTEVNRU5UIHJvb3QgKEF1dGhvciopPg0KICA8IUVMRU1FTlQgQXV0aG9yDQogICAgICAoYXV0aG9ySUQsbmFtZSwob3JnK3xlbWFpbCkrPg0KICA8IUVMRU1FTlQgYXV0aG9ySUQgKCNQQ0RBVEEpPiAgIA0KICA8IUVMRU1FTlQgbmFtZSAoI1BDREFUQSk+DQogIDwhRUxFTUVOVCBvcmcgKCNQQ0RBVEEpPg0KICA8IUVMRU1FTlQgZW1haWwgKCNQQ0RBVEEpPg0KXT4NCmBgYA0KDQojIyMjIExDIEkuMg0KDQpJZGVudGlmeSB0aGUgdHdvIGVycm9ycyBwcmVzZW50IGluIHRoZSBEVEQgYW5kIGNvcnJlc3BvbmRpbmcgWE1MLiBIb3cgd291bGQgeW91IGZpeCB0aGUgRFREIHNvIHRoYXQgaXQgdmFsaWRhdGVzIHRoZSBYTUw/DQoNCmBgYCB4bWwNCjw/eG1sIHZlcnNpb249IjEuMCIgZW5jb2Rpbmc9IlVURi04Ij8+DQoNCjwhRE9DVFlQRSBtc2cNCls8IUVMRU1FTlQgbm90ZSAodG8rLGZyb20saGVhZGluZyxib2R5Pyk+DQo8IUVMRU1FTlQgdG8gKCNQQ0RBVEEpPg0KPCFFTEVNRU5UIGZyb20gKCNQQ0RBVEEpPg0KPCFFTEVNRU5UIGhlYWRpbmcgKCNQQ0RBVEEpPg0KPCFFTEVNRU5UIGJvZHkgKCNQQ0RBVEEpPg0KXT4NCg0KPG5vdGU+ICANCiAgPGZyb20+Q2hyaXM8L2Zyb20+DQogIDx0bz5Ub3BoZXI8L3RvPg0KICA8aGVhZGluZz5SZW1pbmRlcjwvaGVhZGluZz4NCiAgPGJvZHk+DQogICBNZWV0aW5nIHRvZGF5IGF0IDEyIGluIFdWSDE2NiENCiAgPC9ib2R5Pg0KPC9ub3RlPiANCmBgYA0KDQojIyMjIExDIEkuMg0KDQpBICpcfCogaW5kaWNhdGVzIHRoYXQgb25lIG9mIHNldmVyYWwgZWxlbWVudHMgY2FuIGJlIGluY2x1ZGVkLiBTbywgaW4gdGhlIGV4YW1wbGUgYmVsb3cgd2hpY2ggZWxlbWVudHMgbXVzdCBiZSBwcmVzZW50Pw0KDQpgYGAgeG1sDQo8P3htbCB2ZXJzaW9uPSIxLjAiIGVuY29kaW5nPSJVVEYtOCI/Pg0KDQo8IURPQ1RZUEUgbm90ZXMgWw0KPCFFTEVNRU5UIG5vdGVzIChub3RlKyk+DQo8IUVMRU1FTlQgbm90ZSAocmVjaXBpZW50cyxmcm9tLChib2R5fGltYWdlKSk+DQo8IUVMRU1FTlQgcmVjaXBpZW50cyAodG8rKT4NCjwhRUxFTUVOVCB0byAoI1BDREFUQSk+DQo8IUVMRU1FTlQgZnJvbSAoI1BDREFUQSk+DQo8IUVMRU1FTlQgaGVhZGluZyAoI1BDREFUQSk+DQo8IUVMRU1FTlQgYm9keSAoI1BDREFUQSk+DQpdPg0KDQo8bm90ZXM+DQogIDxub3RlPg0KICAgIDxyZWNpcGllbnRzPg0KICAgICAgPHRvPlRvcGhlcjwvdG8+DQogICAgICA8dG8+UGF0PC90bz4NCiAgICA8L3JlY2lwaWVudHM+DQogICAgPGZyb20+Q2hyaXM8L2Zyb20+DQogICAgPGJvZHk+DQogICAgICBNZWV0aW5nIHRvZGF5IGF0IDEyIGluIFdWSDE2NiENCiAgICA8L2JvZHk+DQogIDwvbm90ZT4gDQogIA0KICA8bm90ZT4gIA0KICAgIDxyZWNpcGllbnRzPg0KICAgICAgPHRvPkVyaW48L3RvPg0KICAgICAgPHRvPkJhcmJhcmE8L3RvPg0KICAgIDwvcmVjaXBpZW50cz4NCiAgICA8ZnJvbT5DaHJpczwvZnJvbT4NCiAgICA8aW1hZ2U+DQogICAgICByZWNlaXB0LmpwZw0KICAgIDwvaW1hZ2U+DQogIDwvbm90ZT4gDQo8L25vdGVzPg0KYGBgDQoNCiMjIyMgTEMgSS4zDQoNCkNyZWF0ZSBhIERURCBmb3IgdGhlIFhNTCAoW2NvdXJzZXMueG1sXShjb3Vyc2VzLnhtbCkgYmVsb3cgYXNzdW1pbmcgdGhlIGZvbGxvd2luZyAicnVsZXMiOg0KDQotICAgVGhlcmUgaXMgYXQgbGVhc3Qgb25lIHNlY3Rpb24gcGVyIGNvdXJzZSBidXQgbm8gbGltaXQNCi0gICB0aW1lIGFuZCBsb2NhdGlvbiBtYXkgYXBwZWFyIGluIGFueSBvcmRlciBhbmQgbG9jYXRpb24gaXMgb3B0aW9uYWwNCi0gICBjb3Vyc2VOdW1iZXIgYW5kIGNybiBhcmUgcmVxdWlyZWQga2V5IGF0dHJpYnV0ZXMgdGhhdCBtdXN0IGJlIHVuaXF1ZQ0KLSAgIGVhY2ggY291cnNlIG1heSBvciBtYXkgbm90IHJ1biBhbmQgd2hldGhlciBpdCdzIHJ1bm5pbmcgc2hvdWxkIGJlIHNpZ25hbGVkIHdpdGggYSBCb29sZWFuIGF0dHJpYnV0ZQ0KLSAgIGNhbXB1cyBpcyByZXN0cmljdGVkIHRvIHtPbmxpbmUsQm9zdG9uLFZhbmNvdXZlcixTZWF0dGxlfSBhbmQgZGVmYXVsdHMgdG8gIkJvc3RvbiIgaWYgbm90IHNwZWNpZmllZA0KDQpgYGAgeG1sDQo8P3htbCB2ZXJzaW9uPSIxLjAiIGVuY29kaW5nPSJVVEYtOCI/Pg0KDQo8Y291cnNlcz4NCiAgPGNvdXJzZSBjb3Vyc2VOdW1iZXI9IkNTNTIwMCI+DQogICAgPHRpdGxlPkRhdGFiYXNlIE1hbmFnZW1lbnQgU3lzdGVtczwvdGl0bGU+DQogICAgPGNyZWRpdHM+NC4wPC9jcmVkaXRzPg0KICAgIDxzZWN0aW9ucz4NCiAgICAgIDxzZWN0aW9uIGNybj0iMzc3ODY4IiBtb2RhbGl0eT0iVHJhZGl0aW9uYWwiPg0KICAgICAgICA8ZGF5cz5NUjwvZGF5cz4NCiAgICAgICAgPHRpbWU+MTE6NDVhbSAtIDE6MjVwbTwvdGltZT4NCiAgICAgICAgPGxvY2F0aW9uPlJZMzAxPC9sb2NhdGlvbj4NCiAgICAgIDwvc2VjdGlvbj4NCiAgICAgIDxzZWN0aW9uIGNybj0iNTU2MjI0IiBtb2RhbGl0eT0iRmxpcHBlZCI+DQogICAgICAgIDxkYXlzPlI8L2RheXM+DQogICAgICAgIDx0aW1lPjY6MDBwbSAtIDk6MTVwbTwvdGltZT4NCiAgICAgICAgPGxvY2F0aW9uPjEwMzwvbG9jYXRpb24+DQogICAgICAgIDxjYW1wdXM+VmFuY291dmVyPC9jYW1wdXM+DQogICAgICA8L3NlY3Rpb24+DQogICAgICA8c2VjdGlvbiBjcm49IjgwOTQ0NSIgbW9kYWxpdHk9Ik9ubGluZSIgLz4NCiAgICA8L3NlY3Rpb25zPg0KICA8L2NvdXJzZT4NCjwvY291cnNlcz4NCmBgYA0KDQojIyBCZXN0IFByYWN0aWNlcw0KDQpSZXBlYXRlZCBlbGVtZW50cyBhcmUgb2Z0ZW4gYmVzdCBncm91cGVkIHVuZGVyIGEgY29tbW9uIHBhcmVudCBlbGVtZW50IGFzIGl0IG1ha2VzIGl0IGVhc2llciB0byBwYXJzZS4NCg0KYGBgIHhtbA0KPD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiPz4NCg0KPCFET0NUWVBFIG5vdGUgWw0KPCFFTEVNRU5UIG5vdGUgKHJlY2lwaWVudHMsZnJvbSxoZWFkaW5nLGJvZHkpPg0KPCFFTEVNRU5UUyByZWNpcGllbnRzICh0byspPg0KPCFFTEVNRU5UIHRvICgjUENEQVRBKT4NCjwhRUxFTUVOVCBmcm9tICgjUENEQVRBKT4NCjwhRUxFTUVOVCBoZWFkaW5nICgjUENEQVRBKT4NCjwhRUxFTUVOVCBib2R5ICgjUENEQVRBKT4NCl0+DQoNCjxub3RlPg0KICA8cmVjaXBpZW50cz4NCiAgICA8dG8+VG9waGVyPC90bz4NCiAgICA8dG8+UGF0PC90bz4NCiAgICA8dG8+TWFyaWE8L3RvPg0KICA8L3JlY2lwaWVudHM+DQogIDxmcm9tPkNocmlzPC9mcm9tPg0KICA8aGVhZGluZz5SZW1pbmRlcjwvaGVhZGluZz4NCiAgPGJvZHk+DQogICBNZWV0aW5nIHRvZGF5IGF0IDEyIGluIFdWSDE2NiENCiAgPC9ib2R5Pg0KPC9ub3RlPiANCmBgYA0KDQojIyBQYXJzaW5nIGFuIFhNTCBEb2N1bWVudA0KDQpBbiBYTUwgZG9jdW1lbnQgbXVzdCBnZW5lcmFsbHkgYmUgdmFsaWQgaW4gb3JkZXIgdG8gYmUgZWFzaWx5IHBhcnNhYmxlIGJ5IGEgcHJvZ3JhbSwgKmkuZS4qLCBjb25mb3JtIHRvIHJ1bGVzICgiZ3JhbW1hciIpIHByZXNjcmliZWQgYnkgdGhlIERURC4gQW4gWE1MIGRvY3VtZW50IGNhbiBiZSB3ZWxsLWZvcm1lZCB3aXRob3V0IGJlaW5nIHZhbGlkLCAqaS5lLiosIGVhY2ggZWxlbWVudCBoYXMgYSBwcm9wZXJseSBuZXN0ZWQgc3RhcnQgYW5kIGVuZCB0YWcgYnV0IG1heSBvciBtYXkgbm90IGhhdmUgYSBEVEQgbm90IGNvbmZvcm0gdG8gaXQuIEluIHByYWN0aWNlLCBtYW55IFhNTCBmaWxlcyBkbyBub3QgaGF2ZSBhIERURCB3aGljaCBjYW4gbWFrZSBwYXJzaW5nIChyZWFkaW5nKSB0aGVtIGNoYWxsZW5naW5nIGFuZCBtdWNoIG1vcmUgZGlmZmljdWx0Lg0KDQojIyMgQXBwcm9hY2hlcyB0byBQYXJzaW5nDQoNClRoZXJlIGFyZSB0d28gZ2VuZXJhbCBhcHByb2FjaGVzIHRvICJwYXJzaW5nIiBhbiBYTUwgZG9jdW1lbnQ6IERPTSB2cyBTQVggcGFyc2luZy4gSW4gRE9NIChEb2N1bWVudCBPYmplY3QgTW9kZWwpIHBhcnNpbmcsIHRoZSBlbnRpcmUgWE1MIGRvY3VtZW50IGlzIGNvbnZlcnRlZCB0byBhbiBpbi1tZW1vcnkgdHJlZSBkYXRhIHN0cnVjdHVyZWQgYW5kIHRodXMgcmVzaWRlcyBmdWxseSBpbiBtZW1vcnkgd2hlcmUgaXQgY2FuIGJlIHRyYXZlcnNlZCBub2RlLWJ5LW5vZGUgb3IgdmlhIFhQYXRoLiBJbiBTQVggKFNpbXBsZSBBUEkgZm9yIFhNTCkgcGFyc2luZyBvbmx5IG9uZSBlbGVtZW50IGlzIHBhcnNlZCBhbmQgbG9hZGVkIGludG8gbWVtb3J5IGF0IGEgdGltZSBhbmQgYSBjYWxsYmFjayBmdW5jdGlvbiByZWdpc3RlcmVkIGJ5IHRoZSBwcm9jZXNzaW5nIHByb2dyYW0gaXMgY2FsbGVkIGZvciBlYWNoIGVsZW1lbnQgYXMgaXQgaXMgZW5jb3VudGVyZWQuIFNBWCBwYXJzaW5nIGlzIHByZWZlcmFibGUgZm9yIHZlcnkgbGFyZ2UgWE1MIGRvY3VtZW50cywgYnV0IGl0IGlzIG1vcmUgZGlmZmljdWx0IHRvIHBhcnNlIGFzIGNvbnRleHQgaXMgbG9zdCBhbmQgWFBhdGggY2Fubm90IGJlIHVzZWQuDQoNCldpdGggU0FYIHBhcnNpbmcsIGV2ZW50cyBhcmUgdHJpZ2dlcmVkIHdoZW4gdGhlIFhNTCBpcyBiZWluZyBwYXJzZWQuIFdoZW4gdGhlIHBhcnNlciBpcyBwYXJzaW5nIHRoZSBYTUwsIGFuZCBlbmNvdW50ZXJzIGEgc3RhcnQgdGFnICgqZS5nLiosICpcPHNvbWV0aGluZ1w+KiksIHRoZW4gaXQgdHJpZ2dlcnMgYW4gZXZlbnQgYW5kIGNhbGxzIGEgcHJvY2Vzc2luZyBmdW5jdGlvbiB0aGF0IGlzIHJlZ2lzdGVyZWQgZm9yIHRoYXQgdGFnLCBzbyBkaWZmZXJlbnQgdGFncyBtYXkgaGF2ZSBkaWZmZXJlbnQgZXZlbnQgaGFuZGxlaW5nIGZ1bmN0aW9ucy4gU2ltaWxhcmx5LCB3aGVuIHRoZSBlbmQgb2YgdGhlIHRhZyBpcyBtZXQgd2hpbGUgcGFyc2luZyAoKlw8L3NvbWV0aGluZ1w+KiksIGl0IHRyaWdnZXJzIGFub3RoZXIgZXZlbnQgYW5kIGNhbGxzIGEgcmVnaXN0ZXJlZCBjYWxsYmFjayBmdW5jdGlvbi4gVXNpbmcgYSBTQVggcGFyc2VyIGltcGxpZXMgeW91IG5lZWQgdG8gaGFuZGxlIHRoZXNlIGV2ZW50cyBhbmQgbWFrZSBzZW5zZSBvZiB0aGUgZGF0YSByZXR1cm5lZCB3aXRoIGVhY2ggZXZlbnQsIGJ1dCB5b3UgZG8gbm90IGhhdmUgImNvbnRleHQiIHVubGVzcyB5b3UgdHJhY2sgaXQuDQoNCk9uIHRoZSBvdGhlciBoYW5kLCBpbiBET00gcGFyc2luZywgdGhlcmUgYXJlIG5vIGV2ZW50cyB0cmlnZ2VyZWQgd2hpbGUgcGFyc2luZy4gVGhlIGVudGlyZSBYTUwgaXMgcGFyc2VkIGFuZCBhIERPTSB0cmVlIChvZiB0aGUgbm9kZXMgaW4gdGhlIFhNTCkgaXMgZ2VuZXJhdGVkIGFuZCByZXR1cm5lZC4gT25jZSBwYXJzZWQsIHRoZSB1c2VyIGNhbiBuYXZpZ2F0ZSB0aGUgdHJlZSB0byBhY2Nlc3MgdGhlIHZhcmlvdXMgZGF0YSBwcmV2aW91c2x5IGVtYmVkZGVkIGluIHRoZSB2YXJpb3VzIG5vZGVzIGluIHRoZSBYTUwuIEluIGdlbmVyYWwsIERPTSBpcyBlYXNpZXIgdG8gdXNlIGJ1dCBoYXMgYW4gb3ZlcmhlYWQgb2YgcGFyc2luZyB0aGUgZW50aXJlIFhNTCBiZWZvcmUgeW91IGNhbiBzdGFydCB1c2luZyBpdCBhbmQga2VlcGluZyBpdCBpbiBtZW1vcnkuDQoNCiMjIyMgRE9NDQoNCi0gICB0cmVlIG1vZGVsIHBhcnNlciBjb252ZXJ0aW5nIFhNTCB0byBhbiBpbnRlcm5hbCB0cmVlDQotICAgbG9hZHMgdGhlIGRvY3VtZW50IGZ1bGx5IGludG8gdGhlIG1lbW9yeSBhbmQgdGhlbiBwYXJzZSB0aGUgZG9jdW1lbnQNCi0gICBtZW1vcnkgY29uc3RyYWluZWQgYXMgaXQgbG9hZHMgdGhlIGVudGlyZSBYTUwgYmVmb3JlIHBhcnNpbmcNCi0gICBlbnRpcmUgb2JqZWN0IG1vZGVsIGlzIHN0b3JlZCBhcyBhIHRyZWUgaW4gbWVtb3J5LCB0aHVzIG1lbW9yeSBpbnRlbnNpdmUNCi0gICBvYmplY3QgbW9kZWwgY2FuIGJlIHJlYWQgYW5kIHdyaXR0ZW4sICppLmUuKiwgY2FuIGluc2VydCBvciBkZWxldGUgZWxlbWVudHMNCi0gICBwcmVmZXJhYmxlIGlmIHRoZSBYTUwgY29udGVudCBmaXRzIGludG8gbWVtb3J5DQotICAgY2FuIGJlIHF1ZXJpZWQgdmlhIFhQYXRoDQotICAgZWFzeSB0byBuYXZpZ2F0ZSBpbiBhbnkgZGlyZWN0aW9uDQotICAgc2xvdyB0byBwcm9jZXNzDQoNCiMjIyMgU0FYDQoNCi0gICBldmVudCBiYXNlZCBwYXJzZXIgdXNpbmcgY2FsbGJhY2sgZnVuY3Rpb25zDQotICAgcGFyc2VzIHRoZSBkb2N1bWVudCBhcyBpdCBpcyByZWFkLCAqaS5lLiosIHBhcnNlcyBvbmUgZWxlbWVudCBhdCBhIHRpbWUNCi0gICBubyBtZW1vcnkgY29uc3RyYWludHMgYXMgaXQgZG9lcyBub3Qgc3RvcmUgdGhlIFhNTCBpbiBtZW1vcnkuDQotICAgcmVhZCBvbmx5LCAqaS5lLiosIGNhbid0IGluc2VydCBvciBkZWxldGUgZWxlbWVudHMNCi0gICBwcmVmZXJhYmxlIHdoZW4gWE1MIGlzIGxhcmdlIG9yIG1lbW9yeSBpcyBsaW1pdGVkDQotICAgcmVhZHMgdGhlIFhNTCBkb2N1bWVudCBmcm9tIGJlZ2lubmluZyB0byBlbmQgaW4gb25lIGRpcmVjdGlvbg0KLSAgIGZhc3QgcHJvY2Vzc2luZw0KDQojIyBJbnNwZWN0aW5nIFhNTCBGaWxlcw0KDQpYTUwgZmlsZXMgYXJlIHRleHQgZmlsZXMgYW5kIG11c3QgYmUgdmlld2VkIChhbmQgZWRpdGVkKSB1c2luZyBhIHRleHQgZWRpdG9yIGFuZCBub3QgYSB3b3JkIHByb2Nlc3Nvci4gU28sIHVzZSBUZXh0RWRpdCBvbiBNYWNPUywgb3IgTm90ZXBhZCBvbiBXaW5kb3dzLiBEb24ndCB1c2UgV29yZCBvciBQYWdlcy4gQSBwcm9ncmFtbWluZyBlbnZpcm9ubWVudCAoSURFKSBzdWNoIGFzIFZpc3VhbCBTdHVkaW8sIEVjbGlwc2UsIG9yIFIgU3R1ZGlvIGNhbiBhbHNvIGJlIHVzZWQuDQoNCkZvciBleGFtcGxlLCB0byBpbnNwZWN0IHRoZSAqcHVibWVkX3NhbXBsZS54bWwqIGZpbGUsIHlvdSBjYW4gdXNlIFIgU3R1ZGlvIC0tIGl0IGhhbmRsZXMgWE1MIHdlbGwuIERvd25sb2FkIHRoZSBbKnB1Ym1lZF9zYW1wbGUueG1sKl0ocHVibWVkX3NhbXBsZS54bWwpIGZpcnN0LCB0aGVuIGxhdW5jaCBSIFN0dWRpbywgYW5kIHRoZW4gY2xpY2sgKkZpbGUvT3BlbiBGaWxlLi4uKiBhbmQgY2hvb3NlICpwdWJtZWRfc2FtcGxlLnhtbCogZnJvbSB0aGUgZm9sZGVyIGludG8gd2hpY2ggeW91IGRvd25sb2FkZWQgdGhlIGZpbGUgKGxpa2VseSB0aGUgKkRvd25sb2FkcyogZm9sZGVyKS4gVm9pbGEuLi4NCg0KVGhlIHNob3J0IGRlbW9uc3RyYXRpb24gYmVsb3cgc2hvd3MgaG93IHRvIG9wZW4gYW4gWE1MIGZpbGUgaW4gUiBTdHVkaW8gZm9yIGluc3BlY3Rpb24uDQoNCmBgYHs9aHRtbH0NCjxpZnJhbWUgc3JjPSJodHRwczovL3d3dy55b3V0dWJlLmNvbS9lbWJlZC9DRTBhQlJiTllzbyIgd2lkdGg9IjU2MCIgaGVpZ2h0PSIzMTUiIGFsbG93ZnVsbHNjcmVlbj0iYWxsb3dmdWxsc2NyZWVuIiBhbGxvdz0iYWNjZWxlcm9tZXRlcjsgYXV0b3BsYXk7IHBpY3R1cmUtaW4tcGljdHVyZSIgZGF0YS1leHRlcm5hbD0iMSI+PC9pZnJhbWU+DQo8cC8+DQpgYGANCkluIG1hbnkgY2FzZXMgeW91IGNhbiBvcGVuIHVwIGEgZmlsZSBieSBqdXN0IGRvdWJsZS1jbGlja2luZyBvbiBpdC4gVGhhdCBsYXVuY2hlcyB0aGUgYXBwIHRoYXQncyBhc3NvY2lhdGVkIHdpdGggdGhlIGV4dGVuc2lvbiAoKi54bWwqLCAqLmR0ZCosICpldGMqKS4gWW91IGNhbiBjaGFuZ2UgdGhlIGFzc29jaWF0aW9uIGluIHlvdXIgcHJlZmVyZW5jZSBzZXR0aW5ncy4gQnV0IG9mdGVuIHlvdSB3YW50IHRvIG9wZW4gYSBmaWxlIHdpdGggYSBkaWZmZXJlbnQgYXBwIG9yIHRoZXJlJ3Mgbm8gYXNzb2NpYXRpb24uIFlvdSB3aWxsIG5lZWQgdG8gb3BlbiB0aGUgYXBwIGNvbXBhdGlibGUgd2l0aCB0aGUgZmlsZSB0eXBlIGZpcnN0LCB0aGVuIG9wZW4gdGhlIGZpbGUgZnJvbSB3aXRoaW4gdGhlIGFwcC4NCg0KQSBub3RlIG9uIGRvd25sb2FkaW5nIFhNTCBmaWxlcyBmcm9tIGEgbGluazogdGhpcyBkb2Vzbid0IGdlbmVyYWxseSB3b3JrLCBhcywgYnkgZGVmYXVsdCwgWE1MIGZpbGVzIGFyZSBvcGVuZWQgaW4geW91ciBicm93c2VyIHdoaWNoIHdpbGwgYXR0ZW1wdCB0byByZW5kZXIgdGhlIFhNTC4gU28sIHRvIGRvd25sb2FkIGFuZCBzYXZlIGFuIFhNTCBmaWxlLCBvcGVuIHRoZSBjb250ZXh0IG1lbnUgYnkgKGdlbmVyYWxseSkgcmlnaHQtY2xpY2tpbmcgb24gdGhlIGxpbmsgYW5kIHNlbGVjdGluZyAiU2F2ZSBBcyIgb3IgYSBzaW1pbGFyIGNob2ljZSBkZXBlbmRpbmcgb24geW91ciBvcGVyYXRpbmcgc3lzdGVtLg0KDQojIyBTdW1tYXJ5DQoNClhNTCBpcyBhbiBpbXBvcnRhbnQgbWVjaGFuaXNtIGZvciB0cmFuc2ZlcnJpbmcgZGF0YSBiZXR3ZWVuIHBhcnRpZXMgYW5kIGluZm9ybWF0aW9uIHN5c3RlbXMuDQoNCiMjIFR1dG9yaWFsIEk6IE92ZXJ2aWV3IG9mIFhNTCBhbmQgRFREDQoNCkluIHRoaXMgbmFycmF0ZWQgY2hhbGstdGFsaywgS2hvdXJ5IEJvc3RvbidzIFByb2YuIFNjaGVkbGJhdWVyIHByb3ZpZGVzIGFuIG92ZXJ2aWV3IG9mIFhNTCwgY29tbW9uIHVzZSwgYW5kIHRoZSByb2xlIG9mIERURCBhcyBhIGRlZmluaW5nIGdyYW1tYXIgb2YgYW4gWE1MIHNjaGVtYVteMV0uDQoNClteMV06IFRoaXMgY2hhbGstdGFsayB3YXMgb3JpZ2luYWxseSBwcmVwYXJlZCBmb3IgSVMyMDAwIC0tIFByaW5jaXBsZXMgb2YgSW5mb3JtYXRpb24gU3lzdGVtcyBhbmQgbWF5IGNvbnRhaW4gcmVmZXJlbmNlcyB0byB0aGF0IGNvdXJzZQ0KDQpgYGB7PWh0bWx9DQo8aWZyYW1lIHN0eWxlPSJib3JkZXI6IDBweCBzb2xpZCAjNDY0NjQ2OyIgc3JjPSJodHRwczovL3d3dy55b3V0dWJlLmNvbS9lbWJlZC9talhINUljVU81USIgd2lkdGg9IjQ4MCIgaGVpZ2h0PSIyNzAiIGFsbG93ZnVsbHNjcmVlbj0iYWxsb3dmdWxsc2NyZWVuIiBhbGxvdz0iYWNjZWxlcm9tZXRlcjsgYXV0b3BsYXk7IGNsaXBib2FyZC13cml0ZTsgZW5jcnlwdGVkLW1lZGlhOyBneXJvc2NvcGU7IHBpY3R1cmUtaW4tcGljdHVyZSIgZGF0YS1leHRlcm5hbD0iMSI+PC9pZnJhbWU+DQpgYGANCiMjIFR1dG9yaWFsIElJOiBFc3NlbnRpYWwgWE1MIHdpdGggRXhhbXBsZXMNCg0KSW4gdGhlc2UgdHdvIHZpZGVvIHR1dG9yaWFscywgS2hvdXJ5IEJvc3RvbidzIFByb2YuIFNjaGVkbGJhdWVyIGV4cGxhaW5zIGZpcnN0IGhvdyB0byBidWlsZCBhbiBYTUwgZG9jdW1lbnQgd2l0aCBhbiBhc3NvY2lhdGVkIERURCBmb2xsb3dlZCBieSBhIGNvZGUgd2Fsay10aHJvdWdoIGluIDxodHRwczovL3d3dy54bWx2YWxpZGF0aW9uLmNvbT4uDQoNClRoZSB0dXRvcmlhbCB1c2VzIHRoZSBYTUwgZmlsZToNCg0KLSAgIFtUZWFtUm9zdGVycy54bWxdKFRlYW1Sb3N0ZXJzLnhtbCkNCg0KYGBgez1odG1sfQ0KPGlmcmFtZSBzdHlsZT0iYm9yZGVyOiAwcHggc29saWQgIzQ2NDY0NjsiIHNyYz0iaHR0cHM6Ly9ub3J0aGVhc3Rlcm4uaG9zdGVkLnBhbm9wdG8uY29tL1Bhbm9wdG8vUGFnZXMvRW1iZWQuYXNweD9pZD1lNWYxNzg2Ny05ZTYwLTRkZDItYjc2Yy1hYzNkMDE0MmQwM2MmYW1wO2F1dG9wbGF5PWZhbHNlJmFtcDtvZmZlcnZpZXdlcj1mYWxzZSZhbXA7c2hvd3RpdGxlPWZhbHNlJmFtcDtzaG93YnJhbmQ9ZmFsc2UmYW1wO3N0YXJ0PTAmYW1wO2ludGVyYWN0aXZpdHk9YWxsIiB3aWR0aD0iNDgwIiBoZWlnaHQ9IjI3MCIgYWxsb3dmdWxsc2NyZWVuPSJhbGxvd2Z1bGxzY3JlZW4iIGFsbG93PSJhdXRvcGxheSIgZGF0YS1leHRlcm5hbD0iMSI+PC9pZnJhbWU+DQo8cC8+DQo8aWZyYW1lIHN0eWxlPSJib3JkZXI6IDBweCBzb2xpZCAjNDY0NjQ2OyIgc3JjPSJodHRwczovL25vcnRoZWFzdGVybi5ob3N0ZWQucGFub3B0by5jb20vUGFub3B0by9QYWdlcy9FbWJlZC5hc3B4P2lkPTc3MDMwOTIwLTg3ZjMtNDU2My04NDJiLWFjM2QwMTU3MWNhNSZhbXA7YXV0b3BsYXk9ZmFsc2UmYW1wO29mZmVydmlld2VyPWZhbHNlJmFtcDtzaG93dGl0bGU9ZmFsc2UmYW1wO3Nob3dicmFuZD1mYWxzZSZhbXA7c3RhcnQ9MCZhbXA7aW50ZXJhY3Rpdml0eT1hbGwiIHdpZHRoPSI0ODAiIGhlaWdodD0iMjcwIiBhbGxvd2Z1bGxzY3JlZW49ImFsbG93ZnVsbHNjcmVlbiIgYWxsb3c9ImF1dG9wbGF5IiBkYXRhLWV4dGVybmFsPSIxIj48L2lmcmFtZT4NCmBgYA0KDQotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0NCg0KIyMgRmlsZXMgJiBSZXNvdXJjZXMNCg0KYGBge3IgemlwRmlsZXMsIGVjaG89RkFMU0V9DQp6aXBOYW1lID0gc3ByaW50ZigiTGVzc29uRmlsZXMtJXMtJXMuemlwIiwgDQogICAgICAgICAgICAgICAgIHBhcmFtcyRjYXRlZ29yeSwNCiAgICAgICAgICAgICAgICAgcGFyYW1zJG51bWJlcikNCg0KdGV4dEFMaW5rID0gcGFzdGUwKCJBbGwgRmlsZXMgZm9yIExlc3NvbiAiLCANCiAgICAgICAgICAgICAgIHBhcmFtcyRjYXRlZ29yeSwiLiIscGFyYW1zJG51bWJlcikNCg0KIyBkb3dubG9hZEZpbGVzTGluaygpIGlzIGluY2x1ZGVkIGZyb20gX2luc2VydDJEQi5SDQprbml0cjo6cmF3X2h0bWwoZG93bmxvYWRGaWxlc0xpbmsoIi4iLCB6aXBOYW1lLCB0ZXh0QUxpbmspKQ0KYGBgDQoNCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQ0KDQojIyBSZWZlcmVuY2VzDQoNCi0gICBbRGVmaW5pdGlvbiBvZiBEVEQgUnVsZXMuXShodHRwczovL3d3dy5saXF1aWQtdGVjaG5vbG9naWVzLmNvbS9EVEQvT3ZlcnZpZXcuYXNweCkNCi0gICBbTWFyY2hhbCwgQi4gKDIwMDEpLiBUaGUgRFREIFN5bnRheC4gRXhjZXJwdCBmcm9tIFhNTCBCeSBFeGFtcGxlXShodHRwczovL3d3dy5pbmZvcm1pdC5jb20vYXJ0aWNsZXMvYXJ0aWNsZS5hc3B4P3A9MTMxMzQ4KQ0KLSAgIDxodHRwczovL3d3dy53My5vcmcvc3RhbmRhcmRzL3htbC9jb3JlLmh0bWw+DQotICAgPGh0dHBzOi8vd3d3LnczLm9yZy9zdGFuZGFyZHMveG1sL3NjaGVtYS5odG1sPg0KDQojIyBFcnJhdGENCg0KW0xldCB1cyBrbm93XShodHRwczovL2Zvcm0uam90Zm9ybS5jb20vMjEyMTg3MDcyNzg0MTU3KXt0YXJnZXQ9Il9ibGFuayJ9Lg0K