2002-12-31 12:20:34 +00:00
|
|
|
/*
|
|
|
|
* fromucs.c - convert Unicode to other character sets.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "charset.h"
|
|
|
|
#include "internal.h"
|
|
|
|
|
|
|
|
struct charset_emit_param {
|
|
|
|
char *output;
|
|
|
|
int outlen;
|
|
|
|
const char *errstr;
|
|
|
|
int errlen;
|
|
|
|
int stopped;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void charset_emit(void *ctx, long int output)
|
|
|
|
{
|
|
|
|
struct charset_emit_param *param = (struct charset_emit_param *)ctx;
|
|
|
|
char outval;
|
|
|
|
char const *p;
|
|
|
|
int outlen;
|
|
|
|
|
|
|
|
if (output == ERROR) {
|
2019-09-08 19:29:00 +00:00
|
|
|
p = param->errstr;
|
|
|
|
outlen = param->errlen;
|
2002-12-31 12:20:34 +00:00
|
|
|
} else {
|
2019-09-08 19:29:00 +00:00
|
|
|
outval = output;
|
|
|
|
p = &outval;
|
|
|
|
outlen = 1;
|
2002-12-31 12:20:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (param->outlen >= outlen) {
|
2019-09-08 19:29:00 +00:00
|
|
|
while (outlen > 0) {
|
|
|
|
*param->output++ = *p++;
|
|
|
|
param->outlen--;
|
|
|
|
outlen--;
|
|
|
|
}
|
2002-12-31 12:20:34 +00:00
|
|
|
} else {
|
2019-09-08 19:29:00 +00:00
|
|
|
param->stopped = 1;
|
2002-12-31 12:20:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-16 19:18:52 +00:00
|
|
|
int charset_from_unicode(const wchar_t **input, int *inlen,
|
|
|
|
char *output, int outlen,
|
2019-09-08 19:29:00 +00:00
|
|
|
int charset, charset_state *state,
|
|
|
|
const char *errstr, int errlen)
|
2002-12-31 12:20:34 +00:00
|
|
|
{
|
|
|
|
charset_spec const *spec = charset_find_spec(charset);
|
|
|
|
charset_state localstate;
|
|
|
|
struct charset_emit_param param;
|
|
|
|
|
|
|
|
param.output = output;
|
|
|
|
param.outlen = outlen;
|
|
|
|
param.stopped = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* charset_emit will expect a valid errstr.
|
|
|
|
*/
|
|
|
|
if (!errstr) {
|
2019-09-08 19:29:00 +00:00
|
|
|
/* *shrug* this is good enough, and consistent across all SBCS... */
|
|
|
|
param.errstr = ".";
|
|
|
|
param.errlen = 1;
|
2002-12-31 12:20:34 +00:00
|
|
|
}
|
|
|
|
param.errstr = errstr;
|
|
|
|
param.errlen = errlen;
|
|
|
|
|
|
|
|
if (!state) {
|
2019-09-08 19:29:00 +00:00
|
|
|
localstate.s0 = 0;
|
2002-12-31 12:20:34 +00:00
|
|
|
} else {
|
2019-09-08 19:29:00 +00:00
|
|
|
localstate = *state; /* structure copy */
|
2002-12-31 12:20:34 +00:00
|
|
|
}
|
|
|
|
state = &localstate;
|
|
|
|
|
|
|
|
while (*inlen > 0) {
|
2019-09-08 19:29:00 +00:00
|
|
|
int lenbefore = param.output - output;
|
|
|
|
spec->write(spec, **input, &localstate, charset_emit, ¶m);
|
|
|
|
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 */
|
|
|
|
(*input)++;
|
|
|
|
(*inlen)--;
|
2002-12-31 12:20:34 +00:00
|
|
|
}
|
|
|
|
return param.output - output;
|
|
|
|
}
|