#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <iconv.h>
#include <libintl.h>
#include <sys/u8_textprep.h>
#include <netsmb/smb_lib.h>
#include "charsets.h"
size_t
unicode_strlen(const uint16_t *us)
{
size_t len = 0;
while (*us++)
len++;
return (len);
}
static char *convert_ucs2xx_to_utf8(iconv_t, const uint16_t *);
char *
convert_unicode_to_utf8(uint16_t *us)
{
static iconv_t cd1 = (iconv_t)-1;
if (cd1 == (iconv_t)-1)
cd1 = iconv_open("UTF-8", "UCS-2");
return (convert_ucs2xx_to_utf8(cd1, us));
}
char *
convert_leunicode_to_utf8(unsigned short *us)
{
static iconv_t cd2 = (iconv_t)-1;
if (cd2 == (iconv_t)-1)
cd2 = iconv_open("UTF-8", "UCS-2LE");
return (convert_ucs2xx_to_utf8(cd2, us));
}
static char *
convert_ucs2xx_to_utf8(iconv_t cd, const uint16_t *us)
{
char *obuf, *optr;
const char *iptr;
size_t ileft, obsize, oleft, ret;
if (cd == (iconv_t)-1) {
smb_error(dgettext(TEXT_DOMAIN,
"iconv_open(UTF-8/UCS-2)"), -1);
return (NULL);
}
iptr = (const char *)us;
ileft = unicode_strlen(us);
ileft *= 2;
oleft = ileft * 2;
obsize = oleft + 2;
obuf = malloc(obsize);
if (!obuf)
return (NULL);
optr = obuf;
ret = iconv(cd, &iptr, &ileft, &optr, &oleft);
*optr = '\0';
if (ret == (size_t)-1) {
smb_error(dgettext(TEXT_DOMAIN,
"iconv(%s) failed"), errno, obuf);
}
if (ileft) {
smb_error(dgettext(TEXT_DOMAIN,
"iconv(%s) failed"), -1, obuf);
}
return (obuf);
}
static uint16_t *convert_utf8_to_ucs2xx(iconv_t, const char *);
uint16_t *
convert_utf8_to_unicode(const char *utf8_string)
{
static iconv_t cd3 = (iconv_t)-1;
if (cd3 == (iconv_t)-1)
cd3 = iconv_open("UCS-2", "UTF-8");
return (convert_utf8_to_ucs2xx(cd3, utf8_string));
}
uint16_t *
convert_utf8_to_leunicode(const char *utf8_string)
{
static iconv_t cd4 = (iconv_t)-1;
if (cd4 == (iconv_t)-1)
cd4 = iconv_open("UCS-2LE", "UTF-8");
return (convert_utf8_to_ucs2xx(cd4, utf8_string));
}
static uint16_t *
convert_utf8_to_ucs2xx(iconv_t cd, const char *utf8_string)
{
uint16_t *obuf, *optr;
const char *iptr;
size_t ileft, obsize, oleft, ret;
if (cd == (iconv_t)-1) {
smb_error(dgettext(TEXT_DOMAIN,
"iconv_open(UCS-2/UTF-8)"), -1);
return (NULL);
}
iptr = utf8_string;
ileft = strlen(iptr);
oleft = ileft * 2;
obsize = oleft + 2;
obuf = malloc(obsize);
if (!obuf)
return (NULL);
optr = obuf;
ret = iconv(cd, &iptr, &ileft, (char **)&optr, &oleft);
*optr = '\0';
if (ret == (size_t)-1) {
smb_error(dgettext(TEXT_DOMAIN,
"iconv(%s) failed"), errno, utf8_string);
}
if (ileft) {
smb_error(dgettext(TEXT_DOMAIN,
"iconv(%s) failed"), -1, utf8_string);
}
return (obuf);
}
static char *
utf8_str_to_upper_or_lower(const char *s, int upper_lower)
{
char *res = NULL;
char *outs;
size_t inlen, outlen, inbleft, outbleft;
int rc, err;
inlen = inbleft = strlen(s);
outlen = outbleft = inlen + 10;
if ((res = malloc(outlen)) == NULL)
return (NULL);
outs = res;
while ((rc = u8_textprep_str((char *)s, &inbleft, outs,
&outbleft, upper_lower, U8_UNICODE_LATEST, &err)) < 0 &&
err == E2BIG) {
if ((res = realloc(res, outlen + inbleft)) == NULL)
return (NULL);
s += (inlen - inbleft);
outs = res + outlen - outbleft;
outlen += inbleft;
outbleft += inbleft;
}
if (rc < 0) {
free(res);
res = NULL;
return (NULL);
}
res[outlen - outbleft] = '\0';
return (res);
}
char *
utf8_str_toupper(const char *s)
{
return (utf8_str_to_upper_or_lower(s, U8_TEXTPREP_TOUPPER));
}
char *
utf8_str_tolower(const char *s)
{
return (utf8_str_to_upper_or_lower(s, U8_TEXTPREP_TOLOWER));
}