Opentk/Source/Bind/Structures/Function.cs

300 lines
8.4 KiB
C#
Raw Normal View History

2007-08-01 21:14:39 +00:00
#region --- License ---
/* Copyright (c) 2006, 2007 Stefanos Apostolopoulos
* See license.txt for license info
*/
#endregion
using System;
using System.Collections.Generic;
using System.Text;
2007-08-01 21:14:39 +00:00
using System.Text.RegularExpressions;
2007-08-01 09:27:57 +00:00
namespace Bind.Structures
{
2007-08-01 09:27:57 +00:00
public class Function : Delegate
{
2007-08-01 21:14:39 +00:00
internal static FunctionCollection Wrappers;
private static bool loaded;
internal static void Initialize()
{
if (!loaded)
{
Wrappers = new FunctionCollection();
loaded = true;
}
}
private static List<string> endings = new List<string>(
new string[]
{
"fv",
"f",
"dv",
"d",
"i",
"iv",
"s",
"sv",
"b",
"bv",
"ui",
"uiv",
"us",
"usv",
"ub",
"ubv"
});
2007-08-01 09:27:57 +00:00
#region --- Constructors ---
public Function()
2007-08-01 09:27:57 +00:00
: base()
{
Body = new FunctionBody();
}
2007-08-01 21:14:39 +00:00
/*
public Function(Function f)
2007-08-01 09:27:57 +00:00
: base(f)
{
this.Body = new FunctionBody(f.Body);
2007-08-01 21:14:39 +00:00
this.Name = f.Name;
}
2007-08-01 21:14:39 +00:00
*/
2007-08-01 09:27:57 +00:00
public Function(Delegate d)
: base(d)
{
2007-08-01 21:14:39 +00:00
if (d is Function)
this.Body = new FunctionBody((d as Function).Body);
else
this.Body = new FunctionBody();
this.Name = d.Name;
}
#endregion
2007-08-01 21:14:39 +00:00
#region public override bool Unsafe
2007-08-01 09:27:57 +00:00
public override bool Unsafe
{
2007-08-01 09:27:57 +00:00
get
{
2007-08-01 09:27:57 +00:00
if (Settings.Compatibility == Settings.Legacy.Tao)
return false;
2007-08-01 09:27:57 +00:00
return base.Unsafe;
}
}
2007-08-01 21:14:39 +00:00
#endregion
#region public FunctionBody Body
2007-08-01 09:27:57 +00:00
FunctionBody _body;
2007-08-01 09:27:57 +00:00
public FunctionBody Body
{
2007-08-01 09:27:57 +00:00
get { return _body; }
set { _body = value; }
}
#endregion
2007-08-01 21:14:39 +00:00
#region public string TrimmedName
string trimmedName;
/// <summary>
/// Gets or sets the name of the opengl function, trimming the excess 234dfubsiv endings..
/// </summary>
public string TrimmedName
{
get { return trimmedName; }
set
{
if (!String.IsNullOrEmpty(value))
trimmedName = value.Trim();
}
}
#endregion
#region public override string Name
/// <summary>
/// Gets or sets the name of the opengl function.
/// If no Tao compatibility is set, set TrimmedName to Name, after removing
/// [u][bsifd][v].
/// </summary>
public override string Name
{
get { return base.Name; }
set
{
base.Name = value;
// If we don't need compatibility with Tao,
// remove the Extension and the overload information from the name
// (Extension == "ARB", "EXT", etc, overload == [u][bsidf][v])
// TODO: Use some regex's here, to reduce clutter.
if (Settings.Compatibility != Settings.Legacy.Tao)
{
string ext = Utilities.GetGL2Extension(value);
string trimmedName = value;
// Remove extension
if (!String.IsNullOrEmpty(ext))
{
trimmedName = trimmedName.Substring(0, trimmedName.Length - ext.Length);
}
// Remove overload
if (endings.Contains(trimmedName.Substring(trimmedName.Length - 3)))
{
TrimmedName = trimmedName.Substring(0, trimmedName.Length - 3);
return;
}
if (endings.Contains(trimmedName.Substring(trimmedName.Length - 2)))
{
TrimmedName = trimmedName.Substring(0, trimmedName.Length - 2);
return;
}
if (endings.Contains(trimmedName.Substring(trimmedName.Length - 1)))
{
// An ending 's' may be either a plural form (glCallLists), which we
// do not want to change, or an actual overload (glColor3s). We assume
// (perhaps incorrectly), that an 's' preceeded be a digit indicates an
// overload. If there is no digit, we assume a plural form (no change).
if (Char.IsDigit(trimmedName[trimmedName.Length - 2]))
TrimmedName = trimmedName.Substring(0, trimmedName.Length - 1);
return;
}
}
}
}
#endregion
2007-08-01 09:27:57 +00:00
#region public override string ToString()
2007-08-01 09:27:57 +00:00
public override string ToString()
{
StringBuilder sb = new StringBuilder();
2007-08-01 09:27:57 +00:00
sb.Append(Unsafe ? "unsafe " : "");
sb.Append(ReturnType);
sb.Append(" ");
if (Settings.Compatibility == Settings.Legacy.Tao)
{
sb.Append("gl");
}
2007-08-01 21:14:39 +00:00
sb.Append(!String.IsNullOrEmpty(TrimmedName) ? TrimmedName : Name);
2007-08-01 09:27:57 +00:00
sb.Append(Parameters.ToString(true));
if (Body.Count > 0)
{
2007-08-01 09:27:57 +00:00
sb.AppendLine();
sb.Append(Body.ToString());
}
return sb.ToString();
}
#endregion
2007-08-01 09:27:57 +00:00
#region public Function GetCLSCompliantFunction(Dictionary<string, string> CSTypes)
2007-08-01 09:27:57 +00:00
public Function GetCLSCompliantFunction(Dictionary<string, string> CSTypes)
{
2007-08-01 09:27:57 +00:00
Function f = new Function(this);
2007-08-01 09:27:57 +00:00
for (int i = 0; i < f.Parameters.Count; i++)
{
2007-08-01 21:14:39 +00:00
f.Parameters[i].CurrentType = f.Parameters[i].GetCLSCompliantType();
2007-08-01 09:27:57 +00:00
}
2007-08-01 09:27:57 +00:00
f.Body.Clear();
if (!f.NeedsWrapper)
{
2007-08-01 21:14:39 +00:00
f.Body.Add((f.ReturnType.CurrentType != "void" ? "return " + this.CallString() : this.CallString()) + ";");
2007-08-01 09:27:57 +00:00
}
else
{
2007-08-01 21:14:39 +00:00
f.Body.AddRange(this.GetBodyWithPins(true));
}
2007-08-01 09:27:57 +00:00
// The type system cannot differentiate between functions with the same parameters
// but different return types. Tough, only CLS-Compliant function in that case.
//f.ReturnType.Type = f.ReturnType.GetCLSCompliantType(CSTypes);
return f;
}
#endregion
}
#region class FunctionBody : List<string>
public class FunctionBody : List<string>
{
public FunctionBody()
{
}
public FunctionBody(FunctionBody fb)
{
foreach (string s in fb)
{
this.Add(s);
}
}
public override string ToString()
{
if (this.Count == 0)
return String.Empty;
StringBuilder sb = new StringBuilder(this.Count);
2007-08-01 09:27:57 +00:00
sb.AppendLine("{");
foreach (string s in this)
{
2007-08-01 09:27:57 +00:00
sb.AppendLine(" " + s);
}
2007-08-01 09:27:57 +00:00
sb.AppendLine("}");
return sb.ToString();
}
}
#endregion
2007-08-01 09:27:57 +00:00
2007-08-01 21:14:39 +00:00
#region class FunctionCollection : Dictionary<string, List<Function>>
2007-08-01 09:27:57 +00:00
class FunctionCollection : Dictionary<string, List<Function>>
{
public void Add(Function f)
{
if (!this.ContainsKey(f.Extension))
{
this.Add(f.Extension, new List<Function>());
this[f.Extension].Add(f);
}
else
{
this[f.Extension].Add(f);
}
}
public void AddRange(IEnumerable<Function> functions)
{
foreach (Function f in functions)
{
this.Add(f);
}
}
}
2007-08-01 21:14:39 +00:00
#endregion
}