2016-08-03 10:59:49 +00:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// <copyright file="XmlQueryStaticData.cs" company="Microsoft">
|
|
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
|
|
// </copyright>
|
2017-08-21 15:34:15 +00:00
|
|
|
// <owner current="true" primary="true">Microsoft</owner>
|
2016-08-03 10:59:49 +00:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Diagnostics;
|
|
|
|
using System.IO;
|
|
|
|
using System.Reflection;
|
|
|
|
using System.Xml.Xsl.IlGen;
|
|
|
|
using System.Xml.Xsl.Qil;
|
|
|
|
|
|
|
|
namespace System.Xml.Xsl.Runtime {
|
|
|
|
/// <summary>
|
|
|
|
/// Contains all static data that is used by the runtime.
|
|
|
|
/// </summary>
|
|
|
|
internal class XmlQueryStaticData {
|
|
|
|
// Name of the field to serialize to
|
|
|
|
public const string DataFieldName = "staticData";
|
|
|
|
public const string TypesFieldName = "ebTypes";
|
|
|
|
|
|
|
|
// Format version marker to support versioning: (major << 8) | minor
|
|
|
|
private const int CurrentFormatVersion = (0 << 8) | 0;
|
|
|
|
|
|
|
|
private XmlWriterSettings defaultWriterSettings;
|
|
|
|
private IList<WhitespaceRule> whitespaceRules;
|
|
|
|
private string[] names;
|
|
|
|
private StringPair[][] prefixMappingsList;
|
|
|
|
private Int32Pair[] filters;
|
|
|
|
private XmlQueryType[] types;
|
|
|
|
private XmlCollation[] collations;
|
|
|
|
private string[] globalNames;
|
|
|
|
private EarlyBoundInfo[] earlyBound;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Constructor.
|
|
|
|
/// </summary>
|
|
|
|
public XmlQueryStaticData(XmlWriterSettings defaultWriterSettings, IList<WhitespaceRule> whitespaceRules, StaticDataManager staticData) {
|
|
|
|
Debug.Assert(defaultWriterSettings != null && staticData != null);
|
|
|
|
this.defaultWriterSettings = defaultWriterSettings;
|
|
|
|
this.whitespaceRules = whitespaceRules;
|
|
|
|
this.names = staticData.Names;
|
|
|
|
this.prefixMappingsList = staticData.PrefixMappingsList;
|
|
|
|
this.filters = staticData.NameFilters;
|
|
|
|
this.types = staticData.XmlTypes;
|
|
|
|
this.collations = staticData.Collations;
|
|
|
|
this.globalNames = staticData.GlobalNames;
|
|
|
|
this.earlyBound = staticData.EarlyBound;
|
|
|
|
|
|
|
|
#if DEBUG
|
|
|
|
// Round-trip check
|
|
|
|
byte[] data;
|
|
|
|
Type[] ebTypes;
|
|
|
|
this.GetObjectData(out data, out ebTypes);
|
|
|
|
XmlQueryStaticData copy = new XmlQueryStaticData(data, ebTypes);
|
|
|
|
|
|
|
|
this.defaultWriterSettings = copy.defaultWriterSettings;
|
|
|
|
this.whitespaceRules = copy.whitespaceRules;
|
|
|
|
this.names = copy.names;
|
|
|
|
this.prefixMappingsList = copy.prefixMappingsList;
|
|
|
|
this.filters = copy.filters;
|
|
|
|
this.types = copy.types;
|
|
|
|
this.collations = copy.collations;
|
|
|
|
this.globalNames = copy.globalNames;
|
|
|
|
this.earlyBound = copy.earlyBound;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Deserialize XmlQueryStaticData object from a byte array.
|
|
|
|
/// </summary>
|
|
|
|
public XmlQueryStaticData(byte[] data, Type[] ebTypes) {
|
|
|
|
MemoryStream dataStream = new MemoryStream(data, /*writable:*/false);
|
|
|
|
XmlQueryDataReader dataReader = new XmlQueryDataReader(dataStream);
|
|
|
|
int length;
|
|
|
|
|
|
|
|
// Read a format version
|
|
|
|
int formatVersion = dataReader.ReadInt32Encoded();
|
|
|
|
|
|
|
|
// Changes in the major part of version are not supported
|
|
|
|
if ((formatVersion & ~0xff) > CurrentFormatVersion)
|
|
|
|
throw new NotSupportedException();
|
|
|
|
|
|
|
|
// XmlWriterSettings defaultWriterSettings;
|
|
|
|
defaultWriterSettings = new XmlWriterSettings(dataReader);
|
|
|
|
|
|
|
|
// IList<WhitespaceRule> whitespaceRules;
|
|
|
|
length = dataReader.ReadInt32();
|
|
|
|
if (length != 0) {
|
|
|
|
this.whitespaceRules = new WhitespaceRule[length];
|
|
|
|
for (int idx = 0; idx < length; idx++) {
|
|
|
|
this.whitespaceRules[idx] = new WhitespaceRule(dataReader);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// string[] names;
|
|
|
|
length = dataReader.ReadInt32();
|
|
|
|
if (length != 0) {
|
|
|
|
this.names = new string[length];
|
|
|
|
for (int idx = 0; idx < length; idx++) {
|
|
|
|
this.names[idx] = dataReader.ReadString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// StringPair[][] prefixMappingsList;
|
|
|
|
length = dataReader.ReadInt32();
|
|
|
|
if (length != 0) {
|
|
|
|
this.prefixMappingsList = new StringPair[length][];
|
|
|
|
for (int idx = 0; idx < length; idx++) {
|
|
|
|
int length2 = dataReader.ReadInt32();
|
|
|
|
this.prefixMappingsList[idx] = new StringPair[length2];
|
|
|
|
for (int idx2 = 0; idx2 < length2; idx2++) {
|
|
|
|
this.prefixMappingsList[idx][idx2] = new StringPair(dataReader.ReadString(), dataReader.ReadString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Int32Pair[] filters;
|
|
|
|
length = dataReader.ReadInt32();
|
|
|
|
if (length != 0) {
|
|
|
|
this.filters = new Int32Pair[length];
|
|
|
|
for (int idx = 0; idx < length; idx++) {
|
|
|
|
this.filters[idx] = new Int32Pair(dataReader.ReadInt32Encoded(), dataReader.ReadInt32Encoded());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// XmlQueryType[] types;
|
|
|
|
length = dataReader.ReadInt32();
|
|
|
|
if (length != 0) {
|
|
|
|
this.types = new XmlQueryType[length];
|
|
|
|
for (int idx = 0; idx < length; idx++) {
|
|
|
|
this.types[idx] = XmlQueryTypeFactory.Deserialize(dataReader);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// XmlCollation[] collations;
|
|
|
|
length = dataReader.ReadInt32();
|
|
|
|
if (length != 0) {
|
|
|
|
this.collations = new XmlCollation[length];
|
|
|
|
for (int idx = 0; idx < length; idx++) {
|
|
|
|
this.collations[idx] = new XmlCollation(dataReader);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// string[] globalNames;
|
|
|
|
length = dataReader.ReadInt32();
|
|
|
|
if (length != 0) {
|
|
|
|
this.globalNames = new string[length];
|
|
|
|
for (int idx = 0; idx < length; idx++) {
|
|
|
|
this.globalNames[idx] = dataReader.ReadString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// EarlyBoundInfo[] earlyBound;
|
|
|
|
length = dataReader.ReadInt32();
|
|
|
|
if (length != 0) {
|
|
|
|
this.earlyBound = new EarlyBoundInfo[length];
|
|
|
|
for (int idx = 0; idx < length; idx++) {
|
|
|
|
this.earlyBound[idx] = new EarlyBoundInfo(dataReader.ReadString(), ebTypes[idx]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Debug.Assert(formatVersion != CurrentFormatVersion || dataReader.Read() == -1, "Extra data at the end of the stream");
|
|
|
|
dataReader.Close();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Serialize XmlQueryStaticData object into a byte array.
|
|
|
|
/// </summary>
|
|
|
|
public void GetObjectData(out byte[] data, out Type[] ebTypes) {
|
|
|
|
MemoryStream dataStream = new MemoryStream(4096);
|
|
|
|
XmlQueryDataWriter dataWriter = new XmlQueryDataWriter(dataStream);
|
|
|
|
|
|
|
|
// First put the format version
|
|
|
|
dataWriter.WriteInt32Encoded(CurrentFormatVersion);
|
|
|
|
|
|
|
|
// XmlWriterSettings defaultWriterSettings;
|
|
|
|
defaultWriterSettings.GetObjectData(dataWriter);
|
|
|
|
|
|
|
|
// IList<WhitespaceRule> whitespaceRules;
|
|
|
|
if (this.whitespaceRules == null) {
|
|
|
|
dataWriter.Write(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dataWriter.Write(this.whitespaceRules.Count);
|
|
|
|
foreach (WhitespaceRule rule in this.whitespaceRules) {
|
|
|
|
rule.GetObjectData(dataWriter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// string[] names;
|
|
|
|
if (this.names == null) {
|
|
|
|
dataWriter.Write(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dataWriter.Write(this.names.Length);
|
|
|
|
foreach (string name in this.names) {
|
|
|
|
dataWriter.Write(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// StringPair[][] prefixMappingsList;
|
|
|
|
if (this.prefixMappingsList == null) {
|
|
|
|
dataWriter.Write(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dataWriter.Write(this.prefixMappingsList.Length);
|
|
|
|
foreach (StringPair[] mappings in this.prefixMappingsList) {
|
|
|
|
dataWriter.Write(mappings.Length);
|
|
|
|
foreach (StringPair mapping in mappings) {
|
|
|
|
dataWriter.Write(mapping.Left);
|
|
|
|
dataWriter.Write(mapping.Right);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Int32Pair[] filters;
|
|
|
|
if (this.filters == null) {
|
|
|
|
dataWriter.Write(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dataWriter.Write(this.filters.Length);
|
|
|
|
foreach (Int32Pair filter in this.filters) {
|
|
|
|
dataWriter.WriteInt32Encoded(filter.Left);
|
|
|
|
dataWriter.WriteInt32Encoded(filter.Right);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// XmlQueryType[] types;
|
|
|
|
if (this.types == null) {
|
|
|
|
dataWriter.Write(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dataWriter.Write(this.types.Length);
|
|
|
|
foreach (XmlQueryType type in this.types) {
|
|
|
|
XmlQueryTypeFactory.Serialize(dataWriter, type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// XmlCollation[] collations;
|
|
|
|
if (collations == null) {
|
|
|
|
dataWriter.Write(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dataWriter.Write(this.collations.Length);
|
|
|
|
foreach (XmlCollation collation in this.collations) {
|
|
|
|
collation.GetObjectData(dataWriter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// string[] globalNames;
|
|
|
|
if (this.globalNames == null) {
|
|
|
|
dataWriter.Write(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dataWriter.Write(this.globalNames.Length);
|
|
|
|
foreach (string name in this.globalNames) {
|
|
|
|
dataWriter.Write(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// EarlyBoundInfo[] earlyBound;
|
|
|
|
if (this.earlyBound == null) {
|
|
|
|
dataWriter.Write(0);
|
|
|
|
ebTypes = null;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dataWriter.Write(this.earlyBound.Length);
|
|
|
|
ebTypes = new Type[this.earlyBound.Length];
|
|
|
|
int idx = 0;
|
|
|
|
foreach (EarlyBoundInfo info in this.earlyBound) {
|
|
|
|
dataWriter.Write(info.NamespaceUri);
|
|
|
|
ebTypes[idx++] = info.EarlyBoundType;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dataWriter.Close();
|
|
|
|
data = dataStream.ToArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Return the default writer settings.
|
|
|
|
/// </summary>
|
|
|
|
public XmlWriterSettings DefaultWriterSettings {
|
|
|
|
get { return this.defaultWriterSettings; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Return the rules used for whitespace stripping/preservation.
|
|
|
|
/// </summary>
|
|
|
|
public IList<WhitespaceRule> WhitespaceRules {
|
|
|
|
get { return this.whitespaceRules; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Return array of names used by this query.
|
|
|
|
/// </summary>
|
|
|
|
public string[] Names {
|
|
|
|
get { return this.names; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Return array of prefix mappings used by this query.
|
|
|
|
/// </summary>
|
|
|
|
public StringPair[][] PrefixMappingsList {
|
|
|
|
get { return this.prefixMappingsList; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Return array of name filter specifications used by this query.
|
|
|
|
/// </summary>
|
|
|
|
public Int32Pair[] Filters {
|
|
|
|
get { return this.filters; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Return array of types used by this query.
|
|
|
|
/// </summary>
|
|
|
|
public XmlQueryType[] Types {
|
|
|
|
get { return this.types; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Return array of collations used by this query.
|
|
|
|
/// </summary>
|
|
|
|
public XmlCollation[] Collations {
|
|
|
|
get { return this.collations; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Return names of all global variables and parameters used by this query.
|
|
|
|
/// </summary>
|
|
|
|
public string[] GlobalNames {
|
|
|
|
get { return this.globalNames; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Return array of early bound object information related to this query.
|
|
|
|
/// </summary>
|
|
|
|
public EarlyBoundInfo[] EarlyBound {
|
|
|
|
get { return this.earlyBound; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Subclass of BinaryReader used to serialize query static data.
|
|
|
|
/// </summary>
|
|
|
|
internal class XmlQueryDataReader : BinaryReader {
|
|
|
|
public XmlQueryDataReader(Stream input) : base(input) { }
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Read in a 32-bit integer in compressed format.
|
|
|
|
/// </summary>
|
|
|
|
public int ReadInt32Encoded() {
|
|
|
|
return Read7BitEncodedInt();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Read a string value from the stream. Value can be null.
|
|
|
|
/// </summary>
|
|
|
|
public string ReadStringQ() {
|
|
|
|
return ReadBoolean() ? ReadString() : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Read a signed byte value from the stream and check if it belongs to the given diapason.
|
|
|
|
/// </summary>
|
|
|
|
public sbyte ReadSByte(sbyte minValue, sbyte maxValue) {
|
|
|
|
sbyte value = ReadSByte();
|
|
|
|
if (value < minValue)
|
|
|
|
throw new ArgumentOutOfRangeException("minValue");
|
|
|
|
if (maxValue < value)
|
|
|
|
throw new ArgumentOutOfRangeException("maxValue");
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Subclass of BinaryWriter used to deserialize query static data.
|
|
|
|
/// </summary>
|
|
|
|
internal class XmlQueryDataWriter : BinaryWriter {
|
|
|
|
public XmlQueryDataWriter(Stream output) : base(output) { }
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Write a 32-bit integer in a compressed format.
|
|
|
|
/// </summary>
|
|
|
|
public void WriteInt32Encoded(int value) {
|
|
|
|
Write7BitEncodedInt(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Write a string value to the stream. Value can be null.
|
|
|
|
/// </summary>
|
|
|
|
public void WriteStringQ(string value) {
|
|
|
|
Write(value != null);
|
|
|
|
if (value != null) {
|
|
|
|
Write(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|