// GtkSharp.Generation.MethodBody.cs - The MethodBody Generation Class. // // Author: Mike Kestner // // Copyright (c) 2001-2003 Mike Kestner // Copyright (c) 2003-2004 Novell, Inc. // // This program is free software; you can redistribute it and/or // modify it under the terms of version 2 of the GNU General Public // License as published by the Free Software Foundation. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // General Public License for more details. // // You should have received a copy of the GNU General Public // License along with this program; if not, write to the // Free Software Foundation, Inc., 59 Temple Place - Suite 330, // Boston, MA 02111-1307, USA. namespace GtkSharp.Generation { using System; using System.Collections; using System.IO; public class MethodBody { Parameters parameters; private string impl_ns; public MethodBody (Parameters parms, string impl_ns) { parameters = parms; this.impl_ns = impl_ns; } private bool UsesHandle (IGeneratable igen) { return igen is ManualGen || igen is ObjectGen || igen is InterfaceGen || igen is OpaqueGen; } private string CastFromInt (string type) { return type != "int" ? "(" + type + ") " : ""; } private string CallArrayLength (Parameter array, Parameter length) { string result = array.NullOk ? array.Name + " != null ? " : ""; result += CastFromInt (length.CSType) + array.Name + ".Length"; result += array.NullOk ? ": 0" : ""; return length.Generatable.CallByName (result); } public string GetCallString (bool is_set) { if (parameters == null) return ""; string[] result = new string [parameters.Count]; for (int i = 0; i < parameters.Count; i++) { Parameter p = parameters [i]; IGeneratable igen = p.Generatable; if (p.IsCount) { if (i > 0 && parameters [i - 1].IsArray) { result[i] = CallArrayLength (parameters[i - 1], p); continue; } else if (i < parameters.Count - 1 && parameters [i + 1].IsArray) { result[i] = CallArrayLength (parameters[i + 1], p); continue; } } else if (i > 0 && parameters [i - 1].IsString && p.IsLength) { result[i] = igen.CallByName (CastFromInt (p.CSType) + parameters [i - 1].Name + ".Length"); continue; } else if (p.IsArray && p.MarshalType != p.CSType) { result[i] = (is_set && i == 0 ? "native_value" : "native_" + p.Name); continue; } string call_parm = p.CallByName (is_set && i == 0 ? "value" : p.Name); if (p.CType == "GError**") { result [i] += "out "; } else if (p.PassAs != "") { if (p.Generatable is LPGen || p.Generatable is LPUGen || !p.MarshalType.EndsWith ("IntPtr")) result [i] += p.PassAs + " "; if (igen is EnumGen) call_parm = p.Name + "_as_int"; else if (igen is LPUGen || igen is LPGen) call_parm = p.Name + "_as_ptr"; else if (UsesHandle (igen)) { call_parm = p.PassAs + " " + call_parm.Replace (".Handle", "_handle"); } } if (p.CType == "GError**") { call_parm = call_parm.Replace (p.Name, "error"); } else if (p.IsUserData && !parameters.HideData && (i == parameters.Count - 1)) { call_parm = "IntPtr.Zero"; } result [i] += call_parm; } string call_string = String.Join (", ", result); call_string = call_string.Replace ("out ref", "out"); call_string = call_string.Replace ("out out ", "out "); call_string = call_string.Replace ("ref ref", "ref"); return call_string; } public void Initialize (GenerationInfo gen_info, bool is_get, bool is_set, string indent) { if (parameters == null) return; StreamWriter sw = gen_info.Writer; for (int i = 0; i < parameters.Count; i++) { Parameter p = parameters [i]; IGeneratable gen = p.Generatable; string name = p.Name; if (is_set) name = "value"; if ((is_get || p.PassAs == "out") && UsesHandle (gen)) sw.WriteLine(indent + "\t\t\tIntPtr " + name + "_handle;"); if (p.PassAs == "out" && gen is EnumGen) sw.WriteLine(indent + "\t\t\tint " + name + "_as_int;"); if (p.PassAs == "out" && (gen is LPGen || gen is LPUGen)) sw.WriteLine(indent + "\t\t\t" + gen.MarshalType + " " + name + "_as_ptr;"); if (p.IsArray && p.MarshalType != p.CSType) { sw.WriteLine(indent + "\t\t\t{0}[] native_{1} = new {0} [{1}.Length];", p.MarshalType.TrimEnd('[', ']'), name); sw.WriteLine(indent + "\t\t\tfor (int i = 0; i < {0}.Length; i++)", name); sw.WriteLine(indent + "\t\t\t\tnative_{0} [i] = {1};", name, p.CallByName (name + "[i]")); } if (gen is CallbackGen) { CallbackGen cbgen = gen as CallbackGen; string wrapper = cbgen.GenWrapper(impl_ns, gen_info); sw.WriteLine (indent + "\t\t\t{0} {1}_wrapper = null;", wrapper, name); sw.Write (indent + "\t\t\t"); if (p.NullOk) sw.Write ("if ({0} != null) ", name); sw.WriteLine ("{1}_wrapper = new {0} ({1}, {2});", wrapper, p.Name, parameters.Static ? "null" : "this"); } } if (ThrowsException) sw.WriteLine (indent + "\t\t\tIntPtr error = IntPtr.Zero;"); } public void InitAccessor (StreamWriter sw, Signature sig, string indent) { sw.WriteLine (indent + "\t\t\t" + sig.AccessorType + " " + sig.AccessorName + ";"); } public void Finish (StreamWriter sw, string indent) { if (parameters == null) return; for (int i = 0; i < parameters.Count; i++) { Parameter p = parameters [i]; IGeneratable gen = p.Generatable; if (p.PassAs == "out" && gen is EnumGen) sw.WriteLine(indent + "\t\t\t" + p.Name + " = (" + p.CSType + ") " + p.Name + "_as_int;"); if (p.PassAs == "out" && (gen is LPGen || gen is LPUGen)) sw.WriteLine(indent + "\t\t\t" + p.Name + " = (" + p.CSType + ") " + p.Name + "_as_ptr;"); if (p.PassAs == "out" && UsesHandle (gen)) sw.WriteLine(indent + "\t\t\t" + p.Name + " = " + gen.FromNativeReturn (p.Name + "_handle") + ";"); } } public void FinishAccessor (StreamWriter sw, Signature sig, string indent) { sw.WriteLine (indent + "\t\t\treturn " + sig.AccessorName + ";"); } public void HandleException (StreamWriter sw, string indent) { if (!ThrowsException) return; sw.WriteLine (indent + "\t\t\tif (error != IntPtr.Zero) throw new GLib.GException (error);"); } public bool ThrowsException { get { if (parameters == null || parameters.Count < 1) return false; return parameters [parameters.Count - 1].CType == "GError**"; } } } }