2
Unix SMB/CIFS implementation.
5
Copyright (C) Jelmer Vernooij 2003
6
Based on iconv/icon_prog.c from the GNU C Library,
7
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1998.
9
This program is free software; you can redistribute it and/or modify
10
it under the terms of the GNU General Public License as published by
11
the Free Software Foundation; either version 3 of the License, or
12
(at your option) any later version.
14
This program is distributed in the hope that it will be useful,
15
but WITHOUT ANY WARRANTY; without even the implied warranty of
16
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17
GNU General Public License for more details.
19
You should have received a copy of the GNU General Public License
20
along with this program. If not, see <http://www.gnu.org/licenses/>.
25
static int process_block (smb_iconv_t cd, const char *addr, size_t len, FILE *output)
27
#define OUTBUF_SIZE 32768
28
const char *start = addr;
29
char outbuf[OUTBUF_SIZE];
38
n = smb_iconv (cd, &addr, &len, &outptr, &outlen);
42
/* We have something to write out. */
43
int errno_save = errno;
45
if (fwrite (outbuf, 1, outptr - outbuf, output)
46
< (size_t) (outptr - outbuf)
49
/* Error occurred while printing the result. */
50
DEBUG (0, ("conversion stopped due to problem in writing the output"));
59
/* iconv() ran into a problem. */
63
DEBUG(0,("illegal input sequence at position %ld",
64
(long) (addr - start)));
68
incomplete character or shift sequence at end of buffer"));
71
DEBUG(0, ("internal error (illegal descriptor)"));
74
DEBUG(0, ("unknown iconv() error %d", errno));
86
static int process_fd (iconv_t cd, int fd, FILE *output)
88
/* we have a problem with reading from a descriptor since we must not
89
provide the iconv() function an incomplete character or shift
90
sequence at the end of the buffer. Since we have to deal with
91
arbitrary encodings we must read the whole text in a buffer and
92
process it in one step. */
93
static char *inbuf = NULL;
94
static size_t maxlen = 0;
98
while (actlen < maxlen)
100
ssize_t n = read (fd, inptr, maxlen - actlen);
103
/* No more text to read. */
108
/* Error while reading. */
109
DEBUG(0, ("error while reading the input"));
117
if (actlen == maxlen)
123
/* Increase the buffer. */
124
new_inbuf = (char *) realloc (inbuf, maxlen + 32768);
125
if (new_inbuf == NULL)
127
DEBUG(0, ("unable to allocate buffer for input"));
132
inptr = inbuf + actlen;
136
n = read (fd, inptr, maxlen - actlen);
139
/* No more text to read. */
144
/* Error while reading. */
145
DEBUG(0, ("error while reading the input"));
152
while (actlen < maxlen);
155
/* Break again so we leave both loops. */
159
/* Now we have all the input in the buffer. Process it in one run. */
160
return process_block (cd, inbuf, actlen, output);
165
int main(int argc, char *argv[])
167
const char *file = NULL;
171
const char *preload_modules[] = {NULL, NULL};
176
/* make sure the vars that get altered (4th field) are in
177
a fixed location or certain compilers complain */
179
struct poptOption long_options[] = {
181
{ "from-code", 'f', POPT_ARG_STRING, &from, 0, "Encoding of original text" },
182
{ "to-code", 't', POPT_ARG_STRING, &to, 0, "Encoding for output" },
183
{ "output", 'o', POPT_ARG_STRING, &output, 0, "Write output to this file" },
184
{ "preload-modules", 'p', POPT_ARG_STRING, &preload_modules[0], 0, "Modules to load" },
190
pc = poptGetContext("smbiconv", argc, (const char **) argv,
193
poptSetOtherOptionHelp(pc, "[FILE] ...");
195
while(poptGetNextOpt(pc) != -1);
197
if (preload_modules[0]) smb_load_modules(preload_modules);
200
out = fopen(output, "w");
203
DEBUG(0, ("Can't open output file '%s': %s, exiting...\n", output, strerror(errno)));
208
cd = smb_iconv_open_ex(tctx, to, from, lp_parm_bool(tctx->lp_ctx, NULL, "iconv", "native", true));
210
DEBUG(0,("unable to find from or to encoding, exiting...\n"));
211
if (out != stdout) fclose(out);
215
while((file = poptGetArg(pc))) {
216
if(strcmp(file, "-") == 0) fd = 0;
218
fd = open(file, O_RDONLY);
221
DEBUG(0, ("Can't open input file '%s': %s, ignoring...\n", file, strerror(errno)));
226
/* Loop thru all arguments */
227
process_fd(cd, fd, out);