1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
|
/*
* fromucs.c - convert Unicode to other character sets.
*/
#include "charset.h"
#include "internal.h"
struct charset_emit_param {
char *output;
int outlen;
int stopped;
};
static void charset_emit(void *ctx, long int output)
{
struct charset_emit_param *param = (struct charset_emit_param *)ctx;
if (param->outlen > 0) {
*param->output++ = output;
param->outlen--;
} else {
param->stopped = 1;
}
}
int charset_from_unicode(const wchar_t **input, int *inlen,
char *output, int outlen,
int charset, charset_state *state, int *error)
{
charset_spec const *spec = charset_find_spec(charset);
charset_state localstate = CHARSET_INIT_STATE;
struct charset_emit_param param;
int locallen;
if (!input) {
locallen = 1;
inlen = &locallen;
}
param.output = output;
param.outlen = outlen;
param.stopped = 0;
if (state)
localstate = *state; /* structure copy */
if (error)
*error = FALSE;
while (*inlen > 0) {
int lenbefore = param.output - output;
int ret;
if (input)
ret = spec->write(spec, **input, &localstate,
charset_emit, ¶m);
else
ret = spec->write(spec, -1, &localstate, charset_emit, ¶m);
if (error && !ret) {
/*
* We have hit a difficult character, which the user
* wants to know about. Leave now.
*/
*error = TRUE;
return lenbefore;
}
if (param.stopped) {
/*
* The emit function has _tried_ to output some
* characters, but ran up against the end of the
* buffer. Leave immediately, and return what happened
* _before_ attempting to process this character.
*/
return lenbefore;
}
if (state)
*state = localstate; /* structure copy */
if (input)
(*input)++;
(*inlen)--;
}
return param.output - output;
}
|