Files
mercury/extras/xml/xml.parse.m
Julien Fischer 9fb4f15dfc Fix compilation problems in the extras distribution caused by recent
Estimated hours taken: 0.5
Branches: main

Fix compilation problems in the extras distribution caused by recent
changes.

extras/stream/stream.m:
	Provide a definition for the type stream/1.

extras/*/*.m:
	Conform to the recent changes to the standard library.
2006-03-30 01:21:20 +00:00

3732 lines
142 KiB
Mathematica
Raw Blame History

%---------------------------------------------------------------------------%
% Copyright (C) 2000, 2005-2006 The University of Melbourne.
% This file may only be copied under the terms of the GNU Library General
% Public License - see the file COPYING.LIB in the Mercury distribution.
%---------------------------------------------------------------------------%
%
% Main author: conway@cs.mu.oz.au.
%
%---------------------------------------------------------------------------%
%
% W3C REC-xml-19980210
%
% Extensible Markup Language (XML) 1.0
%
% W3C Recommendation 10-February-1998
%
% This version:
% http://www.w3.org/TR/1998/REC-xml-19980210
% http://www.w3.org/TR/1998/REC-xml-19980210.xml
% http://www.w3.org/TR/1998/REC-xml-19980210.html
% http://www.w3.org/TR/1998/REC-xml-19980210.pdf
% http://www.w3.org/TR/1998/REC-xml-19980210.ps
%
% Latest version:
% http://www.w3.org/TR/REC-xml
%
% Previous version:
% http://www.w3.org/TR/PR-xml-971208
%
% Editors:
% Tim Bray (Textuality and Netscape) <tbray@textuality.com>
% Jean Paoli (Microsoft) <jeanpa@microsoft.com>
% C. M. Sperberg-McQueen (University of Illinois at Chicago)
% <cmsmcq@uic.edu>
%
%Abstract
%
% The Extensible Markup Language (XML) is a subset of SGML that is
% completely described in this document. Its goal is to enable generic
% SGML to be served, received, and processed on the Web in the way that
% is now possible with HTML. XML has been designed for ease of
% implementation and for interoperability with both SGML and HTML.
%
%Status of this document
%
% This document has been reviewed by W3C Members and other interested
% parties and has been endorsed by the Director as a W3C Recommendation.
% It is a stable document and may be used as reference material or cited
% as a normative reference from another document. W3C's role in making
% the Recommendation is to draw attention to the specification and to
% promote its widespread deployment. This enhances the functionality and
% interoperability of the Web.
%
% This document specifies a syntax created by subsetting an existing,
% widely used international text processing standard (Standard
% Generalized Markup Language, ISO 8879:1986(E) as amended and
% corrected) for use on the World Wide Web. It is a product of the W3C
% XML Activity, details of which can be found at http://www.w3.org/XML.
% A list of current W3C Recommendations and other technical documents
% can be found at http://www.w3.org/TR.
%
% This specification uses the term URI, which is defined by [Berners-Lee
% et al.], a work in progress expected to update [IETF RFC1738] and
% [IETF RFC1808].
%
% The list of known errors in this specification is available at
% http://www.w3.org/XML/xml-19980210-errata.
%
% Please report errors in this document to xml-editor@w3.org.
%
% Extensible Markup Language (XML) 1.0
%
%Table of Contents
%
% 1. Introduction
% 1.1 Origin and Goals
% 1.2 Terminology
% 2. Documents
% 2.1 Well-Formed XML Documents
% 2.2 Characters
% 2.3 Common Syntactic Constructs
% 2.4 Character Data and Markup
% 2.5 Comments
% 2.6 Processing Instructions
% 2.7 CDATA Sections
% 2.8 Prolog and Document Type Declaration
% 2.9 Standalone Document Declaration
% 2.10 White Space Handling
% 2.11 End-of-Line Handling
% 2.12 Language Identification
% 3. Logical Structures
% 3.1 Start-Tags, End-Tags, and Empty-Element Tags
% 3.2 Element Type Declarations
% 3.2.1 Element Content
% 3.2.2 Mixed Content
% 3.3 Attribute-List Declarations
% 3.3.1 Attribute Types
% 3.3.2 Attribute Defaults
% 3.3.3 Attribute-Value Normalization
% 3.4 Conditional Sections
% 4. Physical Structures
% 4.1 Character and Entity References
% 4.2 Entity Declarations
% 4.2.1 Internal Entities
% 4.2.2 External Entities
% 4.3 Parsed Entities
% 4.3.1 The Text Declaration
% 4.3.2 Well-Formed Parsed Entities
% 4.3.3 Character Encoding in Entities
% 4.4 XML Processor Treatment of Entities and References
% 4.4.1 Not Recognized
% 4.4.2 Included
% 4.4.3 Included If Validating
% 4.4.4 Forbidden
% 4.4.5 Included in Literal
% 4.4.6 Notify
% 4.4.7 Bypassed
% 4.4.8 Included as PE
% 4.5 Construction of Internal Entity Replacement Text
% 4.6 Predefined Entities
% 4.7 Notation Declarations
% 4.8 Document Entity
% 5. Conformance
% 5.1 Validating and Non-Validating Processors
% 5.2 Using XML Processors
% 6. Notation
%
% Appendices
%
% A. References
% A.1 Normative References
% A.2 Other References
% B. Character Classes
% C. XML and SGML (Non-Normative)
% D. Expansion of Entity and Character References (Non-Normative)
% E. Deterministic Content Models (Non-Normative)
% F. Autodetection of Character Encodings (Non-Normative)
% G. W3C XML Working Group (Non-Normative)
% _________________________________________________________________
%
%1. Introduction
%
% Extensible Markup Language, abbreviated XML, describes a class of data
% objects called XML documents and partially describes the behavior of
% computer programs which process them. XML is an application profile or
% restricted form of SGML, the Standard Generalized Markup Language [ISO
% 8879]. By construction, XML documents are conforming SGML documents.
%
% XML documents are made up of storage units called entities, which
% contain either parsed or unparsed data. Parsed data is made up of
% characters, some of which form character data, and some of which form
% markup. Markup encodes a description of the document's storage layout
% and logical structure. XML provides a mechanism to impose constraints
% on the storage layout and logical structure.
%
% A software module called an XML processor is used to read XML
% documents and provide access to their content and structure. It is
% assumed that an XML processor is doing its work on behalf of another
% module, called the application. This specification describes the
% required behavior of an XML processor in terms of how it must read XML
% data and the information it must provide to the application.
%
% 1.1 Origin and Goals
%
% XML was developed by an XML Working Group (originally known as the
% SGML Editorial Review Board) formed under the auspices of the World
% Wide Web Consortium (W3C) in 1996. It was chaired by Jon Bosak of Sun
% Microsystems with the active participation of an XML Special Interest
% Group (previously known as the SGML Working Group) also organized by
% the W3C. The membership of the XML Working Group is given in an
% appendix. Dan Connolly served as the WG's contact with the W3C.
%
% The design goals for XML are:
% 1. XML shall be straightforwardly usable over the Internet.
% 2. XML shall support a wide variety of applications.
% 3. XML shall be compatible with SGML.
% 4. It shall be easy to write programs which process XML documents.
% 5. The number of optional features in XML is to be kept to the
% absolute minimum, ideally zero.
% 6. XML documents should be human-legible and reasonably clear.
% 7. The XML design should be prepared quickly.
% 8. The design of XML shall be formal and concise.
% 9. XML documents shall be easy to create.
% 10. Terseness in XML markup is of minimal importance.
%
% This specification, together with associated standards (Unicode and
% ISO/IEC 10646 for characters, Internet RFC 1766 for language
% identification tags, ISO 639 for language name codes, and ISO 3166 for
% country name codes), provides all the information necessary to
% understand XML Version 1.0 and construct computer programs to process
% it.
%
% This version of the XML specification may be distributed freely, as
% long as all text and legal notices remain intact.
%
% 1.2 Terminology
%
% The terminology used to describe XML documents is defined in the body
% of this specification. The terms defined in the following list are
% used in building those definitions and in describing the actions of an
% XML processor:
%
% may
% Conforming documents and XML processors are permitted to but
% need not behave as described.
%
% must
% Conforming documents and XML processors are required to behave
% as described; otherwise they are in error.
%
% error
% A violation of the rules of this specification; results are
% undefined. Conforming software may detect and report an error
% and may recover from it.
%
% fatal error
% An error which a conforming XML processor must detect and
% report to the application. After encountering a fatal error,
% the processor may continue processing the data to search for
% further errors and may report such errors to the application.
% In order to support correction of errors, the processor may
% make unprocessed data from the document (with intermingled
% character data and markup) available to the application. Once a
% fatal error is detected, however, the processor must not
% continue normal processing (i.e., it must not continue to pass
% character data and information about the document's logical
% structure to the application in the normal way).
%
% at user option
% Conforming software may or must (depending on the modal verb in
% the sentence) behave as described; if it does, it must provide
% users a means to enable or disable the behavior described.
%
% validity constraint
% A rule which applies to all valid XML documents. Violations of
% validity constraints are errors; they must, at user option, be
% reported by validating XML processors.
%
% well-formedness constraint
% A rule which applies to all well-formed XML documents.
% Violations of well-formedness constraints are fatal errors.
%
% match
% (Of strings or names:) Two strings or names being compared must
% be identical. Characters with multiple possible representations
% in ISO/IEC 10646 (e.g. characters with both precomposed and
% base+diacritic forms) match only if they have the same
% representation in both strings. At user option, processors may
% normalize such characters to some canonical form. No case
% folding is performed. (Of strings and rules in the grammar:) A
% string matches a grammatical production if it belongs to the
% language generated by that production. (Of content and content
% models:) An element matches its declaration when it conforms in
% the fashion described in the constraint "Element Valid".
%
% for compatibility
% A feature of XML included solely to ensure that XML remains
% compatible with SGML.
%
% for interoperability
% A non-binding recommendation included to increase the chances
% that XML documents can be processed by the existing installed
% base of SGML processors which predate the WebSGML Adaptations
% Annex to ISO 8879.
%
%2. Documents
%
% A data object is an XML document if it is well-formed, as defined in
% this specification. A well-formed XML document may in addition be
% valid if it meets certain further constraints.
%
% Each XML document has both a logical and a physical structure.
% Physically, the document is composed of units called entities. An
% entity may refer to other entities to cause their inclusion in the
% document. A document begins in a "root" or document entity. Logically,
% the document is composed of declarations, elements, comments,
% character references, and processing instructions, all of which are
% indicated in the document by explicit markup. The logical and physical
% structures must nest properly, as described in "4.3.2 Well-Formed
% Parsed Entities".
%
% 2.1 Well-Formed XML Documents
%
% A textual object is a well-formed XML document if:
% 1. Taken as a whole, it matches the production labeled document.
% 2. It meets all the well-formedness constraints given in this
% specification.
% 3. Each of the parsed entities which is referenced directly or
% indirectly within the document is well-formed.
%
% Document
% [1] document ::= prolog element Misc*
:- module xml.parse.
:- interface.
:- import_module list.
:- import_module parsing.
:- import_module xml.cat.
:- import_module xml.doc.
:- import_module xml.dtd.
%
% The following three globals should be set in the globals included
% in the initial parsing state.
%
:- type gCatalog ---> gCatalog.
:- instance global(gCatalog, catalog).
:- type gDirs ---> gDirs.
:- type dirs ---> dirs(cat.dirs).
:- instance global(gDirs, parse.dirs).
:- type gEncodings ---> gEncodings.
:- type encodings ---> encodings(string -> encoding).
:- instance global(gEncodings, encodings).
:- pred document(pstate(_), pstate((dtd, document))).
:- mode document(in, out) is det.
:- implementation.
:- import_module xml.encoding.
:- import_module unicode.
:- import_module array, char, int, io, map, pair, require, string, unit.
:- instance global(gCatalog, catalog) where [].
:- instance global(gDirs, parse.dirs) where [].
:- instance global(gEncodings, encodings) where [].
:- type gContent ---> gContent.
:- instance global(gContent, contentStore) where [].
:- type gElements ---> gElements.
:- type elements ---> elements(name -> dtd.element).
:- instance global(gElements, elements) where [].
:- type gAttributes ---> gAttributes.
:- type attributes ---> attributes(name -> name -> dtd.attribute).
:- instance global(gAttributes, attributes) where [].
:- type gEntities ---> gEntities.
:- type entities ---> entities(name -> entityDef).
:- instance global(gEntities, entities) where [].
:- type gPEntities ---> gPEntities.
:- type pentities ---> pentities(name -> entityDef).
:- instance global(gPEntities, pentities) where [].
:- type gDTD ---> gDTD.
:- instance global(gDTD, dtd) where [].
:- type gExtEntities ---> gExtEntities.
:- type extEntities ---> extEntities(externalId -> dtd.entity).
:- instance global(gExtEntities, extEntities) where [].
document -->
{ contentStore(Content0) },
set(gContent, Content0),
set(gExtEntities, extEntities(init)),
set(gEntities, entities(entities)),
set(gPEntities, pentities(init)),
set(gElements, elements(init)),
set(gAttributes, attributes(init)),
(prolog then (pred((DTD, PreMisc)::in, pdi, puo) is det -->
(
set(gDTD, DTD),
(element then (pred(Root::in, pdi, puo) is det -->
star(misc) then (pred(PostMisc0::in, pdi, puo) is det -->
get(gContent, Content),
{ filterOpt(PostMisc0, PostMisc) },
{ Doc = doc(PreMisc, Root, PostMisc, array(values(Content^eMap))) },
return((DTD, Doc))
)))))).
:- pred contentStore(contentStore::out) is det.
contentStore(content(0, Map)) :-
init(Map).
:- pred same_type(T::unused, T::unused) is det.
same_type(_, _).
:- func entities = (name -> entityDef).
entities = Entities :-
map__from_assoc_list([
"lt" - internal("&#60;"),
"gt" - internal("&#62;"),
"amp" - internal("&#38;"),
"quot" - internal("&#39;"),
"apos" - internal("&#34;")
], Entities).
:- pred initDTD(name, dtd).
:- mode initDTD(in, out) is det.
initDTD(Root, DTD) :-
init(Elems),
map__from_assoc_list([
"lt" - internal("&#60;"),
"gt" - internal("&#62;"),
"amp" - internal("&#38;"),
"quot" - internal("&#39;"),
"apos" - internal("&#34;")
], Entities),
init(PEntities),
DTD = dtd(Root, Elems, Entities, PEntities).
%
% Matching the document production implies that:
% 1. It contains one or more elements.
% 2. There is exactly one element, called the root, or document
% element, no part of which appears in the content of any other
% element. For all other elements, if the start-tag is in the
% content of another element, the end-tag is in the content of the
% same element. More simply stated, the elements, delimited by
% start- and end-tags, nest properly within each other.
%
% As a consequence of this, for each non-root element C in the document,
% there is one other element P in the document such that C is in the
% content of P, but is not in the content of any other element that is
% in the content of P. P is referred to as the parent of C, and C as a
% child of P.
%
% 2.2 Characters
%
% A parsed entity contains text, a sequence of characters, which may
% represent markup or character data. A character is an atomic unit of
% text as specified by ISO/IEC 10646 [ISO/IEC 10646]. Legal characters
% are tab, carriage return, line feed, and the legal graphic characters
% of Unicode and ISO/IEC 10646. The use of "compatibility characters",
% as defined in section 6.8 of [Unicode], is discouraged.
%
% Character Range
% [2] Char ::= #x9 | #xA | #xD | [#x20-#xD7FF] | [#xE000-#xFFFD]
% | [#x10000-#x10FFFF] /* any Unicode character, excluding the surrogate
% blocks, FFFE, and FFFF. */
:- pred char(pstate(_), pstate(unicode)).
:- mode char(in, out) is det.
char -->
tok then (pred(C::in, pdi, puo) is det -->
( {
C = 0x09
;
C = 0x0A
;
C = 0x0D
;
C >= 0x20, C =< 0xD7FF
;
C >= 0xE000, C =< 0xFFFD
;
C >= 0x10000, C =< 0x10FFFF
} ->
return(C)
;
{ format("Unexpected character `%x'.", [i(C)], Msg) },
error(Msg)
)).
%
% The mechanism for encoding character code points into bit patterns may
% vary from entity to entity. All XML processors must accept the UTF-8
% and UTF-16 encodings of 10646; the mechanisms for signaling which of
% the two is in use, or for bringing other encodings into play, are
% discussed later, in "4.3.3 Character Encoding in Entities".
%
% 2.3 Common Syntactic Constructs
%
% This section defines some symbols used widely in the grammar.
%
% S (white space) consists of one or more space (#x20) characters,
% carriage returns, line feeds, or tabs.
%
% White Space
% [3] S ::= (#x20 | #x9 | #xD | #xA)+
:- pred s(pstate(_), pstate(list(unicode))).
:- mode s(in, out) is det.
s -->
plus(s0).
:- pred s0(pstate(_), pstate(unicode)).
:- mode s0(in, out) is det.
s0 -->
tok then (pred(C::in, pdi, puo) is det -->
( {
C = 0x20
;
C = 0x09
;
C = 0x0D
;
C = 0x0A
} ->
return(C)
;
fail("not whitespace")
)).
%
% Characters are classified for convenience as letters, digits, or other
% characters. Letters consist of an alphabetic or syllabic base
% character possibly followed by one or more combining characters, or of
% an ideographic character. Full definitions of the specific characters
% in each class are given in "B. Character Classes".
%
% A Name is a token beginning with a letter or one of a few punctuation
% characters, and continuing with letters, digits, hyphens, underscores,
% colons, or full stops, together known as name characters. Names
% beginning with the string "xml", or any string which would match
% (('X'|'x') ('M'|'m') ('L'|'l')), are reserved for standardization in
% this or future versions of this specification.
%
% Note: The colon character within XML names is reserved for
% experimentation with name spaces. Its meaning is expected to be
% standardized at some future point, at which point those documents
% using the colon for experimental purposes may need to be updated.
% (There is no guarantee that any name-space mechanism adopted for XML
% will in fact use the colon as a name-space delimiter.) In practice,
% this means that authors should not use the colon in XML names except
% as part of name-space experiments, but that XML processors should
% accept the colon as a name character.
%
% An Nmtoken (name token) is any mixture of name characters.
%
% Names and Tokens
% [4] NameChar ::= Letter | Digit | '.' | '-' | '_' | ':'
% | CombiningChar | Extender
:- pred nameChar(pstate(_), pstate(unicode)).
:- mode nameChar(in, out) is det.
nameChar -->
letter or digit or lit1('.') or lit1('-') or lit1('_') or lit1(':') or
combiningChar or extender.
% [5] Name ::= (Letter | '_' | ':') (NameChar)*
:- pred name(pstate(_), pstate(name)).
:- mode name(in, out) is det.
name -->
letter or lit1('_') or lit1(':') then (pred(C::in, pdi, puo) is det -->
star(nameChar) then (pred(Cs::in, pdi, puo) is det -->
mkString([C|Cs], Name),
return(Name)
)).
% [6] Names ::= Name (S Name)*
:- pred names(pstate(_), pstate(list(name))).
:- mode names(in, out) is det.
names -->
name then (pred(Name::in, pdi, puo) is det -->
star(snd(s and name)) then (pred(Names0::in, pdi, puo) is det -->
{ Names = [Name|Names0] },
return(Names)
)).
% [7] Nmtoken ::= (NameChar)+
:- pred nmtoken(pstate(_), pstate(name)).
:- mode nmtoken(in, out) is det.
nmtoken -->
plus(nameChar) then (pred(Cs::in, pdi, puo) is det -->
mkString(Cs, Name),
return(Name)
).
% [8] Nmtokens ::= Nmtoken (S Nmtoken)*
:- pred nmtokens(pstate(_), pstate(list(name))).
:- mode nmtokens(in, out) is det.
nmtokens -->
nmtoken then (pred(Name::in, pdi, puo) is det -->
star(snd(s and nmtoken)) then (pred(Names::in, pdi, puo) is det -->
return([Name|Names])
)).
%
% Literal data is any quoted string not containing the quotation mark
% used as a delimiter for that string. Literals are used for specifying
% the content of internal entities (EntityValue), the values of
% attributes (AttValue), and external identifiers (SystemLiteral). Note
% that a SystemLiteral can be parsed without scanning for markup.
%
% Literals
% [9] EntityValue ::= '"' ([^%&"] | PEReference | Reference)* '"'
% | "'" ([^%&'] | PEReference | Reference)* "'"
:- pred entityValue(pstate(_), pstate(string)).
:- mode entityValue(in, out) is det.
entityValue -->
quote then (pred(Q::in, pdi, puo) is det -->
entityValue1(Q) then (pred(Chars::in, pdi, puo) is det -->
quote then (pred(EndQ::in, pdi, puo) is det -->
( { Q = EndQ } ->
mkString(Chars, Val),
return(Val)
;
error("mismatched quotes")
)))).
:- pred entityValue1(unicode, pstate(_), pstate(list(unicode))).
:- mode entityValue1(in, in, out) is det.
entityValue1(Q) -->
star(list(charRef) or list(except([('%'), Q])) or
pEReference(star(char)))
then (pred(Css::in, pdi, puo) is det -->
{ condense(Css, Cs) },
return(Cs)
).
% [10] AttValue ::= '"' ([^<&"] | Reference)* '"'
% | "'" ([^<&'] | Reference)* "'"
:- pred attValue(pstate(_), pstate(string)).
:- mode attValue(in, out) is det.
attValue -->
quote then (pred(Q::in, pdi, puo) is det -->
attValue1(Q) then (pred(Chars::in, pdi, puo) is det -->
quote then (pred(EndQ::in, pdi, puo) is det -->
( { Q = EndQ } ->
mkString(Chars, Val),
return(Val)
;
error("mismatched quotes")
)))).
:- pred attValue1(unicode, pstate(_), pstate(list(unicode))).
:- mode attValue1(in, in, out) is det.
attValue1(Q) -->
star(list(charRef) or list(except([('&'), ('<'), Q])) or
entityRef(star(char)))
then (pred(Css::in, pdi, puo) is det -->
{ condense(Css, Cs) },
return(Cs)
).
:- pred attValue2(pstate(_), pstate(list(unicode))).
:- mode attValue2(in, out) is det.
attValue2 -->
star(list(charRef) or entityRef(attValue2) or list(char))
then (pred(Css::in, pdi, puo) is det -->
{ condense(Css, Cs) },
return(Cs)
).
% [11] SystemLiteral ::= ('"' [^"]* '"') | ("'" [^']* "'")
:- pred systemLiteral(pstate(_), pstate(string)).
:- mode systemLiteral(in, out) is det.
systemLiteral -->
quote then (pred(Q::in, pdi, puo) is det -->
star(except([Q])) then (pred(Chars::in, pdi, puo) is det -->
quote then (pred(EndQ::in, pdi, puo) is det -->
( { Q = EndQ } ->
mkString(Chars, Val),
return(Val)
;
error("mismatched quotes")
)))).
% [12] PubidLiteral ::= '"' PubidChar* '"' | "'" (PubidChar - "'")* "'"
:- pred pubidLiteral(pstate(_), pstate(string)).
:- mode pubidLiteral(in, out) is det.
pubidLiteral -->
quote then (pred(Q::in, pdi, puo) is det -->
star(pubidChar(Q)) then (pred(Chars::in, pdi, puo) is det -->
quote then (pred(EndQ::in, pdi, puo) is det -->
( { Q = EndQ } ->
mkString(Chars, Val),
return(Val)
;
error("mismatched quotes")
)))).
% [13] PubidChar ::= #x20 | #xD | #xA | [a-zA-Z0-9]
% | [-'()+,./:=?;!*#@$_%]
:- pred pubidChar(unicode, pstate(_), pstate(unicode)).
:- mode pubidChar(in, in, out) is det.
pubidChar(Q) -->
tok then (pred(C::in, pdi, puo) is det -->
(
{ C \= Q },
{
C = 0x20
;
C = 0x0D
;
C = 0x0A
;
C >= a, C =< z
;
C >= 'A', C =< 'Z'
;
C >= '0', C =< '9'
;
char__to_int(Ch, C),
contains_char("-'()+,./:=?;!*#@$_%\"", Ch)
}
->
return(C)
;
fail("not a publicId char")
)).
%
% 2.4 Character Data and Markup
%
% Text consists of intermingled character data and markup. Markup takes
% the form of start-tags, end-tags, empty-element tags, entity
% references, character references, comments, CDATA section delimiters,
% document type declarations, and processing instructions.
%
% All text that is not markup constitutes the character data of the
% document.
%
% The ampersand character (&) and the left angle bracket (<) may appear
% in their literal form only when used as markup delimiters, or within a
% comment, a processing instruction, or a CDATA section. They are also
% legal within the literal entity value of an internal entity
% declaration; see "4.3.2 Well-Formed Parsed Entities". If they are
% needed elsewhere, they must be escaped using either numeric character
% references or the strings "&amp;" and "&lt;" respectively. The right
% angle bracket (>) may be represented using the string "&gt;", and
% must, for compatibility, be escaped using "&gt;" or a character
% reference when it appears in the string "]]>" in content, when that
% string is not marking the end of a CDATA section.
%
% In the content of elements, character data is any string of characters
% which does not contain the start-delimiter of any markup. In a CDATA
% section, character data is any string of characters not including the
% CDATA-section-close delimiter, "]]>".
%
% To allow attribute values to contain both single and double quotes,
% the apostrophe or single-quote character (') may be represented as
% "&apos;", and the double-quote character (") as "&quot;".
%
% Character Data
% [14] CharData ::= [^<&]* - ([^<&]* ']]>' [^<&]*)
:- pred charData(pstate(_), pstate(ref(doc.content))).
:- mode charData(in, out) is det.
charData -->
plus(except([('<'), ('&')]) or charRef)
then (pred(Chars::in, pdi, puo) is det -->
mkString(Chars, Data),
add(data(Data), Ref),
return(Ref)
).
%
% 2.5 Comments
%
% Comments may appear anywhere in a document outside other markup; in
% addition, they may appear within the document type declaration at
% places allowed by the grammar. They are not part of the document's
% character data; an XML processor may, but need not, make it possible
% for an application to retrieve the text of comments. For
% compatibility, the string "--" (double-hyphen) must not occur within
% comments.
%
% Comments
% [15] Comment ::= '<!--' ((Char - '-') | ('-' (Char - '-')))* '-->'
:- pred comment(pstate(_), pstate(ref(doc.content))).
:- mode comment(in, out) is det.
comment -->
lit("<!--") then (pred(_::in, pdi, puo) is det -->
upto(char, lit("-->")) then (pred((Cs, _)::in, pdi, puo) is det -->
mkString(Cs, Comment),
add(comment(Comment), Ref),
return(Ref)
)).
%
% An example of a comment:
%
% <!-- declarations for <head> & <body> -->
%
% 2.6 Processing Instructions
%
% Processing instructions (PIs) allow documents to contain instructions
% for applications.
%
% Processing Instructions
% [16] PI ::= '<?' PITarget (S (Char* - (Char* '?>' Char*)))? '?>'
:- pred pi(pstate(_), pstate(ref(doc.content))).
:- mode pi(in, out) is det.
pi -->
lit("<?") then (pred(_::in, pdi, puo) is det -->
piTarget then (pred(Target::in, pdi, puo) is det -->
opt(s and upto(char, lit("?>")))
then (pred(MD::in, pdi, puo) is det -->
( { MD = yes((_, (Chars, _))) } ->
mkString(Chars, Data)
;
{ Data = "" }
),
add(pi(Target, Data), Ref),
return(Ref)
))).
% [17] PITarget ::= Name - (('X' | 'x') ('M' | 'm') ('L' | 'l'))
:- pred piTarget(pstate(_), pstate(name)).
:- mode piTarget(in, out) is det.
piTarget -->
name then (pred(Target::in, pdi, puo) is det -->
( { Target = "XML" ; Target = "xml" } ->
fail("(x|X)(m|M)(l|L) is not a valid pi target")
;
return(Target)
)).
%
% PIs are not part of the document's character data, but must be passed
% through to the application. The PI begins with a target (PITarget)
% used to identify the application to which the instruction is directed.
% The target names "XML", "xml", and so on are reserved for
% standardization in this or future versions of this specification. The
% XML Notation mechanism may be used for formal declaration of PI
% targets.
%
% 2.7 CDATA Sections
%
% CDATA sections may occur anywhere character data may occur; they are
% used to escape blocks of text containing characters which would
% otherwise be recognized as markup. CDATA sections begin with the
% string "<![CDATA[" and end with the string "]]>":
%
% CDATA Sections
% [18] CDSect ::= CDStart CData CDEnd
% [19] CDStart ::= '<![CDATA['
% [20] CData ::= (Char* - (Char* ']]>' Char*))
% [21] CDEnd ::= ']]>'
:- pred cdSect(pstate(_), pstate(ref(doc.content))).
:- mode cdSect(in, out) is det.
cdSect -->
lit("<![CDATA[") then (pred(_::in, pdi, puo) is det -->
upto(char, lit("]]>", unit)) then (pred((Cs, _)::in, pdi, puo) is det -->
mkString(Cs, Data),
add(data(Data), Ref),
return(Ref)
)).
%
% Within a CDATA section, only the CDEnd string is recognized as markup,
% so that left angle brackets and ampersands may occur in their literal
% form; they need not (and cannot) be escaped using "&lt;" and "&amp;".
% CDATA sections cannot nest.
%
% An example of a CDATA section, in which "<greeting>" and "</greeting>"
% are recognized as character data, not markup:
%
% <![CDATA[<greeting>Hello, world!</greeting>]]>
%
% 2.8 Prolog and Document Type Declaration
%
% XML documents may, and should, begin with an XML declaration which
% specifies the version of XML being used. For example, the following is
% a complete XML document, well-formed but not valid:
%
% <?xml version="1.0"?>
% <greeting>Hello, world!</greeting>
%
% and so is this:
%
% <greeting>Hello, world!</greeting>
%
% The version number "1.0" should be used to indicate conformance to
% this version of this specification; it is an error for a document to
% use the value "1.0" if it does not conform to this version of this
% specification. It is the intent of the XML working group to give later
% versions of this specification numbers other than "1.0", but this
% intent does not indicate a commitment to produce any future versions
% of XML, nor if any are produced, to use any particular numbering
% scheme. Since future versions are not ruled out, this construct is
% provided as a means to allow the possibility of automatic version
% recognition, should it become necessary. Processors may signal an
% error if they receive documents labeled with versions they do not
% support.
%
% The function of the markup in an XML document is to describe its
% storage and logical structure and to associate attribute-value pairs
% with its logical structures. XML provides a mechanism, the document
% type declaration, to define constraints on the logical structure and
% to support the use of predefined storage units. An XML document is
% valid if it has an associated document type declaration and if the
% document complies with the constraints expressed in it.
%
% The document type declaration must appear before the first element in
% the document.
%
% Prolog
% [22] prolog ::= XMLDecl? Misc* (doctypedecl Misc*)?
:- pred prolog(pstate(_), pstate((dtd, list(ref(doc.content))))).
:- mode prolog(in, out) is det.
prolog -->
opt(xmlDecl) then (pred(_::in, pdi, puo) is det -->
star(misc) then (pred(Misc0::in, pdi, puo) is det -->
opt(doctypedecl and star(misc)) then (pred(MStuff::in, pdi, puo) is det -->
{
MStuff = yes((DTD, Misc1)),
append(Misc0, Misc1, Misc2),
filterOpt(Misc2, Misc)
;
MStuff = no,
init(Elems),
init(Entities),
init(PEntities),
DTD = dtd("", Elems, Entities, PEntities),
filterOpt(Misc0, Misc)
},
return((DTD, Misc))
))).
% [23] XMLDecl ::= '<?xml' VersionInfo EncodingDecl? SDDecl? S? '?>'
:- pred xmlDecl(pstate(_), pstate(unit)).
:- mode xmlDecl(in, out) is det.
xmlDecl -->
lit("<?xml") then (pred(_::in, pdi, puo) is det -->
versionInfo then (pred(_::in, pdi, puo) is det -->
opt(encodingDecl) then (pred(MEnc::in, pdi, puo) is det -->
opt(sdDecl) then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit("?>") then (pred(_::in, pdi, puo) is det -->
(
{ MEnc = yes(EncName) },
get(gEncodings, encodings(Encodings)),
( { search(Encodings, EncName, Encoding) } ->
setEncoding(Encoding),
return
;
{ format("unknown encoding `%s'", [s(EncName)], Msg) },
error(Msg)
)
;
{ MEnc = no },
return
))))))).
% [24] VersionInfo ::= S 'version' Eq (' VersionNum ' | " VersionNum ")
:- pred versionInfo(pstate(_), pstate(unit)).
:- mode versionInfo(in, out) is det.
versionInfo -->
s then (pred(_::in, pdi, puo) is det -->
lit("version") then (pred(_::in, pdi, puo) is det -->
eq then (pred(_::in, pdi, puo) is det -->
quote then (pred(Q::in, pdi, puo) is det -->
versionNum then (pred(_::in, pdi, puo) is det -->
quote then (pred(EndQ::in, pdi, puo) is det -->
( { Q = EndQ } ->
return
;
error("mismatched quotes")
))))))).
% [25] Eq ::= S? '=' S?
:- pred eq(pstate(_), pstate(unit)).
:- mode eq(in, out) is det.
eq -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('=') then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
return
))).
% [26] VersionNum ::= ([a-zA-Z0-9_.:] | '-')+
:- pred versionNum(pstate(_), pstate(list(unicode))).
:- mode versionNum(in, out) is det.
versionNum -->
plus(versionNumChar).
:- pred versionNumChar(pstate(_), pstate(unicode)).
:- mode versionNumChar(in, out) is det.
versionNumChar -->
tok then (pred(C::in, pdi, puo) is det -->
( {
C >= a, C =< z
;
C >= 'A', C =< 'Z'
;
C >= '0', C =< '9'
;
char__to_int(Ch, C),
contains_char("_.:-", Ch)
} ->
return(C)
;
fail("not a versionNum character")
)).
% [27] Misc ::= Comment | PI | S
:- pred misc(pstate(_), pstate(opt(ref(doc.content)))).
:- mode misc(in, out) is det.
misc -->
opt(comment or pi) then (pred(MContent::in, pdi, puo) is det -->
(
{ MContent = yes(_) },
return(MContent)
;
{ MContent = no },
no(s)
)).
:- pred filterOpt(list(opt(T)), list(T)).
:- mode filterOpt(in, out) is det.
filterOpt([], []).
filterOpt([M0|Ms0], Ms) :-
filterOpt(Ms0, Ms1),
(
M0 = yes(M),
Ms = [M|Ms1]
;
M0 = no,
Ms = Ms1
).
%
% The XML document type declaration contains or points to markup
% declarations that provide a grammar for a class of documents. This
% grammar is known as a document type definition, or DTD. The document
% type declaration can point to an external subset (a special kind of
% external entity) containing markup declarations, or can contain the
% markup declarations directly in an internal subset, or can do both.
% The DTD for a document consists of both subsets taken together.
%
% A markup declaration is an element type declaration, an attribute-list
% declaration, an entity declaration, or a notation declaration. These
% declarations may be contained in whole or in part within parameter
% entities, as described in the well-formedness and validity constraints
% below. For fuller information, see "4. Physical Structures".
%
% Document Type Definition
% [28] doctypedecl ::= '<!DOCTYPE' S Name (S ExternalID)? S? ('['
% (markupdecl | PEReference | S)* ']' S?)? '>' [ VC: Root Element Type ]
:- pred doctypedecl(pstate(_), pstate(dtd)).
:- mode doctypedecl(in, out) is det.
doctypedecl -->
(lit("<!DOCTYPE") then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
name then (pred(Root::in, pdi, puo) is det -->
opt(snd(s and externalID)) then (pred(MExId::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
opt(lit1(('['), unit) and
parseInternalSubSet and
(lit1((']'), unit) and opt(s)))
then (pred(_::in, pdi, puo) is det -->
lit1(('>')) then (pred(_::in, pdi, puo) is det -->
opt(MExId, parseExtSubset, return)
then (pred(_::in, pdi, puo) is det -->
get(gEntities, entities(Entities)),
get(gPEntities, pentities(PEntities)),
get(gElements, elements(Elements)),
return(dtd(Root, Elements, Entities, PEntities))
))))))))).
:- pred parseInternalSubSet(pstate(_), pstate(unit)).
:- mode parseInternalSubSet(pdi, puo) is det.
parseInternalSubSet -->
star(no(markupdecl) or no(s) or no(pEReference(parseInternalSubSet)))
then (pred(_::in, pdi, puo) is det -->
return
).
:- pred parseExtSubset(externalId, pstate(_), pstate(unit)).
:- mode parseExtSubset(in, pdi, puo) is det.
parseExtSubset(ExId) -->
getEntity(external(ExId)) then (pred(Entity::in, pdi, puo) is det -->
parseEntity(extSubset, mkEntity(Entity))
then (pred(_::in, pdi, puo) is det -->
return
)).
% [29] markupdecl ::= elementdecl | AttlistDecl | EntityDecl
% | NotationDecl | PI | Comment [ VC: Proper Declaration/PE Nesting ]
% [ WFC: PEs in Internal Subset ]
:- pred markupdecl(pstate(_), pstate(unit)).
:- mode markupdecl(in, out) is det.
markupdecl -->
no(elementdecl) or no(attlistDecl) or no(entityDecl) or
no(pi) or no(comment) then (pred(_::in, pdi, puo) is det -->
return
).
%
% The markup declarations may be made up in whole or in part of the
% replacement text of parameter entities. The productions later in this
% specification for individual nonterminals (elementdecl, AttlistDecl,
% and so on) describe the declarations after all the parameter entities
% have been included.
%
% Validity Constraint: Root Element Type
% The Name in the document type declaration must match the element type
% of the root element.
%
% Validity Constraint: Proper Declaration/PE Nesting
% Parameter-entity replacement text must be properly nested with markup
% declarations. That is to say, if either the first character or the
% last character of a markup declaration (markupdecl above) is contained
% in the replacement text for a parameter-entity reference, both must be
% contained in the same replacement text.
%
% Well-Formedness Constraint: PEs in Internal Subset
% In the internal DTD subset, parameter-entity references can occur only
% where markup declarations can occur, not within markup declarations.
% (This does not apply to references that occur in external parameter
% entities or to the external subset.)
%
% Like the internal subset, the external subset and any external
% parameter entities referred to in the DTD must consist of a series of
% complete markup declarations of the types allowed by the non-terminal
% symbol markupdecl, interspersed with white space or parameter-entity
% references. However, portions of the contents of the external subset
% or of external parameter entities may conditionally be ignored by
% using the conditional section construct; this is not allowed in the
% internal subset.
%
% External Subset
% [30] extSubset ::= TextDecl? extSubsetDecl
:- pred extSubset(pstate(T1), pstate(unit)).
:- mode extSubset(in, out) is det.
extSubset -->
getEncoding(Enc),
(opt(textDecl) then (pred(_::in, pdi, puo) is det -->
extSubsetDecl then (pred(_::in, pdi, puo) is det -->
return
))),
setEncoding(Enc).
% [31] extSubsetDecl ::= ( markupdecl | conditionalSect | PEReference
% | S )*
:- pred extSubsetDecl(pstate(_), pstate(unit)).
:- mode extSubsetDecl(in, out) is det.
extSubsetDecl -->
star(no(markupdecl) or no(conditionalSect) or
no(pEReference(extSubsetDecl)) or no(s))
then (pred(_::in, pdi, puo) is det -->
return
).
%
% The external subset and external parameter entities also differ from
% the internal subset in that in them, parameter-entity references are
% permitted within markup declarations, not only between markup
% declarations.
%
% An example of an XML document with a document type declaration:
%
% <?xml version="1.0"?>
% <!DOCTYPE greeting SYSTEM "hello.dtd">
% <greeting>Hello, world!</greeting>
%
% The system identifier "hello.dtd" gives the URI of a DTD for the
% document.
%
% The declarations can also be given locally, as in this example:
%
% <?xml version="1.0" encoding="UTF-8" ?>
% <!DOCTYPE greeting [
% <!ELEMENT greeting (#PCDATA)>
% ]>
% <greeting>Hello, world!</greeting>
%
% If both the external and internal subsets are used, the internal
% subset is considered to occur before the external subset. This has the
% effect that entity and attribute-list declarations in the internal
% subset take precedence over those in the external subset.
%
% 2.9 Standalone Document Declaration
%
% Markup declarations can affect the content of the document, as passed
% from an XML processor to an application; examples are attribute
% defaults and entity declarations. The standalone document declaration,
% which may appear as a component of the XML declaration, signals
% whether or not there are such declarations which appear external to
% the document entity.
%
% Standalone Document Declaration
% [32] SDDecl ::= S 'standalone' Eq (("'" ('yes' | 'no') "'") | ('"'
% ('yes' | 'no') '"')) [ VC: Standalone Document Declaration ]
:- pred sdDecl(pstate(_), pstate(unit)).
:- mode sdDecl(in, out) is det.
sdDecl -->
s then (pred(_::in, pdi, puo) is det -->
lit("standalone") then (pred(_::in, pdi, puo) is det -->
eq then (pred(_::in, pdi, puo) is det -->
quote then (pred(Q::in, pdi, puo) is det -->
(lit("yes") or lit("no")) then (pred(_::in, pdi, puo) is det -->
quote then (pred(EndQ::in, pdi, puo) is det -->
( { Q = EndQ } ->
return
;
error("mismatched quotes")
))))))).
%
% In a standalone document declaration, the value "yes" indicates that
% there are no markup declarations external to the document entity
% (either in the DTD external subset, or in an external parameter entity
% referenced from the internal subset) which affect the information
% passed from the XML processor to the application. The value "no"
% indicates that there are or may be such external markup declarations.
% Note that the standalone document declaration only denotes the
% presence of external declarations; the presence, in a document, of
% references to external entities, when those entities are internally
% declared, does not change its standalone status.
%
% If there are no external markup declarations, the standalone document
% declaration has no meaning. If there are external markup declarations
% but there is no standalone document declaration, the value "no" is
% assumed.
%
% Any XML document for which standalone="no" holds can be converted
% algorithmically to a standalone document, which may be desirable for
% some network delivery applications.
%
% Validity Constraint: Standalone Document Declaration
% The standalone document declaration must have the value "no" if any
% external markup declarations contain declarations of:
% * attributes with default values, if elements to which these
% attributes apply appear in the document without specifications of
% values for these attributes, or
% * entities (other than amp, lt, gt, apos, quot), if references to
% those entities appear in the document, or
% * attributes with values subject to normalization, where the
% attribute appears in the document with a value which will change
% as a result of normalization, or
% * element types with element content, if white space occurs directly
% within any instance of those types.
%
% An example XML declaration with a standalone document declaration:
%
% <?xml version="1.0" standalone='yes'?>
%
% 2.10 White Space Handling
%
% In editing XML documents, it is often convenient to use "white space"
% (spaces, tabs, and blank lines, denoted by the nonterminal S in this
% specification) to set apart the markup for greater readability. Such
% white space is typically not intended for inclusion in the delivered
% version of the document. On the other hand, "significant" white space
% that should be preserved in the delivered version is common, for
% example in poetry and source code.
%
% An XML processor must always pass all characters in a document that
% are not markup through to the application. A validating XML processor
% must also inform the application which of these characters constitute
% white space appearing in element content.
%
% A special attribute named xml:space may be attached to an element to
% signal an intention that in that element, white space should be
% preserved by applications. In valid documents, this attribute, like
% any other, must be declared if it is used. When declared, it must be
% given as an enumerated type whose only possible values are "default"
% and "preserve". For example:
%
% <!ATTLIST poem xml:space (default|preserve) 'preserve'>
%
% The value "default" signals that applications' default white-space
% processing modes are acceptable for this element; the value "preserve"
% indicates the intent that applications preserve all the white space.
% This declared intent is considered to apply to all elements within the
% content of the element where it is specified, unless overriden with
% another instance of the xml:space attribute.
%
% The root element of any document is considered to have signaled no
% intentions as regards application space handling, unless it provides a
% value for this attribute or the attribute is declared with a default
% value.
%
% 2.11 End-of-Line Handling
%
% XML parsed entities are often stored in computer files which, for
% editing convenience, are organized into lines. These lines are
% typically separated by some combination of the characters
% carriage-return (#xD) and line-feed (#xA).
%
% To simplify the tasks of applications, wherever an external parsed
% entity or the literal entity value of an internal parsed entity
% contains either the literal two-character sequence "#xD#xA" or a
% standalone literal #xD, an XML processor must pass to the application
% the single character #xA. (This behavior can conveniently be produced
% by normalizing all line breaks to #xA on input, before parsing.)
%
% 2.12 Language Identification
%
% In document processing, it is often useful to identify the natural or
% formal language in which the content is written. A special attribute
% named xml:lang may be inserted in documents to specify the language
% used in the contents and attribute values of any element in an XML
% document. In valid documents, this attribute, like any other, must be
% declared if it is used. The values of the attribute are language
% identifiers as defined by [IETF RFC 1766], "Tags for the
% Identification of Languages":
%
% Language Identification
% [33] LanguageID ::= Langcode ('-' Subcode)*
:- pred languageID(pstate(_), pstate(unit)).
:- mode languageID(in, out) is det.
languageID -->
langcode then (pred(_::in, pdi, puo) is det -->
star(lit1('-') and subcode) then (pred(_::in, pdi, puo) is det -->
return
)).
% [34] Langcode ::= ISO639Code | IanaCode | UserCode
:- pred langcode(pstate(_), pstate(unit)).
:- mode langcode(in, out) is det.
langcode -->
(x(iso639Code) or x(ianaCode) or x(userCode))
then (pred(_::in, pdi, puo) is det -->
return
).
% [35] ISO639Code ::= ([a-z] | [A-Z]) ([a-z] | [A-Z])
:- pred iso639Code(pstate(_), pstate((unicode, unicode))).
:- mode iso639Code(in, out) is det.
iso639Code -->
(range(a, z) or range('A', 'Z')) and (range(a, z) or range('A', 'Z')).
% [36] IanaCode ::= ('i' | 'I') '-' ([a-z] | [A-Z])+
:- pred ianaCode(pstate(_), pstate((unicode, unicode, list(unicode)))).
:- mode ianaCode(in, out) is det.
ianaCode -->
(lit1('i') or lit1('I')) and
lit1('-') and
plus(range(a, z) or range('A', 'Z')).
% [37] UserCode ::= ('x' | 'X') '-' ([a-z] | [A-Z])+
:- pred userCode(pstate(_), pstate((unicode, unicode, list(unicode)))).
:- mode userCode(in, out) is det.
userCode -->
(lit1('x') or lit1('X')) and
lit1('-') and
plus(range(a, z) or range('A', 'Z')).
% [38] Subcode ::= ([a-z] | [A-Z])+
:- pred subcode(pstate(_), pstate(list(unicode))).
:- mode subcode(in, out) is det.
subcode -->
plus(range(a, z) or range('A', 'Z')).
%
% The Langcode may be any of the following:
% * a two-letter language code as defined by [ISO 639], "Codes for the
% representation of names of languages"
% * a language identifier registered with the Internet Assigned
% Numbers Authority [IANA]; these begin with the prefix "i-" (or
% "I-")
% * a language identifier assigned by the user, or agreed on between
% parties in private use; these must begin with the prefix "x-" or
% "X-" in order to ensure that they do not conflict with names later
% standardized or registered with IANA
%
% There may be any number of Subcode segments; if the first subcode
% segment exists and the Subcode consists of two letters, then it must
% be a country code from [ISO 3166], "Codes for the representation of
% names of countries." If the first subcode consists of more than two
% letters, it must be a subcode for the language in question registered
% with IANA, unless the Langcode begins with the prefix "x-" or "X-".
%
% It is customary to give the language code in lower case, and the
% country code (if any) in upper case. Note that these values, unlike
% other names in XML documents, are case insensitive.
%
% For example:
%
% <p xml:lang="en">The quick brown fox jumps over the lazy dog.</p>
% <p xml:lang="en-GB">What colour is it?</p>
% <p xml:lang="en-US">What color is it?</p>
% <sp who="Faust" desc='leise' xml:lang="de">
% <l>Habe nun, ach! Philosophie,</l>
% <l>Juristerei, und Medizin</l>
% <l>und leider auch Theologie</l>
% <l>durchaus studiert mit hei<EFBFBD>em Bem<EFBFBD>h'n.</l>
% </sp>
%
% The intent declared with xml:lang is considered to apply to all
% attributes and content of the element where it is specified, unless
% overridden with an instance of xml:lang on another element within that
% content.
%
% A simple declaration for xml:lang might take the form
%
% xml:lang NMTOKEN #IMPLIED
%
% but specific default values may also be given, if appropriate. In a
% collection of French poems for English students, with glosses and
% notes in English, the xml:lang attribute might be declared this way:
%
% <!ATTLIST poem xml:lang NMTOKEN 'fr'>
% <!ATTLIST gloss xml:lang NMTOKEN 'en'>
% <!ATTLIST note xml:lang NMTOKEN 'en'>
%
%3. Logical Structures
%
% Each XML document contains one or more elements, the boundaries of
% which are either delimited by start-tags and end-tags, or, for empty
% elements, by an empty-element tag. Each element has a type, identified
% by name, sometimes called its "generic identifier" (GI), and may have
% a set of attribute specifications. Each attribute specification has a
% name and a value.
%
% Element
% [39] element ::= EmptyElemTag
% | STag content ETag [ WFC: Element Type Match ]
% [ VC: Element Valid ]
:- pred element(pstate(_), pstate(ref(doc.content))).
:- mode element(in, out) is det.
element -->
emptyElemTag or nonEmptyElement.
:- pred nonEmptyElement(pstate(_), pstate(ref(doc.content))).
:- mode nonEmptyElement(in, out) is det.
nonEmptyElement -->
sTag then (pred((Name, Attrs)::in, pdi, puo) is det -->
content then (pred(Content::in, pdi, puo) is det -->
eTag then (pred(EndName::in, pdi, puo) is det -->
( { Name = EndName } ->
{ Element = element(Name, Attrs, Content) },
add(element(Element), Ref),
return(Ref)
;
{ format("start tag name `%s' and end tag name `%s' do not match",
[s(Name), s(EndName)], Msg) },
error(Msg)
)))).
%
% This specification does not constrain the semantics, use, or (beyond
% syntax) names of the element types and attributes, except that names
% beginning with a match to (('X'|'x')('M'|'m')('L'|'l')) are reserved
% for standardization in this or future versions of this specification.
%
% Well-Formedness Constraint: Element Type Match
% The Name in an element's end-tag must match the element type in the
% start-tag.
%
% Validity Constraint: Element Valid
% An element is valid if there is a declaration matching elementdecl
% where the Name matches the element type, and one of the following
% holds:
% 1. The declaration matches EMPTY and the element has no content.
% 2. The declaration matches children and the sequence of child
% elements belongs to the language generated by the regular
% expression in the content model, with optional white space
% (characters matching the nonterminal S) between each pair of child
% elements.
% 3. The declaration matches Mixed and the content consists of
% character data and child elements whose types match names in the
% content model.
% 4. The declaration matches ANY, and the types of any child elements
% have been declared.
%
% 3.1 Start-Tags, End-Tags, and Empty-Element Tags
%
% The beginning of every non-empty XML element is marked by a start-tag.
%
% Start-tag
% [40] STag ::= '<' Name (S Attribute)* S? '>' [ WFC: Unique Att Spec ]
:- pred sTag(pstate(_), pstate((name, list(doc.attribute)))).
:- mode sTag(in, out) is det.
sTag -->
lit1('<') then (pred(_::in, pdi, puo) is det -->
name then (pred(Name::in, pdi, puo) is det -->
star(snd(s and attribute)) then (pred(Attrs::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('>') then (pred(_::in, pdi, puo) is det -->
return((Name, Attrs))
))))).
% [41] Attribute ::= Name Eq AttValue [ VC: Attribute Value Type ]
% [ WFC: No External Entity References ]
% [ WFC: No < in Attribute Values ]
:- pred attribute(pstate(_), pstate(doc.attribute)).
:- mode attribute(in, out) is det.
attribute -->
name then (pred(Name::in, pdi, puo) is det -->
eq then (pred(_::in, pdi, puo) is det -->
attValue then (pred(Value::in, pdi, puo) is det -->
return(attribute(Name, Value))
))).
%
% The Name in the start- and end-tags gives the element's type. The
% Name-AttValue pairs are referred to as the attribute specifications
% of the element, with the Name in each pair referred to as the
% attribute name and the content of the AttValue (the text between the
% ' or " delimiters) as the attribute value.
%
% Well-Formedness Constraint: Unique Att Spec
% No attribute name may appear more than once in the same start-tag or
% empty-element tag.
%
% Validity Constraint: Attribute Value Type
% The attribute must have been declared; the value must be of the type
% declared for it. (For attribute types, see "3.3 Attribute-List
% Declarations".)
%
% Well-Formedness Constraint: No External Entity References
% Attribute values cannot contain direct or indirect entity references
% to external entities.
%
% Well-Formedness Constraint: No < in Attribute Values
% The replacement text of any entity referred to directly or indirectly
% in an attribute value (other than "&lt;") must not contain a <.
%
% An example of a start-tag:
%
% <termdef id="dt-dog" term="dog">
%
% The end of every element that begins with a start-tag must be marked
% by an end-tag containing a name that echoes the element's type as
% given in the start-tag:
%
% End-tag
% [42] ETag ::= '</' Name S? '>'
:- pred eTag(pstate(_), pstate(name)).
:- mode eTag(in, out) is det.
eTag -->
lit("</") then (pred(_::in, pdi, puo) is det -->
name then (pred(Name::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('>') then (pred(_::in, pdi, puo) is det -->
return(Name)
)))).
%
% An example of an end-tag:
%
% </termdef>
%
% The text between the start-tag and end-tag is called the element's
% content:
%
% Content of Elements
% [43] content ::= (element | CharData | Reference | CDSect | PI
% | Comment)*
:- pred content(pstate(_), pstate(list(ref(doc.content)))).
:- mode content(in, out) is det.
content -->
star(list(element) or list(charData) or
list(cdSect) or list(pi) or
list(comment) or entityRef(content))
then (pred(Css::in, pdi, puo) is det -->
{ condense(Css, Cs) },
return(Cs)
).
%
% If an element is empty, it must be represented either by a start-tag
% immediately followed by an end-tag or by an empty-element tag. An
% empty-element tag takes a special form:
%
% Tags for Empty Elements
% [44] EmptyElemTag ::= '<' Name (S Attribute)* S? '/>' [ WFC: Unique
% Att Spec ]
:- pred emptyElemTag(pstate(_), pstate(ref(doc.content))).
:- mode emptyElemTag(in, out) is det.
emptyElemTag -->
lit1('<') then (pred(_::in, pdi, puo) is det -->
name then (pred(Name::in, pdi, puo) is det -->
star(snd(s and attribute)) then (pred(Attrs::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit("/>") then (pred(_::in, pdi, puo) is det -->
{ Element = element(Name, Attrs, []) },
add(element(Element), Ref),
return(Ref)
))))).
%
% Empty-element tags may be used for any element which has no content,
% whether or not it is declared using the keyword EMPTY. For
% interoperability, the empty-element tag must be used, and can only be
% used, for elements which are declared EMPTY.
%
% Examples of empty elements:
%
% <IMG align="left"
% src="http://www.w3.org/Icons/WWW/w3c_home" />
% <br></br>
% <br/>
%
% 3.2 Element Type Declarations
%
% The element structure of an XML document may, for validation purposes,
% be constrained using element type and attribute-list declarations. An
% element type declaration constrains the element's content.
%
% Element type declarations often constrain which element types can
% appear as children of the element. At user option, an XML processor
% may issue a warning when a declaration mentions an element type for
% which no declaration is provided, but this is not an error.
%
% An element type declaration takes the form:
%
% Element Type Declaration
% [45] elementdecl ::= '<!ELEMENT' S Name S contentspec S? '>' [ VC:
% Unique Element Type Declaration ]
:- pred elementdecl(pstate(_), pstate(unit)).
:- mode elementdecl(in, out) is det.
elementdecl -->
lit("<!ELEMENT") then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pe(name) then (pred(Name::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pe(contentspec) then (pred(Content::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('>') then (pred(_::in, pdi, puo) is det -->
{ init(EmptyAttrs) },
{ Element = element(Name, EmptyAttrs, Content) },
get(gElements, elements(Elements0)),
{ set(Elements0, Name, Element, Elements) },
set(gElements, elements(Elements)),
return
))))))).
% [46] contentspec ::= 'EMPTY' | 'ANY' | Mixed | children
:- pred contentspec(pstate(_), pstate(dtd.content)).
:- mode contentspec(in, out) is det.
contentspec -->
lit("EMPTY", empty) or lit("ANY", any) or mixed or children.
%
% where the Name gives the element type being declared.
%
% Validity Constraint: Unique Element Type Declaration
% No element type may be declared more than once.
%
% Examples of element type declarations:
%
% <!ELEMENT br EMPTY>
% <!ELEMENT p (#PCDATA|emph)* >
% <!ELEMENT %name.para; %content.para; >
% <!ELEMENT container ANY>
%
% 3.2.1 Element Content
%
% An element type has element content when elements of that type must
% contain only child elements (no character data), optionally separated
% by white space (characters matching the nonterminal S). In this case,
% the constraint includes a content model, a simple grammar governing
% the allowed types of the child elements and the order in which they
% are allowed to appear. The grammar is built on content particles
% (cps), which consist of names, choice lists of content particles, or
% sequence lists of content particles:
%
% Element-content Models
% [47] children ::= (choice | seq) ('?' | '*' | '+')?
:- pred children(pstate(_), pstate(dtd.content)).
:- mode children(in, out) is det.
children -->
(choice or seq) then (pred(Children::in, pdi, puo) is det -->
multiplicity then (pred(Mult::in, pdi, puo) is det -->
{ Content = children(Children - Mult) },
return(Content)
)).
:- pred multiplicity(pstate(_), pstate(multiplicity)).
:- mode multiplicity(in, out) is det.
multiplicity -->
opt(lit1(('?'), ('?')) or lit1(('*'), ('*')) or lit1(('+'), ('+')), '1').
% [48] cp ::= (Name | choice | seq) ('?' | '*' | '+')?
:- pred cp(pstate(_), pstate(contentParticle)).
:- mode cp(in, out) is det.
cp -->
((wrap(name, nameToChild) or choice or seq) and multiplicity)
then (pred((Kid, Mult)::in, pdi, puo) is det -->
{ CP = (Kid - Mult) },
return(CP)
).
:- pred nameToChild(name, children).
:- mode nameToChild(in, out) is det.
nameToChild(Name, element(Name)).
% [49] choice ::= '(' S? cp ( S? '|' S? cp )* S? ')' [ VC: Proper
% Group/PE Nesting ]
:- pred choice(pstate(_), pstate(children)).
:- mode choice(in, out) is det.
choice -->
lit1('(') then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
pe(cp) then (pred(Child::in, pdi, puo) is det -->
star(snd((opt(s) and lit1('|') and opt(s)) and pe(cp)))
then (pred(Children0::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1(')') then (pred(_::in, pdi, puo) is det -->
{ Children = alt([Child|Children0]) },
return(Children)
)))))).
% [50] seq ::= '(' S? cp ( S? ',' S? cp )* S? ')' [ VC: Proper Group/PE
% Nesting ]
:- pred seq(pstate(_), pstate(children)).
:- mode seq(in, out) is det.
seq -->
lit1('(') then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
pe(cp) then (pred(Child::in, pdi, puo) is det -->
star(snd((opt(s) and lit1(',') and opt(s)) and pe(cp)))
then (pred(Children0::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1(')') then (pred(_::in, pdi, puo) is det -->
{ Children = seq([Child|Children0]) },
return(Children)
)))))).
%
% where each Name is the type of an element which may appear as a child.
% Any content particle in a choice list may appear in the element
% content at the location where the choice list appears in the grammar;
% content particles occurring in a sequence list must each appear in the
% element content in the order given in the list. The optional character
% following a name or list governs whether the element or the content
% particles in the list may occur one or more (+), zero or more (*), or
% zero or one times (?). The absence of such an operator means that the
% element or content particle must appear exactly once. This syntax and
% meaning are identical to those used in the productions in this
% specification.
%
% The content of an element matches a content model if and only if it is
% possible to trace out a path through the content model, obeying the
% sequence, choice, and repetition operators and matching each element
% in the content against an element type in the content model. For
% compatibility, it is an error if an element in the document can match
% more than one occurrence of an element type in the content model. For
% more information, see "E. Deterministic Content Models".
%
% Validity Constraint: Proper Group/PE Nesting
% Parameter-entity replacement text must be properly nested with
% parenthetized groups. That is to say, if either of the opening or
% closing parentheses in a choice, seq, or Mixed construct is contained
% in the replacement text for a parameter entity, both must be contained
% in the same replacement text. For interoperability, if a
% parameter-entity reference appears in a choice, seq, or Mixed
% construct, its replacement text should not be empty, and neither the
% first nor last non-blank character of the replacement text should be a
% connector (| or ,).
%
% Examples of element-content models:
%
% <!ELEMENT spec (front, body, back?)>
% <!ELEMENT div1 (head, (p | list | note)*, div2*)>
% <!ELEMENT dictionary-body (%div.mix; | %dict.mix;)*>
%
% 3.2.2 Mixed Content
%
% An element type has mixed content when elements of that type may
% contain character data, optionally interspersed with child elements.
% In this case, the types of the child elements may be constrained, but
% not their order or their number of occurrences:
%
% Mixed-content Declaration
% [51] Mixed ::= '(' S? '#PCDATA' (S? '|' S? Name)* S? ')*'
% | '(' S? '#PCDATA' S? ')' [ VC: Proper Group/PE Nesting ]
% [ VC: No Duplicate Types ]
:- pred mixed(pstate(_), pstate(dtd.content)).
:- mode mixed(in, out) is det.
mixed -->
mixed1 or mixed2.
:- pred mixed1(pstate(_), pstate(dtd.content)).
:- mode mixed1(in, out) is det.
mixed1 -->
lit1('(') then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit("#PCDATA") then (pred(_::in, pdi, puo) is det -->
star(snd((opt(s) and lit1('|') and opt(s)) and name))
then (pred(Names::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit(")*") then (pred(_::in, pdi, puo) is det -->
{ Content = mixed(mixed(Names)) },
return(Content)
)))))).
:- pred mixed2(pstate(_), pstate(dtd.content)).
:- mode mixed2(in, out) is det.
mixed2 -->
lit1('(') and opt(s) and lit("#PCDATA") and opt(s) and lit1(')')
then (pred(_::in, pdi, puo) is det -->
return(mixed(mixed([])))
).
%
% where the Names give the types of elements that may appear as
% children.
%
% Validity Constraint: No Duplicate Types
% The same name must not appear more than once in a single mixed-content
% declaration.
%
% Examples of mixed content declarations:
%
% <!ELEMENT p (#PCDATA|a|ul|b|i|em)*>
% <!ELEMENT p (#PCDATA | %font; | %phrase; | %special; | %form;)* >
% <!ELEMENT b (#PCDATA)>
%
% 3.3 Attribute-List Declarations
%
% Attributes are used to associate name-value pairs with elements.
% Attribute specifications may appear only within start-tags and
% empty-element tags; thus, the productions used to recognize them
% appear in "3.1 Start-Tags, End-Tags, and Empty-Element Tags".
% Attribute-list declarations may be used:
% * To define the set of attributes pertaining to a given element
% type.
% * To establish type constraints for these attributes.
% * To provide default values for attributes.
%
% Attribute-list declarations specify the name, data type, and default
% value (if any) of each attribute associated with a given element type:
%
% Attribute-list Declaration
% [52] AttlistDecl ::= '<!ATTLIST' S Name AttDef* S? '>'
:- pred attlistDecl(pstate(_), pstate(unit)).
:- mode attlistDecl(in, out) is det.
attlistDecl -->
lit("<!ATTLIST") then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pe(name) then (pred(Name::in, pdi, puo) is det -->
pe(star(attDef)) then (pred(AttsList::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('>') then (pred(_::in, pdi, puo) is det -->
{ init(Atts0) },
(
{ foldl((pred(Att::in, Atts1::in, Atts2::out) is semidet :-
Att = attribute(AttName, _, _),
insert(Atts1, AttName, Att, Atts2)
), AttsList, Atts0, Atts) }
->
get(gElements, elements(Elements0)),
( { search(Elements0, Name, Element0) } ->
{ Attrs = merge(Element0^eAttrs, Atts) },
{ Element = Element0^eAttrs := Attrs },
{ set(Elements0, Name, Element, Elements) },
set(gElements, elements(Elements))
;
get(gAttributes, attributes(Attributes0)),
{ set(Attributes0, Name, Atts, Attributes) },
set(gAttributes, attributes(Attributes))
),
return
;
error("duplicated attribute")
))))))).
% [53] AttDef ::= S Name S AttType S DefaultDecl
:- pred attDef(pstate(_), pstate(dtd.attribute)).
:- mode attDef(in, out) is det.
attDef -->
s then (pred(_::in, pdi, puo) is det -->
pe(name) then (pred(Name::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pe(attType) then (pred(Type::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pe(defaultDecl) then (pred(Default::in, pdi, puo) is det -->
return(attribute(Name, Type, Default))
)))))).
%
% The Name in the AttlistDecl rule is the type of an element. At user
% option, an XML processor may issue a warning if attributes are
% declared for an element type not itself declared, but this is not an
% error. The Name in the AttDef rule is the name of the attribute.
%
% When more than one AttlistDecl is provided for a given element type,
% the contents of all those provided are merged. When more than one
% definition is provided for the same attribute of a given element type,
% the first declaration is binding and later declarations are ignored.
% For interoperability, writers of DTDs may choose to provide at most
% one attribute-list declaration for a given element type, at most one
% attribute definition for a given attribute name, and at least one
% attribute definition in each attribute-list declaration. For
% interoperability, an XML processor may at user option issue a warning
% when more than one attribute-list declaration is provided for a given
% element type, or more than one attribute definition is provided for a
% given attribute, but this is not an error.
%
% 3.3.1 Attribute Types
%
% XML attribute types are of three kinds: a string type, a set of
% tokenized types, and enumerated types. The string type may take any
% literal string as a value; the tokenized types have varying lexical
% and semantic constraints, as noted:
%
% Attribute Types
% [54] AttType ::= StringType | TokenizedType | EnumeratedType
:- pred attType(pstate(_), pstate(type)).
:- mode attType(in, out) is det.
attType -->
stringType or tokenizedType or enumeratedType.
% [55] StringType ::= 'CDATA'
:- pred stringType(pstate(_), pstate(type)).
:- mode stringType(in, out) is det.
stringType -->
lit("CDATA") then (pred(_::in, pdi, puo) is det -->
return(cdata)
).
% [56] TokenizedType ::= 'ID' [ VC: ID ]
% [ VC: One ID per Element Type ]
% [ VC: ID Attribute Default ]
% | 'IDREF' [ VC: IDREF ]
% | 'IDREFS' [ VC: IDREF ]
% | 'ENTITY' [ VC: Entity Name ]
% | 'ENTITIES' [ VC: Entity Name ]
% | 'NMTOKEN' [ VC: Name Token ]
% | 'NMTOKENS' [ VC: Name Token ]
:- pred tokenizedType(pstate(_), pstate(type)).
:- mode tokenizedType(in, out) is det.
tokenizedType -->
% Because or/4 commits to the first match, the ordering here
% is significant because IDREFS matches IDREF matches ID, etc.
lit("IDREFS", idrefs) or lit("IDREF", idref) or lit("ID", id) or
lit("ENTITY", entity) or lit("ENTITIES", entities) or
lit("NMTOKENS", nmtokens) or lit("NMTOKEN", nmtoken).
%
% Validity Constraint: ID
% Values of type ID must match the Name production. A name must not
% appear more than once in an XML document as a value of this type;
% i.e., ID values must uniquely identify the elements which bear them.
%
% Validity Constraint: One ID per Element Type
% No element type may have more than one ID attribute specified.
%
% Validity Constraint: ID Attribute Default
% An ID attribute must have a declared default of #IMPLIED or #REQUIRED.
%
% Validity Constraint: IDREF
% Values of type IDREF must match the Name production, and values of
% type IDREFS must match Names; each Name must match the value of an ID
% attribute on some element in the XML document; i.e. IDREF values must
% match the value of some ID attribute.
%
% Validity Constraint: Entity Name
% Values of type ENTITY must match the Name production, values of type
% ENTITIES must match Names; each Name must match the name of an
% unparsed entity declared in the DTD.
%
% Validity Constraint: Name Token
% Values of type NMTOKEN must match the Nmtoken production; values of
% type NMTOKENS must match Nmtokens.
%
% Enumerated attributes can take one of a list of values provided in the
% declaration. There are two kinds of enumerated types:
%
% Enumerated Attribute Types
% [57] EnumeratedType ::= NotationType | Enumeration
:- pred enumeratedType(pstate(_), pstate(type)).
:- mode enumeratedType(in, out) is det.
enumeratedType -->
notationType or enumeration.
% [58] NotationType ::= 'NOTATION' S '(' S? Name (S? '|' S? Name)* S?
% ')' [ VC: Notation Attributes ]
:- pred notationType(pstate(_), pstate(type)).
:- mode notationType(in, out) is det.
notationType -->
lit("NOTATION") then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
lit1('(') then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
nmtoken then (pred(Token::in, pdi, puo) is det -->
star(snd((opt(s) and lit1('|') and opt(s)) and nmtoken))
then (pred(Tokens::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1(')') then (pred(_::in, pdi, puo) is det -->
{ Type = notation([Token|Tokens]) },
return(Type)
)))))))).
% [59] Enumeration ::= '(' S? Nmtoken (S? '|' S? Nmtoken)* S? ')' [ VC:
% Enumeration ]
:- pred enumeration(pstate(_), pstate(type)).
:- mode enumeration(in, out) is det.
enumeration -->
lit1('(') then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
nmtoken then (pred(Token::in, pdi, puo) is det -->
star(snd((opt(s) and lit1('|') and opt(s)) and nmtoken))
then (pred(Tokens::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1(')') then (pred(_::in, pdi, puo) is det -->
{ Type = notation([Token|Tokens]) },
return(Type)
)))))).
%
% A NOTATION attribute identifies a notation, declared in the DTD with
% associated system and/or public identifiers, to be used in
% interpreting the element to which the attribute is attached.
%
% Validity Constraint: Notation Attributes
% Values of this type must match one of the notation names included in
% the declaration; all notation names in the declaration must be
% declared.
%
% Validity Constraint: Enumeration
% Values of this type must match one of the Nmtoken tokens in the
% declaration.
%
% For interoperability, the same Nmtoken should not occur more than once
% in the enumerated attribute types of a single element type.
%
% 3.3.2 Attribute Defaults
%
% An attribute declaration provides information on whether the
% attribute's presence is required, and if not, how an XML processor
% should react if a declared attribute is absent in a document.
%
% Attribute Defaults
% [60] DefaultDecl ::= '#REQUIRED' | '#IMPLIED'
% | (('#FIXED' S)? AttValue) [ VC: Required Attribute ]
% [ VC: Attribute Default Legal ]
% [ WFC: No < in Attribute Values ]
% [ VC: Fixed Attribute Default ]
:- pred defaultDecl(pstate(_), pstate(default)).
:- mode defaultDecl(in, out) is det.
defaultDecl -->
lit("#REQUIRED", required) or
lit("#IMPLIED", implied) or
wrap(snd((lit("#FIXED", unit) and s) and attValue), wrapFixed) or
wrap(attValue, wrapDefault).
:- pred wrapFixed(string, default).
:- mode wrapFixed(in, out) is det.
wrapFixed(AttValue, fixed(AttValue)).
:- pred wrapDefault(string, default).
:- mode wrapDefault(in, out) is det.
wrapDefault(AttValue, defaulted(AttValue)).
%
% In an attribute declaration, #REQUIRED means that the attribute must
% always be provided, #IMPLIED that no default value is provided. If the
% declaration is neither #REQUIRED nor #IMPLIED, then the AttValue value
% contains the declared default value; the #FIXED keyword states that
% the attribute must always have the default value. If a default value
% is declared, when an XML processor encounters an omitted attribute, it
% is to behave as though the attribute were present with the declared
% default value.
%
% Validity Constraint: Required Attribute
% If the default declaration is the keyword #REQUIRED, then the
% attribute must be specified for all elements of the type in the
% attribute-list declaration.
%
% Validity Constraint: Attribute Default Legal
% The declared default value must meet the lexical constraints of the
% declared attribute type.
%
% Validity Constraint: Fixed Attribute Default
% If an attribute has a default value declared with the #FIXED keyword,
% instances of that attribute must match the default value.
%
% Examples of attribute-list declarations:
%
% <!ATTLIST termdef
% id ID #REQUIRED
% name CDATA #IMPLIED>
% <!ATTLIST list
% type (bullets|ordered|glossary) "ordered">
% <!ATTLIST form
% method CDATA #FIXED "POST">
%
% 3.3.3 Attribute-Value Normalization
%
% Before the value of an attribute is passed to the application or
% checked for validity, the XML processor must normalize it as follows:
% * a character reference is processed by appending the referenced
% character to the attribute value
% * an entity reference is processed by recursively processing the
% replacement text of the entity
% * a whitespace character (#x20, #xD, #xA, #x9) is processed by
% appending #x20 to the normalized value, except that only a single
% #x20 is appended for a "#xD#xA" sequence that is part of an
% external parsed entity or the literal entity value of an internal
% parsed entity
% * other characters are processed by appending them to the normalized
% value
%
% If the declared value is not CDATA, then the XML processor must
% further process the normalized attribute value by discarding any
% leading and trailing space (#x20) characters, and by replacing
% sequences of space (#x20) characters by a single space (#x20)
% character.
%
% All attributes for which no declaration has been read should be
% treated by a non-validating parser as if declared CDATA.
%
% 3.4 Conditional Sections
%
% Conditional sections are portions of the document type declaration
% external subset which are included in, or excluded from, the logical
% structure of the DTD based on the keyword which governs them.
%
% Conditional Section
% [61] conditionalSect ::= includeSect | ignoreSect
:- pred conditionalSect(pstate(_), pstate(unit)).
:- mode conditionalSect(in, out) is det.
conditionalSect -->
includeSect or ignoreSect.
% [62] includeSect ::= '<![' S? 'INCLUDE' S? '[' extSubsetDecl ']]>'
:- pred includeSect(pstate(_), pstate(unit)).
:- mode includeSect(in, out) is det.
includeSect -->
lit("<![") then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit("INCLUDE") then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('[') then (pred(_::in, pdi, puo) is det -->
extSubsetDecl then (pred(_::in, pdi, puo) is det -->
lit("]]>") then (pred(_::in, pdi, puo) is det -->
return
))))))).
% [63] ignoreSect ::= '<![' S? 'IGNORE' S? '[' ignoreSectContents*
% ']]>'
:- pred ignoreSect(pstate(_), pstate(unit)).
:- mode ignoreSect(in, out) is det.
ignoreSect -->
lit("<![") then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit("IGNORE") then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('[') then (pred(_::in, pdi, puo) is det -->
star(ignoreSectContents) then (pred(_::in, pdi, puo) is det -->
return
)))))).
% [64] ignoreSectContents ::= Ignore ('<![' ignoreSectContents ']]>'
% Ignore)*
:- pred ignoreSectContents(pstate(_), pstate(unit)).
:- mode ignoreSectContents(in, out) is det.
ignoreSectContents -->
ignore then (pred(Ign::in, pdi, puo) is det -->
( { Ign = "<![" } ->
(star(ignoreSectContents and lit("]]>") and ignore)
then (pred(_::in, pdi, puo) is det -->
return
))
; % { Ign = "]]>" } ->
return
)).
% [65] Ignore ::= Char* - (Char* ('<![' | ']]>') Char*)
:- pred ignore(pstate(_), pstate(string)).
:- mode ignore(in, out) is det.
ignore -->
upto(char, lit("<![") or lit("]]>"))
then (pred(P::in, pdi, puo) is det -->
{ P = (_Ign, Terminal) },
return(Terminal)
).
%
% Like the internal and external DTD subsets, a conditional section may
% contain one or more complete declarations, comments, processing
% instructions, or nested conditional sections, intermingled with white
% space.
%
% If the keyword of the conditional section is INCLUDE, then the
% contents of the conditional section are part of the DTD. If the
% keyword of the conditional section is IGNORE, then the contents of the
% conditional section are not logically part of the DTD. Note that for
% reliable parsing, the contents of even ignored conditional sections
% must be read in order to detect nested conditional sections and ensure
% that the end of the outermost (ignored) conditional section is
% properly detected. If a conditional section with a keyword of INCLUDE
% occurs within a larger conditional section with a keyword of IGNORE,
% both the outer and the inner conditional sections are ignored.
%
% If the keyword of the conditional section is a parameter-entity
% reference, the parameter entity must be replaced by its content before
% the processor decides whether to include or ignore the conditional
% section.
%
% An example:
%
% <!ENTITY % draft 'INCLUDE' >
% <!ENTITY % final 'IGNORE' >
%
% <![%draft;[
% <!ELEMENT book (comments*, title, body, supplements?)>
% ]]>
% <![%final;[
% <!ELEMENT book (title, body, supplements?)>
% ]]>
%
%4. Physical Structures
%
% An XML document may consist of one or many storage units. These are
% called entities; they all have content and are all (except for the
% document entity, see below, and the external DTD subset) identified by
% name. Each XML document has one entity called the document entity,
% which serves as the starting point for the XML processor and may
% contain the whole document.
%
% Entities may be either parsed or unparsed. A parsed entity's contents
% are referred to as its replacement text; this text is considered an
% integral part of the document.
%
% An unparsed entity is a resource whose contents may or may not be
% text, and if text, may not be XML. Each unparsed entity has an
% associated notation, identified by name. Beyond a requirement that an
% XML processor make the identifiers for the entity and notation
% available to the application, XML places no constraints on the
% contents of unparsed entities.
%
% Parsed entities are invoked by name using entity references; unparsed
% entities by name, given in the value of ENTITY or ENTITIES attributes.
%
% General entities are entities for use within the document content. In
% this specification, general entities are sometimes referred to with
% the unqualified term entity when this leads to no ambiguity. Parameter
% entities are parsed entities for use within the DTD. These two types
% of entities use different forms of reference and are recognized in
% different contexts. Furthermore, they occupy different namespaces; a
% parameter entity and a general entity with the same name are two
% distinct entities.
%
% 4.1 Character and Entity References
%
% A character reference refers to a specific character in the ISO/IEC
% 10646 character set, for example one not directly accessible from
% available input devices.
%
% Character Reference
% [66] CharRef ::= '&#' [0-9]+ ';'
% | '&#x' [0-9a-fA-F]+ ';' [ WFC: Legal Character ]
:- pred charRef(pstate(_), pstate(unicode)).
:- mode charRef(in, out) is det.
charRef -->
(lit("&#x") and
plus(range('a', 'f') or range('A', 'F') or range('0', '9')) and
lit1(';')
then (pred((_, Hs, _)::in, pdi, puo) is det -->
{ hex_digits_to_number(Hs, 0, UniCode) },
return(UniCode)
)) or
(lit("&#") and plus(range('0', '9')) and lit1(';')
then (pred((_, Ds, _)::in, pdi, puo) is det -->
{ decimal_digits_to_number(Ds, 0, UniCode) },
return(UniCode)
)).
:- pred decimal_digits_to_number(list(unicode), int, int).
:- mode decimal_digits_to_number(in, in, out) is det.
decimal_digits_to_number([], U, U).
decimal_digits_to_number([D|Ds], U0, U) :-
U1 = 10 * U0 + (D - '0'),
decimal_digits_to_number(Ds, U1, U).
:- pred hex_digits_to_number(list(unicode), int, int).
:- mode hex_digits_to_number(in, in, out) is det.
hex_digits_to_number([], U, U).
hex_digits_to_number([D|Ds], U0, U) :-
( '0' =< D, D =< '9' ->
U1 = 16 * U0 + (D - '0')
; 'a' =< D, D =< 'f' ->
U1 = 16 * U0 + 10 + (D - 'a')
; 'A' =< D, D =< 'F' ->
U1 = 16 * U0 + 10 + (D - 'a')
;
error("hex_digits_to_number: internal error")
),
hex_digits_to_number(Ds, U1, U).
%
% Well-Formedness Constraint: Legal Character
% Characters referred to using character references must match the
% production for Char.
% If the character reference begins with "&#x", the digits and letters
% up to the terminating ; provide a hexadecimal representation of the
% character's code point in ISO/IEC 10646. If it begins just with "&#",
% the digits up to the terminating ; provide a decimal representation of
% the character's code point.
%
% An entity reference refers to the content of a named entity.
% References to parsed general entities use ampersand (&) and semicolon
% (;) as delimiters. Parameter-entity references use percent-sign (%)
% and semicolon (;) as delimiters.
%
% Entity Reference
% [67] Reference ::= EntityRef | CharRef
%:- pred reference(pstate(_), pstate(unit)).
%:- mode reference(in, out) is det.
%
%reference -->
% % Note: charRef has been migrated to charData, etc. because it
% % is handled differently to normal entity references.
% entityRef.
% [68] EntityRef ::= '&' Name ';' [ WFC: Entity Declared ]
% [ VC: Entity Declared ]
% [ WFC: Parsed Entity ]
% [ WFC: No Recursion ]
:- pred entityRef(parser(T1, T2), pstate(T1), pstate(T2)).
:- mode entityRef(in(parser), in, out) is det.
entityRef(Parser) -->
null then (pred(X::in, pdi, puo) is det -->
entityRef then (pred(Entity::in, pdi, puo) is det -->
return(X),
parseEntity(Parser, mkEntity(Entity))
)).
:- pred entityRef(pstate(_), pstate(dtd.entity)).
:- mode entityRef(in, out) is det.
entityRef -->
lit1('&') then (pred(_::in, pdi, puo) is det -->
name then (pred(Name::in, pdi, puo) is det -->
lit1(';') then (pred(_::in, pdi, puo) is det -->
get(gEntities, entities(Entities)),
( { search(Entities, Name, EntityDef) } ->
(getEntity(EntityDef) then (pred(Entity::in, pdi, puo) is det -->
return(Entity)
))
;
{ format("reference to undefined entity `%s'", [s(Name)], Msg) },
error(Msg)
)))).
% [69] PEReference ::= '%' Name ';' [ VC: Entity Declared ]
% [ WFC: No Recursion ]
% [ WFC: In DTD ]
:- pred pEReference(parser(T1, T2), pstate(T1), pstate(T2)).
:- mode pEReference(in(parser), in, out) is det.
pEReference(Parser) -->
null then (pred(X::in, pdi, puo) is det -->
pEReference then (pred(Entity::in, pdi, puo) is det -->
return(X),
parseEntity(Parser, mkEntity(Entity))
)).
:- pred pEReference(pstate(_), pstate(dtd.entity)).
:- mode pEReference(in, out) is det.
pEReference -->
lit1('%') then (pred(_::in, pdi, puo) is det -->
name then (pred(Name::in, pdi, puo) is det -->
lit1(';') then (pred(_::in, pdi, puo) is det -->
get(gPEntities, pentities(PEntities)),
( { search(PEntities, Name, EntityDef) } ->
(getEntity(EntityDef) then (pred(Entity::in, pdi, puo) is det -->
return(Entity)
))
;
{ format("reference to undefined parameter entity `%s'",
[s(Name)], Msg) },
error(Msg)
)))).
%
% Well-Formedness Constraint: Entity Declared
% In a document without any DTD, a document with only an internal DTD
% subset which contains no parameter entity references, or a document
% with "standalone='yes'", the Name given in the entity reference must
% match that in an entity declaration, except that well-formed
% documents need not declare any of the following entities: amp, lt, gt,
% apos, quot. The declaration of a parameter entity must precede any
% reference to it. Similarly, the declaration of a general entity must
% precede any reference to it which appears in a default value in an
% attribute-list declaration. Note that if entities are declared in the
% external subset or in external parameter entities, a non-validating
% processor is not obligated to read and process their declarations; for
% such documents, the rule that an entity must be declared is a
% well-formedness constraint only if standalone='yes'.
%
% Validity Constraint: Entity Declared
% In a document with an external subset or external parameter entities
% with "standalone='no'", the Name given in the entity reference must
% match that in an entity declaration. For interoperability, valid
% documents should declare the entities amp, lt, gt, apos, quot, in the
% form specified in "4.6 Predefined Entities". The declaration of a
% parameter entity must precede any reference to it. Similarly, the
% declaration of a general entity must precede any reference to it which
% appears in a default value in an attribute-list declaration.
%
% Well-Formedness Constraint: Parsed Entity
% An entity reference must not contain the name of an unparsed entity.
% Unparsed entities may be referred to only in attribute values declared
% to be of type ENTITY or ENTITIES.
%
% Well-Formedness Constraint: No Recursion
% A parsed entity must not contain a recursive reference to itself,
% either directly or indirectly.
%
% Well-Formedness Constraint: In DTD
% Parameter-entity references may only appear in the DTD.
%
% Examples of character and entity references:
%
% Type <key>less-than</key> (&#x3C;) to save options.
% This document was prepared on &docdate; and
% is classified &security-level;.
%
% Example of a parameter-entity reference:
%
% <!-- declare the parameter entity "ISOLat2"... -->
% <!ENTITY % ISOLat2
% SYSTEM "http://www.xml.com/iso/isolat2-xml.entities" >
% <!-- ... now reference it. -->
% %ISOLat2;
%
% 4.2 Entity Declarations
%
% Entities are declared thus:
%
% Entity Declaration
% [70] EntityDecl ::= GEDecl | PEDecl
:- pred entityDecl(pstate(_), pstate(unit)).
:- mode entityDecl(in, out) is det.
entityDecl -->
geDecl or peDecl.
% [71] GEDecl ::= '<!ENTITY' S Name S EntityDef S? '>'
:- pred geDecl(pstate(_), pstate(unit)).
:- mode geDecl(in, out) is det.
geDecl -->
lit("<!ENTITY") then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pe(name) then (pred(Name::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pe(entityDef) then (pred(Value::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('>') then (pred(_::in, pdi, puo) is det -->
get(gEntities, entities(Entities0)),
( { contains(Entities0, Name) } ->
{ format("Multiple declarations of entity `%s'", [s(Name)], Msg) },
warn(Msg),
{ Entities = Entities0 }
;
{ set(Entities0, Name, Value, Entities) }
),
set(gEntities, entities(Entities)),
return
))))))).
% [72] PEDecl ::= '<!ENTITY' S '%' S Name S PEDef S? '>'
:- pred peDecl(pstate(_), pstate(unit)).
:- mode peDecl(in, out) is det.
peDecl -->
lit("<!ENTITY") then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
lit1('%') then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pe(name) then (pred(Name::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pe(peDef) then (pred(Value::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('>') then (pred(_::in, pdi, puo) is det -->
get(gPEntities, pentities(PEntities0)),
( { contains(PEntities0, Name) } ->
{ format("Multiple declarations of parameter entity `%s'",
[s(Name)], Msg) },
warn(Msg),
{ PEntities = PEntities0 }
;
{ set(PEntities0, Name, Value, PEntities) }
),
set(gPEntities, pentities(PEntities)),
return
))))))))).
% [73] EntityDef ::= EntityValue | (ExternalID NDataDecl?)
:- pred entityDef(pstate(_), pstate(entityDef)).
:- mode entityDef(in, out) is det.
entityDef -->
% XXX x(entityValue) or (externalID and opt(nDataDecl)).
(entityValue then (pred(Entity::in, pdi, puo) is det -->
return(internal(Entity))
)) or (
externalID then (pred(ExtId::in, pdi, puo) is det -->
opt(nDataDecl) then (pred(_::in, pdi, puo) is det -->
return(external(ExtId))
))).
% [74] PEDef ::= EntityValue | ExternalID
:- pred peDef(pstate(_), pstate(entityDef)).
:- mode peDef(in, out) is det.
peDef -->
% XXX x(entityValue) or externalID.
entityDef.
%
% The Name identifies the entity in an entity reference or, in the case
% of an unparsed entity, in the value of an ENTITY or ENTITIES
% attribute. If the same entity is declared more than once, the first
% declaration encountered is binding; at user option, an XML processor
% may issue a warning if entities are declared multiple times.
%
% 4.2.1 Internal Entities
%
% If the entity definition is an EntityValue, the defined entity is
% called an internal entity. There is no separate physical storage
% object, and the content of the entity is given in the declaration.
% Note that some processing of entity and character references in the
% literal entity value may be required to produce the correct
% replacement text: see "4.5 Construction of Internal Entity
% Replacement Text".
%
% An internal entity is a parsed entity.
%
% Example of an internal entity declaration:
%
% <!ENTITY Pub-Status "This is a pre-release of the
% specification.">
%
% 4.2.2 External Entities
%
% If the entity is not internal, it is an external entity, declared as
% follows:
%
% External Entity Declaration
% [75] ExternalID ::= 'SYSTEM' S SystemLiteral
% | 'PUBLIC' S PubidLiteral S SystemLiteral
:- pred externalID(pstate(_), pstate(externalId)).
:- mode externalID(in, out) is det.
externalID -->
wrap(snd((lit("SYSTEM", unit) and s) and systemLiteral), wrapSystem) or
wrap(snd((lit("PUBLIC", unit) and s) and pubidLiteral) and
snd(s and systemLiteral), wrapPublic).
:- pred wrapSystem(string, externalId).
:- mode wrapSystem(in, out) is det.
wrapSystem(System, system(System)).
:- pred wrapPublic((string, string), externalId).
:- mode wrapPublic(in, out) is det.
wrapPublic((Public, System), public(Public, System)).
% [76] NDataDecl ::= S 'NDATA' S Name [ VC: Notation Declared ]
:- pred nDataDecl(pstate(_), pstate(name)).
:- mode nDataDecl(in, out) is det.
nDataDecl -->
s then (pred(_::in, pdi, puo) is det -->
lit("NDATA") then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
name then (pred(Name::in, pdi, puo) is det -->
return(Name)
)))).
%
% If the NDataDecl is present, this is a general unparsed entity;
% otherwise it is a parsed entity.
%
% Validity Constraint: Notation Declared
% The Name must match the declared name of a notation.
%
% The SystemLiteral is called the entity's system identifier. It is a
% URI, which may be used to retrieve the entity. Note that the hash mark
% (#) and fragment identifier frequently used with URIs are not,
% formally, part of the URI itself; an XML processor may signal an error
% if a fragment identifier is given as part of a system identifier.
% Unless otherwise provided by information outside the scope of this
% specification (e.g. a special XML element type defined by a particular
% DTD, or a processing instruction defined by a particular application
% specification), relative URIs are relative to the location of the
% resource within which the entity declaration occurs. A URI might thus
% be relative to the document entity, to the entity containing the
% external DTD subset, or to some other external parameter entity.
%
% An XML processor should handle a non-ASCII character in a URI by
% representing the character in UTF-8 as one or more bytes, and then
% escaping these bytes with the URI escaping mechanism (i.e., by
% converting each byte to %HH, where HH is the hexadecimal notation of
% the byte value).
%
% In addition to a system identifier, an external identifier may include
% a public identifier. An XML processor attempting to retrieve the
% entity's content may use the public identifier to try to generate an
% alternative URI. If the processor is unable to do so, it must use the
% URI specified in the system literal. Before a match is attempted, all
% strings of white space in the public identifier must be normalized to
% single space characters (#x20), and leading and trailing white space
% must be removed.
%
% Examples of external entity declarations:
%
% <!ENTITY open-hatch
% SYSTEM "http://www.textuality.com/boilerplate/OpenHatch.xml">
% <!ENTITY open-hatch
% PUBLIC "-//Textuality//TEXT Standard open-hatch boilerplate//
% EN"
% "http://www.textuality.com/boilerplate/OpenHatch.xml">
% <!ENTITY hatch-pic
% SYSTEM "../grafix/OpenHatch.gif"
% NDATA gif >
%
% 4.3 Parsed Entities
%
% 4.3.1 The Text Declaration
%
% External parsed entities may each begin with a text declaration.
%
% Text Declaration
% [77] TextDecl ::= '<?xml' VersionInfo? EncodingDecl S? '?>'
:- pred textDecl(pstate(_), pstate(unit)).
:- mode textDecl(in, out) is det.
textDecl -->
lit("<?xml") then (pred(_::in, pdi, puo) is det -->
opt(versionInfo) then (pred(_::in, pdi, puo) is det -->
encodingDecl then (pred(EncName::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit("?>") then (pred(_::in, pdi, puo) is det -->
get(gEncodings, encodings(Encodings)),
( { search(Encodings, EncName, Enc) } ->
setEncoding(Enc),
return
;
{ format("unknown encoding `%s' in external entity",
[s(EncName)], Msg) },
error(Msg)
)))))).
%
% The text declaration must be provided literally, not by reference to a
% parsed entity. No text declaration may appear at any position other
% than the beginning of an external parsed entity.
%
% 4.3.2 Well-Formed Parsed Entities
%
% The document entity is well-formed if it matches the production
% labeled document. An external general parsed entity is well-formed if
% it matches the production labeled extParsedEnt. An external parameter
% entity is well-formed if it matches the production labeled extPE.
%
% Well-Formed External Parsed Entity
% [78] extParsedEnt ::= TextDecl? content
:- pred extParsedEnt(pstate(_), pstate(list(ref(doc.content)))).
:- mode extParsedEnt(in, out) is det.
extParsedEnt -->
snd(opt(textDecl) and content).
% [79] extPE ::= TextDecl? extSubsetDecl
:- pred extPE(pstate(_), pstate(unit)).
:- mode extPE(in, out) is det.
extPE -->
opt(textDecl) then (pred(_::in, pdi, puo) is det -->
extSubsetDecl then (pred(_::in, pdi, puo) is det -->
return
)).
%
% An internal general parsed entity is well-formed if its replacement
% text matches the production labeled content. All internal parameter
% entities are well-formed by definition.
%
% A consequence of well-formedness in entities is that the logical and
% physical structures in an XML document are properly nested; no
% start-tag, end-tag, empty-element tag, element, comment, processing
% instruction, character reference, or entity reference can begin in one
% entity and end in another.
%
% 4.3.3 Character Encoding in Entities
%
% Each external parsed entity in an XML document may use a different
% encoding for its characters. All XML processors must be able to read
% entities in either UTF-8 or UTF-16.
%
% Entities encoded in UTF-16 must begin with the Byte Order Mark
% described by ISO/IEC 10646 Annex E and Unicode Appendix B (the ZERO
% WIDTH NO-BREAK SPACE character, #xFEFF). This is an encoding
% signature, not part of either the markup or the character data of the
% XML document. XML processors must be able to use this character to
% differentiate between UTF-8 and UTF-16 encoded documents.
%
% Although an XML processor is required to read only entities in the
% UTF-8 and UTF-16 encodings, it is recognized that other encodings are
% used around the world, and it may be desired for XML processors to
% read entities that use them. Parsed entities which are stored in an
% encoding other than UTF-8 or UTF-16 must begin with a text declaration
% containing an encoding declaration:
%
% Encoding Declaration
% [80] EncodingDecl ::= S 'encoding' Eq ('"' EncName '"' | "'" EncName
% "'" )
:- pred encodingDecl(pstate(_), pstate(string)).
:- mode encodingDecl(in, out) is det.
encodingDecl -->
s then (pred(_::in, pdi, puo) is det -->
lit("encoding") then (pred(_::in, pdi, puo) is det -->
eq then (pred(_::in, pdi, puo) is det -->
quote then (pred(Q::in, pdi, puo) is det -->
encName then (pred(Name::in, pdi, puo) is det -->
quote then (pred(EndQ::in, pdi, puo) is det -->
( { Q = EndQ } ->
return(Name)
;
error("mismatched quotes")
))))))).
% [81] EncName ::= [A-Za-z] ([A-Za-z0-9._] | '-')* /* Encoding name
% contains only Latin characters */
:- pred encName(pstate(_), pstate(string)).
:- mode encName(in, out) is det.
encName -->
(range('A', 'Z') or range('a', 'z'))
then (pred(C::in, pdi, puo) is det -->
star(range('A', 'Z') or range('a', 'z') or range('0', '9') or
lit1('.') or lit1('_') or lit1('-'))
then (pred(Cs::in, pdi, puo) is det -->
mkString([C|Cs], EncName),
return(EncName)
)).
%
% In the document entity, the encoding declaration is part of the XML
% declaration. The EncName is the name of the encoding used.
%
% In an encoding declaration, the values "UTF-8", "UTF-16",
% "ISO-10646-UCS-2", and "ISO-10646-UCS-4" should be used for the
% various encodings and transformations of Unicode / ISO/IEC 10646, the
% values "ISO-8859-1", "ISO-8859-2", ... "ISO-8859-9" should be used for
% the parts of ISO 8859, and the values "ISO-2022-JP", "Shift_JIS", and
% "EUC-JP" should be used for the various encoded forms of JIS
% X-0208-1997. XML processors may recognize other encodings; it is
% recommended that character encodings registered (as charsets) with the
% Internet Assigned Numbers Authority [IANA], other than those just
% listed, should be referred to using their registered names. Note that
% these registered names are defined to be case-insensitive, so
% processors wishing to match against them should do so in a
% case-insensitive way.
%
% In the absence of information provided by an external transport
% protocol (e.g. HTTP or MIME), it is an error for an entity including
% an encoding declaration to be presented to the XML processor in an
% encoding other than that named in the declaration, for an encoding
% declaration to occur other than at the beginning of an external
% entity, or for an entity which begins with neither a Byte Order Mark
% nor an encoding declaration to use an encoding other than UTF-8. Note
% that since ASCII is a subset of UTF-8, ordinary ASCII entities do not
% strictly need an encoding declaration.
%
% It is a fatal error when an XML processor encounters an entity with an
% encoding that it is unable to process.
%
% Examples of encoding declarations:
%
% <?xml encoding='UTF-8'?>
% <?xml encoding='EUC-JP'?>
%
% 4.4 XML Processor Treatment of Entities and References
%
% The table below summarizes the contexts in which character references,
% entity references, and invocations of unparsed entities might appear
% and the required behavior of an XML processor in each case. The labels
% in the leftmost column describe the recognition context:
%
% Reference in Content
% as a reference anywhere after the start-tag and before the
% end-tag of an element; corresponds to the nonterminal content.
%
% Reference in Attribute Value
% as a reference within either the value of an attribute in a
% start-tag, or a default value in an attribute declaration;
% corresponds to the nonterminal AttValue.
%
% Occurs as Attribute Value
% as a Name, not a reference, appearing either as the value of an
% attribute which has been declared as type ENTITY, or as one of
% the space-separated tokens in the value of an attribute which
% has been declared as type ENTITIES.
%
% Reference in Entity Value
% as a reference within a parameter or internal entity's literal
% entity value in the entity's declaration; corresponds to the
% nonterminal EntityValue.
%
% Reference in DTD
% as a reference within either the internal or external subsets
% of the DTD, but outside of an EntityValue or AttValue.
%
% Entity Type Character
% Parameter Internal
% General External Parsed
% General Unparsed
% Reference
% in Content Not recognized Included Included if validating Forbidden
% Included
% Reference
% in Attribute Value Not recognized Included in literal Forbidden
% Forbidden Included
% Occurs as
% Attribute Value Not recognized Forbidden Forbidden Notify
% Not recognized
% Reference
% in EntityValue Included in literal Bypassed Bypassed Forbidden
% Included
% Reference
% in DTD Included as PE Forbidden Forbidden Forbidden Forbidden
%
% 4.4.1 Not Recognized
%
% Outside the DTD, the % character has no special significance; thus,
% what would be parameter entity references in the DTD are not
% recognized as markup in content. Similarly, the names of unparsed
% entities are not recognized except when they appear in the value of an
% appropriately declared attribute.
%
% 4.4.2 Included
%
% An entity is included when its replacement text is retrieved and
% processed, in place of the reference itself, as though it were part of
% the document at the location the reference was recognized. The
% replacement text may contain both character data and (except for
% parameter entities) markup, which must be recognized in the usual way,
% except that the replacement text of entities used to escape markup
% delimiters (the entities amp, lt, gt, apos, quot) is always treated as
% data. (The string "AT&amp;T;" expands to "AT&T;" and the remaining
% ampersand is not recognized as an entity-reference delimiter.) A
% character reference is included when the indicated character is
% processed in place of the reference itself.
%
% 4.4.3 Included If Validating
%
% When an XML processor recognizes a reference to a parsed entity, in
% order to validate the document, the processor must include its
% replacement text. If the entity is external, and the processor is not
% attempting to validate the XML document, the processor may, but need
% not, include the entity's replacement text. If a non-validating parser
% does not include the replacement text, it must inform the application
% that it recognized, but did not read, the entity.
%
% This rule is based on the recognition that the automatic inclusion
% provided by the SGML and XML entity mechanism, primarily designed to
% support modularity in authoring, is not necessarily appropriate for
% other applications, in particular document browsing. Browsers, for
% example, when encountering an external parsed entity reference, might
% choose to provide a visual indication of the entity's presence and
% retrieve it for display only on demand.
%
% 4.4.4 Forbidden
%
% The following are forbidden, and constitute fatal errors:
% * the appearance of a reference to an unparsed entity.
% * the appearance of any character or general-entity reference in the
% DTD except within an EntityValue or AttValue.
% * a reference to an external entity in an attribute value.
%
% 4.4.5 Included in Literal
%
% When an entity reference appears in an attribute value, or a parameter
% entity reference appears in a literal entity value, its replacement
% text is processed in place of the reference itself as though it were
% part of the document at the location the reference was recognized,
% except that a single or double quote character in the replacement text
% is always treated as a normal data character and will not terminate
% the literal. For example, this is well-formed:
%
% <!ENTITY % YN '"Yes"' >
% <!ENTITY WhatHeSaid "He said &YN;" >
%
% while this is not:
%
% <!ENTITY EndAttr "27'" >
% <element attribute='a-&EndAttr;>
%
% 4.4.6 Notify
%
% When the name of an unparsed entity appears as a token in the value of
% an attribute of declared type ENTITY or ENTITIES, a validating
% processor must inform the application of the system and public (if
% any) identifiers for both the entity and its associated notation.
%
% 4.4.7 Bypassed
%
% When a general entity reference appears in the EntityValue in an
% entity declaration, it is bypassed and left as is.
%
% 4.4.8 Included as PE
%
% Just as with external parsed entities, parameter entities need only be
% included if validating. When a parameter-entity reference is
% recognized in the DTD and included, its replacement text is enlarged
% by the attachment of one leading and one following space (#x20)
% character; the intent is to constrain the replacement text of
% parameter entities to contain an integral number of grammatical tokens
% in the DTD.
%
% 4.5 Construction of Internal Entity Replacement Text
%
% In discussing the treatment of internal entities, it is useful to
% distinguish two forms of the entity's value. The literal entity value
% is the quoted string actually present in the entity declaration,
% corresponding to the non-terminal EntityValue. The replacement text is
% the content of the entity, after replacement of character references
% and parameter-entity references.
%
% The literal entity value as given in an internal entity declaration
% (EntityValue) may contain character, parameter-entity, and
% general-entity references. Such references must be contained entirely
% within the literal entity value. The actual replacement text that is
% included as described above must contain the replacement text of any
% parameter entities referred to, and must contain the character
% referred to, in place of any character references in the literal
% entity value; however, general-entity references must be left as-is,
% unexpanded. For example, given the following declarations:
%
% <!ENTITY % pub "&#xc9;ditions Gallimard" >
% <!ENTITY rights "All rights reserved" >
% <!ENTITY book "La Peste: Albert Camus,
% &#xA9; 1947 %pub;. &rights;" >
%
% then the replacement text for the entity "book" is:
%
% La Peste: Albert Camus,
% <EFBFBD> 1947 <EFBFBD>ditions Gallimard. &rights;
%
% The general-entity reference "&rights;" would be expanded should the
% reference "&book;" appear in the document's content or an attribute
% value.
%
% These simple rules may have complex interactions; for a detailed
% discussion of a difficult example, see "D. Expansion of Entity and
% Character References".
%
% 4.6 Predefined Entities
%
% Entity and character references can both be used to escape the left
% angle bracket, ampersand, and other delimiters. A set of general
% entities (amp, lt, gt, apos, quot) is specified for this purpose.
% Numeric character references may also be used; they are expanded
% immediately when recognized and must be treated as character data, so
% the numeric character references "&#60;" and "&#38;" may be used to
% escape < and & when they occur in character data.
%
% All XML processors must recognize these entities whether they are
% declared or not. For interoperability, valid XML documents should
% declare these entities, like any others, before using them. If the
% entities in question are declared, they must be declared as internal
% entities whose replacement text is the single character being escaped
% or a character reference to that character, as shown below.
%
% <!ENTITY lt "&#38;#60;">
% <!ENTITY gt "&#62;">
% <!ENTITY amp "&#38;#38;">
% <!ENTITY apos "&#39;">
% <!ENTITY quot "&#34;">
%
% Note that the < and & characters in the declarations of "lt" and "amp"
% are doubly escaped to meet the requirement that entity replacement be
% well-formed.
%
% 4.7 Notation Declarations
%
% Notations identify by name the format of unparsed entities, the format
% of elements which bear a notation attribute, or the application to
% which a processing instruction is addressed.
%
% Notation declarations provide a name for the notation, for use in
% entity and attribute-list declarations and in attribute
% specifications, and an external identifier for the notation which may
% allow an XML processor or its client application to locate a helper
% application capable of processing data in the given notation.
%
% Notation Declarations
% [82] NotationDecl ::= '<!NOTATION' S Name S (ExternalID | PublicID)
% S? '>'
:- pred notationDecl(pstate(_), pstate(unit)).
:- mode notationDecl(in, out) is det.
notationDecl -->
lit("<!NOTATION") then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
name then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
(externalID or publicID) then (pred(_::in, pdi, puo) is det -->
opt(s) then (pred(_::in, pdi, puo) is det -->
lit1('>') then (pred(_::in, pdi, puo) is det -->
return
))))))).
% [83] PublicID ::= 'PUBLIC' S PubidLiteral
:- pred publicID(pstate(_), pstate(externalId)).
:- mode publicID(in, out) is det.
publicID -->
lit("PUBLIC") then (pred(_::in, pdi, puo) is det -->
s then (pred(_::in, pdi, puo) is det -->
pubidLiteral then (pred(Lit::in, pdi, puo) is det -->
return(public(Lit, ""))
))).
%
% XML processors must provide applications with the name and external
% identifier(s) of any notation declared and referred to in an attribute
% value, attribute definition, or entity declaration. They may
% additionally resolve the external identifier into the system
% identifier, file name, or other information needed to allow the
% application to call a processor for data in the notation described.
% (It is not an error, however, for XML documents to declare and refer
% to notations for which notation-specific applications are not
% available on the system where the XML processor or application is
% running.)
%
% 4.8 Document Entity
%
% The document entity serves as the root of the entity tree and a
% starting-point for an XML processor. This specification does not
% specify how the document entity is to be located by an XML processor;
% unlike other entities, the document entity has no name and might well
% appear on a processor input stream without any identification at all.
%
%5. Conformance
%
% 5.1 Validating and Non-Validating Processors
%
% Conforming XML processors fall into two classes: validating and
% non-validating.
%
% Validating and non-validating processors alike must report violations
% of this specification's well-formedness constraints in the content of
% the document entity and any other parsed entities that they read.
%
% Validating processors must report violations of the constraints
% expressed by the declarations in the DTD, and failures to fulfill the
% validity constraints given in this specification. To accomplish this,
% validating XML processors must read and process the entire DTD and all
% external parsed entities referenced in the document.
%
% Non-validating processors are required to check only the document
% entity, including the entire internal DTD subset, for well-formedness.
% While they are not required to check the document for validity, they
% are required to process all the declarations they read in the internal
% DTD subset and in any parameter entity that they read, up to the first
% reference to a parameter entity that they do not read; that is to say,
% they must use the information in those declarations to normalize
% attribute values, include the replacement text of internal entities,
% and supply default attribute values. They must not process entity
% declarations or attribute-list declarations encountered after a
% reference to a parameter entity that is not read, since the entity may
% have contained overriding declarations.
%
% 5.2 Using XML Processors
%
% The behavior of a validating XML processor is highly predictable; it
% must read every piece of a document and report all well-formedness and
% validity violations. Less is required of a non-validating processor;
% it need not read any part of the document other than the document
% entity. This has two effects that may be important to users of XML
% processors:
% * Certain well-formedness errors, specifically those that require
% reading external entities, may not be detected by a non-validating
% processor. Examples include the constraints entitled Entity
% Declared, Parsed Entity, and No Recursion, as well as some of the
% cases described as forbidden in "4.4 XML Processor Treatment of
% Entities and References".
% * The information passed from the processor to the application may
% vary, depending on whether the processor reads parameter and
% external entities. For example, a non-validating processor may not
% normalize attribute values, include the replacement text of
% internal entities, or supply default attribute values, where doing
% so depends on having read declarations in external or parameter
% entities.
%
% For maximum reliability in interoperating between different XML
% processors, applications which use non-validating processors should
% not rely on any behaviors not required of such processors.
% Applications which require facilities such as the use of default
% attributes or internal entities which are declared in external
% entities should use validating XML processors.
%
%6. Notation
%
% The formal grammar of XML is given in this specification using a
% simple Extended Backus-Naur Form (EBNF) notation. Each rule in the
% grammar defines one symbol, in the form
%
% symbol ::= expression
%
% Symbols are written with an initial capital letter if they are defined
% by a regular expression, or with an initial lower case letter
% otherwise. Literal strings are quoted.
%
% Within the expression on the right-hand side of a rule, the following
% expressions are used to match strings of one or more characters:
%
% #xN
% where N is a hexadecimal integer, the expression matches the
% character in ISO/IEC 10646 whose canonical (UCS-4) code value,
% when interpreted as an unsigned binary number, has the value
% indicated. The number of leading zeros in the #xN form is
% insignificant; the number of leading zeros in the corresponding
% code value is governed by the character encoding in use and is
% not significant for XML.
%
% [a-zA-Z], [#xN-#xN]
% matches any character with a value in the range(s) indicated
% (inclusive).
%
% [^a-z], [^#xN-#xN]
% matches any character with a value outside the range indicated.
%
% [^abc], [^#xN#xN#xN]
% matches any character with a value not among the characters
% given.
%
% "string"
% matches a literal string matching that given inside the double
% quotes.
%
% 'string'
% matches a literal string matching that given inside the single
% quotes.
%
% These symbols may be combined to match more complex patterns as
% follows, where A and B represent simple expressions:
%
% (expression)
% expression is treated as a unit and may be combined as
% described in this list.
%
% A?
% matches A or nothing; optional A.
%
% A B
% matches A followed by B.
%
% A | B
% matches A or B but not both.
%
% A - B
% matches any string that matches A but does not match B.
%
% A+
% matches one or more occurrences of A.
%
% A*
% matches zero or more occurrences of A.
%
% Other notations used in the productions are:
%
% /* ... */
% comment.
%
% [ wfc: ... ]
% well-formedness constraint; this identifies by name a
% constraint on well-formed documents associated with a
% production.
%
% [ vc: ... ]
% validity constraint; this identifies by name a constraint on
% valid documents associated with a production.
% _________________________________________________________________
%
% Appendices
%
%A. References
%
% A.1 Normative References
%
% IANA
% (Internet Assigned Numbers Authority) Official Names for
% Character Sets, ed. Keld Simonsen et al. See
% ftp://ftp.isi.edu/in-notes/iana/assignments/character-sets.
%
% IETF RFC 1766
% IETF (Internet Engineering Task Force). RFC 1766: Tags for the
% Identification of Languages, ed. H. Alvestrand. 1995.
%
% ISO 639
% (International Organization for Standardization). ISO 639:1988
% (E). Code for the representation of names of languages.
% [Geneva]: International Organization for Standardization, 1988.
%
% ISO 3166
% (International Organization for Standardization). ISO
% 3166-1:1997 (E). Codes for the representation of names of
% countries and their subdivisions -- Part 1: Country codes
% [Geneva]: International Organization for Standardization, 1997.
%
% ISO/IEC 10646
% ISO (International Organization for Standardization). ISO/IEC
% 10646-1993 (E). Information technology -- Universal
% Multiple-Octet Coded Character Set (UCS) -- Part 1:
% Architecture and Basic Multilingual Plane. [Geneva]:
% International Organization for Standardization, 1993 (plus
% amendments AM 1 through AM 7).
%
% Unicode
% The Unicode Consortium. The Unicode Standard, Version 2.0.
% Reading, Mass.: Addison-Wesley Developers Press, 1996.
%
% A.2 Other References
%
% Aho/Ullman
% Aho, Alfred V., Ravi Sethi, and Jeffrey D. Ullman. Compilers:
% Principles, Techniques, and Tools. Reading: Addison-Wesley,
% 1986, rpt. corr. 1988.
%
% Berners-Lee et al.
% Berners-Lee, T., R. Fielding, and L. Masinter. Uniform Resource
% Identifiers (URI): Generic Syntax and Semantics. 1997. (Work in
% progress; see updates to RFC1738.)
%
% Br<EFBFBD>ggemann-Klein
% Br<EFBFBD>ggemann-Klein, Anne. Regular Expressions into Finite
% Automata. Extended abstract in I. Simon, Hrsg., LATIN 1992, S.
% 97-98. Springer-Verlag, Berlin 1992. Full Version in
% Theoretical Computer Science 120: 197-213, 1993.
%
% Br<EFBFBD>ggemann-Klein and Wood
% Br<EFBFBD>ggemann-Klein, Anne, and Derick Wood. Deterministic Regular
% Languages. Universit<EFBFBD>t Freiburg, Institut f<EFBFBD>r Informatik,
% Bericht 38, Oktober 1991.
%
% Clark
% James Clark. Comparison of SGML and XML. See
% http://www.w3.org/TR/NOTE-sgml-xml-971215.
%
% IETF RFC1738
% IETF (Internet Engineering Task Force). RFC 1738: Uniform
% Resource Locators (URL), ed. T. Berners-Lee, L. Masinter, M.
% McCahill. 1994.
%
% IETF RFC1808
% IETF (Internet Engineering Task Force). RFC 1808: Relative
% Uniform Resource Locators, ed. R. Fielding. 1995.
%
% IETF RFC2141
% IETF (Internet Engineering Task Force). RFC 2141: URN Syntax,
% ed. R. Moats. 1997.
%
% ISO 8879
% ISO (International Organization for Standardization). ISO
% 8879:1986(E). Information processing -- Text and Office Systems
% -- Standard Generalized Markup Language (SGML). First edition
% -- 1986-10-15. [Geneva]: International Organization for
% Standardization, 1986.
%
% ISO/IEC 10744
% ISO (International Organization for Standardization). ISO/IEC
% 10744-1992 (E). Information technology -- Hypermedia/Time-based
% Structuring Language (HyTime). [Geneva]: International
% Organization for Standardization, 1992. Extended Facilities
% Annexe. [Geneva]: International Organization for
% Standardization, 1996.
%
%B. Character Classes
%
% The character classes are in a separate module
% for compile-time performance!
:- include_module xml:parse:chars.
:- import_module xml:parse:chars.
%C. XML and SGML (Non-Normative)
%
% XML is designed to be a subset of SGML, in that every valid XML
% document should also be a conformant SGML document. For a detailed
% comparison of the additional restrictions that XML places on documents
% beyond those of SGML, see [Clark].
%
%D. Expansion of Entity and Character References (Non-Normative)
%
% This appendix contains some examples illustrating the sequence of
% entity- and character-reference recognition and expansion, as
% specified in "4.4 XML Processor Treatment of Entities and References".
%
% If the DTD contains the declaration
%
% <!ENTITY example "<p>An ampersand (&#38;#38;) may be escaped
% numerically (&#38;#38;#38;) or with a general entity
% (&amp;amp;).</p>" >
%
% then the XML processor will recognize the character references when it
% parses the entity declaration, and resolve them before storing the
% following string as the value of the entity "example":
%
% <p>An ampersand (&#38;) may be escaped
% numerically (&#38;#38;) or with a general entity
% (&amp;amp;).</p>
%
% A reference in the document to "&example;" will cause the text to be
% reparsed, at which time the start- and end-tags of the "p" element
% will be recognized and the three references will be recognized and
% expanded, resulting in a "p" element with the following content (all
% data, no delimiters or markup):
%
% An ampersand (&) may be escaped
% numerically (&#38;) or with a general entity
% (&amp;).
%
% A more complex example will illustrate the rules and their effects
% fully. In the following example, the line numbers are solely for
% reference.
%
% 1 <?xml version='1.0'?>
% 2 <!DOCTYPE test [
% 3 <!ELEMENT test (#PCDATA) >
% 4 <!ENTITY % xx '&#37;zz;'>
% 5 <!ENTITY % zz '&#60;!ENTITY tricky "error-prone" >' >
% 6 %xx;
% 7 ]>
% 8 <test>This sample shows a &tricky; method.</test>
%
% This produces the following:
% * in line 4, the reference to character 37 is expanded immediately,
% and the parameter entity "xx" is stored in the symbol table with
% the value "%zz;". Since the replacement text is not rescanned, the
% reference to parameter entity "zz" is not recognized. (And it
% would be an error if it were, since "zz" is not yet declared.)
% * in line 5, the character reference "&#60;" is expanded immediately
% and the parameter entity "zz" is stored with the replacement text
% "<!ENTITY tricky "error-prone" >", which is a well-formed entity
% declaration.
% * in line 6, the reference to "xx" is recognized, and the
% replacement text of "xx" (namely "%zz;") is parsed. The reference
% to "zz" is recognized in its turn, and its replacement text
% ("<!ENTITY tricky "error-prone" >") is parsed. The general entity
% "tricky" has now been declared, with the replacement text
% "error-prone".
% * in line 8, the reference to the general entity "tricky" is
% recognized, and it is expanded, so the full content of the "test"
% element is the self-describing (and ungrammatical) string This
% sample shows a error-prone method.
%
%E. Deterministic Content Models (Non-Normative)
%
% For compatibility, it is required that content models in element type
% declarations be deterministic.
%
% SGML requires deterministic content models (it calls them
% "unambiguous"); XML processors built using SGML systems may flag
% non-deterministic content models as errors.
%
% For example, the content model ((b, c) | (b, d)) is non-deterministic,
% because given an initial b the parser cannot know which b in the model
% is being matched without looking ahead to see which element follows
% the b. In this case, the two references to b can be collapsed into a
% single reference, making the model read (b, (c | d)). An initial b now
% clearly matches only a single name in the content model. The parser
% doesn't need to look ahead to see what follows; either c or d would be
% accepted.
%
% More formally: a finite state automaton may be constructed from the
% content model using the standard algorithms, e.g. algorithm 3.5 in
% section 3.9 of Aho, Sethi, and Ullman [Aho/Ullman]. In many such
% algorithms, a follow set is constructed for each position in the
% regular expression (i.e., each leaf node in the syntax tree for the
% regular expression); if any position has a follow set in which more
% than one following position is labeled with the same element type
% name, then the content model is in error and may be reported as an
% error.
%
% Algorithms exist which allow many but not all non-deterministic
% content models to be reduced automatically to equivalent deterministic
% models; see Br<EFBFBD>ggemann-Klein 1991 [Br<EFBFBD>ggemann-Klein].
%
%F. Autodetection of Character Encodings (Non-Normative)
%
% The XML encoding declaration functions as an internal label on each
% entity, indicating which character encoding is in use. Before an XML
% processor can read the internal label, however, it apparently has to
% know what character encoding is in use--which is what the internal
% label is trying to indicate. In the general case, this is a hopeless
% situation. It is not entirely hopeless in XML, however, because XML
% limits the general case in two ways: each implementation is assumed to
% support only a finite set of character encodings, and the XML encoding
% declaration is restricted in position and content in order to make it
% feasible to autodetect the character encoding in use in each entity in
% normal cases. Also, in many cases other sources of information are
% available in addition to the XML data stream itself. Two cases may be
% distinguished, depending on whether the XML entity is presented to the
% processor without, or with, any accompanying (external) information.
% We consider the first case first.
%
% Because each XML entity not in UTF-8 or UTF-16 format must begin with
% an XML encoding declaration, in which the first characters must be
% '<?xml', any conforming processor can detect, after two to four octets
% of input, which of the following cases apply. In reading this list, it
% may help to know that in UCS-4, '<' is "#x0000003C" and '?' is
% "#x0000003F", and the Byte Order Mark required of UTF-16 data streams
% is "#xFEFF".
%
% * 00 00 00 3C: UCS-4, big-endian machine (1234 order)
% * 3C 00 00 00: UCS-4, little-endian machine (4321 order)
% * 00 00 3C 00: UCS-4, unusual octet order (2143)
% * 00 3C 00 00: UCS-4, unusual octet order (3412)
% * FE FF: UTF-16, big-endian
% * FF FE: UTF-16, little-endian
% * 00 3C 00 3F: UTF-16, big-endian, no Byte Order Mark (and thus,
% strictly speaking, in error)
% * 3C 00 3F 00: UTF-16, little-endian, no Byte Order Mark (and thus,
% strictly speaking, in error)
% * 3C 3F 78 6D: UTF-8, ISO 646, ASCII, some part of ISO 8859,
% Shift-JIS, EUC, or any other 7-bit, 8-bit, or mixed-width encoding
% which ensures that the characters of ASCII have their normal
% positions, width, and values; the actual encoding declaration must
% be read to detect which of these applies, but since all of these
% encodings use the same bit patterns for the ASCII characters, the
% encoding declaration itself may be read reliably
% * 4C 6F A7 94: EBCDIC (in some flavor; the full encoding declaration
% must be read to tell which code page is in use)
% * other: UTF-8 without an encoding declaration, or else the data
% stream is corrupt, fragmentary, or enclosed in a wrapper of some
% kind
%
% This level of autodetection is enough to read the XML encoding
% declaration and parse the character-encoding identifier, which is
% still necessary to distinguish the individual members of each family
% of encodings (e.g. to tell UTF-8 from 8859, and the parts of 8859 from
% each other, or to distinguish the specific EBCDIC code page in use,
% and so on).
%
% Because the contents of the encoding declaration are restricted to
% ASCII characters, a processor can reliably read the entire encoding
% declaration as soon as it has detected which family of encodings is in
% use. Since in practice, all widely used character encodings fall into
% one of the categories above, the XML encoding declaration allows
% reasonably reliable in-band labeling of character encodings, even when
% external sources of information at the operating-system or
% transport-protocol level are unreliable.
%
% Once the processor has detected the character encoding in use, it can
% act appropriately, whether by invoking a separate input routine for
% each case, or by calling the proper conversion function on each
% character of input.
%
% Like any self-labeling system, the XML encoding declaration will not
% work if any software changes the entity's character set or encoding
% without updating the encoding declaration. Implementors of
% character-encoding routines should be careful to ensure the accuracy
% of the internal and external information used to label the entity.
%
% The second possible case occurs when the XML entity is accompanied by
% encoding information, as in some file systems and some network
% protocols. When multiple sources of information are available, their
% relative priority and the preferred method of handling conflict should
% be specified as part of the higher-level protocol used to deliver XML.
% Rules for the relative priority of the internal label and the
% MIME-type label in an external header, for example, should be part of
% the RFC document defining the text/xml and application/xml MIME types.
% In the interests of interoperability, however, the following rules are
% recommended.
% * If an XML entity is in a file, the Byte-Order Mark and
% encoding-declaration PI are used (if present) to determine the
% character encoding. All other heuristics and sources of
% information are solely for error recovery.
% * If an XML entity is delivered with a MIME type of text/xml, then
% the charset parameter on the MIME type determines the character
% encoding method; all other heuristics and sources of information
% are solely for error recovery.
% * If an XML entity is delivered with a MIME type of application/xml,
% then the Byte-Order Mark and encoding-declaration PI are used (if
% present) to determine the character encoding. All other heuristics
% and sources of information are solely for error recovery.
%
% These rules apply only in the absence of protocol-level documentation;
% in particular, when the MIME types text/xml and application/xml are
% defined, the recommendations of the relevant RFC will supersede these
% rules.
%
%G. W3C XML Working Group (Non-Normative)
%
% This specification was prepared and approved for publication by the
% W3C XML Working Group (WG). WG approval of this specification does not
% necessarily imply that all WG members voted for its approval. The
% current and former members of the XML WG are:
% Jon Bosak, Sun (Chair); James Clark (Technical Lead); Tim Bray,
% Textuality and Netscape (XML Co-editor); Jean Paoli, Microsoft (XML
% Co-editor); C. M. Sperberg-McQueen, U. of Ill. (XML Co-editor); Dan
% Connolly, W3C (W3C Liaison); Paula Angerstein, Texcel; Steve DeRose,
% INSO; Dave Hollander, HP; Eliot Kimber, ISOGEN; Eve Maler, ArborText;
% Tom Magliery, NCSA; Murray Maloney, Muzmo and Grif; Makoto Murata,
% Fuji Xerox Information Systems; Joel Nava, Adobe; Conleth O'Connell,
% Vignette; Peter Sharpe, SoftQuad; John Tigue, DataChannel
%
% Copyright <EFBFBD> 1998 W3C (MIT, INRIA, Keio ), All Rights Reserved. W3C
% liability, trademark, document use and software licensing rules
% apply.
:- pred add(doc.content, ref(doc.content), pstate(T), pstate(T)).
:- mode add(in, out, pdi, puo) is det.
add(Cont, Ref) -->
get(gContent, Content0),
{ doc.add(Cont, Ref, Content0, Content) },
set(gContent, Content).
%:- pred psp(pstate(T1), pstate(unit)).
%:- mode psp(pdi, puo) is det.
%
%psp -->
% plus(ps) then (pred(_::in, pdi, puo) is det -->
% return
% ).
%:- pred ps(pstate(T1), pstate(unit)).
%:- mode ps(pdi, puo) is det.
%
%ps -->
% x(s) or x(pEReference) or x(comment).
:- pred pe(parser(T1, T2), pstate(T1), pstate(T2)).
:- mode pe(in(parser), pdi, puo) is det.
pe(P) -->
pEReference(pe(P)) or P.
:- pred null(pstate(T1), pstate(T1)).
:- mode null(pdi, puo) is det.
null --> [].
:- pred getEntity(entityDef, pstate(T1), pstate(dtd.entity)).
:- mode getEntity(in, pdi, puo) is det.
getEntity(internal(Entity)) -->
return(Entity).
getEntity(external(system(SystemId))) -->
get(gDirs, dirs(Dirs)),
io(find(SystemId, Dirs), Res0),
(
{ Res0 = ok(_Path) },
io((pred(Res10::out, di, uo) is det -->
read_file_as_string(Res10)
), Res1),
(
{ Res1 = ok(Entity) },
io(seen),
return(Entity)
;
{ Res1 = error(_, Err) },
{ io__error_message(Err, Msg) },
error(Msg)
)
;
{ Res0 = error(Msg) },
error(Msg)
).
getEntity(external(public(PublicId, SystemId))) -->
( { SystemId \= "" } ->
getEntity(external(system(SystemId)))
;
get(gCatalog, catalog(Catalog)),
( { search(Catalog, PublicId, FoundSystemId) } ->
getEntity(external(system(FoundSystemId)))
;
error("Entity not found")
)
).
:- pred warn(string, pstate(T1), pstate(T1)).
:- mode warn(in, pdi, puo) is det.
warn(Msg) -->
io((pred(di, uo) is det -->
io__stderr_stream(StdErr),
write_string(StdErr, "warning: "),
write_string(StdErr, Msg),
write_string(StdErr, "\n")
)).