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