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