Home | History | Annotate | Line # | Download | only in iconv
iconv.c revision 1.8.2.1
      1 /*	$NetBSD: iconv.c,v 1.8.2.1 2009/05/13 19:18:24 jym 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.8.2.1 2009/05/13 19:18:24 jym 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 != 0) {
     82                 if (p[0] == '/' && p[1] == '/') {
     83                         *p = '\0';
     84                         break;
     85                 }
     86                 p++;
     87         }
     88 
     89 	ret = _citrus_iconv_open(&handle, _PATH_ICONV, in, out_truncated);
     90 	free(out_truncated);
     91 	if (ret) {
     92 		errno = ret == ENOENT? EINVAL : ret;
     93 		return ((iconv_t)-1);
     94 	}
     95 
     96 	return ((iconv_t)(void *)handle);
     97 }
     98 
     99 int
    100 iconv_close(iconv_t handle)
    101 {
    102 	if (ISBADF(handle)) {
    103 		errno = EBADF;
    104 		return (-1);
    105 	}
    106 
    107 	_citrus_iconv_close((struct _citrus_iconv *)(void *)handle);
    108 
    109 	return (0);
    110 }
    111 
    112 size_t
    113 iconv(iconv_t handle, const char **in, size_t *szin, char **out, size_t *szout)
    114 {
    115 	int err;
    116 	size_t ret;
    117 
    118 	if (ISBADF(handle)) {
    119 		errno = EBADF;
    120 		return ((size_t)-1);
    121 	}
    122 
    123 	err = _citrus_iconv_convert(
    124 		(struct _citrus_iconv *)(void *)handle, in, szin, out, szout,
    125 		0, &ret);
    126 	if (err) {
    127 		errno = err;
    128 		ret = (size_t)-1;
    129 	}
    130 
    131 	return (ret);
    132 }
    133 
    134 size_t
    135 __iconv(iconv_t handle, const char **in, size_t *szin, char **out,
    136 	size_t *szout, u_int32_t flags, size_t *invalids)
    137 {
    138 	int err;
    139 	size_t ret;
    140 
    141 	if (ISBADF(handle)) {
    142 		errno = EBADF;
    143 		return ((size_t)-1);
    144 	}
    145 
    146 	err = _citrus_iconv_convert(
    147 		(struct _citrus_iconv *)(void *)handle, in, szin, out, szout,
    148 		flags, &ret);
    149 	if (invalids)
    150 		*invalids = ret;
    151 	if (err) {
    152 		errno = err;
    153 		ret = (size_t)-1;
    154 	}
    155 
    156 	return (ret);
    157 }
    158 
    159 int
    160 __iconv_get_list(char ***rlist, size_t *rsz)
    161 {
    162 	int ret;
    163 
    164 	ret = _citrus_esdb_get_list(rlist, rsz);
    165 	if (ret) {
    166 		errno = ret;
    167 		return -1;
    168 	}
    169 
    170 	return 0;
    171 }
    172 
    173 void
    174 __iconv_free_list(char **list, size_t sz)
    175 {
    176 	_citrus_esdb_free_list(list, sz);
    177 }
    178 
    179 #else
    180 iconv_t
    181 /*ARGSUSED*/
    182 iconv_open(const char *in, const char *out)
    183 {
    184 	errno = EINVAL;
    185 	return ((iconv_t)-1);
    186 }
    187 int
    188 /*ARGSUSED*/
    189 iconv_close(iconv_t handle)
    190 {
    191 	errno = EBADF;
    192 	return (-1);
    193 }
    194 size_t
    195 /*ARGSUSED*/
    196 iconv(iconv_t handle, const char **in, size_t *szin, char **out, size_t *szout)
    197 {
    198 	errno = EBADF;
    199 	return ((size_t)-1);
    200 }
    201 int
    202 /*ARGSUSED*/
    203 __iconv_get_list(char ***rlist, size_t *rsz)
    204 {
    205 	errno = EINVAL;
    206 	return -1;
    207 }
    208 void
    209 /*ARGSUSED*/
    210 __iconv_free_list(char **list, size_t sz)
    211 {
    212 }
    213 #endif
    214