This repository has been archived on 2025-12-24. You can view files and clone it, but you cannot make any changes to it's state, such as pushing and creating new issues, pull requests or comments.
yachat/iris-legacy/iris/libidn/stringprep.c

476 lines
12 KiB
C
Raw Normal View History

2025-12-25 01:37:49 +05:00
/* stringprep.c Core stringprep implementation.
* Copyright (C) 2002, 2003 Simon Josefsson
*
* This file is part of GNU Libidn.
*
* GNU Libidn is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* GNU Libidn 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with GNU Libidn; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
#include "internal.h"
static ssize_t
stringprep_find_character_in_table (my_uint32_t ucs4,
Stringprep_table_element * table)
{
ssize_t i;
for (i = 0; table[i].start; i++)
if (ucs4 >= table[i].start &&
ucs4 <= (table[i].end ? table[i].end : table[i].start))
return i;
return -1;
}
static ssize_t
stringprep_find_string_in_table (my_uint32_t * ucs4,
size_t ucs4len,
size_t * tablepos,
Stringprep_table_element * table)
{
size_t j;
ssize_t pos;
for (j = 0; j < ucs4len; j++)
if ((pos = stringprep_find_character_in_table (ucs4[j], table)) != -1)
{
if (tablepos)
*tablepos = pos;
return j;
}
return -1;
}
static int
stringprep_apply_table_to_string (my_uint32_t * ucs4,
size_t * ucs4len,
size_t maxucs4len,
Stringprep_table_element * table,
const char *tablename)
{
ssize_t pos;
size_t i, maplen;
while ((pos = stringprep_find_string_in_table (ucs4, *ucs4len,
&i, table)) != -1)
{
for (maplen = STRINGPREP_MAX_MAP_CHARS;
maplen > 0 && table[i].map[maplen - 1] == 0; maplen--)
;
if (*ucs4len - 1 + maplen >= maxucs4len)
return STRINGPREP_TOO_SMALL_BUFFER;
memmove (&ucs4[pos + maplen], &ucs4[pos + 1],
*ucs4len * sizeof (my_uint32_t) - (&ucs4[pos + 1] - ucs4));
memcpy (&ucs4[pos], table[i].map, sizeof (my_uint32_t) * maplen);
*ucs4len = *ucs4len - 1 + maplen;
}
return STRINGPREP_OK;
}
#define INVERTED(x) ((x) & ((~0UL) >> 1))
#define UNAPPLICAPLEFLAGS(flags, profileflags) \
((!INVERTED(profileflags) && !(profileflags & flags) && profileflags) || \
( INVERTED(profileflags) && (profileflags & flags)))
/**
* stringprep:
* @in: input/ouput array with string to prepare.
* @maxlen: maximum length of input/output array.
* @flags: optional stringprep profile flags.
* @profile: pointer to stringprep profile to use.
*
* Prepare the input UTF-8 string according to the stringprep profile.
* Normally application programmers use stringprep profile macros such
* as stringprep_nameprep(), stringprep_kerberos5() etc instead of
* calling this function directly.
*
* Since the stringprep operation can expand the string, @maxlen
* indicate how large the buffer holding the string is. The @flags
* are one of Stringprep_profile_flags, or 0. The profile indicates
* processing details specific to that profile. Your application can
* define new profiles, possibly re-using the generic stringprep
* tables that always will be part of the library.
*
* Note that you must convert strings entered in the systems locale
* into UTF-8 before using this function.
*
* Return value: Returns 0 iff successful, or an error code.
**/
int
stringprep (char *in,
size_t maxlen,
Stringprep_profile_flags flags, Stringprep_profile * profile)
{
size_t i, j;
ssize_t k;
int rc;
char *p = 0;
my_uint32_t *q = 0;
my_uint32_t *ucs4;
size_t ucs4len, maxucs4len;
ucs4 = stringprep_utf8_to_ucs4 (in, -1, &ucs4len);
maxucs4len = 4 * ucs4len + 10; /* XXX */
ucs4 = realloc (ucs4, 1 + maxucs4len * sizeof (my_uint32_t));
if (!ucs4)
{
rc = STRINGPREP_MALLOC_ERROR;
goto done;
}
for (i = 0; profile[i].operation; i++)
{
switch (profile[i].operation)
{
case STRINGPREP_NFKC:
if (UNAPPLICAPLEFLAGS (flags, profile[i].flags))
{
break;
}
if (flags & STRINGPREP_NO_NFKC && !profile[i].flags)
{
/* Profile requires NFKC, but callee asked for no NFKC. */
rc = STRINGPREP_FLAG_ERROR;
goto done;
}
q = stringprep_ucs4_nfkc_normalize (ucs4, ucs4len);
if (!q)
{
rc = STRINGPREP_NFKC_FAILED;
goto done;
}
for (j = 0; q[j]; j++)
;
free (ucs4);
ucs4 = q;
ucs4len = j;
q = 0;
break;
case STRINGPREP_PROHIBIT_TABLE:
k = stringprep_find_string_in_table (ucs4, ucs4len,
NULL, profile[i].table);
if (k != -1)
{
rc = STRINGPREP_CONTAINS_PROHIBITED;
goto done;
}
break;
case STRINGPREP_UNASSIGNED_TABLE:
if (UNAPPLICAPLEFLAGS (flags, profile[i].flags))
break;
if (flags & STRINGPREP_NO_UNASSIGNED)
{
k = stringprep_find_string_in_table
(ucs4, ucs4len, NULL, profile[i].table);
if (k != -1)
{
rc = STRINGPREP_CONTAINS_UNASSIGNED;
goto done;
}
}
break;
case STRINGPREP_MAP_TABLE:
if (UNAPPLICAPLEFLAGS (flags, profile[i].flags))
break;
rc = stringprep_apply_table_to_string
(ucs4, &ucs4len, maxucs4len, profile[i].table, profile[i].name);
if (rc != STRINGPREP_OK)
goto done;
break;
case STRINGPREP_BIDI_PROHIBIT_TABLE:
case STRINGPREP_BIDI_RAL_TABLE:
case STRINGPREP_BIDI_L_TABLE:
break;
case STRINGPREP_BIDI:
{
int done_prohibited = 0;
int done_ral = 0;
int done_l = 0;
int contains_ral = -1;
int contains_l = -1;
for (j = 0; profile[j].operation; j++)
if (profile[j].operation == STRINGPREP_BIDI_PROHIBIT_TABLE)
{
done_prohibited = 1;
k = stringprep_find_string_in_table (ucs4, ucs4len,
NULL,
profile[j].table);
if (k != -1)
{
rc = STRINGPREP_BIDI_CONTAINS_PROHIBITED;
goto done;
}
}
else if (profile[j].operation == STRINGPREP_BIDI_RAL_TABLE)
{
done_ral = 1;
if (stringprep_find_string_in_table
(ucs4, ucs4len, NULL, profile[j].table) != -1)
contains_ral = j;
}
else if (profile[j].operation == STRINGPREP_BIDI_L_TABLE)
{
done_l = 1;
if (stringprep_find_string_in_table
(ucs4, ucs4len, NULL, profile[j].table) != -1)
contains_l = j;
}
if (!done_prohibited || !done_ral || !done_l)
{
rc = STRINGPREP_PROFILE_ERROR;
goto done;
}
if (contains_ral != -1 && contains_l != -1)
{
rc = STRINGPREP_BIDI_BOTH_L_AND_RAL;
goto done;
}
if (contains_ral != -1)
{
if (!(stringprep_find_character_in_table
(ucs4[0], profile[contains_ral].table) != -1 &&
stringprep_find_character_in_table
(ucs4[ucs4len - 1], profile[contains_ral].table) != -1))
{
rc = STRINGPREP_BIDI_LEADTRAIL_NOT_RAL;
goto done;
}
}
}
break;
default:
rc = STRINGPREP_PROFILE_ERROR;
goto done;
break;
}
}
p = stringprep_ucs4_to_utf8 (ucs4, ucs4len, 0, 0);
if (strlen (p) >= maxlen)
{
rc = STRINGPREP_TOO_SMALL_BUFFER;
goto done;
}
strcpy (in, p); /* flawfinder: ignore */
rc = STRINGPREP_OK;
done:
if (p)
free (p);
if (q)
free (q);
if (ucs4)
free (ucs4);
return rc;
}
/**
* stringprep_profile:
* @in: input/ouput array with string to prepare.
* @out: output variable with newly allocate string.
* @flags: optional stringprep profile flags.
* @profile: name of stringprep profile to use.
*
* Prepare the input UTF-8 string according to the stringprep profile.
* Normally application programmers use stringprep profile macros such
* as stringprep_nameprep(), stringprep_kerberos5() etc instead of
* calling this function directly.
*
* Note that you must convert strings entered in the systems locale
* into UTF-8 before using this function.
*
* The output @out variable must be deallocated by the caller.
*
* Return value: Returns 0 iff successful, or an error code.
**/
int
stringprep_profile (char *in,
char **out, char *profile, Stringprep_profile_flags flags)
{
Stringprep_profiles *p;
char *str;
size_t len;
int rc;
for (p = &stringprep_profiles[0]; p->name; p++)
if (strcmp (p->name, profile) == 0)
break;
if (!p || !p->name || !p->tables)
return STRINGPREP_UNKNOWN_PROFILE;
len = strlen (in) + BUFSIZ;
str = (char *) malloc (len);
if (str == NULL)
return STRINGPREP_MALLOC_ERROR;
strcpy (str, in);
rc = stringprep (str, len, flags, p->tables);
if (rc == STRINGPREP_OK)
*out = str;
else
free (str);
return rc;
}
/**
* STRINGPREP_VERSION
*
* String defined via CPP denoting the header file version number.
* Used together with stringprep_check_version() to verify header file
* and run-time library consistency.
*/
/**
* STRINGPREP_MAX_MAP_CHARS
*
* Maximum number of code points that can replace a single code point,
* during stringprep mapping.
*/
/**
* Stringprep_rc
*
* Enumerated return codes of stringprep(), stringprep_profile()
* functions (and macros using those functions). The value 0 is
* guaranteed to always correspond to success.
*/
/**
* Stringprep_profile_flags:
* @STRINGPREP_NO_NFKC: Disable the NFKC normalization, as well as
* selecting the non-NFKC case folding tables. Usually the profile
* specifies BIDI and NFKC settings, and applications should not
* override it unless in special situations.
* @STRINGPREP_NO_BIDI: Disable the BIDI step. Usually the profile
* specifies BIDI and NFKC settings, and applications should not
* override it unless in special situations.
* @STRINGPREP_NO_UNASSIGNED: Make the library return with an error if
* string contains unassigned characters according to profile.
*
* Stringprep profile flags.
*/
/**
* Stringprep_profile_steps:
*
* Various steps in the stringprep algorithm. You really want to
* study the source code to understand this one. Only useful if you
* want to add another profile.
*/
/**
* stringprep_nameprep:
* @in: input/ouput array with string to prepare.
* @maxlen: maximum length of input/output array.
*
* Prepare the input UTF-8 string according to the nameprep profile.
* The AllowUnassigned flag is true, use
* stringprep_nameprep_no_unassigned() for false AllowUnassigned.
* Returns 0 iff successful, or an error code.
**/
/**
* stringprep_nameprep_no_unassigned:
* @in: input/ouput array with string to prepare.
* @maxlen: maximum length of input/output array.
*
* Prepare the input UTF-8 string according to the nameprep profile.
* The AllowUnassigned flag is false, use stringprep_nameprep() for
* true AllowUnassigned. Returns 0 iff successful, or an error code.
**/
/**
* stringprep_iscsi:
* @in: input/ouput array with string to prepare.
* @maxlen: maximum length of input/output array.
*
* Prepare the input UTF-8 string according to the draft iSCSI
* stringprep profile. Returns 0 iff successful, or an error code.
**/
/**
* stringprep_kerberos5:
* @in: input/ouput array with string to prepare.
* @maxlen: maximum length of input/output array.
*
* Prepare the input UTF-8 string according to the draft Kerberos5
* stringprep profile. Returns 0 iff successful, or an error code.
**/
/**
* stringprep_plain:
* @in: input/ouput array with string to prepare.
* @maxlen: maximum length of input/output array.
*
* Prepare the input UTF-8 string according to the draft SASL
* ANONYMOUS profile. Returns 0 iff successful, or an error code.
**/
/**
* stringprep_xmpp_nodeprep:
* @in: input/ouput array with string to prepare.
* @maxlen: maximum length of input/output array.
*
* Prepare the input UTF-8 string according to the draft XMPP node
* identifier profile. Returns 0 iff successful, or an error code.
**/
/**
* stringprep_xmpp_resourceprep:
* @in: input/ouput array with string to prepare.
* @maxlen: maximum length of input/output array.
*
* Prepare the input UTF-8 string according to the draft XMPP resource
* identifier profile. Returns 0 iff successful, or an error code.
**/
/**
* stringprep_generic:
* @in: input/ouput array with string to prepare.
* @maxlen: maximum length of input/output array.
*
* Prepare the input UTF-8 string according to a hypotetical "generic"
* stringprep profile. This is mostly used for debugging or when
* constructing new stringprep profiles. Returns 0 iff successful, or
* an error code.
**/