Blame gio/strinfo.c

Packit Service d3d246
/*
Packit Service d3d246
 * Copyright © 2010 Codethink Limited
Packit Service d3d246
 *
Packit Service d3d246
 * This library is free software; you can redistribute it and/or
Packit Service d3d246
 * modify it under the terms of the GNU Lesser General Public
Packit Service d3d246
 * License as published by the Free Software Foundation; either
Packit Service d3d246
 * version 2.1 of the License, or (at your option) any later version.
Packit Service d3d246
 *
Packit Service d3d246
 * This library is distributed in the hope that it will be useful,
Packit Service d3d246
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service d3d246
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service d3d246
 * Lesser General Public License for more details.
Packit Service d3d246
 *
Packit Service d3d246
 * You should have received a copy of the GNU Lesser General Public
Packit Service d3d246
 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
Packit Service d3d246
 *
Packit Service d3d246
 * Author: Ryan Lortie <desrt@desrt.ca>
Packit Service d3d246
 */
Packit Service d3d246
Packit Service d3d246
#include <string.h>
Packit Service d3d246
#include <glib.h>
Packit Service d3d246
Packit Service d3d246
/*
Packit Service d3d246
 * The string info map is an efficient data structure designed to be
Packit Service d3d246
 * used with a small set of items.  It is used by GSettings schemas for
Packit Service d3d246
 * three purposes:
Packit Service d3d246
 *
Packit Service d3d246
 *  1) Implement <choices> with a list of valid strings
Packit Service d3d246
 *
Packit Service d3d246
 *  2) Implement <alias> by mapping one string to another
Packit Service d3d246
 *
Packit Service d3d246
 *  3) Implement enumerated types by mapping strings to integer values
Packit Service d3d246
 *     (and back).
Packit Service d3d246
 *
Packit Service d3d246
 * The map is made out of an array of uint32s.  Each entry in the array
Packit Service d3d246
 * is an integer value, followed by a specially formatted string value:
Packit Service d3d246
 *
Packit Service d3d246
 *   The string starts with the byte 0xff or 0xfe, followed by the
Packit Service d3d246
 *   content of the string, followed by a nul byte, followed by
Packit Service d3d246
 *   additional nul bytes for padding, followed by a 0xff byte.
Packit Service d3d246
 *
Packit Service d3d246
 *   Padding is added so that the entire formatted string takes up a
Packit Service d3d246
 *   multiple of 4 bytes, and not less than 8 bytes.  The requirement
Packit Service d3d246
 *   for a string to take up 8 bytes is so that the scanner doesn't lose
Packit Service d3d246
 *   synch and mistake a string for an integer value.
Packit Service d3d246
 *
Packit Service d3d246
 * The first byte of the formatted string depends on if the integer is
Packit Service d3d246
 * an enum value (0xff) or an alias (0xfe).  If it is an alias then the
Packit Service d3d246
 * number refers to the word offset within the info map at which the
Packit Service d3d246
 * integer corresponding to the "target" value is stored.
Packit Service d3d246
 *
Packit Service d3d246
 * For example, consider the case of the string info map representing an
Packit Service d3d246
 * enumerated type of 'foo' (value 1) and 'bar' (value 2) and 'baz'
Packit Service d3d246
 * (alias for 'bar').  Note that string info maps are always little
Packit Service d3d246
 * endian.
Packit Service d3d246
 *
Packit Service d3d246
 * x01 x00 x00 x00   xff 'f' 'o' 'o'   x00 x00 x00 xff   x02 x00 x00 x00
Packit Service d3d246
 * xff 'b' 'a' 'r'   x00 x00 x00 xff   x03 x00 x00 x00   xfe 'b' 'a' 'z'
Packit Service d3d246
 * x00 x00 x00 xff
Packit Service d3d246
 *
Packit Service d3d246
 *
Packit Service d3d246
 * The operations that someone may want to perform with the map:
Packit Service d3d246
 *
Packit Service d3d246
 *   - lookup if a string is valid (and not an alias)
Packit Service d3d246
 *   - lookup the integer value for a enum 'nick'
Packit Service d3d246
 *   - lookup the integer value for the target of an alias
Packit Service d3d246
 *   - lookup an alias and convert it to its target string
Packit Service d3d246
 *   - lookup the enum nick for a given value
Packit Service d3d246
 *
Packit Service d3d246
 * In order to lookup if a string is valid, it is padded on either side
Packit Service d3d246
 * (as described) and scanned for in the array.  For example, you might
Packit Service d3d246
 * look for "foo":
Packit Service d3d246
 *
Packit Service d3d246
 *                   xff 'f' 'o' 'o'   x00 x00 x00 xff
Packit Service d3d246
 *
Packit Service d3d246
 * In order to lookup the integer value for a nick, the string is padded
Packit Service d3d246
 * on either side and scanned for in the array, as above.  Instead of
Packit Service d3d246
 * merely succeeding, we look at the integer value to the left of the
Packit Service d3d246
 * match.  This is the enum value.
Packit Service d3d246
 *
Packit Service d3d246
 * In order to lookup an alias and convert it to its target enum value,
Packit Service d3d246
 * the string is padded on either side (as described, with 0xfe) and
Packit Service d3d246
 * scanned for.  For example, you might look for "baz":
Packit Service d3d246
 *
Packit Service d3d246
 *                   xfe 'b' 'a' 'z'  x00 x00 x00 xff
Packit Service d3d246
 *
Packit Service d3d246
 * The integer immediately preceding the match then contains the offset
Packit Service d3d246
 * of the integer value of the target.  In our example, that's '3'.
Packit Service d3d246
 * This index is dereferenced to find the enum value of '2'.
Packit Service d3d246
 *
Packit Service d3d246
 * To convert the alias to its target string, 5 bytes just need to be
Packit Service d3d246
 * added past the start of the integer value to find the start of the
Packit Service d3d246
 * string.
Packit Service d3d246
 *
Packit Service d3d246
 * To lookup the enum nick for a given value, the value is searched for
Packit Service d3d246
 * in the array.  To ensure that the value isn't matching the inside of a
Packit Service d3d246
 * string, we must check that it is either the first item in the array or
Packit Service d3d246
 * immediately preceded by the byte 0xff.  It must also be immediately
Packit Service d3d246
 * followed by the byte 0xff.
Packit Service d3d246
 *
Packit Service d3d246
 * Because strings always take up a minimum of 2 words, because 0xff or
Packit Service d3d246
 * 0xfe never appear inside of a utf-8 string and because no two integer
Packit Service d3d246
 * values ever appear in sequence, the only way we can have the
Packit Service d3d246
 * sequence:
Packit Service d3d246
 *
Packit Service d3d246
 *     xff __ __ __ __ xff (or 0xfe)
Packit Service d3d246
 *
Packit Service d3d246
 * is in the event of an integer nested between two strings.
Packit Service d3d246
 *
Packit Service d3d246
 * For implementation simplicity/efficiency, strings may not be more
Packit Service d3d246
 * than 65 characters in length (ie: 17 32bit words after padding).
Packit Service d3d246
 *
Packit Service d3d246
 * In the event that we are doing <choices> (ie: not an enum type) then
Packit Service d3d246
 * the value of each choice is set to zero and ignored.
Packit Service d3d246
 */
Packit Service d3d246
Packit Service d3d246
#define STRINFO_MAX_WORDS   17
Packit Service d3d246
G_GNUC_UNUSED static guint
Packit Service d3d246
strinfo_string_to_words (const gchar *string,
Packit Service d3d246
                         guint32     *words,
Packit Service d3d246
                         gboolean     alias)
Packit Service d3d246
{
Packit Service d3d246
  guint n_words;
Packit Service d3d246
  gsize size;
Packit Service d3d246
Packit Service d3d246
  size = strlen (string);
Packit Service d3d246
Packit Service d3d246
  n_words = MAX (2, (size + 6) >> 2);
Packit Service d3d246
Packit Service d3d246
  if (n_words > STRINFO_MAX_WORDS)
Packit Service d3d246
    return FALSE;
Packit Service d3d246
Packit Service d3d246
  words[0] = GUINT32_TO_LE (alias ? 0xfe : 0xff);
Packit Service d3d246
  words[n_words - 1] = GUINT32_TO_BE (0xff);
Packit Service d3d246
  memcpy (((gchar *) words) + 1, string, size + 1);
Packit Service d3d246
Packit Service d3d246
  return n_words;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static gint
Packit Service d3d246
strinfo_scan (const guint32 *strinfo,
Packit Service d3d246
              guint          length,
Packit Service d3d246
              const guint32 *words,
Packit Service d3d246
              guint          n_words)
Packit Service d3d246
{
Packit Service d3d246
  guint i = 0;
Packit Service d3d246
Packit Service d3d246
  if (length < n_words)
Packit Service d3d246
    return -1;
Packit Service d3d246
Packit Service d3d246
  while (i <= length - n_words)
Packit Service d3d246
    {
Packit Service d3d246
      guint j = 0;
Packit Service d3d246
Packit Service d3d246
      for (j = 0; j < n_words; j++)
Packit Service d3d246
        if (strinfo[i + j] != words[j])
Packit Service d3d246
          break;
Packit Service d3d246
Packit Service d3d246
      if (j == n_words)
Packit Service d3d246
        return i;   /* match */
Packit Service d3d246
Packit Service d3d246
      /* skip at least one word, continue */
Packit Service d3d246
      i += j ? j : 1;
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  return -1;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static gint
Packit Service d3d246
strinfo_find_string (const guint32 *strinfo,
Packit Service d3d246
                     guint          length,
Packit Service d3d246
                     const gchar   *string,
Packit Service d3d246
                     gboolean       alias)
Packit Service d3d246
{
Packit Service d3d246
  guint32 words[STRINFO_MAX_WORDS];
Packit Service d3d246
  guint n_words;
Packit Service d3d246
Packit Service d3d246
  if (length == 0)
Packit Service d3d246
    return -1;
Packit Service d3d246
Packit Service d3d246
  n_words = strinfo_string_to_words (string, words, alias);
Packit Service d3d246
Packit Service d3d246
  return strinfo_scan (strinfo + 1, length - 1, words, n_words);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static gint
Packit Service d3d246
strinfo_find_integer (const guint32 *strinfo,
Packit Service d3d246
                      guint          length,
Packit Service d3d246
                      guint32        value)
Packit Service d3d246
{
Packit Service d3d246
  guint i;
Packit Service d3d246
Packit Service d3d246
  for (i = 0; i < length; i++)
Packit Service d3d246
    if (strinfo[i] == GUINT32_TO_LE (value))
Packit Service d3d246
      {
Packit Service d3d246
        const guchar *charinfo = (const guchar *) &strinfo[i];
Packit Service d3d246
Packit Service d3d246
        /* make sure it has 0xff on either side */
Packit Service d3d246
        if ((i == 0 || charinfo[-1] == 0xff) && charinfo[4] == 0xff)
Packit Service d3d246
          return i;
Packit Service d3d246
      }
Packit Service d3d246
Packit Service d3d246
  return -1;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static gboolean
Packit Service d3d246
strinfo_is_string_valid (const guint32 *strinfo,
Packit Service d3d246
                         guint          length,
Packit Service d3d246
                         const gchar   *string)
Packit Service d3d246
{
Packit Service d3d246
  return strinfo_find_string (strinfo, length, string, FALSE) != -1;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static gboolean
Packit Service d3d246
strinfo_enum_from_string (const guint32 *strinfo,
Packit Service d3d246
                          guint          length,
Packit Service d3d246
                          const gchar   *string,
Packit Service d3d246
                          guint         *result)
Packit Service d3d246
{
Packit Service d3d246
  gint index;
Packit Service d3d246
Packit Service d3d246
  index = strinfo_find_string (strinfo, length, string, FALSE);
Packit Service d3d246
Packit Service d3d246
  if (index < 0)
Packit Service d3d246
    return FALSE;
Packit Service d3d246
Packit Service d3d246
  *result = GUINT32_FROM_LE (strinfo[index]);
Packit Service d3d246
  return TRUE;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static const gchar *
Packit Service d3d246
strinfo_string_from_enum (const guint32 *strinfo,
Packit Service d3d246
                          guint          length,
Packit Service d3d246
                          guint          value)
Packit Service d3d246
{
Packit Service d3d246
  gint index;
Packit Service d3d246
Packit Service d3d246
  index = strinfo_find_integer (strinfo, length, value);
Packit Service d3d246
Packit Service d3d246
  if (index < 0)
Packit Service d3d246
    return NULL;
Packit Service d3d246
Packit Service d3d246
  return 1 + (const gchar *) &strinfo[index + 1];
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static const gchar *
Packit Service d3d246
strinfo_string_from_alias (const guint32 *strinfo,
Packit Service d3d246
                           guint          length,
Packit Service d3d246
                           const gchar   *alias)
Packit Service d3d246
{
Packit Service d3d246
  gint index;
Packit Service d3d246
Packit Service d3d246
  index = strinfo_find_string (strinfo, length, alias, TRUE);
Packit Service d3d246
Packit Service d3d246
  if (index < 0)
Packit Service d3d246
    return NULL;
Packit Service d3d246
Packit Service d3d246
  return 1 + (const gchar *) &strinfo[GUINT32_TO_LE (strinfo[index]) + 1];
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static GVariant *
Packit Service d3d246
strinfo_enumerate (const guint32 *strinfo,
Packit Service d3d246
                   guint          length)
Packit Service d3d246
{
Packit Service d3d246
  GVariantBuilder builder;
Packit Service d3d246
  const gchar *ptr, *end;
Packit Service d3d246
Packit Service d3d246
  ptr = (gpointer) strinfo;
Packit Service d3d246
  end = ptr + 4 * length;
Packit Service d3d246
Packit Service d3d246
  ptr += 4;
Packit Service d3d246
Packit Service d3d246
  g_variant_builder_init (&builder, G_VARIANT_TYPE_STRING_ARRAY);
Packit Service d3d246
Packit Service d3d246
  while (ptr < end)
Packit Service d3d246
    {
Packit Service d3d246
      /* don't include aliases */
Packit Service d3d246
      if (*ptr == '\xff')
Packit Service d3d246
        g_variant_builder_add (&builder, "s", ptr + 1);
Packit Service d3d246
Packit Service d3d246
      /* find the end of this string */
Packit Service d3d246
      ptr = memchr (ptr, '\xff', end - ptr);
Packit Service d3d246
      g_assert (ptr != NULL);
Packit Service d3d246
Packit Service d3d246
      /* skip over the int to the next string */
Packit Service d3d246
      ptr += 5;
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  return g_variant_builder_end (&builder);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static void
Packit Service d3d246
strinfo_builder_append_item (GString     *builder,
Packit Service d3d246
                             const gchar *string,
Packit Service d3d246
                             guint        value)
Packit Service d3d246
{
Packit Service d3d246
  guint32 words[STRINFO_MAX_WORDS];
Packit Service d3d246
  guint n_words;
Packit Service d3d246
Packit Service d3d246
  value = GUINT32_TO_LE (value);
Packit Service d3d246
Packit Service d3d246
  n_words = strinfo_string_to_words (string, words, FALSE);
Packit Service d3d246
  g_string_append_len (builder, (void *) &value, sizeof value);
Packit Service d3d246
  g_string_append_len (builder, (void *) words, 4 * n_words);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static gboolean
Packit Service d3d246
strinfo_builder_append_alias (GString     *builder,
Packit Service d3d246
                              const gchar *alias,
Packit Service d3d246
                              const gchar *target)
Packit Service d3d246
{
Packit Service d3d246
  guint32 words[STRINFO_MAX_WORDS];
Packit Service d3d246
  guint n_words;
Packit Service d3d246
  guint value;
Packit Service d3d246
  gint index;
Packit Service d3d246
Packit Service d3d246
  index = strinfo_find_string ((const guint32 *) builder->str,
Packit Service d3d246
                               builder->len / 4, target, FALSE);
Packit Service d3d246
Packit Service d3d246
  if (index == -1)
Packit Service d3d246
    return FALSE;
Packit Service d3d246
Packit Service d3d246
  value = GUINT32_TO_LE (index);
Packit Service d3d246
Packit Service d3d246
  n_words = strinfo_string_to_words (alias, words, TRUE);
Packit Service d3d246
  g_string_append_len (builder, (void *) &value, sizeof value);
Packit Service d3d246
  g_string_append_len (builder, (void *) words, 4 * n_words);
Packit Service d3d246
Packit Service d3d246
  return TRUE;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static gboolean
Packit Service d3d246
strinfo_builder_contains (GString     *builder,
Packit Service d3d246
                          const gchar *string)
Packit Service d3d246
{
Packit Service d3d246
  return strinfo_find_string ((const guint32 *) builder->str,
Packit Service d3d246
                              builder->len / 4, string, FALSE) != -1 ||
Packit Service d3d246
         strinfo_find_string ((const guint32 *) builder->str,
Packit Service d3d246
                              builder->len / 4, string, TRUE) != -1;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
G_GNUC_UNUSED static gboolean
Packit Service d3d246
strinfo_builder_contains_value (GString *builder,
Packit Service d3d246
                                guint    value)
Packit Service d3d246
{
Packit Service d3d246
  return strinfo_string_from_enum ((const guint32 *) builder->str,
Packit Service d3d246
                                   builder->len / 4, value) != NULL;
Packit Service d3d246
}