Blame glib/glib/ghostutils.c

Packit db3073
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
Packit db3073
Packit db3073
/* GLIB - Library of useful routines for C programming
Packit db3073
 * Copyright (C) 2008 Red Hat, Inc.
Packit db3073
 *
Packit db3073
 * This library is free software; you can redistribute it and/or
Packit db3073
 * modify it under the terms of the GNU Lesser General Public
Packit db3073
 * License as published by the Free Software Foundation; either
Packit db3073
 * version 2 of the License, or (at your option) any later version.
Packit db3073
 *
Packit db3073
 * This library is distributed in the hope that it will be useful,
Packit db3073
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit db3073
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit db3073
 * Lesser General Public License for more details.
Packit db3073
 *
Packit db3073
 * You should have received a copy of the GNU Lesser General
Packit db3073
 * Public License along with this library; if not, write to the
Packit db3073
 * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
Packit db3073
 * Boston, MA 02111-1307, USA.
Packit db3073
 */
Packit db3073
Packit db3073
#include "config.h"
Packit db3073
Packit db3073
#include <string.h>
Packit db3073
Packit db3073
#include "ghostutils.h"
Packit db3073
Packit db3073
#include "garray.h"
Packit db3073
#include "gmem.h"
Packit db3073
#include "gstring.h"
Packit db3073
#include "gstrfuncs.h"
Packit db3073
#include "glibintl.h"
Packit db3073
Packit db3073
Packit db3073
/**
Packit db3073
 * SECTION:ghostutils
Packit db3073
 * @short_description: Internet hostname utilities
Packit db3073
 *
Packit db3073
 * Functions for manipulating internet hostnames; in particular, for
Packit db3073
 * converting between Unicode and ASCII-encoded forms of
Packit db3073
 * Internationalized Domain Names (IDNs).
Packit db3073
 *
Packit db3073
 * The 
Packit db3073
 * url="http://www.ietf.org/rfc/rfc3490.txt">Internationalized Domain
Packit db3073
 * Names for Applications (IDNA)</ulink> standards allow for the use
Packit db3073
 * of Unicode domain names in applications, while providing
Packit db3073
 * backward-compatibility with the old ASCII-only DNS, by defining an
Packit db3073
 * ASCII-Compatible Encoding of any given Unicode name, which can be
Packit db3073
 * used with non-IDN-aware applications and protocols. (For example,
Packit db3073
 * "Παν語.org" maps to "xn--4wa8awb4637h.org".)
Packit db3073
 **/
Packit db3073
Packit db3073
#define IDNA_ACE_PREFIX     "xn--"
Packit db3073
#define IDNA_ACE_PREFIX_LEN 4
Packit db3073
Packit db3073
/* Punycode constants, from RFC 3492. */
Packit db3073
Packit db3073
#define PUNYCODE_BASE          36
Packit db3073
#define PUNYCODE_TMIN           1
Packit db3073
#define PUNYCODE_TMAX          26
Packit db3073
#define PUNYCODE_SKEW          38
Packit db3073
#define PUNYCODE_DAMP         700
Packit db3073
#define PUNYCODE_INITIAL_BIAS  72
Packit db3073
#define PUNYCODE_INITIAL_N   0x80
Packit db3073
Packit db3073
#define PUNYCODE_IS_BASIC(cp) ((guint)(cp) < 0x80)
Packit db3073
Packit db3073
/* Encode/decode a single base-36 digit */
Packit db3073
static inline gchar
Packit db3073
encode_digit (guint dig)
Packit db3073
{
Packit db3073
  if (dig < 26)
Packit db3073
    return dig + 'a';
Packit db3073
  else
Packit db3073
    return dig - 26 + '0';
Packit db3073
}
Packit db3073
Packit db3073
static inline guint
Packit db3073
decode_digit (gchar dig)
Packit db3073
{
Packit db3073
  if (dig >= 'A' && dig <= 'Z')
Packit db3073
    return dig - 'A';
Packit db3073
  else if (dig >= 'a' && dig <= 'z')
Packit db3073
    return dig - 'a';
Packit db3073
  else if (dig >= '0' && dig <= '9')
Packit db3073
    return dig - '0' + 26;
Packit db3073
  else
Packit db3073
    return G_MAXUINT;
Packit db3073
}
Packit db3073
Packit db3073
/* Punycode bias adaptation algorithm, RFC 3492 section 6.1 */
Packit db3073
static guint
Packit db3073
adapt (guint    delta,
Packit db3073
       guint    numpoints,
Packit db3073
       gboolean firsttime)
Packit db3073
{
Packit db3073
  guint k;
Packit db3073
Packit db3073
  delta = firsttime ? delta / PUNYCODE_DAMP : delta / 2;
Packit db3073
  delta += delta / numpoints;
Packit db3073
Packit db3073
  k = 0;
Packit db3073
  while (delta > ((PUNYCODE_BASE - PUNYCODE_TMIN) * PUNYCODE_TMAX) / 2)
Packit db3073
    {
Packit db3073
      delta /= PUNYCODE_BASE - PUNYCODE_TMIN;
Packit db3073
      k += PUNYCODE_BASE;
Packit db3073
    }
Packit db3073
Packit db3073
  return k + ((PUNYCODE_BASE - PUNYCODE_TMIN + 1) * delta /
Packit db3073
	      (delta + PUNYCODE_SKEW));
Packit db3073
}
Packit db3073
Packit db3073
/* Punycode encoder, RFC 3492 section 6.3. The algorithm is
Packit db3073
 * sufficiently bizarre that it's not really worth trying to explain
Packit db3073
 * here.
Packit db3073
 */
Packit db3073
static gboolean
Packit db3073
punycode_encode (const gchar *input_utf8,
Packit db3073
                 gsize        input_utf8_length,
Packit db3073
		 GString     *output)
Packit db3073
{
Packit db3073
  guint delta, handled_chars, num_basic_chars, bias, j, q, k, t, digit;
Packit db3073
  gunichar n, m, *input;
Packit db3073
  glong input_length;
Packit db3073
  gboolean success = FALSE;
Packit db3073
Packit db3073
  /* Convert from UTF-8 to Unicode code points */
Packit db3073
  input = g_utf8_to_ucs4 (input_utf8, input_utf8_length, NULL,
Packit db3073
			  &input_length, NULL);
Packit db3073
  if (!input)
Packit db3073
    return FALSE;
Packit db3073
Packit db3073
  /* Copy basic chars */
Packit db3073
  for (j = num_basic_chars = 0; j < input_length; j++)
Packit db3073
    {
Packit db3073
      if (PUNYCODE_IS_BASIC (input[j]))
Packit db3073
	{
Packit db3073
	  g_string_append_c (output, g_ascii_tolower (input[j]));
Packit db3073
	  num_basic_chars++;
Packit db3073
	}
Packit db3073
    }
Packit db3073
  if (num_basic_chars)
Packit db3073
    g_string_append_c (output, '-');
Packit db3073
Packit db3073
  handled_chars = num_basic_chars;
Packit db3073
Packit db3073
  /* Encode non-basic chars */
Packit db3073
  delta = 0;
Packit db3073
  bias = PUNYCODE_INITIAL_BIAS;
Packit db3073
  n = PUNYCODE_INITIAL_N;
Packit db3073
  while (handled_chars < input_length)
Packit db3073
    {
Packit db3073
      /* let m = the minimum {non-basic} code point >= n in the input */
Packit db3073
      for (m = G_MAXUINT, j = 0; j < input_length; j++)
Packit db3073
	{
Packit db3073
	  if (input[j] >= n && input[j] < m)
Packit db3073
	    m = input[j];
Packit db3073
	}
Packit db3073
Packit db3073
      if (m - n > (G_MAXUINT - delta) / (handled_chars + 1))
Packit db3073
	goto fail;
Packit db3073
      delta += (m - n) * (handled_chars + 1);
Packit db3073
      n = m;
Packit db3073
Packit db3073
      for (j = 0; j < input_length; j++)
Packit db3073
	{
Packit db3073
	  if (input[j] < n)
Packit db3073
	    {
Packit db3073
	      if (++delta == 0)
Packit db3073
		goto fail;
Packit db3073
	    }
Packit db3073
	  else if (input[j] == n)
Packit db3073
	    {
Packit db3073
	      q = delta;
Packit db3073
	      for (k = PUNYCODE_BASE; ; k += PUNYCODE_BASE)
Packit db3073
		{
Packit db3073
		  if (k <= bias)
Packit db3073
		    t = PUNYCODE_TMIN;
Packit db3073
		  else if (k >= bias + PUNYCODE_TMAX)
Packit db3073
		    t = PUNYCODE_TMAX;
Packit db3073
		  else
Packit db3073
		    t = k - bias;
Packit db3073
		  if (q < t)
Packit db3073
		    break;
Packit db3073
		  digit = t + (q - t) % (PUNYCODE_BASE - t);
Packit db3073
		  g_string_append_c (output, encode_digit (digit));
Packit db3073
		  q = (q - t) / (PUNYCODE_BASE - t);
Packit db3073
		}
Packit db3073
Packit db3073
	      g_string_append_c (output, encode_digit (q));
Packit db3073
	      bias = adapt (delta, handled_chars + 1, handled_chars == num_basic_chars);
Packit db3073
	      delta = 0;
Packit db3073
	      handled_chars++;
Packit db3073
	    }
Packit db3073
	}
Packit db3073
Packit db3073
      delta++;
Packit db3073
      n++;
Packit db3073
    }
Packit db3073
Packit db3073
  success = TRUE;
Packit db3073
Packit db3073
 fail:
Packit db3073
  g_free (input);
Packit db3073
  return success;
Packit db3073
}
Packit db3073
Packit db3073
/* From RFC 3454, Table B.1 */
Packit db3073
#define idna_is_junk(ch) ((ch) == 0x00AD || (ch) == 0x1806 || (ch) == 0x200B || (ch) == 0x2060 || (ch) == 0xFEFF || (ch) == 0x034F || (ch) == 0x180B || (ch) == 0x180C || (ch) == 0x180D || (ch) == 0x200C || (ch) == 0x200D || ((ch) >= 0xFE00 && (ch) <= 0xFE0F))
Packit db3073
Packit db3073
/* Scan @str for "junk" and return a cleaned-up string if any junk
Packit db3073
 * is found. Else return %NULL.
Packit db3073
 */
Packit db3073
static gchar *
Packit db3073
remove_junk (const gchar *str,
Packit db3073
             gint         len)
Packit db3073
{
Packit db3073
  GString *cleaned = NULL;
Packit db3073
  const gchar *p;
Packit db3073
  gunichar ch;
Packit db3073
Packit db3073
  for (p = str; len == -1 ? *p : p < str + len; p = g_utf8_next_char (p))
Packit db3073
    {
Packit db3073
      ch = g_utf8_get_char (p);
Packit db3073
      if (idna_is_junk (ch))
Packit db3073
	{
Packit db3073
	  if (!cleaned)
Packit db3073
	    {
Packit db3073
	      cleaned = g_string_new (NULL);
Packit db3073
	      g_string_append_len (cleaned, str, p - str);
Packit db3073
	    }
Packit db3073
	}
Packit db3073
      else if (cleaned)
Packit db3073
	g_string_append_unichar (cleaned, ch);
Packit db3073
    }
Packit db3073
Packit db3073
  if (cleaned)
Packit db3073
    return g_string_free (cleaned, FALSE);
Packit db3073
  else
Packit db3073
    return NULL;
Packit db3073
}
Packit db3073
Packit db3073
static inline gboolean
Packit db3073
contains_uppercase_letters (const gchar *str,
Packit db3073
                            gint         len)
Packit db3073
{
Packit db3073
  const gchar *p;
Packit db3073
Packit db3073
  for (p = str; len == -1 ? *p : p < str + len; p = g_utf8_next_char (p))
Packit db3073
    {
Packit db3073
      if (g_unichar_isupper (g_utf8_get_char (p)))
Packit db3073
	return TRUE;
Packit db3073
    }
Packit db3073
  return FALSE;
Packit db3073
}
Packit db3073
Packit db3073
static inline gboolean
Packit db3073
contains_non_ascii (const gchar *str,
Packit db3073
                    gint         len)
Packit db3073
{
Packit db3073
  const gchar *p;
Packit db3073
Packit db3073
  for (p = str; len == -1 ? *p : p < str + len; p++)
Packit db3073
    {
Packit db3073
      if ((guchar)*p > 0x80)
Packit db3073
	return TRUE;
Packit db3073
    }
Packit db3073
  return FALSE;
Packit db3073
}
Packit db3073
Packit db3073
/* RFC 3454, Appendix C. ish. */
Packit db3073
static inline gboolean
Packit db3073
idna_is_prohibited (gunichar ch)
Packit db3073
{
Packit db3073
  switch (g_unichar_type (ch))
Packit db3073
    {
Packit db3073
    case G_UNICODE_CONTROL:
Packit db3073
    case G_UNICODE_FORMAT:
Packit db3073
    case G_UNICODE_UNASSIGNED:
Packit db3073
    case G_UNICODE_PRIVATE_USE:
Packit db3073
    case G_UNICODE_SURROGATE:
Packit db3073
    case G_UNICODE_LINE_SEPARATOR:
Packit db3073
    case G_UNICODE_PARAGRAPH_SEPARATOR:
Packit db3073
    case G_UNICODE_SPACE_SEPARATOR:
Packit db3073
      return TRUE;
Packit db3073
Packit db3073
    case G_UNICODE_OTHER_SYMBOL:
Packit db3073
      if (ch == 0xFFFC || ch == 0xFFFD ||
Packit db3073
	  (ch >= 0x2FF0 && ch <= 0x2FFB))
Packit db3073
	return TRUE;
Packit db3073
      return FALSE;
Packit db3073
Packit db3073
    case G_UNICODE_NON_SPACING_MARK:
Packit db3073
      if (ch == 0x0340 || ch == 0x0341)
Packit db3073
	return TRUE;
Packit db3073
      return FALSE;
Packit db3073
Packit db3073
    default:
Packit db3073
      return FALSE;
Packit db3073
    }
Packit db3073
}
Packit db3073
Packit db3073
/* RFC 3491 IDN cleanup algorithm. */
Packit db3073
static gchar *
Packit db3073
nameprep (const gchar *hostname,
Packit db3073
          gint         len,
Packit db3073
          gboolean    *is_unicode)
Packit db3073
{
Packit db3073
  gchar *name, *tmp = NULL, *p;
Packit db3073
Packit db3073
  /* It would be nice if we could do this without repeatedly
Packit db3073
   * allocating strings and converting back and forth between
Packit db3073
   * gunichars and UTF-8... The code does at least avoid doing most of
Packit db3073
   * the sub-operations when they would just be equivalent to a
Packit db3073
   * g_strdup().
Packit db3073
   */
Packit db3073
Packit db3073
  /* Remove presentation-only characters */
Packit db3073
  name = remove_junk (hostname, len);
Packit db3073
  if (name)
Packit db3073
    {
Packit db3073
      tmp = name;
Packit db3073
      len = -1;
Packit db3073
    }
Packit db3073
  else
Packit db3073
    name = (gchar *)hostname;
Packit db3073
Packit db3073
  /* Convert to lowercase */
Packit db3073
  if (contains_uppercase_letters (name, len))
Packit db3073
    {
Packit db3073
      name = g_utf8_strdown (name, len);
Packit db3073
      g_free (tmp);
Packit db3073
      tmp = name;
Packit db3073
      len = -1;
Packit db3073
    }
Packit db3073
Packit db3073
  /* If there are no UTF8 characters, we're done. */
Packit db3073
  if (!contains_non_ascii (name, len))
Packit db3073
    {
Packit db3073
      *is_unicode = FALSE;
Packit db3073
      if (name == (gchar *)hostname)
Packit db3073
        return len == -1 ? g_strdup (hostname) : g_strndup (hostname, len);
Packit db3073
      else
Packit db3073
        return name;
Packit db3073
    }
Packit db3073
Packit db3073
  *is_unicode = TRUE;
Packit db3073
Packit db3073
  /* Normalize */
Packit db3073
  name = g_utf8_normalize (name, len, G_NORMALIZE_NFKC);
Packit db3073
  g_free (tmp);
Packit db3073
  tmp = name;
Packit db3073
Packit db3073
  if (!name)
Packit db3073
    return NULL;
Packit db3073
Packit db3073
  /* KC normalization may have created more capital letters (eg,
Packit db3073
   * angstrom -> capital A with ring). So we have to lowercasify a
Packit db3073
   * second time. (This is more-or-less how the nameprep algorithm
Packit db3073
   * does it. If tolower(nfkc(tolower(X))) is guaranteed to be the
Packit db3073
   * same as tolower(nfkc(X)), then we could skip the first tolower,
Packit db3073
   * but I'm not sure it is.)
Packit db3073
   */
Packit db3073
  if (contains_uppercase_letters (name, -1))
Packit db3073
    {
Packit db3073
      name = g_utf8_strdown (name, -1);
Packit db3073
      g_free (tmp);
Packit db3073
      tmp = name;
Packit db3073
    }
Packit db3073
Packit db3073
  /* Check for prohibited characters */
Packit db3073
  for (p = name; *p; p = g_utf8_next_char (p))
Packit db3073
    {
Packit db3073
      if (idna_is_prohibited (g_utf8_get_char (p)))
Packit db3073
	{
Packit db3073
	  name = NULL;
Packit db3073
          g_free (tmp);
Packit db3073
	  goto done;
Packit db3073
	}
Packit db3073
    }
Packit db3073
Packit db3073
  /* FIXME: We're supposed to verify certain constraints on bidi
Packit db3073
   * characters, but glib does not appear to have that information.
Packit db3073
   */
Packit db3073
Packit db3073
 done:
Packit db3073
  return name;
Packit db3073
}
Packit db3073
Packit db3073
/* RFC 3490, section 3.1 says '.', 0x3002, 0xFF0E, and 0xFF61 count as
Packit db3073
 * label-separating dots. @str must be '\0'-terminated.
Packit db3073
 */
Packit db3073
#define idna_is_dot(str) ( \
Packit db3073
  ((guchar)(str)[0] == '.') ||                                                 \
Packit db3073
  ((guchar)(str)[0] == 0xE3 && (guchar)(str)[1] == 0x80 && (guchar)(str)[2] == 0x82) || \
Packit db3073
  ((guchar)(str)[0] == 0xEF && (guchar)(str)[1] == 0xBC && (guchar)(str)[2] == 0x8E) || \
Packit db3073
  ((guchar)(str)[0] == 0xEF && (guchar)(str)[1] == 0xBD && (guchar)(str)[2] == 0xA1) )
Packit db3073
Packit db3073
static const gchar *
Packit db3073
idna_end_of_label (const gchar *str)
Packit db3073
{
Packit db3073
  for (; *str; str = g_utf8_next_char (str))
Packit db3073
    {
Packit db3073
      if (idna_is_dot (str))
Packit db3073
        return str;
Packit db3073
    }
Packit db3073
  return str;
Packit db3073
}
Packit db3073
Packit db3073
/**
Packit db3073
 * g_hostname_to_ascii:
Packit db3073
 * @hostname: a valid UTF-8 or ASCII hostname
Packit db3073
 *
Packit db3073
 * Converts @hostname to its canonical ASCII form; an ASCII-only
Packit db3073
 * string containing no uppercase letters and not ending with a
Packit db3073
 * trailing dot.
Packit db3073
 *
Packit db3073
 * Return value: an ASCII hostname, which must be freed, or %NULL if
Packit db3073
 * @hostname is in some way invalid.
Packit db3073
 *
Packit db3073
 * Since: 2.22
Packit db3073
 **/
Packit db3073
gchar *
Packit db3073
g_hostname_to_ascii (const gchar *hostname)
Packit db3073
{
Packit db3073
  gchar *name, *label, *p;
Packit db3073
  GString *out;
Packit db3073
  gssize llen, oldlen;
Packit db3073
  gboolean unicode;
Packit db3073
Packit db3073
  label = name = nameprep (hostname, -1, &unicode);
Packit db3073
  if (!name || !unicode)
Packit db3073
    return name;
Packit db3073
Packit db3073
  out = g_string_new (NULL);
Packit db3073
Packit db3073
  do
Packit db3073
    {
Packit db3073
      unicode = FALSE;
Packit db3073
      for (p = label; *p && !idna_is_dot (p); p++)
Packit db3073
	{
Packit db3073
	  if ((guchar)*p > 0x80)
Packit db3073
	    unicode = TRUE;
Packit db3073
	}
Packit db3073
Packit db3073
      oldlen = out->len;
Packit db3073
      llen = p - label;
Packit db3073
      if (unicode)
Packit db3073
	{
Packit db3073
          if (!strncmp (label, IDNA_ACE_PREFIX, IDNA_ACE_PREFIX_LEN))
Packit db3073
            goto fail;
Packit db3073
Packit db3073
	  g_string_append (out, IDNA_ACE_PREFIX);
Packit db3073
	  if (!punycode_encode (label, llen, out))
Packit db3073
	    goto fail;
Packit db3073
	}
Packit db3073
      else
Packit db3073
        g_string_append_len (out, label, llen);
Packit db3073
Packit db3073
      if (out->len - oldlen > 63)
Packit db3073
	goto fail;
Packit db3073
Packit db3073
      label += llen;
Packit db3073
      if (*label)
Packit db3073
        label = g_utf8_next_char (label);
Packit db3073
      if (*label)
Packit db3073
        g_string_append_c (out, '.');
Packit db3073
    }
Packit db3073
  while (*label);
Packit db3073
Packit db3073
  g_free (name);
Packit db3073
  return g_string_free (out, FALSE);
Packit db3073
Packit db3073
 fail:
Packit db3073
  g_free (name);
Packit db3073
  g_string_free (out, TRUE);
Packit db3073
  return NULL;
Packit db3073
}
Packit db3073
Packit db3073
/**
Packit db3073
 * g_hostname_is_non_ascii:
Packit db3073
 * @hostname: a hostname
Packit db3073
 *
Packit db3073
 * Tests if @hostname contains Unicode characters. If this returns
Packit db3073
 * %TRUE, you need to encode the hostname with g_hostname_to_ascii()
Packit db3073
 * before using it in non-IDN-aware contexts.
Packit db3073
 *
Packit db3073
 * Note that a hostname might contain a mix of encoded and unencoded
Packit db3073
 * segments, and so it is possible for g_hostname_is_non_ascii() and
Packit db3073
 * g_hostname_is_ascii_encoded() to both return %TRUE for a name.
Packit db3073
 *
Packit db3073
 * Return value: %TRUE if @hostname contains any non-ASCII characters
Packit db3073
 *
Packit db3073
 * Since: 2.22
Packit db3073
 **/
Packit db3073
gboolean
Packit db3073
g_hostname_is_non_ascii (const gchar *hostname)
Packit db3073
{
Packit db3073
  return contains_non_ascii (hostname, -1);
Packit db3073
}
Packit db3073
Packit db3073
/* Punycode decoder, RFC 3492 section 6.2. As with punycode_encode(),
Packit db3073
 * read the RFC if you want to understand what this is actually doing.
Packit db3073
 */
Packit db3073
static gboolean
Packit db3073
punycode_decode (const gchar *input,
Packit db3073
                 gsize        input_length,
Packit db3073
                 GString     *output)
Packit db3073
{
Packit db3073
  GArray *output_chars;
Packit db3073
  gunichar n;
Packit db3073
  guint i, bias;
Packit db3073
  guint oldi, w, k, digit, t;
Packit db3073
  const gchar *split;
Packit db3073
Packit db3073
  n = PUNYCODE_INITIAL_N;
Packit db3073
  i = 0;
Packit db3073
  bias = PUNYCODE_INITIAL_BIAS;
Packit db3073
Packit db3073
  split = input + input_length - 1;
Packit db3073
  while (split > input && *split != '-')
Packit db3073
    split--;
Packit db3073
  if (split > input)
Packit db3073
    {
Packit db3073
      output_chars = g_array_sized_new (FALSE, FALSE, sizeof (gunichar),
Packit db3073
					split - input);
Packit db3073
      input_length -= (split - input) + 1;
Packit db3073
      while (input < split)
Packit db3073
	{
Packit db3073
	  gunichar ch = (gunichar)*input++;
Packit db3073
	  if (!PUNYCODE_IS_BASIC (ch))
Packit db3073
	    goto fail;
Packit db3073
	  g_array_append_val (output_chars, ch);
Packit db3073
	}
Packit db3073
      input++;
Packit db3073
    }
Packit db3073
  else
Packit db3073
    output_chars = g_array_new (FALSE, FALSE, sizeof (gunichar));
Packit db3073
Packit db3073
  while (input_length)
Packit db3073
    {
Packit db3073
      oldi = i;
Packit db3073
      w = 1;
Packit db3073
      for (k = PUNYCODE_BASE; ; k += PUNYCODE_BASE)
Packit db3073
	{
Packit db3073
	  if (!input_length--)
Packit db3073
	    goto fail;
Packit db3073
	  digit = decode_digit (*input++);
Packit db3073
	  if (digit >= PUNYCODE_BASE)
Packit db3073
	    goto fail;
Packit db3073
	  if (digit > (G_MAXUINT - i) / w)
Packit db3073
	    goto fail;
Packit db3073
	  i += digit * w;
Packit db3073
	  if (k <= bias)
Packit db3073
	    t = PUNYCODE_TMIN;
Packit db3073
	  else if (k >= bias + PUNYCODE_TMAX)
Packit db3073
	    t = PUNYCODE_TMAX;
Packit db3073
	  else
Packit db3073
	    t = k - bias;
Packit db3073
	  if (digit < t)
Packit db3073
	    break;
Packit db3073
	  if (w > G_MAXUINT / (PUNYCODE_BASE - t))
Packit db3073
	    goto fail;
Packit db3073
	  w *= (PUNYCODE_BASE - t);
Packit db3073
	}
Packit db3073
Packit db3073
      bias = adapt (i - oldi, output_chars->len + 1, oldi == 0);
Packit db3073
Packit db3073
      if (i / (output_chars->len + 1) > G_MAXUINT - n)
Packit db3073
	goto fail;
Packit db3073
      n += i / (output_chars->len + 1);
Packit db3073
      i %= (output_chars->len + 1);
Packit db3073
Packit db3073
      g_array_insert_val (output_chars, i++, n);
Packit db3073
    }
Packit db3073
Packit db3073
  for (i = 0; i < output_chars->len; i++)
Packit db3073
    g_string_append_unichar (output, g_array_index (output_chars, gunichar, i));
Packit db3073
  g_array_free (output_chars, TRUE);
Packit db3073
  return TRUE;
Packit db3073
Packit db3073
 fail:
Packit db3073
  g_array_free (output_chars, TRUE);
Packit db3073
  return FALSE;
Packit db3073
}
Packit db3073
Packit db3073
/**
Packit db3073
 * g_hostname_to_unicode:
Packit db3073
 * @hostname: a valid UTF-8 or ASCII hostname
Packit db3073
 *
Packit db3073
 * Converts @hostname to its canonical presentation form; a UTF-8
Packit db3073
 * string in Unicode normalization form C, containing no uppercase
Packit db3073
 * letters, no forbidden characters, and no ASCII-encoded segments,
Packit db3073
 * and not ending with a trailing dot.
Packit db3073
 *
Packit db3073
 * Of course if @hostname is not an internationalized hostname, then
Packit db3073
 * the canonical presentation form will be entirely ASCII.
Packit db3073
 *
Packit db3073
 * Return value: a UTF-8 hostname, which must be freed, or %NULL if
Packit db3073
 * @hostname is in some way invalid.
Packit db3073
 *
Packit db3073
 * Since: 2.22
Packit db3073
 **/
Packit db3073
gchar *
Packit db3073
g_hostname_to_unicode (const gchar *hostname)
Packit db3073
{
Packit db3073
  GString *out;
Packit db3073
  gssize llen;
Packit db3073
Packit db3073
  out = g_string_new (NULL);
Packit db3073
Packit db3073
  do
Packit db3073
    {
Packit db3073
      llen = idna_end_of_label (hostname) - hostname;
Packit db3073
      if (!g_ascii_strncasecmp (hostname, IDNA_ACE_PREFIX, IDNA_ACE_PREFIX_LEN))
Packit db3073
	{
Packit db3073
	  hostname += IDNA_ACE_PREFIX_LEN;
Packit db3073
	  llen -= IDNA_ACE_PREFIX_LEN;
Packit db3073
	  if (!punycode_decode (hostname, llen, out))
Packit db3073
	    {
Packit db3073
	      g_string_free (out, TRUE);
Packit db3073
	      return NULL;
Packit db3073
	    }
Packit db3073
	}
Packit db3073
      else
Packit db3073
        {
Packit db3073
          gboolean unicode;
Packit db3073
          gchar *canonicalized = nameprep (hostname, llen, &unicode);
Packit db3073
Packit db3073
          if (!canonicalized)
Packit db3073
            {
Packit db3073
              g_string_free (out, TRUE);
Packit db3073
              return NULL;
Packit db3073
            }
Packit db3073
          g_string_append (out, canonicalized);
Packit db3073
          g_free (canonicalized);
Packit db3073
        }
Packit db3073
Packit db3073
      hostname += llen;
Packit db3073
      if (*hostname)
Packit db3073
        hostname = g_utf8_next_char (hostname);
Packit db3073
      if (*hostname)
Packit db3073
        g_string_append_c (out, '.');
Packit db3073
    }
Packit db3073
  while (*hostname);
Packit db3073
Packit db3073
  return g_string_free (out, FALSE);
Packit db3073
}
Packit db3073
Packit db3073
/**
Packit db3073
 * g_hostname_is_ascii_encoded:
Packit db3073
 * @hostname: a hostname
Packit db3073
 *
Packit db3073
 * Tests if @hostname contains segments with an ASCII-compatible
Packit db3073
 * encoding of an Internationalized Domain Name. If this returns
Packit db3073
 * %TRUE, you should decode the hostname with g_hostname_to_unicode()
Packit db3073
 * before displaying it to the user.
Packit db3073
 *
Packit db3073
 * Note that a hostname might contain a mix of encoded and unencoded
Packit db3073
 * segments, and so it is possible for g_hostname_is_non_ascii() and
Packit db3073
 * g_hostname_is_ascii_encoded() to both return %TRUE for a name.
Packit db3073
 *
Packit db3073
 * Return value: %TRUE if @hostname contains any ASCII-encoded
Packit db3073
 * segments.
Packit db3073
 *
Packit db3073
 * Since: 2.22
Packit db3073
 **/
Packit db3073
gboolean
Packit db3073
g_hostname_is_ascii_encoded (const gchar *hostname)
Packit db3073
{
Packit db3073
  while (1)
Packit db3073
    {
Packit db3073
      if (!g_ascii_strncasecmp (hostname, IDNA_ACE_PREFIX, IDNA_ACE_PREFIX_LEN))
Packit db3073
	return TRUE;
Packit db3073
      hostname = idna_end_of_label (hostname);
Packit db3073
      if (*hostname)
Packit db3073
        hostname = g_utf8_next_char (hostname);
Packit db3073
      if (!*hostname)
Packit db3073
	return FALSE;
Packit db3073
    }
Packit db3073
}
Packit db3073
Packit db3073
/**
Packit db3073
 * g_hostname_is_ip_address:
Packit db3073
 * @hostname: a hostname (or IP address in string form)
Packit db3073
 *
Packit db3073
 * Tests if @hostname is the string form of an IPv4 or IPv6 address.
Packit db3073
 * (Eg, "192.168.0.1".)
Packit db3073
 *
Packit db3073
 * Return value: %TRUE if @hostname is an IP address
Packit db3073
 *
Packit db3073
 * Since: 2.22
Packit db3073
 **/
Packit db3073
gboolean
Packit db3073
g_hostname_is_ip_address (const gchar *hostname)
Packit db3073
{
Packit db3073
  gchar *p, *end;
Packit db3073
  gint nsegments, octet;
Packit db3073
Packit db3073
  /* On Linux we could implement this using inet_pton, but the Windows
Packit db3073
   * equivalent of that requires linking against winsock, so we just
Packit db3073
   * figure this out ourselves. Tested by tests/hostutils.c.
Packit db3073
   */
Packit db3073
Packit db3073
  p = (char *)hostname;
Packit db3073
Packit db3073
  if (strchr (p, ':'))
Packit db3073
    {
Packit db3073
      gboolean skipped;
Packit db3073
Packit db3073
      /* If it contains a ':', it's an IPv6 address (assuming it's an
Packit db3073
       * IP address at all). This consists of eight ':'-separated
Packit db3073
       * segments, each containing a 1-4 digit hex number, except that
Packit db3073
       * optionally: (a) the last two segments can be replaced by an
Packit db3073
       * IPv4 address, and (b) a single span of 1 to 8 "0000" segments
Packit db3073
       * can be replaced with just "::".
Packit db3073
       */
Packit db3073
Packit db3073
      nsegments = 0;
Packit db3073
      skipped = FALSE;
Packit db3073
      while (*p && nsegments < 8)
Packit db3073
        {
Packit db3073
          /* Each segment after the first must be preceded by a ':'.
Packit db3073
           * (We also handle half of the "string starts with ::" case
Packit db3073
           * here.)
Packit db3073
           */
Packit db3073
          if (p != (char *)hostname || (p[0] == ':' && p[1] == ':'))
Packit db3073
            {
Packit db3073
              if (*p != ':')
Packit db3073
                return FALSE;
Packit db3073
              p++;
Packit db3073
            }
Packit db3073
Packit db3073
          /* If there's another ':', it means we're skipping some segments */
Packit db3073
          if (*p == ':' && !skipped)
Packit db3073
            {
Packit db3073
              skipped = TRUE;
Packit db3073
              nsegments++;
Packit db3073
Packit db3073
              /* Handle the "string ends with ::" case */
Packit db3073
              if (!p[1])
Packit db3073
                p++;
Packit db3073
Packit db3073
              continue;
Packit db3073
            }
Packit db3073
Packit db3073
          /* Read the segment, make sure it's valid. */
Packit db3073
          for (end = p; g_ascii_isxdigit (*end); end++)
Packit db3073
            ;
Packit db3073
          if (end == p || end > p + 4)
Packit db3073
            return FALSE;
Packit db3073
Packit db3073
          if (*end == '.')
Packit db3073
            {
Packit db3073
              if ((nsegments == 6 && !skipped) || (nsegments <= 6 && skipped))
Packit db3073
                goto parse_ipv4;
Packit db3073
              else
Packit db3073
                return FALSE;
Packit db3073
            }
Packit db3073
Packit db3073
          nsegments++;
Packit db3073
          p = end;
Packit db3073
        }
Packit db3073
Packit db3073
      return !*p && (nsegments == 8 || skipped);
Packit db3073
    }
Packit db3073
Packit db3073
 parse_ipv4:
Packit db3073
Packit db3073
  /* Parse IPv4: N.N.N.N, where each N <= 255 and doesn't have leading 0s. */
Packit db3073
  for (nsegments = 0; nsegments < 4; nsegments++)
Packit db3073
    {
Packit db3073
      if (nsegments != 0)
Packit db3073
        {
Packit db3073
          if (*p != '.')
Packit db3073
            return FALSE;
Packit db3073
          p++;
Packit db3073
        }
Packit db3073
Packit db3073
      /* Check the segment; a little tricker than the IPv6 case since
Packit db3073
       * we can't allow extra leading 0s, and we can't assume that all
Packit db3073
       * strings of valid length are within range.
Packit db3073
       */
Packit db3073
      octet = 0;
Packit db3073
      if (*p == '0')
Packit db3073
        end = p + 1;
Packit db3073
      else
Packit db3073
        {
Packit db3073
          for (end = p; g_ascii_isdigit (*end); end++)
Packit db3073
            octet = 10 * octet + (*end - '0');
Packit db3073
        }
Packit db3073
      if (end == p || end > p + 3 || octet > 255)
Packit db3073
        return FALSE;
Packit db3073
Packit db3073
      p = end;
Packit db3073
    }
Packit db3073
Packit db3073
  /* If there's nothing left to parse, then it's ok. */
Packit db3073
  return !*p;
Packit db3073
}