mirror of
https://gitlab.gnome.org/GNOME/libxml2
synced 2025-03-28 21:33:13 +00:00
encoding: Support input-only and output-only converters
Make it possible to open an encoding handler only for input or output. This avoids the creation of unnecessary converters. Should also fix #863.
This commit is contained in:
parent
69b83bb68e
commit
84c6524e26
@ -349,7 +349,10 @@
|
|||||||
<exports symbol='XML_ENC_ERR_MEMORY' type='enum'/>
|
<exports symbol='XML_ENC_ERR_MEMORY' type='enum'/>
|
||||||
<exports symbol='XML_ENC_ERR_SPACE' type='enum'/>
|
<exports symbol='XML_ENC_ERR_SPACE' type='enum'/>
|
||||||
<exports symbol='XML_ENC_ERR_SUCCESS' type='enum'/>
|
<exports symbol='XML_ENC_ERR_SUCCESS' type='enum'/>
|
||||||
|
<exports symbol='XML_ENC_INPUT' type='enum'/>
|
||||||
|
<exports symbol='XML_ENC_OUTPUT' type='enum'/>
|
||||||
<exports symbol='xmlCharEncError' type='typedef'/>
|
<exports symbol='xmlCharEncError' type='typedef'/>
|
||||||
|
<exports symbol='xmlCharEncFlags' type='typedef'/>
|
||||||
<exports symbol='xmlCharEncoding' type='typedef'/>
|
<exports symbol='xmlCharEncoding' type='typedef'/>
|
||||||
<exports symbol='xmlCharEncodingHandler' type='typedef'/>
|
<exports symbol='xmlCharEncodingHandler' type='typedef'/>
|
||||||
<exports symbol='xmlCharEncodingHandlerPtr' type='typedef'/>
|
<exports symbol='xmlCharEncodingHandlerPtr' type='typedef'/>
|
||||||
@ -4528,6 +4531,8 @@ and not by parsing an instance'/>
|
|||||||
<enum name='XML_ENC_ERR_MEMORY' file='encoding' value='-4' type='xmlCharEncError'/>
|
<enum name='XML_ENC_ERR_MEMORY' file='encoding' value='-4' type='xmlCharEncError'/>
|
||||||
<enum name='XML_ENC_ERR_SPACE' file='encoding' value='-3' type='xmlCharEncError'/>
|
<enum name='XML_ENC_ERR_SPACE' file='encoding' value='-3' type='xmlCharEncError'/>
|
||||||
<enum name='XML_ENC_ERR_SUCCESS' file='encoding' value='0' type='xmlCharEncError'/>
|
<enum name='XML_ENC_ERR_SUCCESS' file='encoding' value='0' type='xmlCharEncError'/>
|
||||||
|
<enum name='XML_ENC_INPUT' file='encoding' value='1' type='xmlCharEncFlags'/>
|
||||||
|
<enum name='XML_ENC_OUTPUT' file='encoding' value='2' type='xmlCharEncFlags'/>
|
||||||
<enum name='XML_ENTITY_DECL' file='tree' value='17' type='xmlElementType'/>
|
<enum name='XML_ENTITY_DECL' file='tree' value='17' type='xmlElementType'/>
|
||||||
<enum name='XML_ENTITY_NODE' file='tree' value='6' type='xmlElementType' info='unused'/>
|
<enum name='XML_ENTITY_NODE' file='tree' value='6' type='xmlElementType' info='unused'/>
|
||||||
<enum name='XML_ENTITY_REF_NODE' file='tree' value='5' type='xmlElementType'/>
|
<enum name='XML_ENTITY_REF_NODE' file='tree' value='5' type='xmlElementType'/>
|
||||||
@ -5721,6 +5726,7 @@ crash if you try to modify the tree)'/>
|
|||||||
<info>This is a basic byte in an UTF-8 encoded string. It's unsigned allowing to pinpoint case where char * are assigned to xmlChar * (possibly making serialization back impossible).</info>
|
<info>This is a basic byte in an UTF-8 encoded string. It's unsigned allowing to pinpoint case where char * are assigned to xmlChar * (possibly making serialization back impossible).</info>
|
||||||
</typedef>
|
</typedef>
|
||||||
<typedef name='xmlCharEncError' file='encoding' type='enum'/>
|
<typedef name='xmlCharEncError' file='encoding' type='enum'/>
|
||||||
|
<typedef name='xmlCharEncFlags' file='encoding' type='enum'/>
|
||||||
<typedef name='xmlCharEncoding' file='encoding' type='enum'/>
|
<typedef name='xmlCharEncoding' file='encoding' type='enum'/>
|
||||||
<struct name='xmlCharEncodingHandler' file='encoding' type='struct _xmlCharEncodingHandler'>
|
<struct name='xmlCharEncodingHandler' file='encoding' type='struct _xmlCharEncodingHandler'>
|
||||||
<field name='name' type='char *'/>
|
<field name='name' type='char *'/>
|
||||||
@ -8117,11 +8123,11 @@ crash if you try to modify the tree)'/>
|
|||||||
<arg name='flush' type='int' info='end of input'/>
|
<arg name='flush' type='int' info='end of input'/>
|
||||||
</functype>
|
</functype>
|
||||||
<functype name='xmlCharEncConvImpl' file='encoding' module='encoding'>
|
<functype name='xmlCharEncConvImpl' file='encoding' module='encoding'>
|
||||||
<info>If this function returns XML_ERR_OK, it must fill the @out pointer with an encoding handler. The handler can be obtained from xmlCharEncNewCustomHandler.</info>
|
<info>If this function returns XML_ERR_OK, it must fill the @out pointer with an encoding handler. The handler can be obtained from xmlCharEncNewCustomHandler. @flags can contain XML_ENC_INPUT, XML_ENC_OUTPUT or both.</info>
|
||||||
<return type='int' info='an xmlParserErrors code.'/>
|
<return type='int' info='an xmlParserErrors code.'/>
|
||||||
<arg name='vctxt' type='void *' info='user data'/>
|
<arg name='vctxt' type='void *' info='user data'/>
|
||||||
<arg name='name' type='const char *' info='encoding name'/>
|
<arg name='name' type='const char *' info='encoding name'/>
|
||||||
<arg name='output' type='int' info='true if output encoding, false if input'/>
|
<arg name='flags' type='xmlCharEncFlags' info='bit mask of flags'/>
|
||||||
<arg name='out' type='xmlCharEncodingHandler **' info='pointer to resulting handler'/>
|
<arg name='out' type='xmlCharEncodingHandler **' info='pointer to resulting handler'/>
|
||||||
</functype>
|
</functype>
|
||||||
<function name='xmlCharEncFirstLine' file='encoding' module='encoding'>
|
<function name='xmlCharEncFirstLine' file='encoding' module='encoding'>
|
||||||
@ -8368,10 +8374,10 @@ crash if you try to modify the tree)'/>
|
|||||||
<arg name='cur' type='xmlAttrPtr' info='the first attribute'/>
|
<arg name='cur' type='xmlAttrPtr' info='the first attribute'/>
|
||||||
</function>
|
</function>
|
||||||
<function name='xmlCreateCharEncodingHandler' file='encoding' module='encoding'>
|
<function name='xmlCreateCharEncodingHandler' file='encoding' module='encoding'>
|
||||||
<info>Find or create a handler matching the encoding. The following converters are looked up in order: - Built-in handler (UTF-8, UTF-16, ISO-8859-1, ASCII) - Custom implementation if provided - User-registered global handler (deprecated) - iconv if enabled - ICU if enabled The handler must be closed with xmlCharEncCloseFunc. If the encoding is UTF-8, a NULL handler and no error code will be returned. Available since 2.14.0.</info>
|
<info>Find or create a handler matching the encoding. The following converters are looked up in order: - Built-in handler (UTF-8, UTF-16, ISO-8859-1, ASCII) - Custom implementation if provided - User-registered global handler (deprecated) - iconv if enabled - ICU if enabled The handler must be closed with xmlCharEncCloseFunc. If the encoding is UTF-8, a NULL handler and no error code will be returned. @flags can contain XML_ENC_INPUT, XML_ENC_OUTPUT or both. Available since 2.14.0.</info>
|
||||||
<return type='int' info='XML_ERR_OK, XML_ERR_UNSUPPORTED_ENCODING or another xmlParserErrors error code.'/>
|
<return type='int' info='XML_ERR_OK, XML_ERR_UNSUPPORTED_ENCODING or another xmlParserErrors error code.'/>
|
||||||
<arg name='name' type='const char *' info='a string describing the char encoding.'/>
|
<arg name='name' type='const char *' info='a string describing the char encoding.'/>
|
||||||
<arg name='output' type='int' info='boolean, use handler for output'/>
|
<arg name='flags' type='xmlCharEncFlags' info='bit mask of flags'/>
|
||||||
<arg name='impl' type='xmlCharEncConvImpl' info='a conversion implementation (optional)'/>
|
<arg name='impl' type='xmlCharEncConvImpl' info='a conversion implementation (optional)'/>
|
||||||
<arg name='implCtxt' type='void *' info='user data for conversion implementation (optional)'/>
|
<arg name='implCtxt' type='void *' info='user data for conversion implementation (optional)'/>
|
||||||
<arg name='out' type='xmlCharEncodingHandler **' info='pointer to result'/>
|
<arg name='out' type='xmlCharEncodingHandler **' info='pointer to result'/>
|
||||||
|
89
encoding.c
89
encoding.c
@ -239,12 +239,14 @@ static int nbCharEncodingHandler = 0;
|
|||||||
|
|
||||||
#ifdef LIBXML_ICONV_ENABLED
|
#ifdef LIBXML_ICONV_ENABLED
|
||||||
static int
|
static int
|
||||||
xmlCharEncIconv(const char *name, xmlCharEncodingHandler **out);
|
xmlCharEncIconv(const char *name, xmlCharEncFlags flags,
|
||||||
|
xmlCharEncodingHandler **out);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef LIBXML_ICU_ENABLED
|
#ifdef LIBXML_ICU_ENABLED
|
||||||
static int
|
static int
|
||||||
xmlCharEncUconv(const char *name, xmlCharEncodingHandler **out);
|
xmlCharEncUconv(const char *name, xmlCharEncFlags flags,
|
||||||
|
xmlCharEncodingHandler **out);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/************************************************************************
|
/************************************************************************
|
||||||
@ -794,7 +796,7 @@ free_handler:
|
|||||||
* xmlFindExtraHandler:
|
* xmlFindExtraHandler:
|
||||||
* @norig: name of the char encoding
|
* @norig: name of the char encoding
|
||||||
* @name: potentially aliased name of the encoding
|
* @name: potentially aliased name of the encoding
|
||||||
* @output: boolean, use handler for output
|
* @flags: bit mask of flags
|
||||||
* @impl: a conversion implementation (optional)
|
* @impl: a conversion implementation (optional)
|
||||||
* @implCtxt: user data for conversion implementation (optional)
|
* @implCtxt: user data for conversion implementation (optional)
|
||||||
* @out: pointer to resulting handler
|
* @out: pointer to resulting handler
|
||||||
@ -804,7 +806,7 @@ free_handler:
|
|||||||
* Returns an xmlParserErrors error code.
|
* Returns an xmlParserErrors error code.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
xmlFindExtraHandler(const char *norig, const char *name, int output,
|
xmlFindExtraHandler(const char *norig, const char *name, xmlCharEncFlags flags,
|
||||||
xmlCharEncConvImpl impl, void *implCtxt,
|
xmlCharEncConvImpl impl, void *implCtxt,
|
||||||
xmlCharEncodingHandler **out) {
|
xmlCharEncodingHandler **out) {
|
||||||
/*
|
/*
|
||||||
@ -814,7 +816,7 @@ xmlFindExtraHandler(const char *norig, const char *name, int output,
|
|||||||
* alias resolution.
|
* alias resolution.
|
||||||
*/
|
*/
|
||||||
if (impl != NULL)
|
if (impl != NULL)
|
||||||
return(impl(implCtxt, norig, output, out));
|
return(impl(implCtxt, norig, flags, out));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Deprecated
|
* Deprecated
|
||||||
@ -827,7 +829,8 @@ xmlFindExtraHandler(const char *norig, const char *name, int output,
|
|||||||
|
|
||||||
if (!xmlStrcasecmp((const xmlChar *) name,
|
if (!xmlStrcasecmp((const xmlChar *) name,
|
||||||
(const xmlChar *) h->name)) {
|
(const xmlChar *) h->name)) {
|
||||||
if ((output ? h->output.func : h->input.func) != NULL) {
|
if ((((flags & XML_ENC_INPUT) == 0) || (h->input.func)) &&
|
||||||
|
(((flags & XML_ENC_OUTPUT) == 0) || (h->output.func))) {
|
||||||
*out = h;
|
*out = h;
|
||||||
return(XML_ERR_OK);
|
return(XML_ERR_OK);
|
||||||
}
|
}
|
||||||
@ -837,7 +840,7 @@ xmlFindExtraHandler(const char *norig, const char *name, int output,
|
|||||||
|
|
||||||
#ifdef LIBXML_ICONV_ENABLED
|
#ifdef LIBXML_ICONV_ENABLED
|
||||||
{
|
{
|
||||||
int ret = xmlCharEncIconv(name, out);
|
int ret = xmlCharEncIconv(name, flags, out);
|
||||||
|
|
||||||
if (ret == XML_ERR_OK)
|
if (ret == XML_ERR_OK)
|
||||||
return(XML_ERR_OK);
|
return(XML_ERR_OK);
|
||||||
@ -848,7 +851,7 @@ xmlFindExtraHandler(const char *norig, const char *name, int output,
|
|||||||
|
|
||||||
#ifdef LIBXML_ICU_ENABLED
|
#ifdef LIBXML_ICU_ENABLED
|
||||||
{
|
{
|
||||||
int ret = xmlCharEncUconv(name, out);
|
int ret = xmlCharEncUconv(name, flags, out);
|
||||||
|
|
||||||
if (ret == XML_ERR_OK)
|
if (ret == XML_ERR_OK)
|
||||||
return(XML_ERR_OK);
|
return(XML_ERR_OK);
|
||||||
@ -906,9 +909,15 @@ xmlLookupCharEncodingHandler(xmlCharEncoding enc,
|
|||||||
return(XML_ERR_OK);
|
return(XML_ERR_OK);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (handler->name != NULL)
|
if (handler->name != NULL) {
|
||||||
return(xmlFindExtraHandler(handler->name, handler->name, 0,
|
xmlCharEncFlags flags = XML_ENC_INPUT;
|
||||||
|
|
||||||
|
#ifdef LIBXML_OUTPUT_ENABLED
|
||||||
|
flags |= XML_ENC_OUTPUT;
|
||||||
|
#endif
|
||||||
|
return(xmlFindExtraHandler(handler->name, handler->name, flags,
|
||||||
NULL, NULL, out));
|
NULL, NULL, out));
|
||||||
|
}
|
||||||
|
|
||||||
return(XML_ERR_UNSUPPORTED_ENCODING);
|
return(XML_ERR_UNSUPPORTED_ENCODING);
|
||||||
}
|
}
|
||||||
@ -934,7 +943,7 @@ xmlGetCharEncodingHandler(xmlCharEncoding enc) {
|
|||||||
/**
|
/**
|
||||||
* xmlCreateCharEncodingHandler:
|
* xmlCreateCharEncodingHandler:
|
||||||
* @name: a string describing the char encoding.
|
* @name: a string describing the char encoding.
|
||||||
* @output: boolean, use handler for output
|
* @flags: bit mask of flags
|
||||||
* @impl: a conversion implementation (optional)
|
* @impl: a conversion implementation (optional)
|
||||||
* @implCtxt: user data for conversion implementation (optional)
|
* @implCtxt: user data for conversion implementation (optional)
|
||||||
* @out: pointer to result
|
* @out: pointer to result
|
||||||
@ -953,13 +962,15 @@ xmlGetCharEncodingHandler(xmlCharEncoding enc) {
|
|||||||
* If the encoding is UTF-8, a NULL handler and no error code will
|
* If the encoding is UTF-8, a NULL handler and no error code will
|
||||||
* be returned.
|
* be returned.
|
||||||
*
|
*
|
||||||
|
* @flags can contain XML_ENC_INPUT, XML_ENC_OUTPUT or both.
|
||||||
|
*
|
||||||
* Available since 2.14.0.
|
* Available since 2.14.0.
|
||||||
*
|
*
|
||||||
* Returns XML_ERR_OK, XML_ERR_UNSUPPORTED_ENCODING or another
|
* Returns XML_ERR_OK, XML_ERR_UNSUPPORTED_ENCODING or another
|
||||||
* xmlParserErrors error code.
|
* xmlParserErrors error code.
|
||||||
*/
|
*/
|
||||||
int
|
int
|
||||||
xmlCreateCharEncodingHandler(const char *name, int output,
|
xmlCreateCharEncodingHandler(const char *name, xmlCharEncFlags flags,
|
||||||
xmlCharEncConvImpl impl, void *implCtxt,
|
xmlCharEncConvImpl impl, void *implCtxt,
|
||||||
xmlCharEncodingHandler **out) {
|
xmlCharEncodingHandler **out) {
|
||||||
const xmlCharEncodingHandler *handler;
|
const xmlCharEncodingHandler *handler;
|
||||||
@ -970,7 +981,7 @@ xmlCreateCharEncodingHandler(const char *name, int output,
|
|||||||
return(XML_ERR_ARGUMENT);
|
return(XML_ERR_ARGUMENT);
|
||||||
*out = NULL;
|
*out = NULL;
|
||||||
|
|
||||||
if (name == NULL)
|
if ((name == NULL) || (flags == 0))
|
||||||
return(XML_ERR_ARGUMENT);
|
return(XML_ERR_ARGUMENT);
|
||||||
|
|
||||||
norig = name;
|
norig = name;
|
||||||
@ -986,13 +997,14 @@ xmlCreateCharEncodingHandler(const char *name, int output,
|
|||||||
|
|
||||||
if ((enc > 0) && ((size_t) enc < NUM_DEFAULT_HANDLERS)) {
|
if ((enc > 0) && ((size_t) enc < NUM_DEFAULT_HANDLERS)) {
|
||||||
handler = &defaultHandlers[enc];
|
handler = &defaultHandlers[enc];
|
||||||
if ((output ? handler->output.func : handler->input.func) != NULL) {
|
if ((((flags & XML_ENC_INPUT) == 0) || (handler->input.func)) &&
|
||||||
|
(((flags & XML_ENC_OUTPUT) == 0) || (handler->output.func))) {
|
||||||
*out = (xmlCharEncodingHandler *) handler;
|
*out = (xmlCharEncodingHandler *) handler;
|
||||||
return(XML_ERR_OK);
|
return(XML_ERR_OK);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return(xmlFindExtraHandler(norig, name, output, impl, implCtxt, out));
|
return(xmlFindExtraHandler(norig, name, flags, impl, implCtxt, out));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1022,7 +1034,9 @@ xmlCreateCharEncodingHandler(const char *name, int output,
|
|||||||
int
|
int
|
||||||
xmlOpenCharEncodingHandler(const char *name, int output,
|
xmlOpenCharEncodingHandler(const char *name, int output,
|
||||||
xmlCharEncodingHandler **out) {
|
xmlCharEncodingHandler **out) {
|
||||||
return(xmlCreateCharEncodingHandler(name, output, NULL, NULL, out));
|
xmlCharEncFlags flags = output ? XML_ENC_OUTPUT : XML_ENC_INPUT;
|
||||||
|
|
||||||
|
return(xmlCreateCharEncodingHandler(name, flags, NULL, NULL, out));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1041,6 +1055,7 @@ xmlOpenCharEncodingHandler(const char *name, int output,
|
|||||||
xmlCharEncodingHandlerPtr
|
xmlCharEncodingHandlerPtr
|
||||||
xmlFindCharEncodingHandler(const char *name) {
|
xmlFindCharEncodingHandler(const char *name) {
|
||||||
xmlCharEncodingHandler *ret;
|
xmlCharEncodingHandler *ret;
|
||||||
|
xmlCharEncFlags flags;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This handler shouldn't be used, but we must return a non-NULL
|
* This handler shouldn't be used, but we must return a non-NULL
|
||||||
@ -1051,7 +1066,11 @@ xmlFindCharEncodingHandler(const char *name) {
|
|||||||
return((xmlCharEncodingHandlerPtr)
|
return((xmlCharEncodingHandlerPtr)
|
||||||
&defaultHandlers[XML_CHAR_ENCODING_UTF8]);
|
&defaultHandlers[XML_CHAR_ENCODING_UTF8]);
|
||||||
|
|
||||||
xmlOpenCharEncodingHandler(name, 0, &ret);
|
flags = XML_ENC_INPUT;
|
||||||
|
#ifdef LIBXML_OUTPUT_ENABLED
|
||||||
|
flags |= XML_ENC_OUTPUT;
|
||||||
|
#endif
|
||||||
|
xmlCreateCharEncodingHandler(name, flags, NULL, NULL, &ret);
|
||||||
return(ret);
|
return(ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1131,6 +1150,9 @@ static void
|
|||||||
xmlIconvFree(void *vctxt) {
|
xmlIconvFree(void *vctxt) {
|
||||||
xmlIconvCtxt *ctxt = vctxt;
|
xmlIconvCtxt *ctxt = vctxt;
|
||||||
|
|
||||||
|
if (ctxt == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
if (ctxt->cd != (iconv_t) -1)
|
if (ctxt->cd != (iconv_t) -1)
|
||||||
iconv_close(ctxt->cd);
|
iconv_close(ctxt->cd);
|
||||||
|
|
||||||
@ -1164,7 +1186,9 @@ xmlEncodingMatch(const char *name1, const char *name2) {
|
|||||||
#endif /* FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION */
|
#endif /* FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION */
|
||||||
|
|
||||||
static int
|
static int
|
||||||
xmlCharEncIconv(const char *name, xmlCharEncodingHandler **out) {
|
xmlCharEncIconv(const char *name, xmlCharEncFlags flags,
|
||||||
|
xmlCharEncodingHandler **out) {
|
||||||
|
xmlCharEncConvFunc inFunc = NULL, outFunc = NULL;
|
||||||
xmlIconvCtxt *inputCtxt = NULL, *outputCtxt = NULL;
|
xmlIconvCtxt *inputCtxt = NULL, *outputCtxt = NULL;
|
||||||
iconv_t icv_in;
|
iconv_t icv_in;
|
||||||
iconv_t icv_out;
|
iconv_t icv_out;
|
||||||
@ -1217,6 +1241,7 @@ xmlCharEncIconv(const char *name, xmlCharEncodingHandler **out) {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
if (flags & XML_ENC_INPUT) {
|
||||||
inputCtxt = xmlMalloc(sizeof(xmlIconvCtxt));
|
inputCtxt = xmlMalloc(sizeof(xmlIconvCtxt));
|
||||||
if (inputCtxt == NULL) {
|
if (inputCtxt == NULL) {
|
||||||
ret = XML_ERR_NO_MEMORY;
|
ret = XML_ERR_NO_MEMORY;
|
||||||
@ -1236,6 +1261,10 @@ xmlCharEncIconv(const char *name, xmlCharEncodingHandler **out) {
|
|||||||
}
|
}
|
||||||
inputCtxt->cd = icv_in;
|
inputCtxt->cd = icv_in;
|
||||||
|
|
||||||
|
inFunc = xmlIconvConvert;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (flags & XML_ENC_OUTPUT) {
|
||||||
outputCtxt = xmlMalloc(sizeof(xmlIconvCtxt));
|
outputCtxt = xmlMalloc(sizeof(xmlIconvCtxt));
|
||||||
if (outputCtxt == NULL) {
|
if (outputCtxt == NULL) {
|
||||||
ret = XML_ERR_NO_MEMORY;
|
ret = XML_ERR_NO_MEMORY;
|
||||||
@ -1255,9 +1284,11 @@ xmlCharEncIconv(const char *name, xmlCharEncodingHandler **out) {
|
|||||||
}
|
}
|
||||||
outputCtxt->cd = icv_out;
|
outputCtxt->cd = icv_out;
|
||||||
|
|
||||||
return(xmlCharEncNewCustomHandler(name, xmlIconvConvert, xmlIconvConvert,
|
outFunc = xmlIconvConvert;
|
||||||
xmlIconvFree, inputCtxt, outputCtxt,
|
}
|
||||||
out));
|
|
||||||
|
return(xmlCharEncNewCustomHandler(name, inFunc, outFunc, xmlIconvFree,
|
||||||
|
inputCtxt, outputCtxt, out));
|
||||||
|
|
||||||
error:
|
error:
|
||||||
if (inputCtxt != NULL)
|
if (inputCtxt != NULL)
|
||||||
@ -1444,21 +1475,29 @@ xmlUconvFree(void *vctxt) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
xmlCharEncUconv(const char *name, xmlCharEncodingHandler **out) {
|
xmlCharEncUconv(const char *name, xmlCharEncFlags flags,
|
||||||
|
xmlCharEncodingHandler **out) {
|
||||||
|
xmlCharEncConvFunc inFunc = NULL, outFunc = NULL;
|
||||||
xmlUconvCtxt *ucv_in = NULL;
|
xmlUconvCtxt *ucv_in = NULL;
|
||||||
xmlUconvCtxt *ucv_out = NULL;
|
xmlUconvCtxt *ucv_out = NULL;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
if (flags & XML_ENC_INPUT) {
|
||||||
ret = openIcuConverter(name, 1, &ucv_in);
|
ret = openIcuConverter(name, 1, &ucv_in);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
inFunc = xmlUconvConvert;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (flags & XML_ENC_OUTPUT) {
|
||||||
ret = openIcuConverter(name, 0, &ucv_out);
|
ret = openIcuConverter(name, 0, &ucv_out);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
outFunc = xmlUconvConvert;
|
||||||
|
}
|
||||||
|
|
||||||
return(xmlCharEncNewCustomHandler(name, xmlUconvConvert, xmlUconvConvert,
|
return(xmlCharEncNewCustomHandler(name, inFunc, outFunc, xmlUconvFree,
|
||||||
xmlUconvFree, ucv_in, ucv_out,
|
ucv_in, ucv_out, out));
|
||||||
out));
|
|
||||||
|
|
||||||
error:
|
error:
|
||||||
if (ucv_in != NULL)
|
if (ucv_in != NULL)
|
||||||
|
@ -171,22 +171,29 @@ icuConvCtxtDtor(void *vctxt) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
icuConvImpl(void *vctxt, const char *name, int output,
|
icuConvImpl(void *vctxt, const char *name, xmlCharEncFlags flags,
|
||||||
xmlCharEncodingHandler **out) {
|
xmlCharEncodingHandler **result) {
|
||||||
|
xmlCharEncConvFunc inFunc = NULL, outFunc = NULL;
|
||||||
myConvCtxt *inputCtxt = NULL;
|
myConvCtxt *inputCtxt = NULL;
|
||||||
myConvCtxt *outputCtxt = NULL;
|
myConvCtxt *outputCtxt = NULL;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
if (flags & XML_ENC_INPUT) {
|
||||||
ret = icuOpen(name, 1, &inputCtxt);
|
ret = icuOpen(name, 1, &inputCtxt);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
inFunc = icuConvert;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (flags & XML_ENC_OUTPUT) {
|
||||||
ret = icuOpen(name, 0, &outputCtxt);
|
ret = icuOpen(name, 0, &outputCtxt);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
outFunc = icuConvert;
|
||||||
|
}
|
||||||
|
|
||||||
return xmlCharEncNewCustomHandler(name, icuConvert, icuConvert,
|
return xmlCharEncNewCustomHandler(name, inFunc, outFunc, icuConvCtxtDtor,
|
||||||
icuConvCtxtDtor, inputCtxt, outputCtxt,
|
inputCtxt, outputCtxt, result);
|
||||||
out);
|
|
||||||
|
|
||||||
error:
|
error:
|
||||||
if (inputCtxt != NULL)
|
if (inputCtxt != NULL)
|
||||||
|
@ -83,6 +83,11 @@ typedef enum {
|
|||||||
XML_CHAR_ENCODING_8859_16= 30 /* ISO-8859-16 */
|
XML_CHAR_ENCODING_8859_16= 30 /* ISO-8859-16 */
|
||||||
} xmlCharEncoding;
|
} xmlCharEncoding;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
XML_ENC_INPUT = (1 << 0),
|
||||||
|
XML_ENC_OUTPUT = (1 << 1)
|
||||||
|
} xmlCharEncFlags;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* xmlCharEncodingInputFunc:
|
* xmlCharEncodingInputFunc:
|
||||||
* @out: a pointer to an array of bytes to store the UTF-8 result
|
* @out: a pointer to an array of bytes to store the UTF-8 result
|
||||||
@ -179,17 +184,19 @@ struct _xmlCharEncodingHandler {
|
|||||||
* xmlCharEncConvImpl:
|
* xmlCharEncConvImpl:
|
||||||
* @vctxt: user data
|
* @vctxt: user data
|
||||||
* @name: encoding name
|
* @name: encoding name
|
||||||
* @output: true if output encoding, false if input
|
* @flags: bit mask of flags
|
||||||
* @out: pointer to resulting handler
|
* @out: pointer to resulting handler
|
||||||
*
|
*
|
||||||
* If this function returns XML_ERR_OK, it must fill the @out
|
* If this function returns XML_ERR_OK, it must fill the @out
|
||||||
* pointer with an encoding handler. The handler can be obtained
|
* pointer with an encoding handler. The handler can be obtained
|
||||||
* from xmlCharEncNewCustomHandler.
|
* from xmlCharEncNewCustomHandler.
|
||||||
*
|
*
|
||||||
|
* @flags can contain XML_ENC_INPUT, XML_ENC_OUTPUT or both.
|
||||||
|
*
|
||||||
* Returns an xmlParserErrors code.
|
* Returns an xmlParserErrors code.
|
||||||
*/
|
*/
|
||||||
typedef int
|
typedef int
|
||||||
(*xmlCharEncConvImpl)(void *vctxt, const char *name, int output,
|
(*xmlCharEncConvImpl)(void *vctxt, const char *name, xmlCharEncFlags flags,
|
||||||
xmlCharEncodingHandler **out);
|
xmlCharEncodingHandler **out);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -212,7 +219,7 @@ XMLPUBFUN int
|
|||||||
xmlCharEncodingHandlerPtr *out);
|
xmlCharEncodingHandlerPtr *out);
|
||||||
XMLPUBFUN int
|
XMLPUBFUN int
|
||||||
xmlCreateCharEncodingHandler (const char *name,
|
xmlCreateCharEncodingHandler (const char *name,
|
||||||
int output,
|
xmlCharEncFlags flags,
|
||||||
xmlCharEncConvImpl impl,
|
xmlCharEncConvImpl impl,
|
||||||
void *implCtxt,
|
void *implCtxt,
|
||||||
xmlCharEncodingHandlerPtr *out);
|
xmlCharEncodingHandlerPtr *out);
|
||||||
|
@ -1142,7 +1142,7 @@ xmlDetectEBCDIC(xmlParserCtxtPtr ctxt, xmlCharEncodingHandlerPtr *hout) {
|
|||||||
* To detect the EBCDIC code page, we convert the first 200 bytes
|
* To detect the EBCDIC code page, we convert the first 200 bytes
|
||||||
* to IBM037 (EBCDIC-US) and try to find the encoding declaration.
|
* to IBM037 (EBCDIC-US) and try to find the encoding declaration.
|
||||||
*/
|
*/
|
||||||
res = xmlCreateCharEncodingHandler("IBM037", /* output */ 0,
|
res = xmlCreateCharEncodingHandler("IBM037", XML_ENC_INPUT,
|
||||||
ctxt->convImpl, ctxt->convCtxt, &handler);
|
ctxt->convImpl, ctxt->convCtxt, &handler);
|
||||||
if (res != 0)
|
if (res != 0)
|
||||||
return(res);
|
return(res);
|
||||||
@ -1188,7 +1188,7 @@ xmlDetectEBCDIC(xmlParserCtxtPtr ctxt, xmlCharEncodingHandlerPtr *hout) {
|
|||||||
out[i] = 0;
|
out[i] = 0;
|
||||||
xmlCharEncCloseFunc(handler);
|
xmlCharEncCloseFunc(handler);
|
||||||
res = xmlCreateCharEncodingHandler((char *) out + start,
|
res = xmlCreateCharEncodingHandler((char *) out + start,
|
||||||
/* output */ 0, ctxt->convImpl, ctxt->convCtxt,
|
XML_ENC_INPUT, ctxt->convImpl, ctxt->convCtxt,
|
||||||
&handler);
|
&handler);
|
||||||
if (res != 0)
|
if (res != 0)
|
||||||
return(res);
|
return(res);
|
||||||
@ -1202,7 +1202,7 @@ done:
|
|||||||
* Encoding handlers are stateful, so we have to recreate them.
|
* Encoding handlers are stateful, so we have to recreate them.
|
||||||
*/
|
*/
|
||||||
xmlCharEncCloseFunc(handler);
|
xmlCharEncCloseFunc(handler);
|
||||||
res = xmlCreateCharEncodingHandler("IBM037", /* output */ 0,
|
res = xmlCreateCharEncodingHandler("IBM037", XML_ENC_INPUT,
|
||||||
ctxt->convImpl, ctxt->convCtxt, &handler);
|
ctxt->convImpl, ctxt->convCtxt, &handler);
|
||||||
if (res != 0)
|
if (res != 0)
|
||||||
return(res);
|
return(res);
|
||||||
@ -1265,7 +1265,7 @@ xmlSwitchInputEncodingName(xmlParserCtxtPtr ctxt, xmlParserInputPtr input,
|
|||||||
if (encoding == NULL)
|
if (encoding == NULL)
|
||||||
return(-1);
|
return(-1);
|
||||||
|
|
||||||
res = xmlCreateCharEncodingHandler(encoding, /* output */ 0,
|
res = xmlCreateCharEncodingHandler(encoding, XML_ENC_INPUT,
|
||||||
ctxt->convImpl, ctxt->convCtxt, &handler);
|
ctxt->convImpl, ctxt->convCtxt, &handler);
|
||||||
if (res == XML_ERR_UNSUPPORTED_ENCODING) {
|
if (res == XML_ERR_UNSUPPORTED_ENCODING) {
|
||||||
xmlWarningMsg(ctxt, XML_ERR_UNSUPPORTED_ENCODING,
|
xmlWarningMsg(ctxt, XML_ERR_UNSUPPORTED_ENCODING,
|
||||||
@ -1618,7 +1618,7 @@ xmlSetDeclaredEncoding(xmlParserCtxtPtr ctxt, xmlChar *encoding) {
|
|||||||
* declaration.
|
* declaration.
|
||||||
*/
|
*/
|
||||||
res = xmlCreateCharEncodingHandler((const char *) encoding,
|
res = xmlCreateCharEncodingHandler((const char *) encoding,
|
||||||
/* output */ 0, ctxt->convImpl, ctxt->convCtxt, &handler);
|
XML_ENC_INPUT, ctxt->convImpl, ctxt->convCtxt, &handler);
|
||||||
if (res != XML_ERR_OK) {
|
if (res != XML_ERR_OK) {
|
||||||
xmlFatalErr(ctxt, res, (const char *) encoding);
|
xmlFatalErr(ctxt, res, (const char *) encoding);
|
||||||
xmlFree(encoding);
|
xmlFree(encoding);
|
||||||
|
10
testparser.c
10
testparser.c
@ -1064,20 +1064,20 @@ rot13ConvCtxtDtor(void *vctxt) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
rot13ConvImpl(void *vctxt ATTRIBUTE_UNUSED, const char *name, int output,
|
rot13ConvImpl(void *vctxt ATTRIBUTE_UNUSED, const char *name,
|
||||||
xmlCharEncodingHandler **out) {
|
xmlCharEncFlags flags, xmlCharEncodingHandler **out) {
|
||||||
int *inputCtxt;
|
int *inputCtxt;
|
||||||
|
|
||||||
if (strcmp(name, "rot13") != 0)
|
if (strcmp(name, "rot13") != 0)
|
||||||
return xmlCreateCharEncodingHandler(name, output, NULL, NULL, out);
|
return xmlCreateCharEncodingHandler(name, flags, NULL, NULL, out);
|
||||||
|
|
||||||
if (output)
|
if (flags & XML_ENC_OUTPUT)
|
||||||
return XML_ERR_UNSUPPORTED_ENCODING;
|
return XML_ERR_UNSUPPORTED_ENCODING;
|
||||||
|
|
||||||
inputCtxt = xmlMalloc(sizeof(*inputCtxt));
|
inputCtxt = xmlMalloc(sizeof(*inputCtxt));
|
||||||
*inputCtxt = 13;
|
*inputCtxt = 13;
|
||||||
|
|
||||||
return xmlCharEncNewCustomHandler(name, rot13Convert, rot13Convert,
|
return xmlCharEncNewCustomHandler(name, rot13Convert, NULL,
|
||||||
rot13ConvCtxtDtor, inputCtxt, NULL,
|
rot13ConvCtxtDtor, inputCtxt, NULL,
|
||||||
out);
|
out);
|
||||||
}
|
}
|
||||||
|
@ -523,6 +523,8 @@ for enum in enums:
|
|||||||
#
|
#
|
||||||
if (name == None) or ((name not in argtypes) and (name not in rettypes)):
|
if (name == None) or ((name not in argtypes) and (name not in rettypes)):
|
||||||
continue;
|
continue;
|
||||||
|
if name == 'xmlCharEncFlags':
|
||||||
|
continue
|
||||||
define = 0
|
define = 0
|
||||||
|
|
||||||
if (name in argtypes) and is_known_param_type(name) == 0:
|
if (name in argtypes) and is_known_param_type(name) == 0:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user