Home | History | Annotate | Line # | Download | only in iconv
iconv.c revision 1.10
      1 /*	$NetBSD: iconv.c,v 1.10 2009/03/03 15:42:43 explorer Exp $	*/
      2 
      3 /*-
      4  * Copyright (c)2003 Citrus Project,
      5  * All rights reserved.
      6  *
      7  * Redistribution and use in source and binary forms, with or without
      8  * modification, are permitted provided that the following conditions
      9  * are met:
     10  * 1. Redistributions of source code must retain the above copyright
     11  *    notice, this list of conditions and the following disclaimer.
     12  * 2. Redistributions in binary form must reproduce the above copyright
     13  *    notice, this list of conditions and the following disclaimer in the
     14  *    documentation and/or other materials provided with the distribution.
     15  *
     16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
     17  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     19  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
     20  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     21  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     22  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     23  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     24  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     25  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     26  * SUCH DAMAGE.
     27  */
     28 
     29 #include <sys/cdefs.h>
     30 #if defined(LIBC_SCCS) && !defined(lint)
     31 __RCSID("$NetBSD: iconv.c,v 1.10 2009/03/03 15:42:43 explorer Exp $");
     32 #endif /* LIBC_SCCS and not lint */
     33 
     34 #include "namespace.h"
     35 #include <assert.h>
     36 #include <errno.h>
     37 #include <paths.h>
     38 #include <sys/queue.h>
     39 
     40 #include <iconv.h>
     41 
     42 #ifdef __weak_alias
     43 __weak_alias(iconv, _iconv)
     44 __weak_alias(iconv_open, _iconv_open)
     45 __weak_alias(iconv_close, _iconv_close)
     46 #endif
     47 
     48 #ifdef HAVE_CITRUS
     49 #include <sys/types.h>
     50 #include <string.h>
     51 #include <stdlib.h>
     52 #include "citrus_types.h"
     53 #include "citrus_module.h"
     54 #include "citrus_esdb.h"
     55 #include "citrus_hash.h"
     56 #include "citrus_iconv.h"
     57 
     58 #define ISBADF(_h_)	(!(_h_) || (_h_) == (iconv_t)-1)
     59 
     60 iconv_t
     61 iconv_open(const char *out, const char *in)
     62 {
     63 	int ret;
     64 	struct _citrus_iconv *handle;
     65 	char *out_truncated;
     66 	char *p;
     67 
     68 	/*
     69 	 * Remove anything following a //, as these are options (like
     70 	 * //ignore, //translate, etc) and we just don't handle them.
     71 	 * This is for compatibilty wiht software that uses thees
     72 	 * blindly.
     73 	 */
     74 	out_truncated = strdup(out);
     75 	if (out_truncated == NULL) {
     76 		errno = ENOMEM;
     77 		return ((iconv_t)-1);
     78 	}
     79 
     80 	p = out_truncated;
     81 	while (*p != '/' && *p != 0)
     82 		p++;
     83 	if (p[0] == '/' && p[1] == '/')
     84 		p[0] = 0;
     85 
     86 	ret = _citrus_iconv_open(&handle, _PATH_ICONV, in, out_truncated);
     87 	free(out_truncated);
     88 	if (ret) {
     89 		errno = ret == ENOENT? EINVAL : ret;
     90 		return ((iconv_t)-1);
     91 	}
     92 
     93 	return ((iconv_t)(void *)handle);
     94 }
     95 
     96 int
     97 iconv_close(iconv_t handle)
     98 {
     99 	if (ISBADF(handle)) {
    100 		errno = EBADF;
    101 		return (-1);
    102 	}
    103 
    104 	_citrus_iconv_close((struct _citrus_iconv *)(void *)handle);
    105 
    106 	return (0);
    107 }
    108 
    109 size_t
    110 iconv(iconv_t handle, const char **in, size_t *szin, char **out, size_t *szout)
    111 {
    112 	int err;
    113 	size_t ret;
    114 
    115 	if (ISBADF(handle)) {
    116 		errno = EBADF;
    117 		return ((size_t)-1);
    118 	}
    119 
    120 	err = _citrus_iconv_convert(
    121 		(struct _citrus_iconv *)(void *)handle, in, szin, out, szout,
    122 		0, &ret);
    123 	if (err) {
    124 		errno = err;
    125 		ret = (size_t)-1;
    126 	}
    127 
    128 	return (ret);
    129 }
    130 
    131 size_t
    132 __iconv(iconv_t handle, const char **in, size_t *szin, char **out,
    133 	size_t *szout, u_int32_t flags, size_t *invalids)
    134 {
    135 	int err;
    136 	size_t ret;
    137 
    138 	if (ISBADF(handle)) {
    139 		errno = EBADF;
    140 		return ((size_t)-1);
    141 	}
    142 
    143 	err = _citrus_iconv_convert(
    144 		(struct _citrus_iconv *)(void *)handle, in, szin, out, szout,
    145 		flags, &ret);
    146 	if (invalids)
    147 		*invalids = ret;
    148 	if (err) {
    149 		errno = err;
    150 		ret = (size_t)-1;
    151 	}
    152 
    153 	return (ret);
    154 }
    155 
    156 int
    157 __iconv_get_list(char ***rlist, size_t *rsz)
    158 {
    159 	int ret;
    160 
    161 	ret = _citrus_esdb_get_list(rlist, rsz);
    162 	if (ret) {
    163 		errno = ret;
    164 		return -1;
    165 	}
    166 
    167 	return 0;
    168 }
    169 
    170 void
    171 __iconv_free_list(char **list, size_t sz)
    172 {
    173 	_citrus_esdb_free_list(list, sz);
    174 }
    175 
    176 #else
    177 iconv_t
    178 /*ARGSUSED*/
    179 iconv_open(const char *in, const char *out)
    180 {
    181 	errno = EINVAL;
    182 	return ((iconv_t)-1);
    183 }
    184 int
    185 /*ARGSUSED*/
    186 iconv_close(iconv_t handle)
    187 {
    188 	errno = EBADF;
    189 	return (-1);
    190 }
    191 size_t
    192 /*ARGSUSED*/
    193 iconv(iconv_t handle, const char **in, size_t *szin, char **out, size_t *szout)
    194 {
    195 	errno = EBADF;
    196 	return ((size_t)-1);
    197 }
    198 int
    199 /*ARGSUSED*/
    200 __iconv_get_list(char ***rlist, size_t *rsz)
    201 {
    202 	errno = EINVAL;
    203 	return -1;
    204 }
    205 void
    206 /*ARGSUSED*/
    207 __iconv_free_list(char **list, size_t sz)
    208 {
    209 }
    210 #endif
    211