|
@@ -1,1334 +0,0 @@
|
|
|
-/* Copyright information is at end of file */
|
|
|
|
|
-
|
|
|
|
|
-#include "xmlrpc_config.h"
|
|
|
|
|
-
|
|
|
|
|
-#include <stddef.h>
|
|
|
|
|
-#include <stdlib.h>
|
|
|
|
|
-#include <stdarg.h>
|
|
|
|
|
-#include <string.h>
|
|
|
|
|
-
|
|
|
|
|
-#include "bool.h"
|
|
|
|
|
-#include "xmlrpc.h"
|
|
|
|
|
-#include "xmlrpc_int.h"
|
|
|
|
|
-
|
|
|
|
|
-/* Borrowed from Python 1.5.2.
|
|
|
|
|
-** MPW pushes 'extended' for float and double types with varargs */
|
|
|
|
|
-#ifdef MPW
|
|
|
|
|
-typedef extended va_double;
|
|
|
|
|
-#else
|
|
|
|
|
-typedef double va_double;
|
|
|
|
|
-#endif
|
|
|
|
|
-
|
|
|
|
|
-/* Borrowed from Python 1.5.2.
|
|
|
|
|
-** Python copies its va_list objects before using them in certain
|
|
|
|
|
-** tricky fashions. We don't why Python does this, but since we're
|
|
|
|
|
-** abusing our va_list objects in a similar fashion, we'll copy them
|
|
|
|
|
-** too. */
|
|
|
|
|
-#ifdef HAS_VA_COPY
|
|
|
|
|
-# define VA_LIST_COPY(dest,src) va_copy((dest), (src))
|
|
|
|
|
-#else
|
|
|
|
|
-# if VA_LIST_IS_ARRAY
|
|
|
|
|
-# define VA_LIST_COPY(dest,src) memcpy((dest), (src), sizeof(va_list))
|
|
|
|
|
-# else
|
|
|
|
|
-# define VA_LIST_COPY(dest,src) ((dest) = (src))
|
|
|
|
|
-# endif
|
|
|
|
|
-#endif
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-destroyValue(xmlrpc_value * const valueP) {
|
|
|
|
|
-
|
|
|
|
|
- /* First, we need to destroy this value's contents, if any. */
|
|
|
|
|
- switch (valueP->_type) {
|
|
|
|
|
- case XMLRPC_TYPE_INT:
|
|
|
|
|
- case XMLRPC_TYPE_BOOL:
|
|
|
|
|
- case XMLRPC_TYPE_DOUBLE:
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case XMLRPC_TYPE_ARRAY:
|
|
|
|
|
- xmlrpc_destroyArrayContents(valueP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case XMLRPC_TYPE_STRING:
|
|
|
|
|
-#ifdef HAVE_UNICODE_WCHAR
|
|
|
|
|
- if (valueP->_wcs_block)
|
|
|
|
|
- xmlrpc_mem_block_free(valueP->_wcs_block);
|
|
|
|
|
-#endif /* HAVE_UNICODE_WCHAR */
|
|
|
|
|
- /* Fall through. */
|
|
|
|
|
-
|
|
|
|
|
- case XMLRPC_TYPE_DATETIME:
|
|
|
|
|
- case XMLRPC_TYPE_BASE64:
|
|
|
|
|
- xmlrpc_mem_block_clean(&valueP->_block);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case XMLRPC_TYPE_STRUCT:
|
|
|
|
|
- xmlrpc_destroyStruct(valueP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case XMLRPC_TYPE_C_PTR:
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case XMLRPC_TYPE_DEAD:
|
|
|
|
|
- XMLRPC_ASSERT(FALSE); /* Can't happen, per entry conditions */
|
|
|
|
|
-
|
|
|
|
|
- default:
|
|
|
|
|
- XMLRPC_ASSERT(FALSE); /* There are no other possible values */
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- /* Next, we mark this value as invalid, to help catch refcount
|
|
|
|
|
- ** errors. */
|
|
|
|
|
- valueP->_type = XMLRPC_TYPE_DEAD;
|
|
|
|
|
-
|
|
|
|
|
- /* Finally, we destroy the value itself. */
|
|
|
|
|
- free(valueP);
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-/*=========================================================================
|
|
|
|
|
-** Reference Counting
|
|
|
|
|
-**=========================================================================
|
|
|
|
|
-** Some simple reference-counting code. The xmlrpc_DECREF routine is in
|
|
|
|
|
-** charge of destroying values when their reference count equals zero.
|
|
|
|
|
-*/
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_INCREF (xmlrpc_value * const valueP) {
|
|
|
|
|
-
|
|
|
|
|
- XMLRPC_ASSERT_VALUE_OK(valueP);
|
|
|
|
|
- XMLRPC_ASSERT(valueP->_refcount > 0);
|
|
|
|
|
-
|
|
|
|
|
- valueP->_refcount++;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_DECREF (xmlrpc_value * const valueP) {
|
|
|
|
|
-
|
|
|
|
|
- XMLRPC_ASSERT_VALUE_OK(valueP);
|
|
|
|
|
- XMLRPC_ASSERT(valueP->_refcount > 0);
|
|
|
|
|
- XMLRPC_ASSERT(valueP->_type != XMLRPC_TYPE_DEAD);
|
|
|
|
|
-
|
|
|
|
|
- valueP->_refcount--;
|
|
|
|
|
-
|
|
|
|
|
- /* If we have no more refs, we need to deallocate this value. */
|
|
|
|
|
- if (valueP->_refcount == 0)
|
|
|
|
|
- destroyValue(valueP);
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-/*=========================================================================
|
|
|
|
|
- Utiltiies
|
|
|
|
|
-=========================================================================*/
|
|
|
|
|
-
|
|
|
|
|
-static const char *
|
|
|
|
|
-typeName(xmlrpc_type const type) {
|
|
|
|
|
-
|
|
|
|
|
- switch(type) {
|
|
|
|
|
-
|
|
|
|
|
- case XMLRPC_TYPE_INT: return "INT";
|
|
|
|
|
- case XMLRPC_TYPE_BOOL: return "BOOL";
|
|
|
|
|
- case XMLRPC_TYPE_DOUBLE: return "DOUBLE";
|
|
|
|
|
- case XMLRPC_TYPE_DATETIME: return "DATETIME";
|
|
|
|
|
- case XMLRPC_TYPE_STRING: return "STRING";
|
|
|
|
|
- case XMLRPC_TYPE_BASE64: return "BASE64";
|
|
|
|
|
- case XMLRPC_TYPE_ARRAY: return "ARRAY";
|
|
|
|
|
- case XMLRPC_TYPE_STRUCT: return "STRUCT";
|
|
|
|
|
- case XMLRPC_TYPE_C_PTR: return "C_PTR";
|
|
|
|
|
- case XMLRPC_TYPE_DEAD: return "DEAD";
|
|
|
|
|
- default: return "???";
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-verifyNoNulls(xmlrpc_env * const envP,
|
|
|
|
|
- const char * const contents,
|
|
|
|
|
- unsigned int const len) {
|
|
|
|
|
-/*----------------------------------------------------------------------------
|
|
|
|
|
- Verify that the character array 'contents', which is 'len' bytes long,
|
|
|
|
|
- does not contain any NUL characters, which means it can be made into
|
|
|
|
|
- a passable ASCIIZ string just by adding a terminating NUL.
|
|
|
|
|
-
|
|
|
|
|
- Fail if the array contains a NUL.
|
|
|
|
|
------------------------------------------------------------------------------*/
|
|
|
|
|
- unsigned int i;
|
|
|
|
|
-
|
|
|
|
|
- for (i = 0; i < len && !envP->fault_occurred; i++)
|
|
|
|
|
- if (contents[i] == '\0')
|
|
|
|
|
- xmlrpc_env_set_fault_formatted(
|
|
|
|
|
- envP, XMLRPC_TYPE_ERROR,
|
|
|
|
|
- "String must not contain NUL characters");
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-verifyNoNullsW(xmlrpc_env * const envP,
|
|
|
|
|
- const wchar_t * const contents,
|
|
|
|
|
- unsigned int const len) {
|
|
|
|
|
-/*----------------------------------------------------------------------------
|
|
|
|
|
- Same as verifyNoNulls(), but for wide characters.
|
|
|
|
|
------------------------------------------------------------------------------*/
|
|
|
|
|
- unsigned int i;
|
|
|
|
|
-
|
|
|
|
|
- for (i = 0; i < len && !envP->fault_occurred; i++)
|
|
|
|
|
- if (contents[i] == '\0')
|
|
|
|
|
- xmlrpc_env_set_fault_formatted(
|
|
|
|
|
- envP, XMLRPC_TYPE_ERROR,
|
|
|
|
|
- "String must not contain NUL characters");
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-validateType(xmlrpc_env * const envP,
|
|
|
|
|
- const xmlrpc_value * const valueP,
|
|
|
|
|
- xmlrpc_type const expectedType) {
|
|
|
|
|
-
|
|
|
|
|
- if (valueP->_type != expectedType) {
|
|
|
|
|
- xmlrpc_env_set_fault_formatted(
|
|
|
|
|
- envP, XMLRPC_TYPE_ERROR, "Value of type %s supplied where "
|
|
|
|
|
- "type %s was expected.",
|
|
|
|
|
- typeName(valueP->_type), typeName(expectedType));
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-/*=========================================================================
|
|
|
|
|
- Extracting XML-RPC value
|
|
|
|
|
-===========================================================================
|
|
|
|
|
- These routines extract XML-RPC values into ordinary C data types.
|
|
|
|
|
-
|
|
|
|
|
- For array and struct values, see the separates files xmlrpc_array.c
|
|
|
|
|
- and xmlrpc_struct.c.
|
|
|
|
|
-=========================================================================*/
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_read_int(xmlrpc_env * const envP,
|
|
|
|
|
- const xmlrpc_value * const valueP,
|
|
|
|
|
- xmlrpc_int32 * const intValueP) {
|
|
|
|
|
-
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_INT);
|
|
|
|
|
- if (!envP->fault_occurred)
|
|
|
|
|
- *intValueP = valueP->_value.i;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_read_double(xmlrpc_env * const envP,
|
|
|
|
|
- const xmlrpc_value * const valueP,
|
|
|
|
|
- xmlrpc_double * const doubleValueP) {
|
|
|
|
|
-
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_DOUBLE);
|
|
|
|
|
- if (!envP->fault_occurred)
|
|
|
|
|
- *doubleValueP = valueP->_value.d;
|
|
|
|
|
-
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_read_bool(xmlrpc_env * const envP,
|
|
|
|
|
- const xmlrpc_value * const valueP,
|
|
|
|
|
- xmlrpc_bool * const boolValueP) {
|
|
|
|
|
-
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_BOOL);
|
|
|
|
|
- if (!envP->fault_occurred)
|
|
|
|
|
- *boolValueP = valueP->_value.b;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_read_string(xmlrpc_env * const envP,
|
|
|
|
|
- const xmlrpc_value * const valueP,
|
|
|
|
|
- const char ** const stringValueP) {
|
|
|
|
|
-/*----------------------------------------------------------------------------
|
|
|
|
|
- Read the value of an XML-RPC string an ASCIIZ string.
|
|
|
|
|
-
|
|
|
|
|
- Fail if the string contains null characters (which means it wasn't
|
|
|
|
|
- really a string, but XML-RPC doesn't seem to understand what a string
|
|
|
|
|
- is, and such values are possible).
|
|
|
|
|
------------------------------------------------------------------------------*/
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_STRING);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- size_t const size =
|
|
|
|
|
- XMLRPC_MEMBLOCK_SIZE(char, &valueP->_block);
|
|
|
|
|
- const char * const contents =
|
|
|
|
|
- XMLRPC_MEMBLOCK_CONTENTS(char, &valueP->_block);
|
|
|
|
|
-
|
|
|
|
|
- verifyNoNulls(envP, contents, (unsigned int)size);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- char * stringValue;
|
|
|
|
|
-
|
|
|
|
|
- stringValue = malloc(size+1);
|
|
|
|
|
- if (stringValue == NULL)
|
|
|
|
|
- xmlrpc_env_set_fault_formatted(
|
|
|
|
|
- envP, XMLRPC_INTERNAL_ERROR, "Unable to allocate space "
|
|
|
|
|
- "for %u-character string", size);
|
|
|
|
|
- else {
|
|
|
|
|
- memcpy(stringValue,
|
|
|
|
|
- XMLRPC_MEMBLOCK_CONTENTS(char, &valueP->_block), size);
|
|
|
|
|
- stringValue[size] = '\0';
|
|
|
|
|
-
|
|
|
|
|
- *stringValueP = stringValue;
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_read_string_lp(xmlrpc_env * const envP,
|
|
|
|
|
- const xmlrpc_value * const valueP,
|
|
|
|
|
- unsigned int * const lengthP,
|
|
|
|
|
- const char ** const stringValueP) {
|
|
|
|
|
-
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_STRING);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- size_t const size =
|
|
|
|
|
- XMLRPC_MEMBLOCK_SIZE(char, &valueP->_block);
|
|
|
|
|
- const char * const contents =
|
|
|
|
|
- XMLRPC_MEMBLOCK_CONTENTS(char, &valueP->_block);
|
|
|
|
|
-
|
|
|
|
|
- char * stringValue;
|
|
|
|
|
-
|
|
|
|
|
- stringValue = malloc(size);
|
|
|
|
|
- if (stringValue == NULL)
|
|
|
|
|
- xmlrpc_env_set_fault_formatted(
|
|
|
|
|
- envP, XMLRPC_INTERNAL_ERROR, "Unable to allocate %u bytes "
|
|
|
|
|
- "for string.", size);
|
|
|
|
|
- else {
|
|
|
|
|
- memcpy(stringValue, contents, size);
|
|
|
|
|
- *stringValueP = stringValue;
|
|
|
|
|
- *lengthP = (unsigned int)size;
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-/*=========================================================================
|
|
|
|
|
-** Building XML-RPC values.
|
|
|
|
|
-**=========================================================================
|
|
|
|
|
-** Build new XML-RPC values from a format string. This code is heavily
|
|
|
|
|
-** inspired by Py_BuildValue from Python 1.5.2. In particular, our
|
|
|
|
|
-** particular abuse of the va_list data type is copied from the equivalent
|
|
|
|
|
-** Python code in modsupport.c. Since Python is portable, our code should
|
|
|
|
|
-** (in theory) also be portable.
|
|
|
|
|
-*/
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-xmlrpc_type xmlrpc_value_type (xmlrpc_value* value)
|
|
|
|
|
-{
|
|
|
|
|
- XMLRPC_ASSERT_VALUE_OK(value);
|
|
|
|
|
- return value->_type;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-createXmlrpcValue(xmlrpc_env * const envP,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-/*----------------------------------------------------------------------------
|
|
|
|
|
- Create a blank xmlrpc_value to be filled in.
|
|
|
|
|
-
|
|
|
|
|
- Set the reference count to 1.
|
|
|
|
|
------------------------------------------------------------------------------*/
|
|
|
|
|
- xmlrpc_value * valP;
|
|
|
|
|
-
|
|
|
|
|
- valP = (xmlrpc_value*) malloc(sizeof(xmlrpc_value));
|
|
|
|
|
- if (!valP)
|
|
|
|
|
- xmlrpc_env_set_fault(envP, XMLRPC_INTERNAL_ERROR,
|
|
|
|
|
- "Could not allocate memory for xmlrpc_value");
|
|
|
|
|
- else
|
|
|
|
|
- valP->_refcount = 1;
|
|
|
|
|
-
|
|
|
|
|
- *valPP = valP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-mkInt(xmlrpc_env * const envP,
|
|
|
|
|
- xmlrpc_int32 const value,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * valP;
|
|
|
|
|
-
|
|
|
|
|
- createXmlrpcValue(envP, &valP);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- valP->_type = XMLRPC_TYPE_INT;
|
|
|
|
|
- valP->_value.i = value;
|
|
|
|
|
- }
|
|
|
|
|
- *valPP = valP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-mkBool(xmlrpc_env * const envP,
|
|
|
|
|
- xmlrpc_bool const value,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * valP;
|
|
|
|
|
-
|
|
|
|
|
- createXmlrpcValue(envP, &valP);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- valP->_type = XMLRPC_TYPE_BOOL;
|
|
|
|
|
- valP->_value.b = value;
|
|
|
|
|
- }
|
|
|
|
|
- *valPP = valP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-mkDouble(xmlrpc_env * const envP,
|
|
|
|
|
- double const value,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * valP;
|
|
|
|
|
-
|
|
|
|
|
- createXmlrpcValue(envP, &valP);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- valP->_type = XMLRPC_TYPE_DOUBLE;
|
|
|
|
|
- valP->_value.d = value;
|
|
|
|
|
- }
|
|
|
|
|
- *valPP = valP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-#ifdef HAVE_UNICODE_WCHAR
|
|
|
|
|
-#define MAKE_WCS_BLOCK_NULL(val) ((val)->_wcs_block = NULL)
|
|
|
|
|
-#else
|
|
|
|
|
-#define MAKE_WCS_BLOCK_NULL(val) do {} while(0)
|
|
|
|
|
-#endif
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-mkString(xmlrpc_env * const envP,
|
|
|
|
|
- const char * const value,
|
|
|
|
|
- unsigned int const length,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * valP;
|
|
|
|
|
-
|
|
|
|
|
- createXmlrpcValue(envP, &valP);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- valP->_type = XMLRPC_TYPE_STRING;
|
|
|
|
|
- MAKE_WCS_BLOCK_NULL(valP);
|
|
|
|
|
- XMLRPC_MEMBLOCK_INIT(char, envP, &valP->_block, length + 1);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- char * const contents =
|
|
|
|
|
- XMLRPC_MEMBLOCK_CONTENTS(char, &valP->_block);
|
|
|
|
|
- memcpy(contents, value, length);
|
|
|
|
|
- contents[length] = '\0';
|
|
|
|
|
- }
|
|
|
|
|
- if (envP->fault_occurred)
|
|
|
|
|
- free(valP);
|
|
|
|
|
- }
|
|
|
|
|
- *valPP = valP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-getString(xmlrpc_env * const envP,
|
|
|
|
|
- const char ** const formatP,
|
|
|
|
|
- va_list * const args,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- const char * str;
|
|
|
|
|
- size_t len;
|
|
|
|
|
-
|
|
|
|
|
- str = (const char*) va_arg(*args, char*);
|
|
|
|
|
- if (**formatP == '#') {
|
|
|
|
|
- (*formatP)++;
|
|
|
|
|
- len = (size_t) va_arg(*args, size_t);
|
|
|
|
|
- } else
|
|
|
|
|
- len = strlen(str);
|
|
|
|
|
-
|
|
|
|
|
- mkString(envP, str, (unsigned int)len, valPP);
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-#ifdef HAVE_UNICODE_WCHAR
|
|
|
|
|
-static void
|
|
|
|
|
-mkWideString(xmlrpc_env * const envP,
|
|
|
|
|
- wchar_t * const wcs,
|
|
|
|
|
- size_t const wcs_len,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * valP;
|
|
|
|
|
- char *contents;
|
|
|
|
|
- wchar_t *wcs_contents;
|
|
|
|
|
- int block_is_inited;
|
|
|
|
|
- xmlrpc_mem_block *utf8_block;
|
|
|
|
|
- char *utf8_contents;
|
|
|
|
|
- size_t utf8_len;
|
|
|
|
|
-
|
|
|
|
|
- /* Error-handling preconditions. */
|
|
|
|
|
- utf8_block = NULL;
|
|
|
|
|
- block_is_inited = 0;
|
|
|
|
|
-
|
|
|
|
|
- /* Initialize our XML-RPC value. */
|
|
|
|
|
- valP = (xmlrpc_value*) malloc(sizeof(xmlrpc_value));
|
|
|
|
|
- XMLRPC_FAIL_IF_NULL(valP, envP, XMLRPC_INTERNAL_ERROR,
|
|
|
|
|
- "Could not allocate memory for wide string");
|
|
|
|
|
- valP->_refcount = 1;
|
|
|
|
|
- valP->_type = XMLRPC_TYPE_STRING;
|
|
|
|
|
-
|
|
|
|
|
- /* More error-handling preconditions. */
|
|
|
|
|
- valP->_wcs_block = NULL;
|
|
|
|
|
-
|
|
|
|
|
- /* Build our wchar_t block first. */
|
|
|
|
|
- valP->_wcs_block =
|
|
|
|
|
- XMLRPC_TYPED_MEM_BLOCK_NEW(wchar_t, envP, wcs_len + 1);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(envP);
|
|
|
|
|
- wcs_contents =
|
|
|
|
|
- XMLRPC_TYPED_MEM_BLOCK_CONTENTS(wchar_t, valP->_wcs_block);
|
|
|
|
|
- memcpy(wcs_contents, wcs, wcs_len * sizeof(wchar_t));
|
|
|
|
|
- wcs_contents[wcs_len] = '\0';
|
|
|
|
|
-
|
|
|
|
|
- /* Convert the wcs block to UTF-8. */
|
|
|
|
|
- utf8_block = xmlrpc_wcs_to_utf8(envP, wcs_contents, wcs_len + 1);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(envP);
|
|
|
|
|
- utf8_contents = XMLRPC_TYPED_MEM_BLOCK_CONTENTS(char, utf8_block);
|
|
|
|
|
- utf8_len = XMLRPC_TYPED_MEM_BLOCK_SIZE(char, utf8_block);
|
|
|
|
|
-
|
|
|
|
|
- /* XXX - We need an extra memcopy to initialize _block. */
|
|
|
|
|
- XMLRPC_TYPED_MEM_BLOCK_INIT(char, envP, &valP->_block, utf8_len);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(envP);
|
|
|
|
|
- block_is_inited = 1;
|
|
|
|
|
- contents = XMLRPC_TYPED_MEM_BLOCK_CONTENTS(char, &valP->_block);
|
|
|
|
|
- memcpy(contents, utf8_contents, utf8_len);
|
|
|
|
|
-
|
|
|
|
|
- cleanup:
|
|
|
|
|
- if (utf8_block)
|
|
|
|
|
- xmlrpc_mem_block_free(utf8_block);
|
|
|
|
|
- if (envP->fault_occurred) {
|
|
|
|
|
- if (valP) {
|
|
|
|
|
- if (valP->_wcs_block)
|
|
|
|
|
- xmlrpc_mem_block_free(valP->_wcs_block);
|
|
|
|
|
- if (block_is_inited)
|
|
|
|
|
- xmlrpc_mem_block_clean(&valP->_block);
|
|
|
|
|
- free(valP);
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- *valPP = valP;
|
|
|
|
|
-}
|
|
|
|
|
-#endif /* HAVE_UNICODE_WCHAR */
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-getWideString(xmlrpc_env * const envP,
|
|
|
|
|
- const char ** const formatP,
|
|
|
|
|
- va_list * const args,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-#ifdef HAVE_UNICODE_WCHAR
|
|
|
|
|
-
|
|
|
|
|
- wchar_t *wcs;
|
|
|
|
|
- size_t len;
|
|
|
|
|
-
|
|
|
|
|
- wcs = (wchar_t*) va_arg(*args, wchar_t*);
|
|
|
|
|
- if (**formatP == '#') {
|
|
|
|
|
- (*formatP)++;
|
|
|
|
|
- len = (size_t) va_arg(*args, size_t);
|
|
|
|
|
- } else
|
|
|
|
|
- len = wcslen(wcs);
|
|
|
|
|
-
|
|
|
|
|
- mkWideString(envP, wcs, len, valPP);
|
|
|
|
|
-
|
|
|
|
|
-#endif /* HAVE_UNICODE_WCHAR */
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-mkDatetime(xmlrpc_env * const envP,
|
|
|
|
|
- const char * const value,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * valP;
|
|
|
|
|
-
|
|
|
|
|
- createXmlrpcValue(envP, &valP);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- valP->_type = XMLRPC_TYPE_DATETIME;
|
|
|
|
|
-
|
|
|
|
|
- XMLRPC_TYPED_MEM_BLOCK_INIT(
|
|
|
|
|
- char, envP, &valP->_block, strlen(value) + 1);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- char * const contents =
|
|
|
|
|
- XMLRPC_TYPED_MEM_BLOCK_CONTENTS(char, &valP->_block);
|
|
|
|
|
- strcpy(contents, value);
|
|
|
|
|
- }
|
|
|
|
|
- if (envP->fault_occurred)
|
|
|
|
|
- free(valP);
|
|
|
|
|
- }
|
|
|
|
|
- *valPP = valP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-mkBase64(xmlrpc_env * const envP,
|
|
|
|
|
- const unsigned char * const value,
|
|
|
|
|
- size_t const length,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * valP;
|
|
|
|
|
-
|
|
|
|
|
- createXmlrpcValue(envP, &valP);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- valP->_type = XMLRPC_TYPE_BASE64;
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_mem_block_init(envP, &valP->_block, length);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- char * const contents =
|
|
|
|
|
- xmlrpc_mem_block_contents(&valP->_block);
|
|
|
|
|
- memcpy(contents, value, length);
|
|
|
|
|
- }
|
|
|
|
|
- if (envP->fault_occurred)
|
|
|
|
|
- free(valP);
|
|
|
|
|
- }
|
|
|
|
|
- *valPP = valP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-getBase64(xmlrpc_env * const envP,
|
|
|
|
|
- va_list * const args,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- unsigned char * value;
|
|
|
|
|
- size_t length;
|
|
|
|
|
-
|
|
|
|
|
- value = (unsigned char*) va_arg(*args, unsigned char*);
|
|
|
|
|
- length = (size_t) va_arg(*args, size_t);
|
|
|
|
|
-
|
|
|
|
|
- mkBase64(envP, value, length, valPP);
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-mkCPtr(xmlrpc_env * const envP,
|
|
|
|
|
- void * const value,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * valP;
|
|
|
|
|
-
|
|
|
|
|
- createXmlrpcValue(envP, &valP);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- valP->_type = XMLRPC_TYPE_C_PTR;
|
|
|
|
|
- valP->_value.c_ptr = value;
|
|
|
|
|
- }
|
|
|
|
|
- *valPP = valP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-mkArrayFromVal(xmlrpc_env * const envP,
|
|
|
|
|
- xmlrpc_value * const value,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- if (xmlrpc_value_type(value) != XMLRPC_TYPE_ARRAY)
|
|
|
|
|
- xmlrpc_env_set_fault(envP, XMLRPC_INTERNAL_ERROR,
|
|
|
|
|
- "Array format ('A'), non-array xmlrpc_value");
|
|
|
|
|
- else
|
|
|
|
|
- xmlrpc_INCREF(value);
|
|
|
|
|
-
|
|
|
|
|
- *valPP = value;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-mkStructFromVal(xmlrpc_env * const envP,
|
|
|
|
|
- xmlrpc_value * const value,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-
|
|
|
|
|
- if (xmlrpc_value_type(value) != XMLRPC_TYPE_STRUCT)
|
|
|
|
|
- xmlrpc_env_set_fault(envP, XMLRPC_INTERNAL_ERROR,
|
|
|
|
|
- "Struct format ('S'), non-struct xmlrpc_value");
|
|
|
|
|
- else
|
|
|
|
|
- xmlrpc_INCREF(value);
|
|
|
|
|
-
|
|
|
|
|
- *valPP = value;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-getValue(xmlrpc_env * const envP,
|
|
|
|
|
- const char** const format,
|
|
|
|
|
- va_list * const args,
|
|
|
|
|
- xmlrpc_value ** const valPP);
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-createXmlrpcArray(xmlrpc_env * const envP,
|
|
|
|
|
- xmlrpc_value ** const arrayPP) {
|
|
|
|
|
-/*----------------------------------------------------------------------------
|
|
|
|
|
- Create an empty array xmlrpc_value.
|
|
|
|
|
------------------------------------------------------------------------------*/
|
|
|
|
|
- xmlrpc_value * arrayP;
|
|
|
|
|
-
|
|
|
|
|
- createXmlrpcValue(envP, &arrayP);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- arrayP->_type = XMLRPC_TYPE_ARRAY;
|
|
|
|
|
- XMLRPC_TYPED_MEM_BLOCK_INIT(xmlrpc_value*, envP, &arrayP->_block, 0);
|
|
|
|
|
- if (envP->fault_occurred)
|
|
|
|
|
- free(arrayP);
|
|
|
|
|
- }
|
|
|
|
|
- *arrayPP = arrayP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-getArray(xmlrpc_env * const envP,
|
|
|
|
|
- const char ** const formatP,
|
|
|
|
|
- char const delimiter,
|
|
|
|
|
- va_list * const args,
|
|
|
|
|
- xmlrpc_value ** const arrayPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * arrayP;
|
|
|
|
|
-
|
|
|
|
|
- createXmlrpcArray(envP, &arrayP);
|
|
|
|
|
-
|
|
|
|
|
- /* Add items to the array until we hit our delimiter. */
|
|
|
|
|
-
|
|
|
|
|
- while (**formatP != delimiter && !envP->fault_occurred) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * itemP;
|
|
|
|
|
-
|
|
|
|
|
- if (**formatP == '\0')
|
|
|
|
|
- xmlrpc_env_set_fault(
|
|
|
|
|
- envP, XMLRPC_INTERNAL_ERROR,
|
|
|
|
|
- "format string ended before closing ')'.");
|
|
|
|
|
- else {
|
|
|
|
|
- getValue(envP, formatP, args, &itemP);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- xmlrpc_array_append_item(envP, arrayP, itemP);
|
|
|
|
|
- xmlrpc_DECREF(itemP);
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- if (envP->fault_occurred)
|
|
|
|
|
- xmlrpc_DECREF(arrayP);
|
|
|
|
|
-
|
|
|
|
|
- *arrayPP = arrayP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-getStructMember(xmlrpc_env * const envP,
|
|
|
|
|
- const char ** const formatP,
|
|
|
|
|
- va_list * const args,
|
|
|
|
|
- xmlrpc_value ** const keyPP,
|
|
|
|
|
- xmlrpc_value ** const valuePP) {
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
- /* Get the key */
|
|
|
|
|
- getValue(envP, formatP, args, keyPP);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- if (**formatP != ':')
|
|
|
|
|
- xmlrpc_env_set_fault(
|
|
|
|
|
- envP, XMLRPC_INTERNAL_ERROR,
|
|
|
|
|
- "format string does not have ':' after a "
|
|
|
|
|
- "structure member key.");
|
|
|
|
|
- else {
|
|
|
|
|
- /* Skip over colon that separates key from value */
|
|
|
|
|
- (*formatP)++;
|
|
|
|
|
-
|
|
|
|
|
- /* Get the value */
|
|
|
|
|
- getValue(envP, formatP, args, valuePP);
|
|
|
|
|
- }
|
|
|
|
|
- if (envP->fault_occurred)
|
|
|
|
|
- xmlrpc_DECREF(*keyPP);
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-getStruct(xmlrpc_env * const envP,
|
|
|
|
|
- const char ** const formatP,
|
|
|
|
|
- char const delimiter,
|
|
|
|
|
- va_list * const args,
|
|
|
|
|
- xmlrpc_value ** const structPP) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value * structP;
|
|
|
|
|
-
|
|
|
|
|
- structP = xmlrpc_struct_new(envP);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- while (**formatP != delimiter && !envP->fault_occurred) {
|
|
|
|
|
- xmlrpc_value * keyP;
|
|
|
|
|
- xmlrpc_value * valueP;
|
|
|
|
|
-
|
|
|
|
|
- getStructMember(envP, formatP, args, &keyP, &valueP);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- if (**formatP == ',')
|
|
|
|
|
- (*formatP)++; /* Skip over the comma */
|
|
|
|
|
- else if (**formatP == delimiter) {
|
|
|
|
|
- /* End of the line */
|
|
|
|
|
- } else
|
|
|
|
|
- xmlrpc_env_set_fault(
|
|
|
|
|
- envP, XMLRPC_INTERNAL_ERROR,
|
|
|
|
|
- "format string does not have ',' or ')' after "
|
|
|
|
|
- "a structure member");
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred)
|
|
|
|
|
- /* Add the new member to the struct. */
|
|
|
|
|
- xmlrpc_struct_set_value_v(envP, structP, keyP, valueP);
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_DECREF(valueP);
|
|
|
|
|
- xmlrpc_DECREF(keyP);
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- if (envP->fault_occurred)
|
|
|
|
|
- xmlrpc_DECREF(structP);
|
|
|
|
|
- }
|
|
|
|
|
- *structPP = structP;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-getValue(xmlrpc_env * const envP,
|
|
|
|
|
- const char** const formatP,
|
|
|
|
|
- va_list * const args,
|
|
|
|
|
- xmlrpc_value ** const valPP) {
|
|
|
|
|
-/*----------------------------------------------------------------------------
|
|
|
|
|
- Get the next value from the list. *formatP points to the specifier
|
|
|
|
|
- for the next value in the format string (i.e. to the type code
|
|
|
|
|
- character) and we move *formatP past the whole specifier for the
|
|
|
|
|
- next value. We read the required arguments from 'args'. We return
|
|
|
|
|
- the value as *valPP with a reference to it.
|
|
|
|
|
-
|
|
|
|
|
- For example, if *formatP points to the "i" in the string "sis",
|
|
|
|
|
- we read one argument from 'args' and return as *valP an integer whose
|
|
|
|
|
- value is the argument we read. We advance *formatP to point to the
|
|
|
|
|
- last 's' and advance 'args' to point to the argument that belongs to
|
|
|
|
|
- that 's'.
|
|
|
|
|
------------------------------------------------------------------------------*/
|
|
|
|
|
- char const formatChar = *(*formatP)++;
|
|
|
|
|
-
|
|
|
|
|
- switch (formatChar) {
|
|
|
|
|
- case 'i':
|
|
|
|
|
- mkInt(envP, (xmlrpc_int32) va_arg(*args, xmlrpc_int32), valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'b':
|
|
|
|
|
- mkBool(envP, (xmlrpc_bool) va_arg(*args, xmlrpc_bool), valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'd':
|
|
|
|
|
- mkDouble(envP, (double) va_arg(*args, va_double), valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 's':
|
|
|
|
|
- getString(envP, formatP, args, valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'w':
|
|
|
|
|
- getWideString(envP, formatP, args, valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- /* The code 't' is reserved for a better, time_t based
|
|
|
|
|
- implementation of dateTime conversion.
|
|
|
|
|
- */
|
|
|
|
|
- case '8':
|
|
|
|
|
- mkDatetime(envP, (char*) va_arg(*args, char*), valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case '6':
|
|
|
|
|
- getBase64(envP, args, valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'p':
|
|
|
|
|
- /* We might someday want to use the code 'p!' to read in a
|
|
|
|
|
- cleanup function for this pointer.
|
|
|
|
|
- */
|
|
|
|
|
- mkCPtr(envP, (void*) va_arg(*args, void*), valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'A':
|
|
|
|
|
- mkArrayFromVal(envP, (xmlrpc_value*) va_arg(*args, xmlrpc_value*),
|
|
|
|
|
- valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'S':
|
|
|
|
|
- mkStructFromVal(envP, (xmlrpc_value*) va_arg(*args, xmlrpc_value*),
|
|
|
|
|
- valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'V':
|
|
|
|
|
- *valPP = (xmlrpc_value*) va_arg(*args, xmlrpc_value*);
|
|
|
|
|
- xmlrpc_INCREF(*valPP);
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case '(':
|
|
|
|
|
- getArray(envP, formatP, ')', args, valPP);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- XMLRPC_ASSERT(**formatP == ')');
|
|
|
|
|
- (*formatP)++; /* Skip over closing parenthesis */
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case '{':
|
|
|
|
|
- getStruct(envP, formatP, '}', args, valPP);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- XMLRPC_ASSERT(**formatP == '}');
|
|
|
|
|
- (*formatP)++; /* Skip over closing brace */
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- default: {
|
|
|
|
|
- const char * const badCharacter = xmlrpc_makePrintableChar(formatChar);
|
|
|
|
|
- xmlrpc_env_set_fault_formatted(
|
|
|
|
|
- envP, XMLRPC_INTERNAL_ERROR,
|
|
|
|
|
- "Unexpected character '%s' in format string", badCharacter);
|
|
|
|
|
- xmlrpc_strfree(badCharacter);
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_build_value_va(xmlrpc_env * const envP,
|
|
|
|
|
- const char * const format,
|
|
|
|
|
- va_list args,
|
|
|
|
|
- xmlrpc_value ** const valPP,
|
|
|
|
|
- const char ** const tailP) {
|
|
|
|
|
-
|
|
|
|
|
- const char * formatCursor;
|
|
|
|
|
- va_list args_copy;
|
|
|
|
|
-
|
|
|
|
|
- XMLRPC_ASSERT_ENV_OK(envP);
|
|
|
|
|
- XMLRPC_ASSERT(format != NULL);
|
|
|
|
|
-
|
|
|
|
|
- if (strlen(format) == 0)
|
|
|
|
|
- xmlrpc_env_set_fault_formatted(
|
|
|
|
|
- envP, XMLRPC_INTERNAL_ERROR, "Format string is empty.");
|
|
|
|
|
- else {
|
|
|
|
|
- formatCursor = &format[0];
|
|
|
|
|
- VA_LIST_COPY(args_copy, args);
|
|
|
|
|
- getValue(envP, &formatCursor, &args_copy, valPP);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred)
|
|
|
|
|
- XMLRPC_ASSERT_VALUE_OK(*valPP);
|
|
|
|
|
-
|
|
|
|
|
- *tailP = formatCursor;
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-xmlrpc_value *
|
|
|
|
|
-xmlrpc_build_value(xmlrpc_env * const envP,
|
|
|
|
|
- const char * const format,
|
|
|
|
|
- ...) {
|
|
|
|
|
-
|
|
|
|
|
- va_list args;
|
|
|
|
|
- xmlrpc_value* retval;
|
|
|
|
|
- const char * suffix;
|
|
|
|
|
-
|
|
|
|
|
- va_start(args, format);
|
|
|
|
|
- xmlrpc_build_value_va(envP, format, args, &retval, &suffix);
|
|
|
|
|
- va_end(args);
|
|
|
|
|
-
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- if (*suffix != '\0')
|
|
|
|
|
- xmlrpc_env_set_fault_formatted(
|
|
|
|
|
- envP, XMLRPC_INTERNAL_ERROR, "Junk after the argument "
|
|
|
|
|
- "specifier: '%s'. There must be exactly one arument.",
|
|
|
|
|
- suffix);
|
|
|
|
|
-
|
|
|
|
|
- if (envP->fault_occurred)
|
|
|
|
|
- xmlrpc_DECREF(retval);
|
|
|
|
|
- }
|
|
|
|
|
- return retval;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-/*=========================================================================
|
|
|
|
|
-** Parsing XML-RPC values.
|
|
|
|
|
-**=========================================================================
|
|
|
|
|
-** Parse an XML-RPC value based on a format string. This code is heavily
|
|
|
|
|
-** inspired by Py_BuildValue from Python 1.5.2.
|
|
|
|
|
-*/
|
|
|
|
|
-
|
|
|
|
|
-/* Prototype for recursive invocation: */
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-parsevalue(xmlrpc_env * const env,
|
|
|
|
|
- xmlrpc_value * const val,
|
|
|
|
|
- const char ** const format,
|
|
|
|
|
- va_list * args);
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-parsearray(xmlrpc_env * const env,
|
|
|
|
|
- const xmlrpc_value * const array,
|
|
|
|
|
- const char ** const format,
|
|
|
|
|
- char const delimiter,
|
|
|
|
|
- va_list * args) {
|
|
|
|
|
-
|
|
|
|
|
- int size, i;
|
|
|
|
|
- xmlrpc_value *item;
|
|
|
|
|
-
|
|
|
|
|
- /* Fetch the array size. */
|
|
|
|
|
- size = xmlrpc_array_size(env, array);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(env);
|
|
|
|
|
-
|
|
|
|
|
- /* Loop over the items in the array. */
|
|
|
|
|
- for (i = 0; i < size; i++) {
|
|
|
|
|
- /* Bail out if the caller didn't care about the rest of the items. */
|
|
|
|
|
- if (**format == '*')
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- item = xmlrpc_array_get_item(env, array, i);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(env);
|
|
|
|
|
-
|
|
|
|
|
- XMLRPC_ASSERT(**format != '\0');
|
|
|
|
|
- if (**format == delimiter)
|
|
|
|
|
- XMLRPC_FAIL(env, XMLRPC_INDEX_ERROR, "Too many items in array");
|
|
|
|
|
- parsevalue(env, item, format, args);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(env);
|
|
|
|
|
- }
|
|
|
|
|
- if (**format == '*')
|
|
|
|
|
- (*format)++;
|
|
|
|
|
- if (**format != delimiter)
|
|
|
|
|
- XMLRPC_FAIL(env, XMLRPC_INDEX_ERROR, "Not enough items in array");
|
|
|
|
|
-
|
|
|
|
|
- cleanup:
|
|
|
|
|
- return;
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-parsestruct(xmlrpc_env * const env,
|
|
|
|
|
- xmlrpc_value * const strct,
|
|
|
|
|
- const char ** const format,
|
|
|
|
|
- char const delimiter,
|
|
|
|
|
- va_list * args) {
|
|
|
|
|
-
|
|
|
|
|
- xmlrpc_value *key, *value;
|
|
|
|
|
- char *keystr;
|
|
|
|
|
- size_t keylen;
|
|
|
|
|
-
|
|
|
|
|
- /* Set up error handling preconditions. */
|
|
|
|
|
- key = NULL;
|
|
|
|
|
-
|
|
|
|
|
- /* Build the members of our struct. */
|
|
|
|
|
- while (**format != '*' && **format != delimiter && **format != '\0') {
|
|
|
|
|
-
|
|
|
|
|
- /* Get our key, and skip over the ':' character. Notice the
|
|
|
|
|
- ** sudden call to getValue--we're going in the opposite direction. */
|
|
|
|
|
- getValue(env, format, args, &key);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(env);
|
|
|
|
|
- XMLRPC_ASSERT(**format == ':');
|
|
|
|
|
- (*format)++;
|
|
|
|
|
-
|
|
|
|
|
- /* Look up the value for our key. */
|
|
|
|
|
- xmlrpc_parse_value(env, key, "s#", &keystr, &keylen);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(env);
|
|
|
|
|
- value = xmlrpc_struct_get_value_n(env, strct, keystr, keylen);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(env);
|
|
|
|
|
-
|
|
|
|
|
- /* Get our value, and skip over the ',' character (if present). */
|
|
|
|
|
- parsevalue(env, value, format, args);
|
|
|
|
|
- XMLRPC_FAIL_IF_FAULT(env);
|
|
|
|
|
- XMLRPC_ASSERT(**format == ',' || **format == delimiter);
|
|
|
|
|
- if (**format == ',')
|
|
|
|
|
- (*format)++;
|
|
|
|
|
-
|
|
|
|
|
- /* Release our reference, and restore our invariant. */
|
|
|
|
|
- xmlrpc_DECREF(key);
|
|
|
|
|
- key = NULL;
|
|
|
|
|
- }
|
|
|
|
|
- if (**format == '*') {
|
|
|
|
|
- (*format)++;
|
|
|
|
|
- if (**format != delimiter && **format != '\0')
|
|
|
|
|
- XMLRPC_FAIL(env, XMLRPC_INTERNAL_ERROR,
|
|
|
|
|
- "* can appear only at the end "
|
|
|
|
|
- "of a structure format specifier");
|
|
|
|
|
- } else {
|
|
|
|
|
- /* Here we're supposed to fail if he didn't extract all the
|
|
|
|
|
- members. But we don't know how to determine whether he
|
|
|
|
|
- specified all the members, so we always fail.
|
|
|
|
|
- */
|
|
|
|
|
- XMLRPC_FAIL(env, XMLRPC_INTERNAL_ERROR, "You must specify '*' as the "
|
|
|
|
|
- "last member of a structure in a format specifier "
|
|
|
|
|
- "used for parsing an xmlrpc_value");
|
|
|
|
|
- }
|
|
|
|
|
- XMLRPC_ASSERT(**format == delimiter || **format == '\0');
|
|
|
|
|
-
|
|
|
|
|
-cleanup:
|
|
|
|
|
- if (key)
|
|
|
|
|
- xmlrpc_DECREF(key);
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-static void
|
|
|
|
|
-parsevalue(xmlrpc_env * const envP,
|
|
|
|
|
- xmlrpc_value * const valueP,
|
|
|
|
|
- const char ** const format,
|
|
|
|
|
- va_list * args) {
|
|
|
|
|
-
|
|
|
|
|
- char formatSpecChar;
|
|
|
|
|
-
|
|
|
|
|
- formatSpecChar = *(*format)++;
|
|
|
|
|
-
|
|
|
|
|
- switch (formatSpecChar) {
|
|
|
|
|
- case 'i':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_INT);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- xmlrpc_int32 * const int32ptr =
|
|
|
|
|
- (xmlrpc_int32*) va_arg(*args, xmlrpc_int32*);
|
|
|
|
|
- *int32ptr = valueP->_value.i;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'b':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_BOOL);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- xmlrpc_bool * const boolptr =
|
|
|
|
|
- (xmlrpc_bool*) va_arg(*args, xmlrpc_bool*);
|
|
|
|
|
- *boolptr = valueP->_value.b;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'd':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_DOUBLE);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- double * const doubleptr = (double*) va_arg(*args, double*);
|
|
|
|
|
- *doubleptr = valueP->_value.d;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 's':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_STRING);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- char * const contents =
|
|
|
|
|
- XMLRPC_MEMBLOCK_CONTENTS(char, &valueP->_block);
|
|
|
|
|
- size_t const len = XMLRPC_MEMBLOCK_SIZE(char, &valueP->_block) - 1;
|
|
|
|
|
-
|
|
|
|
|
- char ** const strptr = (char**) va_arg(*args, char**);
|
|
|
|
|
- if (**format == '#') {
|
|
|
|
|
- size_t * const sizeptr = (size_t*) va_arg(*args, size_t**);
|
|
|
|
|
- (*format)++;
|
|
|
|
|
- *sizeptr = len;
|
|
|
|
|
- } else
|
|
|
|
|
- verifyNoNulls(envP, contents, (unsigned int)len);
|
|
|
|
|
- *strptr = contents;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
-#ifdef HAVE_UNICODE_WCHAR
|
|
|
|
|
- case 'w':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_STRING);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- if (!valueP->_wcs_block) {
|
|
|
|
|
- /* Allocate a wchar_t string if we don't have one. */
|
|
|
|
|
- char * const contents =
|
|
|
|
|
- XMLRPC_MEMBLOCK_CONTENTS(char, &valueP->_block);
|
|
|
|
|
- size_t const len =
|
|
|
|
|
- XMLRPC_MEMBLOCK_SIZE(char, &valueP->_block) - 1;
|
|
|
|
|
- valueP->_wcs_block =
|
|
|
|
|
- xmlrpc_utf8_to_wcs(envP, contents, len + 1);
|
|
|
|
|
- }
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- wchar_t * const wcontents =
|
|
|
|
|
- XMLRPC_MEMBLOCK_CONTENTS(wchar_t, valueP->_wcs_block);
|
|
|
|
|
- size_t const len =
|
|
|
|
|
- XMLRPC_MEMBLOCK_SIZE(wchar_t, valueP->_wcs_block) - 1;
|
|
|
|
|
-
|
|
|
|
|
- wchar_t ** const wcsptr = (wchar_t**) va_arg(*args, wchar_t**);
|
|
|
|
|
- if (**format == '#') {
|
|
|
|
|
- size_t * const sizeptr = (size_t*) va_arg(*args, size_t**);
|
|
|
|
|
- (*format)++;
|
|
|
|
|
- *sizeptr = len;
|
|
|
|
|
- } else
|
|
|
|
|
- verifyNoNullsW(envP, wcontents, (unsigned int)len);
|
|
|
|
|
- *wcsptr = wcontents;
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-#endif /* HAVE_UNICODE_WCHAR */
|
|
|
|
|
-
|
|
|
|
|
- case '8':
|
|
|
|
|
- /* The code 't' is reserved for a better, time_t based
|
|
|
|
|
- ** implementation of dateTime conversion. */
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_DATETIME);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- char * const contents =
|
|
|
|
|
- XMLRPC_MEMBLOCK_CONTENTS(char, &valueP->_block);
|
|
|
|
|
- char ** const strptr = (char**) va_arg(*args, char**);
|
|
|
|
|
- *strptr = contents;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case '6':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_BASE64);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- unsigned char * const bin_data =
|
|
|
|
|
- XMLRPC_MEMBLOCK_CONTENTS(unsigned char,
|
|
|
|
|
- &valueP->_block);
|
|
|
|
|
- size_t const len = XMLRPC_MEMBLOCK_SIZE(char, &valueP->_block);
|
|
|
|
|
- unsigned char ** const binptr =
|
|
|
|
|
- (unsigned char**) va_arg(*args, unsigned char**);
|
|
|
|
|
- size_t * const sizeptr = (size_t*) va_arg(*args, size_t**);
|
|
|
|
|
- *binptr = bin_data;
|
|
|
|
|
- *sizeptr = len;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'p':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_C_PTR);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- void ** const voidptrptr = (void**) va_arg(*args, void**);
|
|
|
|
|
- *voidptrptr = valueP->_value.c_ptr;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'V': {
|
|
|
|
|
- xmlrpc_value ** const valptr =
|
|
|
|
|
- (xmlrpc_value**) va_arg(*args, xmlrpc_value**);
|
|
|
|
|
- *valptr = valueP;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'A':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_ARRAY);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- xmlrpc_value ** const valptr =
|
|
|
|
|
- (xmlrpc_value**) va_arg(*args, xmlrpc_value**);
|
|
|
|
|
- *valptr = valueP;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case 'S':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_STRUCT);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- xmlrpc_value ** const valptr =
|
|
|
|
|
- (xmlrpc_value**) va_arg(*args, xmlrpc_value**);
|
|
|
|
|
- *valptr = valueP;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case '(':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_ARRAY);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- parsearray(envP, valueP, format, ')', args);
|
|
|
|
|
- (*format)++;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- case '{':
|
|
|
|
|
- validateType(envP, valueP, XMLRPC_TYPE_STRUCT);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- parsestruct(envP, valueP, format, '}', args);
|
|
|
|
|
- (*format)++;
|
|
|
|
|
- }
|
|
|
|
|
- break;
|
|
|
|
|
-
|
|
|
|
|
- default:
|
|
|
|
|
- xmlrpc_env_set_fault_formatted(
|
|
|
|
|
- envP, XMLRPC_INTERNAL_ERROR, "Invalid format character '%c'",
|
|
|
|
|
- formatSpecChar);
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_parse_value_va(xmlrpc_env * const envP,
|
|
|
|
|
- xmlrpc_value * const value,
|
|
|
|
|
- const char * const format,
|
|
|
|
|
- va_list args) {
|
|
|
|
|
-
|
|
|
|
|
- const char *format_copy;
|
|
|
|
|
- va_list args_copy;
|
|
|
|
|
-
|
|
|
|
|
- XMLRPC_ASSERT_ENV_OK(envP);
|
|
|
|
|
- XMLRPC_ASSERT_VALUE_OK(value);
|
|
|
|
|
- XMLRPC_ASSERT(format != NULL);
|
|
|
|
|
-
|
|
|
|
|
- format_copy = format;
|
|
|
|
|
- VA_LIST_COPY(args_copy, args);
|
|
|
|
|
- parsevalue(envP, value, &format_copy, &args_copy);
|
|
|
|
|
- if (!envP->fault_occurred) {
|
|
|
|
|
- XMLRPC_ASSERT(*format_copy == '\0');
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-void
|
|
|
|
|
-xmlrpc_parse_value(xmlrpc_env * const envP,
|
|
|
|
|
- xmlrpc_value * const value,
|
|
|
|
|
- const char * const format,
|
|
|
|
|
- ...) {
|
|
|
|
|
-
|
|
|
|
|
- va_list args;
|
|
|
|
|
-
|
|
|
|
|
- va_start(args, format);
|
|
|
|
|
- xmlrpc_parse_value_va(envP, value, format, args);
|
|
|
|
|
- va_end(args);
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-/* Copyright (C) 2001 by First Peer, Inc. All rights reserved.
|
|
|
|
|
-** Copyright (C) 2001 by Eric Kidd. All rights reserved.
|
|
|
|
|
-**
|
|
|
|
|
-** Redistribution and use in source and binary forms, with or without
|
|
|
|
|
-** modification, are permitted provided that the following conditions
|
|
|
|
|
-** are met:
|
|
|
|
|
-** 1. Redistributions of source code must retain the above copyright
|
|
|
|
|
-** notice, this list of conditions and the following disclaimer.
|
|
|
|
|
-** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
|
-** notice, this list of conditions and the following disclaimer in the
|
|
|
|
|
-** documentation and/or other materials provided with the distribution.
|
|
|
|
|
-** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
|
-** derived from this software without specific prior written permission.
|
|
|
|
|
-**
|
|
|
|
|
-** THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
|
-** ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
|
-** IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
|
-** ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
|
|
|
-** FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
|
-** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
|
-** OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
|
-** HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
|
-** LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
|
-** OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
|
-** SUCH DAMAGE. */
|
|
|