mirror of
https://github.com/Ryujinx/Opentk.git
synced 2025-01-05 22:05:29 +00:00
11070bb52a
Functions marked with the obsolete attribute will now add [Obsolete] to the generated output.
463 lines
19 KiB
C#
463 lines
19 KiB
C#
#region License
|
|
//
|
|
// The Open Toolkit Library License
|
|
//
|
|
// Copyright (c) 2006 - 2013 Stefanos Apostolopoulos for the Open Toolkit library.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
// in the Software without restriction, including without limitation the rights to
|
|
// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
// the Software, and to permit persons to whom the Software is furnished to do
|
|
// so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
// copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
|
// OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
// WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
// OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
#endregion
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Xml.Linq;
|
|
using System.Xml.XPath;
|
|
using Bind.Structures;
|
|
|
|
namespace Bind
|
|
{
|
|
using Delegate = Bind.Structures.Delegate;
|
|
using Enum = Bind.Structures.Enum;
|
|
|
|
class XmlSpecReader : ISpecReader
|
|
{
|
|
Settings Settings { get; set; }
|
|
|
|
#region Constructors
|
|
|
|
public XmlSpecReader(Settings settings)
|
|
{
|
|
if (settings == null)
|
|
throw new ArgumentNullException("settings");
|
|
Settings = settings;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ISpecReader Members
|
|
|
|
public void ReadDelegates(string file, DelegateCollection delegates, string apiname, string apiversions)
|
|
{
|
|
var specs = new XPathDocument(file);
|
|
|
|
// The pre-GL4.4 spec format does not distinguish between
|
|
// different apinames (it is assumed that different APIs
|
|
// are stored in distinct signature.xml files).
|
|
// To maintain compatibility, we detect the version of the
|
|
// signatures.xml file and ignore apiname if it is version 1.
|
|
var specversion = GetSpecVersion(specs);
|
|
if (specversion == "1")
|
|
{
|
|
apiname = null;
|
|
}
|
|
|
|
foreach (var apiversion in apiversions.Split('|'))
|
|
{
|
|
string xpath_add, xpath_delete;
|
|
GetSignaturePaths(apiname, apiversion, out xpath_add, out xpath_delete);
|
|
|
|
foreach (XPathNavigator nav in specs.CreateNavigator().Select(xpath_delete))
|
|
{
|
|
foreach (XPathNavigator node in nav.SelectChildren("function", String.Empty))
|
|
delegates.Remove(node.GetAttribute("name", String.Empty));
|
|
}
|
|
foreach (XPathNavigator nav in specs.CreateNavigator().Select(xpath_add))
|
|
{
|
|
delegates.AddRange(ReadDelegates(nav, apiversion));
|
|
}
|
|
}
|
|
}
|
|
|
|
public void ReadEnums(string file, EnumCollection enums, string apiname, string apiversions)
|
|
{
|
|
var specs = new XPathDocument(file);
|
|
|
|
// The pre-GL4.4 spec format does not distinguish between
|
|
// different apinames (it is assumed that different APIs
|
|
// are stored in distinct signature.xml files).
|
|
// To maintain compatibility, we detect the version of the
|
|
// signatures.xml file and ignore apiname if it is version 1.
|
|
var specversion = GetSpecVersion(specs);
|
|
if (specversion == "1")
|
|
{
|
|
apiname = null;
|
|
}
|
|
|
|
foreach (var apiversion in apiversions.Split('|'))
|
|
{
|
|
string xpath_add, xpath_delete;
|
|
GetSignaturePaths(apiname, apiversion, out xpath_add, out xpath_delete);
|
|
|
|
// First, read all enum definitions from spec and override file.
|
|
// Afterwards, read all token/enum overrides from overrides file.
|
|
foreach (XPathNavigator nav in specs.CreateNavigator().Select(xpath_delete))
|
|
{
|
|
foreach (XPathNavigator node in nav.SelectChildren("enum", String.Empty))
|
|
enums.Remove(node.GetAttribute("name", String.Empty));
|
|
}
|
|
foreach (XPathNavigator nav in specs.CreateNavigator().Select(xpath_add))
|
|
{
|
|
Utilities.Merge(enums, ReadEnums(nav));
|
|
}
|
|
}
|
|
}
|
|
|
|
public Dictionary<string, string> ReadTypeMap(string file)
|
|
{
|
|
using (var sr = new StreamReader(file))
|
|
{
|
|
Console.WriteLine("Reading opengl types.");
|
|
Dictionary<string, string> GLTypes = new Dictionary<string, string>();
|
|
|
|
if (sr == null)
|
|
return GLTypes;
|
|
|
|
do
|
|
{
|
|
string line = sr.ReadLine();
|
|
|
|
if (String.IsNullOrEmpty(line) || line.StartsWith("#"))
|
|
continue;
|
|
|
|
string[] words = line.Split(" ,*\t".ToCharArray(), StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
if (words[0].ToLower() == "void")
|
|
{
|
|
// Special case for "void" -> "". We make it "void" -> "void"
|
|
GLTypes.Add(words[0], "void");
|
|
}
|
|
else if (words[0] == "VoidPointer" || words[0] == "ConstVoidPointer")
|
|
{
|
|
// "(Const)VoidPointer" -> "void*"
|
|
GLTypes.Add(words[0], "void*");
|
|
}
|
|
else if (words[0] == "CharPointer" || words[0] == "charPointerARB" ||
|
|
words[0] == "ConstCharPointer")
|
|
{
|
|
// The typematching logic cannot handle pointers to pointers, e.g. CharPointer* -> char** -> string* -> string[].
|
|
// Hence we give it a push.
|
|
// Note: When both CurrentType == "String" and Pointer == true, the typematching is hardcoded to use
|
|
// String[] or StringBuilder[].
|
|
GLTypes.Add(words[0], "String");
|
|
}
|
|
/*else if (words[0].Contains("Pointer"))
|
|
{
|
|
GLTypes.Add(words[0], words[1].Replace("Pointer", "*"));
|
|
}*/
|
|
else if (words[1].Contains("GLvoid"))
|
|
{
|
|
GLTypes.Add(words[0], "void");
|
|
}
|
|
else if (words[1] == "const" && words[2] == "GLubyte")
|
|
{
|
|
GLTypes.Add(words[0], "String");
|
|
}
|
|
else if (words[1] == "struct")
|
|
{
|
|
GLTypes.Add(words[0], words[2]);
|
|
}
|
|
else
|
|
{
|
|
GLTypes.Add(words[0], words[1]);
|
|
}
|
|
}
|
|
while (!sr.EndOfStream);
|
|
|
|
return GLTypes;
|
|
}
|
|
}
|
|
|
|
public Dictionary<string, string> ReadCSTypeMap(string file)
|
|
{
|
|
using (var sr = new StreamReader(file))
|
|
{
|
|
Dictionary<string, string> CSTypes = new Dictionary<string, string>();
|
|
Console.WriteLine("Reading C# types.");
|
|
|
|
while (!sr.EndOfStream)
|
|
{
|
|
string line = sr.ReadLine();
|
|
if (String.IsNullOrEmpty(line) || line.StartsWith("#"))
|
|
continue;
|
|
|
|
string[] words = line.Split(" ,\t".ToCharArray(), StringSplitOptions.RemoveEmptyEntries);
|
|
if (words.Length < 2)
|
|
continue;
|
|
|
|
if (((Settings.Compatibility & Settings.Legacy.NoBoolParameters) != Settings.Legacy.None) && words[1] == "bool")
|
|
words[1] = "Int32";
|
|
|
|
CSTypes.Add(words[0], words[1]);
|
|
}
|
|
|
|
return CSTypes;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Private Members
|
|
|
|
static void GetSignaturePaths(string apiname, string apiversion, out string xpath_add, out string xpath_delete)
|
|
{
|
|
xpath_add = "/signatures/add";
|
|
xpath_delete = "/signatures/delete";
|
|
|
|
if (!String.IsNullOrEmpty(apiname) && !String.IsNullOrEmpty(apiversion))
|
|
{
|
|
var match = String.Format(
|
|
"[contains(concat('|', @name, '|'), '|{0}|') and " +
|
|
"(contains(concat('|', @version, '|'), '|{1}|') or not(boolean(@version)))]",
|
|
apiname,
|
|
apiversion);
|
|
xpath_add += match;
|
|
xpath_delete += match;
|
|
}
|
|
else if (!String.IsNullOrEmpty(apiname))
|
|
{
|
|
var match = String.Format("[contains(concat('|', @name, '|'), '|{0}|')]", apiname);
|
|
xpath_add += match;
|
|
xpath_delete += match;
|
|
}
|
|
}
|
|
|
|
string GetSpecVersion(XPathDocument specs)
|
|
{
|
|
var version =
|
|
specs.CreateNavigator().SelectSingleNode("/signatures")
|
|
.GetAttribute("version", String.Empty);
|
|
if (String.IsNullOrEmpty(version))
|
|
{
|
|
version = "1";
|
|
}
|
|
return version;
|
|
}
|
|
|
|
DelegateCollection ReadDelegates(XPathNavigator specs, string apiversion)
|
|
{
|
|
DelegateCollection delegates = new DelegateCollection();
|
|
var extensions = new List<string>();
|
|
|
|
string path = "function";
|
|
foreach (XPathNavigator node in specs.SelectChildren(path, String.Empty))
|
|
{
|
|
var name = node.GetAttribute("name", String.Empty).Trim();
|
|
var version = node.GetAttribute("version", String.Empty).Trim();
|
|
|
|
// Ignore functions that have a higher version number than
|
|
// our current apiversion. Extensions do not have a version,
|
|
// so we add them anyway (which is desirable).
|
|
if (!String.IsNullOrEmpty(version) && !String.IsNullOrEmpty(apiversion) &&
|
|
Decimal.Parse(version) > Decimal.Parse(apiversion))
|
|
continue;
|
|
|
|
// Check whether we are adding to an existing delegate or creating a new one.
|
|
var d = new Delegate
|
|
{
|
|
Name = name,
|
|
EntryPoint = name,
|
|
Version = node.GetAttribute("version", String.Empty).Trim(),
|
|
Category = node.GetAttribute("category", String.Empty).Trim(),
|
|
DeprecatedVersion = node.GetAttribute("deprecated", String.Empty).Trim(),
|
|
Deprecated = !String.IsNullOrEmpty(node.GetAttribute("deprecated", String.Empty)),
|
|
Extension = node.GetAttribute("extension", String.Empty).Trim() ?? "Core",
|
|
Obsolete = node.GetAttribute("obsolete", String.Empty).Trim()
|
|
};
|
|
if (!extensions.Contains(d.Extension))
|
|
extensions.Add(d.Extension);
|
|
|
|
foreach (XPathNavigator param in node.SelectChildren(XPathNodeType.Element))
|
|
{
|
|
switch (param.Name)
|
|
{
|
|
case "returns":
|
|
d.ReturnType.CurrentType = param.GetAttribute("type", String.Empty).Trim();
|
|
break;
|
|
|
|
case "param":
|
|
Parameter p = new Parameter();
|
|
p.CurrentType = param.GetAttribute("type", String.Empty).Trim();
|
|
p.Name = param.GetAttribute("name", String.Empty).Trim();
|
|
|
|
string element_count = param.GetAttribute("elementcount", String.Empty).Trim();
|
|
if (String.IsNullOrEmpty(element_count))
|
|
{
|
|
element_count = param.GetAttribute("count", String.Empty).Trim();
|
|
if (!String.IsNullOrEmpty(element_count))
|
|
{
|
|
int count;
|
|
if (Int32.TryParse(element_count, out count))
|
|
{
|
|
p.ElementCount = count;
|
|
}
|
|
}
|
|
}
|
|
|
|
p.Flow = Parameter.GetFlowDirection(param.GetAttribute("flow", String.Empty).Trim());
|
|
|
|
d.Parameters.Add(p);
|
|
break;
|
|
}
|
|
}
|
|
|
|
delegates.Add(d);
|
|
}
|
|
|
|
Utilities.InitExtensions(extensions);
|
|
return delegates;
|
|
}
|
|
|
|
EnumCollection ReadEnums(XPathNavigator nav)
|
|
{
|
|
EnumCollection enums = new EnumCollection();
|
|
Enum all = new Enum() { Name = Settings.CompleteEnumName };
|
|
|
|
if (nav != null)
|
|
{
|
|
var reuse_list = new List<KeyValuePair<Enum, string>>();
|
|
|
|
// First pass: collect all available tokens and enums
|
|
foreach (XPathNavigator node in nav.SelectChildren("enum", String.Empty))
|
|
{
|
|
Enum e = new Enum()
|
|
{
|
|
Name = node.GetAttribute("name", String.Empty).Trim(),
|
|
Type = node.GetAttribute("type", String.Empty).Trim()
|
|
};
|
|
|
|
if (String.IsNullOrEmpty(e.Name))
|
|
throw new InvalidOperationException(String.Format("Empty name for enum element {0}", node.ToString()));
|
|
|
|
// It seems that all flag collections contain "Mask" in their names.
|
|
// This looks like a heuristic, but it holds 100% in practice
|
|
// (checked all enums to make sure).
|
|
e.IsFlagCollection = e.Name.ToLower().Contains("mask");
|
|
|
|
foreach (XPathNavigator param in node.SelectChildren(XPathNodeType.Element))
|
|
{
|
|
Constant c = null;
|
|
switch (param.Name)
|
|
{
|
|
case "token":
|
|
c = new Constant
|
|
{
|
|
Name = param.GetAttribute("name", String.Empty).Trim(),
|
|
Value = param.GetAttribute("value", String.Empty).Trim()
|
|
};
|
|
break;
|
|
|
|
case "use":
|
|
c = new Constant
|
|
{
|
|
Name = param.GetAttribute("token", String.Empty).Trim(),
|
|
Reference = param.GetAttribute("enum", String.Empty).Trim(),
|
|
Value = param.GetAttribute("token", String.Empty).Trim(),
|
|
};
|
|
break;
|
|
|
|
case "reuse":
|
|
var reuse_enum = param.GetAttribute("enum", String.Empty).Trim();
|
|
reuse_list.Add(new KeyValuePair<Enum, string>(e, reuse_enum));
|
|
break;
|
|
|
|
default:
|
|
throw new NotSupportedException();
|
|
}
|
|
|
|
if (c != null)
|
|
{
|
|
Utilities.Merge(all, c);
|
|
try
|
|
{
|
|
if (!e.ConstantCollection.ContainsKey(c.Name))
|
|
{
|
|
e.ConstantCollection.Add(c.Name, c);
|
|
}
|
|
else if (e.ConstantCollection[c.Name].Value != c.Value)
|
|
{
|
|
var existing = e.ConstantCollection[c.Name];
|
|
if (existing.Reference != null && c.Reference == null)
|
|
{
|
|
e.ConstantCollection[c.Name] = c;
|
|
}
|
|
else if (existing.Reference == null && c.Reference != null)
|
|
{
|
|
// Keep existing
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine("[Warning] Conflicting token {0}.{1} with value {2} != {3}",
|
|
e.Name, c.Name, e.ConstantCollection[c.Name].Value, c.Value);
|
|
}
|
|
}
|
|
}
|
|
catch (ArgumentException ex)
|
|
{
|
|
Console.WriteLine("[Warning] Failed to add constant {0} to enum {1}: {2}", c.Name, e.Name, ex.Message);
|
|
}
|
|
}
|
|
}
|
|
|
|
Utilities.Merge(enums, e);
|
|
}
|
|
|
|
// Second pass: resolve "reuse" directives
|
|
restart:
|
|
foreach (var pair in reuse_list)
|
|
{
|
|
var e = pair.Key;
|
|
var reuse = pair.Value;
|
|
var count = e.ConstantCollection.Count;
|
|
|
|
if (enums.ContainsKey(reuse))
|
|
{
|
|
var reuse_enum = enums[reuse];
|
|
foreach (var token in reuse_enum.ConstantCollection.Values)
|
|
{
|
|
Utilities.Merge(e, token);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine("[Warning] Reuse token not found: {0}", reuse);
|
|
}
|
|
|
|
if (count != e.ConstantCollection.Count)
|
|
{
|
|
// Restart resolution of reuse directives whenever
|
|
// we modify an enum. This is the simplest (brute) way
|
|
// to resolve chains of reuse directives:
|
|
// e.g. enum A reuses B which reuses C
|
|
goto restart;
|
|
}
|
|
}
|
|
}
|
|
|
|
Utilities.Merge(enums, all);
|
|
return enums;
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|