c6a3bf4b73
* generator/CallbackGen.cs: connect return value count parameters * generator/ManagedCallString.cs: out count params aren't special * generator/MethodBody.cs: don't finish hidden params * generator/Parameters.cs: explicit IsCount setting support, with Parameters.GetCountParameter(name) interface. Clear IsCount values during validation if they have no associated array. Any remaining count params must be associated with a retval, so hide them. * generator/ReturnValue.cs: support array_length_param attribute to associate a return value with a "count" param. Use new helper methods to marshal these array retvals to IntPtr and back. * glib/Marshaller.cs: support for byte[] marshaling to and from IntPtr using a count and type in the from native direction. * gtk/Gtk.metadata: mark TextBufferSerializeFunc return type as an array with the length specific in the length param.
189 lines
5.7 KiB
C#
189 lines
5.7 KiB
C#
// GtkSharp.Generation.ReturnValue.cs - The ReturnValue Generatable.
|
|
//
|
|
// Author: Mike Kestner <mkestner@novell.com>
|
|
//
|
|
// Copyright (c) 2004-2005 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.Xml;
|
|
|
|
public class ReturnValue {
|
|
|
|
|
|
bool is_null_term;
|
|
bool is_array;
|
|
bool elements_owned;
|
|
bool owned;
|
|
string array_length_param = String.Empty;
|
|
string ctype = String.Empty;
|
|
string default_value = String.Empty;
|
|
string element_ctype = String.Empty;
|
|
Parameter count_param;
|
|
|
|
public ReturnValue (XmlElement elem)
|
|
{
|
|
if (elem != null) {
|
|
is_null_term = elem.HasAttribute ("null_term_array");
|
|
is_array = elem.HasAttribute ("array") || elem.HasAttribute ("array_length_param");
|
|
array_length_param = elem.GetAttribute ("array_length_param");
|
|
elements_owned = elem.GetAttribute ("elements_owned") == "true";
|
|
owned = elem.GetAttribute ("owned") == "true";
|
|
ctype = elem.GetAttribute("type");
|
|
default_value = elem.GetAttribute ("default_value");
|
|
element_ctype = elem.GetAttribute ("element_type");
|
|
}
|
|
}
|
|
|
|
public Parameter CountParameter {
|
|
get { return count_param; }
|
|
set { count_param = value; }
|
|
}
|
|
|
|
public string CountParameterName {
|
|
get { return array_length_param; }
|
|
}
|
|
|
|
public string CType {
|
|
get {
|
|
return ctype;
|
|
}
|
|
}
|
|
|
|
public string CSType {
|
|
get {
|
|
if (IGen == null)
|
|
return String.Empty;
|
|
|
|
if (ElementType != String.Empty)
|
|
return ElementType + "[]";
|
|
|
|
return IGen.QualifiedName + (is_array || is_null_term ? "[]" : String.Empty);
|
|
}
|
|
}
|
|
|
|
public string DefaultValue {
|
|
get {
|
|
if (default_value != null && default_value.Length > 0)
|
|
return default_value;
|
|
if (IGen == null)
|
|
return String.Empty;
|
|
return IGen.DefaultValue;
|
|
}
|
|
}
|
|
|
|
string ElementType {
|
|
get {
|
|
if (element_ctype.Length > 0)
|
|
return SymbolTable.Table.GetCSType (element_ctype);
|
|
|
|
return String.Empty;
|
|
}
|
|
}
|
|
|
|
IGeneratable igen;
|
|
public IGeneratable IGen {
|
|
get {
|
|
if (igen == null)
|
|
igen = SymbolTable.Table [CType];
|
|
return igen;
|
|
}
|
|
}
|
|
|
|
public bool IsVoid {
|
|
get {
|
|
return CSType == "void";
|
|
}
|
|
}
|
|
|
|
public string MarshalType {
|
|
get {
|
|
if (IGen == null)
|
|
return String.Empty;
|
|
else if (is_array || is_null_term)
|
|
return "IntPtr";
|
|
return IGen.MarshalType;
|
|
}
|
|
}
|
|
|
|
public string ToNativeType {
|
|
get {
|
|
if (IGen == null)
|
|
return String.Empty;
|
|
return IGen.MarshalType + (is_array || is_null_term ? "[]" : String.Empty);
|
|
}
|
|
}
|
|
|
|
public string FromNative (string var)
|
|
{
|
|
if (IGen == null)
|
|
return String.Empty;
|
|
|
|
if (ElementType != String.Empty) {
|
|
string args = (owned ? "true" : "false") + ", " + (elements_owned ? "true" : "false");
|
|
if (IGen.QualifiedName == "GLib.PtrArray")
|
|
return String.Format ("({0}[]) GLib.Marshaller.PtrArrayToArray ({1}, {2}, typeof({0}))", ElementType, var, args);
|
|
else
|
|
return String.Format ("({0}[]) GLib.Marshaller.ListPtrToArray ({1}, typeof({2}), {3}, typeof({4}))", ElementType, var, IGen.QualifiedName, args, element_ctype == "gfilename*" ? "GLib.ListBase.FilenameString" : ElementType);
|
|
} else if (IGen is IOwnable)
|
|
return ((IOwnable)IGen).FromNative (var, owned);
|
|
else if (is_null_term)
|
|
return String.Format ("GLib.Marshaller.NullTermPtrToStringArray ({0}, {1})", var, owned ? "true" : "false");
|
|
else if (is_array)
|
|
return String.Format ("({0}) GLib.Marshaller.ArrayPtrToArray ({1}, typeof ({2}), (int){3}native_{4}, true)", CSType, var, IGen.QualifiedName, CountParameter.CSType == "int" ? String.Empty : "(" + CountParameter.CSType + ")", CountParameter.Name);
|
|
else
|
|
return IGen.FromNative (var);
|
|
}
|
|
|
|
public string ToNative (string var)
|
|
{
|
|
if (IGen == null)
|
|
return String.Empty;
|
|
|
|
if (ElementType.Length > 0) {
|
|
string args = ", typeof (" + ElementType + "), " + (owned ? "true" : "false") + ", " + (elements_owned ? "true" : "false");
|
|
var = "new " + IGen.QualifiedName + "(" + var + args + ")";
|
|
} else if (is_null_term)
|
|
return String.Format ("GLib.Marshaller.StringArrayToNullTermPointer ({0})", var);
|
|
else if (is_array)
|
|
return String.Format ("GLib.Marshaller.ArrayToArrayPtr ({0})", var);
|
|
|
|
if (IGen is IManualMarshaler)
|
|
return (IGen as IManualMarshaler).AllocNative (var);
|
|
else if (IGen is ObjectGen && owned)
|
|
return var + " == null ? IntPtr.Zero : " + var + ".OwnedHandle";
|
|
else if (IGen is OpaqueGen && owned)
|
|
return var + " == null ? IntPtr.Zero : " + var + ".OwnedCopy";
|
|
else
|
|
return IGen.CallByName (var);
|
|
}
|
|
|
|
public bool Validate (LogWriter log)
|
|
{
|
|
if (MarshalType == "" || CSType == "") {
|
|
log.Warn ("Unknown return type: {0}", CType);
|
|
return false;
|
|
} else if ((CSType == "GLib.List" || CSType == "GLib.SList") && String.IsNullOrEmpty (ElementType))
|
|
log.Warn ("Returns {0} with unknown element type. Add element_type attribute with gapi-fixup.", CType);
|
|
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|