* xmlread.pp: separated DOM-specific code into DoStartElement(); introduced FNameTable.

git-svn-id: trunk@16270 -
This commit is contained in:
sergei 2010-10-31 02:23:42 +00:00
parent 4e39959ca1
commit b066395a2b

View File

@ -280,6 +280,7 @@ type
TXMLReader = class
private
FSource: TXMLCharSource;
FNameTable: THashTable;
FCtrl: TDOMParser;
FXML11: Boolean;
FState: TXMLReadState;
@ -394,6 +395,7 @@ type
procedure ParseMarkupDecl; // [29]
procedure ParseStartTag; // [39]
procedure ParseEndTag; // [42]
procedure DoStartElement;
procedure DoEndElement;
procedure ParseAttribute(ElDef: TElementDecl);
procedure ParseContent; // [43]
@ -1313,15 +1315,16 @@ begin
FStdPrefix_xml := FNSHelper.GetPrefix(@PrefixDefault, 3);
FStdPrefix_xmlns := FNSHelper.GetPrefix(@PrefixDefault, 5);
FStdUri_xmlns := doc.Names.FindOrAdd(PWideChar(stduri_xmlns), Length(stduri_xmlns));
FStdUri_xml := doc.Names.FindOrAdd(PWideChar(stduri_xml), Length(stduri_xml));
FStdUri_xmlns := FNameTable.FindOrAdd(PWideChar(stduri_xmlns), Length(stduri_xmlns));
FStdUri_xml := FNameTable.FindOrAdd(PWideChar(stduri_xml), Length(stduri_xml));
end;
end;
procedure TXMLReader.ProcessXML(ASource: TXMLCharSource);
begin
doc := TXMLDocument.Create;
doc.documentURI := ASource.SystemID; // TODO: to be changed to URI or BaseURI
doc.documentURI := ASource.SystemID; // TODO: to be changed to URI or BaseURI
FNameTable := doc.Names;
FState := rsProlog;
FNesting := 0;
FCurrNode := @FNodeStack[0];
@ -1343,6 +1346,7 @@ end;
procedure TXMLReader.ProcessFragment(ASource: TXMLCharSource; AOwner: TDOMNode);
begin
doc := AOwner.OwnerDocument;
FNameTable := doc.Names;
FState := rsRoot;
FNesting := 0;
FCurrNode := @FNodeStack[0];
@ -1926,7 +1930,7 @@ begin
if not SkipUntilSeq(GT_Delim, '?') then
FatalError('Unterminated processing instruction', -1);
SetNodeInfoWithValue(ntProcessingInstruction,
doc.Names.FindOrAdd(FName.Buffer, FName.Length));
FNameTable.FindOrAdd(FName.Buffer, FName.Length));
end;
procedure TXMLReader.CreatePINode;
@ -2165,7 +2169,7 @@ var
p: PHashItem;
begin
CheckName;
p := doc.Names.FindOrAdd(FName.Buffer, FName.Length);
p := FNameTable.FindOrAdd(FName.Buffer, FName.Length);
Result := TElementDecl(p^.Data);
if Result = nil then
begin
@ -2343,7 +2347,7 @@ begin
begin
CheckName;
ExpectWhitespace;
attrName := doc.Names.FindOrAdd(FName.Buffer, FName.Length);
attrName := FNameTable.FindOrAdd(FName.Buffer, FName.Length);
AttDef := TAttributeDef.Create(attrName, FColonPos);
try
AttDef.ExternallyDeclared := FSource.DTDSubsetType <> dsInternal;
@ -2656,6 +2660,7 @@ end;
procedure TXMLReader.ProcessDTD(ASource: TXMLCharSource);
begin
doc := TXMLDocument.Create;
FNameTable := doc.Names;
FDocType := TDOMDocumentTypeEx.Create(doc);
// TODO: DTD labeled version 1.1 will be rejected - must set FXML11 flag
doc.AppendChild(FDocType);
@ -2675,6 +2680,28 @@ begin
cur.AppendChild(doc.CreateEntityReference(s));
end;
procedure TXMLReader.DoStartElement;
var
NewElem: TDOMElement;
Attr: TDOMAttr;
i: Integer;
begin
with FCurrNode^.FQName^ do
NewElem := doc.CreateElementBuf(PWideChar(Key), Length(Key));
FCursorStack[FNesting-1].InternalAppend(NewElem);
FCursorStack[FNesting] := NewElem;
if Assigned(FCurrNode^.FNsUri) then
NewElem.SetNSI(FCurrNode^.FNsUri^.Key, FCurrNode^.FColonPos+1);
for i := 1 to FAttrCount do
begin
Attr := LoadAttribute(doc, @FNodeStack[FNesting+i]);
NewElem.SetAttributeNode(Attr);
// Attach element to ID map entry if necessary
if Assigned(FNodeStack[FNesting+i].FIDEntry) then
FNodeStack[FNesting+i].FIDEntry^.Data := NewElem;
end;
end;
// The code below does the bulk of the parsing, and must be as fast as possible.
// To minimize CPU cache effects, methods from different classes are kept together
@ -2743,6 +2770,8 @@ begin
CreatePINode;
xtComment:
DoComment(FCurrNode^.FValueStart, FCurrNode^.FValueLength);
xtElement:
DoStartElement;
xtEndElement:
DoEndElement;
xtDoctype:
@ -2942,13 +2971,10 @@ end;
// Element name already in FNameBuffer
procedure TXMLReader.ParseStartTag; // [39] [40] [44]
var
NewElem: TDOMElement;
Attr: TDOMAttr;
ElDef: TElementDecl;
IsEmpty: Boolean;
ElName: PHashItem;
b: TBinding;
i: Integer;
begin
if FState > rsRoot then
FatalError('Only one top-level element allowed', FName.Length)
@ -2962,11 +2988,8 @@ begin
// we're about to process a new set of attributes
Inc(FAttrTag);
NewElem := doc.CreateElementBuf(FName.Buffer, FName.Length);
FCursorStack[FNesting].InternalAppend(NewElem);
// Remember the hash entry, we'll need it often
ElName := NewElem.NSI.QName;
ElName := FNameTable.FindOrAdd(FName.Buffer, FName.Length);
// Find declaration for this element
ElDef := TElementDecl(ElName^.Data);
@ -2982,7 +3005,6 @@ begin
FPrefixedAttrs := 0;
FSpecifiedAttrs := 0;
PushVC(ElDef); // this increases FNesting
FCursorStack[FNesting] := NewElem;
FCurrNode^.FQName := ElName;
FCurrNode^.FNodeType := ntElement;
@ -3031,29 +3053,16 @@ begin
FTokenStart := FCurrNode^.FLoc;
FatalError('Unbound element name prefix "%s"', [FCurrNode^.FPrefix^.Key],-1);
end;
FCurrNode^.FNsUri := doc.Names.FindOrAdd(PWideChar(b.uri), Length(b.uri));
NewElem.SetNSI(b.uri, FCurrNode^.FColonPos+1);
FCurrNode^.FNsUri := FNameTable.FindOrAdd(PWideChar(b.uri), Length(b.uri));
end
else
begin
b := FNSHelper.DefaultNSBinding;
if Assigned(b) then
begin
FCurrNode^.FNsUri := doc.Names.FindOrAdd(PWideChar(b.uri), Length(b.uri));
NewElem.SetNSI(b.uri, FCurrNode^.FColonPos+1);
end;
FCurrNode^.FNsUri := FNameTable.FindOrAdd(PWideChar(b.uri), Length(b.uri));
end;
end;
for i := 1 to FAttrCount do
begin
Attr := LoadAttribute(doc, @FNodeStack[FNesting+i]);
NewElem.SetAttributeNode(Attr);
// Attach element to ID map entry if necessary
if Assigned(FNodeStack[FNesting+i].FIDEntry) then
FNodeStack[FNesting+i].FIDEntry^.Data := NewElem;
end;
if not IsEmpty then
begin
if not FPreserveWhitespace then // critical for testsuite compliance
@ -3119,7 +3128,7 @@ end;
begin
CheckName;
attrName := doc.Names.FindOrAdd(FName.Buffer, FName.Length);
attrName := FNameTable.FindOrAdd(FName.Buffer, FName.Length);
attrData := AllocAttributeData(attrName);
attrData^.FColonPos := FColonPos;
StoreLocation(attrData^.FLoc);
@ -3264,7 +3273,7 @@ function TXMLReader.AddBinding(attrData: PNodeData): Boolean;
var
nsUri, Pfx: PHashItem;
begin
nsUri := doc.Names.FindOrAdd(PWideChar(attrData^.FValueStr), Length(attrData^.FValueStr));
nsUri := FNameTable.FindOrAdd(PWideChar(attrData^.FValueStr), Length(attrData^.FValueStr));
if attrData^.FColonPos > 0 then
Pfx := FNSHelper.GetPrefix(@attrData^.FQName^.key[7], Length(attrData^.FQName^.key)-6)
else
@ -3320,7 +3329,7 @@ begin
if FNsAttHash.Locate(@b.uri, @AttrName^.Key[J], Length(AttrName^.Key) - J+1) then
DoErrorPos(esFatal, 'Duplicate prefixed attribute', attrData^.FLoc);
attrData^.FNsUri := doc.Names.FindOrAdd(PWideChar(b.uri), Length(b.uri));
attrData^.FNsUri := FNameTable.FindOrAdd(PWideChar(b.uri), Length(b.uri));
end;
end;