[glib/gvariant] add gvariant format string documentation
- From: Ryan Lortie <ryanl src gnome org>
- To: svn-commits-list gnome org
- Cc:
- Subject: [glib/gvariant] add gvariant format string documentation
- Date: Thu, 14 Jan 2010 20:52:47 +0000 (UTC)
commit 20314907f230d3fcb2d5a71dc63121979fe61dae
Author: Ryan Lortie <desrt desrt ca>
Date: Thu Jan 14 15:52:25 2010 -0500
add gvariant format string documentation
docs/reference/glib/glib-docs.sgml | 2 +
docs/reference/glib/gvariant-varargs.xml | 1025 ++++++++++++++++++++++++++++++
2 files changed, 1027 insertions(+), 0 deletions(-)
---
diff --git a/docs/reference/glib/glib-docs.sgml b/docs/reference/glib/glib-docs.sgml
index 2e84b9e..83496f5 100644
--- a/docs/reference/glib/glib-docs.sgml
+++ b/docs/reference/glib/glib-docs.sgml
@@ -82,6 +82,7 @@
<!ENTITY glib-gettextize SYSTEM "glib-gettextize.xml">
<!ENTITY gtester SYSTEM "gtester.xml">
<!ENTITY gtester-report SYSTEM "gtester-report.xml">
+<!ENTITY gvariant-varargs SYSTEM "gvariant-varargs.xml">
<!ENTITY version SYSTEM "version.xml">
]>
@@ -203,6 +204,7 @@ synchronize their operation.
&glib-Memory-Allocators;
&glib-GVariantType;
&glib-GVariant;
+ &gvariant-varargs;
</chapter>
<chapter id="tools">
diff --git a/docs/reference/glib/gvariant-varargs.xml b/docs/reference/glib/gvariant-varargs.xml
new file mode 100644
index 0000000..f4217e1
--- /dev/null
+++ b/docs/reference/glib/gvariant-varargs.xml
@@ -0,0 +1,1025 @@
+<?xml version='1.0' encoding='utf-8'?>
+
+<refentry id='gvariant-varargs'>
+ <refentrytitle>GVariant Format Strings</refentrytitle>
+
+ <refsect1>
+ <title>Variable Argument Conversions</title>
+
+ <para>
+ This page attempts to document how to perform variable argument conversions with GVariant.
+ </para>
+ <para>
+ A conversion string is a two-way mapping between a single <link linkend='GVariant'>GVariant</link> value and one or
+ more C values.
+ </para>
+ <para>
+ A conversion from C values into a <link linkend='GVariant'>GVariant</link> value is made using the
+ <link linkend='g-variant-new'><function>g_variant_new()</function></link> function. A conversion from a
+ <link linkend='GVariant'>GVariant</link> into C values is made using the
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> function.
+ </para>
+ </refsect1>
+
+ <refsect1>
+ <title>Syntax</title>
+
+ <para>
+ This section exhaustively describes all possibilities for GVariant format strings. There are no valid forms of
+ format strings other than those described here. Please note that the format string syntax is likely to expand in the
+ future.
+ </para>
+ <para>
+ Valid format strings have one of the following forms:
+ </para>
+ <itemizedlist>
+ <listitem>
+ <para>any type string</para>
+ </listitem>
+ <listitem>
+ <para>
+ a type string prefixed with a '<literal>@</literal>'
+ </para>
+ </listitem>
+ <listitem>
+ <para>
+ a type string of a concrete, fixed-sized type, prefixed with a '<literal>&</literal>'
+ </para>
+ </listitem>
+ <listitem>
+ <para>
+ a type string of a concrete, fixed-sized type, prefixed with a '<literal>&a</literal>'
+ </para>
+ </listitem>
+ <listitem>
+ <para>
+ '<literal>&s</literal>' '<literal>&o</literal>', '<literal>&g</literal>', '<literal>^as</literal>',
+ '<literal>^ao</literal>', '<literal>^ag</literal>', '<literal>^a&s</literal>', '<literal>^a&o</literal>' or
+ '<literal>^a&g</literal>'
+ </para>
+ </listitem>
+ <listitem>
+ <para>
+ any conversion string, prefixed with an '<literal>m</literal>'
+ </para>
+ </listitem>
+ <listitem>
+ <para>
+ a sequence of zero or more format strings strings, concatenated and enclosed in parentheses
+ </para>
+ </listitem>
+ <listitem>
+ <para>
+ an opening brace, followed by two format strings, followed by a closing brace (subject to the constraint that the
+ first format string correspond to a type valid for use as the key type of a dictionary)
+ </para>
+ </listitem>
+ </itemizedlist>
+ </refsect1>
+ <refsect1>
+ <title>Symbols</title>
+
+ <para>
+ The following table describes the rough meaning of symbols that may appear inside a GVariant format string. Each
+ symbol is described in detail in its own section, including usage examples.
+ </para>
+
+ <informaltable>
+ <tgroup cols='2'>
+ <colspec colname='col_0'/>
+ <colspec colname='col_1'/>
+ <tbody>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>Symbol</emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>Meaning</emphasis>
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>b</literal>, <literal>y</literal>, <literal>n</literal>, <literal>q</literal>, <literal>i</literal>,
+ <literal>u</literal>, <literal>x</literal>, <literal>t</literal>, <literal>h</literal>, <literal>d</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used for building or deconstructing boolean, byte and numeric types. See
+ <link linkend='gvariant-format-strings-numeric-types'>Numeric Types</link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>s</literal>, <literal>o</literal>, <literal>g</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used for building or deconstructing string types. See
+ <link linkend='gvariant-format-strings-strings'>Strings</link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'><literal>v</literal></emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used for building or deconstructing variant types. See
+ <link linkend='gvariant-format-strings-variant'>Variants</link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>a</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used for building or deconstructing arrays. See
+ <link linkend='gvariant-format-strings-arrays'>Arrays</link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>m</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used for building or deconstructing maybe types. See
+ <link linkend='gvariant-format-strings-nullable-types'>Nullable Types</link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>()</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used for building or deconstructing tuples. See
+ <link linkend='gvariant-format-strings-tuples'>Tuples</link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>{}</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used for building or deconstructing dictionary entries. See
+ <link linkend='gvariant-format-strings-dictionaries'>Dictionaries</link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>@</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used as a prefix on a GVariant type string (not format string). Denotes that a pointer to a
+ <link linkend='GVariant'>GVariant</link> should be used in place of the normal C type or types. For
+ <link linkend='g-variant-new'><function>g_variant_new()</function></link> this means that you must pass a
+ non-<link linkend='NULL--CAPS'><literal>NULL</literal></link> <code>(<link linkend='GVariant'>GVariant</link>
+ *)</code>. For <link linkend='g-variant-get'><function>g_variant_get()</function></link> this means that you
+ must pass a pointer to a <code>(<link linkend='GVariant'>GVariant</link> *)</code> for the value to be returned
+ by reference or <link linkend='NULL--CAPS'><literal>NULL</literal></link> to ignore the value. See
+ <link linkend='gvariant-format-strings-gvariant'><code>GVariant *</code></link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>*</literal>, <literal>?</literal>, <literal>r</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Exactly equivalent to <literal>@*</literal>, <literal>@?</literal> and <literal>@r</literal>. Provided only for
+ completeness so that all GVariant type strings can be used also as format strings. See <link
+ linkend='gvariant-format-strings-gvariant'><code>GVariant *</code></link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'><literal>&</literal></emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used as a prefix on a GVariant type string (not format string). Denotes that a C pointer to serialised data
+ should be used in place of the normal C type or types. See
+ <link linkend='gvariant-format-strings-pointers'>Pointers</link> below.
+ </para>
+ </entry>
+ </row>
+
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'><literal>^</literal></emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ Used as a prefix on some specific types of format strings. See
+ <link linkend='gvariant-format-strings-convenience'>Convenience Conversions</link> below.
+ </para>
+ </entry>
+ </row>
+ </tbody>
+ </tgroup>
+ </informaltable>
+
+
+ <refsect2 id='gvariant-format-strings-numeric-types'>
+ <title>Numeric Types</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <literal>b</literal>, <literal>y</literal>, <literal>n</literal>, <literal>q</literal>,
+ <literal>i</literal>, <literal>u</literal>, <literal>x</literal>, <literal>t</literal>, <literal>h</literal>,
+ <literal>d</literal>
+ </emphasis>
+ </para>
+
+ <para>
+ Variable argument conversions from numeric types work in the most obvious way possible. Upon encountering one of
+ these characters, <link linkend='g-variant-new'><function>g_variant_new()</function></link> takes the equivalent C
+ type as an argument. <link linkend='g-variant-get'><function>g_variant_get()</function></link> takes a pointer to
+ the equivalent C type (or <link linkend='NULL--CAPS'><literal>NULL</literal></link> to ignore the value).
+ </para>
+
+ <para>
+ The equivalent C types are as follows:
+ </para>
+
+ <informaltable>
+ <tgroup cols='2'>
+ <colspec colname='col_0'/><colspec colname='col_1'/>
+ <tbody>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>Character</emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>Equivalent C type</emphasis>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>b</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='gboolean'><type>gboolean</type></link>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>y</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='guchar'><type>guchar</type></link>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>n</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='gint16'><type>gint16</type></link>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>q</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='guint16'><type>guint16</type></link>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>i</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='gint32'><type>gint32</type></link>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>u</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='guint32'><type>guint32</type></link>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>x</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='gint64'><type>gint64</type></link>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>t</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='guint64'><type>guint64</type></link>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>h</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='gint32'><type>gint32</type></link>
+ </para>
+ </entry>
+ </row>
+ <row rowsep='1'>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <emphasis role='strong'>
+ <literal>d</literal>
+ </emphasis>
+ </para>
+ </entry>
+ <entry colsep='1' rowsep='1'>
+ <para>
+ <link linkend='gdouble'><type>gdouble</type></link>
+ </para>
+ </entry>
+ </row>
+ </tbody>
+ </tgroup>
+ </informaltable>
+
+ <para>
+ Note that in C, small integer types in variable argument lists are promoted up to <link
+ linkend='gint'><type>int</type></link> or <link linkend='guint'><type>unsigned int</type></link> as appropriate, and
+ read back accordingly. <link linkend='gint'><type>int</type></link> is 32 bits on every platform on which GLib is
+ currently suported. This means that you can use C expressions of type <link linkend='gint'><type>int</type></link>
+ with <link linkend='g-variant-new'><function>g_variant_new()</function></link> and format characters
+ '<literal>b</literal>', '<literal>y</literal>', '<literal>n</literal>', '<literal>q</literal>',
+ '<literal>i</literal>', '<literal>u</literal>' and '<literal>h</literal>'. Specifically, you can use integer
+ literals with these characters.
+ </para>
+
+ <para>
+ When using the '<literal>x</literal>' and '<literal>t</literal>' characters, you must ensure that the value that you
+ provide is 64 bit. This means that you should use a cast or make use of the
+ <link linkend='G-GINT64-CONSTANT--CAPS'><literal>G_GINT64_CONSTANT</literal></link> or
+ <link linkend='G-GUINT64-CONSTANT--CAPS'><literal>G_GUINT64_CONSTANT</literal></link> macros.
+ </para>
+
+ <para>
+ No type promotion occurs when using <link linkend='g-variant-get'><function>g_variant_get()</function></link> since
+ it operates with pointers. The pointers must always point to a memory region of exactly the correct size.
+ </para>
+
+ <refsect3>
+ <title>Examples</title>
+ <informalexample><programlisting>
+<![CDATA[GVariant *value1, *value2, *value3, *value4;
+
+value1 = g_variant_new ("y", 200);
+value2 = g_variant_new ("b", TRUE);
+value3 = g_variant_new ("d", 37.5):
+value4 = g_variant_new ("x", G_GINT64_CONSTANT (998877665544332211));
+
+{
+ gdouble floating;
+ gboolean truth;
+ gint64 bignum;
+
+
+ g_variant_get (value1, "y", NULL); /* ignore the value. */
+ g_variant_get (value2, "b", &truth);
+ g_variant_get (value3, "d", &floating);
+ g_variant_get (value4, "x", &bignum);
+}]]></programlisting></informalexample>
+ </refsect3>
+ </refsect2>
+
+ <refsect2 id='gvariant-format-strings-strings'>
+ <title>Strings</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <literal>s</literal>, <literal>o</literal>, <literal>g</literal>
+ </emphasis>
+ </para>
+
+ <para>
+ String conversions occur to and from standard nul-terminated C strings. Upon encountering an
+ '<literal>s</literal>', '<literal>o</literal>' or '<literal>g</literal>' in a format string,
+ <link linkend='g-variant-new'><function>g_variant_new()</function></link> takes a <code>(const
+ <link linkend='gchar'>gchar</link> *)</code> and makes a copy of it.
+ <link linkend='NULL--CAPS'><literal>NULL</literal></link> is not a valid string. If the '<literal>o</literal>' or
+ '<literal>g</literal>' characters are used, care must be taken to ensure that the passed string is a valid DBus
+ object path or DBus type signature, respectively.
+ </para>
+ <para>
+ Upon encounting '<literal>s</literal>', '<literal>o</literal>' or '<literal>g</literal>', <link
+ linkend='g-variant-get'><function>g_variant_get()</function></link> takes a pointer to a
+ <code>(<link linkend='gchar'>gchar</link> *)</code> (ie: <code>(<link linkend='gchar'>gchar</link> **)</code>) and
+ sets it to a newly-allocated copy of the string. It is appropriate to free this copy using
+ <link linkend='g-free'><function>g_free()</function></link>.
+ <link linkend='NULL--CAPS'><literal>NULL</literal></link> may also be passed to indicate that the value of the
+ string should be ignored (in which case no copy is made).
+ </para>
+
+ <refsect3>
+ <title>Examples</title>
+ <informalexample><programlisting>
+<![CDATA[GVariant *value1, *value2, *value3;
+
+value1 = g_variant_new ("s", "hello world!");
+value2 = g_variant_new ("o", "/must/be/a/valid/path");
+value3 = g_variant_new ("g", "iias");
+
+#if 0
+ g_variant_new ("s", NULL); /* not valid: NULL is not a string. */
+#endif
+
+{
+ gchar *result;
+
+ g_variant_get (value1, "s", &result);
+ g_print ("It was '%s'\n", result);
+ g_free (result);
+}]]></programlisting></informalexample>
+ </refsect3>
+ </refsect2>
+
+ <refsect2 id='gvariant-format-strings-variants'>
+ <title>Variants</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <literal>v</literal>
+ </emphasis>
+ </para>
+
+ <para>
+ Upon encountering a '<literal>v</literal>',
+ <link linkend='g-variant-new'><function>g_variant_new()</function></link> takes a <code>(<link
+ linkend='GVariant'>GVariant</link> *)</code>. The value of the
+ <link linkend='GVariant'><type>GVariant</type></link> is used as the contents of the variant value.
+ </para>
+ <para>
+ Upon encountering a '<literal>v</literal>', <link
+ linkend='g-variant-get'><function>g_variant_get()</function></link> takes a pointer to a
+ <code>(<link linkend='GVariant'>GVariant</link> *)</code> (ie: <code>(<link linkend='GVariant'>GVariant</link> **)
+ </code>). It is set to a new reference to a <link linkend='GVariant'><type>GVariant</type></link> instance
+ containing the contents of the variant value. It is appropriate to free this reference using
+ <link linkend='g-variant-unref'><function>g_variant_unref()</function></link>.
+ <link linkend='NULL--CAPS'><literal>NULL</literal></link> may also be passed to indicate that the value should be
+ ignored (in which case no new reference is created).
+ </para>
+
+ <refsect3>
+ <title>Examples</title>
+ <informalexample><programlisting>
+<![CDATA[GVariant *x, *y;
+
+/* the following two lines are equivalent: */
+x = g_variant_new ("v", y);
+x = g_variant_new_variant (y);
+
+/* as are these: */
+g_variant_get (x, "v", &y);
+y = g_variant_get_variant (x);]]></programlisting></informalexample>
+ </refsect3>
+ </refsect2>
+
+
+ <refsect2 id='gvariant-format-strings-arrays'>
+ <title>Arrays</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <literal>a</literal>
+ </emphasis>
+ </para>
+
+ <para>
+ Upon encountering an '<literal>a</literal>' character followed by a type string,
+ <link linkend='g-variant-new'><function>g_variant_new()</function></link> will take a
+ <code>(<link linkend='GVariantBuilder'>GVariantBuilder</link> *)</code> that has been created as an array builder
+ for an array of the type given in the type string. The builder will have
+ <link linkend='g-variant-builder-end'><function>g_variant_builder_end()</function></link> called on it and the
+ result will be used as the value. As a special exception, if the given type string is a definite type, then
+ <link linkend='NULL--CAPS'><literal>NULL</literal></link> may be given to mean an empty array of that type.
+ </para>
+
+ <para>
+ Upon encountering an '<literal>a</literal>' character followed by a type string,
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> will take a pointer to a
+ <link linkend='GVariantIter'><type>GVariantIter</type></link> (ie:
+ <code>(<link linkend='GVariantIter'>GVariantIter</link> *)</code>). The iter (which is assumed to be completely
+ uninitialised) is initialised for iterating over the elements of the array.
+ <link linkend='NULL--CAPS'><literal>NULL</literal></link> may also be given to indicate that the value of the array
+ should be ignored.
+ </para>
+
+ <refsect3>
+ <title>Examples</title>
+ <informalexample><programlisting>
+<![CDATA[GVariantBuilder *builder;
+GVariant *value;
+
+builder = g_variant_builder_new (G_VARIANT_TYPE_CLASS_ARRAY, NULL);
+g_variant_builder_add (builder, "s", "when");
+g_variant_builder_add (builder, "s", "in");
+g_variant_builder_add (builder, "s", "the");
+g_variant_builder_add (builder, "s", "course");
+value = g_variant_new ("as", builder);
+
+{
+ GVariantIter iter;
+ gchar *str;
+
+ g_variant_get (value, "as", &iter);
+ while (g_variant_iter_next (&iter, "s", &str))
+ g_print ("%s\n");
+}]]></programlisting></informalexample>
+ </refsect3>
+ </refsect2>
+
+ <refsect2 id='gvariant-format-strings-nullable-types'>
+ <title>Nullable Types</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <literal>m</literal>
+ </emphasis>
+ </para>
+ <para>
+ Nullable types are handled in two separate ways depending on the format string that follows the
+ '<literal>m</literal>'. The method that is used always depends entirely on the character immediately following the
+ '<literal>m</literal>'.
+ </para>
+
+ <para>
+ The first way is used with format strings starting with '<literal>s</literal>', '<literal>o</literal>',
+ '<literal>g</literal>', '<literal>v</literal>', '<literal>@</literal>', '<literal>*</literal>',
+ '<literal>?</literal>', '<literal>r</literal>', '<literal>&</literal>', or '<literal>^</literal>'. In all of
+ these cases, for non-nullable types, <link linkend='g-variant-new'><function>g_variant_new()</function></link> takes
+ a pointer to a non-<link linkend='NULL--CAPS'><literal>NULL</literal></link> value and
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> returns (by reference) a
+ non-<link linkend='NULL--CAPS'><literal>NULL</literal></link> pointer. When any of these format strings are
+ prefixed with an '<literal>m</literal>', the type of arguments that are collected does not change in any way, but
+ <link linkend='NULL--CAPS'><literal>NULL</literal></link> becomes a permissable value. If the format string
+ normally involves taking of other arguments (such as an integer for the length of an array) then those values are
+ still taken as normal, but in the <link linkend='NULL--CAPS'><literal>NULL</literal></link> case, they are ignored.
+ </para>
+ <para>
+ The second way is used with all other format strings. For both
+ <link linkend='g-variant-new'><function>g_variant_new()</function></link> and
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> an additional
+ <code>(<link linkend='gboolean'>gboolean</link> *)</code> argument is collected. If
+ <link linkend='NULL--CAPS'><literal>NULL</literal></link> is given then no further arguments will be collected for the
+ format string. If a non-<link linkend='NULL--CAPS'><literal>NULL</literal></link> value is given then the arguments
+ that are normally collected for the equivalent non-nullable type will be collected following the
+ <code>(<link linkend='gboolean'>gboolean</link> *)</code>.
+ </para>
+ <para>
+ If <link linkend='NULL--CAPS'><literal>NULL</literal></link>, or a pointer to
+ <link linkend='FALSE--CAPS'><literal>FALSE</literal></link> is given to
+ <link linkend='g-variant-new'><function>g_variant_new()</function></link> then the null value is constructed. In
+ the case that the arguments were collected, they will be ignored. Otherwise (if a pointer to
+ <link linkend='TRUE--CAPS'><literal>TRUE</literal></link> was given), the arguments are used in the normal way to
+ create the non-null value.
+ </para>
+ <para>
+ If <link linkend='NULL--CAPS'><literal>NULL</literal></link> is given to
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> then the value is ignored. If a
+ non-<link linkend='NULL--CAPS'><literal>NULL</literal></link> pointer is given then it is used to return by reference
+ whether the value was non-null. In the case that the value was non-null, the
+ <link linkend='gboolean'><type>gboolean</type></link> will be set to
+ <link linkend='TRUE--CAPS'><literal>TRUE</literal></link> and the value will be stored in the arguments in the usual
+ way. In the case that the value was null, the <link linkend='gboolean'><type>gboolean</type></link> will be set to
+ <link linkend='FALSE--CAPS'><literal>FALSE</literal></link> and the arguments will be collected in the normal way,
+ but they values that they point to will be ignored.
+ </para>
+ <para>
+ The macro <link linkend='G-VARIANT-JUST--CAPS'><literal>G_VARIANT_JUST</literal></link> is provided. It evaluates
+ to a pointer to a <link linkend='gboolean'><type>gboolean</type></link> containing
+ <link linkend='TRUE--CAPS'><literal>TRUE</literal></link>.
+ </para>
+
+ <refsect3>
+ <title>Examples</title>
+ <informalexample><programlisting>
+<![CDATA[GVariant *value1, *value2, *value3, *value4, *value5, *value6;
+gboolean f = FALSE;
+
+value1 = g_variant_new ("ms", "Hello world");
+value2 = g_variant_new ("ms", NULL);
+value3 = g_variant_new ("(m(ii)s)", G_VARIANT_JUST, 123, 456, "Done");
+value4 = g_variant_new ("(m(ii)s)", &f, -1, -1, "Done"); /* both '-1' are ignored. */
+value5 = g_variant_new ("(m(ii)s)", NULL, "Done");
+
+{
+ GVariant *contents;
+ const gchar *cstr;
+ gboolean just;
+ gint32 x, y;
+ gchar *str;
+
+ g_variant_get (value1, "ms", &str);
+ if (str != NULL)
+ g_print ("str: %s\n", str);
+ else
+ g_print ("it was null\n");
+ g_free (str);
+
+
+ g_variant_get (value2, "m&s", &cstr);
+ if (cstr != NULL)
+ g_print ("str: %s\n", cstr);
+ else
+ g_print ("it was null\n");
+ /* don't free 'cstr' */
+
+
+ /* NULL passed for the gboolean *, so 'gint32 *' not collected */
+ g_variant_get (value3, "(m(ii)s)", NULL, &str); /* only interested in the string. */
+ g_print ("string is %s\n", str);
+ g_free (str);
+
+
+ /* non-NULL passed for gboolean *, so 2 'gint32 *' are collected */
+ g_variant_get (value4, "(m(ii)&s)", &just, &x, &y, &cstr);
+ if (just)
+ g_print ("it was (%d, %d)\n", x, y);
+ else
+ g_print ("it was null\n");
+ g_print ("string is %s\n", cstr);
+ /* don't free 'cstr' */
+
+
+ g_variant_get (value5, "(m*s)", &contents, NULL); /* ignore the string. */
+ if (contents != NULL)
+ {
+ g_variant_get (contents, "(ii)", &x, &y);
+ g_print ("it was (%d, %d)\n", x, y);
+ g_variant_unref (contents);
+ }
+ else
+ g_print ("it was null\n");
+}]]></programlisting></informalexample>
+ </refsect3>
+ </refsect2>
+
+ <refsect2 id='gvariant-format-strings-tuples'>
+ <title>Tuples</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <code>()</code>
+ </emphasis>
+ </para>
+
+ <para>
+ Tuples are handled by handling each item in the tuple, in sequence. Each item is handled in the usual way.
+ </para>
+
+ <refsect3>
+ <title>Examples</title>
+ <informalexample><programlisting>
+<![CDATA[GVariant *value1, *value2;
+
+value1 = g_variant_new ("(s(ii))", "Hello", 55, 77);
+value2 = g_variant_new ("()");
+
+{
+ gchar *string;
+ gint x, y;
+
+ g_variant_get (value1, "(s(ii))", &string, &x, &y);
+ g_print ("%s, %d, %d\n", string, x, y);
+ g_free (string);
+
+ g_variant_get (value2, "()"); /* do nothing... */
+}]]></programlisting></informalexample>
+ </refsect3>
+ </refsect2>
+
+ <refsect2 id='gvariant-format-strings-dictionaries'>
+ <title>Dictionaries</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <code>{}</code>
+ </emphasis>
+ </para>
+
+ <para>
+ Dictionary entries are handled by handling first the key, then the value. Each is handled in the usual way.
+ </para>
+
+ <refsect3>
+ <title>Examples</title>
+ <informalexample><programlisting>
+<![CDATA[GVariantBuilder *b;
+GVariant *dict;
+
+b = g_variant_builder_new (G_VARIANT_TYPE_CLASS_ARRAY, G_VARIANT_TYPE ("a{sv}"));
+g_variant_builder_add (b, "{sv}", "name", g_variant_new_string ("foo"));
+g_variant_builder_add (b, "{sv}", "timeout", g_variant_new_int32 (10));
+dict = g_variant_builder_end (b);]]></programlisting></informalexample>
+ </refsect3>
+ </refsect2>
+
+ <refsect2 id='gvariant-format-strings-gvariant'>
+ <title>GVariant *</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <literal>@</literal>, <literal>*</literal>, <literal>?</literal>, <literal>r</literal>
+ </emphasis>
+
+ </para>
+ <para>
+ Upon encountering a '<literal>@</literal>' in front of a type string,
+ <link linkend='g-variant-new'><function>g_variant_new()</function></link> takes a
+ non-<link linkend='NULL--CAPS'><literal>NULL</literal></link> pointer to a
+ <link linkend='GVariant'><type>GVariant</type></link> and uses its value directly instead of collecting arguments to
+ create the value. The provided <link linkend='GVariant'><type>GVariant</type></link> must have a type that matches the
+ type string following the '<literal>@</literal>'. '<literal>*</literal>' is
+ the same as '<literal>@*</literal>' (ie: take a <link linkend='GVariant'><type>GVariant</type></link> of any type).
+ '<literal>?</literal>' is the same as '<literal>@?</literal>' (ie: take a
+ <link linkend='GVariant'><type>GVariant</type></link> of any basic type). '<literal>r</literal>' is the same as
+ '<literal>@r</literal>' (ie: take a <link linkend='GVariant'><type>GVariant</type></link> of any tuple type).
+ </para>
+ <para>
+ Upon encountering a '<literal>@</literal>' in front of a type string,
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link>
+ takes a pointer to a <code>(<link linkend='GVariant'>GVariant</link> *)</code> (ie: a
+ <code>(<link linkend='GVariant'>GVariant</link> **)</code>) and sets it to a new reference to a
+ <link linkend='GVariant'><type>GVariant</type></link> containing the value (instead of deconstructing the value into
+ C types in the usual way). <link linkend='NULL--CAPS'><literal>NULL</literal></link> can be given to ignore the
+ value. '<literal>*</literal>', '<literal>?</literal>' and '<literal>r</literal>' are handled in a way analogous to
+ what is stated above.
+ </para>
+ <para>
+ You can always use '<literal>*</literal>' as an alternative to '<literal>?</literal>', '<literal>r</literal>' or any
+ use of '<literal>@</literal>'. Using the other characters where possible is recommended, however, due to the
+ improvements in type safety and code self-documentation.
+ </para>
+
+ <refsect3>
+ <title>Examples</title>
+ <informalexample><programlisting>
+<![CDATA[GVariant *value1, *value2;
+
+value1 = g_variant_new ("(i ii)", 44, g_variant_new_int32 (55), 66);
+
+/* note: consumes floating reference count on 'value1' */
+value2 = g_variant_new ("(@(iii)*)", value1, g_variant_new_string ("foo"));
+
+{
+ const gchar *string;
+ GVariant *tmp;
+ gsize length;
+ gint x, y, z;
+
+ g_variant_get (value2, "((iii)*)", &x, &y, &z, &tmp);
+ string = g_variant_get_string (tmp, &length);
+ g_print ("it is %d %d %d %s (length=%d)\n", x, y, z, string, (int) length);
+ g_variant_unref (tmp);
+
+ /* quick way to skip all the values in a tuple */
+ g_variant_get (value2, "(rs)", NULL, &string); /* or "(@(iii)s)" */
+ g_print ("i only got the string: %s\n", string);
+ g_free (string);
+}]]></programlisting></informalexample>
+ </refsect3>
+ </refsect2>
+
+ <refsect2 id='gvariant-format-strings-pointers'>
+ <title>Pointers</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <code>&</code>
+ </emphasis>
+ </para>
+
+ <para>
+ The '<code>&</code>' character has 3 closely-related uses. In all cases, when used with
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> it means that a pointer directly to the
+ serialised data stored inside the GVariant is returned (either directly or indirectly). This pointer is valid only
+ for as long as the GVariant exists.
+ </para>
+ <para>
+ The three cases are as follows:
+ </para>
+ <para>
+ The simplest case is when it is applied to a string (ie: '<literal>&s</literal>', '<literal>&o</literal>' or
+ '<code>&g</code>'). For <link linkend='g-variant-new'><function>g_variant_new()</function></link> this has
+ absolutely no effect. The string is collected and duplicated normally. For
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> it means that instead of creating a newly
+ allocated copy of the string, a pointer to the serialised data is returned. This pointer should not be freed.
+ Validity checks are performed to ensure that the string data will always be properly nul-terminated.
+ </para>
+ <para>
+ The next case is when '<literal>&</literal>' is used to prefix a type string corresponding to a fixed-size type
+ (ie: a type whose values are always the same size). This includes booleans, bytes, integers, doubles and tuples or
+ dictionary entries consisting entirely of those types (recursively). Arrays, nullable types, strings, variants and
+ tuples or dictionaries containing those types are not fixed-size types. In this case, instead of collecting the
+ arguments in the usual way, a single pointer is collected.
+ <link linkend='g-variant-new'><function>g_variant_new()</function></link> collects a pointer to the data as it would
+ appear in serialised form and makes a copy.
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> collects a pointer to a
+ <link linkend='gpointer'><type>gpointer</type></link> and sets it to point to the serialised data. In the
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> case,
+ <link linkend='NULL--CAPS'><literal>NULL</literal></link> may be given to ignore the value.
+ </para>
+ <para>
+ The last case is when '<literal>&a</literal>' is used to prefix a type string corresponding to a fixed-size type
+ (ie: an array in which each item has a fixed size). In this case, in addition to collecting the pointer to the data
+ itself, <link linkend='g-variant-new'><function>g_variant_new()</function></link> collects a
+ <link linkend='gsize'><type>gsize</type></link> and
+ <link linkend='g-variant-get'><function>g_variant_get()</function></link> collects a
+ <code>(<link linkend='gsize'>gsize</link> *)</code>. These are the length of the array; not its size in bytes.
+ </para>
+ <para>
+ <emphasis role='string'>Note:</emphasis> even though it will work on 32bit platforms, an
+ <link linkend='gint'><type>int</type></link> is not normally the same thing as a
+ <link linkend='gsize'><type>gsize</type></link>. This means that passing an
+ <link linkend='gint'><type>int</type></link> (including integer literals) as the size of an array is not correct and
+ will cause problems on 64bit platforms. Always cast to <link linkend='gsize'><type>gsize</type></link> first, or
+ better yet, use <link linkend='G-N-ELEMENTS--CAPS'><function>G_N_ELEMENTS()</function></link> (which returns
+ <link linkend='gsize'><type>gsize</type></link>).
+ </para>
+
+ <refsect3>
+ <title>Examples</title>
+ <informalexample><programlisting>
+<![CDATA[struct {
+ gint32 x, y;
+} struc = {
+ 11, 22
+};
+GVariant *value1, *value2, *value3;
+gint32 array[] = { 33, 44, 55 };
+
+value1 = g_variant_new ("&ai", array, G_N_ELEMENTS (array));
+value2 = g_variant_new_string ("hello world");
+value3 = g_variant_new ("&(ii)", &struc);]]></programlisting></informalexample>
+ </refsect3>
+ </refsect2>
+
+ <refsect2 id='gvariant-format-strings-convenience'>
+ <title>Convenience Conversions</title>
+ <para>
+ <emphasis role='strong'>
+ Characters: <literal>^</literal>
+ </emphasis>
+ </para>
+
+ <para>
+ The '<literal>^</literal>' character currently only has one purpose: to convert to and from
+ <link linkend='G-TYPE-STRV'><literal>G_TYPE_STRV</literal></link> type arrays of strings. It is always used with
+ arrays of strings (or other string types). It has two forms.
+ </para>
+ <itemizedlist>
+ <listitem>
+ <para>
+ '<literal>^as</literal>' (or <literal>o</literal> or <literal>g</literal>)
+ </para>
+ </listitem>
+ <listitem>
+ <para>
+ '<literal>^a&s</literal>' (or <literal>o</literal> or <literal>g</literal>)
+ </para>
+ </listitem>
+ </itemizedlist>
+ <para>
+ When used with <link linkend='g-variant-new'><function>g_variant_new()</function></link> both forms are equivalent.
+ A <code>(const <link linkend='gchar'>gchar</link> * const *)</code> is collected. This must be a pointer to the
+ array of <link linkend='NULL--CAPS'><literal>NULL</literal></link>-terminated pointers to strings. This array is
+ converted to a <link linkend='GVariant'><type>GVariant</type></link> instance. Copies are made, so the original
+ array may be freed immediately.
+ </para>
+ <para>
+ When used with <link linkend='g-variant-get'><function>g_variant_get()</function></link> the two forms have
+ different meaning. Both return a freshly allocated
+ <link linkend='NULL--CAPS'><literal>NULL</literal></link>-terminated array of pointers to strings. In the case of
+ '<literal>^as</literal>', the strings are owned by the caller -- it is appropriate to free the array with
+ <link linkend='g-strfreev'><function>g_strfreev()</function></link>. In the case of '<literal>^a&s</literal>',
+ a shallow copy is made; the strings themselves are embedded in the serialised data and owned by the original
+ <link linkend='GVariant'><type>GVariant</type></link> instance -- it is only appropriate to free the outer array
+ with <link linkend='g-free'><function>g_free()</function></link>.
+ </para>
+ </refsect2>
+ </refsect1>
+</refentry>
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]